LyoqCiAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAgKiBAZmlsZSAgICBzdG0zMmg3eHhfaGFsX2NyeXAuYwogICogQGF1dGhvciAgTUNEIEFwcGxpY2F0aW9uIFRlYW0KICAqIEBicmllZiAgIENSWVAgSEFMIG1vZHVsZSBkcml2ZXIuCiAgKiAgICAgICAgICBUaGlzIGZpbGUgcHJvdmlkZXMgZmlybXdhcmUgZnVuY3Rpb25zIHRvIG1hbmFnZSB0aGUgZm9sbG93aW5nCiAgKiAgICAgICAgICBmdW5jdGlvbmFsaXRpZXMgb2YgdGhlIENyeXB0b2dyYXBoeSAoQ1JZUCkgcGVyaXBoZXJhbDoKICAqICAgICAgICAgICArIEluaXRpYWxpemF0aW9uIGFuZCBkZS1pbml0aWFsaXphdGlvbiBmdW5jdGlvbnMKICAqICAgICAgICAgICArIEFFUyBwcm9jZXNzaW5nIGZ1bmN0aW9ucwogICogICAgICAgICAgICsgREVTIHByb2Nlc3NpbmcgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBUREVTIHByb2Nlc3NpbmcgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBETUEgY2FsbGJhY2sgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBDUllQIElSUSBoYW5kbGVyIG1hbmFnZW1lbnQKICAqICAgICAgICAgICArIFBlcmlwaGVyYWwgU3RhdGUgZnVuY3Rpb25zCiAgKgogIEB2ZXJiYXRpbQogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgICAgICAgICAgICAgICAgICAjIyMjIyBIb3cgdG8gdXNlIHRoaXMgZHJpdmVyICMjIyMjCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICBbLi5dCiAgICAgIFRoZSBDUllQIEhBTCBkcml2ZXIgY2FuIGJlIHVzZWQgaW4gQ1JZUCBJUCBhcyBmb2xsb3dzOgoKICAgICAgKCMpSW5pdGlhbGl6ZSB0aGUgQ1JZUCBsb3cgbGV2ZWwgcmVzb3VyY2VzIGJ5IGltcGxlbWVudGluZyB0aGUgSEFMX0NSWVBfTXNwSW5pdCgpOgogICAgICAgICAoIyMpIEVuYWJsZSB0aGUgQ1JZUCBpbnRlcmZhY2UgY2xvY2sgdXNpbmcgX19IQUxfUkNDX0NSWVBfQ0xLX0VOQUJMRSgpCiAgICAgICAgICgjIykgSW4gY2FzZSBvZiB1c2luZyBpbnRlcnJ1cHRzIChlLmcuIEhBTF9DUllQX0VuY3J5cHRfSVQoKSkKICAgICAgICAgICAgICgrKyspIENvbmZpZ3VyZSB0aGUgQ1JZUCBpbnRlcnJ1cHQgcHJpb3JpdHkgdXNpbmcgSEFMX05WSUNfU2V0UHJpb3JpdHkoKQogICAgICAgICAgICAgKCsrKykgRW5hYmxlIHRoZSBDUllQIElSUSBoYW5kbGVyIHVzaW5nIEhBTF9OVklDX0VuYWJsZUlSUSgpCiAgICAgICAgICAgICAoKysrKSBJbiBDUllQIElSUSBoYW5kbGVyLCBjYWxsIEhBTF9DUllQX0lSUUhhbmRsZXIoKQogICAgICAgICAoIyMpIEluIGNhc2Ugb2YgdXNpbmcgRE1BIHRvIGNvbnRyb2wgZGF0YSB0cmFuc2ZlciAoZS5nLiBIQUxfQ1JZUF9FbmNyeXB0X0RNQSgpKQogICAgICAgICAgICAgKCsrKykgRW5hYmxlIHRoZSBETUF4IGludGVyZmFjZSBjbG9jayB1c2luZyBfX1JDQ19ETUF4X0NMS19FTkFCTEUoKQogICAgICAgICAgICAgKCsrKykgQ29uZmlndXJlIGFuZCBlbmFibGUgdHdvIERNQSBzdHJlYW1zIG9uZSBmb3IgbWFuYWdpbmcgZGF0YSB0cmFuc2ZlciBmcm9tCiAgICAgICAgICAgICAgICAgbWVtb3J5IHRvIHBlcmlwaGVyYWwgKGlucHV0IHN0cmVhbSkgYW5kIGFub3RoZXIgc3RyZWFtIGZvciBtYW5hZ2luZyBkYXRhCiAgICAgICAgICAgICAgICAgdHJhbnNmZXIgZnJvbSBwZXJpcGhlcmFsIHRvIG1lbW9yeSAob3V0cHV0IHN0cmVhbSkKICAgICAgICAgICAgICgrKyspIEFzc29jaWF0ZSB0aGUgaW5pdGlhbGl6ZWQgRE1BIGhhbmRsZSB0byB0aGUgQ1JZUCBETUEgaGFuZGxlCiAgICAgICAgICAgICAgICAgdXNpbmcgIF9fSEFMX0xJTktETUEoKQogICAgICAgICAgICAgKCsrKykgQ29uZmlndXJlIHRoZSBwcmlvcml0eSBhbmQgZW5hYmxlIHRoZSBOVklDIGZvciB0aGUgdHJhbnNmZXIgY29tcGxldGUKICAgICAgICAgICAgICAgICBpbnRlcnJ1cHQgb24gdGhlIHR3byBETUEgU3RyZWFtcy4gVGhlIG91dHB1dCBzdHJlYW0gc2hvdWxkIGhhdmUgaGlnaGVyCiAgICAgICAgICAgICAgICAgcHJpb3JpdHkgdGhhbiB0aGUgaW5wdXQgc3RyZWFtIEhBTF9OVklDX1NldFByaW9yaXR5KCkgYW5kIEhBTF9OVklDX0VuYWJsZUlSUSgpCgogICAgICAoIylJbml0aWFsaXplIHRoZSBDUllQIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMgOgogICAgICAgICAoIyMpIFRoZSBkYXRhIHR5cGU6IDEtYml0LCA4LWJpdCwgMTYtYml0IG9yIDMyLWJpdC4KICAgICAgICAgKCMjKSBUaGUga2V5IHNpemU6IDEyOCwgMTkyIG9yIDI1Ni4KICAgICAgICAgKCMjKSBUaGUgQWxnb01vZGUgREVTLyBUREVTIEFsZ29yaXRobSBFQ0IvQ0JDIG9yIEFFUyBBbGdvcml0aG0gRUNCL0NCQy9DVFIvR0NNIG9yIENDTS4KICAgICAgICAgKCMjKSBUaGUgaW5pdGlhbGl6YXRpb24gdmVjdG9yIChjb3VudGVyKS4gSXQgaXMgbm90IHVzZWQgaW4gRUNCIG1vZGUuCiAgICAgICAgICgjIykgVGhlIGtleSBidWZmZXIgdXNlZCBmb3IgZW5jcnlwdGlvbi9kZWNyeXB0aW9uLgogICAgICAgICAoIyMpIFRoZSBIZWFkZXIgdXNlZCBvbmx5IGluIEFFUyBHQ00gYW5kIENDTSBBbGdvcml0aG0gZm9yIGF1dGhlbnRpY2F0aW9uLgogICAgICAgICAoIyMpIFRoZSBIZWFkZXJTaXplIFRoZSBzaXplIG9mIGhlYWRlciBidWZmZXIgaW4gd29yZC4KICAgICAgICAgKCMjKSBUaGUgQjAgYmxvY2sgaXMgdGhlIGZpcnN0IGF1dGhlbnRpY2F0aW9uIGJsb2NrIHVzZWQgb25seSAgaW4gQUVTIENDTSBtb2RlLgoKICAgICAgKCMpVGhyZWUgcHJvY2Vzc2luZyAoZW5jcnlwdGlvbi9kZWNyeXB0aW9uKSBmdW5jdGlvbnMgYXJlIGF2YWlsYWJsZToKICAgICAgICAgKCMjKSBQb2xsaW5nIG1vZGU6IGVuY3J5cHRpb24gYW5kIGRlY3J5cHRpb24gQVBJcyBhcmUgYmxvY2tpbmcgZnVuY3Rpb25zCiAgICAgICAgICAgICAgaS5lLiB0aGV5IHByb2Nlc3MgdGhlIGRhdGEgYW5kIHdhaXQgdGlsbCB0aGUgcHJvY2Vzc2luZyBpcyBmaW5pc2hlZCwKICAgICAgICAgICAgICBlLmcuIEhBTF9DUllQX0VuY3J5cHQgJiBIQUxfQ1JZUF9EZWNyeXB0CiAgICAgICAgICgjIykgSW50ZXJydXB0IG1vZGU6IGVuY3J5cHRpb24gYW5kIGRlY3J5cHRpb24gQVBJcyBhcmUgbm90IGJsb2NraW5nIGZ1bmN0aW9ucwogICAgICAgICAgICAgIGkuZS4gdGhleSBwcm9jZXNzIHRoZSBkYXRhIHVuZGVyIGludGVycnVwdCwKICAgICAgICAgICAgICBlLmcuIEhBTF9DUllQX0VuY3J5cHRfSVQgJiBIQUxfQ1JZUF9EZWNyeXB0X0lUCiAgICAgICAgICgjIykgRE1BIG1vZGU6IGVuY3J5cHRpb24gYW5kIGRlY3J5cHRpb24gQVBJcyBhcmUgbm90IGJsb2NraW5nIGZ1bmN0aW9ucwogICAgICAgICAgICAgIGkuZS4gdGhlIGRhdGEgdHJhbnNmZXIgaXMgZW5zdXJlZCBieSBETUEsCiAgICAgICAgICAgICAgZS5nLiBIQUxfQ1JZUF9FbmNyeXB0X0RNQSAmIEhBTF9DUllQX0RlY3J5cHRfRE1BCgogICAgICAoIylXaGVuIHRoZSBwcm9jZXNzaW5nIGZ1bmN0aW9uIGlzIGNhbGxlZCBhdCBmaXJzdCB0aW1lIGFmdGVyIEhBTF9DUllQX0luaXQoKQogICAgICAgICB0aGUgQ1JZUCBwZXJpcGhlcmFsIGlzIGNvbmZpZ3VyZWQgYW5kIHByb2Nlc3NlcyB0aGUgYnVmZmVyIGluIGlucHV0LgogICAgICAgICBBdCBzZWNvbmQgY2FsbCwgbm8gbmVlZCB0byBJbml0aWFsaXplIHRoZSBDUllQLCB1c2VyIGhhdmUgdG8gZ2V0IGN1cnJlbnQgY29uZmlndXJhdGlvbiB2aWEKICAgICAgICAgSEFMX0NSWVBfR2V0Q29uZmlnKCkgQVBJLCB0aGVuIG9ubHkgIEhBTF9DUllQX1NldENvbmZpZygpIGlzIHJlcXVlc3RlZCB0byBzZXQKICAgICAgICAgbmV3IHBhcmFtZXRyZXMsIGZpbmFsbHkgdXNlciBjYW4gIHN0YXJ0IGVuY3J5cHRpb24vZGVjcnlwdGlvbi4KCiAgICAgICAoIylDYWxsIEhBTF9DUllQX0RlSW5pdCgpIHRvIGRlaW5pdGlhbGl6ZSB0aGUgQ1JZUCBwZXJpcGhlcmFsLgogICAgICAgCiAgICAgICAoIylUbyBwcm9jZXNzIGEgc2luZ2xlIG1lc3NhZ2Ugd2l0aCBjb25zZWN1dGl2ZSBjYWxscyB0byBIQUxfQ1JZUF9FbmNyeXB0KCkgb3IgSEFMX0NSWVBfRGVjcnlwdCgpCiAgICAgICAgICB3aXRob3V0IGhhdmluZyB0byBjb25maWd1cmUgYWdhaW4gdGhlIEtleSBvciB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yIGJldHdlZW4gZWFjaCBBUEkgY2FsbCwKICAgICAgICAgIHRoZSBmaWVsZCBLZXlJVkNvbmZpZ1NraXAgb2YgdGhlIGluaXRpYWxpemF0aW9uIHN0cnVjdHVyZSBtdXN0IGJlIHNldCB0byBDUllQX0tFWUlWQ09ORklHX09OQ0UuCiAgICAgICAgICBTYW1lIGlzIHRydWUgZm9yIGNvbnNlY3V0aXZlIGNhbGxzIG9mIEhBTF9DUllQX0VuY3J5cHRfSVQoKSwgSEFMX0NSWVBfRGVjcnlwdF9JVCgpLCBIQUxfQ1JZUF9FbmNyeXB0X0RNQSgpCiAgICAgICAgICBvciBIQUxfQ1JZUF9EZWNyeXB0X0RNQSgpLgogICAgICAgICAgCiAgICBbLi5dCiAgICAgIFRoZSBjcnlwdG9ncmFwaGljIHByb2Nlc3NvciBzdXBwb3J0cyBmb2xsb3dpbmcgc3RhbmRhcmRzOgogICAgICAoIykgVGhlIGRhdGEgZW5jcnlwdGlvbiBzdGFuZGFyZCAoREVTKSBhbmQgVHJpcGxlLURFUyAoVERFUykgc3VwcG9ydGVkIG9ubHkgYnkgQ1JZUDEgSVA6CiAgICAgICAgICgjIyk2NC1iaXQgZGF0YSBibG9jayBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgY2hhaW5pbmcgbW9kZXMgc3VwcG9ydGVkIDoKICAgICAgICAgICAgICgrKyspICBFbGVjdHJvbmljIENvZGUgQm9vayhFQ0IpCiAgICAgICAgICAgICAoKysrKSAgQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICAgICgjIykga2V5cyBsZW5ndGggc3VwcG9ydGVkIDo2NC1iaXQsIDEyOC1iaXQgYW5kIDE5Mi1iaXQuCiAgICAgICgjKSBUaGUgYWR2YW5jZWQgZW5jcnlwdGlvbiBzdGFuZGFyZCAoQUVTKSBzdXBwb3J0ZWQgIGJ5IENSWVAxOgogICAgICAgICAoIyMpMTI4LWJpdCBkYXRhIGJsb2NrIHByb2Nlc3NpbmcKICAgICAgICAgKCMjKSBjaGFpbmluZyBtb2RlcyBzdXBwb3J0ZWQgOgogICAgICAgICAgICAgKCsrKykgIEVsZWN0cm9uaWMgQ29kZSBCb29rKEVDQikKICAgICAgICAgICAgICgrKyspICBDaXBoZXIgQmxvY2sgQ2hhaW5pbmcgKENCQykKICAgICAgICAgICAgICgrKyspICBDb3VudGVyIG1vZGUgKENUUikKICAgICAgICAgICAgICgrKyspICBHYWxvaXMvY291bnRlciBtb2RlIChHQ00vR01BQykKICAgICAgICAgICAgICgrKyspICBDb3VudGVyIHdpdGggQ2lwaGVyIEJsb2NrIENoYWluaW5nLU1lc3NhZ2UoQ0NNKQogICAgICAgICAoIyMpIGtleXMgbGVuZ3RoIFN1cHBvcnRlZCA6CiAgICAgICAgICAgICAoKysrKSBmb3IgQ1JZUDEgSVA6IDEyOC1iaXQsIDE5Mi1iaXQgYW5kIDI1Ni1iaXQuCgogICAgWy4uXSAgVGhpcyBzZWN0aW9uIGRlc2NyaWJlcyB0aGUgQUVTIEdhbG9pcy9jb3VudGVyIG1vZGUgKEdDTSkgc3VwcG9ydGVkIGJ5IGJvdGggQ1JZUDEgSVA6CiAgICAgICgjKSAgQWxnb3JpdGhtIHN1cHBvcnRlZCA6CiAgICAgICAgICgjIykgR2Fsb2lzL2NvdW50ZXIgbW9kZSAoR0NNKQogICAgICAgICAoIyMpIEdhbG9pcyBtZXNzYWdlIGF1dGhlbnRpY2F0aW9uIGNvZGUgKEdNQUMpIDppcyBleGFjdGx5IHRoZSBzYW1lIGFzCiAgICAgICAgICAgICAgR0NNIGFsZ29yaXRobSBjb21wb3NlZCBvbmx5IGJ5IGFuIGhlYWRlci4KICAgICAgKCMpICBGb3VyIHBoYXNlcyBhcmUgcGVyZm9ybWVkIGluIEdDTSA6CiAgICAgICAgICgjIykgSW5pdCBwaGFzZTogSVAgcHJlcGFyZXMgdGhlIEdDTSBoYXNoIHN1YmtleSAoSCkgYW5kIGRvIHRoZSBJViBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgSGVhZGVyIHBoYXNlOiBJUCBwcm9jZXNzZXMgdGhlIEFkZGl0aW9uYWwgQXV0aGVudGljYXRlZCBEYXRhIChBQUQpLCB3aXRoIGhhc2gKICAgICAgICAgIGNvbXB1dGF0aW9uIG9ubHkuCiAgICAgICAgICgjIykgUGF5bG9hZCBwaGFzZTogSVAgcHJvY2Vzc2VzIHRoZSBwbGFpbnRleHQgKFApIHdpdGggaGFzaCBjb21wdXRhdGlvbiArIGtleXN0cmVhbQogICAgICAgICAgZW5jcnlwdGlvbiArIGRhdGEgWE9SaW5nLiBJdCB3b3JrcyBpbiBhIHNpbWlsYXIgd2F5IGZvciBjaXBoZXJ0ZXh0IChDKS4KICAgICAgICAgKCMjKSBGaW5hbCBwaGFzZTogSVAgZ2VuZXJhdGVzIHRoZSBhdXRoZW50aWNhdGVkIHRhZyAoVCkgdXNpbmcgdGhlIGxhc3QgYmxvY2sgb2YgZGF0YS4KICAgICAgICAgIEhBTF9DUllQRXhfQUVTR0NNX0dlbmVyYXRlQXV0aFRBRyBBUEkgdXNlZCBpbiB0aGlzIHBoYXNlIHRvIGdlbmVyYXRlIDQgd29yZHMgd2hpY2ggY29ycmVzcG9uZAogICAgICAgICAgdG8gdGhlIFRhZy4gdXNlciBzaG91bGQgY29uc2lkZXIgb25seSBwYXJ0IG9mIHRoaXMgNCB3b3JkcywgaWYgVGFnIGxlbmd0aCBpcyBsZXNzIHRoYW4gMTI4IGJpdHMuIAogICAgICAoIykgIHN0cnVjdHVyZSBvZiBtZXNzYWdlIGNvbnN0cnVjdGlvbiBpbiBHQ00gaXMgZGVmaW5lZCBhcyBiZWxvdyAgOgogICAgICAgICAoIyMpIDE2IGJ5dGVzIEluaXRpYWwgQ291bnRlciBCbG9jayAoSUNCKWNvbXBvc2VkIG9mIElWIGFuZCBjb3VudGVyCiAgICAgICAgICgjIykgVGhlIGF1dGhlbnRpY2F0ZWQgaGVhZGVyIEEgKGFsc28ga25vd3MgYXMgQWRkaXRpb25hbCBBdXRoZW50aWNhdGlvbiBEYXRhIEFBRCkKICAgICAgICAgIHRoaXMgcGFydCBvZiB0aGUgbWVzc2FnZSBpcyBvbmx5IGF1dGhlbnRpY2F0ZWQsIG5vdCBlbmNyeXB0ZWQuCiAgICAgICAgICgjIykgVGhlIHBsYWludGV4dCBtZXNzYWdlIFAgaXMgYm90aCBhdXRoZW50aWNhdGVkIGFuZCBlbmNyeXB0ZWQgYXMgY2lwaGVydGV4dC4KICAgICAgICAgIEdDTSBzdGFuZGFyZCBzcGVjaWZpZXMgdGhhdCBjaXBoZXJ0ZXh0IGhhcyBzYW1lIGJpdCBsZW5ndGggYXMgdGhlIHBsYWludGV4dC4KICAgICAgICAgKCMjKSBUaGUgbGFzdCBibG9jayBpcyBjb21wb3NlZCBvZiB0aGUgbGVuZ3RoIG9mIEEgKG9uIDY0IGJpdHMpIGFuZCB0aGUgbGVuZ3RoIG9mIGNpcGhlcnRleHQKICAgICAgICAgIChvbiA2NCBiaXRzKQoKICAgIFsuLl0gIFRoaXMgc2VjdGlvbiBkZXNjcmliZSBUaGUgQUVTIENvdW50ZXIgd2l0aCBDaXBoZXIgQmxvY2sgQ2hhaW5pbmctTWVzc2FnZQogICAgICAgICAgQXV0aGVudGljYXRpb24gQ29kZSAoQ0NNKSBzdXBwb3J0ZWQgYnkgYm90aCBDUllQMSBJUDoKICAgICAgKCMpICBTcGVjaWZpYyBwYXJhbWV0ZXJzIGZvciBDQ00gIDoKCiAgICAgICAgICgjIykgQjAgYmxvY2sgIDogQWNjb3JkaW5nIHRvIE5JU1QgU3BlY2lhbCBQdWJsaWNhdGlvbiA4MDAtMzhDLAogICAgICAgICAgICBUaGUgZmlyc3QgYmxvY2sgQjAgaXMgZm9ybWF0dGVkIGFzIGZvbGxvd3MsIHdoZXJlIGwobSkgaXMgZW5jb2RlZCBpbgogICAgICAgICAgICBtb3N0LXNpZ25pZmljYW50LWJ5dGUgZmlyc3Qgb3JkZXIoc2VlIGJlbG93IHRhYmxlIDMpCgogICAgICAgICAgICAgICgrKyspICBROiBhIGJpdCBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIG9jdGV0IGxlbmd0aCBvZiBQIChwbGFpbnRleHQpCiAgICAgICAgICAgICAgKCsrKykgIHEgVGhlIG9jdGV0IGxlbmd0aCBvZiB0aGUgYmluYXJ5IHJlcHJlc2VudGF0aW9uIG9mIHRoZSBvY3RldCBsZW5ndGggb2YgdGhlIHBheWxvYWQKICAgICAgICAgICAgICAoKysrKSAgQSBub25jZSAoTiksIG4gVGhlIG9jdGV0IGxlbmd0aCBvZiB0aGUgd2hlcmUgbitxPTE1LgogICAgICAgICAgICAgICgrKyspICBGbGFnczogbW9zdCBzaWduaWZpY2FudCBvY3RldCBjb250YWluaW5nIGZvdXIgZmxhZ3MgZm9yIGNvbnRyb2wgaW5mb3JtYXRpb24sCiAgICAgICAgICAgICAgKCsrKykgIHQgVGhlIG9jdGV0IGxlbmd0aCBvZiB0aGUgTUFDLgogICAgICAgICAoIyMpIEIxIGJsb2NrIChoZWFkZXIpIDogYXNzb2NpYXRlZCBkYXRhIGxlbmd0aChhKSBjb25jYXRlbmF0ZWQgd2l0aCBBc3NvY2lhdGVkIERhdGEgKEEpCiAgICAgICAgICAgICAgdGhlIGFzc29jaWF0ZWQgZGF0YSBsZW5ndGggZXhwcmVzc2VkIGluIGJ5dGVzIChhKSBkZWZpbmVkIGFzIGJlbG93OgogICAgICAgICAgICAoKysrKSAgSWYgMCA8IGEgPCAyMTYtMjgsIHRoZW4gaXQgaXMgZW5jb2RlZCBhcyBbYV0xNiwgaS5lLiB0d28gb2N0ZXRzCiAgICAgICAgICAgICgrKyspICBJZiAyMTYtMjggPCBhIDwgMjMyLCB0aGVuIGl0IGlzIGVuY29kZWQgYXMgMHhmZiB8fCAweGZlIHx8IFthXTMyLCBpLmUuIHNpeCBvY3RldHMKICAgICAgICAgICAgKCsrKykgIElmIDIzMiA8IGEgPCAyNjQsIHRoZW4gaXQgaXMgZW5jb2RlZCBhcyAweGZmIHx8IDB4ZmYgfHwgW2FdNjQsIGkuZS4gdGVuIG9jdGV0cwogICAgICAgICAoIyMpIENUUnggYmxvY2sgIDogY29udHJvbCBibG9ja3MKICAgICAgICAgICAgKCsrKykgR2VuZXJhdGlvbiBvZiBDVFIxIGZyb20gZmlyc3QgYmxvY2sgQjAgaW5mb3JtYXRpb24gOgogICAgICAgICAgICAgIGVxdWFsIHRvIEIwIHdpdGggZmlyc3QgNSBiaXRzIHplcm9lZCBhbmQgbW9zdCBzaWduaWZpY2FudCBiaXRzIHN0b3Jpbmcgb2N0ZXQKICAgICAgICAgICAgICBsZW5ndGggb2YgUCBhbHNvIHplcm9lZCwgdGhlbiBpbmNyZW1lbnRlZCBieSBvbmUgKCBzZWUgYmVsb3cgVGFibGUgNCkKICAgICAgICAgICAgKCsrKykgR2VuZXJhdGlvbiBvZiBDVFIwOiBzYW1lIGFzIENUUjEgd2l0aCBiaXRbMF0gc2V0IHRvIHplcm8uCgogICAgICAoIykgIEZvdXIgcGhhc2VzIGFyZSBwZXJmb3JtZWQgaW4gQ0NNIGZvciBDUllQMSBJUDoKICAgICAgICAgKCMjKSBJbml0IHBoYXNlOiBJUCBwcmVwYXJlcyB0aGUgR0NNIGhhc2ggc3Via2V5IChIKSBhbmQgZG8gdGhlIElWIHByb2Nlc3NpbmcKICAgICAgICAgKCMjKSBIZWFkZXIgcGhhc2U6IElQIHByb2Nlc3NlcyB0aGUgQWRkaXRpb25hbCBBdXRoZW50aWNhdGVkIERhdGEgKEFBRCksIHdpdGggaGFzaAogICAgICAgICAgY29tcHV0YXRpb24gb25seS4KICAgICAgICAgKCMjKSBQYXlsb2FkIHBoYXNlOiBJUCBwcm9jZXNzZXMgdGhlIHBsYWludGV4dCAoUCkgd2l0aCBoYXNoIGNvbXB1dGF0aW9uICsga2V5c3RyZWFtCiAgICAgICAgICBlbmNyeXB0aW9uICsgZGF0YSBYT1JpbmcuIEl0IHdvcmtzIGluIGEgc2ltaWxhciB3YXkgZm9yIGNpcGhlcnRleHQgKEMpLgogICAgICAgICAoIyMpIEZpbmFsIHBoYXNlOiBJUCBnZW5lcmF0ZXMgdGhlIGF1dGhlbnRpY2F0ZWQgdGFnIChUKSB1c2luZyB0aGUgbGFzdCBibG9jayBvZiBkYXRhLgogICAgICAgICBIQUxfQ1JZUEV4X0FFU0NDTV9HZW5lcmF0ZUF1dGhUQUcgQVBJIHVzZWQgaW4gdGhpcyBwaGFzZSB0byBnZW5lcmF0ZSA0IHdvcmRzIHdoaWNoIGNvcnJlc3BvbmQgdG8gdGhlIFRhZy4KICAgICAgICAgdXNlciBzaG91bGQgY29uc2lkZXIgb25seSBwYXJ0IG9mIHRoaXMgNCB3b3JkcywgaWYgVGFnIGxlbmd0aCBpcyBsZXNzIHRoYW4gMTI4IGJpdHMKCiAgKioqIENhbGxiYWNrIHJlZ2lzdHJhdGlvbiAqKioKICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PQoKICBbLi5dCiAgVGhlIGNvbXBpbGF0aW9uIGRlZmluZSAgVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyB3aGVuIHNldCB0byAxCiAgYWxsb3dzIHRoZSB1c2VyIHRvIGNvbmZpZ3VyZSBkeW5hbWljYWxseSB0aGUgZHJpdmVyIGNhbGxiYWNrcy4KICBVc2UgRnVuY3Rpb25zIEByZWYgSEFMX0NSWVBfUmVnaXN0ZXJDYWxsYmFjaygpIG9yIEhBTF9DUllQX1JlZ2lzdGVyWFhYQ2FsbGJhY2soKQogIHRvIHJlZ2lzdGVyIGFuIGludGVycnVwdCBjYWxsYmFjay4KCiAgWy4uXQogIEZ1bmN0aW9uIEByZWYgSEFMX0NSWVBfUmVnaXN0ZXJDYWxsYmFjaygpIGFsbG93cyB0byByZWdpc3RlciBmb2xsb3dpbmcgY2FsbGJhY2tzOgogICAgKCspIEluQ3BsdENhbGxiYWNrICAgICA6ICBJbnB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAgICgrKSBPdXRDcGx0Q2FsbGJhY2sgICAgOiBPdXRwdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2suCiAgICAoKykgRXJyb3JDYWxsYmFjayAgICAgIDogY2FsbGJhY2sgZm9yIGVycm9yIGRldGVjdGlvbi4KICAgICgrKSBNc3BJbml0Q2FsbGJhY2sgICAgOiBDUllQIE1zcEluaXQuCiAgICAoKykgTXNwRGVJbml0Q2FsbGJhY2sgIDogQ1JZUCBNc3BEZUluaXQuCiAgVGhpcyBmdW5jdGlvbiB0YWtlcyBhcyBwYXJhbWV0ZXJzIHRoZSBIQUwgcGVyaXBoZXJhbCBoYW5kbGUsIHRoZSBDYWxsYmFjayBJRAogIGFuZCBhIHBvaW50ZXIgdG8gdGhlIHVzZXIgY2FsbGJhY2sgZnVuY3Rpb24uCgogIFsuLl0KICBVc2UgZnVuY3Rpb24gQHJlZiBIQUxfQ1JZUF9VblJlZ2lzdGVyQ2FsbGJhY2soKSB0byByZXNldCBhIGNhbGxiYWNrIHRvIHRoZSBkZWZhdWx0CiAgd2VhayBmdW5jdGlvbi4KICBAcmVmIEhBTF9DUllQX1VuUmVnaXN0ZXJDYWxsYmFjaygpIHRha2VzIGFzIHBhcmFtZXRlcnMgdGhlIEhBTCBwZXJpcGhlcmFsIGhhbmRsZSwKICBhbmQgdGhlIENhbGxiYWNrIElELgogIFRoaXMgZnVuY3Rpb24gYWxsb3dzIHRvIHJlc2V0IGZvbGxvd2luZyBjYWxsYmFja3M6CiAgICAoKykgSW5DcGx0Q2FsbGJhY2sgICAgIDogIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrLgogICAgKCspIE91dENwbHRDYWxsYmFjayAgICA6IE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAgICgrKSBFcnJvckNhbGxiYWNrICAgICAgOiBjYWxsYmFjayBmb3IgZXJyb3IgZGV0ZWN0aW9uLgogICAgKCspIE1zcEluaXRDYWxsYmFjayAgICA6IENSWVAgTXNwSW5pdC4KICAgICgrKSBNc3BEZUluaXRDYWxsYmFjayAgOiBDUllQIE1zcERlSW5pdC4KCiAgWy4uXQogIEJ5IGRlZmF1bHQsIGFmdGVyIHRoZSBAcmVmIEhBTF9DUllQX0luaXQoKSBhbmQgd2hlbiB0aGUgc3RhdGUgaXMgSEFMX0NSWVBfU1RBVEVfUkVTRVQKICBhbGwgY2FsbGJhY2tzIGFyZSBzZXQgdG8gdGhlIGNvcnJlc3BvbmRpbmcgd2VhayBmdW5jdGlvbnMgOgogIGV4YW1wbGVzIEByZWYgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soKSAsIEByZWYgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKCkuCiAgRXhjZXB0aW9uIGRvbmUgZm9yIE1zcEluaXQgYW5kIE1zcERlSW5pdCBmdW5jdGlvbnMgdGhhdCBhcmUKICByZXNldCB0byB0aGUgbGVnYWN5IHdlYWsgZnVuY3Rpb24gaW4gdGhlIEByZWYgSEFMX0NSWVBfSW5pdCgpLyBAcmVmIEhBTF9DUllQX0RlSW5pdCgpIG9ubHkgd2hlbgogIHRoZXNlIGNhbGxiYWNrcyBhcmUgbnVsbCAobm90IHJlZ2lzdGVyZWQgYmVmb3JlaGFuZCkuCiAgaWYgbm90LCBNc3BJbml0IG9yIE1zcERlSW5pdCBhcmUgbm90IG51bGwsIHRoZSBAcmVmIEhBTF9DUllQX0luaXQoKSAvIEByZWYgSEFMX0NSWVBfRGVJbml0KCkKICBrZWVwIGFuZCB1c2UgdGhlIHVzZXIgTXNwSW5pdC9Nc3BEZUluaXQgZnVuY3Rpb25zIChyZWdpc3RlcmVkIGJlZm9yZWhhbmQpCgogIFsuLl0KICBDYWxsYmFja3MgY2FuIGJlIHJlZ2lzdGVyZWQvdW5yZWdpc3RlcmVkIGluIEhBTF9DUllQX1NUQVRFX1JFQURZIHN0YXRlIG9ubHkuCiAgRXhjZXB0aW9uIGRvbmUgTXNwSW5pdC9Nc3BEZUluaXQgY2FsbGJhY2tzIHRoYXQgY2FuIGJlIHJlZ2lzdGVyZWQvdW5yZWdpc3RlcmVkCiAgaW4gSEFMX0NSWVBfU1RBVEVfUkVBRFkgb3IgSEFMX0NSWVBfU1RBVEVfUkVTRVQgc3RhdGUsCiAgdGh1cyByZWdpc3RlcmVkICh1c2VyKSBNc3BJbml0L0RlSW5pdCBjYWxsYmFja3MgY2FuIGJlIHVzZWQgZHVyaW5nIHRoZSBJbml0L0RlSW5pdC4KICBJbiB0aGF0IGNhc2UgZmlyc3QgcmVnaXN0ZXIgdGhlIE1zcEluaXQvTXNwRGVJbml0IHVzZXIgY2FsbGJhY2tzCiAgdXNpbmcgQHJlZiBIQUxfQ1JZUF9SZWdpc3RlckNhbGxiYWNrKCkgYmVmb3JlIGNhbGxpbmcgQHJlZiBIQUxfQ1JZUF9EZUluaXQoKQogIG9yIEByZWYgSEFMX0NSWVBfSW5pdCgpIGZ1bmN0aW9uLgoKICBbLi5dCiAgV2hlbiBUaGUgY29tcGlsYXRpb24gZGVmaW5lIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgaXMgc2V0IHRvIDAgb3IKICBub3QgZGVmaW5lZCwgdGhlIGNhbGxiYWNrIHJlZ2lzdHJhdGlvbiBmZWF0dXJlIGlzIG5vdCBhdmFpbGFibGUgYW5kIGFsbCBjYWxsYmFja3MKICBhcmUgc2V0IHRvIHRoZSBjb3JyZXNwb25kaW5nIHdlYWsgZnVuY3Rpb25zLgoKICBAZW5kdmVyYmF0aW0KCiAgVGFibGUgMS4gSW5pdGlhbCBDb3VudGVyIEJsb2NrIChJQ0IpCiAgICAgICAgICArLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKICAgICAgICAgIHwgICAgICAgSW5pdGlhbGl6YXRpb24gdmVjdG9yIChJVikgICAgICB8ICBDb3VudGVyICAgICAgfAogICAgICAgICAgfC0tLS0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLXwtLS0tLS0tLS18CiAgICAgICAgIDEyNyAgICAgICAgICAgICAgOTUgICAgICAgICAgICAgICAgNjMgICAgICAgICAgICAzMSAgICAgICAwCgoKICAgICAgICAgICAgICBCaXQgTnVtYmVyICAgIFJlZ2lzdGVyICAgICAgICAgICBDb250ZW50cwogICAgICAgICAgICAgIC0tLS0tLS0tLS0gICAtLS0tLS0tLS0tLS0tLS0gICAgICAgLS0tLS0tLS0tLS0KICAgICAgICAgICAgICAxMjcgLi4uOTYgICAgQ1JZUF9JVjFSWzMxOjBdICAgICBJQ0JbMTI3Ojk2XQogICAgICAgICAgICAgIDk1ICAuLi42NCAgICBDUllQX0lWMUxbMzE6MF0gICAgIEIwWzk1OjY0XQogICAgICAgICAgICAgIDYzIC4uLiAzMiAgICBDUllQX0lWMFJbMzE6MF0gICAgIElDQls2MzozMl0KICAgICAgICAgICAgICAzMSAuLi4gMCAgICAgQ1JZUF9JVjBMWzMxOjBdICAgICBJQ0JbMzE6MF0sIHdoZXJlIDMyLWJpdCBjb3VudGVyPSAweDIKCiAgVGFibGUgMi4gIEdDTSBsYXN0IGJsb2NrIGRlZmluaXRpb24KCiAgICAgICAgICArLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKICAgICAgICAgIHwgIEJpdFswXSAgIHwgIEJpdFszMl0gICAgICAgICAgIHwgIEJpdFs2NF0gIHwgQml0Wzk2XSAgICAgICAgICAgICAgfAogICAgICAgICAgfC0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CiAgICAgICAgICB8ICAgMHgwICAgICB8IEhlYWRlciBsZW5ndGhbMzE6MF18ICAgICAweDAgICB8IFBheWxvYWQgbGVuZ3RoWzMxOjBdIHwKICAgICAgICAgIHwtLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAoKICBUYWJsZSAzLiBCMCBibG9jawogICAgICAgICAgICAgICAgT2N0ZXQgTnVtYmVyICAgQ29udGVudHMKICAgICAgICAgICAgICAgIC0tLS0tLS0tLS0tLSAgIC0tLS0tLS0tLQogICAgICAgICAgICAgICAgMCAgICAgICAgICAgICAgRmxhZ3MKICAgICAgICAgICAgICAgIDEgLi4uIDE1LXEgICAgIE5vbmNlIE4KICAgICAgICAgICAgICAgIDE2LXEgLi4uIDE1ICAgIFEKCiAgICAgICAgICAgIHRoZSBGbGFncyBmaWVsZCBpcyBmb3JtYXR0ZWQgYXMgZm9sbG93czoKCiAgICAgICAgICAgICAgICBCaXQgTnVtYmVyICAgQ29udGVudHMKICAgICAgICAgICAgICAgIC0tLS0tLS0tLS0gICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICAgICAgICAgICAgICA3ICAgICAgICAgICAgUmVzZXJ2ZWQgKGFsd2F5cyB6ZXJvKQogICAgICAgICAgICAgICAgNiAgICAgICAgICAgIEFkYXRhCiAgICAgICAgICAgICAgICA1IC4uLiAzICAgICAgKHQtMikvMgogICAgICAgICAgICAgICAgMiAuLi4gMCAgICAgIFtxLTFdMwoKIFRhYmxlIDQuIENUUnggYmxvY2sKICAgICAgICAgICAgICAgIEJpdCBOdW1iZXIgICAgUmVnaXN0ZXIgICAgICAgICAgIENvbnRlbnRzCiAgICAgICAgICAgICAgICAtLS0tLS0tLS0tICAgLS0tLS0tLS0tLS0tLS0tICAgICAgIC0tLS0tLS0tLS0tCiAgICAgICAgICAgICAgICAxMjcgLi4uOTYgICAgQ1JZUF9JVjFSWzMxOjBdICAgICBCMFsxMjc6OTZdLCB3aGVyZSBRIGxlbmd0aCBiaXRzIGFyZSBzZXQgdG8gMCwgZXhjZXB0IGZvcgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYml0IDAgdGhhdCBpcyBzZXQgdG8gMQogICAgICAgICAgICAgICAgOTUgIC4uLjY0ICAgIENSWVBfSVYxTFszMTowXSAgICAgQjBbOTU6NjRdCiAgICAgICAgICAgICAgICA2MyAuLi4gMzIgICAgQ1JZUF9JVjBSWzMxOjBdICAgICBCMFs2MzozMl0KICAgICAgICAgICAgICAgIDMxIC4uLiAwICAgICBDUllQX0lWMExbMzE6MF0gICAgIEIwWzMxOjBdLCB3aGVyZSBmbGFnIGJpdHMgc2V0IHRvIDAKCgogICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogICogQGF0dGVudGlvbgogICoKICAqIDxoMj48Y2VudGVyPiZjb3B5OyBDb3B5cmlnaHQgKGMpIDIwMTcgU1RNaWNyb2VsZWN0cm9uaWNzLgogICogQWxsIHJpZ2h0cyByZXNlcnZlZC48L2NlbnRlcj48L2gyPgogICoKICAqIFRoaXMgc29mdHdhcmUgY29tcG9uZW50IGlzIGxpY2Vuc2VkIGJ5IFNUIHVuZGVyIEJTRCAzLUNsYXVzZSBsaWNlbnNlLAogICogdGhlICJMaWNlbnNlIjsgWW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZQogICogTGljZW5zZS4gWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0OgogICogICAgICAgICAgICAgICAgICAgICAgICBvcGVuc291cmNlLm9yZy9saWNlbnNlcy9CU0QtMy1DbGF1c2UKICAqCiAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAgKi8KCi8qIEluY2x1ZGVzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCiNpbmNsdWRlICJzdG0zMmg3eHhfaGFsLmgiCgovKiogQGFkZHRvZ3JvdXAgU1RNMzJIN3h4X0hBTF9Ecml2ZXIKICAqIEB7CiAgKi8KCiNpZiBkZWZpbmVkIChDUllQKQoKLyoqIEBkZWZncm91cCBDUllQIENSWVAKICAqIEBicmllZiBDUllQIEhBTCBtb2R1bGUgZHJpdmVyLgogICogQHsKICAqLwoKCiNpZmRlZiBIQUxfQ1JZUF9NT0RVTEVfRU5BQkxFRAoKLyogUHJpdmF0ZSB0eXBlZGVmIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyogUHJpdmF0ZSBkZWZpbmUgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyoqIEBhZGR0b2dyb3VwIENSWVBfUHJpdmF0ZV9EZWZpbmVzCiAgKiBAewogICovCiNkZWZpbmUgQ1JZUF9USU1FT1VUX0tFWVBSRVBBUkFUSU9OICAgICAgODJVICAgICAgICAgLypUaGUgbGF0ZW5jeSBvZiBrZXkgcHJlcGFyYXRpb24gb3BlcmF0aW9uIGlzIDgyIGNsb2NrIGN5Y2xlcy4qLwojZGVmaW5lIENSWVBfVElNRU9VVF9HQ01DQ01JTklUUEhBU0UgICAgIDI5OVUgICAgICAgIC8qICBUaGUgbGF0ZW5jeSBvZiAgR0NNL0NDTSBpbml0IHBoYXNlIHRvIHByZXBhcmUgaGFzaCBzdWJrZXkgaXMgMjk5IGNsb2NrIGN5Y2xlcy4qLwojZGVmaW5lIENSWVBfVElNRU9VVF9HQ01DQ01IRUFERVJQSEFTRSAgIDI5MFUgICAgICAgIC8qICBUaGUgbGF0ZW5jeSBvZiAgR0NNL0NDTSBoZWFkZXIgcGhhc2UgaXMgMjkwIGNsb2NrIGN5Y2xlcy4qLwoKI2RlZmluZSAgQ1JZUF9QSEFTRV9SRUFEWSAgICAgICAgICAgICAgICAweDAwMDAwMDAxVSAvKiE8IENSWVAgcGVyaXBoZXJhbCBpcyByZWFkeSBmb3IgaW5pdGlhbGl6YXRpb24uICovCiNkZWZpbmUgIENSWVBfUEhBU0VfUFJPQ0VTUyAgICAgICAgICAgICAgMHgwMDAwMDAwMlUgLyohPCBDUllQIHBlcmlwaGVyYWwgaXMgaW4gcHJvY2Vzc2luZyBwaGFzZSAqLwoKI2RlZmluZSBDUllQX1BIQVNFX0lOSVQgICAgICAgICAgICAgICAgICAweDAwMDAwMDAwVSAgICAgICAgICAgICAvKiE8IEdDTS9HTUFDIChvciBDQ00pIGluaXQgcGhhc2UgKi8KI2RlZmluZSBDUllQX1BIQVNFX0hFQURFUiAgICAgICAgICAgICAgICBDUllQX0NSX0dDTV9DQ01QSF8wICAgICAvKiE8IEdDTS9HTUFDIG9yIENDTSBoZWFkZXIgcGhhc2UgKi8KI2RlZmluZSBDUllQX1BIQVNFX1BBWUxPQUQgICAgICAgICAgICAgICBDUllQX0NSX0dDTV9DQ01QSF8xICAgICAvKiE8IEdDTSgvQ0NNKSBwYXlsb2FkIHBoYXNlICAgICAgKi8KI2RlZmluZSBDUllQX1BIQVNFX0ZJTkFMICAgICAgICAgICAgICAgICBDUllQX0NSX0dDTV9DQ01QSCAgICAgICAvKiE8IEdDTS9HTUFDIG9yIENDTSAgZmluYWwgcGhhc2UgKi8KI2RlZmluZSBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCAgICAgICAweDAwMDAwMDAwVSAgICAgICAgICAgICAvKiE8IEVuY3J5cHRpb24gbW9kZSAgICovCiNkZWZpbmUgQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQgICAgICAgQ1JZUF9DUl9BTEdPRElSICAgICAgICAgLyohPCBEZWNyeXB0aW9uICAgICAgICAqLwoKCi8qICBDVFIxIGluZm9ybWF0aW9uIHRvIHVzZSBpbiBDQ00gYWxnb3JpdGhtICovCiNkZWZpbmUgQ1JZUF9DQ01fQ1RSMV8wICAgICAgICAgICAgICAgICAgMHgwN0ZGRkZGRlUKI2RlZmluZSBDUllQX0NDTV9DVFIxXzEgICAgICAgICAgICAgICAgICAweEZGRkZGRjAwVQojZGVmaW5lIENSWVBfQ0NNX0NUUjFfMiAgICAgICAgICAgICAgICAgIDB4MDAwMDAwMDFVCgovKioKICAqIEB9CiAgKi8KCgovKiBQcml2YXRlIG1hY3JvIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiogQGFkZHRvZ3JvdXAgQ1JZUF9Qcml2YXRlX01hY3JvcwogICogQHsKICAqLwoKI2RlZmluZSBDUllQX1NFVF9QSEFTRShfX0hBTkRMRV9fLCBfX1BIQVNFX18pICBkb3soX19IQU5ETEVfXyktPkluc3RhbmNlLT5DUiAmPSAodWludDMyX3QpKH5DUllQX0NSX0dDTV9DQ01QSCk7XAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChfX0hBTkRMRV9fKS0+SW5zdGFuY2UtPkNSIHw9ICh1aW50MzJfdCkoX19QSEFTRV9fKTtcCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9d2hpbGUoMCkKCiNkZWZpbmUgSEFMX0NSWVBfRklGT19GTFVTSChfX0hBTkRMRV9fKSAoKF9fSEFORExFX18pLT5JbnN0YW5jZS0+Q1IgfD0gIENSWVBfQ1JfRkZMVVNIKQoKCi8qKgogICogQH0KICAqLwoKLyogUHJpdmF0ZSBzdHJ1Y3QgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi8qIFByaXZhdGUgdmFyaWFibGVzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi8qIFByaXZhdGUgZnVuY3Rpb24gcHJvdG90eXBlcyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi8qKiBAYWRkdG9ncm91cCBDUllQX1ByaXZhdGVfRnVuY3Rpb25zX3Byb3RvdHlwZXMKICAqIEB7CiAgKi8KCnN0YXRpYyB2b2lkIENSWVBfU2V0RE1BQ29uZmlnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IGlucHV0YWRkciwgdWludDE2X3QgU2l6ZSwgdWludDMyX3Qgb3V0cHV0YWRkcik7CnN0YXRpYyB2b2lkIENSWVBfRE1BSW5DcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKTsKc3RhdGljIHZvaWQgQ1JZUF9ETUFPdXRDcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKTsKc3RhdGljIHZvaWQgQ1JZUF9ETUFFcnJvcihETUFfSGFuZGxlVHlwZURlZiAqaGRtYSk7CnN0YXRpYyB2b2lkIENSWVBfU2V0S2V5KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IEtleVNpemUpOwpzdGF0aWMgdm9pZCBDUllQX0FFU19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgdm9pZCBDUllQX0dDTUNDTV9TZXRQYXlsb2FkUGhhc2VfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyB2b2lkIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9HQ01DQ01fU2V0SGVhZGVyUGhhc2VfRE1BKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwojaWYgIWRlZmluZWQgKENSWVBfVkVSXzJfMikKc3RhdGljIHZvaWQgQ1JZUF9Xb3JrYXJvdW5kKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwojZW5kaWYgLypFbmQgb2Ygbm90IGRlZmluZWQgQ1JZUF9WRVJfMl8yKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTR0NNX1Byb2Nlc3NfRE1BKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNHQ01fUHJvY2Vzc19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTR0NNX1Byb2Nlc3MoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU0NDTV9Qcm9jZXNzKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNDQ01fUHJvY2Vzc19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTQ0NNX1Byb2Nlc3NfRE1BKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgdm9pZCBDUllQX0FFU19Qcm9jZXNzRGF0YShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwdCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19FbmNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0RlY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19FbmNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyB2b2lkIENSWVBfVERFU19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uSUZFTUZsYWcoY29uc3QgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1dhaXRPbkJVU1lGbGFnKGNvbnN0IENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9XYWl0T25PRk5FRmxhZyhjb25zdCBDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfVERFU19Qcm9jZXNzKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwoKLyoqCiAgKiBAfQogICovCgovKiBFeHBvcnRlZCBmdW5jdGlvbnMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KCi8qKiBAZGVmZ3JvdXAgQ1JZUF9FeHBvcnRlZF9GdW5jdGlvbnMgQ1JZUCBFeHBvcnRlZCBGdW5jdGlvbnMKICAqIEB7CiAgKi8KCgovKiogQGRlZmdyb3VwIENSWVBfRXhwb3J0ZWRfRnVuY3Rpb25zX0dyb3VwMSBJbml0aWFsaXphdGlvbiBhbmQgZGUtaW5pdGlhbGl6YXRpb24gZnVuY3Rpb25zCiAgKiAgQGJyaWVmICAgIENSWVAgIEluaXRpYWxpemF0aW9uIGFuZCBDb25maWd1cmF0aW9uIGZ1bmN0aW9ucy4KICAqCkB2ZXJiYXRpbQogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICAgICAjIyMjIyBJbml0aWFsaXphdGlvbiwgZGUtaW5pdGlhbGl6YXRpb24gYW5kIFNldCBhbmQgR2V0IGNvbmZpZ3VyYXRpb24gZnVuY3Rpb25zICMjIyMjCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgWy4uXSAgVGhpcyBzZWN0aW9uIHByb3ZpZGVzIGZ1bmN0aW9ucyBhbGxvd2luZyB0bzoKICAgICAgKCspIEluaXRpYWxpemUgdGhlIENSWVAKICAgICAgKCspIERlSW5pdGlhbGl6ZSB0aGUgQ1JZUAogICAgICAoKykgSW5pdGlhbGl6ZSB0aGUgQ1JZUCBNU1AKICAgICAgKCspIERlSW5pdGlhbGl6ZSB0aGUgQ1JZUCBNU1AKICAgICAgKCspIGNvbmZpZ3VyZSBDUllQIChIQUxfQ1JZUF9TZXRDb25maWcpIHdpdGggdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzIGluIHRoZSBDUllQX0NvbmZpZ1R5cGVEZWYKICAgICAgICAgIFBhcmFtZXRlcnMgd2hpY2ggYXJlIGNvbmZpZ3VyZWQgaW4gVGhpcyBzZWN0aW9uIGFyZSA6CiAgICAgICAgICAoKyspIEtleSBzaXplCiAgICAgICAgICAoKyspIERhdGEgVHlwZSA6IDMyLDE2LCA4IG9yIDFiaXQKICAgICAgICAgICgrKykgQWxnb01vZGUgOiBmb3IgQ1JZUDEgSVAKICAgICAgICAgICAgICAgICBFQ0IgYW5kIENCQyBpbiBERVMvVERFUyBTdGFuZGFyZAogICAgICAgICAgICAgICAgIEVDQixDQkMsQ1RSLEdDTS9HTUFDIGFuZCBDQ00gaW4gQUVTIFN0YW5kYXJkLgogICAgICAoKykgR2V0IENSWVAgY29uZmlndXJhdGlvbiAoSEFMX0NSWVBfR2V0Q29uZmlnKSBmcm9tIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycyBpbiB0aGUgQ1JZUF9IYW5kbGVUeXBlRGVmCgoKQGVuZHZlcmJhdGltCiAgKiBAewogICovCgoKLyoqCiAgKiBAYnJpZWYgIEluaXRpYWxpemVzIHRoZSBDUllQIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkCiAgKiAgICAgICAgIHBhcmFtZXRlcnMgaW4gdGhlIENSWVBfQ29uZmlnVHlwZURlZiBhbmQgY3JlYXRlcyB0aGUgYXNzb2NpYXRlZCBoYW5kbGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfSW5pdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgLyogQ2hlY2sgdGhlIENSWVAgaGFuZGxlIGFsbG9jYXRpb24gKi8KICBpZiAoaGNyeXAgPT0gTlVMTCkKICB7CiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KCiAgLyogQ2hlY2sgcGFyYW1ldGVycyAqLwogIGFzc2VydF9wYXJhbShJU19DUllQX0tFWVNJWkUoaGNyeXAtPkluaXQuS2V5U2l6ZSkpOwogIGFzc2VydF9wYXJhbShJU19DUllQX0RBVEFUWVBFKGhjcnlwLT5Jbml0LkRhdGFUeXBlKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfQUxHT1JJVEhNKGhjcnlwLT5Jbml0LkFsZ29yaXRobSkpOwogIGFzc2VydF9wYXJhbShJU19DUllQX0lOSVQoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwKSk7CgojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFU0VUKQogIHsKICAgIC8qIEFsbG9jYXRlIGxvY2sgcmVzb3VyY2UgYW5kIGluaXRpYWxpemUgaXQgKi8KICAgIGhjcnlwLT5Mb2NrID0gSEFMX1VOTE9DS0VEOwoKICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayAgPSBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjazsgIC8qIExlZ2FjeSB3ZWFrICBJbkNwbHRDYWxsYmFjayAgKi8KICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2s7IC8qIExlZ2FjeSB3ZWFrIE91dENwbHRDYWxsYmFjayAgKi8KICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrICAgPSBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrOyAgIC8qIExlZ2FjeSB3ZWFrIEVycm9yQ2FsbGJhY2sgICAgKi8KCiAgICBpZiAoaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9PSBOVUxMKQogICAgewogICAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwSW5pdDsgLyogTGVnYWN5IHdlYWsgTXNwSW5pdCAgKi8KICAgIH0KCiAgICAvKiBJbml0IHRoZSBsb3cgbGV2ZWwgaGFyZHdhcmUgKi8KICAgIGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2soaGNyeXApOwogIH0KI2Vsc2UKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFU0VUKQogIHsKICAgIC8qIEFsbG9jYXRlIGxvY2sgcmVzb3VyY2UgYW5kIGluaXRpYWxpemUgaXQgKi8KICAgIGhjcnlwLT5Mb2NrID0gSEFMX1VOTE9DS0VEOwoKICAgIC8qIEluaXQgdGhlIGxvdyBsZXZlbCBoYXJkd2FyZSAqLwogICAgSEFMX0NSWVBfTXNwSW5pdChoY3J5cCk7CiAgfQojZW5kaWYgLyogKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MpICovCgogIC8qIFNldCB0aGUga2V5IHNpemUoVGhpcyBiaXQgZmllbGQgaXMgkWRvbpJ0IGNhcmWSIGluIHRoZSBERVMgb3IgVERFUyBtb2RlcykgZGF0YSB0eXBlIGFuZCBBbGdvcml0aG0gKi8KICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfREFUQVRZUEUgfCBDUllQX0NSX0tFWVNJWkUgfCBDUllQX0NSX0FMR09NT0RFLAogICAgICAgICAgICAgaGNyeXAtPkluaXQuRGF0YVR5cGUgfCBoY3J5cC0+SW5pdC5LZXlTaXplIHwgaGNyeXAtPkluaXQuQWxnb3JpdGhtKTsKI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgLyogUmVhZCBEZXZpY2UgSUQgdG8gaW5kaWNhdGUgQ1JZUDEgSVAgVmVyc2lvbiAqLwogIGhjcnlwLT5WZXJzaW9uID0gSEFMX0dldFJFVklEKCk7CiNlbmRpZiAvKkVuZCBvZiBub3QgZGVmaW5lZCBDUllQX1ZFUl8yXzIqLwogIC8qIFJlc2V0IEVycm9yIENvZGUgZmllbGQgKi8KICBoY3J5cC0+RXJyb3JDb2RlID0gSEFMX0NSWVBfRVJST1JfTk9ORTsKCiAgLyogUmVzZXQgcGVyaXBoZXJhbCBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gZmxhZyAqLwogIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDBVOwogIAogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBTZXQgdGhlIGRlZmF1bHQgQ1JZUCBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUkVBRFk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgRGUtSW5pdGlhbGl6ZXMgdGhlIENSWVAgcGVyaXBoZXJhbC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9EZUluaXQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIENoZWNrIHRoZSBDUllQIGhhbmRsZSBhbGxvY2F0aW9uICovCiAgaWYgKGhjcnlwID09IE5VTEwpCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIFNldCB0aGUgZGVmYXVsdCBDUllQIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9SRUFEWTsKCiAgLyogUmVzZXQgQ3J5cEluQ291bnQgYW5kIENyeXBPdXRDb3VudCAqLwogIGhjcnlwLT5DcnlwSW5Db3VudCA9IDA7CiAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IDA7CiAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDA7CgogIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICBpZiAoaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID09IE5VTEwpCiAgewogICAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwRGVJbml0OyAvKiBMZWdhY3kgd2VhayBNc3BEZUluaXQgICovCiAgfQogIC8qIERlSW5pdCB0aGUgbG93IGxldmVsIGhhcmR3YXJlICovCiAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrKGhjcnlwKTsKCiNlbHNlCiAgLyogRGVJbml0IHRoZSBsb3cgbGV2ZWwgaGFyZHdhcmU6IENMT0NLLCBOVklDLiovCiAgSEFMX0NSWVBfTXNwRGVJbml0KGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KCiAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVTRVQ7CgogIC8qIFJlbGVhc2UgTG9jayAqLwogIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgQ29uZmlndXJlIHRoZSBDUllQIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkCiAgKiAgICAgICAgIHBhcmFtZXRlcnMgaW4gdGhlIENSWVBfQ29uZmlnVHlwZURlZgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUKICAqIEBwYXJhbSAgcENvbmY6IHBvaW50ZXIgdG8gYSBDUllQX0NvbmZpZ1R5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9TZXRDb25maWcoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgQ1JZUF9Db25maWdUeXBlRGVmICpwQ29uZikKewogIC8qIENoZWNrIHRoZSBDUllQIGhhbmRsZSBhbGxvY2F0aW9uICovCiAgaWYgKChoY3J5cCA9PSBOVUxMKSB8fCAocENvbmYgPT0gTlVMTCkpCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIENoZWNrIHBhcmFtZXRlcnMgKi8KICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9LRVlTSVpFKHBDb25mLT5LZXlTaXplKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfREFUQVRZUEUocENvbmYtPkRhdGFUeXBlKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfQUxHT1JJVEhNKHBDb25mLT5BbGdvcml0aG0pKTsKCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qIFNldCAgQ1JZUCBwYXJhbWV0ZXJzICAqLwogICAgaGNyeXAtPkluaXQuRGF0YVR5cGUgICAgID0gcENvbmYtPkRhdGFUeXBlOwogICAgaGNyeXAtPkluaXQucEtleSAgICAgICAgID0gcENvbmYtPnBLZXk7CiAgICBoY3J5cC0+SW5pdC5BbGdvcml0aG0gICAgPSBwQ29uZi0+QWxnb3JpdGhtOwogICAgaGNyeXAtPkluaXQuS2V5U2l6ZSAgICAgID0gcENvbmYtPktleVNpemU7CiAgICBoY3J5cC0+SW5pdC5wSW5pdFZlY3QgICAgPSBwQ29uZi0+cEluaXRWZWN0OwogICAgaGNyeXAtPkluaXQuSGVhZGVyICAgICAgID0gcENvbmYtPkhlYWRlcjsKICAgIGhjcnlwLT5Jbml0LkhlYWRlclNpemUgICA9IHBDb25mLT5IZWFkZXJTaXplOwogICAgaGNyeXAtPkluaXQuQjAgICAgICAgICAgID0gcENvbmYtPkIwOwogICAgaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdCA9IHBDb25mLT5EYXRhV2lkdGhVbml0OwoKICAgIC8qIFNldCB0aGUga2V5IHNpemUoVGhpcyBiaXQgZmllbGQgaXMgkWRvbpJ0IGNhcmWSIGluIHRoZSBERVMgb3IgVERFUyBtb2RlcykgZGF0YSB0eXBlLCBBbGdvTW9kZSBhbmQgb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0RBVEFUWVBFIHwgQ1JZUF9DUl9LRVlTSVpFIHwgQ1JZUF9DUl9BTEdPTU9ERSwKICAgICAgICAgICAgICAgaGNyeXAtPkluaXQuRGF0YVR5cGUgfCBoY3J5cC0+SW5pdC5LZXlTaXplIHwgaGNyeXAtPkluaXQuQWxnb3JpdGhtKTsKCiAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIFJlc2V0IEVycm9yIENvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgPSBIQUxfQ1JZUF9FUlJPUl9OT05FOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogU2V0IHRoZSBkZWZhdWx0IENSWVAgcGhhc2UgKi8KICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUkVBRFk7CgogICAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogICAgcmV0dXJuIEhBTF9PSzsKICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KfQoKLyoqCiAgKiBAYnJpZWYgIEdldCBDUllQIENvbmZpZ3VyYXRpb24gcGFyYW1ldGVycyBpbiBhc3NvY2lhdGVkIGhhbmRsZS4KICAqIEBwYXJhbSAgcENvbmY6IHBvaW50ZXIgdG8gYSBDUllQX0NvbmZpZ1R5cGVEZWYgc3RydWN0dXJlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfR2V0Q29uZmlnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIENSWVBfQ29uZmlnVHlwZURlZiAqcENvbmYpCnsKICAvKiBDaGVjayB0aGUgQ1JZUCBoYW5kbGUgYWxsb2NhdGlvbiAqLwogIGlmICgoaGNyeXAgPT0gTlVMTCkgfHwgKHBDb25mID09IE5VTEwpKQogIHsKICAgIHJldHVybiBIQUxfRVJST1I7CiAgfQoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogR2V0ICBDUllQIHBhcmFtZXRlcnMgICovCiAgICBwQ29uZi0+RGF0YVR5cGUgICAgICAgID0gaGNyeXAtPkluaXQuRGF0YVR5cGU7CiAgICBwQ29uZi0+cEtleSAgICAgICAgICAgID0gaGNyeXAtPkluaXQucEtleTsKICAgIHBDb25mLT5BbGdvcml0aG0gICAgICAgPSBoY3J5cC0+SW5pdC5BbGdvcml0aG07CiAgICBwQ29uZi0+S2V5U2l6ZSAgICAgICAgID0gaGNyeXAtPkluaXQuS2V5U2l6ZSA7CiAgICBwQ29uZi0+cEluaXRWZWN0ICAgICAgID0gaGNyeXAtPkluaXQucEluaXRWZWN0OwogICAgcENvbmYtPkhlYWRlciAgICAgICAgICA9IGhjcnlwLT5Jbml0LkhlYWRlciA7CiAgICBwQ29uZi0+SGVhZGVyU2l6ZSAgICAgID0gaGNyeXAtPkluaXQuSGVhZGVyU2l6ZTsKICAgIHBDb25mLT5CMCAgICAgICAgICAgICAgPSBoY3J5cC0+SW5pdC5CMDsKICAgIHBDb25mLT5EYXRhV2lkdGhVbml0ICAgID0gaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdDsKCiAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogICAgcmV0dXJuIEhBTF9PSzsKICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KfQovKioKICAqIEBicmllZiAgSW5pdGlhbGl6ZXMgdGhlIENSWVAgTVNQLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgTm9uZQogICovCl9fd2VhayB2b2lkIEhBTF9DUllQX01zcEluaXQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIFByZXZlbnQgdW51c2VkIGFyZ3VtZW50KHMpIGNvbXBpbGF0aW9uIHdhcm5pbmcgKi8KICBVTlVTRUQoaGNyeXApOwoKICAvKiBOT1RFIDogVGhpcyBmdW5jdGlvbiBTaG91bGQgbm90IGJlIG1vZGlmaWVkLCB3aGVuIHRoZSBjYWxsYmFjayBpcyBuZWVkZWQsCiAgICAgICAgICAgIHRoZSBIQUxfQ1JZUF9Nc3BJbml0IGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIERlSW5pdGlhbGl6ZXMgQ1JZUCBNU1AuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBOb25lCiAgKi8KX193ZWFrIHZvaWQgSEFMX0NSWVBfTXNwRGVJbml0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBQcmV2ZW50IHVudXNlZCBhcmd1bWVudChzKSBjb21waWxhdGlvbiB3YXJuaW5nICovCiAgVU5VU0VEKGhjcnlwKTsKCiAgLyogTk9URSA6IFRoaXMgZnVuY3Rpb24gU2hvdWxkIG5vdCBiZSBtb2RpZmllZCwgd2hlbiB0aGUgY2FsbGJhY2sgaXMgbmVlZGVkLAogICAgICAgICAgICB0aGUgSEFMX0NSWVBfTXNwRGVJbml0IGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCi8qKgogICogQGJyaWVmICBSZWdpc3RlciBhIFVzZXIgQ1JZUCBDYWxsYmFjawogICogICAgICAgICBUbyBiZSB1c2VkIGluc3RlYWQgb2YgdGhlIHdlYWsgcHJlZGVmaW5lZCBjYWxsYmFjawogICogQHBhcmFtIGhjcnlwIGNyeXAgaGFuZGxlCiAgKiBAcGFyYW0gQ2FsbGJhY2tJRCBJRCBvZiB0aGUgY2FsbGJhY2sgdG8gYmUgcmVnaXN0ZXJlZAogICogICAgICAgIFRoaXMgcGFyYW1ldGVyIGNhbiBiZSBvbmUgb2YgdGhlIGZvbGxvd2luZyB2YWx1ZXM6CiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfSU5QVVRfQ09NUExFVEVfQ0JfSUQgSW5wdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9PVVRQVVRfQ09NUExFVEVfQ0JfSUQgT3V0cHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrIElECiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfRVJST1JfQ0JfSUQgUnggSGFsZiBFcnJvciBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgTXNwSW5pdCBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX01TUERFSU5JVF9DQl9JRCBNc3BEZUluaXQgY2FsbGJhY2sgSUQKICAqIEBwYXJhbSBwQ2FsbGJhY2sgcG9pbnRlciB0byB0aGUgQ2FsbGJhY2sgZnVuY3Rpb24KICAqIEByZXR2YWwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfUmVnaXN0ZXJDYWxsYmFjayhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCBIQUxfQ1JZUF9DYWxsYmFja0lEVHlwZURlZiBDYWxsYmFja0lELAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBDUllQX0NhbGxiYWNrVHlwZURlZiBwQ2FsbGJhY2spCnsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CgogIGlmIChwQ2FsbGJhY2sgPT0gTlVMTCkKICB7CiAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfSU5WQUxJRF9DQUxMQkFDSzsKCiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogIF9fSEFMX0xPQ0soaGNyeXApOwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIHN3aXRjaCAoQ2FsbGJhY2tJRCkKICAgIHsKICAgICAgY2FzZSBIQUxfQ1JZUF9JTlBVVF9DT01QTEVURV9DQl9JRCA6CiAgICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrID0gcENhbGxiYWNrOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBIQUxfQ1JZUF9PVVRQVVRfQ09NUExFVEVfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX0VSUk9SX0NCX0lEIDoKICAgICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfTVNQSU5JVF9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfTVNQREVJTklUX0NCX0lEIDoKICAgICAgICBoY3J5cC0+TXNwRGVJbml0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0IDoKICAgICAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CiAgICAgICAgLyogUmV0dXJuIGVycm9yIHN0YXR1cyAqLwogICAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgICAgYnJlYWs7CiAgICB9CiAgfQogIGVsc2UgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRVNFVCkKICB7CiAgICBzd2l0Y2ggKENhbGxiYWNrSUQpCiAgICB7CiAgICAgIGNhc2UgSEFMX0NSWVBfTVNQSU5JVF9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfTVNQREVJTklUX0NCX0lEIDoKICAgICAgICBoY3J5cC0+TXNwRGVJbml0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0IDoKICAgICAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CiAgICAgICAgLyogUmV0dXJuIGVycm9yIHN0YXR1cyAqLwogICAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgICAgYnJlYWs7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfSU5WQUxJRF9DQUxMQkFDSzsKICAgIC8qIFJldHVybiBlcnJvciBzdGF0dXMgKi8KICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgfQoKICAvKiBSZWxlYXNlIExvY2sgKi8KICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICByZXR1cm4gc3RhdHVzOwp9CgovKioKICAqIEBicmllZiAgVW5yZWdpc3RlciBhbiBDUllQIENhbGxiYWNrCiAgKiAgICAgICAgIENSWVAgY2FsbGFiY2sgaXMgcmVkaXJlY3RlZCB0byB0aGUgd2VhayBwcmVkZWZpbmVkIGNhbGxiYWNrCiAgKiBAcGFyYW0gaGNyeXAgY3J5cCBoYW5kbGUKICAqIEBwYXJhbSBDYWxsYmFja0lEIElEIG9mIHRoZSBjYWxsYmFjayB0byBiZSB1bnJlZ2lzdGVyZWQKICAqICAgICAgICBUaGlzIHBhcmFtZXRlciBjYW4gYmUgb25lIG9mIHRoZSBmb2xsb3dpbmcgdmFsdWVzOgogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX0lOUFVUX0NPTVBMRVRFX0NCX0lEIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrIElECiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfT1VUUFVUX0NPTVBMRVRFX0NCX0lEIE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX0VSUk9SX0NCX0lEIFJ4IEhhbGYgRXJyb3IgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9NU1BJTklUX0NCX0lEIE1zcEluaXQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgTXNwRGVJbml0IGNhbGxiYWNrIElECiAgKiBAcmV0dmFsIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX1VuUmVnaXN0ZXJDYWxsYmFjayhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCBIQUxfQ1JZUF9DYWxsYmFja0lEVHlwZURlZiBDYWxsYmFja0lEKQp7CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogIF9fSEFMX0xPQ0soaGNyeXApOwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIHN3aXRjaCAoQ2FsbGJhY2tJRCkKICAgIHsKICAgICAgY2FzZSBIQUxfQ1JZUF9JTlBVVF9DT01QTEVURV9DQl9JRCA6CiAgICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrID0gSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2s7ICAvKiBMZWdhY3kgd2VhayAgSW5DcGx0Q2FsbGJhY2sgICovCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX09VVFBVVF9DT01QTEVURV9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk91dENwbHRDYWxsYmFjayA9IEhBTF9DUllQX091dENwbHRDYWxsYmFjazsgICAgICAgICAvKiBMZWdhY3kgd2VhayBPdXRDcGx0Q2FsbGJhY2sgICAgICAgKi8KICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfRVJST1JfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrID0gSEFMX0NSWVBfRXJyb3JDYWxsYmFjazsgICAgICAgICAgIC8qIExlZ2FjeSB3ZWFrIEVycm9yQ2FsbGJhY2sgICAgICAgICovCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9Nc3BJbml0OwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5Nc3BEZUluaXRDYWxsYmFjayA9IEhBTF9DUllQX01zcERlSW5pdDsKICAgICAgICBicmVhazsKCiAgICAgIGRlZmF1bHQgOgogICAgICAgIC8qIFVwZGF0ZSB0aGUgZXJyb3IgY29kZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfSU5WQUxJRF9DQUxMQkFDSzsKICAgICAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KICB9CiAgZWxzZSBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFU0VUKQogIHsKICAgIHN3aXRjaCAoQ2FsbGJhY2tJRCkKICAgIHsKICAgICAgY2FzZSBIQUxfQ1JZUF9NU1BJTklUX0NCX0lEIDoKICAgICAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwSW5pdDsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfTVNQREVJTklUX0NCX0lEIDoKICAgICAgICBoY3J5cC0+TXNwRGVJbml0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9Nc3BEZUluaXQ7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0IDoKICAgICAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CiAgICAgICAgLyogUmV0dXJuIGVycm9yIHN0YXR1cyAqLwogICAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgICAgYnJlYWs7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfSU5WQUxJRF9DQUxMQkFDSzs7CiAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogIH0KCiAgLyogUmVsZWFzZSBMb2NrICovCiAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgcmV0dXJuIHN0YXR1czsKfQojZW5kaWYgLyogVVNFX0hBTF9VQVJUX1JFR0lTVEVSX0NBTExCQUNLUyAqLwoKLyoqCiAgKiBAfQogICovCgovKiogQGRlZmdyb3VwIENSWVBfRXhwb3J0ZWRfRnVuY3Rpb25zX0dyb3VwMiAgRW5jcnlwdCBEZWNyeXB0IGZ1bmN0aW9ucwogICogIEBicmllZiAgIENSWVAgcHJvY2Vzc2luZyBmdW5jdGlvbnMuCiAgKgpAdmVyYmF0aW0KICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICAgICAgICAgICAgICAgICAgICAgICMjIyMjIEVuY3J5cHQgRGVjcnlwdCAgZnVuY3Rpb25zICMjIyMjCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICBbLi5dICBUaGlzIHNlY3Rpb24gcHJvdmlkZXMgQVBJIGFsbG93aW5nIHRvIEVuY3J5cHQvRGVjcnlwdCBEYXRhIGZvbGxvd2luZwogICAgICAgICAgU3RhbmRhcmQgREVTL1RERVMgb3IgQUVTLCBhbmQgQWxnb3JpdGhtIGNvbmZpZ3VyZWQgYnkgdGhlIHVzZXI6CiAgICAgICgrKSBTdGFuZGFyZCBERVMvVERFUyBvbmx5IHN1cHBvcnRlZCBieSBDUllQMSBJUCwgYmVsb3cgbGlzdCBvZiBBbGdvcml0aG0gc3VwcG9ydGVkIDoKICAgICAgICAgICAoKyspICBFbGVjdHJvbmljIENvZGUgQm9vayhFQ0IpCiAgICAgICAgICAgKCsrKSBDaXBoZXIgQmxvY2sgQ2hhaW5pbmcgKENCQykKICAgICAgKCspIFN0YW5kYXJkIEFFUyAgc3VwcG9ydGVkIGJ5IENSWVAxIElQICwgbGlzdCBvZiBBbGdvcml0aG0gc3VwcG9ydGVkOgogICAgICAgICAgICgrKykgRWxlY3Ryb25pYyBDb2RlIEJvb2soRUNCKQogICAgICAgICAgICgrKykgQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICAgICAgKCsrKSBDb3VudGVyIG1vZGUgKENUUikKICAgICAgICAgICAoKyspIENpcGhlciBCbG9jayBDaGFpbmluZyAoQ0JDKQogICAgICAgICAgICgrKykgQ291bnRlciBtb2RlIChDVFIpCiAgICAgICAgICAgKCsrKSBHYWxvaXMvY291bnRlciBtb2RlIChHQ00pCiAgICAgICAgICAgKCsrKSBDb3VudGVyIHdpdGggQ2lwaGVyIEJsb2NrIENoYWluaW5nLU1lc3NhZ2UoQ0NNKQogICAgWy4uXSAgVGhyZWUgcHJvY2Vzc2luZyBmdW5jdGlvbnMgYXJlIGF2YWlsYWJsZToKICAgICAgKCspIFBvbGxpbmcgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHQgJiBIQUxfQ1JZUF9EZWNyeXB0CiAgICAgICgrKSBJbnRlcnJ1cHQgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHRfSVQgJiBIQUxfQ1JZUF9EZWNyeXB0X0lUCiAgICAgICgrKSBETUEgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHRfRE1BICYgSEFMX0NSWVBfRGVjcnlwdF9ETUEKCkBlbmR2ZXJiYXRpbQogICogQHsKICAqLwoKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChwbGFpbnRleHQpCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBlaXRoZXIgaW4gd29yZCBvciBpbiBieXRlLCBhY2NvcmRpbmcgdG8gRGF0YVdpZHRoVW5pdC4KICAqIEBwYXJhbSAgT3V0cHV0OiBQb2ludGVyIHRvIHRoZSBvdXRwdXQgYnVmZmVyKGNpcGhlcnRleHQpCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFNwZWNpZnkgVGltZW91dCB2YWx1ZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfRW5jcnlwdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCAqSW5wdXQsIHVpbnQxNl90IFNpemUsIHVpbnQzMl90ICpPdXRwdXQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzOwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSBzdGF0ZSBCdXN5ICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9CVVNZOwoKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgICAvKiAgUmVzZXQgQ3J5cEluQ291bnQsIENyeXBPdXRDb3VudCBhbmQgSW5pdGlhbGl6ZSBwQ3J5cEluQnVmZlB0ciwgcENyeXBPdXRCdWZmUHRyIGFuZCBTaXplIHBhcmFtZXRlcnMqLwogICAgaGNyeXAtPkNyeXBJbkNvdW50ID0gMFU7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50ID0gMFU7CiAgICBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgPSBJbnB1dDsKICAgIGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgPSBPdXRwdXQ7CgogICAgLyogIENhbGN1bGF0ZSBTaXplIHBhcmFtZXRlciBpbiBCeXRlKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID09IENSWVBfREFUQVdJRFRIVU5JVF9XT1JEKQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemUgKiA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplOwogICAgfQoKICAgIC8qIFNldCBFbmNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPRElSLCBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCk7CgogICAgLyogYWxnbyBnZXQgYWxnb3JpdGhtIHNlbGVjdGVkICovCiAgICBhbGdvID0gaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT01PREU7CgogICAgc3dpdGNoIChhbGdvKQogICAgewogICAgICBjYXNlIENSWVBfREVTX0VDQjoKICAgICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgICAgY2FzZSBDUllQX1RERVNfQ0JDOgoKICAgICAgICAvKlNldCBLZXkgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDEpOwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDQpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgICAgfQoKICAgICAgICAvKlNldCBJbml0aWFsaXphdGlvbiBWZWN0b3IgKElWKSovCiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgICB9CgogICAgICAgIC8qIEZsdXNoIEZJRk8gKi8KICAgICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAgICAgLyogU3RhdHJ0IERFUy9UREVTIGVuY3J5cHRpb24gcHJvY2VzcyAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfVERFU19Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIC8qIEFFUyBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRW5jcnlwdChoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0dDTToKCiAgICAgICAgLyogQUVTIEdDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2VzcyhoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2VzcyhoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICBzdGF0dXMgPSBIQUxfRVJST1I7CiAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKHN0YXR1cyA9PSBIQUxfT0spCiAgICB7CiAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHN0YXR1cyA9IEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gc3RhdHVzIDsKfQoKLyoqCiAgKiBAYnJpZWYgIERlY3J5cHRpb24gbW9kZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIElucHV0OiBQb2ludGVyIHRvIHRoZSBpbnB1dCBidWZmZXIgKGNpcGhlcnRleHQgKQogICogQHBhcmFtICBTaXplOiBMZW5ndGggb2YgdGhlIHBsYWludGV4dCBidWZmZXIgZWl0aGVyIGluIHdvcmQgb3IgaW4gYnl0ZSwgYWNjb3JkaW5nIHRvIERhdGFXaWR0aFVuaXQKICAqIEBwYXJhbSAgT3V0cHV0OiBQb2ludGVyIHRvIHRoZSBvdXRwdXQgYnVmZmVyKHBsYWludGV4dCkKICAqIEBwYXJhbSAgVGltZW91dDogU3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9EZWNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBUaW1lb3V0KQp7CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzOwogIHVpbnQzMl90IGFsZ287CgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyLCBwQ3J5cE91dEJ1ZmZQdHIgYW5kIFNpemUgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgogICAgLyogU2V0IERlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERTsKCiAgICBzd2l0Y2ggKGFsZ28pCiAgICB7CiAgICAgIGNhc2UgQ1JZUF9ERVNfRUNCOgogICAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgICAgY2FzZSBDUllQX1RERVNfRUNCOgogICAgICBjYXNlIENSWVBfVERFU19DQkM6CgogICAgICAgIC8qU2V0IEtleSAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgICB7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAyKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA1KTsKICAgICAgICB9CgogICAgICAgIC8qU2V0IEluaXRpYWxpemF0aW9uIFZlY3RvciAoSVYpKi8KICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICAgIH0KCiAgICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgICAvKiBTdGFydCBERVMvVERFUyBkZWNyeXB0aW9uIHByb2Nlc3MgKi8KICAgICAgICBzdGF0dXMgPSBDUllQX1RERVNfUHJvY2VzcyhoY3J5cCwgVGltZW91dCk7CgogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0JDOgogICAgICBjYXNlIENSWVBfQUVTX0NUUjoKCiAgICAgICAgLyogQUVTIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU19EZWNyeXB0KGhjcnlwLCBUaW1lb3V0KTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNOgoKICAgICAgICAvKiBBRVMgR0NNIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KSA7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2VzcyhoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICBzdGF0dXMgPSBIQUxfRVJST1I7CiAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKHN0YXR1cyA9PSBIQUxfT0spCiAgICB7CiAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHN0YXR1cyA9IEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gc3RhdHVzOwp9CgovKioKICAqIEBicmllZiAgRW5jcnlwdGlvbiBpbiBpbnRlcnJ1cHQgbW9kZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIElucHV0OiBQb2ludGVyIHRvIHRoZSBpbnB1dCBidWZmZXIgKHBsYWludGV4dCkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGVpdGhlciBpbiB3b3JkIG9yIGluIGJ5dGUsIGFjY29yZGluZyB0byBEYXRhV2lkdGhVbml0CiAgKiBAcGFyYW0gIE91dHB1dDogUG9pbnRlciB0byB0aGUgb3V0cHV0IGJ1ZmZlcihjaXBoZXJ0ZXh0KQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfRW5jcnlwdF9JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCAqSW5wdXQsIHVpbnQxNl90IFNpemUsIHVpbnQzMl90ICpPdXRwdXQpCnsKICB1aW50MzJfdCBhbGdvOwogIEhBTF9TdGF0dXNUeXBlRGVmIHN0YXR1czsKCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2Ugc3RhdGUgQnVzeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogIFJlc2V0IENyeXBJbkNvdW50LCBDcnlwT3V0Q291bnQgYW5kIEluaXRpYWxpemUgcENyeXBJbkJ1ZmZQdHIsIHBDcnlwT3V0QnVmZlB0ciBhbmQgU2l6ZSBwYXJhbWV0ZXJzKi8KICAgIGhjcnlwLT5DcnlwSW5Db3VudCA9IDBVOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IDBVOwogICAgaGNyeXAtPnBDcnlwSW5CdWZmUHRyID0gSW5wdXQ7CiAgICBoY3J5cC0+cENyeXBPdXRCdWZmUHRyID0gT3V0cHV0OwoKICAgIC8qICBDYWxjdWxhdGUgU2l6ZSBwYXJhbWV0ZXIgaW4gQnl0ZSovCiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdCA9PSBDUllQX0RBVEFXSURUSFVOSVRfV09SRCkKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplICogNFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZTsKICAgIH0KCiAgICAvKiBTZXQgZW5jcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT0RJUiwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERSk7CgogICAgc3dpdGNoIChhbGdvKQogICAgewogICAgICBjYXNlIENSWVBfREVTX0VDQjoKICAgICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgICAgY2FzZSBDUllQX1RERVNfQ0JDOgoKICAgICAgICAvKlNldCBLZXkgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDEpOwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDQpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgICAgfQogICAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICAgIH0KCiAgICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgICAgIC8qIEVuYWJsZSBDUllQIHRvIHN0YXJ0IERFUy9UREVTIHByb2Nlc3MqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICAgICAgICAKICAgICAgICBzdGF0dXMgPSBIQUxfT0s7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICBzdGF0dXMgPSBDUllQX0FFU19FbmNyeXB0X0lUKGhjcnlwKTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNOgoKICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzX0lUKGhjcnlwKSA7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2Vzc19JVChoY3J5cCk7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXMgOwp9CgovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBpdG50ZXJydXB0IG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChjaXBoZXJ0ZXh0ICkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGVpdGhlciBpbiB3b3JkIG9yIGluIGJ5dGUsIGFjY29yZGluZyB0byBEYXRhV2lkdGhVbml0CiAgKiBAcGFyYW0gIE91dHB1dDogUG9pbnRlciB0byB0aGUgb3V0cHV0IGJ1ZmZlcihwbGFpbnRleHQpCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9EZWNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSBzdGF0ZSBCdXN5ICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9CVVNZOwoKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgICAvKiAgUmVzZXQgQ3J5cEluQ291bnQsIENyeXBPdXRDb3VudCBhbmQgSW5pdGlhbGl6ZSBwQ3J5cEluQnVmZlB0ciwgcENyeXBPdXRCdWZmUHRyIGFuZCBTaXplIHBhcmFtZXRlcnMqLwogICAgaGNyeXAtPkNyeXBJbkNvdW50ID0gMFU7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50ID0gMFU7CiAgICBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgPSBJbnB1dDsKICAgIGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgPSBPdXRwdXQ7CgogICAgLyogIENhbGN1bGF0ZSBTaXplIHBhcmFtZXRlciBpbiBCeXRlKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID09IENSWVBfREFUQVdJRFRIVU5JVF9XT1JEKQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemUgKiA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplOwogICAgfQoKICAgIC8qIFNldCBkZWNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPRElSLCBDUllQX09QRVJBVElOR01PREVfREVDUllQVCk7CgogICAgLyogYWxnbyBnZXQgYWxnb3JpdGhtIHNlbGVjdGVkICovCiAgICBhbGdvID0gaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT01PREU7CgogICAgc3dpdGNoIChhbGdvKQogICAgewogICAgICBjYXNlIENSWVBfREVTX0VDQjoKICAgICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgICAgY2FzZSBDUllQX1RERVNfQ0JDOgoKICAgICAgICAvKlNldCBLZXkgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDEpOwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDQpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgICAgfQoKICAgICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgICB9CiAgICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgICAgIC8qIEVuYWJsZSBDUllQIGFuZCBzdGFydCBERVMvVERFUyBwcm9jZXNzKi8KICAgICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0JDOgogICAgICBjYXNlIENSWVBfQUVTX0NUUjoKCiAgICAgICAgLyogQUVTIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU19EZWNyeXB0X0lUKGhjcnlwKTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNOgoKICAgICAgICAvKiBBRVMgR0NNIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzX0lUKGhjcnlwKSA7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTWRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0NDTV9Qcm9jZXNzX0lUKGhjcnlwKTsKICAgICAgICBicmVhazsKCiAgICAgIGRlZmF1bHQ6CiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9OT1RfU1VQUE9SVEVEOwogICAgICAgIHN0YXR1cyA9IEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgc3RhdHVzID0gSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXM7Cn0KCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIGluIERNQSBtb2RlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgSW5wdXQ6IFBvaW50ZXIgdG8gdGhlIGlucHV0IGJ1ZmZlciAocGxhaW50ZXh0KQogICogQHBhcmFtICBTaXplOiBMZW5ndGggb2YgdGhlIHBsYWludGV4dCBidWZmZXIgZWl0aGVyIGluIHdvcmQgb3IgaW4gYnl0ZSwgYWNjb3JkaW5nIHRvIERhdGFXaWR0aFVuaXQKICAqIEBwYXJhbSAgT3V0cHV0OiBQb2ludGVyIHRvIHRoZSBvdXRwdXQgYnVmZmVyKGNpcGhlcnRleHQpCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9FbmNyeXB0X0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCAqSW5wdXQsIHVpbnQxNl90IFNpemUsIHVpbnQzMl90ICpPdXRwdXQpCnsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CiAgdWludDMyX3QgYWxnbzsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSBzdGF0ZSBCdXN5ICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9CVVNZOwoKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgICAvKiAgUmVzZXQgQ3J5cEluQ291bnQsIENyeXBPdXRDb3VudCBhbmQgSW5pdGlhbGl6ZSBwQ3J5cEluQnVmZlB0ciwgcENyeXBPdXRCdWZmUHRyIGFuZCBTaXplIHBhcmFtZXRlcnMqLwogICAgaGNyeXAtPkNyeXBJbkNvdW50ID0gMFU7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50ID0gMFU7CiAgICBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgPSBJbnB1dDsKICAgIGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgPSBPdXRwdXQ7CgogICAgLyogIENhbGN1bGF0ZSBTaXplIHBhcmFtZXRlciBpbiBCeXRlKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID09IENSWVBfREFUQVdJRFRIVU5JVF9XT1JEKQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemUgKiA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplOwogICAgfQoKICAgIC8qIFNldCBlbmNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPRElSLCBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCk7CgogICAgLyogYWxnbyBnZXQgYWxnb3JpdGhtIHNlbGVjdGVkICovCiAgICBhbGdvID0gaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT01PREU7CgogICAgc3dpdGNoIChhbGdvKQogICAgewogICAgICBjYXNlIENSWVBfREVTX0VDQjoKICAgICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgICAgY2FzZSBDUllQX1RERVNfQ0JDOgoKICAgICAgICAvKlNldCBLZXkgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDEpOwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDQpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgICAgfQoKICAgICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgICB9CgogICAgICAgIC8qIEZsdXNoIEZJRk8gKi8KICAgICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAgICAgLyogU3RhcnQgRE1BIHByb2Nlc3MgdHJhbnNmZXIgZm9yIERFUy9UREVTICovCiAgICAgICAgQ1JZUF9TZXRETUFDb25maWcoaGNyeXAsICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwSW5CdWZmUHRyKSwgKGhjcnlwLT5TaXplIC8gNFUpLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICAgICAgICB7CiAgICAgICAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgICAgICAgewogICAgICAgICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICAgICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgICAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgICAgIH0KICAgICAgICAgIGVsc2UKICAgICAgICAgIHsKICAgICAgICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgICAgICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgICAgICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgICAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICAgICAgfQogICAgICAgIH0KCiAgICAgICAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgICAgICAgewogICAgICAgICAgLyogIFNldCB0aGUgS2V5Ki8KICAgICAgICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAgICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgICAgICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0VDQikKICAgICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxVSk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyVSk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzVSk7CiAgICAgICAgICB9CiAgICAgICAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KCiAgICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAgICAgLyogU3RhcnQgRE1BIHByb2Nlc3MgdHJhbnNmZXIgZm9yIEFFUyAqLwogICAgICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksIChoY3J5cC0+U2l6ZSAvIDRVKSwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0dDTToKCiAgICAgICAgLyogQUVTIEdDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2Vzc19ETUEoaGNyeXApIDsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0NNOgoKICAgICAgICAvKiBBRVMgQ0NNIGVuY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0NDTV9Qcm9jZXNzX0RNQShoY3J5cCk7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXM7Cn0KCi8qKgogICogQGJyaWVmICBEZWNyeXB0aW9uIGluIERNQSBtb2RlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgSW5wdXQ6IFBvaW50ZXIgdG8gdGhlIGlucHV0IGJ1ZmZlciAoY2lwaGVydGV4dCApCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBlaXRoZXIgaW4gd29yZCBvciBpbiBieXRlLCBhY2NvcmRpbmcgdG8gRGF0YVdpZHRoVW5pdAogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIocGxhaW50ZXh0KQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgKklucHV0LCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCAqT3V0cHV0KQp7CiAgdWludDMyX3QgYWxnbzsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyLCBwQ3J5cE91dEJ1ZmZQdHIgYW5kIFNpemUgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgogICAgLyogU2V0IGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERTsKCiAgICBzd2l0Y2ggKGFsZ28pCiAgICB7CiAgICAgIGNhc2UgQ1JZUF9ERVNfRUNCOgogICAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgICAgY2FzZSBDUllQX1RERVNfRUNCOgogICAgICBjYXNlIENSWVBfVERFU19DQkM6CgogICAgICAgIC8qU2V0IEtleSAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgICB7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAyKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA1KTsKICAgICAgICB9CgogICAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICAgIH0KCiAgICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgICAvKiBTdGFydCBETUEgcHJvY2VzcyB0cmFuc2ZlciBmb3IgREVTL1RERVMgKi8KICAgICAgICBDUllQX1NldERNQUNvbmZpZyhoY3J5cCwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCAoaGNyeXAtPlNpemUgLyA0VSksICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwT3V0QnVmZlB0cikpOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0JDOgogICAgICBjYXNlIENSWVBfQUVTX0NUUjoKCiAgICAgICAgLyogQUVTIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU19EZWNyeXB0X0RNQShoY3J5cCk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0dDTToKCiAgICAgICAgLyogQUVTIEdDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2Vzc19ETUEoaGNyeXApIDsKCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2Vzc19ETUEoaGNyeXApOwogICAgICAgIGJyZWFrOwoKICAgICAgZGVmYXVsdDoKICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgc3RhdHVzID0gSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXM7Cn0KCi8qKgogICogQH0KICAqLwoKLyoqIEBkZWZncm91cCBDUllQX0V4cG9ydGVkX0Z1bmN0aW9uc19Hcm91cDMgQ1JZUCBJUlEgaGFuZGxlciBtYW5hZ2VtZW50CiAgKiAgQGJyaWVmICAgIENSWVAgSVJRIGhhbmRsZXIuCiAgKgpAdmVyYmF0aW0KICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICAgICAgICAgICAgICAgICMjIyMjIENSWVAgSVJRIGhhbmRsZXIgbWFuYWdlbWVudCAjIyMjIwogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQpbLi5dICBUaGlzIHNlY3Rpb24gcHJvdmlkZXMgQ1JZUCBJUlEgaGFuZGxlciBhbmQgY2FsbGJhY2sgZnVuY3Rpb25zLgogICAgICAoKykgSEFMX0NSWVBfSVJRSGFuZGxlciBDUllQIGludGVycnVwdCByZXF1ZXN0CiAgICAgICgrKSBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrCiAgICAgICgrKSBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2sgb3V0cHV0IGRhdGEgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sKICAgICAgKCspIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2sgIENSWVAgZXJyb3IgY2FsbGJhY2sKICAgICAgKCspIEhBTF9DUllQX0dldFN0YXRlIHJldHVybiB0aGUgQ1JZUCBzdGF0ZQogICAgICAoKykgSEFMX0NSWVBfR2V0RXJyb3IgcmV0dXJuIHRoZSBDUllQIGVycm9yIGNvZGUKQGVuZHZlcmJhdGltCiAgKiBAewogICovCgovKioKICAqIEBicmllZiAgVGhpcyBmdW5jdGlvbiBoYW5kbGVzIGNyeXB0b2dyYXBoaWMgaW50ZXJydXB0IHJlcXVlc3QuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBOb25lCiAgKi8Kdm9pZCBIQUxfQ1JZUF9JUlFIYW5kbGVyKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICB1aW50MzJfdCBpdHN0YXR1cyA9IGhjcnlwLT5JbnN0YW5jZS0+TUlTUjsKCiAgaWYgKChpdHN0YXR1cyAmIChDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSkpICE9IDBVKQogIHsKICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgIHsKICAgICAgQ1JZUF9UREVTX0lUKGhjcnlwKTsgLyogREVTIG9yIFRERVMqLwogICAgfQogICAgZWxzZSBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0NUUikpCiAgICB7CiAgICAgIENSWVBfQUVTX0lUKGhjcnlwKTsgLypBRVMqLwogICAgfQoKICAgIGVsc2UgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfR0NNKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQ1JfQUxHT01PREVfQUVTX0NDTSkpCiAgICB7CiAgICAgIC8qIGlmIGhlYWRlciBwaGFzZSAqLwogICAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX1BIQVNFX0hFQURFUikgPT0gQ1JZUF9QSEFTRV9IRUFERVIpCiAgICAgIHsKICAgICAgICBDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZV9JVChoY3J5cCk7CiAgICAgIH0KICAgICAgZWxzZSAgLyogaWYgcGF5bG9hZCBwaGFzZSAqLwogICAgICB7CiAgICAgICAgQ1JZUF9HQ01DQ01fU2V0UGF5bG9hZFBoYXNlX0lUKGhjcnlwKTsKICAgICAgfQogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBOb3RoaW5nIHRvIGRvICovCiAgICB9CiAgfQp9CgovKioKICAqIEBicmllZiAgUmV0dXJuIHRoZSBDUllQIGVycm9yIGNvZGUuCiAgKiBAcGFyYW0gIGhjcnlwIDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciB0aGUgIENSWVAgSVAKICAqIEByZXR2YWwgQ1JZUCBlcnJvciBjb2RlCiAgKi8KdWludDMyX3QgSEFMX0NSWVBfR2V0RXJyb3IoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHJldHVybiBoY3J5cC0+RXJyb3JDb2RlOwp9CgovKioKICAqIEBicmllZiAgUmV0dXJucyB0aGUgQ1JZUCBzdGF0ZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHJldHZhbCBIQUwgc3RhdGUKICAqLwpIQUxfQ1JZUF9TVEFURVR5cGVEZWYgSEFMX0NSWVBfR2V0U3RhdGUoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHJldHVybiBoY3J5cC0+U3RhdGU7Cn0KCi8qKgogICogQGJyaWVmICBJbnB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHJldHZhbCBOb25lCiAgKi8KX193ZWFrIHZvaWQgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIFByZXZlbnQgdW51c2VkIGFyZ3VtZW50KHMpIGNvbXBpbGF0aW9uIHdhcm5pbmcgKi8KICBVTlVTRUQoaGNyeXApOwoKICAvKiBOT1RFIDogVGhpcyBmdW5jdGlvbiBTaG91bGQgbm90IGJlIG1vZGlmaWVkLCB3aGVuIHRoZSBjYWxsYmFjayBpcyBuZWVkZWQsCiAgICAgICAgICAgIHRoZSBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayBjb3VsZCBiZSBpbXBsZW1lbnRlZCBpbiB0aGUgdXNlciBmaWxlCiAgICovCn0KCi8qKgogICogQGJyaWVmICBPdXRwdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEByZXR2YWwgTm9uZQogICovCl9fd2VhayB2b2lkIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgLyogUHJldmVudCB1bnVzZWQgYXJndW1lbnQocykgY29tcGlsYXRpb24gd2FybmluZyAqLwogIFVOVVNFRChoY3J5cCk7CgogIC8qIE5PVEUgOiBUaGlzIGZ1bmN0aW9uIFNob3VsZCBub3QgYmUgbW9kaWZpZWQsIHdoZW4gdGhlIGNhbGxiYWNrIGlzIG5lZWRlZCwKICAgICAgICAgICAgdGhlIEhBTF9DUllQX091dENwbHRDYWxsYmFjayBjb3VsZCBiZSBpbXBsZW1lbnRlZCBpbiB0aGUgdXNlciBmaWxlCiAgICovCn0KCi8qKgogICogQGJyaWVmICBDUllQIGVycm9yIGNhbGxiYWNrLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpfX3dlYWsgdm9pZCBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBQcmV2ZW50IHVudXNlZCBhcmd1bWVudChzKSBjb21waWxhdGlvbiB3YXJuaW5nICovCiAgVU5VU0VEKGhjcnlwKTsKCiAgLyogTk9URSA6IFRoaXMgZnVuY3Rpb24gU2hvdWxkIG5vdCBiZSBtb2RpZmllZCwgd2hlbiB0aGUgY2FsbGJhY2sgaXMgbmVlZGVkLAogICAgICAgICAgICB0aGUgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayBjb3VsZCBiZSBpbXBsZW1lbnRlZCBpbiB0aGUgdXNlciBmaWxlCiAgICovCn0KLyoqCiAgKiBAfQogICovCgovKiBQcml2YXRlIGZ1bmN0aW9ucyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiogQGFkZHRvZ3JvdXAgQ1JZUF9Qcml2YXRlX0Z1bmN0aW9ucwogICogQHsKICAqLwoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRUNCL0NCQyBBbGdvcml0aG0gd2l0aCBERVMvVERFUyBzdGFuZGFyZC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1RERVNfUHJvY2VzcyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CgogIHVpbnQzMl90IHRlbXA7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MTZfdCBpbmNvdW50OyAvKiBUZW1wb3JhcnkgQ3J5cEluQ291bnQgVmFsdWUgKi8KICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KCiAgLyogRW5hYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogIC8qU3RhcnQgcHJvY2Vzc2luZyovCiAgd2hpbGUgKChoY3J5cC0+Q3J5cEluQ291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpICYmIChvdXRjb3VudCA8IChoY3J5cC0+U2l6ZSAvIDRVKSkpCiAgewogICAgLyogVGVtcG9yYXJ5IENyeXBJbkNvdW50IFZhbHVlICovCiAgICBpbmNvdW50ID0gaGNyeXAtPkNyeXBJbkNvdW50OwogICAgLyogV3JpdGUgcGxhaW4gZGF0YSBhbmQgZ2V0IGNpcGhlciBkYXRhICovCiAgICBpZiAoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX0lGTkYpICE9IDB4MFUpICYmIChpbmNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSkKICAgIHsKICAgICAgLyogV3JpdGUgdGhlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICB9CgogICAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgaWYgKENSWVBfV2FpdE9uT0ZORUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgIHsKICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKiBDaGFuZ2Ugc3RhdGUgJiBlcnJvckNvZGUqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQoKICAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogICAgaWYgKCgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19PRk5FKSAhPSAweDBVKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpKQogICAgewogICAgICAvKiBSZWFkIHRoZSBvdXRwdXQgYmxvY2sgZnJvbSB0aGUgT3V0cHV0IEZJRk8gYW5kIHB1dCB0aGVtIGluIHRlbXBvcmFyeSBCdWZmZXIgdGhlbiBnZXQgQ3J5cE91dEJ1ZmYgZnJvbSB0ZW1wb3JhcnkgYnVmZmVyICAqLwogICAgICB0ZW1wID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gdGVtcDsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgICB0ZW1wID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gdGVtcDsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgfQogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKICB9CiAgLyogRGlzYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBDUllQIGJsb2NrIGlucHV0L291dHB1dCBkYXRhIGhhbmRsaW5nIHVuZGVyIGludGVycnVwdGlvbiB3aXRoIERFUy9UREVTIHN0YW5kYXJkLgogICogQG5vdGUgICBUaGUgZnVuY3Rpb24gaXMgY2FsbGVkIHVuZGVyIGludGVycnVwdGlvbiBvbmx5LCBvbmNlCiAgKiAgICAgICAgIGludGVycnVwdGlvbnMgaGF2ZSBiZWVuIGVuYWJsZWQgYnkgQ1JZUF9EZWNyeXB0X0lUKCkgYW5kIENSWVBfRW5jcnlwdF9JVCgpLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgdm9pZCBDUllQX1RERVNfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHVpbnQzMl90IHRlbXA7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9CVVNZKQogIHsKICAgIGlmIChfX0hBTF9DUllQX0dFVF9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpICE9IDB4MFUpCiAgICB7CiAgICAgIGlmKF9fSEFMX0NSWVBfR0VUX0ZMQUcoaGNyeXAsIENSWVBfRkxBR19JTlJJUykgIT0gMHgwVSkKICAgICAgewogICAgICAgIC8qIFdyaXRlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgICAgCiAgICAgICAgaWYgKGhjcnlwLT5DcnlwSW5Db3VudCA9PSAoaGNyeXAtPlNpemUgLyA0VSkpCiAgICAgICAgewogICAgICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRpb24gKi8KICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwogICAgICAgICAgCiAgICAgICAgICAvKiBDYWxsIHRoZSBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgICBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgICAgfQogICAgICB9CiAgICB9CgogICAgaWYgKF9fSEFMX0NSWVBfR0VUX0lUKGhjcnlwLCBDUllQX0lUX09VVEkpICE9IDB4MFUpCiAgICB7CiAgICAgIGlmKF9fSEFMX0NSWVBfR0VUX0ZMQUcoaGNyeXAsIENSWVBfRkxBR19PVVRSSVMpICE9IDB4MFUpCiAgICAgIHsKICAgICAgICAvKiBSZWFkIHRoZSBvdXRwdXQgYmxvY2sgZnJvbSB0aGUgT3V0cHV0IEZJRk8gYW5kIHB1dCB0aGVtIGluIHRlbXBvcmFyeSBCdWZmZXIgdGhlbiBnZXQgQ3J5cE91dEJ1ZmYgZnJvbSB0ZW1wb3JhcnkgYnVmZmVyICAqLwogICAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgKGhjcnlwLT5DcnlwT3V0Q291bnQpKSA9IHRlbXA7CiAgICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgKGhjcnlwLT5DcnlwT3V0Q291bnQpKSA9IHRlbXA7CiAgICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgICAgIGlmIChoY3J5cC0+Q3J5cE91dENvdW50ID09IChoY3J5cC0+U2l6ZSAvIDRVKSkKICAgICAgICB7CiAgICAgICAgICAvKiBEaXNhYmxlIGludGVycnVwdGlvbiAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFX0lUKGhjcnlwLCBDUllQX0lUX09VVEkpOwogICAgICAgICAgCiAgICAgICAgICAvKiBEaXNhYmxlIENSWVAgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CiAgICAgICAgICAKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICAKICAgICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICAgICAgICAKICAgICAgICAgIC8qIENhbGwgb3V0cHV0IHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgT3V0cHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgICBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICAgICAgCiAgICAgICAgfQogICAgICB9CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogIH0KfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRUNCL0NCQyAmIENUUiBBbGdvcml0aG0gd2l0aCBBRVMgU3RhbmRhcmQKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IHNwZWNpZnkgVGltZW91dCB2YWx1ZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0VuY3J5cHQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogIHsKICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICB9CiAgfQoKICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgewogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDFVKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMlUpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzVSk7CiAgICB9CiAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KCiAgLyogU2V0IHRoZSBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgLyogRW5hYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogIHdoaWxlICgoaGNyeXAtPkNyeXBJbkNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpKQogIHsKICAgIC8qIFdyaXRlIHBsYWluIERkdGEgYW5kIGdldCBjaXBoZXIgZGF0YSAqLwogICAgQ1JZUF9BRVNfUHJvY2Vzc0RhdGEoaGNyeXAsIFRpbWVvdXQpOwogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKICB9CgogIC8qIERpc2FibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRUNCL0NCQyAmIENUUiBtb2RlIHdpdGggQUVTIFN0YW5kYXJkIHVzaW5nIGludGVycnVwdCBtb2RlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0VuY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogIHsKICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICB9CiAgfQoKICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgewogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDFVKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMlUpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzVSk7CiAgICB9CiAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KCiAgLyogU2V0IHRoZSBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgaWYgKGhjcnlwLT5TaXplICE9IDBVKQogIHsKICAgIC8qIEVuYWJsZSBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkgfCBDUllQX0lUX09VVEkpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgfQogIGVsc2UKICB7CiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIERlY3J5cHRpb24gaW4gRUNCL0NCQyAmIENUUiBtb2RlIHdpdGggQUVTIFN0YW5kYXJkCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZQogICogQHBhcmFtICBUaW1lb3V0OiBTcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19EZWNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgfQogIH0KCiAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgewogIC8qICBLZXkgcHJlcGFyYXRpb24gZm9yIEVDQi9DQkMgKi8KICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0NUUikgICAvKkVDQiBvciBDQkMqLwogIHsKICAgIC8qIGNoYW5nZSBBTEdPTU9ERSB0byBrZXkgcHJlcGFyYXRpb24gZm9yIGRlY3J5cHRpb24qLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0NSX0FMR09NT0RFX0FFU19LRVkpOwoKICAgIC8qICBTZXQgdGhlIEtleSovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgLyogRW5hYmxlIENSWVAgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAvKiBXYWl0IGZvciBCVVNZIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICBpZiAoQ1JZUF9XYWl0T25CVVNZRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgfQogICAgLyogVHVybiBiYWNrIHRvIEFMR09NT0RFIG9mIHRoZSBjb25maWd1cmF0aW9uICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIGhjcnlwLT5Jbml0LkFsZ29yaXRobSk7CiAgfQogIGVsc2UgIC8qQWxnb3JpdGhtIENUUiAqLwogIHsKICAgIC8qICBTZXQgdGhlIEtleSovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CiAgfQoKICAvKiBTZXQgSVYgKi8KICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0VDQikKICB7CiAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwogIH0KfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAgIAogIC8qIFNldCB0aGUgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogIC8qIEVuYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgd2hpbGUgKChoY3J5cC0+Q3J5cEluQ291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpICYmIChvdXRjb3VudCA8IChoY3J5cC0+U2l6ZSAvIDRVKSkpCiAgewogICAgLyogV3JpdGUgcGxhaW4gZGF0YSBhbmQgZ2V0IGNpcGhlciBkYXRhICovCiAgICBDUllQX0FFU19Qcm9jZXNzRGF0YShoY3J5cCwgVGltZW91dCk7CiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwogIH0KCiAgLyogRGlzYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9Ci8qKgogICogQGJyaWVmICBEZWNyeXB0aW9uIGluIEVDQi9DQkMgJiBDVFIgbW9kZSB3aXRoIEFFUyBTdGFuZGFyZCB1c2luZyBpbnRlcnJ1cHQgbW9kZQogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19EZWNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICBfX0lPIHVpbnQzMl90IGNvdW50ID0gMFU7CiAgIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogIHsKICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICB9CiAgfQoKICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgLyogIEtleSBwcmVwYXJhdGlvbiBmb3IgRUNCL0NCQyAqLwogIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfQ1RSKQogIHsKICAgIC8qIGNoYW5nZSBBTEdPTU9ERSB0byBrZXkgcHJlcGFyYXRpb24gZm9yIGRlY3J5cHRpb24qLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0NSX0FMR09NT0RFX0FFU19LRVkpOwoKICAgIC8qICBTZXQgdGhlIEtleSovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgLyogRW5hYmxlIENSWVAgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAvKiBXYWl0IGZvciBCVVNZIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICBjb3VudCA9IENSWVBfVElNRU9VVF9LRVlQUkVQQVJBVElPTjsKICAgIGRvCiAgICB7CiAgICAgIGNvdW50LS0gOwogICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgIHsKICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0gd2hpbGUgKEhBTF9JU19CSVRfU0VUKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19CVVNZKSk7CgogICAgLyogVHVybiBiYWNrIHRvIEFMR09NT0RFIG9mIHRoZSBjb25maWd1cmF0aW9uICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIGhjcnlwLT5Jbml0LkFsZ29yaXRobSk7CiAgfQogIGVsc2UgIC8qQWxnb3JpdGhtIENUUiAqLwogIHsKICAgIC8qICBTZXQgdGhlIEtleSovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CiAgfQoKICAvKiBTZXQgSVYgKi8KICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0VDQikKICB7CiAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwogIH0KfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAgIAogIC8qIFNldCB0aGUgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CiAgaWYgKGhjcnlwLT5TaXplICE9IDBVKQogIHsKICAgIC8qIEVuYWJsZSBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkgfCBDUllQX0lUX09VVEkpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogIH0KICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBFQ0IvQ0JDICYgQ1RSIG1vZGUgd2l0aCBBRVMgU3RhbmRhcmQgdXNpbmcgRE1BIG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgfQogIH0KCiAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgewogIC8qICBLZXkgcHJlcGFyYXRpb24gZm9yIEVDQi9DQkMgKi8KICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0NUUikKICB7CiAgICAvKiBjaGFuZ2UgQUxHT01PREUgdG8ga2V5IHByZXBhcmF0aW9uIGZvciBkZWNyeXB0aW9uKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPTU9ERSwgQ1JZUF9DUl9BTEdPTU9ERV9BRVNfS0VZKTsKCiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogV2FpdCBmb3IgQlVTWSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT047CiAgICBkbwogICAgewogICAgICBjb3VudC0tIDsKICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9IHdoaWxlIChIQUxfSVNfQklUX1NFVChoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfQlVTWSkpOwoKICAgIC8qIFR1cm4gYmFjayB0byBBTEdPTU9ERSBvZiB0aGUgY29uZmlndXJhdGlvbiAqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBoY3J5cC0+SW5pdC5BbGdvcml0aG0pOwogIH0KICBlbHNlICAvKkFsZ29yaXRobSBDVFIgKi8KICB7CiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwogIH0KCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgewogICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKICB9Cn0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCgogIC8qIFNldCB0aGUgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogIGlmIChoY3J5cC0+U2l6ZSAhPSAwVSkKICB7CiAgICAvKiBTZXQgdGhlIGlucHV0IGFuZCBvdXRwdXQgYWRkcmVzc2VzIGFuZCBzdGFydCBETUEgdHJhbnNmZXIgKi8KICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksIChoY3J5cC0+U2l6ZSAvIDRVKSwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKCi8qKgogICogQGJyaWVmICBETUEgQ1JZUCBpbnB1dCBkYXRhIHByb2Nlc3MgY29tcGxldGUgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhkbWE6IERNQSBoYW5kbGUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfRE1BSW5DcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKQp7CiAgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCA9IChDUllQX0hhbmRsZVR5cGVEZWYgKikoKERNQV9IYW5kbGVUeXBlRGVmICopaGRtYSktPlBhcmVudDsKCiAgLyogRGlzYWJsZSB0aGUgRE1BIHRyYW5zZmVyIGZvciBpbnB1dCBGSUZPIHJlcXVlc3QgYnkgcmVzZXR0aW5nIHRoZSBESUVOIGJpdAogIGluIHRoZSBETUFDUiByZWdpc3RlciAqLwogIGhjcnlwLT5JbnN0YW5jZS0+RE1BQ1IgJj0gKHVpbnQzMl90KSh+Q1JZUF9ETUFDUl9ESUVOKTsKCiAgLyogQ2FsbCBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwp9CgovKioKICAqIEBicmllZiAgRE1BIENSWVAgb3V0cHV0IGRhdGEgcHJvY2VzcyBjb21wbGV0ZSBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGRtYTogRE1BIGhhbmRsZQogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9ETUFPdXRDcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKQp7CiAgdWludDMyX3QgY291bnQ7CiAgdWludDMyX3QgbnBibGI7CiAgdWludDMyX3QgbGFzdHdvcmRzaXplOwogIHVpbnQzMl90IHRlbXA7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MzJfdCB0ZW1wX2NyX2FsZ29kaXI7ICAKICBDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwID0gKENSWVBfSGFuZGxlVHlwZURlZiAqKSgoRE1BX0hhbmRsZVR5cGVEZWYgKiloZG1hKS0+UGFyZW50OwoKCiAgLyogRGlzYWJsZSB0aGUgRE1BIHRyYW5zZmVyIGZvciBvdXRwdXQgRklGTyAqLwogIGhjcnlwLT5JbnN0YW5jZS0+RE1BQ1IgJj0gKHVpbnQzMl90KSh+Q1JZUF9ETUFDUl9ET0VOKTsKICAKICAvKiBMYXN0IGJsb2NrIHRyYW5zZmVyIGluIGNhc2Ugb2YgR0NNIG9yIENDTSB3aXRoIFNpemUgbm90ICUxNiovCiAgaWYgKCgoaGNyeXAtPlNpemUpICUgMTZVKSAhPSAwVSkKICB7CiAgICAvKiBzZXQgQ3J5cEluQ291bnQgYW5kIENyeXBPdXRDb3VudCB0byBleGFjdCBudW1iZXIgb2Ygd29yZCBhbHJlYWR5IGNvbXB1dGVkIHZpYSBETUEgICovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAoaGNyeXAtPlNpemUgLyAxNlUpICogNFUgOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IGhjcnlwLT5DcnlwSW5Db3VudDsKICAgIAogICAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgICBucGJsYiA9ICgoKCh1aW50MzJfdCkoaGNyeXAtPlNpemUpIC8gMTZVKSArIDFVKSAqIDE2VSkgLSAodWludDMyX3QpKGhjcnlwLT5TaXplKTsKICAgIAojaWYgIWRlZmluZWQgKENSWVBfVkVSXzJfMikKICAgIGlmIChoY3J5cC0+VmVyc2lvbiA+PSBSRVZfSURfQikKI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgICB7CiAgICAvKiBDYXNlIG9mIEFFUyBHQ00gcGF5bG9hZCBlbmNyeXB0aW9uIG9yIEFFUyBDQ00gcGF5bG9hZCBkZWNyeXB0aW9uIHRvIGdldCByaWdodCB0YWcgKi8KICAgICAgdGVtcF9jcl9hbGdvZGlyID0gaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUjsKICAgICAgaWYgKCgodGVtcF9jcl9hbGdvZGlyID09IENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKSAmJiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0dDTSkpIHx8CiAgICAgICAgICAoKHRlbXBfY3JfYWxnb2RpciA9PSBDUllQX09QRVJBVElOR01PREVfREVDUllQVCkgJiYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19DQ00pKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBTcGVjaWZ5IHRoZSBudW1iZXIgb2Ygbm9uLXZhbGlkIGJ5dGVzIHVzaW5nIE5QQkxCIHJlZ2lzdGVyKi8KICAgICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfTlBCTEIsIG5wYmxiIDw8IDIwKTsKCiAgICAgICAgLyogRW5hYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogICAgICB9CiAgICB9CiAgICAKICAgIC8qIE51bWJlciBvZiB2YWxpZCB3b3JkcyAobGFzdHdvcmRzaXplKSBpbiBsYXN0IGJsb2NrICovCiAgICBpZiAoKG5wYmxiICUgNFUpID09IDBVKQogICAgewogICAgICBsYXN0d29yZHNpemUgPSAoMTZVIC0gbnBibGIpIC8gNFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgoMTZVIC0gbnBibGIpIC8gNFUpICsgMVU7CiAgICB9CiAgICAvKiBXcml0ZSB0aGUgbGFzdCBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgZm9yIChjb3VudCA9IDBVOyBjb3VudCA8IGxhc3R3b3Jkc2l6ZTsgY291bnQgKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICB9CiAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgIHdoaWxlIChjb3VudCA8IDRVKQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAwVTsKICAgICAgY291bnQrKzsKICAgIH0KICAgIC8qIFdhaXQgZm9yIE9GTkUgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgIGNvdW50ID0gQ1JZUF9USU1FT1VUX0dDTUNDTUhFQURFUlBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgfQogICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQ1JZUF9GTEFHX09GTkUpKTsKCiAgICAvKlJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyAqLwogICAgZm9yIChjb3VudCA9IDBVOyBjb3VudCA8IDRVOyBjb3VudCsrKQogICAgewogICAgICAvKiBSZWFkIHRoZSBvdXRwdXQgYmxvY2sgZnJvbSB0aGUgb3V0cHV0IEZJRk8gYW5kIHB1dCB0aGVtIGluIHRlbXBvcmFyeSBidWZmZXIgdGhlbiBnZXQgQ3J5cE91dEJ1ZmYgZnJvbSB0ZW1wb3JhcnkgYnVmZmVyICovCiAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CgogICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gdGVtcDsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgfQogIH0gLypFbmQgb2YgbGFzdCBibG9jayB0cmFuc2ZlciBpbiBjYXNlIG9mIEdDTSBvciBDQ00gKi8KCiAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gJiBDUllQX0FFU19HQ00pICE9IENSWVBfQUVTX0dDTSkKICB7CiAgICAvKiBEaXNhYmxlIENSWVAgIChub3QgYWxsb3dlZCBpbiAgR0NNKSovCiAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwogIH0KCiAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlIHRvIHJlYWR5ICovCiAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAvKiBDYWxsIG91dHB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIERNQSBDUllQIGNvbW11bmljYXRpb24gZXJyb3IgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhkbWE6IERNQSBoYW5kbGUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfRE1BRXJyb3IoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpCnsKICBDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwID0gKENSWVBfSGFuZGxlVHlwZURlZiAqKSgoRE1BX0hhbmRsZVR5cGVEZWYgKiloZG1hKS0+UGFyZW50OwoKICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBETUEgZXJyb3IgY29kZSBmaWVsZCAqLwogIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfRE1BOwoKICAvKiBDYWxsIGVycm9yIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIFNldCB0aGUgRE1BIGNvbmZpZ3VyYXRpb24gYW5kIHN0YXJ0IHRoZSBETUEgdHJhbnNmZXIKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIGlucHV0YWRkcjogYWRkcmVzcyBvZiB0aGUgaW5wdXQgYnVmZmVyCiAgKiBAcGFyYW0gIFNpemU6IHNpemUgb2YgdGhlIGlucHV0IGJ1ZmZlciwgbXVzdCBiZSBhIG11bHRpcGxlIG9mIDE2LgogICogQHBhcmFtICBvdXRwdXRhZGRyOiBhZGRyZXNzIG9mIHRoZSBvdXRwdXQgYnVmZmVyCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpzdGF0aWMgdm9pZCBDUllQX1NldERNQUNvbmZpZyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBpbnB1dGFkZHIsIHVpbnQxNl90IFNpemUsIHVpbnQzMl90IG91dHB1dGFkZHIpCnsKICAvKiBTZXQgdGhlIENSWVAgRE1BIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiAgaGNyeXAtPmhkbWFpbi0+WGZlckNwbHRDYWxsYmFjayA9IENSWVBfRE1BSW5DcGx0OwoKICAvKiBTZXQgdGhlIERNQSBpbnB1dCBlcnJvciBjYWxsYmFjayAqLwogIGhjcnlwLT5oZG1haW4tPlhmZXJFcnJvckNhbGxiYWNrID0gQ1JZUF9ETUFFcnJvcjsKCiAgLyogU2V0IHRoZSBDUllQIERNQSB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwogIGhjcnlwLT5oZG1hb3V0LT5YZmVyQ3BsdENhbGxiYWNrID0gQ1JZUF9ETUFPdXRDcGx0OwoKICAvKiBTZXQgdGhlIERNQSBvdXRwdXQgZXJyb3IgY2FsbGJhY2sgKi8KICBoY3J5cC0+aGRtYW91dC0+WGZlckVycm9yQ2FsbGJhY2sgPSBDUllQX0RNQUVycm9yOwoKICAvKiBFbmFibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLyogRW5hYmxlIHRoZSBpbnB1dCBETUEgU3RyZWFtICovCiAgaWYgKEhBTF9ETUFfU3RhcnRfSVQoaGNyeXAtPmhkbWFpbiwgaW5wdXRhZGRyLCAodWludDMyX3QpJmhjcnlwLT5JbnN0YW5jZS0+RElOLCBTaXplKSAhPSBIQUxfT0spCiAgewogICAgLyogRE1BIGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfRE1BOwoKICAgIC8qIENhbGwgZXJyb3IgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQoKICAvKiBFbmFibGUgdGhlIG91dHB1dCBETUEgU3RyZWFtICovCiAgaWYgKEhBTF9ETUFfU3RhcnRfSVQoaGNyeXAtPmhkbWFvdXQsICh1aW50MzJfdCkmaGNyeXAtPkluc3RhbmNlLT5ET1VULCBvdXRwdXRhZGRyLCBTaXplKSAhPSBIQUxfT0spCiAgewogICAgLyogRE1BIGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfRE1BOwoKICAgIC8qIENhbGwgZXJyb3IgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQogIC8qIEVuYWJsZSBJbi9PdXQgRE1BIHJlcXVlc3QgKi8KICBoY3J5cC0+SW5zdGFuY2UtPkRNQUNSID0gQ1JZUF9ETUFDUl9ET0VOIHwgQ1JZUF9ETUFDUl9ESUVOOwp9CgovKioKICAqIEBicmllZiAgUHJvY2VzcyBEYXRhOiBXcml0ZSBJbnB1dCBkYXRhIGluIHBvbGxpbmcgbW9kZSBhbmQgdXNlZCBpbiBBRVMgZnVuY3Rpb25zLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgVGltZW91dDogU3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpzdGF0aWMgdm9pZCBDUllQX0FFU19Qcm9jZXNzRGF0YShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CgogIHVpbnQzMl90IHRlbXBbNF07ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MTZfdCBpbmNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBJbkNvdW50IFZhbHVlICovCiAgdWludDE2X3Qgb3V0Y291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlICovCiAgdWludDMyX3QgaTsKCiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBpbmNvdW50ID0gaGNyeXAtPkNyeXBJbkNvdW50OwoKICBpZiAoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX0lGTkYpICE9IDB4MFUpICYmIChpbmNvdW50IDwgKChoY3J5cC0+U2l6ZSkgLyA0VSkpKQogIHsKICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgfQoKICAvKiBXYWl0IGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgaWYgKENSWVBfV2FpdE9uT0ZORUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICB7CiAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgLyogQ2hhbmdlIHN0YXRlICYgZXJyb3IgY29kZSovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICB9CiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogIGlmICgoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSkgIT0gMHgwVSkgJiYgKG91dGNvdW50IDwgKChoY3J5cC0+U2l6ZSkgLyA0VSkpKQogIHsKICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBPdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgICovCiAgICBmb3IgKGkgPSAwVTsgaSA8IDRVOyBpKyspCiAgICB7CiAgICAgIHRlbXBbaV0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICB9CiAgICBpID0gMFU7CiAgICB3aGlsZSgoKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoKGhjcnlwLT5TaXplKS80VSkpKSAmJiAoaTw0VSkpCiAgICB7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgICBpKys7CiAgICB9CiAgfQp9CgovKioKICAqIEBicmllZiAgSGFuZGxlIENSWVAgYmxvY2sgaW5wdXQvb3V0cHV0IGRhdGEgaGFuZGxpbmcgdW5kZXIgaW50ZXJydXB0aW9uLgogICogQG5vdGUgICBUaGUgZnVuY3Rpb24gaXMgY2FsbGVkIHVuZGVyIGludGVycnVwdGlvbiBvbmx5LCBvbmNlCiAgKiAgICAgICAgIGludGVycnVwdGlvbnMgaGF2ZSBiZWVuIGVuYWJsZWQgYnkgSEFMX0NSWVBfRW5jcnlwdF9JVCBvciBIQUxfQ1JZUF9EZWNyeXB0X0lULgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgdm9pZCBDUllQX0FFU19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgdWludDMyX3QgdGVtcFs0XTsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwogIHVpbnQxNl90IGluY291bnQ7IC8qIFRlbXBvcmFyeSBDcnlwSW5Db3VudCBWYWx1ZSAqLwogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwogIHVpbnQzMl90IGk7CgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfQlVTWSkKICB7CiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgaW5jb3VudCA9IGhjcnlwLT5DcnlwSW5Db3VudDsKCiAgICBpZiAoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX0lGTkYpICE9IDB4MFUpICYmIChpbmNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSkKICAgIHsKICAgICAgLyogV3JpdGUgdGhlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGlmIChoY3J5cC0+Q3J5cEluQ291bnQgPT0gKGhjcnlwLT5TaXplIC8gNFUpKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSk7CgogICAgICAgIC8qIENhbGwgdGhlIGlucHV0IGRhdGEgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KCiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAgIGlmICgoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSkgIT0gMHgwVSkgJiYgKG91dGNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSkKICAgIHsKICAgICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIG91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgYnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAgKi8KICAgICAgZm9yIChpID0gMFU7IGkgPCA0VTsgaSsrKQogICAgICB7CiAgICAgICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICAgfQogICAgICBpID0gMFU7CiAgICAgIHdoaWxlKCgoaGNyeXAtPkNyeXBPdXRDb3VudCA8ICgoaGNyeXAtPlNpemUpLzRVKSkpICYmIChpPDRVKSkKICAgICAgewogICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgaSsrOwogICAgICB9CiAgICAgIGlmIChoY3J5cC0+Q3J5cE91dENvdW50ID09IChoY3J5cC0+U2l6ZSAvIDRVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIERpc2FibGUgQ1JZUCAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgICAgICAvKiBDYWxsIG91dHB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQp9CgovKioKICAqIEBicmllZiAgV3JpdGVzIEtleSBpbiBLZXkgcmVnaXN0ZXJzLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgS2V5U2l6ZTogU2l6ZSBvZiBLZXkKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfU2V0S2V5KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IEtleVNpemUpCnsKICBzd2l0Y2ggKEtleVNpemUpCiAgewogICAgY2FzZSBDUllQX0tFWVNJWkVfMjU2QjoKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDUpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA2KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNyk7CiAgICAgIGJyZWFrOwogICAgY2FzZSBDUllQX0tFWVNJWkVfMTkyQjoKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDUpOwogICAgICBicmVhazsKICAgIGNhc2UgQ1JZUF9LRVlTSVpFXzEyOEI6CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDEpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMyk7CgogICAgICBicmVhazsKICAgIGRlZmF1bHQ6CiAgICAgIGJyZWFrOwogIH0KfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24vRGVjcnlwdGlvbiBwcm9jZXNzIGluIEFFUyBHQ00gbW9kZSBhbmQgcHJlcGFyZSB0aGUgYXV0aGVudGljYXRpb24gVEFHCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBUaW1lb3V0OiBUaW1lb3V0IGR1cmF0aW9uCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNHQ01fUHJvY2VzcyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgdGlja3N0YXJ0OwogIHVpbnQzMl90IHdvcmRzaXplID0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSkgLyA0VTsKICB1aW50MzJfdCBucGJsYiA7CiAgdWludDMyX3QgdGVtcFs0XTsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwogIHVpbnQzMl90IGluZGV4IDsKICB1aW50MzJfdCBsYXN0d29yZHNpemUgOwogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogIHsKICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gKz0gaGNyeXAtPlNpemU7IC8qIENvbXB1dGUgbWVzc2FnZSB0b3RhbCBwYXlsb2FkIGxlbmd0aCAqLwogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICAgIGhjcnlwLT5TaXplc1N1bSA9IGhjcnlwLT5TaXplOyAvKiBNZXJlbHkgc3RvcmUgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICB9CiAgZWxzZQogIHsKICAgIGhjcnlwLT5TaXplc1N1bSA9IGhjcnlwLT5TaXplOwogIH0KCiAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgewogIC8qICBSZXNldCBDcnlwSGVhZGVyQ291bnQgKi8KICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50ID0gMFU7CgogIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogSW5pdCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9JTklUKTsKCiAgLyogU2V0IHRoZSBrZXkgKi8KICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogIC8qIFNldCB0aGUgaW5pdGlhbGl6YXRpb24gdmVjdG9yIGFuZCB0aGUgY291bnRlciA6IEluaXRpYWwgQ291bnRlciBCbG9jayAoSUNCKSovCiAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKCiAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogIC8qIEdldCB0aWNrICovCiAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKCiAgLypXYWl0IGZvciB0aGUgQ1JZUEVOIGJpdCB0byBiZSBjbGVhcmVkKi8KICB3aGlsZSAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0NSWVBFTikgPT0gQ1JZUF9DUl9DUllQRU4pCiAgewogICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICBpZiAoVGltZW91dCAhPSBIQUxfTUFYX0RFTEFZKQogICAgewogICAgICBpZiAoKChIQUxfR2V0VGljaygpIC0gdGlja3N0YXJ0KSA+IFRpbWVvdXQpIHx8IChUaW1lb3V0ID09IDBVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KCiAgLyoqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCiAgaWYgKENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlKGhjcnlwLCAgVGltZW91dCkgIT0gSEFMX09LKQogIHsKICAgIHJldHVybiBIQUxfRVJST1I7CiAgfQoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKlBheWxvYWQgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgaWYgKGhjcnlwLT5WZXJzaW9uID49IFJFVl9JRF9CKQojZW5kaWYgLypFbmQgb2Ygbm90IGRlZmluZWQgQ1JZUF9WRVJfMl8yKi8KICB7CiAgICAvKiBTZXQgdG8gMCB0aGUgbnVtYmVyIG9mIG5vbi12YWxpZCBieXRlcyB1c2luZyBOUEJMQiByZWdpc3RlciovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfTlBCTEIsIDBVKTsKICB9CgogIC8qIFNlbGVjdCBwYXlsb2FkIHBoYXNlIG9uY2UgdGhlIGhlYWRlciBwaGFzZSBpcyBwZXJmb3JtZWQgKi8KICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9QQVlMT0FEKTsKCiAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7Cn0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCiAgCiAgaWYgKChoY3J5cC0+U2l6ZSAlIDE2VSkgIT0gMFUpCiAgewogICAgLyogcmVjYWxjdWxhdGUgIHdvcmRzaXplICovCiAgICB3b3Jkc2l6ZSA9ICgod29yZHNpemUgLyA0VSkgKiA0VSkgOwogIH0KCiAgLyogR2V0IHRpY2sgKi8KICB0aWNrc3RhcnQgPSBIQUxfR2V0VGljaygpOwogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAvKiBXcml0ZSBpbnB1dCBkYXRhIGFuZCBnZXQgb3V0cHV0IERhdGEgKi8KICB3aGlsZSAoKGhjcnlwLT5DcnlwSW5Db3VudCA8IHdvcmRzaXplKSAmJiAob3V0Y291bnQgPCB3b3Jkc2l6ZSkpCiAgewogICAgLyogV3JpdGUgcGxhaW4gZGF0YSBhbmQgZ2V0IGNpcGhlciBkYXRhICovCiAgICBDUllQX0FFU19Qcm9jZXNzRGF0YShoY3J5cCwgVGltZW91dCk7CgogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICYgZXJyb3IgY29kZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KCiAgaWYgKChoY3J5cC0+U2l6ZSAlIDE2VSkgIT0gMFUpCiAgewoKI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgICBpZiAoaGNyeXAtPlZlcnNpb24gPj0gUkVWX0lEX0IpCiNlbmRpZiAvKkVuZCBvZiBub3QgZGVmaW5lZCBDUllQX1ZFUl8yXzIqLwogICAgewogICAgICAvKiBDb21wdXRlIHRoZSBudW1iZXIgb2YgcGFkZGluZyBieXRlcyBpbiBsYXN0IGJsb2NrIG9mIHBheWxvYWQgKi8KICAgICAgbnBibGIgPSAoKCgodWludDMyX3QpKGhjcnlwLT5TaXplKSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogICAgICAvKiAgU2V0IE5wYmxiIGluIGNhc2Ugb2YgQUVTIEdDTSBwYXlsb2FkIGVuY3J5cHRpb24gdG8gZ2V0IHJpZ2h0IHRhZyovCiAgICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogU3BlY2lmeSB0aGUgbnVtYmVyIG9mIG5vbi12YWxpZCBieXRlcyB1c2luZyBOUEJMQiByZWdpc3RlciovCiAgICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX05QQkxCLCBucGJsYiA8PCAyMCk7CgogICAgICAgIC8qIEVuYWJsZSBDUllQIHRvIHN0YXJ0IHRoZSBmaW5hbCBwaGFzZSAqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICAgICAgfQogICAgICAvKiBOdW1iZXIgb2YgdmFsaWQgd29yZHMgKGxhc3R3b3Jkc2l6ZSkgaW4gbGFzdCBibG9jayAqLwogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDBVKQogICAgICB7CiAgICAgICAgbGFzdHdvcmRzaXplID0gKDE2VSAtIG5wYmxiKSAvIDRVOwogICAgICB9CiAgICAgIGVsc2UKICAgICAgewogICAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgoMTZVIC0gbnBibGIpIC8gNFUpICsgMVU7CiAgICAgIH0KCiAgICAgIC8qIFdyaXRlIHRoZSBsYXN0IGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCBsYXN0d29yZHNpemU7IGluZGV4ICsrKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICB9CgogICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgd2hpbGUgKGluZGV4IDwgNFUpCiAgICAgIHsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAwVTsKICAgICAgICBpbmRleCsrOwogICAgICB9CgogICAgICAvKiBXYWl0IGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICAgIGlmIChDUllQX1dhaXRPbk9GTkVGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KCiAgICAgIC8qUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIG91dHB1dCBGSUZPICovCiAgICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19PRk5FKSAhPSAweDBVKQogICAgICB7CiAgICAgICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IDRVOyBpbmRleCsrKQogICAgICAgIHsKICAgICAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgKi8KICAgICAgICAgIHRlbXBbaW5kZXhdID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgICAgIH0KICAgICAgICBmb3IgKGluZGV4PTA7IGluZGV4PGxhc3R3b3Jkc2l6ZTsgaW5kZXgrKykKICAgICAgICB7CiAgICAgICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gdGVtcFtpbmRleF07CiAgICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgfQogICAgICB9CiAgICB9CiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogICAgZWxzZSAvKiAgV29ya2Fyb3VuZCB0byBiZSB1c2VkICovCiAgICB7CiAgICAgIC8qICBXb3JrYXJvdW5kIDIgZm9yIFNUTTMySDcgYmVsb3cgcmV2LkIgVG8gZ2VuZXJhdGUgY29ycmVjdCBUQUcgb25seSB3aGVuIHNpemUgb2YgdGhlIGxhc3QgYmxvY2sgb2YKICAgICAgcGF5bG9hZCBpcyBpbmZlcmlvciB0byAxMjggYml0cywgaW4gY2FzZSBvZiBHQ00gZW5jcnlwdGlvbiBvciBDQ00gZGVjcnlwdGlvbiovCiAgICAgIENSWVBfV29ya2Fyb3VuZChoY3J5cCwgVGltZW91dCk7CiAgICB9IC8qIGVuZCBvZiBOUEJMQiBvciBXb3JrYXJvdW5kKi8KI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24vRGVjcnlwdGlvbiBwcm9jZXNzIGluIEFFUyBHQ00gbW9kZSBhbmQgcHJlcGFyZSB0aGUgYXV0aGVudGljYXRpb24gVEFHIGluIGludGVycnVwdCBtb2RlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTR0NNX1Byb2Nlc3NfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIC8qIENvbmZpZ3VyZSBLZXksIElWIGFuZCBwcm9jZXNzIG1lc3NhZ2UgKGhlYWRlciBhbmQgcGF5bG9hZCkgKi8KICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgLyogIFJlc2V0IENyeXBIZWFkZXJDb3VudCAqLwogIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQgPSAwVTsKCiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogSW5pdCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0lOSVQpOwoKICAvKiBTZXQgdGhlIGtleSAqLwogIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgYW5kIHRoZSBjb3VudGVyIDogSW5pdGlhbCBDb3VudGVyIEJsb2NrIChJQ0IpKi8KICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogIGhjcnlwLT5JbnN0YW5jZS0+SVYxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDIpOwogIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwoKICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLypXYWl0IGZvciB0aGUgQ1JZUEVOIGJpdCB0byBiZSBjbGVhcmVkKi8KICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01JTklUUEhBU0U7CiAgZG8KICB7CiAgICBjb3VudC0tIDsKICAgIGlmIChjb3VudCA9PSAwVSkKICAgIHsKICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KICB9IHdoaWxlICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQ1JZUEVOKSA9PSBDUllQX0NSX0NSWVBFTik7CgogIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAvKiBTZWxlY3QgaGVhZGVyIHBoYXNlICovCiAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKICB9IC8qIGVuZCBvZiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSk7CgogIC8qIEVuYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uL0RlY3J5cHRpb24gcHJvY2VzcyBpbiBBRVMgR0NNIG1vZGUgYW5kIHByZXBhcmUgdGhlIGF1dGhlbnRpY2F0aW9uIFRBRyB1c2luZyBETUEKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNHQ01fUHJvY2Vzc19ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKICB1aW50MzJfdCB3b3Jkc2l6ZSA9ICh1aW50MzJfdCkoaGNyeXAtPlNpemUpIC8gNFUgOwogIHVpbnQzMl90IGluZGV4OwogIHVpbnQzMl90IG5wYmxiOwogIHVpbnQzMl90IGxhc3R3b3Jkc2l6ZTsKICB1aW50MzJfdCB0ZW1wWzRdOyAgLyogVGVtcG9yYXJ5IENyeXBPdXRCdWZmICovCiAgdWludDMyX3QgRG9LZXlJVkNvbmZpZyA9IDFVOyAvKiBCeSBkZWZhdWx0LCBjYXJyeSBvdXQgcGVyaXBoZXJhbCBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gKi8KCiAgaWYgKGhjcnlwLT5Jbml0LktleUlWQ29uZmlnU2tpcCA9PSBDUllQX0tFWUlWQ09ORklHX09OQ0UpCiAgewogICAgaWYgKGhjcnlwLT5LZXlJVkNvbmZpZyA9PSAxVSkKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgICAgYW5kIGlmIGl0IGhhcyBhbHJlYWR5IGJlZW4gZG9uZSwgc2tpcCBpdCAqLwogICAgICBEb0tleUlWQ29uZmlnID0gMFU7CiAgICAgIGhjcnlwLT5TaXplc1N1bSArPSBoY3J5cC0+U2l6ZTsgLyogQ29tcHV0ZSBtZXNzYWdlIHRvdGFsIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgIGFuZCBpZiBpdCBoYXMgbm90IGJlZW4gZG9uZSBhbHJlYWR5LCBkbyBpdCBhbmQgc2V0IEtleUlWQ29uZmlnCiAgICAgICAgIHRvIGtlZXAgdHJhY2sgaXQgd29uJ3QgaGF2ZSB0byBiZSBkb25lIGFnYWluIG5leHQgdGltZSAqLwogICAgICBoY3J5cC0+S2V5SVZDb25maWcgPSAxVTsKICAgICAgaGNyeXAtPlNpemVzU3VtID0gaGNyeXAtPlNpemU7IC8qIE1lcmVseSBzdG9yZSBwYXlsb2FkIGxlbmd0aCAqLwogICAgfQogIH0KICBlbHNlCiAgewogICAgaGNyeXAtPlNpemVzU3VtID0gaGNyeXAtPlNpemU7CiAgfQoKICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAvKiAgUmVzZXQgQ3J5cEhlYWRlckNvdW50ICovCiAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDBVOwoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqIEluaXQgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9JTklUKTsKCiAgLyogU2V0IHRoZSBrZXkgKi8KICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogIC8qIFNldCB0aGUgaW5pdGlhbGl6YXRpb24gdmVjdG9yIGFuZCB0aGUgY291bnRlciA6IEluaXRpYWwgQ291bnRlciBCbG9jayAoSUNCKSovCiAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKCiAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogIC8qV2FpdCBmb3IgdGhlIENSWVBFTiBiaXQgdG8gYmUgY2xlYXJlZCovCiAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogIGRvCiAgewogICAgY291bnQtLSA7CiAgICBpZiAoY291bnQgPT0gMFUpCiAgICB7CiAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICB9CiAgfSB3aGlsZSAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0NSWVBFTikgPT0gQ1JZUF9DUl9DUllQRU4pOwoKICAvKioqKioqKioqKioqKioqKioqKioqKioqIEhlYWRlciBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBpZiAoQ1JZUF9HQ01DQ01fU2V0SGVhZGVyUGhhc2VfRE1BKGhjcnlwKSAhPSBIQUxfT0spCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qKioqKioqKioqKioqKioqKioqKioqKiogUGF5bG9hZCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogIC8qIFNldCB0aGUgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgojaWYgIWRlZmluZWQgKENSWVBfVkVSXzJfMikKICBpZiAoaGNyeXAtPlZlcnNpb24gPj0gUkVWX0lEX0IpCiNlbmRpZiAvKkVuZCBvZiBub3QgZGVmaW5lZCBDUllQX1ZFUl8yXzIqLwogIHsKICAgIC8qIFNldCB0byAwIHRoZSBudW1iZXIgb2Ygbm9uLXZhbGlkIGJ5dGVzIHVzaW5nIE5QQkxCIHJlZ2lzdGVyKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9OUEJMQiwgMFUpOwogIH0KCiAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX1BBWUxPQUQpOwoKfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAgIAogIGlmIChoY3J5cC0+U2l6ZSA9PSAwVSkKICB7CiAgICAvKiBQcm9jZXNzIHVuTG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSBhbmQgcGhhc2UgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogIH0KICBlbHNlIGlmIChoY3J5cC0+U2l6ZSA+PSAxNlUpCiAgewogICAgLyogZm9yIFNUTTMySDcgYmVsb3cgcmV2LkIgOiBTaXplIHNob3VsZCBiZSAlNCAgb3RoZXJ3aXNlIFRhZyB3aWxsICBiZSBpbmNvcnJlY3RseSBnZW5lcmF0ZWQgZm9yIEdDTSBFbmNyeXB0aW9uOgogICAgV29ya2Fyb3VuZCBpcyBpbXBsZW1lbnRlZCBpbiBwb2xsaW5nIG1vZGUsIHNvIGlmIGxhc3QgYmxvY2sgb2YgcGF5bG9hZCA8MTI4Yml0IGRvbid0IHVzZSBETUEgbW9kZSBvdGhlcndpc2UgVEFHIGlzIGluY29ycmVjdGx5IGdlbmVyYXRlZCAqLwoKICAgIC8qRE1BIHRyYW5zZmVyIG11c3Qgbm90IGluY2x1ZGUgdGhlIGxhc3QgYmxvY2sgaW4gY2FzZSBvZiBTaXplIGlzIG5vdCAlMTYgKi8KICAgIHdvcmRzaXplID0gd29yZHNpemUgLSAod29yZHNpemUgJSA0VSk7CgogICAgLypETUEgdHJhbnNmZXIgKi8KICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksICh1aW50MTZfdCl3b3Jkc2l6ZSwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgfQogIGVsc2UgLyogbGVuZ3RoIG9mIGlucHV0IGRhdGEgaXMgPCAxNiAqLwogIHsKICAgIC8qIENvbXB1dGUgdGhlIG51bWJlciBvZiBwYWRkaW5nIGJ5dGVzIGluIGxhc3QgYmxvY2sgb2YgcGF5bG9hZCAqLwogICAgbnBibGIgPSAxNlUgLSAodWludDMyX3QpaGNyeXAtPlNpemU7CgojaWYgIWRlZmluZWQgKENSWVBfVkVSXzJfMikKICAgIGlmIChoY3J5cC0+VmVyc2lvbiA+PSBSRVZfSURfQikKI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgICB7CiAgICAgIC8qIFNldCBOcGJsYiBpbiBjYXNlIG9mIEFFUyBHQ00gcGF5bG9hZCBlbmNyeXB0aW9uIHRvIGdldCByaWdodCB0YWcqLwogICAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09ESVIpID09IENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKQogICAgICB7CiAgICAgICAgLyogU3BlY2lmeSB0aGUgbnVtYmVyIG9mIG5vbi12YWxpZCBieXRlcyB1c2luZyBOUEJMQiByZWdpc3RlciovCiAgICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX05QQkxCLCBucGJsYiA8PCAyMCk7CiAgICAgIH0KICAgIH0KICAgIC8qIEVuYWJsZSBDUllQIHRvIHN0YXJ0IHRoZSBmaW5hbCBwaGFzZSAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIC8qIE51bWJlciBvZiB2YWxpZCB3b3JkcyAobGFzdHdvcmRzaXplKSBpbiBsYXN0IGJsb2NrICovCiAgICBpZiAoKG5wYmxiICUgNFUpID09IDBVKQogICAgewogICAgICBsYXN0d29yZHNpemUgPSAoMTZVIC0gbnBibGIpIC8gNFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgoMTZVIC0gbnBibGIpIC8gNFUpICsgMVU7CiAgICB9CgogICAgLyogV3JpdGUgdGhlIGxhc3QgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgIGZvciAoaW5kZXggPSAwOyBpbmRleCA8IGxhc3R3b3Jkc2l6ZTsgaW5kZXggKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICB9CgogICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICB3aGlsZSAoaW5kZXggPCA0VSkKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gMFU7CiAgICAgIGluZGV4Kys7CiAgICB9CgogICAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICBkbwogICAgewogICAgICBjb3VudC0tIDsKICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICB9CiAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfT0ZORSkpOwoKICAgIC8qUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIG91dHB1dCBGSUZPICovCiAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgICB7CiAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgKi8KICAgICAgdGVtcFtpbmRleF0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICB9CiAgICBmb3IgKGluZGV4PTA7IGluZGV4PGxhc3R3b3Jkc2l6ZTsgaW5kZXgrKykKICAgIHsKICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2luZGV4XTsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgfQoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSB0byByZWFkeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAYnJpZWYgIEFFUyBDQ00gZW5jcnlwdGlvbi9kZWNyeXB0aW9uIHByb2Nlc3NpbmcgaW4gcG9sbGluZyBtb2RlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBUaW1lb3V0OiBUaW1lb3V0IGR1cmF0aW9uCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNDQ01fUHJvY2VzcyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgdGlja3N0YXJ0OwogIHVpbnQzMl90IHdvcmRzaXplID0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSkgLyA0VTsKICB1aW50MzJfdCBucGJsYiA7CiAgdWludDMyX3QgbGFzdHdvcmRzaXplIDsKICB1aW50MzJfdCB0ZW1wWzRdIDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwogIHVpbnQzMl90IGluZGV4IDsKICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKICAvKiAgUmVzZXQgQ3J5cEhlYWRlckNvdW50ICovCiAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDBVOwoKICAvKioqKioqKioqKioqKioqKioqKioqKiBJbml0IHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCiAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSU5JVCk7CgogIC8qIFNldCB0aGUga2V5ICovCiAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAvKiBTZXQgdGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciAoSVYpIHdpdGggQ1RSMSBpbmZvcm1hdGlvbiAqLwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAoaGNyeXAtPkluaXQuQjBbMF0pICYgQ1JZUF9DQ01fQ1RSMV8wOwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSBoY3J5cC0+SW5pdC5CMFsxXTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gaGNyeXAtPkluaXQuQjBbMl07CiAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9IChoY3J5cC0+SW5pdC5CMFszXSAmIENSWVBfQ0NNX0NUUjFfMSkgfCAgQ1JZUF9DQ01fQ1RSMV8yOwoKICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiNpZiBkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgewogICAgLyogZm9yIFNUTTMySDcgcmV2LkIgYW5kIGFib3ZlIFdyaXRlICBCMCBwYWNrZXQgaW50byBDUllQX0RSKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDIpOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMyk7CiAgfQojZWxzZQogIGlmIChoY3J5cC0+VmVyc2lvbiA+PSBSRVZfSURfQikKICB7CiAgICAvKiBmb3IgU1RNMzJINyByZXYuQiBhbmQgYWJvdmUgV3JpdGUgIEIwIHBhY2tldCBpbnRvIENSWVBfRFIqLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKTsKICB9CiAgZWxzZSAvKiBkYXRhIGhhcyB0byBiZSBzd2FwcGVkIGFjY29yZGluZyB0byB0aGUgREFUQVRZUEUgKi8KICB7CiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV84QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSk7CiAgICB9CiAgICBlbHNlIGlmIChoY3J5cC0+SW5pdC5EYXRhVHlwZSA9PSBDUllQX0RBVEFUWVBFXzE2QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSwgMTYpOwogICAgfQogICAgZWxzZSBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV8xQikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMykpOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwogICAgfQogIH0KI2VuZGlmCiAgLyogR2V0IHRpY2sgKi8KICB0aWNrc3RhcnQgPSBIQUxfR2V0VGljaygpOwoKICAvKldhaXQgZm9yIHRoZSBDUllQRU4gYml0IHRvIGJlIGNsZWFyZWQqLwogIHdoaWxlICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQ1JZUEVOKSA9PSBDUllQX0NSX0NSWVBFTikKICB7CiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9CiAgfQoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KICAvKiBIZWFkZXIgYmxvY2soQjEpIDogYXNzb2NpYXRlZCBkYXRhIGxlbmd0aCBleHByZXNzZWQgaW4gYnl0ZXMgY29uY2F0ZW5hdGVkCiAgd2l0aCBBc3NvY2lhdGVkIERhdGEgKEEpKi8KCiAgaWYgKENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CiAgLyoqKioqKioqKioqKioqKioqKioqKiogUGF5bG9hZCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogIC8qIFNldCB0aGUgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogIGlmIChoY3J5cC0+VmVyc2lvbiA+PSBSRVZfSURfQikKI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgewogICAgLyogU2V0IHRvIDAgdGhlIG51bWJlciBvZiBub24tdmFsaWQgYnl0ZXMgdXNpbmcgTlBCTEIgcmVnaXN0ZXIqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX05QQkxCLCAwVSk7CiAgfQoKICAvKiBTZWxlY3QgcGF5bG9hZCBwaGFzZSBvbmNlIHRoZSBoZWFkZXIgcGhhc2UgaXMgcGVyZm9ybWVkICovCiAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfUEFZTE9BRCk7CgogIC8qIEVuYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsICovCiAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAgCiAgaWYgKChoY3J5cC0+U2l6ZSAlIDE2VSkgIT0gMFUpCiAgewogICAgLyogcmVjYWxjdWxhdGUgIHdvcmRzaXplICovCiAgICB3b3Jkc2l6ZSA9ICgod29yZHNpemUgLyA0VSkgKiA0VSkgOwogIH0KICAvKiBHZXQgdGljayAqLwogIHRpY2tzdGFydCA9IEhBTF9HZXRUaWNrKCk7CgogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAvKiBXcml0ZSBpbnB1dCBkYXRhIGFuZCBnZXQgb3V0cHV0IGRhdGEgKi8KICB3aGlsZSAoKGhjcnlwLT5DcnlwSW5Db3VudCA8IHdvcmRzaXplKSAmJiAob3V0Y291bnQgPCB3b3Jkc2l6ZSkpCiAgewogICAgLyogV3JpdGUgcGxhaW4gZGF0YSBhbmQgZ2V0IGNpcGhlciBkYXRhICovCiAgICBDUllQX0FFU19Qcm9jZXNzRGF0YShoY3J5cCwgVGltZW91dCk7CgogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9CiAgfQoKICBpZiAoKGhjcnlwLT5TaXplICUgMTZVKSAhPSAwVSkKICB7CiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogICAgaWYgKGhjcnlwLT5WZXJzaW9uID49IFJFVl9JRF9CKQojZW5kaWYgLypFbmQgb2Ygbm90IGRlZmluZWQgQ1JZUF9WRVJfMl8yKi8KICAgIHsKICAgICAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgICAgIG5wYmxiID0gKCgoKHVpbnQzMl90KShoY3J5cC0+U2l6ZSkgLyAxNlUpICsgMVUpICogMTZVKSAtICh1aW50MzJfdCkoaGNyeXAtPlNpemUpOwoKICAgICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPRElSKSA9PSBDUllQX09QRVJBVElOR01PREVfREVDUllQVCkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBTZXQgTnBibGIgaW4gY2FzZSBvZiBBRVMgQ0NNIHBheWxvYWQgZGVjcnlwdGlvbiB0byBnZXQgcmlnaHQgdGFnICAqLwogICAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9OUEJMQiwgbnBibGIgPDwgMjApOwoKICAgICAgICAvKiBFbmFibGUgQ1JZUCB0byBzdGFydCB0aGUgZmluYWwgcGhhc2UgKi8KICAgICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgICAgIH0KCiAgICAgIC8qIE51bWJlciBvZiB2YWxpZCB3b3JkcyAobGFzdHdvcmRzaXplKSBpbiBsYXN0IGJsb2NrICovCiAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gMFUpCiAgICAgIHsKICAgICAgICBsYXN0d29yZHNpemUgPSAoMTZVIC0gbnBibGIpIC8gNFU7CiAgICAgIH0KICAgICAgZWxzZQogICAgICB7CiAgICAgICAgbGFzdHdvcmRzaXplID0gKCgxNlUgLSBucGJsYikgLyA0VSkgKyAxVTsKICAgICAgfQoKICAgICAgLyogV3JpdGUgdGhlIGxhc3QgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IGxhc3R3b3Jkc2l6ZTsgaW5kZXggKyspCiAgICAgIHsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIH0KCiAgICAgIC8qIFBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgICB3aGlsZSAoaW5kZXggPCA0VSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9IDBVOwogICAgICAgIGluZGV4Kys7CiAgICAgIH0KCiAgICAgIC8qIFdhaXQgZm9yIE9GTkUgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgICAgaWYgKENSWVBfV2FpdE9uT0ZORUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgfQoKICAgICAgLypSZWFkIHRoZSBvdXRwdXQgYmxvY2sgZnJvbSB0aGUgb3V0cHV0IEZJRk8gKi8KICAgICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpCiAgICAgIHsKICAgICAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgICAgICAgewogICAgICAgICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIG91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgYnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAqLwogICAgICAgICAgdGVtcFtpbmRleF0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICAgfQogICAgICAgIGZvciAoaW5kZXg9MDsgaW5kZXg8bGFzdHdvcmRzaXplOyBpbmRleCsrKQogICAgICAgIHsKICAgICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpbmRleF07CiAgICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgfSAKICAgICAgfQogICAgfQojaWYgIWRlZmluZWQgKENSWVBfVkVSXzJfMikKICAgIGVsc2UgLyogTm8gTlBCTEIsIFdvcmthcm91bmQgdG8gYmUgdXNlZCAqLwogICAgewogICAgICAvKiBDUllQIFdvcmthcm91bmQgOiAgQ1JZUDEgZ2VuZXJhdGVzIGNvcnJlY3QgVEFHICBkdXJpbmcgQ0NNIGRlY3J5cHRpb24gb25seSB3aGVuIGNpcGhlcnRleHQgYmxvY2tzIHNpemUgaXMgbXVsdGlwbGUgb2YKICAgICAgMTI4IGJpdHMuIElmIGx0aGUgc2l6ZSBvZiB0aGUgbGFzdCBibG9jayBvZiBwYXlsb2FkIGlzIGluZmVyaW9yIHRvIDEyOCBiaXRzLCB3aGVuIENDTSBkZWNyeXB0aW9uCiAgICAgIGlzIHNlbGVjdGVkLCB0aGVuIHRoZSBUQUcgbWVzc2FnZSB3aWxsIGJlIHdyb25nLiovCiAgICAgIENSWVBfV29ya2Fyb3VuZChoY3J5cCwgVGltZW91dCk7CiAgICB9CiNlbmRpZiAvKkVuZCBvZiBub3QgZGVmaW5lZCBDUllQX1ZFUl8yXzIqLwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBBRVMgQ0NNIGVuY3J5cHRpb24vZGVjcnlwdGlvbiBwcm9jZXNzIGluIGludGVycnVwdCBtb2RlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTQ0NNX1Byb2Nlc3NfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIC8qIENvbmZpZ3VyZSBLZXksIElWIGFuZCBwcm9jZXNzIG1lc3NhZ2UgKGhlYWRlciBhbmQgcGF5bG9hZCkgKi8KICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgLyogIFJlc2V0IENyeXBIZWFkZXJDb3VudCAqLwogIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQgPSAwVTsKCiAgLyoqKioqKioqKioqKiBJbml0IHBoYXNlICoqKioqKioqKioqKi8KCiAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSU5JVCk7CgogIC8qIFNldCB0aGUga2V5ICovCiAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAvKiBTZXQgdGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciAoSVYpIHdpdGggQ1RSMSBpbmZvcm1hdGlvbiAqLwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAoaGNyeXAtPkluaXQuQjBbMF0pICYgQ1JZUF9DQ01fQ1RSMV8wOwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSBoY3J5cC0+SW5pdC5CMFsxXTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gaGNyeXAtPkluaXQuQjBbMl07CiAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9IChoY3J5cC0+SW5pdC5CMFszXSAmIENSWVBfQ0NNX0NUUjFfMSkgfCAgQ1JZUF9DQ01fQ1RSMV8yOwoKICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLypXcml0ZSB0aGUgQjAgcGFja2V0IGludG8gQ1JZUF9EUiovCiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogIGlmIChoY3J5cC0+VmVyc2lvbiA+PSBSRVZfSURfQikKI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgewogICAgLyogZm9yIFNUTTMySDcgcmV2LkIgYW5kIGFib3ZlIGRhdGEgaGFzIG5vdCB0byBiZSBzd2FwcGVkICovCiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwogIH0KI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgZWxzZSAvKiBkYXRhIGhhcyB0byBiZSBzd2FwcGVkIGFjY29yZGluZyB0byB0aGUgREFUQVRZUEUgKi8KICB7CiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV84QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSk7CiAgICB9CiAgICBlbHNlIGlmIChoY3J5cC0+SW5pdC5EYXRhVHlwZSA9PSBDUllQX0RBVEFUWVBFXzE2QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSwgMTYpOwogICAgfQogICAgZWxzZSBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV8xQikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMykpOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwogICAgfQogIH0KI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgLypXYWl0IGZvciB0aGUgQ1JZUEVOIGJpdCB0byBiZSBjbGVhcmVkKi8KICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01JTklUUEhBU0U7CiAgZG8KICB7CiAgICBjb3VudC0tIDsKICAgIGlmIChjb3VudCA9PSAwVSkKICAgIHsKICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KICB9IHdoaWxlICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQ1JZUEVOKSA9PSBDUllQX0NSX0NSWVBFTik7CgogIC8qIFNlbGVjdCBoZWFkZXIgcGhhc2UgKi8KICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9IRUFERVIpOwp9IC8qIGVuZCBvZiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSk7CgogIC8qIEVuYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQovKioKICAqIEBicmllZiAgQUVTIENDTSBlbmNyeXB0aW9uL2RlY3J5cHRpb24gcHJvY2VzcyBpbiBETUEgbW9kZQogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU0NDTV9Qcm9jZXNzX0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgX19JTyB1aW50MzJfdCBjb3VudCAgICA9IDBVOwogIHVpbnQzMl90IHdvcmRzaXplID0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSkgLyA0VSA7CiAgdWludDMyX3QgaW5kZXg7CiAgdWludDMyX3QgbnBibGI7CiAgdWludDMyX3QgbGFzdHdvcmRzaXplOwogIHVpbnQzMl90IHRlbXBbNF07ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKICAvKiAgUmVzZXQgQ3J5cEhlYWRlckNvdW50ICovCiAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDBVOwoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKiogSW5pdCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCiAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSU5JVCk7CgogIC8qIFNldCB0aGUga2V5ICovCiAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAvKiBTZXQgdGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciAoSVYpIHdpdGggQ1RSMSBpbmZvcm1hdGlvbiAqLwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAoaGNyeXAtPkluaXQuQjBbMF0pICYgQ1JZUF9DQ01fQ1RSMV8wOwogIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSBoY3J5cC0+SW5pdC5CMFsxXTsKICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gaGNyeXAtPkluaXQuQjBbMl07CiAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9IChoY3J5cC0+SW5pdC5CMFszXSAmIENSWVBfQ0NNX0NUUjFfMSkgfCAgQ1JZUF9DQ01fQ1RSMV8yOwoKICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLypXcml0ZSB0aGUgQjAgcGFja2V0IGludG8gQ1JZUF9EUiovCiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogIGlmIChoY3J5cC0+VmVyc2lvbiA+PSBSRVZfSURfQikKI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgewogICAgLyogZm9yIFNUTTMySDcgcmV2LkIgYW5kIGFib3ZlIGRhdGEgaGFzIG5vdCB0byBiZSBzd2FwcGVkICovCiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwogIH0KI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgZWxzZSAvKiBkYXRhIGhhcyB0byBiZSBzd2FwcGVkIGFjY29yZGluZyB0byB0aGUgREFUQVRZUEUgKi8KICB7CiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV84QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSk7CiAgICB9CiAgICBlbHNlIGlmIChoY3J5cC0+SW5pdC5EYXRhVHlwZSA9PSBDUllQX0RBVEFUWVBFXzE2QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSwgMTYpOwogICAgfQogICAgZWxzZSBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV8xQikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMykpOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwogICAgfQogIH0KI2VuZGlmIC8qRW5kIG9mIG5vdCBkZWZpbmVkIENSWVBfVkVSXzJfMiovCiAgLypXYWl0IGZvciB0aGUgQ1JZUEVOIGJpdCB0byBiZSBjbGVhcmVkKi8KICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01JTklUUEhBU0U7CiAgZG8KICB7CiAgICBjb3VudC0tIDsKICAgIGlmIChjb3VudCA9PSAwVSkKICAgIHsKICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KICB9IHdoaWxlICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQ1JZUEVOKSA9PSBDUllQX0NSX0NSWVBFTik7CgogIC8qKioqKioqKioqKioqKioqKioqKiogSGVhZGVyIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBpZiAoQ1JZUF9HQ01DQ01fU2V0SGVhZGVyUGhhc2VfRE1BKGhjcnlwKSAhPSBIQUxfT0spCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qKioqKioqKioqKioqKioqKioqKiBQYXlsb2FkIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwojaWYgIWRlZmluZWQgKENSWVBfVkVSXzJfMikKICBpZiAoaGNyeXAtPlZlcnNpb24gPj0gUkVWX0lEX0IpCiNlbmRpZiAvKkVuZCBvZiBub3QgZGVmaW5lZCBDUllQX1ZFUl8yXzIqLwogIHsKICAgIC8qIFNldCB0byAwIHRoZSBudW1iZXIgb2Ygbm9uLXZhbGlkIGJ5dGVzIHVzaW5nIE5QQkxCIHJlZ2lzdGVyKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9OUEJMQiwgMFUpOwogIH0KCiAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX1BBWUxPQUQpOwogIH0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCgogIGlmIChoY3J5cC0+U2l6ZSA9PSAwVSkKICB7CiAgICAvKiBQcm9jZXNzIHVuTG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSBhbmQgcGhhc2UgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogIH0KICBlbHNlIGlmIChoY3J5cC0+U2l6ZSA+PSAxNlUpCiAgewogICAgLyogZm9yIFNUTTMySDcgYmVsb3cgcmV2LkIgOjogIFNpemUgc2hvdWxkIGJlICU0ICBvdGhlcndpc2UgVGFnIHdpbGwgIGJlIGluY29ycmVjdGx5IGdlbmVyYXRlZCBmb3IgQ0NNIERlY3J5cHRpb24sIFdvcmthcm91bmQgaXMgaW1wbGVtZW50ZWQgaW4gcG9sbGluZyBtb2RlKi8KICAgIC8qRE1BIHRyYW5zZmVyIG11c3Qgbm90IGluY2x1ZGUgdGhlIGxhc3QgYmxvY2sgaW4gY2FzZSBvZiBTaXplIGlzIG5vdCAlMTYgKi8KICAgIHdvcmRzaXplID0gd29yZHNpemUgLSAod29yZHNpemUgJSA0VSk7CgogICAgLypETUEgdHJhbnNmZXIgKi8KICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksICh1aW50MTZfdCkgd29yZHNpemUsICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwT3V0QnVmZlB0cikpOwogIH0KICBlbHNlIC8qIGxlbmd0aCBvZiBpbnB1dCBkYXRhIGlzICA8IDE2VSAqLwogIHsKICAgIC8qIENvbXB1dGUgdGhlIG51bWJlciBvZiBwYWRkaW5nIGJ5dGVzIGluIGxhc3QgYmxvY2sgb2YgcGF5bG9hZCAqLwogICAgbnBibGIgPSAxNlUgLSAodWludDMyX3QpKGhjcnlwLT5TaXplKTsKCiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogICAgaWYgKGhjcnlwLT5WZXJzaW9uID49IFJFVl9JRF9CKQojZW5kaWYgLypFbmQgb2Ygbm90IGRlZmluZWQgQ1JZUF9WRVJfMl8yKi8KICAgIHsKICAgICAgLyogU2V0IE5wYmxiIGluIGNhc2Ugb2YgQUVTIENDTSBwYXlsb2FkIGRlY3J5cHRpb24gdG8gZ2V0IHJpZ2h0IHRhZyovCiAgICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpCiAgICAgIHsKICAgICAgICAvKiBTcGVjaWZ5IHRoZSBudW1iZXIgb2Ygbm9uLXZhbGlkIGJ5dGVzIHVzaW5nIE5QQkxCIHJlZ2lzdGVyKi8KICAgICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfTlBCTEIsIG5wYmxiIDw8IDIwKTsKICAgICAgfQogICAgfQogICAgLyogRW5hYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogTnVtYmVyIG9mIHZhbGlkIHdvcmRzIChsYXN0d29yZHNpemUpIGluIGxhc3QgYmxvY2sgKi8KICAgIGlmICgobnBibGIgJSA0VSkgPT0gMFUpCiAgICB7CiAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgxNlUgLSBucGJsYikgLyA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgbGFzdHdvcmRzaXplID0gKCgxNlUgLSBucGJsYikgLyA0VSkgKyAxVTsKICAgIH0KCiAgICAvKiBXcml0ZSB0aGUgbGFzdCBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IGxhc3R3b3Jkc2l6ZTsgaW5kZXggKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICB9CgogICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICB3aGlsZSAoaW5kZXggPCA0VSkKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gMFU7CiAgICAgIGluZGV4Kys7CiAgICB9CgogICAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICBkbwogICAgewogICAgICBjb3VudC0tIDsKICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICB9CiAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfT0ZORSkpOwoKICAgIC8qUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIG91dHB1dCBGSUZPICovCiAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgICB7CiAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgKi8KICAgICAgdGVtcFtpbmRleF0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICB9CiAgICBmb3IgKGluZGV4PTA7IGluZGV4PGxhc3R3b3Jkc2l6ZTsgaW5kZXgrKykKICAgIHsKICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2luZGV4XTsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgfQoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSB0byByZWFkeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgU2V0cyB0aGUgcGF5bG9hZCBwaGFzZSBpbiBpbnRlcnJ1cHQgbW9kZQogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgc3RhdGUKICAqLwpzdGF0aWMgdm9pZCBDUllQX0dDTUNDTV9TZXRQYXlsb2FkUGhhc2VfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHVpbnQzMl90IGxvb3Bjb3VudGVyOwogIHVpbnQzMl90IHRlbXBbNF07ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MzJfdCBsYXN0d29yZHNpemU7CiAgdWludDMyX3QgbnBibGI7CiAgdWludDMyX3QgdGVtcF9jcl9hbGdvZGlyOwogIHVpbnQ4X3QgbmVnYXRpdmUgPSAwVTsKICB1aW50MzJfdCBpOwoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogUGF5bG9hZCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBpZiAoKGhjcnlwLT5TaXplIC8gNFUpIDwgaGNyeXAtPkNyeXBJbkNvdW50KQogIHsKICAgIG5lZ2F0aXZlID0gMVU7CiAgfQoKICBpZiAoaGNyeXAtPlNpemUgPT0gMFUpCiAgewogICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JIHwgQ1JZUF9JVF9PVVRJKTsKCiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgfQogIAogIGVsc2UgaWYgKCgoKGhjcnlwLT5TaXplIC8gNFUpIC0gKGhjcnlwLT5DcnlwSW5Db3VudCkpID49IDRVKSAmJgogICAgICAgICAgIChuZWdhdGl2ZSA9PSAwVSkpCiAgewogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPklNU0NSICYgQ1JZUF9JTVNDUl9JTklNKSE9IDB4MFUpCiAgICB7CiAgICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBpZiAoKChoY3J5cC0+U2l6ZSAvIDRVKSA9PSBoY3J5cC0+Q3J5cEluQ291bnQpICYmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwogICAgICAgIC8qIENhbGwgdGhlIGlucHV0IGRhdGEgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDFVKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICB9CiAgICAgIAogICAgICBpZiAoaGNyeXAtPkNyeXBPdXRDb3VudCA8IChoY3J5cC0+U2l6ZSAvIDRVKSkgICAgIAogICAgICB7CiAgICAgICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpCiAgICAgICAgewogICAgICAgICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIE91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgYnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAgKi8KICAgICAgICAgIGZvciAoaSA9IDBVOyBpIDwgNFU7IGkrKykKICAgICAgICAgIHsKICAgICAgICAgICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICAgICAgIH0KICAgICAgICAgIGkgPSAwVTsKICAgICAgICAgIHdoaWxlKCgoaGNyeXAtPkNyeXBPdXRDb3VudCA8ICgoaGNyeXAtPlNpemUpLzRVKSkpICYmIChpPDRVKSkKICAgICAgICAgIHsKICAgICAgICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2ldOwogICAgICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgICAgIGkrKzsKICAgICAgICAgIH0KICAgICAgICAgIGlmICgoKGhjcnlwLT5TaXplIC8gNFUpID09IGhjcnlwLT5DcnlwT3V0Q291bnQpICYmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKSkKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAgICAgCiAgICAgICAgICAgIC8qIERpc2FibGUgQ1JZUCAqLwogICAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwogICAgICAgICAgICAKICAgICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgICAKICAgICAgICAgICAgLyogQ2FsbCBvdXRwdXQgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDFVKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgICAgIH0KICAgICAgICB9CiAgICAgIH0KICAgIH0KICB9CiAgZWxzZSBpZiAoKGhjcnlwLT5TaXplICUgMTZVKSAhPSAwVSkKICB7CgkvKiBTZXQgcGFkZGluZyBvbmx5IGluIGNhc2Ugb2YgaW5wdXQgZmlmbyBpbnRlcnJ1cHQgKi8KCWlmICgoaGNyeXAtPkluc3RhbmNlLT5JTVNDUiAmIENSWVBfSU1TQ1JfSU5JTSkhPSAweDBVKQogewogICAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgICBucGJsYiA9ICgoKCh1aW50MzJfdCloY3J5cC0+U2l6ZSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CiAgICAKI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCiAgICBpZiAoaGNyeXAtPlZlcnNpb24gPj0gUkVWX0lEX0IpCiNlbmRpZiAvKkVuZCBvZiBub3QgZGVmaW5lZCBDUllQX1ZFUl8yXzIqLwogICAgewogICAgICAvKiBTZXQgTnBibGIgaW4gY2FzZSBvZiBBRVMgR0NNIHBheWxvYWQgZW5jcnlwdGlvbiBhbmQgQ0NNIGRlY3J5cHRpb24gdG8gZ2V0IHJpZ2h0IHRhZyAqLwogICAgICB0ZW1wX2NyX2FsZ29kaXIgPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPRElSOwoKICAgICAgaWYgKCgodGVtcF9jcl9hbGdvZGlyID09IENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKSAmJiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0dDTSkpIHx8CiAgICAgICAgICAoKHRlbXBfY3JfYWxnb2RpciA9PSBDUllQX09QRVJBVElOR01PREVfREVDUllQVCkgJiYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19DQ00pKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBTcGVjaWZ5IHRoZSBudW1iZXIgb2Ygbm9uLXZhbGlkIGJ5dGVzIHVzaW5nIE5QQkxCIHJlZ2lzdGVyKi8KICAgICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfTlBCTEIsIG5wYmxiIDw8IDIwKTsKCiAgICAgICAgLyogRW5hYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogICAgICB9CiAgICB9CgogICAgLyogTnVtYmVyIG9mIHZhbGlkIHdvcmRzIChsYXN0d29yZHNpemUpIGluIGxhc3QgYmxvY2sgKi8KICAgIGlmICgobnBibGIgJSA0VSkgPT0gMFUpCiAgICB7CiAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgxNlUgLSBucGJsYikgLyA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgbGFzdHdvcmRzaXplID0gKCgxNlUgLSBucGJsYikgLyA0VSkgKyAxVTsKICAgIH0KCiAgICAvKiBXcml0ZSB0aGUgbGFzdCBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgZm9yIChsb29wY291bnRlciA9IDBVOyBsb29wY291bnRlciA8IGxhc3R3b3Jkc2l6ZTsgbG9vcGNvdW50ZXIrKykKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIH0KICAgIC8qIFBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgd2hpbGUgKGxvb3Bjb3VudGVyIDwgNFUpCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9IDBVOwogICAgICBsb29wY291bnRlcisrOwogICAgfQoKICAgIC8qIERpc2FibGUgdGhlIGlucHV0IEZJRk8gSW50ZXJydXB0ICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JKTsKCX0KCiAgICAvKlJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyAqLwogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpCiAgICB7CiAgICAgIGZvciAoaSA9IDBVOyBpIDwgNFU7IGkrKykKICAgICAgewogICAgICAgIHRlbXBbaV0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgIH0KICAgICAgaWYgKCggKGhjcnlwLT5TaXplKS80VSk9PTBVKQogICAgICB7CiAgICAgICAgZm9yIChpID0gMFU7ICh1aW50MTZfdClpPCgoaGNyeXAtPlNpemUpJTRVKTsgaSsrKQogICAgICAgIHsKICAgICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgICB9ICAgICAKICAgICAgfQogICAgICBpID0gMFU7CiAgICAgIHdoaWxlKCgoaGNyeXAtPkNyeXBPdXRDb3VudCA8ICgoaGNyeXAtPlNpemUpLzRVKSkpICYmIChpPDRVKSkKICAgICAgewogICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgaSsrOwogICAgICB9CiAgICB9CgogICAgLyogRGlzYWJsZSB0aGUgb3V0cHV0IEZJRk8gSW50ZXJydXB0ICovCiAgICBpZiAoaGNyeXAtPkNyeXBPdXRDb3VudCA+PSAoKGhjcnlwLT5TaXplKSAvIDRVKSkKICAgIHsKICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJIHwgQ1JZUF9JVF9JTkkpOwoKICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgICAvKiBDYWxsIG91dHB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMVUpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgT3V0cHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIE5vdGhpbmcgdG8gZG8gKi8KICB9Cn0KCgovKioKICAqIEBicmllZiAgU2V0cyB0aGUgaGVhZGVyIHBoYXNlIGluIHBvbGxpbmcgbW9kZQogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUoSGVhZGVyICYgSGVhZGVyU2l6ZSkKICAqIEBwYXJhbSAgVGltZW91dDogVGltZW91dCB2YWx1ZQogICogQHJldHZhbCBzdGF0ZQogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgbG9vcGNvdW50ZXI7CgogIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgZm9yIEdDTS9HTUFDIG9yIENDTSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogIGlmICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAhPSAwVSkpCiAgewogICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIGlmICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSA9PSAwVSkKICAgIHsKICAgICAgLyogSGVhZGVyU2l6ZSAlNCwgbm8gcGFkZGluZyAqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IGhjcnlwLT5Jbml0LkhlYWRlclNpemUpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwoKICAgICAgICAvKiBXYWl0IGZvciBJRkVNIHRvIGJlIHJhaXNlZCAqLwogICAgICAgIGlmIChDUllQX1dhaXRPbklGRU1GbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICAgICAgewogICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgfQogICAgICB9CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qV3JpdGUgaGVhZGVyIGJsb2NrIGluIHRoZSBJTiBGSUZPIHdpdGhvdXQgbGFzdCBibG9jayAqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8ICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSkgLSAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSkpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwoKICAgICAgICAvKiBXYWl0IGZvciBJRkVNIHRvIGJlIHJhaXNlZCAqLwogICAgICAgIGlmIChDUllQX1dhaXRPbklGRU1GbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICAgICAgewogICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgfQogICAgICB9CiAgICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpKTsgbG9vcGNvdW50ZXIrKykKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogcGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAweDBVOwogICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgIH0KICAgICAgLyogV2FpdCBmb3IgQ0NGIElGRU0gdG8gYmUgcmFpc2VkICovCiAgICAgIGlmIChDUllQX1dhaXRPbklGRU1GbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0KICAgIC8qIFdhaXQgdW50aWwgdGhlIGNvbXBsZXRlIG1lc3NhZ2UgaGFzIGJlZW4gcHJvY2Vzc2VkICovCiAgICBpZiAoQ1JZUF9XYWl0T25CVVNZRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAmIHJldHVybiBlcnJvciAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgfQogIH0KICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIFNldHMgdGhlIGhlYWRlciBwaGFzZSB3aGVuIHVzaW5nIERNQSBpbiBwcm9jZXNzCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZShIZWFkZXIgJiBIZWFkZXJTaXplKQogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgX19JTyB1aW50MzJfdCBjb3VudCAgPSAwVTsKICB1aW50MzJfdCBsb29wY291bnRlcjsKCiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqIEhlYWRlciBwaGFzZSBmb3IgR0NNL0dNQUMgb3IgQ0NNICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KICBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgIT0gMFUpKQogIHsKICAgIC8qIFNlbGVjdCBoZWFkZXIgcGhhc2UgKi8KICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0hFQURFUik7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSkgPT0gMFUpCiAgICB7CiAgICAgIC8qIEhlYWRlclNpemUgJTQsIG5vIHBhZGRpbmcgKi8KICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyAobG9vcGNvdW50ZXIgPCBoY3J5cC0+SW5pdC5IZWFkZXJTaXplKTsgbG9vcGNvdW50ZXIgKz0gNFUpCiAgICAgIHsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKCiAgICAgICAgLyogV2FpdCBmb3IgSUZFTSB0byBiZSByYWlzZWQgKi8KICAgICAgICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01IRUFERVJQSEFTRTsKICAgICAgICBkbwogICAgICAgIHsKICAgICAgICAgIGNvdW50LS0gOwogICAgICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICAgICAgewogICAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgICAgfQogICAgICAgIH0gd2hpbGUgKEhBTF9JU19CSVRfQ0xSKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19JRkVNKSk7CiAgICAgIH0KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLypXcml0ZSBoZWFkZXIgYmxvY2sgaW4gdGhlIElOIEZJRk8gd2l0aG91dCBsYXN0IGJsb2NrICovCiAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgKGxvb3Bjb3VudGVyIDwgKChoY3J5cC0+SW5pdC5IZWFkZXJTaXplKSAtIChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpKSk7IGxvb3Bjb3VudGVyICs9IDRVKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CgogICAgICAgIC8qIFdhaXQgZm9yIElGRU0gdG8gYmUgcmFpc2VkICovCiAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICBjb3VudC0tIDsKICAgICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgIH0KICAgICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfSUZFTSkpOwogICAgICB9CiAgICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpKTsgbG9vcGNvdW50ZXIrKykKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAweDBVOwogICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgIH0KICAgICAgLyogV2FpdCBmb3IgSUZFTSB0byBiZSByYWlzZWQgKi8KICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICAgIGRvCiAgICAgIHsKICAgICAgICBjb3VudC0tIDsKICAgICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgICAgewogICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwogICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgIH0KICAgICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQ1JZUF9GTEFHX0lGRU0pKTsKICAgIH0KICAgIC8qIFdhaXQgdW50aWwgdGhlIGNvbXBsZXRlIG1lc3NhZ2UgaGFzIGJlZW4gcHJvY2Vzc2VkICovCiAgICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01IRUFERVJQSEFTRTsKICAgIGRvCiAgICB7CiAgICAgIGNvdW50LS0gOwogICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9IHdoaWxlIChIQUxfSVNfQklUX1NFVChoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfQlVTWSkpOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBTZXRzIHRoZSBoZWFkZXIgcGhhc2UgaW4gaW50ZXJydXB0IG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlKEhlYWRlciAmIEhlYWRlclNpemUpCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpzdGF0aWMgdm9pZCBDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZV9JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgdWludDMyX3QgbG9vcGNvdW50ZXI7CgogIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBpZiAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSA9PSAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCkKICB7CiAgICAvKiBEaXNhYmxlIGludGVycnVwdHMgKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwoKICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiNpZiAhZGVmaW5lZCAoQ1JZUF9WRVJfMl8yKQogICAgaWYgKGhjcnlwLT5WZXJzaW9uID49IFJFVl9JRF9CKQojZW5kaWYgLypFbmQgb2Ygbm90IGRlZmluZWQgQ1JZUF9WRVJfMl8yKi8KICAgIHsKICAgICAgLyogU2V0IHRvIDAgdGhlIG51bWJlciBvZiBub24tdmFsaWQgYnl0ZXMgdXNpbmcgTlBCTEIgcmVnaXN0ZXIqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfTlBCTEIsIDBVKTsKICAgIH0KCiAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfUEFZTE9BRCk7CgogICAgLyogRW5hYmxlIEludGVycnVwdHMgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICB9CiAgZWxzZSBpZiAoKChoY3J5cC0+SW5pdC5IZWFkZXJTaXplKSAtIChoY3J5cC0+Q3J5cEhlYWRlckNvdW50KSkgPj0gNFUpCgogIHsKICAgIC8qIEhlYWRlclNpemUgJTQsIG5vIHBhZGRpbmcgKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgIDsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogIH0KICBlbHNlCiAgewogICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IGxvb3Bjb3VudGVyIDwgKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSk7IGxvb3Bjb3VudGVyKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgIH0KICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgewogICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAweDBVOwogICAgICBsb29wY291bnRlcisrOwogICAgfQogIH0KfQoKI2lmICFkZWZpbmVkIChDUllQX1ZFUl8yXzIpCi8qKgogICogQGJyaWVmICBXb3JrYXJvdW5kIHVzZWQgZm9yIEdDTS9DQ00gbW9kZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfV29ya2Fyb3VuZChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgIGl2MXRlbXA7CiAgdWludDMyX3QgIHRlbXBbNF0gPSB7MH07CiAgdWludDMyX3QgIHRlbXAyWzRdID0gezB9OwogIHVpbnQzMl90IGludGVybWVkaWF0ZV9kYXRhWzRdID0gezB9OwogIHVpbnQzMl90IGluZGV4OwogIHVpbnQzMl90IGxhc3R3b3Jkc2l6ZTsKICB1aW50MzJfdCBucGJsYjsKCiAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgbnBibGIgPSAoKCgodWludDMyX3QpKGhjcnlwLT5TaXplKSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogIC8qIE51bWJlciBvZiB2YWxpZCB3b3JkcyAobGFzdHdvcmRzaXplKSBpbiBsYXN0IGJsb2NrICovCiAgaWYgKChucGJsYiAlIDRVKSA9PSAwVSkKICB7CiAgICBsYXN0d29yZHNpemUgPSAoMTZVIC0gbnBibGIpIC8gNFU7CiAgfQogIGVsc2UKICB7CiAgICBsYXN0d29yZHNpemUgPSAoKDE2VSAtIG5wYmxiKSAvIDRVKSArIDFVOwogIH0KCiAgLyogV29ya2Fyb3VuZCAyLCBjYXNlIEdDTSBlbmNyeXB0aW9uICovCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19HQ00pCiAgewogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPRElSKSA9PSBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCkKICAgIHsKICAgICAgLypXb3JrYXJvdW5kIGluIG9yZGVyIHRvIHByb3Blcmx5IGNvbXB1dGUgYXV0aGVudGljYXRpb24gdGFncyB3aGlsZSBkb2luZwogICAgICAgYSBHQ00gZW5jcnlwdGlvbiB3aXRoIHRoZSBsYXN0IGJsb2NrIG9mIHBheWxvYWQgc2l6ZSBpbmZlcmlvciB0byAxMjggYml0cyovCiAgICAgIC8qIERpc2FibGUgQ1JZUCB0byBzdGFydCB0aGUgZmluYWwgcGhhc2UgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qVXBkYXRlIENSWVBfSVYxUiByZWdpc3RlciBhbmQgQUxHT01PREUqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKChoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNN1IpLTFVKTsKICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0FFU19DVFIpOwoKICAgICAgLyogRW5hYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICAgIH0KCiAgICBmb3IgKGluZGV4ID0gMDsgaW5kZXggPCBsYXN0d29yZHNpemUgOyBpbmRleCArKykKICAgIHsKICAgICAgLyogV3JpdGUgdGhlIGxhc3QgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIH0KICAgIHdoaWxlIChpbmRleCA8IDRVKQogICAgewogICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gMFU7CiAgICAgIGluZGV4Kys7CiAgICB9CiAgICAvKiBXYWl0IGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICBpZiAoQ1JZUF9XYWl0T25PRk5FRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpCiAgICB7CiAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCA0VTsgaW5kZXgrKykKICAgICAgewogICAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyAqLwogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKCiAgICAgICAgLyogSW50ZXJtZWRpYXRlIGRhdGEgYnVmZmVyIHRvIGJlIHVzZWQgaW4gZm9yIHRoZSB3b3JrYXJvdW5kKi8KICAgICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gaW50ZXJtZWRpYXRlX2RhdGFbaW5kZXhdOwogICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgfQogICAgfQoKICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpCiAgICB7CiAgICAgIC8qd29ya2Fyb3VuZCBpbiBvcmRlciB0byBwcm9wZXJseSBjb21wdXRlIGF1dGhlbnRpY2F0aW9uIHRhZ3Mgd2hpbGUgZG9pbmcKICAgICAgYSBHQ00gZW5jcnlwdGlvbiB3aXRoIHRoZSBsYXN0IGJsb2NrIG9mIHBheWxvYWQgc2l6ZSBpbmZlcmlvciB0byAxMjggYml0cyovCiAgICAgIC8qIENoYW5nZSB0aGUgQUVTIG1vZGUgdG8gR0NNIG1vZGUgYW5kIFNlbGVjdCBGaW5hbCBwaGFzZSAqLwogICAgICAvKiBjb25maWd1cmVkICBDSE1PRCBHQ00gICAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIENSWVBfQUVTX0dDTSk7CgogICAgICAvKiBjb25maWd1cmVkICBmaW5hbCBwaGFzZSAgKi8KICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0dDTV9DQ01QSCwgQ1JZUF9QSEFTRV9GSU5BTCk7CgogICAgICBpZiAoIChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9EQVRBVFlQRSkgPT0gQ1JZUF9EQVRBVFlQRV8zMkIpCiAgICAgIHsKICAgICAgICBpZiAoKG5wYmxiICU0VSk9PTFVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZS0xVV0gJj0gMHhGRkZGRkYwMFU7CiAgICAgICAgfQogICAgICAgIGlmICgobnBibGIgJTRVKT09MlUpCiAgICAgICAgewogICAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplLTFVXSAmPSAweEZGRkYwMDAwVTsKICAgICAgICB9CiAgICAgICAgaWYgKChucGJsYiAlNFUpPT0zVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUtMVVdICY9IDB4RkYwMDAwMDBVOwogICAgICAgIH0KICAgICAgfQogICAgICBlbHNlIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfREFUQVRZUEUpID09IENSWVBfREFUQVRZUEVfOEIpCiAgICAgIHsKICAgICAgICBpZiAoKG5wYmxiICU0VSk9PTFVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZS0xVV0gJj0gX19SRVYoMHhGRkZGRkYwMFUpOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICU0VSk9PTJVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZS0xVV0gJj0gX19SRVYoMHhGRkZGMDAwMFUpOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICU0VSk9PTNVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZS0xVV0gJj0gX19SRVYoMHhGRjAwMDAwMFUpOwogICAgICAgIH0KICAgICAgfQogICAgICBlbHNlIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfREFUQVRZUEUpID09IENSWVBfREFUQVRZUEVfMTZCKQogICAgICB7CiAgICAgICAgaWYgKChucGJsYiAlNFUpPT0xVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUtMVVdICY9IF9fUk9SKCgweEZGRkZGRjAwVSksIDE2KTsKICAgICAgICB9CiAgICAgICAgaWYgKChucGJsYiAlNFUpPT0yVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUtMVVdICY9IF9fUk9SKCgweEZGRkYwMDAwVSksIDE2KTsKICAgICAgICB9CiAgICAgICAgaWYgKChucGJsYiAlNFUpPT0zVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUtMVVdICY9IF9fUk9SKCgweEZGMDAwMDAwVSksIDE2KTsKICAgICAgICB9CiAgICAgIH0KICAgICAgZWxzZSAvKkNSWVBfREFUQVRZUEVfMUIqLwogICAgICB7CiAgICAgICAgaWYgKChucGJsYiAlNFUpPT0xVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUtMVVdICY9IF9fUkJJVCgweEZGRkZGRjAwVSk7CiAgICAgICAgfQogICAgICAgIGlmICgobnBibGIgJTRVKT09MlUpCiAgICAgICAgewogICAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplLTFVXSAmPSBfX1JCSVQoMHhGRkZGMDAwMFUpOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICU0VSk9PTNVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZS0xVV0gJj0gX19SQklUKDB4RkYwMDAwMDBVKTsKICAgICAgICB9CiAgICAgIH0KICAgICAgCiAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCBsYXN0d29yZHNpemUgOyBpbmRleCArKykKICAgICAgewogICAgICAgIC8qV3JpdGUgdGhlIGludGVybWVkaWF0ZV9kYXRhIGluIHRoZSBJTiBGSUZPICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF07CiAgICAgIH0KICAgICAgd2hpbGUgKGluZGV4IDwgNFUpCiAgICAgIHsKICAgICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAweDBVOwogICAgICAgIGluZGV4Kys7CiAgICAgIH0KICAgICAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgICBpZiAoQ1JZUF9XYWl0T25PRk5FRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDFVKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgfQoKICAgICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpCiAgICAgIHsKICAgICAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgICAgICAgewogICAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbaW5kZXhdID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgICAgIH0KICAgICAgfQogICAgfQogIH0gLyogRW5kIG9mIEdDTSBlbmNyeXB0aW9uICovCiAgZWxzZQogIHsKICAgIC8qIFdvcmthcm91bmQgMiwgY2FzZSBDQ00gZGVjcnlwdGlvbiwgaW4gb3JkZXIgdG8gcHJvcGVybHkgY29tcHV0ZQogICAgICBhdXRoZW50aWNhdGlvbiB0YWdzIHdoaWxlIGRvaW5nIGEgQ0NNIGRlY3J5cHRpb24gd2l0aCB0aGUgbGFzdCBibG9jawogICAgICBvZiBwYXlsb2FkIHNpemUgaW5mZXJpb3IgdG8gMTI4IGJpdHMqLwoKICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpCiAgICB7CiAgICAgIGl2MXRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNN1I7CgogICAgICAvKiBEaXNhYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICB0ZW1wWzBdID0gIGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ00wUjsKICAgICAgdGVtcFsxXSA9ICBoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNMVI7CiAgICAgIHRlbXBbMl0gPSAgaGNyeXAtPkluc3RhbmNlLT5DU0dDTUNDTTJSOwogICAgICB0ZW1wWzNdID0gIGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ00zUjsKCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSBpdjF0ZW1wOwoKICAgICAgLyogQ29uZmlndXJlZCAgQ0hNT0QgQ1RSICAgKi8KICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0FFU19DVFIpOwoKICAgICAgLyogRW5hYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICAgIH0KICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IGxhc3R3b3Jkc2l6ZTsgaW5kZXggKyspCiAgICB7CiAgICAgIC8qIFdyaXRlIHRoZSBsYXN0IElucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICB9CiAgICB3aGlsZSAoaW5kZXggPCA0VSkKICAgIHsKICAgICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9IDBVOwogICAgICBpbmRleCsrOwogICAgfQogICAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgaWYgKENSWVBfV2FpdE9uT0ZORUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgIHsKICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgIH0KCiAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSkgIT0gMHgwVSkKICAgIHsKICAgICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IDRVOyBpbmRleCsrKQogICAgICB7CiAgICAgICAgLyogUmVhZCB0aGUgT3V0cHV0IGJsb2NrIGZyb20gdGhlIE91dHB1dCBGSUZPICovCiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbaW5kZXhdID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwoKICAgICAgICAvKmludGVybWVkaWF0ZSBkYXRhIGJ1ZmZlciB0byBiZSB1c2VkIGluIGZvciB0aGUgd29ya2Fyb3VuZCovCiAgICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgKGhjcnlwLT5DcnlwT3V0Q291bnQpKSA9IGludGVybWVkaWF0ZV9kYXRhW2luZGV4XTsKICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIH0KICAgIH0KCiAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09ESVIpID09IENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKQogICAgewogICAgICB0ZW1wMlswXSA9ICBoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNMFI7CiAgICAgIHRlbXAyWzFdID0gIGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ00xUjsKICAgICAgdGVtcDJbMl0gPSAgaGNyeXAtPkluc3RhbmNlLT5DU0dDTUNDTTJSOwogICAgICB0ZW1wMlszXSA9ICBoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNM1I7CgogICAgICAvKiBjb25maWd1cmVkICBDSE1PRCBDQ00gICAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIENSWVBfQUVTX0NDTSk7CgogICAgICAvKiBjb25maWd1cmVkICBIZWFkZXIgcGhhc2UgICovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9HQ01fQ0NNUEgsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAgIC8qc2V0IHRvIHplcm8gdGhlIGJpdHMgY29ycmVzcG9uZGluZyB0byB0aGUgcGFkZGVkIGJpdHMqLwogICAgICBmb3IgKGluZGV4ID0gbGFzdHdvcmRzaXplOyBpbmRleCA8IDRVOyBpbmRleCArKykKICAgICAgewogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSA9IDBVOwogICAgICB9CgogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDFVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IDB4RkZGRkZGMDBVOwogICAgICB9CiAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gMlUpCiAgICAgIHsKICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gMHhGRkZGMDAwMFU7CiAgICAgIH0KICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAzVSkKICAgICAgewogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSAweEZGMDAwMDAwVTsKICAgICAgfQoKICAgICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IDRVIDsgaW5kZXggKyspCiAgICAgIHsKICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF0gXj0gIHRlbXBbaW5kZXhdOwogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSBePSAgdGVtcDJbaW5kZXhdOwogICAgICB9CiAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCA0VTsgaW5kZXggKyspCiAgICAgIHsKICAgICAgICAvKiBXcml0ZSB0aGUgbGFzdCBJbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9IGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSA7CiAgICAgIH0KCiAgICAgIC8qIFdhaXQgZm9yIEJVU1kgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgICAgaWYgKENSWVBfV2FpdE9uQlVTWUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgfQogICAgfQogIH0gLyogRW5kIG9mIENDTSBXS0EqLwoKICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgX19IQUxfVU5MT0NLKGhjcnlwKTsKfQojZW5kaWYgLypFbmQgb2Ygbm90IGRlZmluZWQgQ1JZUF9WRVJfMl8yKi8KCi8qKgogICogQGJyaWVmICBIYW5kbGUgQ1JZUCBoYXJkd2FyZSBibG9jayBUaW1lb3V0IHdoZW4gd2FpdGluZyBmb3IgSUZFTSBmbGFnIHRvIGJlIHJhaXNlZC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHBhcmFtICBUaW1lb3V0OiBUaW1lb3V0IGR1cmF0aW9uLgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uSUZFTUZsYWcoY29uc3QgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IHRpY2tzdGFydDsKCiAgLyogR2V0IHRpbWVvdXQgKi8KICB0aWNrc3RhcnQgPSBIQUxfR2V0VGljaygpOwoKICB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQ1JZUF9GTEFHX0lGRU0pKQogIHsKICAgIC8qIENoZWNrIGZvciB0aGUgVGltZW91dCAqLwogICAgaWYgKFRpbWVvdXQgIT0gSEFMX01BWF9ERUxBWSkKICAgIHsKICAgICAgaWYgKCgoSEFMX0dldFRpY2soKSAtIHRpY2tzdGFydCkgPiBUaW1lb3V0KSB8fCAoVGltZW91dCA9PSAwVSkpCiAgICAgIHsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9CiAgfQogIHJldHVybiBIQUxfT0s7Cn0KLyoqCiAgKiBAYnJpZWYgIEhhbmRsZSBDUllQIGhhcmR3YXJlIGJsb2NrIFRpbWVvdXQgd2hlbiB3YWl0aW5nIGZvciBCVVNZIGZsYWcgdG8gYmUgcmFpc2VkLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgZHVyYXRpb24uCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9XYWl0T25CVVNZRmxhZyhjb25zdCBDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgdGlja3N0YXJ0OwoKICAvKiBHZXQgdGltZW91dCAqLwogIHRpY2tzdGFydCA9IEhBTF9HZXRUaWNrKCk7CgogIHdoaWxlIChIQUxfSVNfQklUX1NFVChoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfQlVTWSkpCiAgewogICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICBpZiAoVGltZW91dCAhPSBIQUxfTUFYX0RFTEFZKQogICAgewogICAgICBpZiAoKChIQUxfR2V0VGljaygpIC0gdGlja3N0YXJ0KSA+IFRpbWVvdXQpIHx8IChUaW1lb3V0ID09IDBVKSkKICAgICAgewogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0KICB9CiAgcmV0dXJuIEhBTF9PSzsKfQoKCi8qKgogICogQGJyaWVmICBIYW5kbGUgQ1JZUCBoYXJkd2FyZSBibG9jayBUaW1lb3V0IHdoZW4gd2FpdGluZyBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHBhcmFtICBUaW1lb3V0OiBUaW1lb3V0IGR1cmF0aW9uLgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uT0ZORUZsYWcoY29uc3QgQ1JZUF9IYW5kbGVUeXBlRGVmICAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKICB1aW50MzJfdCB0aWNrc3RhcnQ7CgogIC8qIEdldCB0aW1lb3V0ICovCiAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKCiAgd2hpbGUgKEhBTF9JU19CSVRfQ0xSKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19PRk5FKSkKICB7CiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAfQogICovCgoKCi8qKgogICogQH0KICAqLwoKLyoqCiAgKiBAfQogICovCgojZW5kaWYgLyogSEFMX0NSWVBfTU9EVUxFX0VOQUJMRUQgKi8KCgovKioKICAqIEB9CiAgKi8KI2VuZGlmIC8qIENSWVAgKi8KLyoqCiAgKiBAfQogICovCgovKioqKioqKioqKioqKioqKioqKioqKioqIChDKSBDT1BZUklHSFQgU1RNaWNyb2VsZWN0cm9uaWNzICoqKioqRU5EIE9GIEZJTEUqKioqLwo=