LyoqCiAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAgKiBAZmlsZSAgICBzdG0zMmY3eHhfaGFsX2NyeXAuYwogICogQGF1dGhvciAgTUNEIEFwcGxpY2F0aW9uIFRlYW0KICAqIEBicmllZiAgIENSWVAgSEFMIG1vZHVsZSBkcml2ZXIuCiAgKiAgICAgICAgICBUaGlzIGZpbGUgcHJvdmlkZXMgZmlybXdhcmUgZnVuY3Rpb25zIHRvIG1hbmFnZSB0aGUgZm9sbG93aW5nCiAgKiAgICAgICAgICBmdW5jdGlvbmFsaXRpZXMgb2YgdGhlIENyeXB0b2dyYXBoeSAoQ1JZUCkgcGVyaXBoZXJhbDoKICAqICAgICAgICAgICArIEluaXRpYWxpemF0aW9uLCBkZS1pbml0aWFsaXphdGlvbiwgc2V0IGNvbmZpZyBhbmQgZ2V0IGNvbmZpZyAgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBERVMvVERFUywgQUVTIHByb2Nlc3NpbmcgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBETUEgY2FsbGJhY2sgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBDUllQIElSUSBoYW5kbGVyIG1hbmFnZW1lbnQKICAqICAgICAgICAgICArIFBlcmlwaGVyYWwgU3RhdGUgZnVuY3Rpb25zCiAgKgogIEB2ZXJiYXRpbQogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgICAgICAgICAgICAgICAgICAjIyMjIyBIb3cgdG8gdXNlIHRoaXMgZHJpdmVyICMjIyMjCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICBbLi5dCiAgICAgIFRoZSBDUllQIEhBTCBkcml2ZXIgY2FuIGJlIHVzZWQgaW4gQ1JZUCBvciBUaW55QUVTIElQIGFzIGZvbGxvd3M6CgogICAgICAoIylJbml0aWFsaXplIHRoZSBDUllQIGxvdyBsZXZlbCByZXNvdXJjZXMgYnkgaW1wbGVtZW50aW5nIHRoZSBIQUxfQ1JZUF9Nc3BJbml0KCk6CiAgICAgICAgICgjIykgRW5hYmxlIHRoZSBDUllQIGludGVyZmFjZSBjbG9jayB1c2luZyBfX0hBTF9SQ0NfQ1JZUF9DTEtfRU5BQkxFKClvciBfX0hBTF9SQ0NfQUVTX0NMS19FTkFCTEUgZm9yIFRpbnlBRVMgSVAKICAgICAgICAgKCMjKSBJbiBjYXNlIG9mIHVzaW5nIGludGVycnVwdHMgKGUuZy4gSEFMX0NSWVBfRW5jcnlwdF9JVCgpKQogICAgICAgICAgICAgKCsrKykgQ29uZmlndXJlIHRoZSBDUllQIGludGVycnVwdCBwcmlvcml0eSB1c2luZyBIQUxfTlZJQ19TZXRQcmlvcml0eSgpCiAgICAgICAgICAgICAoKysrKSBFbmFibGUgdGhlIENSWVAgSVJRIGhhbmRsZXIgdXNpbmcgSEFMX05WSUNfRW5hYmxlSVJRKCkKICAgICAgICAgICAgICgrKyspIEluIENSWVAgSVJRIGhhbmRsZXIsIGNhbGwgSEFMX0NSWVBfSVJRSGFuZGxlcigpCiAgICAgICAgICgjIykgSW4gY2FzZSBvZiB1c2luZyBETUEgdG8gY29udHJvbCBkYXRhIHRyYW5zZmVyIChlLmcuIEhBTF9DUllQX0VuY3J5cHRfRE1BKCkpCiAgICAgICAgICAgICAoKysrKSBFbmFibGUgdGhlIERNQXggaW50ZXJmYWNlIGNsb2NrIHVzaW5nIF9fUkNDX0RNQXhfQ0xLX0VOQUJMRSgpCiAgICAgICAgICAgICAoKysrKSBDb25maWd1cmUgYW5kIGVuYWJsZSB0d28gRE1BIHN0cmVhbXMgb25lIGZvciBtYW5hZ2luZyBkYXRhIHRyYW5zZmVyIGZyb20KICAgICAgICAgICAgICAgICBtZW1vcnkgdG8gcGVyaXBoZXJhbCAoaW5wdXQgc3RyZWFtKSBhbmQgYW5vdGhlciBzdHJlYW0gZm9yIG1hbmFnaW5nIGRhdGEKICAgICAgICAgICAgICAgICB0cmFuc2ZlciBmcm9tIHBlcmlwaGVyYWwgdG8gbWVtb3J5IChvdXRwdXQgc3RyZWFtKQogICAgICAgICAgICAgKCsrKykgQXNzb2NpYXRlIHRoZSBpbml0aWFsaXplZCBETUEgaGFuZGxlIHRvIHRoZSBDUllQIERNQSBoYW5kbGUKICAgICAgICAgICAgICAgICB1c2luZyAgX19IQUxfTElOS0RNQSgpCiAgICAgICAgICAgICAoKysrKSBDb25maWd1cmUgdGhlIHByaW9yaXR5IGFuZCBlbmFibGUgdGhlIE5WSUMgZm9yIHRoZSB0cmFuc2ZlciBjb21wbGV0ZQogICAgICAgICAgICAgICAgIGludGVycnVwdCBvbiB0aGUgdHdvIERNQSBTdHJlYW1zLiBUaGUgb3V0cHV0IHN0cmVhbSBzaG91bGQgaGF2ZSBoaWdoZXIKICAgICAgICAgICAgICAgICBwcmlvcml0eSB0aGFuIHRoZSBpbnB1dCBzdHJlYW0gSEFMX05WSUNfU2V0UHJpb3JpdHkoKSBhbmQgSEFMX05WSUNfRW5hYmxlSVJRKCkKCiAgICAgICgjKUluaXRpYWxpemUgdGhlIENSWVAgYWNjb3JkaW5nIHRvIHRoZSBzcGVjaWZpZWQgcGFyYW1ldGVycyA6CiAgICAgICAgICgjIykgVGhlIGRhdGEgdHlwZTogMS1iaXQsIDgtYml0LCAxNi1iaXQgb3IgMzItYml0LgogICAgICAgICAoIyMpIFRoZSBrZXkgc2l6ZTogMTI4LCAxOTIgb3IgMjU2LgogICAgICAgICAoIyMpIFRoZSBBbGdvTW9kZSBERVMvIFRERVMgQWxnb3JpdGhtIEVDQi9DQkMgb3IgQUVTIEFsZ29yaXRobSBFQ0IvQ0JDL0NUUi9HQ00gb3IgQ0NNLgogICAgICAgICAoIyMpIFRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgKGNvdW50ZXIpLiBJdCBpcyBub3QgdXNlZCBpbiBFQ0IgbW9kZS4KICAgICAgICAgKCMjKSBUaGUga2V5IGJ1ZmZlciB1c2VkIGZvciBlbmNyeXB0aW9uL2RlY3J5cHRpb24uCiAgICAgICAgICgjIykgVGhlIEhlYWRlciB1c2VkIG9ubHkgaW4gQUVTIEdDTSBhbmQgQ0NNIEFsZ29yaXRobSBmb3IgYXV0aGVudGljYXRpb24uCiAgICAgICAgICgjIykgVGhlIEhlYWRlclNpemUgVGhlIHNpemUgb2YgaGVhZGVyIGJ1ZmZlciBpbiB3b3JkLgogICAgICAgICAoIyMpIFRoZSBCMCBibG9jayBpcyB0aGUgZmlyc3QgYXV0aGVudGljYXRpb24gYmxvY2sgdXNlZCBvbmx5ICBpbiBBRVMgQ0NNIG1vZGUuCgogICAgICAoIylUaHJlZSBwcm9jZXNzaW5nIChlbmNyeXB0aW9uL2RlY3J5cHRpb24pIGZ1bmN0aW9ucyBhcmUgYXZhaWxhYmxlOgogICAgICAgICAoIyMpIFBvbGxpbmcgbW9kZTogZW5jcnlwdGlvbiBhbmQgZGVjcnlwdGlvbiBBUElzIGFyZSBibG9ja2luZyBmdW5jdGlvbnMKICAgICAgICAgICAgICBpLmUuIHRoZXkgcHJvY2VzcyB0aGUgZGF0YSBhbmQgd2FpdCB0aWxsIHRoZSBwcm9jZXNzaW5nIGlzIGZpbmlzaGVkLAogICAgICAgICAgICAgIGUuZy4gSEFMX0NSWVBfRW5jcnlwdCAmIEhBTF9DUllQX0RlY3J5cHQKICAgICAgICAgKCMjKSBJbnRlcnJ1cHQgbW9kZTogZW5jcnlwdGlvbiBhbmQgZGVjcnlwdGlvbiBBUElzIGFyZSBub3QgYmxvY2tpbmcgZnVuY3Rpb25zCiAgICAgICAgICAgICAgaS5lLiB0aGV5IHByb2Nlc3MgdGhlIGRhdGEgdW5kZXIgaW50ZXJydXB0LAogICAgICAgICAgICAgIGUuZy4gSEFMX0NSWVBfRW5jcnlwdF9JVCAmIEhBTF9DUllQX0RlY3J5cHRfSVQKICAgICAgICAgKCMjKSBETUEgbW9kZTogZW5jcnlwdGlvbiBhbmQgZGVjcnlwdGlvbiBBUElzIGFyZSBub3QgYmxvY2tpbmcgZnVuY3Rpb25zCiAgICAgICAgICAgICAgaS5lLiB0aGUgZGF0YSB0cmFuc2ZlciBpcyBlbnN1cmVkIGJ5IERNQSwKICAgICAgICAgICAgICBlLmcuIEhBTF9DUllQX0VuY3J5cHRfRE1BICYgSEFMX0NSWVBfRGVjcnlwdF9ETUEKCiAgICAgICgjKVdoZW4gdGhlIHByb2Nlc3NpbmcgZnVuY3Rpb24gaXMgY2FsbGVkIGF0IGZpcnN0IHRpbWUgYWZ0ZXIgSEFMX0NSWVBfSW5pdCgpCiAgICAgICAgIHRoZSBDUllQIHBlcmlwaGVyYWwgaXMgY29uZmlndXJlZCBhbmQgcHJvY2Vzc2VzIHRoZSBidWZmZXIgaW4gaW5wdXQuCiAgICAgICAgIEF0IHNlY29uZCBjYWxsLCBubyBuZWVkIHRvIEluaXRpYWxpemUgdGhlIENSWVAsIHVzZXIgaGF2ZSB0byBnZXQgY3VycmVudCBjb25maWd1cmF0aW9uIHZpYQogICAgICAgICBIQUxfQ1JZUF9HZXRDb25maWcoKSBBUEksIHRoZW4gb25seSAgSEFMX0NSWVBfU2V0Q29uZmlnKCkgaXMgcmVxdWVzdGVkIHRvIHNldAogICAgICAgICBuZXcgcGFyYW1ldHJlcywgZmluYWxseSB1c2VyIGNhbiAgc3RhcnQgZW5jcnlwdGlvbi9kZWNyeXB0aW9uLgoKICAgICAgICgjKUNhbGwgSEFMX0NSWVBfRGVJbml0KCkgdG8gZGVpbml0aWFsaXplIHRoZSBDUllQIHBlcmlwaGVyYWwuCgogICAgICAgKCMpVG8gcHJvY2VzcyBhIHNpbmdsZSBtZXNzYWdlIHdpdGggY29uc2VjdXRpdmUgY2FsbHMgdG8gSEFMX0NSWVBfRW5jcnlwdCgpIG9yIEhBTF9DUllQX0RlY3J5cHQoKQogICAgICAgICAgd2l0aG91dCBoYXZpbmcgdG8gY29uZmlndXJlIGFnYWluIHRoZSBLZXkgb3IgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciBiZXR3ZWVuIGVhY2ggQVBJIGNhbGwsCiAgICAgICAgICB0aGUgZmllbGQgS2V5SVZDb25maWdTa2lwIG9mIHRoZSBpbml0aWFsaXphdGlvbiBzdHJ1Y3R1cmUgbXVzdCBiZSBzZXQgdG8gQ1JZUF9LRVlJVkNPTkZJR19PTkNFLgogICAgICAgICAgU2FtZSBpcyB0cnVlIGZvciBjb25zZWN1dGl2ZSBjYWxscyBvZiBIQUxfQ1JZUF9FbmNyeXB0X0lUKCksIEhBTF9DUllQX0RlY3J5cHRfSVQoKSwgSEFMX0NSWVBfRW5jcnlwdF9ETUEoKQogICAgICAgICAgb3IgSEFMX0NSWVBfRGVjcnlwdF9ETUEoKS4KCiAgICBbLi5dCiAgICAgIFRoZSBjcnlwdG9ncmFwaGljIHByb2Nlc3NvciBzdXBwb3J0cyBmb2xsb3dpbmcgc3RhbmRhcmRzOgogICAgICAoIykgVGhlIGRhdGEgZW5jcnlwdGlvbiBzdGFuZGFyZCAoREVTKSBhbmQgVHJpcGxlLURFUyAoVERFUykgc3VwcG9ydGVkIG9ubHkgYnkgQ1JZUDEgSVA6CiAgICAgICAgICgjIyk2NC1iaXQgZGF0YSBibG9jayBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgY2hhaW5pbmcgbW9kZXMgc3VwcG9ydGVkIDoKICAgICAgICAgICAgICgrKyspICBFbGVjdHJvbmljIENvZGUgQm9vayhFQ0IpCiAgICAgICAgICAgICAoKysrKSAgQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICAgICgjIykga2V5cyBsZW5ndGggc3VwcG9ydGVkIDo2NC1iaXQsIDEyOC1iaXQgYW5kIDE5Mi1iaXQuCiAgICAgICgjKSBUaGUgYWR2YW5jZWQgZW5jcnlwdGlvbiBzdGFuZGFyZCAoQUVTKSBzdXBwb3J0ZWQgIGJ5IENSWVAxICYgVGlueUFFUyBJUDoKICAgICAgICAgKCMjKTEyOC1iaXQgZGF0YSBibG9jayBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgY2hhaW5pbmcgbW9kZXMgc3VwcG9ydGVkIDoKICAgICAgICAgICAgICgrKyspICBFbGVjdHJvbmljIENvZGUgQm9vayhFQ0IpCiAgICAgICAgICAgICAoKysrKSAgQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICAgICAgICAoKysrKSAgQ291bnRlciBtb2RlIChDVFIpCiAgICAgICAgICAgICAoKysrKSAgR2Fsb2lzL2NvdW50ZXIgbW9kZSAoR0NNL0dNQUMpCiAgICAgICAgICAgICAoKysrKSAgQ291bnRlciB3aXRoIENpcGhlciBCbG9jayBDaGFpbmluZy1NZXNzYWdlKENDTSkKICAgICAgICAgKCMjKSBrZXlzIGxlbmd0aCBTdXBwb3J0ZWQgOgogICAgICAgICAgICAgKCsrKykgZm9yIENSWVAxIElQOiAxMjgtYml0LCAxOTItYml0IGFuZCAyNTYtYml0LgogICAgICAgICAgICAgKCsrKykgZm9yIFRpbnlBRVMgSVA6ICAxMjgtYml0IGFuZCAyNTYtYml0CgogICAgWy4uXSAgVGhpcyBzZWN0aW9uIGRlc2NyaWJlcyB0aGUgQUVTIEdhbG9pcy9jb3VudGVyIG1vZGUgKEdDTSkgc3VwcG9ydGVkIGJ5IGJvdGggQ1JZUDEgSVA6CiAgICAgICgjKSAgQWxnb3JpdGhtIHN1cHBvcnRlZCA6CiAgICAgICAgICgjIykgR2Fsb2lzL2NvdW50ZXIgbW9kZSAoR0NNKQogICAgICAgICAoIyMpIEdhbG9pcyBtZXNzYWdlIGF1dGhlbnRpY2F0aW9uIGNvZGUgKEdNQUMpIDppcyBleGFjdGx5IHRoZSBzYW1lIGFzCiAgICAgICAgICAgICAgR0NNIGFsZ29yaXRobSBjb21wb3NlZCBvbmx5IGJ5IGFuIGhlYWRlci4KICAgICAgKCMpICBGb3VyIHBoYXNlcyBhcmUgcGVyZm9ybWVkIGluIEdDTSA6CiAgICAgICAgICgjIykgSW5pdCBwaGFzZTogSVAgcHJlcGFyZXMgdGhlIEdDTSBoYXNoIHN1YmtleSAoSCkgYW5kIGRvIHRoZSBJViBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgSGVhZGVyIHBoYXNlOiBJUCBwcm9jZXNzZXMgdGhlIEFkZGl0aW9uYWwgQXV0aGVudGljYXRlZCBEYXRhIChBQUQpLCB3aXRoIGhhc2gKICAgICAgICAgIGNvbXB1dGF0aW9uIG9ubHkuCiAgICAgICAgICgjIykgUGF5bG9hZCBwaGFzZTogSVAgcHJvY2Vzc2VzIHRoZSBwbGFpbnRleHQgKFApIHdpdGggaGFzaCBjb21wdXRhdGlvbiArIGtleXN0cmVhbQogICAgICAgICAgZW5jcnlwdGlvbiArIGRhdGEgWE9SaW5nLiBJdCB3b3JrcyBpbiBhIHNpbWlsYXIgd2F5IGZvciBjaXBoZXJ0ZXh0IChDKS4KICAgICAgICAgKCMjKSBGaW5hbCBwaGFzZTogSVAgZ2VuZXJhdGVzIHRoZSBhdXRoZW50aWNhdGVkIHRhZyAoVCkgdXNpbmcgdGhlIGxhc3QgYmxvY2sgb2YgZGF0YS4KICAgICAgKCMpICBzdHJ1Y3R1cmUgb2YgbWVzc2FnZSBjb25zdHJ1Y3Rpb24gaW4gR0NNIGlzIGRlZmluZWQgYXMgYmVsb3cgIDoKICAgICAgICAgKCMjKSAxNiBieXRlcyBJbml0aWFsIENvdW50ZXIgQmxvY2sgKElDQiljb21wb3NlZCBvZiBJViBhbmQgY291bnRlcgogICAgICAgICAoIyMpIFRoZSBhdXRoZW50aWNhdGVkIGhlYWRlciBBIChhbHNvIGtub3dzIGFzIEFkZGl0aW9uYWwgQXV0aGVudGljYXRpb24gRGF0YSBBQUQpCiAgICAgICAgICB0aGlzIHBhcnQgb2YgdGhlIG1lc3NhZ2UgaXMgb25seSBhdXRoZW50aWNhdGVkLCBub3QgZW5jcnlwdGVkLgogICAgICAgICAoIyMpIFRoZSBwbGFpbnRleHQgbWVzc2FnZSBQIGlzIGJvdGggYXV0aGVudGljYXRlZCBhbmQgZW5jcnlwdGVkIGFzIGNpcGhlcnRleHQuCiAgICAgICAgICBHQ00gc3RhbmRhcmQgc3BlY2lmaWVzIHRoYXQgY2lwaGVydGV4dCBoYXMgc2FtZSBiaXQgbGVuZ3RoIGFzIHRoZSBwbGFpbnRleHQuCiAgICAgICAgICgjIykgVGhlIGxhc3QgYmxvY2sgaXMgY29tcG9zZWQgb2YgdGhlIGxlbmd0aCBvZiBBIChvbiA2NCBiaXRzKSBhbmQgdGhlIGxlbmd0aCBvZiBjaXBoZXJ0ZXh0CiAgICAgICAgICAob24gNjQgYml0cykKCiAgICBbLi5dICBUaGlzIHNlY3Rpb24gZGVzY3JpYmUgVGhlIEFFUyBDb3VudGVyIHdpdGggQ2lwaGVyIEJsb2NrIENoYWluaW5nLU1lc3NhZ2UKICAgICAgICAgIEF1dGhlbnRpY2F0aW9uIENvZGUgKENDTSkgc3VwcG9ydGVkIGJ5IGJvdGggQ1JZUDEgSVA6CiAgICAgICgjKSAgU3BlY2lmaWMgcGFyYW1ldGVycyBmb3IgQ0NNICA6CgogICAgICAgICAoIyMpIEIwIGJsb2NrICA6IEFjY29yZGluZyB0byBOSVNUIFNwZWNpYWwgUHVibGljYXRpb24gODAwLTM4QywKICAgICAgICAgICAgVGhlIGZpcnN0IGJsb2NrIEIwIGlzIGZvcm1hdHRlZCBhcyBmb2xsb3dzLCB3aGVyZSBsKG0pIGlzIGVuY29kZWQgaW4KICAgICAgICAgICAgbW9zdC1zaWduaWZpY2FudC1ieXRlIGZpcnN0IG9yZGVyKHNlZSBiZWxvdyB0YWJsZSAzKQoKICAgICAgICAgICAgICAoKysrKSAgUTogYSBiaXQgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBvY3RldCBsZW5ndGggb2YgUCAocGxhaW50ZXh0KQogICAgICAgICAgICAgICgrKyspICBxIFRoZSBvY3RldCBsZW5ndGggb2YgdGhlIGJpbmFyeSByZXByZXNlbnRhdGlvbiBvZiB0aGUgb2N0ZXQgbGVuZ3RoIG9mIHRoZSBwYXlsb2FkCiAgICAgICAgICAgICAgKCsrKykgIEEgbm9uY2UgKE4pLCBuIFRoZSBvY3RldCBsZW5ndGggb2YgdGhlIHdoZXJlIG4rcT0xNS4KICAgICAgICAgICAgICAoKysrKSAgRmxhZ3M6IG1vc3Qgc2lnbmlmaWNhbnQgb2N0ZXQgY29udGFpbmluZyBmb3VyIGZsYWdzIGZvciBjb250cm9sIGluZm9ybWF0aW9uLAogICAgICAgICAgICAgICgrKyspICB0IFRoZSBvY3RldCBsZW5ndGggb2YgdGhlIE1BQy4KICAgICAgICAgKCMjKSBCMSBibG9jayAoaGVhZGVyKSA6IGFzc29jaWF0ZWQgZGF0YSBsZW5ndGgoYSkgY29uY2F0ZW5hdGVkIHdpdGggQXNzb2NpYXRlZCBEYXRhIChBKQogICAgICAgICAgICAgIHRoZSBhc3NvY2lhdGVkIGRhdGEgbGVuZ3RoIGV4cHJlc3NlZCBpbiBieXRlcyAoYSkgZGVmaW5lZCBhcyBiZWxvdzoKICAgICAgICAgICAgKCsrKykgIElmIDAgPCBhIDwgMjE2LTI4LCB0aGVuIGl0IGlzIGVuY29kZWQgYXMgW2FdMTYsIGkuZS4gdHdvIG9jdGV0cwogICAgICAgICAgICAoKysrKSAgSWYgMjE2LTI4IDwgYSA8IDIzMiwgdGhlbiBpdCBpcyBlbmNvZGVkIGFzIDB4ZmYgfHwgMHhmZSB8fCBbYV0zMiwgaS5lLiBzaXggb2N0ZXRzCiAgICAgICAgICAgICgrKyspICBJZiAyMzIgPCBhIDwgMjY0LCB0aGVuIGl0IGlzIGVuY29kZWQgYXMgMHhmZiB8fCAweGZmIHx8IFthXTY0LCBpLmUuIHRlbiBvY3RldHMKICAgICAgICAgKCMjKSBDVFJ4IGJsb2NrICA6IGNvbnRyb2wgYmxvY2tzCiAgICAgICAgICAgICgrKyspIEdlbmVyYXRpb24gb2YgQ1RSMSBmcm9tIGZpcnN0IGJsb2NrIEIwIGluZm9ybWF0aW9uIDoKICAgICAgICAgICAgICBlcXVhbCB0byBCMCB3aXRoIGZpcnN0IDUgYml0cyB6ZXJvZWQgYW5kIG1vc3Qgc2lnbmlmaWNhbnQgYml0cyBzdG9yaW5nIG9jdGV0CiAgICAgICAgICAgICAgbGVuZ3RoIG9mIFAgYWxzbyB6ZXJvZWQsIHRoZW4gaW5jcmVtZW50ZWQgYnkgb25lICggc2VlIGJlbG93IFRhYmxlIDQpCiAgICAgICAgICAgICgrKyspIEdlbmVyYXRpb24gb2YgQ1RSMDogc2FtZSBhcyBDVFIxIHdpdGggYml0WzBdIHNldCB0byB6ZXJvLgoKICAgICAgKCMpICBGb3VyIHBoYXNlcyBhcmUgcGVyZm9ybWVkIGluIENDTSBmb3IgQ1JZUDEgSVA6CiAgICAgICAgICgjIykgSW5pdCBwaGFzZTogSVAgcHJlcGFyZXMgdGhlIEdDTSBoYXNoIHN1YmtleSAoSCkgYW5kIGRvIHRoZSBJViBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgSGVhZGVyIHBoYXNlOiBJUCBwcm9jZXNzZXMgdGhlIEFkZGl0aW9uYWwgQXV0aGVudGljYXRlZCBEYXRhIChBQUQpLCB3aXRoIGhhc2gKICAgICAgICAgIGNvbXB1dGF0aW9uIG9ubHkuCiAgICAgICAgICgjIykgUGF5bG9hZCBwaGFzZTogSVAgcHJvY2Vzc2VzIHRoZSBwbGFpbnRleHQgKFApIHdpdGggaGFzaCBjb21wdXRhdGlvbiArIGtleXN0cmVhbQogICAgICAgICAgZW5jcnlwdGlvbiArIGRhdGEgWE9SaW5nLiBJdCB3b3JrcyBpbiBhIHNpbWlsYXIgd2F5IGZvciBjaXBoZXJ0ZXh0IChDKS4KICAgICAgICAgKCMjKSBGaW5hbCBwaGFzZTogSVAgZ2VuZXJhdGVzIHRoZSBhdXRoZW50aWNhdGVkIHRhZyAoVCkgdXNpbmcgdGhlIGxhc3QgYmxvY2sgb2YgZGF0YS4KCiAgKioqIENhbGxiYWNrIHJlZ2lzdHJhdGlvbiAqKioKICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCiAgVGhlIGNvbXBpbGF0aW9uIGRlZmluZSAgVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyB3aGVuIHNldCB0byAxCiAgYWxsb3dzIHRoZSB1c2VyIHRvIGNvbmZpZ3VyZSBkeW5hbWljYWxseSB0aGUgZHJpdmVyIGNhbGxiYWNrcy4KICBVc2UgRnVuY3Rpb25zIEByZWYgSEFMX0NSWVBfUmVnaXN0ZXJDYWxsYmFjaygpIG9yIEhBTF9DUllQX1JlZ2lzdGVyWFhYQ2FsbGJhY2soKQogIHRvIHJlZ2lzdGVyIGFuIGludGVycnVwdCBjYWxsYmFjay4KCiAgRnVuY3Rpb24gQHJlZiBIQUxfQ1JZUF9SZWdpc3RlckNhbGxiYWNrKCkgYWxsb3dzIHRvIHJlZ2lzdGVyIGZvbGxvd2luZyBjYWxsYmFja3M6CiAgICAoKykgSW5DcGx0Q2FsbGJhY2sgICAgIDogIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrLgogICAgKCspIE91dENwbHRDYWxsYmFjayAgICA6IE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAgICgrKSBFcnJvckNhbGxiYWNrICAgICAgOiBjYWxsYmFjayBmb3IgZXJyb3IgZGV0ZWN0aW9uLgogICAgKCspIE1zcEluaXRDYWxsYmFjayAgICA6IENSWVAgTXNwSW5pdC4KICAgICgrKSBNc3BEZUluaXRDYWxsYmFjayAgOiBDUllQIE1zcERlSW5pdC4KICBUaGlzIGZ1bmN0aW9uIHRha2VzIGFzIHBhcmFtZXRlcnMgdGhlIEhBTCBwZXJpcGhlcmFsIGhhbmRsZSwgdGhlIENhbGxiYWNrIElECiAgYW5kIGEgcG9pbnRlciB0byB0aGUgdXNlciBjYWxsYmFjayBmdW5jdGlvbi4KCiAgVXNlIGZ1bmN0aW9uIEByZWYgSEFMX0NSWVBfVW5SZWdpc3RlckNhbGxiYWNrKCkgdG8gcmVzZXQgYSBjYWxsYmFjayB0byB0aGUgZGVmYXVsdAogIHdlYWsgZnVuY3Rpb24uCiAgQHJlZiBIQUxfQ1JZUF9VblJlZ2lzdGVyQ2FsbGJhY2soKSB0YWtlcyBhcyBwYXJhbWV0ZXJzIHRoZSBIQUwgcGVyaXBoZXJhbCBoYW5kbGUsCiAgYW5kIHRoZSBDYWxsYmFjayBJRC4KICBUaGlzIGZ1bmN0aW9uIGFsbG93cyB0byByZXNldCBmb2xsb3dpbmcgY2FsbGJhY2tzOgogICAgKCspIEluQ3BsdENhbGxiYWNrICAgICA6ICBJbnB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAgICgrKSBPdXRDcGx0Q2FsbGJhY2sgICAgOiBPdXRwdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2suCiAgICAoKykgRXJyb3JDYWxsYmFjayAgICAgIDogY2FsbGJhY2sgZm9yIGVycm9yIGRldGVjdGlvbi4KICAgICgrKSBNc3BJbml0Q2FsbGJhY2sgICAgOiBDUllQIE1zcEluaXQuCiAgICAoKykgTXNwRGVJbml0Q2FsbGJhY2sgIDogQ1JZUCBNc3BEZUluaXQuCgogIEJ5IGRlZmF1bHQsIGFmdGVyIHRoZSBAcmVmIEhBTF9DUllQX0luaXQoKSBhbmQgd2hlbiB0aGUgc3RhdGUgaXMgSEFMX0NSWVBfU1RBVEVfUkVTRVQKICBhbGwgY2FsbGJhY2tzIGFyZSBzZXQgdG8gdGhlIGNvcnJlc3BvbmRpbmcgd2VhayBmdW5jdGlvbnMgOgogIGV4YW1wbGVzIEByZWYgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soKSAsIEByZWYgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKCkuCiAgRXhjZXB0aW9uIGRvbmUgZm9yIE1zcEluaXQgYW5kIE1zcERlSW5pdCBmdW5jdGlvbnMgdGhhdCBhcmUKICByZXNldCB0byB0aGUgbGVnYWN5IHdlYWsgZnVuY3Rpb24gaW4gdGhlIEByZWYgSEFMX0NSWVBfSW5pdCgpLyBAcmVmIEhBTF9DUllQX0RlSW5pdCgpIG9ubHkgd2hlbgogIHRoZXNlIGNhbGxiYWNrcyBhcmUgbnVsbCAobm90IHJlZ2lzdGVyZWQgYmVmb3JlaGFuZCkuCiAgaWYgbm90LCBNc3BJbml0IG9yIE1zcERlSW5pdCBhcmUgbm90IG51bGwsIHRoZSBAcmVmIEhBTF9DUllQX0luaXQoKSAvIEByZWYgSEFMX0NSWVBfRGVJbml0KCkKICBrZWVwIGFuZCB1c2UgdGhlIHVzZXIgTXNwSW5pdC9Nc3BEZUluaXQgZnVuY3Rpb25zIChyZWdpc3RlcmVkIGJlZm9yZWhhbmQpCgogIENhbGxiYWNrcyBjYW4gYmUgcmVnaXN0ZXJlZC91bnJlZ2lzdGVyZWQgaW4gSEFMX0NSWVBfU1RBVEVfUkVBRFkgc3RhdGUgb25seS4KICBFeGNlcHRpb24gZG9uZSBNc3BJbml0L01zcERlSW5pdCBjYWxsYmFja3MgdGhhdCBjYW4gYmUgcmVnaXN0ZXJlZC91bnJlZ2lzdGVyZWQKICBpbiBIQUxfQ1JZUF9TVEFURV9SRUFEWSBvciBIQUxfQ1JZUF9TVEFURV9SRVNFVCBzdGF0ZSwKICB0aHVzIHJlZ2lzdGVyZWQgKHVzZXIpIE1zcEluaXQvRGVJbml0IGNhbGxiYWNrcyBjYW4gYmUgdXNlZCBkdXJpbmcgdGhlIEluaXQvRGVJbml0LgogIEluIHRoYXQgY2FzZSBmaXJzdCByZWdpc3RlciB0aGUgTXNwSW5pdC9Nc3BEZUluaXQgdXNlciBjYWxsYmFja3MKICB1c2luZyBAcmVmIEhBTF9DUllQX1JlZ2lzdGVyQ2FsbGJhY2soKSBiZWZvcmUgY2FsbGluZyBAcmVmIEhBTF9DUllQX0RlSW5pdCgpCiAgb3IgQHJlZiBIQUxfQ1JZUF9Jbml0KCkgZnVuY3Rpb24uCgogIFdoZW4gVGhlIGNvbXBpbGF0aW9uIGRlZmluZSBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTIGlzIHNldCB0byAwIG9yCiAgbm90IGRlZmluZWQsIHRoZSBjYWxsYmFjayByZWdpc3RyYXRpb24gZmVhdHVyZSBpcyBub3QgYXZhaWxhYmxlIGFuZCBhbGwgY2FsbGJhY2tzCiAgYXJlIHNldCB0byB0aGUgY29ycmVzcG9uZGluZyB3ZWFrIGZ1bmN0aW9ucy4KCiAgVGFibGUgMS4gSW5pdGlhbCBDb3VudGVyIEJsb2NrIChJQ0IpCiAgICAgICAgICArLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKICAgICAgICAgIHwgICAgICAgSW5pdGlhbGl6YXRpb24gdmVjdG9yIChJVikgICAgICB8ICBDb3VudGVyICAgICAgfAogICAgICAgICAgfC0tLS0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLXwtLS0tLS0tLS18CiAgICAgICAgIDEyNyAgICAgICAgICAgICAgOTUgICAgICAgICAgICAgICAgNjMgICAgICAgICAgICAzMSAgICAgICAwCgoKICAgICAgICAgICAgICBCaXQgTnVtYmVyICAgIFJlZ2lzdGVyICAgICAgICAgICBDb250ZW50cwogICAgICAgICAgICAgIC0tLS0tLS0tLS0gICAtLS0tLS0tLS0tLS0tLS0gICAgICAgLS0tLS0tLS0tLS0KICAgICAgICAgICAgICAxMjcgLi4uOTYgICAgQ1JZUF9JVjFSWzMxOjBdICAgICBJQ0JbMTI3Ojk2XQogICAgICAgICAgICAgIDk1ICAuLi42NCAgICBDUllQX0lWMUxbMzE6MF0gICAgIEIwWzk1OjY0XQogICAgICAgICAgICAgIDYzIC4uLiAzMiAgICBDUllQX0lWMFJbMzE6MF0gICAgIElDQls2MzozMl0KICAgICAgICAgICAgICAzMSAuLi4gMCAgICAgQ1JZUF9JVjBMWzMxOjBdICAgICBJQ0JbMzE6MF0sIHdoZXJlIDMyLWJpdCBjb3VudGVyPSAweDIKCiAgVGFibGUgMi4gIEdDTSBsYXN0IGJsb2NrIGRlZmluaXRpb24KCiAgICAgICAgICArLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKICAgICAgICAgIHwgIEJpdFswXSAgIHwgIEJpdFszMl0gICAgICAgICAgIHwgIEJpdFs2NF0gIHwgQml0Wzk2XSAgICAgICAgICAgICAgfAogICAgICAgICAgfC0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CiAgICAgICAgICB8ICAgMHgwICAgICB8IEhlYWRlciBsZW5ndGhbMzE6MF18ICAgICAweDAgICB8IFBheWxvYWQgbGVuZ3RoWzMxOjBdIHwKICAgICAgICAgIHwtLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAoKICBUYWJsZSAzLiBCMCBibG9jawogICAgICAgICAgICAgICAgT2N0ZXQgTnVtYmVyICAgQ29udGVudHMKICAgICAgICAgICAgICAgIC0tLS0tLS0tLS0tLSAgIC0tLS0tLS0tLQogICAgICAgICAgICAgICAgMCAgICAgICAgICAgICAgRmxhZ3MKICAgICAgICAgICAgICAgIDEgLi4uIDE1LXEgICAgIE5vbmNlIE4KICAgICAgICAgICAgICAgIDE2LXEgLi4uIDE1ICAgIFEKCiAgICAgICAgICAgIHRoZSBGbGFncyBmaWVsZCBpcyBmb3JtYXR0ZWQgYXMgZm9sbG93czoKCiAgICAgICAgICAgICAgICBCaXQgTnVtYmVyICAgQ29udGVudHMKICAgICAgICAgICAgICAgIC0tLS0tLS0tLS0gICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICAgICAgICAgICAgICA3ICAgICAgICAgICAgUmVzZXJ2ZWQgKGFsd2F5cyB6ZXJvKQogICAgICAgICAgICAgICAgNiAgICAgICAgICAgIEFkYXRhCiAgICAgICAgICAgICAgICA1IC4uLiAzICAgICAgKHQtMikvMgogICAgICAgICAgICAgICAgMiAuLi4gMCAgICAgIFtxLTFdMwoKIFRhYmxlIDQuIENUUnggYmxvY2sKICAgICAgICAgICAgICAgIEJpdCBOdW1iZXIgICAgUmVnaXN0ZXIgICAgICAgICAgIENvbnRlbnRzCiAgICAgICAgICAgICAgICAtLS0tLS0tLS0tICAgLS0tLS0tLS0tLS0tLS0tICAgICAgIC0tLS0tLS0tLS0tCiAgICAgICAgICAgICAgICAxMjcgLi4uOTYgICAgQ1JZUF9JVjFSWzMxOjBdICAgICBCMFsxMjc6OTZdLCB3aGVyZSBRIGxlbmd0aCBiaXRzIGFyZSBzZXQgdG8gMCwgZXhjZXB0IGZvcgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYml0IDAgdGhhdCBpcyBzZXQgdG8gMQogICAgICAgICAgICAgICAgOTUgIC4uLjY0ICAgIENSWVBfSVYxTFszMTowXSAgICAgQjBbOTU6NjRdCiAgICAgICAgICAgICAgICA2MyAuLi4gMzIgICAgQ1JZUF9JVjBSWzMxOjBdICAgICBCMFs2MzozMl0KICAgICAgICAgICAgICAgIDMxIC4uLiAwICAgICBDUllQX0lWMExbMzE6MF0gICAgIEIwWzMxOjBdLCB3aGVyZSBmbGFnIGJpdHMgc2V0IHRvIDAKCiAgQGVuZHZlcmJhdGltCiAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAgKiBAYXR0ZW50aW9uCiAgKgogICogPGgyPjxjZW50ZXI+JmNvcHk7IENvcHlyaWdodCAoYykgMjAxNiBTVE1pY3JvZWxlY3Ryb25pY3MuCiAgKiBBbGwgcmlnaHRzIHJlc2VydmVkLjwvY2VudGVyPjwvaDI+CiAgKgogICogVGhpcyBzb2Z0d2FyZSBjb21wb25lbnQgaXMgbGljZW5zZWQgYnkgU1QgdW5kZXIgQlNEIDMtQ2xhdXNlIGxpY2Vuc2UsCiAgKiB0aGUgIkxpY2Vuc2UiOyBZb3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlCiAgKiBMaWNlbnNlLiBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQ6CiAgKiAgICAgICAgICAgICAgICAgICAgICAgIG9wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL0JTRC0zLUNsYXVzZQogICoKICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICAqLwoKLyogSW5jbHVkZXMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KI2luY2x1ZGUgInN0bTMyZjd4eF9oYWwuaCIKCi8qKiBAYWRkdG9ncm91cCBTVE0zMkY3eHhfSEFMX0RyaXZlcgogICogQHsKICAqLwoKI2lmIGRlZmluZWQgKEFFUykgIHx8IGRlZmluZWQgKENSWVApCgovKiogQGRlZmdyb3VwIENSWVAgQ1JZUAogICogQGJyaWVmIENSWVAgSEFMIG1vZHVsZSBkcml2ZXIuCiAgKiBAewogICovCgoKI2lmZGVmIEhBTF9DUllQX01PRFVMRV9FTkFCTEVECgovKiBQcml2YXRlIHR5cGVkZWYgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiBQcml2YXRlIGRlZmluZSAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiogQGFkZHRvZ3JvdXAgQ1JZUF9Qcml2YXRlX0RlZmluZXMKICAqIEB7CiAgKi8KI2RlZmluZSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT04gICAgICA4MlUgICAgICAgICAvKlRoZSBsYXRlbmN5IG9mIGtleSBwcmVwYXJhdGlvbiBvcGVyYXRpb24gaXMgODIgY2xvY2sgY3ljbGVzLiovCiNkZWZpbmUgQ1JZUF9USU1FT1VUX0dDTUNDTUlOSVRQSEFTRSAgICAgMjk5VSAgICAgICAgLyogIFRoZSBsYXRlbmN5IG9mICBHQ00vQ0NNIGluaXQgcGhhc2UgdG8gcHJlcGFyZSBoYXNoIHN1YmtleSBpcyAyOTkgY2xvY2sgY3ljbGVzLiovCiNkZWZpbmUgQ1JZUF9USU1FT1VUX0dDTUNDTUhFQURFUlBIQVNFICAgMjkwVSAgICAgICAgLyogIFRoZSBsYXRlbmN5IG9mICBHQ00vQ0NNIGhlYWRlciBwaGFzZSBpcyAyOTAgY2xvY2sgY3ljbGVzLiovCgojZGVmaW5lICBDUllQX1BIQVNFX1JFQURZICAgICAgICAgICAgICAgIDB4MDAwMDAwMDFVIC8qITwgQ1JZUCBwZXJpcGhlcmFsIGlzIHJlYWR5IGZvciBpbml0aWFsaXphdGlvbi4gKi8KI2RlZmluZSAgQ1JZUF9QSEFTRV9QUk9DRVNTICAgICAgICAgICAgICAweDAwMDAwMDAyVSAvKiE8IENSWVAgcGVyaXBoZXJhbCBpcyBpbiBwcm9jZXNzaW5nIHBoYXNlICovCgojaWYgZGVmaW5lZChBRVMpCiNkZWZpbmUgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQgICAgICAgICAgICAgICAgICAgMHgwMDAwMDAwMFUgICAgIC8qITwgRW5jcnlwdGlvbiBtb2RlKE1vZGUgMSkgICovCiNkZWZpbmUgQ1JZUF9PUEVSQVRJTkdNT0RFX0tFWURFUklWQVRJT04gICAgICAgICAgICAgQUVTX0NSX01PREVfMCAgIC8qITwgS2V5IGRlcml2YXRpb24gbW9kZSAgb25seSB1c2VkIHdoZW4gcGVyZm9ybWluZyBFQ0IgYW5kIENCQyBkZWNyeXB0aW9ucyAoTW9kZSAyKSAqLwojZGVmaW5lIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUICAgICAgICAgICAgICAgICAgIEFFU19DUl9NT0RFXzEgICAvKiE8IERlY3J5cHRpb24gICAgKE1vZGUgMykgICAgKi8KI2RlZmluZSBDUllQX09QRVJBVElOR01PREVfS0VZREVSSVZBVElPTl9ERUNSWVBUICAgICBBRVNfQ1JfTU9ERSAgICAgLyohPCBLZXkgZGVyaXZhdGlvbiBhbmQgZGVjcnlwdGlvbiBvbmx5IHVzZWQgd2hlbiBwZXJmb3JtaW5nIEVDQiBhbmQgQ0JDIGRlY3J5cHRpb25zIChNb2RlIDQpICovCiNkZWZpbmUgQ1JZUF9QSEFTRV9JTklUICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMHgwMDAwMDAwMFUgICAgICAgIC8qITwgR0NNL0dNQUMgKG9yIENDTSkgaW5pdCBwaGFzZSAqLwojZGVmaW5lIENSWVBfUEhBU0VfSEVBREVSICAgICAgICAgICAgICAgICAgICAgICAgICAgIEFFU19DUl9HQ01QSF8wICAgICAvKiE8IEdDTS9HTUFDIG9yIENDTSBoZWFkZXIgcGhhc2UgKi8KI2RlZmluZSBDUllQX1BIQVNFX1BBWUxPQUQgICAgICAgICAgICAgICAgICAgICAgICAgICBBRVNfQ1JfR0NNUEhfMSAgICAgLyohPCBHQ00oL0NDTSkgcGF5bG9hZCBwaGFzZSAgICAgICovCiNkZWZpbmUgQ1JZUF9QSEFTRV9GSU5BTCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQUVTX0NSX0dDTVBIICAgICAgIC8qITwgR0NNL0dNQUMgb3IgQ0NNICBmaW5hbCBwaGFzZSAqLwojZWxzZSAgLyogQ1JZUCAqLwojZGVmaW5lIENSWVBfUEhBU0VfSU5JVCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDB4MDAwMDAwMDBVICAgICAgICAgICAgIC8qITwgR0NNL0dNQUMgKG9yIENDTSkgaW5pdCBwaGFzZSAqLwojZGVmaW5lIENSWVBfUEhBU0VfSEVBREVSICAgICAgICAgICAgICAgICAgICAgICAgICAgIENSWVBfQ1JfR0NNX0NDTVBIXzAgICAgIC8qITwgR0NNL0dNQUMgb3IgQ0NNIGhlYWRlciBwaGFzZSAqLwojZGVmaW5lIENSWVBfUEhBU0VfUEFZTE9BRCAgICAgICAgICAgICAgICAgICAgICAgICAgIENSWVBfQ1JfR0NNX0NDTVBIXzEgICAgIC8qITwgR0NNKC9DQ00pIHBheWxvYWQgcGhhc2UgICAgICAqLwojZGVmaW5lIENSWVBfUEhBU0VfRklOQUwgICAgICAgICAgICAgICAgICAgICAgICAgICAgIENSWVBfQ1JfR0NNX0NDTVBIICAgICAgIC8qITwgR0NNL0dNQUMgb3IgQ0NNICBmaW5hbCBwaGFzZSAqLwojZGVmaW5lIENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUICAgICAgICAgICAgICAgICAgIDB4MDAwMDAwMDBVICAgICAgICAgICAgIC8qITwgRW5jcnlwdGlvbiBtb2RlICAgKi8KI2RlZmluZSBDUllQX09QRVJBVElOR01PREVfREVDUllQVCAgICAgICAgICAgICAgICAgICBDUllQX0NSX0FMR09ESVIgICAgICAgICAvKiE8IERlY3J5cHRpb24gICAgICAgICovCiNlbmRpZiAvKiBFbmQgQ1JZUCBvciAgQUVTICovCgovKiAgQ1RSMSBpbmZvcm1hdGlvbiB0byB1c2UgaW4gQ0NNIGFsZ29yaXRobSAqLwojZGVmaW5lIENSWVBfQ0NNX0NUUjFfMCAgICAgICAgICAgICAgICAgIDB4MDdGRkZGRkZVCiNkZWZpbmUgQ1JZUF9DQ01fQ1RSMV8xICAgICAgICAgICAgICAgICAgMHhGRkZGRkYwMFUKI2RlZmluZSBDUllQX0NDTV9DVFIxXzIgICAgICAgICAgICAgICAgICAweDAwMDAwMDAxVQoKCi8qKgogICogQH0KICAqLwoKCi8qIFByaXZhdGUgbWFjcm8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi8qKiBAYWRkdG9ncm91cCBDUllQX1ByaXZhdGVfTWFjcm9zCiAgKiBAewogICovCgojaWYgZGVmaW5lZChDUllQKQoKI2RlZmluZSBDUllQX1NFVF9QSEFTRShfX0hBTkRMRV9fLCBfX1BIQVNFX18pICBkb3soX19IQU5ETEVfXyktPkluc3RhbmNlLT5DUiAmPSAodWludDMyX3QpKH5DUllQX0NSX0dDTV9DQ01QSCk7XAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChfX0hBTkRMRV9fKS0+SW5zdGFuY2UtPkNSIHw9ICh1aW50MzJfdCkoX19QSEFTRV9fKTtcCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9d2hpbGUoMCkKCiNkZWZpbmUgSEFMX0NSWVBfRklGT19GTFVTSChfX0hBTkRMRV9fKSAoKF9fSEFORExFX18pLT5JbnN0YW5jZS0+Q1IgfD0gIENSWVBfQ1JfRkZMVVNIKQoKI2Vsc2UgLypBRVMqLwojZGVmaW5lIENSWVBfU0VUX1BIQVNFKF9fSEFORExFX18sIF9fUEhBU0VfXykgIGRveyhfX0hBTkRMRV9fKS0+SW5zdGFuY2UtPkNSICY9ICh1aW50MzJfdCkofkFFU19DUl9HQ01QSCk7XAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChfX0hBTkRMRV9fKS0+SW5zdGFuY2UtPkNSIHw9ICh1aW50MzJfdCkoX19QSEFTRV9fKTtcCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9d2hpbGUoMCkKI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCovCgoKLyoqCiAgKiBAfQogICovCgovKiBQcml2YXRlIHN0cnVjdCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyogUHJpdmF0ZSB2YXJpYWJsZXMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyogUHJpdmF0ZSBmdW5jdGlvbiBwcm90b3R5cGVzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyoqIEBhZGR0b2dyb3VwIENSWVBfUHJpdmF0ZV9GdW5jdGlvbnNfcHJvdG90eXBlcwogICogQHsKICAqLwoKc3RhdGljIHZvaWQgQ1JZUF9TZXRETUFDb25maWcoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgaW5wdXRhZGRyLCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCBvdXRwdXRhZGRyKTsKc3RhdGljIHZvaWQgQ1JZUF9ETUFJbkNwbHQoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpOwpzdGF0aWMgdm9pZCBDUllQX0RNQU91dENwbHQoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpOwpzdGF0aWMgdm9pZCBDUllQX0RNQUVycm9yKERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKTsKc3RhdGljIHZvaWQgQ1JZUF9TZXRLZXkoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgS2V5U2l6ZSk7CnN0YXRpYyB2b2lkIENSWVBfQUVTX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwoKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgdm9pZCBDUllQX0dDTUNDTV9TZXRQYXlsb2FkUGhhc2VfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyB2b2lkIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9HQ01DQ01fU2V0SGVhZGVyUGhhc2VfRE1BKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgdm9pZCBDUllQX1dvcmthcm91bmQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU0dDTV9Qcm9jZXNzX0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTR0NNX1Byb2Nlc3NfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU0dDTV9Qcm9jZXNzKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNDQ01fUHJvY2VzcyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTQ0NNX1Byb2Nlc3NfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU0NDTV9Qcm9jZXNzX0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKCnN0YXRpYyB2b2lkIENSWVBfQUVTX1Byb2Nlc3NEYXRhKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXB0LCB1aW50MzJfdCBUaW1lb3V0KTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0VuY3J5cHQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19EZWNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0VuY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19EZWNyeXB0X0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKI2lmIGRlZmluZWQgKENSWVApCnN0YXRpYyB2b2lkIENSWVBfVERFU19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1dhaXRPbklGRU1GbGFnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwoKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uQlVTWUZsYWcoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1dhaXRPbk9GTkVGbGFnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9UREVTX1Byb2Nlc3MoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CiNlbHNlIC8qQUVTKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uQ0NGbGFnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwojZW5kaWYgLyogRW5kIENSWVAgb3IgQUVTICovCgovKioKICAqIEB9CiAgKi8KCi8qIEV4cG9ydGVkIGZ1bmN0aW9ucyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwoKLyoqIEBkZWZncm91cCBDUllQX0V4cG9ydGVkX0Z1bmN0aW9ucyBDUllQIEV4cG9ydGVkIEZ1bmN0aW9ucwogICogQHsKICAqLwoKCi8qKiBAZGVmZ3JvdXAgQ1JZUF9FeHBvcnRlZF9GdW5jdGlvbnNfR3JvdXAxIEluaXRpYWxpemF0aW9uIGFuZCBkZS1pbml0aWFsaXphdGlvbiBmdW5jdGlvbnMKICAqICBAYnJpZWYgICAgSW5pdGlhbGl6YXRpb24gYW5kIENvbmZpZ3VyYXRpb24gZnVuY3Rpb25zLgogICoKQHZlcmJhdGltCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgICMjIyMjIEluaXRpYWxpemF0aW9uLCBkZS1pbml0aWFsaXphdGlvbiBhbmQgU2V0IGFuZCBHZXQgY29uZmlndXJhdGlvbiBmdW5jdGlvbnMgIyMjIyMKICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICBbLi5dICBUaGlzIHNlY3Rpb24gcHJvdmlkZXMgZnVuY3Rpb25zIGFsbG93aW5nIHRvOgogICAgICAoKykgSW5pdGlhbGl6ZSB0aGUgQ1JZUAogICAgICAoKykgRGVJbml0aWFsaXplIHRoZSBDUllQCiAgICAgICgrKSBJbml0aWFsaXplIHRoZSBDUllQIE1TUAogICAgICAoKykgRGVJbml0aWFsaXplIHRoZSBDUllQIE1TUAogICAgICAoKykgY29uZmlndXJlIENSWVAgKEhBTF9DUllQX1NldENvbmZpZykgd2l0aCB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMgaW4gdGhlIENSWVBfQ29uZmlnVHlwZURlZgogICAgICAgICAgUGFyYW1ldGVycyB3aGljaCBhcmUgY29uZmlndXJlZCBpbiBUaGlzIHNlY3Rpb24gYXJlIDoKICAgICAgICAgICgrKSBLZXkgc2l6ZQogICAgICAgICAgKCspIERhdGEgVHlwZSA6IDMyLDE2LCA4IG9yIDFiaXQKICAgICAgICAgICgrKSBBbGdvTW9kZSA6CiAgICAgICAgICAgICAgLSBmb3IgQ1JZUDEgSVAgOgogICAgICAgICAgICAgICAgIEVDQiBhbmQgQ0JDIGluIERFUy9UREVTIFN0YW5kYXJkCiAgICAgICAgICAgICAgICAgRUNCLENCQyxDVFIsR0NNL0dNQUMgYW5kIENDTSBpbiBBRVMgU3RhbmRhcmQuCiAgICAgICAgICAgICAgLSBmb3IgVGlueUFFUzIgSVAsIG9ubHkgRUNCLENCQyxDVFIsR0NNL0dNQUMgYW5kIENDTSBpbiBBRVMgU3RhbmRhcmQgYXJlIHN1cHBvcnRlZC4KICAgICAgKCspIEdldCBDUllQIGNvbmZpZ3VyYXRpb24gKEhBTF9DUllQX0dldENvbmZpZykgZnJvbSB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMgaW4gdGhlIENSWVBfSGFuZGxlVHlwZURlZgoKCkBlbmR2ZXJiYXRpbQogICogQHsKICAqLwoKCi8qKgogICogQGJyaWVmICBJbml0aWFsaXplcyB0aGUgQ1JZUCBhY2NvcmRpbmcgdG8gdGhlIHNwZWNpZmllZAogICogICAgICAgICBwYXJhbWV0ZXJzIGluIHRoZSBDUllQX0NvbmZpZ1R5cGVEZWYgYW5kIGNyZWF0ZXMgdGhlIGFzc29jaWF0ZWQgaGFuZGxlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0luaXQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIENoZWNrIHRoZSBDUllQIGhhbmRsZSBhbGxvY2F0aW9uICovCiAgaWYgKGhjcnlwID09IE5VTEwpCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIENoZWNrIHBhcmFtZXRlcnMgKi8KICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9LRVlTSVpFKGhjcnlwLT5Jbml0LktleVNpemUpKTsKICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9EQVRBVFlQRShoY3J5cC0+SW5pdC5EYXRhVHlwZSkpOwogIGFzc2VydF9wYXJhbShJU19DUllQX0FMR09SSVRITShoY3J5cC0+SW5pdC5BbGdvcml0aG0pKTsKICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9JTklUKGhjcnlwLT5Jbml0LktleUlWQ29uZmlnU2tpcCkpOwoKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRVNFVCkKICB7CiAgICAvKiBBbGxvY2F0ZSBsb2NrIHJlc291cmNlIGFuZCBpbml0aWFsaXplIGl0ICovCiAgICBoY3J5cC0+TG9jayA9IEhBTF9VTkxPQ0tFRDsKCiAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2sgID0gSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2s7ICAvKiBMZWdhY3kgd2VhayBJbkNwbHRDYWxsYmFjayAgICovCiAgICBoY3J5cC0+T3V0Q3BsdENhbGxiYWNrID0gSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrOyAvKiBMZWdhY3kgd2VhayBPdXRDcGx0Q2FsbGJhY2sgICovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayAgID0gSEFMX0NSWVBfRXJyb3JDYWxsYmFjazsgICAvKiBMZWdhY3kgd2VhayBFcnJvckNhbGxiYWNrICAgICovCgogICAgaWYgKGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2sgPT0gTlVMTCkKICAgIHsKICAgICAgaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9IEhBTF9DUllQX01zcEluaXQ7IC8qIExlZ2FjeSB3ZWFrIE1zcEluaXQgICovCiAgICB9CgogICAgLyogSW5pdCB0aGUgbG93IGxldmVsIGhhcmR3YXJlICovCiAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrKGhjcnlwKTsKICB9CiNlbHNlCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRVNFVCkKICB7CiAgICAvKiBBbGxvY2F0ZSBsb2NrIHJlc291cmNlIGFuZCBpbml0aWFsaXplIGl0ICovCiAgICBoY3J5cC0+TG9jayA9IEhBTF9VTkxPQ0tFRDsKCiAgICAvKiBJbml0IHRoZSBsb3cgbGV2ZWwgaGFyZHdhcmUgKi8KICAgIEhBTF9DUllQX01zcEluaXQoaGNyeXApOwogIH0KI2VuZGlmIC8qIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTKSAqLwoKICAvKiBTZXQgdGhlIGtleSBzaXplKFRoaXMgYml0IGZpZWxkIGlzIGRvbpJ0IGNhcmUgaW4gdGhlIERFUyBvciBUREVTIG1vZGVzKSBkYXRhIHR5cGUgYW5kIEFsZ29yaXRobSAqLwojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0RBVEFUWVBFIHwgQ1JZUF9DUl9LRVlTSVpFIHwgQ1JZUF9DUl9BTEdPTU9ERSwKICAgICAgICAgICAgIGhjcnlwLT5Jbml0LkRhdGFUeXBlIHwgaGNyeXAtPkluaXQuS2V5U2l6ZSB8IGhjcnlwLT5Jbml0LkFsZ29yaXRobSk7CgojZWxzZSAvKkFFUyovCgogIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RBVEFUWVBFIHwgQUVTX0NSX0tFWVNJWkUgfCBBRVNfQ1JfQ0hNT0QsCiAgICAgICAgICAgICBoY3J5cC0+SW5pdC5EYXRhVHlwZSB8IGhjcnlwLT5Jbml0LktleVNpemUgfCBoY3J5cC0+SW5pdC5BbGdvcml0aG0pOwoKI2VuZGlmICAvKiBFbmQgQUVTIG9yIENSWVAqLwoKICAvKiBSZXNldCBFcnJvciBDb2RlIGZpZWxkICovCiAgaGNyeXAtPkVycm9yQ29kZSA9IEhBTF9DUllQX0VSUk9SX05PTkU7CgogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBTZXQgdGhlIGRlZmF1bHQgQ1JZUCBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUkVBRFk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgRGUtSW5pdGlhbGl6ZXMgdGhlIENSWVAgcGVyaXBoZXJhbC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9EZUluaXQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIENoZWNrIHRoZSBDUllQIGhhbmRsZSBhbGxvY2F0aW9uICovCiAgaWYgKGhjcnlwID09IE5VTEwpCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIFNldCB0aGUgZGVmYXVsdCBDUllQIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9SRUFEWTsKCiAgLyogUmVzZXQgQ3J5cEluQ291bnQgYW5kIENyeXBPdXRDb3VudCAqLwogIGhjcnlwLT5DcnlwSW5Db3VudCA9IDA7CiAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IDA7CiAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDA7CgogIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKCiAgaWYgKGhjcnlwLT5Nc3BEZUluaXRDYWxsYmFjayA9PSBOVUxMKQogIHsKICAgIGhjcnlwLT5Nc3BEZUluaXRDYWxsYmFjayA9IEhBTF9DUllQX01zcERlSW5pdDsgLyogTGVnYWN5IHdlYWsgTXNwRGVJbml0ICAqLwogIH0KICAvKiBEZUluaXQgdGhlIGxvdyBsZXZlbCBoYXJkd2FyZSAqLwogIGhjcnlwLT5Nc3BEZUluaXRDYWxsYmFjayhoY3J5cCk7CgojZWxzZQoKICAvKiBEZUluaXQgdGhlIGxvdyBsZXZlbCBoYXJkd2FyZTogQ0xPQ0ssIE5WSUMuKi8KICBIQUxfQ1JZUF9Nc3BEZUluaXQoaGNyeXApOwoKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KCiAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVTRVQ7CgogIC8qIFJlbGVhc2UgTG9jayAqLwogIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgQ29uZmlndXJlIHRoZSBDUllQIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkCiAgKiAgICAgICAgIHBhcmFtZXRlcnMgaW4gdGhlIENSWVBfQ29uZmlnVHlwZURlZgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUKICAqIEBwYXJhbSAgcENvbmY6IHBvaW50ZXIgdG8gYSBDUllQX0NvbmZpZ1R5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9TZXRDb25maWcoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgQ1JZUF9Db25maWdUeXBlRGVmICpwQ29uZikKewogIC8qIENoZWNrIHRoZSBDUllQIGhhbmRsZSBhbGxvY2F0aW9uICovCiAgaWYgKChoY3J5cCA9PSBOVUxMKSB8fCAocENvbmYgPT0gTlVMTCkpCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIENoZWNrIHBhcmFtZXRlcnMgKi8KICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9LRVlTSVpFKHBDb25mLT5LZXlTaXplKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfREFUQVRZUEUocENvbmYtPkRhdGFUeXBlKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfQUxHT1JJVEhNKHBDb25mLT5BbGdvcml0aG0pKTsKCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qIFNldCAgQ1JZUCBwYXJhbWV0ZXJzICAqLwogICAgaGNyeXAtPkluaXQuRGF0YVR5cGUgICA9IHBDb25mLT5EYXRhVHlwZTsKICAgIGhjcnlwLT5Jbml0LnBLZXkgICAgICAgPSBwQ29uZi0+cEtleTsKICAgIGhjcnlwLT5Jbml0LkFsZ29yaXRobSAgPSBwQ29uZi0+QWxnb3JpdGhtOwogICAgaGNyeXAtPkluaXQuS2V5U2l6ZSAgICA9IHBDb25mLT5LZXlTaXplOwogICAgaGNyeXAtPkluaXQucEluaXRWZWN0ICA9IHBDb25mLT5wSW5pdFZlY3Q7CiAgICBoY3J5cC0+SW5pdC5IZWFkZXIgICAgID0gcENvbmYtPkhlYWRlcjsKICAgIGhjcnlwLT5Jbml0LkhlYWRlclNpemUgPSBwQ29uZi0+SGVhZGVyU2l6ZTsKICAgIGhjcnlwLT5Jbml0LkIwICAgICAgICAgPSBwQ29uZi0+QjA7CiAgICBoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID0gcENvbmYtPkRhdGFXaWR0aFVuaXQ7CgogICAgLyogU2V0IHRoZSBrZXkgc2l6ZShUaGlzIGJpdCBmaWVsZCBpcyBkb26SdCBjYXJlIGluIHRoZSBERVMgb3IgVERFUyBtb2RlcykgZGF0YSB0eXBlLCBBbGdvTW9kZSBhbmQgb3BlcmF0aW5nIG1vZGUqLwojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfREFUQVRZUEUgfCBDUllQX0NSX0tFWVNJWkUgfCBDUllQX0NSX0FMR09NT0RFLAogICAgICAgICAgICAgICBoY3J5cC0+SW5pdC5EYXRhVHlwZSB8IGhjcnlwLT5Jbml0LktleVNpemUgfCBoY3J5cC0+SW5pdC5BbGdvcml0aG0pOwoKI2Vsc2UgLypBRVMqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfREFUQVRZUEUgfCBBRVNfQ1JfS0VZU0laRSB8IEFFU19DUl9DSE1PRCwKICAgICAgICAgICAgICAgaGNyeXAtPkluaXQuRGF0YVR5cGUgfCBoY3J5cC0+SW5pdC5LZXlTaXplIHwgaGNyeXAtPkluaXQuQWxnb3JpdGhtKTsKCiAgICAvKmNsZWFyIGVycm9yIGZsYWdzKi8KICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9FUlJfQ0xFQVIpOwoKI2VuZGlmICAvKiBFbmQgQUVTIG9yIENSWVAgKi8KCiAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIFJlc2V0IEVycm9yIENvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgPSBIQUxfQ1JZUF9FUlJPUl9OT05FOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogU2V0IHRoZSBkZWZhdWx0IENSWVAgcGhhc2UgKi8KICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUkVBRFk7CgogICAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogICAgcmV0dXJuIEhBTF9PSzsKICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KfQoKLyoqCiAgKiBAYnJpZWYgIEdldCBDUllQIENvbmZpZ3VyYXRpb24gcGFyYW1ldGVycyBpbiBhc3NvY2lhdGVkIGhhbmRsZS4KICAqIEBwYXJhbSAgcENvbmY6IHBvaW50ZXIgdG8gYSBDUllQX0NvbmZpZ1R5cGVEZWYgc3RydWN0dXJlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfR2V0Q29uZmlnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIENSWVBfQ29uZmlnVHlwZURlZiAqcENvbmYpCnsKICAvKiBDaGVjayB0aGUgQ1JZUCBoYW5kbGUgYWxsb2NhdGlvbiAqLwogIGlmICgoaGNyeXAgPT0gTlVMTCkgfHwgKHBDb25mID09IE5VTEwpKQogIHsKICAgIHJldHVybiBIQUxfRVJST1I7CiAgfQoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogR2V0ICBDUllQIHBhcmFtZXRlcnMgICovCiAgICBwQ29uZi0+RGF0YVR5cGUgICAgICAgID0gaGNyeXAtPkluaXQuRGF0YVR5cGU7CiAgICBwQ29uZi0+cEtleSAgICAgICAgICAgID0gaGNyeXAtPkluaXQucEtleTsKICAgIHBDb25mLT5BbGdvcml0aG0gICAgICAgPSBoY3J5cC0+SW5pdC5BbGdvcml0aG07CiAgICBwQ29uZi0+S2V5U2l6ZSAgICAgICAgID0gaGNyeXAtPkluaXQuS2V5U2l6ZSA7CiAgICBwQ29uZi0+cEluaXRWZWN0ICAgICAgID0gaGNyeXAtPkluaXQucEluaXRWZWN0OwogICAgcENvbmYtPkhlYWRlciAgICAgICAgICA9IGhjcnlwLT5Jbml0LkhlYWRlciA7CiAgICBwQ29uZi0+SGVhZGVyU2l6ZSAgICAgID0gaGNyeXAtPkluaXQuSGVhZGVyU2l6ZTsKICAgIHBDb25mLT5CMCAgICAgICAgICAgICAgPSBoY3J5cC0+SW5pdC5CMDsKICAgIHBDb25mLT5EYXRhV2lkdGhVbml0ICAgID0gaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdDsKCiAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogICAgcmV0dXJuIEhBTF9PSzsKICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KfQovKioKICAqIEBicmllZiAgSW5pdGlhbGl6ZXMgdGhlIENSWVAgTVNQLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgTm9uZQogICovCl9fd2VhayB2b2lkIEhBTF9DUllQX01zcEluaXQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIFByZXZlbnQgdW51c2VkIGFyZ3VtZW50KHMpIGNvbXBpbGF0aW9uIHdhcm5pbmcgKi8KICBVTlVTRUQoaGNyeXApOwoKICAvKiBOT1RFIDogVGhpcyBmdW5jdGlvbiBTaG91bGQgbm90IGJlIG1vZGlmaWVkLCB3aGVuIHRoZSBjYWxsYmFjayBpcyBuZWVkZWQsCiAgICAgICAgICAgIHRoZSBIQUxfQ1JZUF9Nc3BJbml0IGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIERlSW5pdGlhbGl6ZXMgQ1JZUCBNU1AuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBOb25lCiAgKi8KX193ZWFrIHZvaWQgSEFMX0NSWVBfTXNwRGVJbml0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBQcmV2ZW50IHVudXNlZCBhcmd1bWVudChzKSBjb21waWxhdGlvbiB3YXJuaW5nICovCiAgVU5VU0VEKGhjcnlwKTsKCiAgLyogTk9URSA6IFRoaXMgZnVuY3Rpb24gU2hvdWxkIG5vdCBiZSBtb2RpZmllZCwgd2hlbiB0aGUgY2FsbGJhY2sgaXMgbmVlZGVkLAogICAgICAgICAgICB0aGUgSEFMX0NSWVBfTXNwRGVJbml0IGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCi8qKgogICogQGJyaWVmICBSZWdpc3RlciBhIFVzZXIgQ1JZUCBDYWxsYmFjawogICogICAgICAgICBUbyBiZSB1c2VkIGluc3RlYWQgb2YgdGhlIHdlYWsgcHJlZGVmaW5lZCBjYWxsYmFjawogICogQHBhcmFtIGhjcnlwIGNyeXAgaGFuZGxlCiAgKiBAcGFyYW0gQ2FsbGJhY2tJRCBJRCBvZiB0aGUgY2FsbGJhY2sgdG8gYmUgcmVnaXN0ZXJlZAogICogICAgICAgIFRoaXMgcGFyYW1ldGVyIGNhbiBiZSBvbmUgb2YgdGhlIGZvbGxvd2luZyB2YWx1ZXM6CiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfSU5QVVRfQ09NUExFVEVfQ0JfSUQgSW5wdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9PVVRQVVRfQ09NUExFVEVfQ0JfSUQgT3V0cHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrIElECiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfRVJST1JfQ0JfSUQgRXJyb3IgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9NU1BJTklUX0NCX0lEIE1zcEluaXQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgTXNwRGVJbml0IGNhbGxiYWNrIElECiAgKiBAcGFyYW0gcENhbGxiYWNrIHBvaW50ZXIgdG8gdGhlIENhbGxiYWNrIGZ1bmN0aW9uCiAgKiBAcmV0dmFsIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX1JlZ2lzdGVyQ2FsbGJhY2soQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgSEFMX0NSWVBfQ2FsbGJhY2tJRFR5cGVEZWYgQ2FsbGJhY2tJRCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwQ1JZUF9DYWxsYmFja1R5cGVEZWYgcENhbGxiYWNrKQp7CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICBpZiAocENhbGxiYWNrID09IE5VTEwpCiAgewogICAgLyogVXBkYXRlIHRoZSBlcnJvciBjb2RlICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CgogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CiAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICBzd2l0Y2ggKENhbGxiYWNrSUQpCiAgICB7CiAgICAgIGNhc2UgSEFMX0NSWVBfSU5QVVRfQ09NUExFVEVfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfT1VUUFVUX0NPTVBMRVRFX0NCX0lEIDoKICAgICAgICBoY3J5cC0+T3V0Q3BsdENhbGxiYWNrID0gcENhbGxiYWNrOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBIQUxfQ1JZUF9FUlJPUl9DQl9JRCA6CiAgICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX01TUERFSU5JVF9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID0gcENhbGxiYWNrOwogICAgICAgIGJyZWFrOwoKICAgICAgZGVmYXVsdCA6CiAgICAgICAgLyogVXBkYXRlIHRoZSBlcnJvciBjb2RlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgICAgIC8qIFJldHVybiBlcnJvciBzdGF0dXMgKi8KICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVTRVQpCiAgewogICAgc3dpdGNoIChDYWxsYmFja0lEKQogICAgewogICAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX01TUERFSU5JVF9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID0gcENhbGxiYWNrOwogICAgICAgIGJyZWFrOwoKICAgICAgZGVmYXVsdCA6CiAgICAgICAgLyogVXBkYXRlIHRoZSBlcnJvciBjb2RlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgICAgIC8qIFJldHVybiBlcnJvciBzdGF0dXMgKi8KICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogVXBkYXRlIHRoZSBlcnJvciBjb2RlICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CiAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogIH0KCiAgLyogUmVsZWFzZSBMb2NrICovCiAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgcmV0dXJuIHN0YXR1czsKfQoKLyoqCiAgKiBAYnJpZWYgIFVucmVnaXN0ZXIgYW4gQ1JZUCBDYWxsYmFjawogICogICAgICAgICBDUllQIGNhbGxiYWNrIGlzIHJlZGlyZWN0ZWQgdG8gdGhlIHdlYWsgcHJlZGVmaW5lZCBjYWxsYmFjawogICogQHBhcmFtIGhjcnlwIGNyeXAgaGFuZGxlCiAgKiBAcGFyYW0gQ2FsbGJhY2tJRCBJRCBvZiB0aGUgY2FsbGJhY2sgdG8gYmUgdW5yZWdpc3RlcmVkCiAgKiAgICAgICAgVGhpcyBwYXJhbWV0ZXIgY2FuIGJlIG9uZSBvZiB0aGUgZm9sbG93aW5nIHZhbHVlczoKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9JTlBVVF9DT01QTEVURV9DQl9JRCBJbnB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX09VVFBVVF9DT01QTEVURV9DQl9JRCBPdXRwdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9FUlJPUl9DQl9JRCBFcnJvciBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgTXNwSW5pdCBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX01TUERFSU5JVF9DQl9JRCBNc3BEZUluaXQgY2FsbGJhY2sgSUQKICAqIEByZXR2YWwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfVW5SZWdpc3RlckNhbGxiYWNrKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIEhBTF9DUllQX0NhbGxiYWNrSURUeXBlRGVmIENhbGxiYWNrSUQpCnsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CgogIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgX19IQUxfTE9DSyhoY3J5cCk7CgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgc3dpdGNoIChDYWxsYmFja0lEKQogICAgewogICAgICBjYXNlIEhBTF9DUllQX0lOUFVUX0NPTVBMRVRFX0NCX0lEIDoKICAgICAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjazsgIC8qIExlZ2FjeSB3ZWFrICBJbkNwbHRDYWxsYmFjayAgKi8KICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfT1VUUFVUX0NPTVBMRVRFX0NCX0lEIDoKICAgICAgICBoY3J5cC0+T3V0Q3BsdENhbGxiYWNrID0gSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrOyAgICAgICAgIC8qIExlZ2FjeSB3ZWFrIE91dENwbHRDYWxsYmFjayAgICAgICAqLwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBIQUxfQ1JZUF9FUlJPUl9DQl9JRCA6CiAgICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2sgPSBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrOyAgICAgICAgICAgLyogTGVnYWN5IHdlYWsgRXJyb3JDYWxsYmFjayAgICAgICAgKi8KICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgSEFMX0NSWVBfTVNQSU5JVF9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9IEhBTF9DUllQX01zcEluaXQ7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIEhBTF9DUllQX01TUERFSU5JVF9DQl9JRCA6CiAgICAgICAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwRGVJbml0OwogICAgICAgIGJyZWFrOwoKICAgICAgZGVmYXVsdCA6CiAgICAgICAgLyogVXBkYXRlIHRoZSBlcnJvciBjb2RlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgICAgIC8qIFJldHVybiBlcnJvciBzdGF0dXMgKi8KICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVTRVQpCiAgewogICAgc3dpdGNoIChDYWxsYmFja0lEKQogICAgewogICAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9Nc3BJbml0OwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgOgogICAgICAgIGhjcnlwLT5Nc3BEZUluaXRDYWxsYmFjayA9IEhBTF9DUllQX01zcERlSW5pdDsKICAgICAgICBicmVhazsKCiAgICAgIGRlZmF1bHQgOgogICAgICAgIC8qIFVwZGF0ZSB0aGUgZXJyb3IgY29kZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfSU5WQUxJRF9DQUxMQkFDSzsKICAgICAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIFVwZGF0ZSB0aGUgZXJyb3IgY29kZSAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgLyogUmV0dXJuIGVycm9yIHN0YXR1cyAqLwogICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICB9CgogIC8qIFJlbGVhc2UgTG9jayAqLwogIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogIHJldHVybiBzdGF0dXM7Cn0KI2VuZGlmIC8qIFVTRV9IQUxfVUFSVF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KLyoqCiAgKiBAfQogICovCgovKiogQGRlZmdyb3VwIENSWVBfRXhwb3J0ZWRfRnVuY3Rpb25zX0dyb3VwMiAgRW5jcnlwdCBEZWNyeXB0IGZ1bmN0aW9ucwogICogIEBicmllZiAgIHByb2Nlc3NpbmcgZnVuY3Rpb25zLgogICoKQHZlcmJhdGltCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICAgICAgICAgICAgICAgICAgICAjIyMjIyBFbmNyeXB0IERlY3J5cHQgIGZ1bmN0aW9ucyAjIyMjIwogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgWy4uXSAgVGhpcyBzZWN0aW9uIHByb3ZpZGVzIEFQSSBhbGxvd2luZyB0byBFbmNyeXB0L0RlY3J5cHQgRGF0YSBmb2xsb3dpbmcKICAgICAgICAgIFN0YW5kYXJkIERFUy9UREVTIG9yIEFFUywgYW5kIEFsZ29yaXRobSBjb25maWd1cmVkIGJ5IHRoZSB1c2VyOgogICAgICAoKykgU3RhbmRhcmQgREVTL1RERVMgb25seSBzdXBwb3J0ZWQgYnkgQ1JZUDEgSVAsIGJlbG93IGxpc3Qgb2YgQWxnb3JpdGhtIHN1cHBvcnRlZCA6CiAgICAgICAgICAgLSBFbGVjdHJvbmljIENvZGUgQm9vayhFQ0IpCiAgICAgICAgICAgLSBDaXBoZXIgQmxvY2sgQ2hhaW5pbmcgKENCQykKICAgICAgKCspIFN0YW5kYXJkIEFFUyAgc3VwcG9ydGVkIGJ5IENSWVAxIElQICYgVGlueUFFUywgbGlzdCBvZiBBbGdvcml0aG0gc3VwcG9ydGVkOgogICAgICAgICAgIC0gRWxlY3Ryb25pYyBDb2RlIEJvb2soRUNCKQogICAgICAgICAgIC0gQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICAgICAgLSBDb3VudGVyIG1vZGUgKENUUikKICAgICAgICAgICAtIENpcGhlciBCbG9jayBDaGFpbmluZyAoQ0JDKQogICAgICAgICAgIC0gQ291bnRlciBtb2RlIChDVFIpCiAgICAgICAgICAgLSBHYWxvaXMvY291bnRlciBtb2RlIChHQ00pCiAgICAgICAgICAgLSBDb3VudGVyIHdpdGggQ2lwaGVyIEJsb2NrIENoYWluaW5nLU1lc3NhZ2UoQ0NNKQogICAgWy4uXSAgVGhyZWUgcHJvY2Vzc2luZyBmdW5jdGlvbnMgYXJlIGF2YWlsYWJsZToKICAgICAgKCspIFBvbGxpbmcgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHQgJiBIQUxfQ1JZUF9EZWNyeXB0CiAgICAgICgrKSBJbnRlcnJ1cHQgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHRfSVQgJiBIQUxfQ1JZUF9EZWNyeXB0X0lUCiAgICAgICgrKSBETUEgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHRfRE1BICYgSEFMX0NSWVBfRGVjcnlwdF9ETUEKCkBlbmR2ZXJiYXRpbQogICogQHsKICAqLwoKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChwbGFpbnRleHQpCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBpbiB3b3JkLgogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIoY2lwaGVydGV4dCkKICAqIEBwYXJhbSAgVGltZW91dDogU3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9FbmNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgYWxnbzsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXM7CgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyIGFuZCBwQ3J5cE91dEJ1ZmZQdHIgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgojaWYgZGVmaW5lZCAoQ1JZUCkKICAgIC8qIFNldCBFbmNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPRElSLCBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCk7CgogICAgLyogYWxnbyBnZXQgYWxnb3JpdGhtIHNlbGVjdGVkICovCiAgICBhbGdvID0gaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT01PREU7CgogICAgc3dpdGNoIChhbGdvKQogICAgewogICAgICBjYXNlIENSWVBfREVTX0VDQjoKICAgICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgICAgY2FzZSBDUllQX1RERVNfQ0JDOgoKICAgICAgICAvKlNldCBLZXkgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDEpOwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDQpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgICAgfQoKICAgICAgICAvKlNldCBJbml0aWFsaXphdGlvbiBWZWN0b3IgKElWKSovCiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgICB9CgogICAgICAgIC8qIEZsdXNoIEZJRk8gKi8KICAgICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAgICAgLyogU3RhdHJ0IERFUy9UREVTIGVuY3J5cHRpb24gcHJvY2VzcyAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfVERFU19Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIC8qIEFFUyBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRW5jcnlwdChoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CiAgICAgICAgCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNOgoKICAgICAgICAvKiBBRVMgR0NNIGVuY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KTsKCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2VzcyhoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CiAgICAgICAgCiAgICAgIGRlZmF1bHQ6CiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9OT1RfU1VQUE9SVEVEOwogICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KCiNlbHNlIC8qQUVTKi8KCiAgICAvKiBTZXQgdGhlIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQUVTX0NSX0NITU9EOwoKICAgIHN3aXRjaCAoYWxnbykKICAgIHsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIC8qIEFFUyBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRW5jcnlwdChoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0dDTV9HTUFDOgoKICAgICAgICAvKiBBRVMgR0NNIGVuY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KSA7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2VzcyhoY3J5cCwgVGltZW91dCk7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICB9CiNlbmRpZiAvKmVuZCBBRVMgb3IgQ1JZUCAqLwoKICAgIGlmIChzdGF0dXMgPT0gSEFMX09LKQogICAgewogICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBtb2RlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgSW5wdXQ6IFBvaW50ZXIgdG8gdGhlIGlucHV0IGJ1ZmZlciAoY2lwaGVydGV4dCApCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBpbiB3b3JkLgogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIocGxhaW50ZXh0KQogICogQHBhcmFtICBUaW1lb3V0OiBTcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0RlY3J5cHQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgKklucHV0LCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCAqT3V0cHV0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQzMl90IFRpbWVvdXQpCnsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXM7CiAgdWludDMyX3QgYWxnbzsKCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2Ugc3RhdGUgQnVzeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogIFJlc2V0IENyeXBJbkNvdW50LCBDcnlwT3V0Q291bnQgYW5kIEluaXRpYWxpemUgcENyeXBJbkJ1ZmZQdHIgYW5kIHBDcnlwT3V0QnVmZlB0ciAgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgICAvKiBTZXQgRGVjcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT0RJUiwgQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09NT0RFOwoKICAgIHN3aXRjaCAoYWxnbykKICAgIHsKICAgICAgY2FzZSBDUllQX0RFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9ERVNfQ0JDOgogICAgICBjYXNlIENSWVBfVERFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0NCQzoKCiAgICAgICAgLypTZXQgS2V5ICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAxKTsKICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfRUNCKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMyk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDUpOwogICAgICAgIH0KCiAgICAgICAgLypTZXQgSW5pdGlhbGl6YXRpb24gVmVjdG9yIChJVikqLwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfREVTX0NCQykgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgICB7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICAgICAgfQoKICAgICAgICAvKiBGbHVzaCBGSUZPICovCiAgICAgICAgSEFMX0NSWVBfRklGT19GTFVTSChoY3J5cCk7CgogICAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgICAgIC8qIFN0YXJ0IERFUy9UREVTIGRlY3J5cHRpb24gcHJvY2VzcyAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfVERFU19Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KTsKCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICAvKiBBRVMgZGVjcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTX0RlY3J5cHQoaGNyeXAsIFRpbWVvdXQpOwogICAgICAgIGJyZWFrOwogICAgICAgIAogICAgICBjYXNlIENSWVBfQUVTX0dDTToKCiAgICAgICAgLyogQUVTIEdDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2VzcyhoY3J5cCwgVGltZW91dCkgOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19DQ006CgogICAgICAgIC8qIEFFUyBDQ00gZGVjcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTQ0NNX1Byb2Nlc3MoaGNyeXAsIFRpbWVvdXQpOwogICAgICAgIGJyZWFrOwogICAgICAgIAogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICB9CgojZWxzZSAvKkFFUyovCgogICAgLyogU2V0IERlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfTU9ERSwgQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBBRVNfQ1JfQ0hNT0Q7CgogICAgc3dpdGNoIChhbGdvKQogICAgewoKICAgICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0JDOgogICAgICBjYXNlIENSWVBfQUVTX0NUUjoKCiAgICAgICAgLyogQUVTIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU19EZWNyeXB0KGhjcnlwLCBUaW1lb3V0KTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNX0dNQUM6CgogICAgICAgIC8qIEFFUyBHQ00gZGVjcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTR0NNX1Byb2Nlc3MoaGNyeXAsIFRpbWVvdXQpIDsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0NNOgoKICAgICAgICAvKiBBRVMgQ0NNIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0NDTV9Qcm9jZXNzKGhjcnlwLCBUaW1lb3V0KTsKICAgICAgICBicmVhazsKCiAgICAgIGRlZmF1bHQ6CiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9OT1RfU1VQUE9SVEVEOwogICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAqLwoKICAgIGlmIChzdGF0dXMgPT0gSEFMX09LKQogICAgewogICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgIH0KICB9CiAgZWxzZQogIHsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgRW5jcnlwdGlvbiBpbiBpbnRlcnJ1cHQgbW9kZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIElucHV0OiBQb2ludGVyIHRvIHRoZSBpbnB1dCBidWZmZXIgKHBsYWludGV4dCkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGluIHdvcmQKICAqIEBwYXJhbSAgT3V0cHV0OiBQb2ludGVyIHRvIHRoZSBvdXRwdXQgYnVmZmVyKGNpcGhlcnRleHQpCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9FbmNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSBzdGF0ZSBCdXN5ICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9CVVNZOwoKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgICAvKiAgUmVzZXQgQ3J5cEluQ291bnQsIENyeXBPdXRDb3VudCBhbmQgSW5pdGlhbGl6ZSBwQ3J5cEluQnVmZlB0ciBhbmQgcENyeXBPdXRCdWZmUHRyIHBhcmFtZXRlcnMqLwogICAgaGNyeXAtPkNyeXBJbkNvdW50ID0gMFU7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50ID0gMFU7CiAgICBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgPSBJbnB1dDsKICAgIGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgPSBPdXRwdXQ7CgogICAgLyogIENhbGN1bGF0ZSBTaXplIHBhcmFtZXRlciBpbiBCeXRlKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID09IENSWVBfREFUQVdJRFRIVU5JVF9XT1JEKQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemUgKiA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplOwogICAgfQoKI2lmIGRlZmluZWQgKENSWVApCgogICAgLyogU2V0IGVuY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSAoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT01PREUpOwoKICAgIHN3aXRjaCAoYWxnbykKICAgIHsKICAgICAgY2FzZSBDUllQX0RFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9ERVNfQ0JDOgogICAgICBjYXNlIENSWVBfVERFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0NCQzoKCiAgICAgICAgLypTZXQgS2V5ICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAxKTsKICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfRUNCKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMyk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDUpOwogICAgICAgIH0KICAgICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgICB9CgogICAgICAgIC8qIEZsdXNoIEZJRk8gKi8KICAgICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAgICAgLyogRW5hYmxlIGludGVycnVwdHMgKi8KICAgICAgICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkgfCBDUllQX0lUX09VVEkpOwoKICAgICAgICAvKiBFbmFibGUgQ1JZUCB0byBzdGFydCBERVMvVERFUyBwcm9jZXNzKi8KICAgICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICBzdGF0dXMgPSBDUllQX0FFU19FbmNyeXB0X0lUKGhjcnlwKTsKICAgICAgICBicmVhazsKICAgICAgICAKICAgICAgY2FzZSBDUllQX0FFU19HQ006CgogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTR0NNX1Byb2Nlc3NfSVQoaGNyeXApIDsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0NNOgoKICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0NDTV9Qcm9jZXNzX0lUKGhjcnlwKTsKICAgICAgICBicmVhazsKICAgICAgICAKICAgICAgZGVmYXVsdDoKICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQoKI2Vsc2UgLyogQUVTICovCgogICAgLyogU2V0IGVuY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfTU9ERSwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBBRVNfQ1JfQ0hNT0Q7CgogICAgc3dpdGNoIChhbGdvKQogICAgewogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICAvKiBBRVMgZW5jcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTX0VuY3J5cHRfSVQoaGNyeXApOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19HQ01fR01BQzoKCiAgICAgICAgLyogQUVTIEdDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2Vzc19JVChoY3J5cCkgOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19DQ006CgogICAgICAgIC8qIEFFUyBDQ00gZW5jcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTQ0NNX1Byb2Nlc3NfSVQoaGNyeXApOwogICAgICAgIGJyZWFrOwoKICAgICAgZGVmYXVsdDoKICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQojZW5kaWYgLyplbmQgQUVTIG9yIENSWVAqLwoKICB9CiAgZWxzZQogIHsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gc3RhdHVzOwp9CgovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBpdG50ZXJydXB0IG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChjaXBoZXJ0ZXh0ICkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGluIHdvcmQuCiAgKiBAcGFyYW0gIE91dHB1dDogUG9pbnRlciB0byB0aGUgb3V0cHV0IGJ1ZmZlcihwbGFpbnRleHQpCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9EZWNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSBzdGF0ZSBCdXN5ICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9CVVNZOwoKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgICAvKiAgUmVzZXQgQ3J5cEluQ291bnQsIENyeXBPdXRDb3VudCBhbmQgSW5pdGlhbGl6ZSBwQ3J5cEluQnVmZlB0ciBhbmQgcENyeXBPdXRCdWZmUHRyIHBhcmFtZXRlcnMqLwogICAgaGNyeXAtPkNyeXBJbkNvdW50ID0gMFU7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50ID0gMFU7CiAgICBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgPSBJbnB1dDsKICAgIGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgPSBPdXRwdXQ7CgogICAgLyogIENhbGN1bGF0ZSBTaXplIHBhcmFtZXRlciBpbiBCeXRlKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID09IENSWVBfREFUQVdJRFRIVU5JVF9XT1JEKQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemUgKiA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplOwogICAgfQoKI2lmIGRlZmluZWQgKENSWVApCgogICAgLyogU2V0IGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERTsKCiAgICBzd2l0Y2ggKGFsZ28pCiAgICB7CiAgICAgIGNhc2UgQ1JZUF9ERVNfRUNCOgogICAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgICAgY2FzZSBDUllQX1RERVNfRUNCOgogICAgICBjYXNlIENSWVBfVERFU19DQkM6CgogICAgICAgIC8qU2V0IEtleSAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgICB7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAyKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA1KTsKICAgICAgICB9CgogICAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICAgIH0KICAgICAgICAvKiBGbHVzaCBGSUZPICovCiAgICAgICAgSEFMX0NSWVBfRklGT19GTFVTSChoY3J5cCk7CgogICAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgICAgIC8qIEVuYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgICAgX19IQUxfQ1JZUF9FTkFCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JIHwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgICAgLyogRW5hYmxlIENSWVAgYW5kIHN0YXJ0IERFUy9UREVTIHByb2Nlc3MqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICAvKiBBRVMgZGVjcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTX0RlY3J5cHRfSVQoaGNyeXApOwogICAgICAgIGJyZWFrOwogICAgICAgIAogICAgICBjYXNlIENSWVBfQUVTX0dDTToKCiAgICAgICAgLyogQUVTIEdDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2Vzc19JVChoY3J5cCkgOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19DQ006CgogICAgICAgIC8qIEFFUyBDQ01kZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2Vzc19JVChoY3J5cCk7CiAgICAgICAgYnJlYWs7CiAgICAgICAgCiAgICAgIGRlZmF1bHQ6CiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9OT1RfU1VQUE9SVEVEOwogICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KCiNlbHNlIC8qQUVTKi8KCiAgICAvKiBTZXQgZGVjcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9NT0RFLCBDUllQX09QRVJBVElOR01PREVfREVDUllQVCk7CgogICAgLyogYWxnbyBnZXQgYWxnb3JpdGhtIHNlbGVjdGVkICovCiAgICBhbGdvID0gaGNyeXAtPkluc3RhbmNlLT5DUiAmIEFFU19DUl9DSE1PRDsKCiAgICBzd2l0Y2ggKGFsZ28pCiAgICB7CiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIC8qIEFFUyBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRGVjcnlwdF9JVChoY3J5cCk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0dDTV9HTUFDOgoKICAgICAgICAvKiBBRVMgR0NNIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzX0lUKGhjcnlwKSA7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKCiAgICAgICAgLyogQUVTIENDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2Vzc19JVChoY3J5cCk7CiAgICAgICAgYnJlYWs7CgogICAgICBkZWZhdWx0OgogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgICAgYnJlYWs7CiAgICB9CiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KCiAgfQogIGVsc2UKICB7CiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIHN0YXR1czsKfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRE1BIG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChwbGFpbnRleHQpCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBpbiB3b3JkLgogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIoY2lwaGVydGV4dCkKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0VuY3J5cHRfRE1BKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyIGFuZCBwQ3J5cE91dEJ1ZmZQdHIgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgICAvKiBTZXQgZW5jcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT0RJUiwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09NT0RFOwoKICAgIHN3aXRjaCAoYWxnbykKICAgIHsKICAgICAgY2FzZSBDUllQX0RFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9ERVNfQ0JDOgogICAgICBjYXNlIENSWVBfVERFU19FQ0I6CiAgICAgIGNhc2UgQ1JZUF9UREVTX0NCQzoKCiAgICAgICAgLypTZXQgS2V5ICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAxKTsKICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfRUNCKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMyk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDUpOwogICAgICAgIH0KCiAgICAgICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwogICAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfREVTX0NCQykgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgICB7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICAgICAgfQoKICAgICAgICAvKiBGbHVzaCBGSUZPICovCiAgICAgICAgSEFMX0NSWVBfRklGT19GTFVTSChoY3J5cCk7CgogICAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgICAgIC8qIFN0YXJ0IERNQSBwcm9jZXNzIHRyYW5zZmVyIGZvciBERVMvVERFUyAqLwogICAgICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksICgodWludDE2X3QpKGhjcnlwLT5TaXplKSAvIDRVKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogICAgICAgIHsKICAgICAgICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICAgICAgICB7CiAgICAgICAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgICAgICAgfQogICAgICAgICAgZWxzZQogICAgICAgICAgewogICAgICAgICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICAgICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICAgICAgICB9CiAgICAgICAgfQoKICAgICAgICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICAgICAgICB7CiAgICAgICAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgICAgICAgewogICAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxVSk7CiAgICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDJVKTsKICAgICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgM1UpOwogICAgICAgICAgfQogICAgICAgIH0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCgogICAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgICAgIC8qIFN0YXJ0IERNQSBwcm9jZXNzIHRyYW5zZmVyIGZvciBBRVMgKi8KICAgICAgICBDUllQX1NldERNQUNvbmZpZyhoY3J5cCwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCAoKHVpbnQxNl90KShoY3J5cC0+U2l6ZSkgLyA0VSksCiAgICAgICAgICAgICAgICAgICAgICAgICAgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgICAgICAgYnJlYWs7CiAgICAgICAgCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNOgogICAgICAgIC8qIEFFUyBHQ00gZW5jcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTR0NNX1Byb2Nlc3NfRE1BKGhjcnlwKSA7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0NDTToKICAgICAgICAvKiBBRVMgQ0NNIGVuY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0NDTV9Qcm9jZXNzX0RNQShoY3J5cCk7CiAgICAgICAgYnJlYWs7CiAgICAgICAgCiAgICAgIGRlZmF1bHQ6CiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9OT1RfU1VQUE9SVEVEOwogICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KCiNlbHNlIC8qQUVTKi8KICAgIC8qIFNldCBlbmNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQUVTX0NSX0NITU9EOwoKICAgIHN3aXRjaCAoYWxnbykKICAgIHsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogICAgICAgIHsKICAgICAgICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICAgICAgICB7CiAgICAgICAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgICAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgICAgICAgfQogICAgICAgICAgZWxzZQogICAgICAgICAgewogICAgICAgICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICAgICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICAgICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICAgICAgICB9CiAgICAgICAgfQoKICAgICAgICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICAgICAgICB7CiAgICAgICAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgICAgICAgewogICAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjMgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMSA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMCA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CiAgICAgICAgICB9CiAgICAgICAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAgICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgICAvKiBTdGFydCBETUEgcHJvY2VzcyB0cmFuc2ZlciBmb3IgQUVTICovCiAgICAgICAgQ1JZUF9TZXRETUFDb25maWcoaGNyeXAsICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwSW5CdWZmUHRyKSwgKGhjcnlwLT5TaXplIC8gNFUpLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfR0NNX0dNQUM6CiAgICAgICAgLyogQUVTIEdDTSBlbmNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2Vzc19ETUEoaGNyeXApIDsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0NNOgogICAgICAgIC8qIEFFUyBDQ00gZW5jcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTQ0NNX1Byb2Nlc3NfRE1BKGhjcnlwKTsKICAgICAgICBicmVhazsKCiAgICAgIGRlZmF1bHQ6CiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9OT1RfU1VQUE9SVEVEOwogICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgICBicmVhazsKICAgIH0KI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAqLwoKICB9CiAgZWxzZQogIHsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gc3RhdHVzOwp9CgovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBETUEgbW9kZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIElucHV0OiBQb2ludGVyIHRvIHRoZSBpbnB1dCBidWZmZXIgKGNpcGhlcnRleHQgKQogICogQHBhcmFtICBTaXplOiBMZW5ndGggb2YgdGhlIHBsYWludGV4dCBidWZmZXIgaW4gd29yZAogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIocGxhaW50ZXh0KQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgKklucHV0LCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCAqT3V0cHV0KQp7CiAgdWludDMyX3QgYWxnbzsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CgogIGlmIChoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewoKICAgIC8qIENoYW5nZSBzdGF0ZSBCdXN5ICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9CVVNZOwoKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9MT0NLKGhjcnlwKTsKCiAgICAvKiAgUmVzZXQgQ3J5cEluQ291bnQsIENyeXBPdXRDb3VudCBhbmQgSW5pdGlhbGl6ZSBwQ3J5cEluQnVmZlB0ciwgcENyeXBPdXRCdWZmUHRyIGFuZCBTaXplIHBhcmFtZXRlcnMqLwogICAgaGNyeXAtPkNyeXBJbkNvdW50ID0gMFU7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50ID0gMFU7CiAgICBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgPSBJbnB1dDsKICAgIGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgPSBPdXRwdXQ7CgogICAgLyogIENhbGN1bGF0ZSBTaXplIHBhcmFtZXRlciBpbiBCeXRlKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhV2lkdGhVbml0ID09IENSWVBfREFUQVdJRFRIVU5JVF9XT1JEKQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemUgKiA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplOwogICAgfQoKI2lmIGRlZmluZWQgKENSWVApCgogICAgLyogU2V0IGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERTsKCiAgICBzd2l0Y2ggKGFsZ28pCiAgICB7CiAgICAgIGNhc2UgQ1JZUF9ERVNfRUNCOgogICAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgICAgY2FzZSBDUllQX1RERVNfRUNCOgogICAgICBjYXNlIENSWVBfVERFU19DQkM6CgogICAgICAgIC8qU2V0IEtleSAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMVJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgICB7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAyKTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA1KTsKICAgICAgICB9CgogICAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICAgIH0KCiAgICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgICAvKiBTdGFydCBETUEgcHJvY2VzcyB0cmFuc2ZlciBmb3IgREVTL1RERVMgKi8KICAgICAgICBDUllQX1NldERNQUNvbmZpZyhoY3J5cCwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCAoKHVpbnQxNl90KShoY3J5cC0+U2l6ZSkgLyA0VSksCiAgICAgICAgICAgICAgICAgICAgICAgICAgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgICAgICAgYnJlYWs7CgogICAgICBjYXNlIENSWVBfQUVTX0VDQjoKICAgICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgICAvKiBBRVMgZGVjcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTX0RlY3J5cHRfRE1BKGhjcnlwKTsKICAgICAgICBicmVhazsKICAgICAgICAKICAgICAgY2FzZSBDUllQX0FFU19HQ006CiAgICAgICAgLyogQUVTIEdDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNHQ01fUHJvY2Vzc19ETUEoaGNyeXApIDsKICAgICAgICBicmVhazsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfQ0NNOgogICAgICAgIC8qIEFFUyBDQ00gZGVjcnlwdGlvbiAqLwogICAgICAgIHN0YXR1cyA9IENSWVBfQUVTQ0NNX1Byb2Nlc3NfRE1BKGhjcnlwKTsKICAgICAgICBicmVhazsKICAgICAgICAKICAgICAgZGVmYXVsdDoKICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQoKI2Vsc2UgLypBRVMqLwoKICAgIC8qIFNldCBkZWNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQUVTX0NSX0NITU9EOwoKICAgIHN3aXRjaCAoYWxnbykKICAgIHsKCiAgICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgICAgY2FzZSBDUllQX0FFU19DVFI6CgogICAgICAgIC8qIEFFUyBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRGVjcnlwdF9ETUEoaGNyeXApOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19HQ01fR01BQzoKICAgICAgICAvKiBBRVMgR0NNIGRlY3J5cHRpb24gKi8KICAgICAgICBzdGF0dXMgPSBDUllQX0FFU0dDTV9Qcm9jZXNzX0RNQShoY3J5cCkgOwogICAgICAgIGJyZWFrOwoKICAgICAgY2FzZSBDUllQX0FFU19DQ006CiAgICAgICAgLyogQUVTIENDTSBkZWNyeXB0aW9uICovCiAgICAgICAgc3RhdHVzID0gQ1JZUF9BRVNDQ01fUHJvY2Vzc19ETUEoaGNyeXApOwogICAgICAgIGJyZWFrOwoKICAgICAgZGVmYXVsdDoKICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogICAgICAgIGJyZWFrOwogICAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCiAgfQogIGVsc2UKICB7CiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIHN0YXR1czsKfQoKLyoqCiAgKiBAfQogICovCgovKiogQGRlZmdyb3VwIENSWVBfRXhwb3J0ZWRfRnVuY3Rpb25zX0dyb3VwMyBDUllQIElSUSBoYW5kbGVyIG1hbmFnZW1lbnQKICAqICBAYnJpZWYgICBDUllQIElSUSBoYW5kbGVyLgogICoKQHZlcmJhdGltCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICAgICAgICAgICAgICAjIyMjIyBDUllQIElSUSBoYW5kbGVyIG1hbmFnZW1lbnQgIyMjIyMKICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KWy4uXSAgVGhpcyBzZWN0aW9uIHByb3ZpZGVzIENSWVAgSVJRIGhhbmRsZXIgYW5kIGNhbGxiYWNrIGZ1bmN0aW9ucy4KICAgICAgKCspIEhBTF9DUllQX0lSUUhhbmRsZXIgQ1JZUCBpbnRlcnJ1cHQgcmVxdWVzdAogICAgICAoKykgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2sgaW5wdXQgZGF0YSB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjawogICAgICAoKykgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrIG91dHB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrCiAgICAgICgrKSBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrICBDUllQIGVycm9yIGNhbGxiYWNrCiAgICAgICgrKSBIQUxfQ1JZUF9HZXRTdGF0ZSByZXR1cm4gdGhlIENSWVAgc3RhdGUKICAgICAgKCspIEhBTF9DUllQX0dldEVycm9yIHJldHVybiB0aGUgQ1JZUCBlcnJvciBjb2RlCkBlbmR2ZXJiYXRpbQogICogQHsKICAqLwoKLyoqCiAgKiBAYnJpZWYgIFRoaXMgZnVuY3Rpb24gaGFuZGxlcyBjcnlwdG9ncmFwaGljIGludGVycnVwdCByZXF1ZXN0LgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgTm9uZQogICovCnZvaWQgSEFMX0NSWVBfSVJRSGFuZGxlcihDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CgojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgdWludDMyX3QgaXRzdGF0dXMgPSBoY3J5cC0+SW5zdGFuY2UtPk1JU1I7CgogIGlmICgoaXRzdGF0dXMgJiAoQ1JZUF9JVF9JTkkgfCBDUllQX0lUX09VVEkpKSAhPSAwVSkKICB7CiAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKQogICAgICAgIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgIHsKICAgICAgQ1JZUF9UREVTX0lUKGhjcnlwKTsgLyogREVTIG9yIFRERVMqLwogICAgfQogICAgZWxzZSBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfQ0JDKQogICAgICAgICAgICAgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19DVFIpKQogICAgewogICAgICBDUllQX0FFU19JVChoY3J5cCk7IC8qQUVTKi8KICAgIH0KICAgIAogICAgZWxzZSBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19HQ00pIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9DUl9BTEdPTU9ERV9BRVNfQ0NNKSkKICAgIHsKICAgICAgLyogaWYgaGVhZGVyIHBoYXNlICovCiAgICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfUEhBU0VfSEVBREVSKSA9PSBDUllQX1BIQVNFX0hFQURFUikKICAgICAgewogICAgICAgIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0lUKGhjcnlwKTsKICAgICAgfQogICAgICBlbHNlICAvKiBpZiBwYXlsb2FkIHBoYXNlICovCiAgICAgIHsKICAgICAgICBDUllQX0dDTUNDTV9TZXRQYXlsb2FkUGhhc2VfSVQoaGNyeXApOwogICAgICB9CiAgICB9CiAgICAKICAgIGVsc2UKICAgIHsKICAgICAgLyogTm90aGluZyB0byBkbyAqLwogICAgfQogIH0KCiNlbHNlIC8qQUVTKi8KICBpZiAoX19IQUxfQ1JZUF9HRVRfRkxBRyhoY3J5cCwgQ1JZUF9JVF9DQ0YpICE9IFJFU0VUKQogIHsKICAgIGlmIChfX0hBTF9DUllQX0dFVF9JVF9TT1VSQ0UoaGNyeXAsIENSWVBfSVRfQ0NGSUUpICE9IFJFU0VUKQogICAgewoKICAgICAgLyogQ2xlYXIgY29tcHV0YXRpb24gY29tcGxldGUgZmxhZyAqLwogICAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiAgICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfR0NNX0dNQUMpCiAgICAgIHsKCiAgICAgICAgLyogaWYgaGVhZGVyIHBoYXNlICovCiAgICAgICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9QSEFTRV9IRUFERVIpID09IENSWVBfUEhBU0VfSEVBREVSKQogICAgICAgIHsKICAgICAgICAgIENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0lUKGhjcnlwKTsKICAgICAgICB9CiAgICAgICAgZWxzZSAgLyogaWYgcGF5bG9hZCBwaGFzZSAqLwogICAgICAgIHsKICAgICAgICAgIENSWVBfR0NNQ0NNX1NldFBheWxvYWRQaGFzZV9JVChoY3J5cCk7CiAgICAgICAgfQogICAgICB9CiAgICAgIGVsc2UgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19DQ00pCiAgICAgIHsKICAgICAgICAvKiBpZiBoZWFkZXIgcGhhc2UgKi8KICAgICAgICBpZiAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSA+PSAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCkKICAgICAgICB7CiAgICAgICAgICBDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZV9JVChoY3J5cCk7CiAgICAgICAgfQogICAgICAgIGVsc2UgICAvKiBpZiBwYXlsb2FkIHBoYXNlICovCiAgICAgICAgewogICAgICAgICAgQ1JZUF9HQ01DQ01fU2V0UGF5bG9hZFBoYXNlX0lUKGhjcnlwKTsKICAgICAgICB9CiAgICAgIH0KICAgICAgZWxzZSAgLyogQUVTIEFsZ29yaXRobSBFQ0IsQ0JDIG9yIENUUiovCiAgICAgIHsKICAgICAgICBDUllQX0FFU19JVChoY3J5cCk7CiAgICAgIH0KICAgIH0KICB9CiAgLyogQ2hlY2sgaWYgZXJyb3Igb2NjdXJyZWQgKi8KICBpZiAoX19IQUxfQ1JZUF9HRVRfSVRfU09VUkNFKGhjcnlwLCBDUllQX0lUX0VSUklFKSAhPSBSRVNFVCkKICB7CiAgICAvKiBJZiB3cml0ZSBFcnJvciBvY2N1cnJlZCAqLwogICAgaWYgKF9fSEFMX0NSWVBfR0VUX0ZMQUcoaGNyeXAsIENSWVBfSVRfV1JFUlIpICE9IFJFU0VUKQogICAgewogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1dSSVRFOwogICAgfQogICAgLyogSWYgcmVhZCBFcnJvciBvY2N1cnJlZCAqLwogICAgaWYgKF9fSEFMX0NSWVBfR0VUX0ZMQUcoaGNyeXAsIENSWVBfSVRfUkRFUlIpICE9IFJFU0VUKQogICAgewogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1JFQUQ7CiAgICB9CiAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCn0KCi8qKgogICogQGJyaWVmICBSZXR1cm4gdGhlIENSWVAgZXJyb3IgY29kZS4KICAqIEBwYXJhbSAgaGNyeXAgOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIHRoZSAgQ1JZUCBJUAogICogQHJldHZhbCBDUllQIGVycm9yIGNvZGUKICAqLwp1aW50MzJfdCBIQUxfQ1JZUF9HZXRFcnJvcihDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgcmV0dXJuIGhjcnlwLT5FcnJvckNvZGU7Cn0KCi8qKgogICogQGJyaWVmICBSZXR1cm5zIHRoZSBDUllQIHN0YXRlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIEhBTCBzdGF0ZQogICovCkhBTF9DUllQX1NUQVRFVHlwZURlZiBIQUxfQ1JZUF9HZXRTdGF0ZShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgcmV0dXJuIGhjcnlwLT5TdGF0ZTsKfQoKLyoqCiAgKiBAYnJpZWYgIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpfX3dlYWsgdm9pZCBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgLyogUHJldmVudCB1bnVzZWQgYXJndW1lbnQocykgY29tcGlsYXRpb24gd2FybmluZyAqLwogIFVOVVNFRChoY3J5cCk7CgogIC8qIE5PVEUgOiBUaGlzIGZ1bmN0aW9uIFNob3VsZCBub3QgYmUgbW9kaWZpZWQsIHdoZW4gdGhlIGNhbGxiYWNrIGlzIG5lZWRlZCwKICAgICAgICAgICAgdGhlIEhBTF9DUllQX0luQ3BsdENhbGxiYWNrIGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHJldHZhbCBOb25lCiAgKi8KX193ZWFrIHZvaWQgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBQcmV2ZW50IHVudXNlZCBhcmd1bWVudChzKSBjb21waWxhdGlvbiB3YXJuaW5nICovCiAgVU5VU0VEKGhjcnlwKTsKCiAgLyogTk9URSA6IFRoaXMgZnVuY3Rpb24gU2hvdWxkIG5vdCBiZSBtb2RpZmllZCwgd2hlbiB0aGUgY2FsbGJhY2sgaXMgbmVlZGVkLAogICAgICAgICAgICB0aGUgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrIGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIENSWVAgZXJyb3IgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEByZXR2YWwgTm9uZQogICovCl9fd2VhayB2b2lkIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIFByZXZlbnQgdW51c2VkIGFyZ3VtZW50KHMpIGNvbXBpbGF0aW9uIHdhcm5pbmcgKi8KICBVTlVTRUQoaGNyeXApOwoKICAvKiBOT1RFIDogVGhpcyBmdW5jdGlvbiBTaG91bGQgbm90IGJlIG1vZGlmaWVkLCB3aGVuIHRoZSBjYWxsYmFjayBpcyBuZWVkZWQsCiAgICAgICAgICAgIHRoZSBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrIGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQovKioKICAqIEB9CiAgKi8KCi8qIFByaXZhdGUgZnVuY3Rpb25zIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi8qKiBAYWRkdG9ncm91cCBDUllQX1ByaXZhdGVfRnVuY3Rpb25zCiAgKiBAewogICovCgojaWYgZGVmaW5lZCAoQ1JZUCkKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIGluIEVDQi9DQkMgQWxnb3JpdGhtIHdpdGggREVTL1RERVMgc3RhbmRhcmQuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBUaW1lb3V0OiBzcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1RERVNfUHJvY2VzcyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgdGVtcFsyXTsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwogIHVpbnQxNl90IGluY291bnQ7IC8qIFRlbXBvcmFyeSBDcnlwSW5Db3VudCBWYWx1ZSAqLwogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwogIHVpbnQzMl90IGk7CgogIC8qIEVuYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAvKlN0YXJ0IHByb2Nlc3NpbmcqLwogIHdoaWxlICgoaGNyeXAtPkNyeXBJbkNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpKQogIHsKICAgIC8qIFRlbXBvcmFyeSBDcnlwSW5Db3VudCBWYWx1ZSAqLwogICAgaW5jb3VudCA9IGhjcnlwLT5DcnlwSW5Db3VudDsKICAgIC8qIFdyaXRlIHBsYWluIGRhdGEgYW5kIGdldCBjaXBoZXIgZGF0YSAqLwogICAgaWYgKCgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19JRk5GKSAhPSAweDBVKSAmJiAoaW5jb3VudCA8IChoY3J5cC0+U2l6ZSAvIDRVKSkpCiAgICB7CiAgICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgfQoKICAgIC8qIFdhaXQgZm9yIE9GTkUgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgIGlmIChDUllQX1dhaXRPbk9GTkVGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICB7CiAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgLyogQ2hhbmdlIHN0YXRlICYgZXJyb3JDb2RlKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgIH0KCiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAgIGlmICgoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSkgIT0gMHgwVSkgJiYgKG91dGNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSkKICAgIHsKICAgICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIE91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgQnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAgKi8KICAgICAgZm9yIChpID0gMFU7IGkgPCAyVTsgaSsrKQogICAgICB7CiAgICAgICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICAgfQogICAgICBpID0gMFU7CiAgICAgIHdoaWxlICgoKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoKGhjcnlwLT5TaXplKSAvIDRVKSkpICYmIChpIDwgMlUpKQogICAgICB7CiAgICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2ldOwogICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgICBpKys7CiAgICAgIH0KICAgIH0KICAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CiAgfQogIC8qIERpc2FibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CiAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgQ1JZUCBibG9jayBpbnB1dC9vdXRwdXQgZGF0YSBoYW5kbGluZyB1bmRlciBpbnRlcnJ1cHRpb24gd2l0aCBERVMvVERFUyBzdGFuZGFyZC4KICAqIEBub3RlICAgVGhlIGZ1bmN0aW9uIGlzIGNhbGxlZCB1bmRlciBpbnRlcnJ1cHRpb24gb25seSwgb25jZQogICogICAgICAgICBpbnRlcnJ1cHRpb25zIGhhdmUgYmVlbiBlbmFibGVkIGJ5IENSWVBfRGVjcnlwdF9JVCgpIGFuZCBDUllQX0VuY3J5cHRfSVQoKS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHJldHZhbCBub25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9UREVTX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICB1aW50MzJfdCB0ZW1wWzJdOyAgLyogVGVtcG9yYXJ5IENyeXBPdXRCdWZmICovCiAgdWludDMyX3QgaTsKCiAgaWYgKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9CVVNZKQogIHsKICAgIGlmIChfX0hBTF9DUllQX0dFVF9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpICE9IDB4MFUpCiAgICB7CiAgICAgIGlmIChfX0hBTF9DUllQX0dFVF9GTEFHKGhjcnlwLCBDUllQX0ZMQUdfSU5SSVMpICE9IDB4MFUpCiAgICAgIHsKICAgICAgICAvKiBXcml0ZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwoKICAgICAgICBpZiAoaGNyeXAtPkNyeXBJbkNvdW50ID09ICgodWludDE2X3QpKGhjcnlwLT5TaXplKSAvIDRVKSkKICAgICAgICB7CiAgICAgICAgICAvKiBEaXNhYmxlIGludGVycnVwdGlvbiAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSk7CiAgICAgICAgICAvKiBDYWxsIHRoZSBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxVSkKICAgICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICAgIH0KICAgICAgfQogICAgfQogICAgaWYgKF9fSEFMX0NSWVBfR0VUX0lUKGhjcnlwLCBDUllQX0lUX09VVEkpICE9IDB4MFUpCiAgICB7CiAgICAgIGlmIChfX0hBTF9DUllQX0dFVF9GTEFHKGhjcnlwLCBDUllQX0ZMQUdfT1VUUklTKSAhPSAweDBVKQogICAgICB7CiAgICAgICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIE91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgQnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAgKi8KICAgICAgICBmb3IgKGkgPSAwVTsgaSA8IDJVOyBpKyspCiAgICAgICAgewogICAgICAgICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICAgICB9CiAgICAgICAgaSA9IDBVOwogICAgICAgIHdoaWxlICgoKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoKGhjcnlwLT5TaXplKSAvIDRVKSkpICYmIChpIDwgMlUpKQogICAgICAgIHsKICAgICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgICAgIGkrKzsKICAgICAgICB9CiAgICAgICAgaWYgKGhjcnlwLT5DcnlwT3V0Q291bnQgPT0gKCh1aW50MTZfdCkoaGNyeXAtPlNpemUpIC8gNFUpKQogICAgICAgIHsKICAgICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0aW9uICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfT1VUSSk7CgogICAgICAgICAgLyogRGlzYWJsZSBDUllQICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICAgICAgICAgIC8qIENhbGwgb3V0cHV0IHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgT3V0cHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgICBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICAgIH0KICAgICAgfQogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICB9Cn0KCiNlbmRpZiAvKiBDUllQICovCgovKioKICAqIEBicmllZiAgRW5jcnlwdGlvbiBpbiBFQ0IvQ0JDICYgQ1RSIEFsZ29yaXRobSB3aXRoIEFFUyBTdGFuZGFyZAogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUKICAqIEBwYXJhbSAgVGltZW91dDogc3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRW5jcnlwdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDE2X3Qgb3V0Y291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlICovCiAgdWludDMyX3QgRG9LZXlJVkNvbmZpZyA9IDFVOyAvKiBCeSBkZWZhdWx0LCBjYXJyeSBvdXQgcGVyaXBoZXJhbCBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gKi8KCiAgaWYgKGhjcnlwLT5Jbml0LktleUlWQ29uZmlnU2tpcCA9PSBDUllQX0tFWUlWQ09ORklHX09OQ0UpCiAgewogICAgaWYgKGhjcnlwLT5LZXlJVkNvbmZpZyA9PSAxVSkKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBhbHJlYWR5IGJlZW4gZG9uZSwgc2tpcCBpdCAqLwogICAgICBEb0tleUlWQ29uZmlnID0gMFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgIGFuZCBpZiBpdCBoYXMgbm90IGJlZW4gZG9uZSBhbHJlYWR5LCBkbyBpdCBhbmQgc2V0IEtleUlWQ29uZmlnCiAgICAgIHRvIGtlZXAgdHJhY2sgaXQgd29uJ3QgaGF2ZSB0byBiZSBkb25lIGFnYWluIG5leHQgdGltZSAqLwogICAgICBoY3J5cC0+S2V5SVZDb25maWcgPSAxVTsKICAgIH0KICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKCiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgewogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiNpZiBkZWZpbmVkIChBRVMpCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMyA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIyID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIxID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIwID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKI2Vsc2UgLyogQ1JZUCAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCiAgICB9CiAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KCiAgLyogU2V0IHRoZSBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgLyogRW5hYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICB3aGlsZSAoKGhjcnlwLT5DcnlwSW5Db3VudCA8IChoY3J5cC0+U2l6ZSAvIDRVKSkgJiYgKG91dGNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSkKICB7CiAgICAvKiBXcml0ZSBwbGFpbiBEZHRhIGFuZCBnZXQgY2lwaGVyIGRhdGEgKi8KICAgIENSWVBfQUVTX1Byb2Nlc3NEYXRhKGhjcnlwLCBUaW1lb3V0KTsKICAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CiAgfQoKICAvKiBEaXNhYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIGluIEVDQi9DQkMgJiBDVFIgbW9kZSB3aXRoIEFFUyBTdGFuZGFyZCB1c2luZyBpbnRlcnJ1cHQgbW9kZQogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19FbmNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgfQogIH0KCiAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgewogICAgLyogIFNldCB0aGUgS2V5Ki8KICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0VDQikKICAgIHsKICAgICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwojaWYgZGVmaW5lZCAoQUVTKQogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjMgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMSA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMCA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CgojZWxzZSAvKiBDUllQICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KICAgIH0KICB9IC8qIGlmIChEb0tleUlWQ29uZmlnID09IDFVKSAqLwoKICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICBpZiAoaGNyeXAtPlNpemUgIT0gMFUpCiAgewojaWYgZGVmaW5lZCAoQUVTKQoKICAgIC8qIEVuYWJsZSBjb21wdXRhdGlvbiBjb21wbGV0ZSBmbGFnIGFuZCBlcnJvciBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9DQ0ZJRSB8IENSWVBfSVRfRVJSSUUpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogV3JpdGUgdGhlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CgojZWxzZSAvKiBDUllQICovCgogICAgLyogRW5hYmxlIGludGVycnVwdHMgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgLyogRW5hYmxlIENSWVAgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiNlbmRpZiAvKiAgRW5kIEFFUyBvciBDUllQICAqLwogIH0KICBlbHNlCiAgewogICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBEZWNyeXB0aW9uIGluIEVDQi9DQkMgJiBDVFIgbW9kZSB3aXRoIEFFUyBTdGFuZGFyZAogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUKICAqIEBwYXJhbSAgVGltZW91dDogU3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDE2X3Qgb3V0Y291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlICovCiAgdWludDMyX3QgRG9LZXlJVkNvbmZpZyA9IDFVOyAvKiBCeSBkZWZhdWx0LCBjYXJyeSBvdXQgcGVyaXBoZXJhbCBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gKi8KCiAgaWYgKGhjcnlwLT5Jbml0LktleUlWQ29uZmlnU2tpcCA9PSBDUllQX0tFWUlWQ09ORklHX09OQ0UpCiAgewogICAgaWYgKGhjcnlwLT5LZXlJVkNvbmZpZyA9PSAxVSkKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBhbHJlYWR5IGJlZW4gZG9uZSwgc2tpcCBpdCAqLwogICAgICBEb0tleUlWQ29uZmlnID0gMFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgIGFuZCBpZiBpdCBoYXMgbm90IGJlZW4gZG9uZSBhbHJlYWR5LCBkbyBpdCBhbmQgc2V0IEtleUlWQ29uZmlnCiAgICAgIHRvIGtlZXAgdHJhY2sgaXQgd29uJ3QgaGF2ZSB0byBiZSBkb25lIGFnYWluIG5leHQgdGltZSAqLwogICAgICBoY3J5cC0+S2V5SVZDb25maWcgPSAxVTsKICAgIH0KICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKICAgIC8qICBLZXkgcHJlcGFyYXRpb24gZm9yIEVDQi9DQkMgKi8KICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfQ1RSKQogICAgewojaWYgZGVmaW5lZCAoQUVTKQogICAgICBpZiAoaGNyeXAtPkF1dG9LZXlEZXJpdmF0aW9uID09IERJU0FCTEUpLypNb2RlIDIgS2V5IHByZXBhcmF0aW9uKi8KICAgICAgewogICAgICAgIC8qIFNldCBrZXkgcHJlcGFyYXRpb24gZm9yIGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9LRVlERVJJVkFUSU9OKTsKCiAgICAgICAgLyogIFNldCB0aGUgS2V5Ki8KICAgICAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgICAgIC8qIEVuYWJsZSBDUllQICovCiAgICAgICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgICAgICAvKiBXYWl0IGZvciBDQ0YgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgICAgICBpZiAoQ1JZUF9XYWl0T25DQ0ZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgICB7CiAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICYgZXJyb3IgY29kZSovCiAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICB9CiAgICAgICAgLyogQ2xlYXIgQ0NGIEZsYWcgKi8KICAgICAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiAgICAgICAgLyogUmV0dXJuIHRvIGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUoTW9kZSAzKSovCiAgICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfTU9ERSwgQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpOwogICAgICB9CiAgICAgIGVsc2UgLypNb2RlIDQgOiBkZWNyeXB0aW9uICYgS2V5IHByZXBhcmF0aW9uKi8KICAgICAgewogICAgICAgIC8qICBTZXQgdGhlIEtleSovCiAgICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgICAvKiBTZXQgZGVjcnlwdGlvbiAmIEtleSBwcmVwYXJhdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfTU9ERSwgQ1JZUF9PUEVSQVRJTkdNT0RFX0tFWURFUklWQVRJT05fREVDUllQVCk7CiAgICAgIH0KI2Vsc2UgLyogQ1JZUCAqLwogICAgICAvKiBjaGFuZ2UgQUxHT01PREUgdG8ga2V5IHByZXBhcmF0aW9uIGZvciBkZWNyeXB0aW9uKi8KICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0NSX0FMR09NT0RFX0FFU19LRVkpOwoKICAgICAgLyogIFNldCB0aGUgS2V5Ki8KICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgLyogRW5hYmxlIENSWVAgKi8KICAgICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgICAgLyogV2FpdCBmb3IgQlVTWSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgICBpZiAoQ1JZUF9XYWl0T25CVVNZRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICAgIC8qIFR1cm4gYmFjayB0byBBTEdPTU9ERSBvZiB0aGUgY29uZmlndXJhdGlvbiAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIGhjcnlwLT5Jbml0LkFsZ29yaXRobSk7CgojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICAqLwogICAgfQogICAgZWxzZSAgLypBbGdvcml0aG0gQ1RSICovCiAgICB7CiAgICAgIC8qICBTZXQgdGhlIEtleSovCiAgICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKICAgIH0KCiAgICAvKiBTZXQgSVYgKi8KICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgewogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiNpZiBkZWZpbmVkIChBRVMpCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMyA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIyID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIxID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIwID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKI2Vsc2UgLyogQ1JZUCAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCiAgICB9CiAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAvKiBFbmFibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogIHdoaWxlICgoaGNyeXAtPkNyeXBJbkNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpKQogIHsKICAgIC8qIFdyaXRlIHBsYWluIGRhdGEgYW5kIGdldCBjaXBoZXIgZGF0YSAqLwogICAgQ1JZUF9BRVNfUHJvY2Vzc0RhdGEoaGNyeXAsIFRpbWVvdXQpOwogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKICB9CgogIC8qIERpc2FibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBFQ0IvQ0JDICYgQ1RSIG1vZGUgd2l0aCBBRVMgU3RhbmRhcmQgdXNpbmcgaW50ZXJydXB0IG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgX19JTyB1aW50MzJfdCBjb3VudCA9IDBVOwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogIHsKICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICB9CiAgfQoKICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgICAvKiAgS2V5IHByZXBhcmF0aW9uIGZvciBFQ0IvQ0JDICovCiAgICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0NUUikKICAgIHsKI2lmIGRlZmluZWQgKEFFUykKICAgICAgaWYgKGhjcnlwLT5BdXRvS2V5RGVyaXZhdGlvbiA9PSBESVNBQkxFKS8qTW9kZSAyIEtleSBwcmVwYXJhdGlvbiovCiAgICAgIHsKICAgICAgICAvKiBTZXQga2V5IHByZXBhcmF0aW9uIGZvciBkZWNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9NT0RFLCBDUllQX09QRVJBVElOR01PREVfS0VZREVSSVZBVElPTik7CgogICAgICAgIC8qICBTZXQgdGhlIEtleSovCiAgICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgICAvKiBFbmFibGUgQ1JZUCAqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAgICAgLyogV2FpdCBmb3IgQ0NGIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT047CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICBjb3VudC0tIDsKICAgICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgIH0KICAgICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBBRVNfU1JfQ0NGKSk7CgogICAgICAgIC8qIENsZWFyIENDRiBGbGFnICovCiAgICAgICAgX19IQUxfQ1JZUF9DTEVBUl9GTEFHKGhjcnlwLCBDUllQX0NDRl9DTEVBUik7CgogICAgICAgIC8qIFJldHVybiB0byBkZWNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKE1vZGUgMykqLwogICAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKICAgICAgfQogICAgICBlbHNlIC8qTW9kZSA0IDogZGVjcnlwdGlvbiAmIGtleSBwcmVwYXJhdGlvbiovCiAgICAgIHsKICAgICAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAgICAgLyogU2V0IGRlY3J5cHRpb24gJiBrZXkgcHJlcGFyYXRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9LRVlERVJJVkFUSU9OX0RFQ1JZUFQpOwogICAgICB9CiNlbHNlIC8qIENSWVAgKi8KCiAgICAgIC8qIGNoYW5nZSBBTEdPTU9ERSB0byBrZXkgcHJlcGFyYXRpb24gZm9yIGRlY3J5cHRpb24qLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIENSWVBfQ1JfQUxHT01PREVfQUVTX0tFWSk7CgogICAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgICAvKiBFbmFibGUgQ1JZUCAqLwogICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgICAvKiBXYWl0IGZvciBCVVNZIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICAgIGNvdW50ID0gQ1JZUF9USU1FT1VUX0tFWVBSRVBBUkFUSU9OOwogICAgICBkbwogICAgICB7CiAgICAgICAgY291bnQtLSA7CiAgICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICAgIHsKICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgfQogICAgICB9IHdoaWxlIChIQUxfSVNfQklUX1NFVChoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfQlVTWSkpOwoKICAgICAgLyogVHVybiBiYWNrIHRvIEFMR09NT0RFIG9mIHRoZSBjb25maWd1cmF0aW9uICovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPTU9ERSwgaGNyeXAtPkluaXQuQWxnb3JpdGhtKTsKCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KICAgIH0KCiAgICBlbHNlICAvKkFsZ29yaXRobSBDVFIgKi8KICAgIHsKICAgICAgLyogIFNldCB0aGUgS2V5Ki8KICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwogICAgfQoKICAgIC8qIFNldCBJViAqLwogICAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgICB7CiAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KI2lmIGRlZmluZWQgKEFFUykKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIzID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjEgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDIpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwojZWxzZSAvKiBDUllQICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KICAgIH0KICB9IC8qIGlmIChEb0tleUlWQ29uZmlnID09IDFVKSAqLwoKICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwogIGlmIChoY3J5cC0+U2l6ZSAhPSAwVSkKICB7CgojaWYgZGVmaW5lZCAoQUVTKQoKICAgIC8qIEVuYWJsZSBjb21wdXRhdGlvbiBjb21wbGV0ZSBmbGFnIGFuZCBlcnJvciBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9DQ0ZJRSB8IENSWVBfSVRfRVJSSUUpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogV3JpdGUgdGhlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CgojZWxzZSAvKiBDUllQICovCgogICAgLyogRW5hYmxlIGludGVycnVwdHMgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgLyogRW5hYmxlIENSWVAgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBFQ0IvQ0JDICYgQ1RSIG1vZGUgd2l0aCBBRVMgU3RhbmRhcmQgdXNpbmcgRE1BIG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgfQogIH0KICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgICAvKiAgS2V5IHByZXBhcmF0aW9uIGZvciBFQ0IvQ0JDICovCiAgICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtICE9IENSWVBfQUVTX0NUUikKICAgIHsKI2lmIGRlZmluZWQgKEFFUykKICAgICAgaWYgKGhjcnlwLT5BdXRvS2V5RGVyaXZhdGlvbiA9PSBESVNBQkxFKS8qTW9kZSAyIGtleSBwcmVwYXJhdGlvbiovCiAgICAgIHsKICAgICAgICAvKiBTZXQga2V5IHByZXBhcmF0aW9uIGZvciBkZWNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKi8KICAgICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9NT0RFLCBDUllQX09QRVJBVElOR01PREVfS0VZREVSSVZBVElPTik7CgogICAgICAgIC8qICBTZXQgdGhlIEtleSovCiAgICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgICAvKiBFbmFibGUgQ1JZUCAqLwogICAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAgICAgLyogV2FpdCBmb3IgQ0NGIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT047CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICBjb3VudC0tIDsKICAgICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgIH0KICAgICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBBRVNfU1JfQ0NGKSk7CgogICAgICAgIC8qIENsZWFyIENDRiBGbGFnICovCiAgICAgICAgX19IQUxfQ1JZUF9DTEVBUl9GTEFHKGhjcnlwLCBDUllQX0NDRl9DTEVBUik7CgogICAgICAgIC8qIFJldHVybiB0byBkZWNyeXB0aW9uIG9wZXJhdGluZyBtb2RlKE1vZGUgMykqLwogICAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKICAgICAgfQogICAgICBlbHNlIC8qTW9kZSA0IDogZGVjcnlwdGlvbiAmIGtleSBwcmVwYXJhdGlvbiovCiAgICAgIHsKICAgICAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAgICAgLyogU2V0IGRlY3J5cHRpb24gJiBLZXkgcHJlcGFyYXRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX01PREUsIENSWVBfT1BFUkFUSU5HTU9ERV9LRVlERVJJVkFUSU9OX0RFQ1JZUFQpOwogICAgICB9CiNlbHNlIC8qIENSWVAgKi8KICAgICAgLyogY2hhbmdlIEFMR09NT0RFIHRvIGtleSBwcmVwYXJhdGlvbiBmb3IgZGVjcnlwdGlvbiovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPTU9ERSwgQ1JZUF9DUl9BTEdPTU9ERV9BRVNfS0VZKTsKCiAgICAgIC8qICBTZXQgdGhlIEtleSovCiAgICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAgIC8qIEVuYWJsZSBDUllQICovCiAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAgIC8qIFdhaXQgZm9yIEJVU1kgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT047CiAgICAgIGRvCiAgICAgIHsKICAgICAgICBjb3VudC0tIDsKICAgICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgICAgewogICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgfQogICAgICB9IHdoaWxlIChIQUxfSVNfQklUX1NFVChoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfQlVTWSkpOwoKICAgICAgLyogVHVybiBiYWNrIHRvIEFMR09NT0RFIG9mIHRoZSBjb25maWd1cmF0aW9uICovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPTU9ERSwgaGNyeXAtPkluaXQuQWxnb3JpdGhtKTsKCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgICovCiAgICB9CiAgICBlbHNlICAvKkFsZ29yaXRobSBDVFIgKi8KICAgIHsKICAgICAgLyogIFNldCB0aGUgS2V5Ki8KICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwogICAgfQoKICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogICAgewogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiNpZiBkZWZpbmVkIChBRVMpCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMyA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIyID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIxID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVlIwID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKI2Vsc2UgLyogQ1JZUCAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICAqLwogICAgfQogIH0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCgogIC8qIFNldCB0aGUgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogIGlmIChoY3J5cC0+U2l6ZSAhPSAwVSkKICB7CiAgICAvKiBTZXQgdGhlIGlucHV0IGFuZCBvdXRwdXQgYWRkcmVzc2VzIGFuZCBzdGFydCBETUEgdHJhbnNmZXIgKi8KICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksIChoY3J5cC0+U2l6ZSAvIDRVKSwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKCi8qKgogICogQGJyaWVmICBETUEgQ1JZUCBpbnB1dCBkYXRhIHByb2Nlc3MgY29tcGxldGUgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhkbWE6IERNQSBoYW5kbGUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfRE1BSW5DcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKQp7CiAgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCA9IChDUllQX0hhbmRsZVR5cGVEZWYgKikoKERNQV9IYW5kbGVUeXBlRGVmICopaGRtYSktPlBhcmVudDsKCiAgLyogRGlzYWJsZSB0aGUgRE1BIHRyYW5zZmVyIGZvciBpbnB1dCBGSUZPIHJlcXVlc3QgYnkgcmVzZXR0aW5nIHRoZSBESUVOIGJpdAogIGluIHRoZSBETUFDUiByZWdpc3RlciAqLwojaWYgZGVmaW5lZCAoQ1JZUCkKICBoY3J5cC0+SW5zdGFuY2UtPkRNQUNSICY9ICh1aW50MzJfdCkofkNSWVBfRE1BQ1JfRElFTik7CgojZWxzZSAvKiBBRVMgKi8KICBDTEVBUl9CSVQoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RNQUlORU4pOwoKICAvKiBUaW55QUVTMiwgTm8gb3V0cHV0IG9uIENDTSBBRVMsIHVubG9jayBzaG91bGQgYmUgZG9uZSB3aGVuIGlucHV0IGRhdGEgcHJvY2VzcyBjb21wbGV0ZSAqLwogIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtICYgQ1JZUF9BRVNfQ0NNKSA9PSBDUllQX0FFU19DQ00pCiAgewogICAgLyogQ2xlYXIgQ0NGIGZsYWcgKi8KICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSB0byByZWFkeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICB9CiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KCiAgLyogQ2FsbCBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwp9CgovKioKICAqIEBicmllZiAgRE1BIENSWVAgb3V0cHV0IGRhdGEgcHJvY2VzcyBjb21wbGV0ZSBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGRtYTogRE1BIGhhbmRsZQogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9ETUFPdXRDcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKQp7CiAgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCA9IChDUllQX0hhbmRsZVR5cGVEZWYgKikoKERNQV9IYW5kbGVUeXBlRGVmICopaGRtYSktPlBhcmVudDsKCiAgLyogRGlzYWJsZSB0aGUgRE1BIHRyYW5zZmVyIGZvciBvdXRwdXQgRklGTyByZXF1ZXN0IGJ5IHJlc2V0dGluZwogIHRoZSBET0VOIGJpdCBpbiB0aGUgRE1BQ1IgcmVnaXN0ZXIgKi8KCiNpZiBkZWZpbmVkIChDUllQKQoKICBoY3J5cC0+SW5zdGFuY2UtPkRNQUNSICY9ICh1aW50MzJfdCkofkNSWVBfRE1BQ1JfRE9FTik7CiAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gJiBDUllQX0FFU19HQ00pICE9IENSWVBfQUVTX0dDTSkKICB7CiAgICAvKiBEaXNhYmxlIENSWVAgIChub3QgYWxsb3dlZCBpbiAgR0NNKSovCiAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwogIH0KI2Vsc2UgLyogQUVTICovCgogIENMRUFSX0JJVChoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfRE1BT1VURU4pOwoKICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAmIENSWVBfQUVTX0dDTV9HTUFDKSAhPSBDUllQX0FFU19HQ01fR01BQykKICB7CiAgICAvKiBEaXNhYmxlIENSWVAgKG5vdCBhbGxvd2VkIGluICBHQ00pKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CiAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCgogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSB0byByZWFkeSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAvKiBDYWxsIG91dHB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIERNQSBDUllQIGNvbW11bmljYXRpb24gZXJyb3IgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhkbWE6IERNQSBoYW5kbGUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfRE1BRXJyb3IoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpCnsKICBDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwID0gKENSWVBfSGFuZGxlVHlwZURlZiAqKSgoRE1BX0hhbmRsZVR5cGVEZWYgKiloZG1hKS0+UGFyZW50OwoKICAvKiBDaGFuZ2UgdGhlIENSWVAgcGVyaXBoZXJhbCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBETUEgZXJyb3IgY29kZSBmaWVsZCAqLwogIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfRE1BOwoKI2lmIGRlZmluZWQgKEFFUykKCiAgLyogQ2xlYXIgQ0NGIGZsYWcgKi8KICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiNlbmRpZiAvKiBBRVMgKi8KCiAgLyogQ2FsbCBlcnJvciBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCn0KCi8qKgogICogQGJyaWVmICBTZXQgdGhlIERNQSBjb25maWd1cmF0aW9uIGFuZCBzdGFydCB0aGUgRE1BIHRyYW5zZmVyCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBpbnB1dGFkZHI6IGFkZHJlc3Mgb2YgdGhlIGlucHV0IGJ1ZmZlcgogICogQHBhcmFtICBTaXplOiBzaXplIG9mIHRoZSBpbnB1dCBidWZmZXIsIG11c3QgYmUgYSBtdWx0aXBsZSBvZiAxNi4KICAqIEBwYXJhbSAgb3V0cHV0YWRkcjogYWRkcmVzcyBvZiB0aGUgb3V0cHV0IGJ1ZmZlcgogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9TZXRETUFDb25maWcoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgaW5wdXRhZGRyLCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCBvdXRwdXRhZGRyKQp7CiAgLyogU2V0IHRoZSBDUllQIERNQSB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwogIGhjcnlwLT5oZG1haW4tPlhmZXJDcGx0Q2FsbGJhY2sgPSBDUllQX0RNQUluQ3BsdDsKCiAgLyogU2V0IHRoZSBETUEgaW5wdXQgZXJyb3IgY2FsbGJhY2sgKi8KICBoY3J5cC0+aGRtYWluLT5YZmVyRXJyb3JDYWxsYmFjayA9IENSWVBfRE1BRXJyb3I7CgogIC8qIFNldCB0aGUgQ1JZUCBETUEgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KICBoY3J5cC0+aGRtYW91dC0+WGZlckNwbHRDYWxsYmFjayA9IENSWVBfRE1BT3V0Q3BsdDsKCiAgLyogU2V0IHRoZSBETUEgb3V0cHV0IGVycm9yIGNhbGxiYWNrICovCiAgaGNyeXAtPmhkbWFvdXQtPlhmZXJFcnJvckNhbGxiYWNrID0gQ1JZUF9ETUFFcnJvcjsKCiNpZiBkZWZpbmVkIChDUllQKQoKICAvKiBFbmFibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLyogRW5hYmxlIHRoZSBpbnB1dCBETUEgU3RyZWFtICovCiAgaWYgKEhBTF9ETUFfU3RhcnRfSVQoaGNyeXAtPmhkbWFpbiwgaW5wdXRhZGRyLCAodWludDMyX3QpJmhjcnlwLT5JbnN0YW5jZS0+RElOLCBTaXplKSAhPSBIQUxfT0spCiAgewogICAgLyogRE1BIGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfRE1BOwoKICAgIC8qIENhbGwgZXJyb3IgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQogIC8qIEVuYWJsZSB0aGUgb3V0cHV0IERNQSBTdHJlYW0gKi8KICBpZiAoSEFMX0RNQV9TdGFydF9JVChoY3J5cC0+aGRtYW91dCwgKHVpbnQzMl90KSZoY3J5cC0+SW5zdGFuY2UtPkRPVVQsIG91dHB1dGFkZHIsIFNpemUpICE9IEhBTF9PSykKICB7CiAgICAvKiBETUEgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9ETUE7CgogICAgLyogQ2FsbCBlcnJvciBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICB9CiAgLyogRW5hYmxlIEluL091dCBETUEgcmVxdWVzdCAqLwogIGhjcnlwLT5JbnN0YW5jZS0+RE1BQ1IgPSBDUllQX0RNQUNSX0RPRU4gfCBDUllQX0RNQUNSX0RJRU47CgojZWxzZSAvKiBBRVMgKi8KCiAgaWYgKCgoaGNyeXAtPkluaXQuQWxnb3JpdGhtICYgQ1JZUF9BRVNfR0NNX0dNQUMpICE9IENSWVBfQUVTX0dDTV9HTUFDKQogICAgICAmJiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAmIENSWVBfQUVTX0NDTSkgIT0gQ1JZUF9BRVNfQ0NNKSkKICB7CiAgICAvKiBFbmFibGUgQ1JZUCAobm90IGFsbG93ZWQgaW4gIEdDTSAmIENDTSkqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogIH0KCiAgLyogRW5hYmxlIHRoZSBETUEgaW5wdXQgc3RyZWFtICovCiAgaWYgKEhBTF9ETUFfU3RhcnRfSVQoaGNyeXAtPmhkbWFpbiwgaW5wdXRhZGRyLCAodWludDMyX3QpJmhjcnlwLT5JbnN0YW5jZS0+RElOUiwgU2l6ZSkgIT0gSEFMX09LKQogIHsKICAgIC8qIERNQSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0RNQTsKCiAgICAvKiBDYWxsIGVycm9yIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogIH0KICAvKiBFbmFibGUgdGhlIERNQSBvdXRwdXQgc3RyZWFtICovCiAgaWYgKEhBTF9ETUFfU3RhcnRfSVQoaGNyeXAtPmhkbWFvdXQsICh1aW50MzJfdCkmaGNyeXAtPkluc3RhbmNlLT5ET1VUUiwgb3V0cHV0YWRkciwgU2l6ZSkgIT0gSEFMX09LKQogIHsKICAgIC8qIERNQSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0RNQTsKCiAgICAvKiBDYWxsIGVycm9yIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogIH0KICAvKkFFUzJ2MS4xLjEgOiBDQ00gYXV0aGVudGljYXRpb24gOiBubyBpbml0IHBoYXNlLCBvbmx5IGhlYWRlciBhbmQgZmluYWwgcGhhc2UgKi8KICAvKiBFbmFibGUgSW4gYW5kIE91dCBETUEgcmVxdWVzdHMgKi8KICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAmIENSWVBfQUVTX0NDTSkgPT0gQ1JZUF9BRVNfQ0NNKQogIHsKICAgIC8qIEVuYWJsZSBvbmx5IEluIERNQSByZXF1ZXN0cyBmb3IgQ0NNKi8KICAgIFNFVF9CSVQoaGNyeXAtPkluc3RhbmNlLT5DUiwgKEFFU19DUl9ETUFJTkVOKSk7CiAgfQogIGVsc2UKICB7CiAgICAvKiBFbmFibGUgSW4gYW5kIE91dCBETUEgcmVxdWVzdHMgKi8KICAgIFNFVF9CSVQoaGNyeXAtPkluc3RhbmNlLT5DUiwgKEFFU19DUl9ETUFJTkVOIHwgQUVTX0NSX0RNQU9VVEVOKSk7CiAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCn0KCi8qKgogICogQGJyaWVmICBQcm9jZXNzIERhdGE6IFdyaXRlIElucHV0IGRhdGEgaW4gcG9sbGluZyBtb2RlIGFuZCB1c2VkIGluIEFFUyBmdW5jdGlvbnMuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBUaW1lb3V0OiBTcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfQUVTX1Byb2Nlc3NEYXRhKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKCiAgdWludDMyX3QgdGVtcFs0XTsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwogIHVpbnQzMl90IGk7CiNpZiBkZWZpbmVkIChDUllQKQogIHVpbnQxNl90IGluY291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cEluQ291bnQgVmFsdWUgKi8KICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KI2VuZGlmCgojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBpbmNvdW50ID0gaGNyeXAtPkNyeXBJbkNvdW50OwoKICBpZiAoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX0lGTkYpICE9IDB4MFUpICYmIChpbmNvdW50IDwgKGhjcnlwLT5TaXplIC8gNFUpKSkKICB7CiAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogIH0KCiAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogIGlmIChDUllQX1dhaXRPbk9GTkVGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgewogICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgIC8qIENoYW5nZSBzdGF0ZSAmIGVycm9yIGNvZGUqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQoKICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgaWYgKCgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19PRk5FKSAhPSAweDBVKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpKQogIHsKICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBPdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgICovCiAgICBmb3IgKGkgPSAwVTsgaSA8IDRVOyBpKyspCiAgICB7CiAgICAgIHRlbXBbaV0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICB9CiAgICBpID0gMFU7CiAgICB3aGlsZSAoKChoY3J5cC0+Q3J5cE91dENvdW50IDwgKChoY3J5cC0+U2l6ZSkgLyA0VSkpKSAmJiAoaSA8IDRVKSkKICAgIHsKICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2ldOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIGkrKzsKICAgIH0KICB9CgojZWxzZSAvKiBBRVMgKi8KCiAgLyogV3JpdGUgdGhlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKCiAgLyogV2FpdCBmb3IgQ0NGIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgaWYgKENSWVBfV2FpdE9uQ0NGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgewogICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQoKICAvKiBDbGVhciBDQ0YgRmxhZyAqLwogIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKICAvKiBSZWFkIHRoZSBvdXRwdXQgYmxvY2sgZnJvbSB0aGUgb3V0cHV0IEZJRk8gYW5kIHB1dCB0aGVtIGluIHRlbXBvcmFyeSBidWZmZXIgdGhlbiBnZXQgQ3J5cE91dEJ1ZmYgZnJvbSB0ZW1wb3JhcnkgYnVmZmVyKi8KICBmb3IgKGkgPSAwVTsgaSA8IDRVOyBpKyspCiAgewogICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVFI7CiAgfQogIGkgPSAwVTsKICB3aGlsZSAoKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoKGhjcnlwLT5TaXplICsgM1UpIC8gNFUpKSAmJiAoaSA8IDRVKSkKICB7CiAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cE91dENvdW50KSA9IHRlbXBbaV07CiAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICBpKys7CiAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCn0KCi8qKgogICogQGJyaWVmICBIYW5kbGUgQ1JZUCBibG9jayBpbnB1dC9vdXRwdXQgZGF0YSBoYW5kbGluZyB1bmRlciBpbnRlcnJ1cHRpb24uCiAgKiBAbm90ZSAgIFRoZSBmdW5jdGlvbiBpcyBjYWxsZWQgdW5kZXIgaW50ZXJydXB0aW9uIG9ubHksIG9uY2UKICAqICAgICAgICAgaW50ZXJydXB0aW9ucyBoYXZlIGJlZW4gZW5hYmxlZCBieSBIQUxfQ1JZUF9FbmNyeXB0X0lUIG9yIEhBTF9DUllQX0RlY3J5cHRfSVQuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyB2b2lkIENSWVBfQUVTX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICB1aW50MzJfdCB0ZW1wWzRdOyAgLyogVGVtcG9yYXJ5IENyeXBPdXRCdWZmICovCiAgdWludDMyX3QgaTsKI2lmIGRlZmluZWQgKENSWVApCiAgdWludDE2X3QgaW5jb3VudDsgLyogVGVtcG9yYXJ5IENyeXBJbkNvdW50IFZhbHVlICovCiAgdWludDE2X3Qgb3V0Y291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlICovCiNlbmRpZgoKICBpZiAoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX0JVU1kpCiAgewojaWYgZGVmaW5lZCAoQ1JZUCkKCiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgaW5jb3VudCA9IGhjcnlwLT5DcnlwSW5Db3VudDsKICAgIGlmICgoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfSUZORikgIT0gMHgwVSkgJiYgKGluY291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpKQogICAgewogICAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaWYgKGhjcnlwLT5DcnlwSW5Db3VudCA9PSAoKHVpbnQxNl90KShoY3J5cC0+U2l6ZSkgLyA0VSkpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIGludGVycnVwdHMgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JKTsKCiAgICAgICAgLyogQ2FsbCB0aGUgaW5wdXQgZGF0YSB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX0luQ3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgfQogICAgfQogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgICBpZiAoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpICYmIChvdXRjb3VudCA8IChoY3J5cC0+U2l6ZSAvIDRVKSkpCiAgICB7CiAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgICovCiAgICAgIGZvciAoaSA9IDBVOyBpIDwgNFU7IGkrKykKICAgICAgewogICAgICAgIHRlbXBbaV0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgIH0KICAgICAgaSA9IDBVOwogICAgICB3aGlsZSAoKChoY3J5cC0+Q3J5cE91dENvdW50IDwgKChoY3J5cC0+U2l6ZSkgLyA0VSkpKSAmJiAoaSA8IDRVKSkKICAgICAgewogICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgaSsrOwogICAgICB9CiAgICAgIGlmIChoY3J5cC0+Q3J5cE91dENvdW50ID09ICgodWludDE2X3QpKGhjcnlwLT5TaXplKSAvIDRVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIERpc2FibGUgQ1JZUCAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgICAgICAvKiBDYWxsIE91dHB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KCiNlbHNlIC8qQUVTKi8KCiAgICAvKiBSZWFkIHRoZSBvdXRwdXQgYmxvY2sgZnJvbSB0aGUgb3V0cHV0IEZJRk8gYW5kIHB1dCB0aGVtIGluIHRlbXBvcmFyeSBidWZmZXIgdGhlbiBnZXQgQ3J5cE91dEJ1ZmYgZnJvbSB0ZW1wb3JhcnkgYnVmZmVyKi8KICAgIGZvciAoaSA9IDBVOyBpIDwgNFU7IGkrKykKICAgIHsKICAgICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVFI7CiAgICB9CiAgICBpID0gMFU7CiAgICB3aGlsZSAoKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoKGhjcnlwLT5TaXplICsgM1UpIC8gNFUpKSAmJiAoaSA8IDRVKSkKICAgIHsKICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2ldOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIGkrKzsKICAgIH0KCiAgICBpZiAoaGNyeXAtPkNyeXBPdXRDb3VudCA9PSAoaGNyeXAtPlNpemUgLyA0VSkpCiAgICB7CiAgICAgIC8qIERpc2FibGUgQ29tcHV0YXRpb24gQ29tcGxldGUgZmxhZyBhbmQgZXJyb3JzIGludGVycnVwdHMgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFX0lUKGhjcnlwLCBDUllQX0lUX0NDRklFIHwgQ1JZUF9JVF9FUlJJRSk7CgogICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBEaXNhYmxlIENSWVAgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAgIC8qIENhbGwgT3V0cHV0IHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICBoY3J5cC0+T3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKCiAgICAgIGlmIChoY3J5cC0+Q3J5cEluQ291bnQgPT0gKGhjcnlwLT5TaXplIC8gNFUpKQogICAgICB7CiAgICAgICAgLyogQ2FsbCBJbnB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMVUpCiAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAgKi8KCiAgfQogIGVsc2UKICB7CiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQp9CgovKioKICAqIEBicmllZiAgV3JpdGVzIEtleSBpbiBLZXkgcmVnaXN0ZXJzLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgS2V5U2l6ZTogU2l6ZSBvZiBLZXkKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfU2V0S2V5KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IEtleVNpemUpCnsKI2lmIGRlZmluZWQgKENSWVApCgogIHN3aXRjaCAoS2V5U2l6ZSkKICB7CiAgICBjYXNlIENSWVBfS0VZU0laRV8yNTZCOgogICAgICBoY3J5cC0+SW5zdGFuY2UtPkswTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkswUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksyTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA0KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA3KTsKICAgICAgYnJlYWs7CiAgICBjYXNlIENSWVBfS0VZU0laRV8xOTJCOgogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA0KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNSk7CiAgICAgIGJyZWFrOwogICAgY2FzZSBDUllQX0tFWVNJWkVfMTI4QjoKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAzKTsKCiAgICAgIGJyZWFrOwogICAgZGVmYXVsdDoKICAgICAgYnJlYWs7CiAgfQojZWxzZSAvKkFFUyovCiAgc3dpdGNoIChLZXlTaXplKQogIHsKICAgIGNhc2UgQ1JZUF9LRVlTSVpFXzI1NkI6CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+S0VZUjcgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPktFWVI2ID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+S0VZUjUgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LRVlSNCA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDMpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPktFWVIzID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+S0VZUjIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyA1KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LRVlSMSA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPktFWVIwID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgNyk7CiAgICAgIGJyZWFrOwogICAgY2FzZSBDUllQX0tFWVNJWkVfMTI4QjoKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LRVlSMyA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+S0VZUjIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBLZXkgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LRVlSMSA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEtleSArIDIpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPktFWVIwID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wS2V5ICsgMyk7CgogICAgICBicmVhazsKICAgIGRlZmF1bHQ6CiAgICAgIGJyZWFrOwogIH0KI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAgKi8KfQoKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uL0RlY3J5cHRpb24gcHJvY2VzcyBpbiBBRVMgR0NNIG1vZGUgYW5kIHByZXBhcmUgdGhlIGF1dGhlbnRpY2F0aW9uIFRBRwogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgVGltZW91dDogVGltZW91dCBkdXJhdGlvbgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTR0NNX1Byb2Nlc3MoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IHRpY2tzdGFydDsKICB1aW50MzJfdCB3b3Jkc2l6ZSA9ICh1aW50MzJfdCkoaGNyeXAtPlNpemUpIC8gNFUgOwogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCgogIGlmIChoY3J5cC0+SW5pdC5LZXlJVkNvbmZpZ1NraXAgPT0gQ1JZUF9LRVlJVkNPTkZJR19PTkNFKQogIHsKICAgIGlmIChoY3J5cC0+S2V5SVZDb25maWcgPT0gMVUpCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgIGFuZCBpZiBpdCBoYXMgYWxyZWFkeSBiZWVuIGRvbmUsIHNraXAgaXQgKi8KICAgICAgRG9LZXlJVkNvbmZpZyA9IDBVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gKz0gaGNyeXAtPlNpemU7IC8qIENvbXB1dGUgbWVzc2FnZSB0b3RhbCBwYXlsb2FkIGxlbmd0aCAqLwogICAgfQogICAgZWxzZQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIG5vdCBiZWVuIGRvbmUgYWxyZWFkeSwgZG8gaXQgYW5kIHNldCBLZXlJVkNvbmZpZwogICAgICB0byBrZWVwIHRyYWNrIGl0IHdvbid0IGhhdmUgdG8gYmUgZG9uZSBhZ2FpbiBuZXh0IHRpbWUgKi8KICAgICAgaGNyeXAtPktleUlWQ29uZmlnID0gMVU7CiAgICAgIGhjcnlwLT5TaXplc1N1bSA9IGhjcnlwLT5TaXplOyAvKiBNZXJlbHkgc3RvcmUgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICB9CiAgZWxzZQogIHsKICAgIGhjcnlwLT5TaXplc1N1bSA9IGhjcnlwLT5TaXplOwogIH0KCiAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgewogICAgLyogIFJlc2V0IENyeXBIZWFkZXJDb3VudCAqLwogICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDBVOwoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogSW5pdCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0lOSVQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgojaWYgZGVmaW5lZChDUllQKQoKICAgIC8qIFNldCB0aGUgaW5pdGlhbGl6YXRpb24gdmVjdG9yIGFuZCB0aGUgY291bnRlciA6IEluaXRpYWwgQ291bnRlciBCbG9jayAoSUNCKSovCiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwoKICAgIC8qIEVuYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogR2V0IHRpY2sgKi8KICAgIHRpY2tzdGFydCA9IEhBTF9HZXRUaWNrKCk7CgogICAgLypXYWl0IGZvciB0aGUgQ1JZUEVOIGJpdCB0byBiZSBjbGVhcmVkKi8KICAgIHdoaWxlICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQ1JZUEVOKSA9PSBDUllQX0NSX0NSWVBFTikKICAgIHsKICAgICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICAgIHsKICAgICAgICBpZiAoKChIQUxfR2V0VGljaygpIC0gdGlja3N0YXJ0KSA+IFRpbWVvdXQpIHx8IChUaW1lb3V0ID09IDBVKSkKICAgICAgICB7CiAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICB9CiAgICAgIH0KICAgIH0KCiNlbHNlIC8qIEFFUyAqLwogICAgLyogV29ya2Fyb3VuZCAxIDogb25seSBBRVMuCiAgICBEYXRhdHlwZSBjb25maWd1cmF0aW9uIG11c3QgYmUgMzIgYml0cyBkdXJpbmcgSW5pdCBwaGFzZS4gT25seSwgYWZ0ZXIgSW5pdCwgYW5kIGJlZm9yZSByZQogICAgZW5hYmxpbmcgdGhlIElQLCBkYXRhdHlwZSBkaWZmZXJlbnQgZnJvbSAzMiBiaXRzIGNhbiBiZSBjb25maWd1cmVkLiovCiAgICAvKiBTZWxlY3QgREFUQVRZUEUgMzIgICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9EQVRBVFlQRSwgQ1JZUF9EQVRBVFlQRV8zMkIpOwoKICAgIC8qIFNldCB0aGUgaW5pdGlhbGl6YXRpb24gdmVjdG9yIGFuZCB0aGUgY291bnRlciA6IEluaXRpYWwgQ291bnRlciBCbG9jayAoSUNCKSovCiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjMgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIxID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMCA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAvKiBqdXN0IHdhaXQgZm9yIGhhc2ggY29tcHV0YXRpb24gKi8KICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgJiByZXR1cm4gZXJyb3IgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgICovCgogICAgLyoqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCiAgICBpZiAoQ1JZUF9HQ01DQ01fU2V0SGVhZGVyUGhhc2UoaGNyeXAsICBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICB7CiAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICB9CgogICAgLyoqKioqKioqKioqKioqKioqKioqKioqKipQYXlsb2FkIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCiAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgojaWYgZGVmaW5lZChDUllQKQoKICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAvKiBTZWxlY3QgcGF5bG9hZCBwaGFzZSBvbmNlIHRoZSBoZWFkZXIgcGhhc2UgaXMgcGVyZm9ybWVkICovCiAgICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9QQVlMT0FEKTsKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKI2Vsc2UgLyogQUVTICovCgogICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfUEFZTE9BRCk7CgojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICAqLwogIH0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCgogIGlmICgoaGNyeXAtPlNpemUgJSAxNlUpICE9IDBVKQogIHsKICAgIC8qIHJlY2FsY3VsYXRlICB3b3Jkc2l6ZSAqLwogICAgd29yZHNpemUgPSAoKHdvcmRzaXplIC8gNFUpICogNFUpIDsKICB9CgogIC8qIEdldCB0aWNrICovCiAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgLyogV3JpdGUgaW5wdXQgZGF0YSBhbmQgZ2V0IG91dHB1dCBEYXRhICovCiAgd2hpbGUgKChoY3J5cC0+Q3J5cEluQ291bnQgPCB3b3Jkc2l6ZSkgJiYgKG91dGNvdW50IDwgd29yZHNpemUpKQogIHsKICAgIC8qIFdyaXRlIHBsYWluIGRhdGEgYW5kIGdldCBjaXBoZXIgZGF0YSAqLwogICAgQ1JZUF9BRVNfUHJvY2Vzc0RhdGEoaGNyeXAsIFRpbWVvdXQpOwoKICAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICBpZiAoVGltZW91dCAhPSBIQUxfTUFYX0RFTEFZKQogICAgewogICAgICBpZiAoKChIQUxfR2V0VGljaygpIC0gdGlja3N0YXJ0KSA+IFRpbWVvdXQpIHx8IChUaW1lb3V0ID09IDBVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAmIGVycm9yIGNvZGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0KICB9CgogIGlmICgoaGNyeXAtPlNpemUgJSAxNlUpICE9IDBVKQogIHsKICAgIC8qICBXb3JrYXJvdW5kIDIgOiAgQ1JZUDEgJiBBRVMgZ2VuZXJhdGVzIGNvcnJlY3QgVEFHIGZvciBHQ00gbW9kZSBvbmx5IHdoZW4gaW5wdXQgYmxvY2sgc2l6ZSBpcyBtdWx0aXBsZSBvZgogICAgMTI4IGJpdHMuIElmIGx0aGUgc2l6ZSBvZiB0aGUgbGFzdCBibG9jayBvZiBwYXlsb2FkIGlzIGluZmVyaW9yIHRvIDEyOCBiaXRzLCB3aGVuIEdDTSBlbmNyeXB0aW9uCiAgICBpcyBzZWxlY3RlZCwgdGhlbiB0aGUgVEFHIG1lc3NhZ2Ugd2lsbCBiZSB3cm9uZy4qLwogICAgQ1JZUF9Xb3JrYXJvdW5kKGhjcnlwLCBUaW1lb3V0KTsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgRW5jcnlwdGlvbi9EZWNyeXB0aW9uIHByb2Nlc3MgaW4gQUVTIEdDTSBtb2RlIGFuZCBwcmVwYXJlIHRoZSBhdXRoZW50aWNhdGlvbiBUQUcgaW4gaW50ZXJydXB0IG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNHQ01fUHJvY2Vzc19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgX19JTyB1aW50MzJfdCBjb3VudCA9IDBVOwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCiNpZiBkZWZpbmVkKEFFUykKICB1aW50MzJfdCBsb29wY291bnRlcjsKICB1aW50MzJfdCBsYXN0d29yZHNpemU7CiAgdWludDMyX3QgbnBibGI7CiNlbmRpZiAvKiBBRVMgKi8KCiAgaWYgKGhjcnlwLT5Jbml0LktleUlWQ29uZmlnU2tpcCA9PSBDUllQX0tFWUlWQ09ORklHX09OQ0UpCiAgewogICAgaWYgKGhjcnlwLT5LZXlJVkNvbmZpZyA9PSAxVSkKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBhbHJlYWR5IGJlZW4gZG9uZSwgc2tpcCBpdCAqLwogICAgICBEb0tleUlWQ29uZmlnID0gMFU7CiAgICAgIGhjcnlwLT5TaXplc1N1bSArPSBoY3J5cC0+U2l6ZTsgLyogQ29tcHV0ZSBtZXNzYWdlIHRvdGFsIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qIElmIHRoZSBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gaGFzIHRvIGJlIGRvbmUgb25seSBvbmNlCiAgICAgIGFuZCBpZiBpdCBoYXMgbm90IGJlZW4gZG9uZSBhbHJlYWR5LCBkbyBpdCBhbmQgc2V0IEtleUlWQ29uZmlnCiAgICAgIHRvIGtlZXAgdHJhY2sgaXQgd29uJ3QgaGF2ZSB0byBiZSBkb25lIGFnYWluIG5leHQgdGltZSAqLwogICAgICBoY3J5cC0+S2V5SVZDb25maWcgPSAxVTsKICAgICAgaGNyeXAtPlNpemVzU3VtID0gaGNyeXAtPlNpemU7IC8qIE1lcmVseSBzdG9yZSBwYXlsb2FkIGxlbmd0aCAqLwogICAgfQogIH0KICBlbHNlCiAgewogICAgaGNyeXAtPlNpemVzU3VtID0gaGNyeXAtPlNpemU7CiAgfQoKICAvKiBDb25maWd1cmUgS2V5LCBJViBhbmQgcHJvY2VzcyBtZXNzYWdlIChoZWFkZXIgYW5kIHBheWxvYWQpICovCiAgaWYgKERvS2V5SVZDb25maWcgPT0gMVUpCiAgewogICAgLyogIFJlc2V0IENyeXBIZWFkZXJDb3VudCAqLwogICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCA9IDBVOwoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqIEluaXQgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0lOSVQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgojaWYgZGVmaW5lZChDUllQKQogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgYW5kIHRoZSBjb3VudGVyIDogSW5pdGlhbCBDb3VudGVyIEJsb2NrIChJQ0IpKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDIpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAvKldhaXQgZm9yIHRoZSBDUllQRU4gYml0IHRvIGJlIGNsZWFyZWQqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfSB3aGlsZSAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0NSWVBFTikgPT0gQ1JZUF9DUl9DUllQRU4pOwoKI2Vsc2UgLyogQUVTICovCgogICAgLyogV29ya2Fyb3VuZCAxIDogb25seSBBRVMKICAgIERhdGF0eXBlIGNvbmZpZ3VyYXRpb24gbXVzdCBiZSAzMiBiaXRzIGR1cmluZyBJTklUIHBoYXNlLiBPbmx5LCBhZnRlciBJTklULCBhbmQgYmVmb3JlIHJlCiAgICBlbmFibGluZyB0aGUgSVAsIGRhdGF0eXBlIGRpZmZlcmVudCBmcm9tIDMyIGJpdHMgY2FuIGJlIGNvbmZpZ3VyZWQuKi8KICAgIC8qIFNlbGVjdCBEQVRBVFlQRSAzMiAgKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RBVEFUWVBFLCBDUllQX0RBVEFUWVBFXzMyQik7CgogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgYW5kIHRoZSBjb3VudGVyIDogSW5pdGlhbCBDb3VudGVyIEJsb2NrIChJQ0IpKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMyA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjEgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDIpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIwID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAzKTsKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIC8qIGp1c3Qgd2FpdCBmb3IgaGFzaCBjb21wdXRhdGlvbiAqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQUVTX1NSX0NDRikpOwoKICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KCiAgICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogSGVhZGVyIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCiNpZiBkZWZpbmVkKENSWVApCgogICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JKTsKCiAgICAvKiBFbmFibGUgQ1JZUCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKI2Vsc2UgLyogQUVTICovCgogICAgLyogV29ya2Fyb3VuZCAxOiBvbmx5IEFFUyAsIGJlZm9yZSByZS1lbmFibGluZyB0aGUgSVAsIGRhdGF0eXBlIGNhbiBiZSBjb25maWd1cmVkKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RBVEFUWVBFLCBoY3J5cC0+SW5pdC5EYXRhVHlwZSk7CgogICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAvKiBFbmFibGUgY29tcHV0YXRpb24gY29tcGxldGUgZmxhZyBhbmQgZXJyb3IgaW50ZXJydXB0cyAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEVfSVQoaGNyeXAsIENSWVBfSVRfQ0NGSUUgfCBDUllQX0lUX0VSUklFKTsKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIGlmIChoY3J5cC0+SW5pdC5IZWFkZXJTaXplID09IDBVKSAvKmhlYWRlciBwaGFzZSBpcyAgc2tpcHBlZCovCiAgICB7CiAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9HQ01QSCwgQ1JZUF9QSEFTRV9QQVlMT0FEKTsKCiAgICAgIC8qIFdyaXRlIHRoZSBwYXlsb2FkIElucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGlmIChoY3J5cC0+U2l6ZSA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9DQ0ZJRSB8IENSWVBfSVRfRVJSSUUpOwoKICAgICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgIH0KICAgICAgZWxzZSBpZiAoaGNyeXAtPlNpemUgPj0gMTZVKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgICBpZiAoaGNyeXAtPkNyeXBJbkNvdW50ID09IChoY3J5cC0+U2l6ZSAvIDRVKSkKICAgICAgICB7CiAgICAgICAgICAvKiBDYWxsIElucHV0IHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgICBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgICAgfQogICAgICB9CiAgICAgIGVsc2UgLyogU2l6ZSA8IDE2Qnl0ZXMgIDogZmlyc3QgYmxvY2sgaXMgdGhlIGxhc3QgYmxvY2sqLwogICAgICB7CiAgICAgICAgLyogV29ya2Fyb3VuZCBub3QgaW1wbGVtZW50ZWQqLwogICAgICAgIC8qIFNpemUgc2hvdWxkIGJlICU0ICBvdGhlcndpc2UgVGFnIHdpbGwgIGJlIGluY29ycmVjdGx5IGdlbmVyYXRlZCBmb3IgR0NNIEVuY3J5cHRpb246CiAgICAgICAgV29ya2Fyb3VuZCBpcyBpbXBsZW1lbnRlZCBpbiBwb2xsaW5nIG1vZGUsIHNvIGlmIGxhc3QgYmxvY2sgb2YKICAgICAgICBwYXlsb2FkIDwxMjhiaXQgZG9uJ3QgdXNlIENSWVBfRW5jcnlwdF9JVCBvdGhlcndpc2UgVEFHIGlzIGluY29ycmVjdGx5IGdlbmVyYXRlZCBmb3IgR0NNIEVuY3J5cHRpb24uICovCgogICAgICAgIC8qIENvbXB1dGUgdGhlIG51bWJlciBvZiBwYWRkaW5nIGJ5dGVzIGluIGxhc3QgYmxvY2sgb2YgcGF5bG9hZCAqLwogICAgICAgIG5wYmxiID0gMTZVIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogICAgICAgIC8qIE51bWJlciBvZiB2YWxpZCB3b3JkcyAobGFzdHdvcmRzaXplKSBpbiBsYXN0IGJsb2NrICovCiAgICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAwVSkKICAgICAgICB7CiAgICAgICAgICBsYXN0d29yZHNpemUgPSAoMTZVIC0gbnBibGIpIC8gNFU7CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICBsYXN0d29yZHNpemUgPSAoKDE2VSAtIG5wYmxiKSAvIDRVKSArIDFVOwogICAgICAgIH0KCiAgICAgICAgLyogIGxhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyBsb29wY291bnRlciA8IGxhc3R3b3Jkc2l6ZSA7IGxvb3Bjb3VudGVyKyspCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgICAgfQogICAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICAgIHsKICAgICAgICAgIC8qIHBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gMHgwVTsKICAgICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgICAgfQogICAgICB9CiAgICB9CiAgICBlbHNlIGlmICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSkgPCA0VSkKICAgIHsKICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyBsb29wY291bnRlciA8IGhjcnlwLT5Jbml0LkhlYWRlclNpemUgOyBsb29wY291bnRlcisrKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogcGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gMHgwVTsKICAgICAgICBsb29wY291bnRlcisrOwogICAgICB9CiAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9QQVlMT0FEKTsKCiAgICAgIC8qIENhbGwgSW5wdXQgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQogICAgZWxzZSBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUpID49IDRVKQogICAgewogICAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKzsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogTm90aGluZyB0byBkbyAqLwogICAgfQoKI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAqLwogIH0gLyogZW5kIG9mIGlmIChEb0tleUlWQ29uZmlnID09IDFVKSAqLwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uL0RlY3J5cHRpb24gcHJvY2VzcyBpbiBBRVMgR0NNIG1vZGUgYW5kIHByZXBhcmUgdGhlIGF1dGhlbnRpY2F0aW9uIFRBRyB1c2luZyBETUEKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNHQ01fUHJvY2Vzc19ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKICB1aW50MzJfdCB3b3Jkc2l6ZTsKICB1aW50MzJfdCBEb0tleUlWQ29uZmlnID0gMVU7IC8qIEJ5IGRlZmF1bHQsIGNhcnJ5IG91dCBwZXJpcGhlcmFsIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKICAgIC8qICBSZXNldCBDcnlwSGVhZGVyQ291bnQgKi8KICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQgPSAwVTsKCiAgICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqIEluaXQgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0lOSVQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgojaWYgZGVmaW5lZChDUllQKQogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgYW5kIHRoZSBjb3VudGVyIDogSW5pdGlhbCBDb3VudGVyIEJsb2NrIChJQ0IpKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxTFIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDIpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMyk7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAvKldhaXQgZm9yIHRoZSBDUllQRU4gYml0IHRvIGJlIGNsZWFyZWQqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfSB3aGlsZSAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0NSWVBFTikgPT0gQ1JZUF9DUl9DUllQRU4pOwoKI2Vsc2UgLyogQUVTICovCgogICAgLypXb3JrYXJvdW5kIDEgOiBvbmx5IEFFUwogICAgRGF0YXR5cGUgY29uZmlndXJhdGlvbiBtdXN0IGJlIDMyIGJpdHMgZHVyaW5nIEluaXQgcGhhc2UuIE9ubHksIGFmdGVyIEluaXQsIGFuZCBiZWZvcmUgcmUKICAgIGVuYWJsaW5nIHRoZSBJUCwgZGF0YXR5cGUgZGlmZmVyZW50IGZyb20gMzIgYml0cyBjYW4gYmUgY29uZmlndXJlZC4qLwogICAgLyogU2VsZWN0IERBVEFUWVBFIDMyICAqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfREFUQVRZUEUsIENSWVBfREFUQVRZUEVfMzJCKTsKCiAgICAvKiBTZXQgdGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciBhbmQgdGhlIGNvdW50ZXIgOiBJbml0aWFsIENvdW50ZXIgQmxvY2sgKElDQikqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIzID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIyID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QgKyAxKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMSA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQucEluaXRWZWN0ICsgMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCArIDMpOwoKICAgIC8qIEVuYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyoganVzdCB3YWl0IGZvciBoYXNoIGNvbXB1dGF0aW9uICovCiAgICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01JTklUUEhBU0U7CiAgICBkbwogICAgewogICAgICBjb3VudC0tIDsKICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBBRVNfU1JfQ0NGKSk7CgogICAgLyogQ2xlYXIgQ0NGIGZsYWcgKi8KICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAqLwoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqKiogSGVhZGVyIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogICAgaWYgKENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0RNQShoY3J5cCkgIT0gSEFMX09LKQogICAgewogICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgfQoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqKiogUGF5bG9hZCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKI2lmIGRlZmluZWQoQ1JZUCkKCiAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgojZW5kaWYgLyogQ1JZUCAqLwoKICAgIC8qIFNlbGVjdCBwYXlsb2FkIHBoYXNlIG9uY2UgdGhlIGhlYWRlciBwaGFzZSBpcyBwZXJmb3JtZWQgKi8KICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX1BBWUxPQUQpOwoKICB9IC8qIGlmIChEb0tleUlWQ29uZmlnID09IDFVKSAqLwoKICBpZiAoaGNyeXAtPlNpemUgIT0gMFUpCiAgewogICAgLyogQ1JZUDEgSVAgViA8IDIuMi4xICBTaXplIHNob3VsZCBiZSAlNCAgb3RoZXJ3aXNlIFRhZyB3aWxsICBiZSBpbmNvcnJlY3RseSBnZW5lcmF0ZWQgZm9yIEdDTSBFbmNyeXB0aW9uOgogICAgV29ya2Fyb3VuZCBpcyBpbXBsZW1lbnRlZCBpbiBwb2xsaW5nIG1vZGUsIHNvIGlmIGxhc3QgYmxvY2sgb2YKICAgIHBheWxvYWQgPDEyOGJpdCBkb24ndCB1c2UgRE1BIG1vZGUgb3RoZXJ3aXNlIFRBRyBpcyBpbmNvcnJlY3RseSBnZW5lcmF0ZWQgLiAqLwogICAgLyogU2V0IHRoZSBpbnB1dCBhbmQgb3V0cHV0IGFkZHJlc3NlcyBhbmQgc3RhcnQgRE1BIHRyYW5zZmVyICovCiAgICBpZiAoKGhjcnlwLT5TaXplICUgMTZVKSA9PSAwVSkKICAgIHsKICAgICAgQ1JZUF9TZXRETUFDb25maWcoaGNyeXAsICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwSW5CdWZmUHRyKSwgKGhjcnlwLT5TaXplIC8gNFUpLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICAgIH0KICAgIGVsc2UgLyp0byBjb21wdXRlIGxhc3Qgd29yZDwxMjhiaXRzLCBvdGhlcndpc2UgaXQgd2lsbCBub3QgYmUgZW5jcnlwdGVkL2RlY3J5cHRlZCAqLwogICAgewogICAgICB3b3Jkc2l6ZSA9ICh1aW50MzJfdCkoaGNyeXAtPlNpemUpICsgKDE2VSAtICgodWludDMyX3QpKGhjcnlwLT5TaXplKSAlIDE2VSkpIDsKCiAgICAgIC8qIFNldCB0aGUgaW5wdXQgYW5kIG91dHB1dCBhZGRyZXNzZXMgYW5kIHN0YXJ0IERNQSB0cmFuc2ZlciwgcENyeXBPdXRCdWZmUHRyIHNpemUgc2hvdWxkIGJlICU0ICovCiAgICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksICgodWludDE2X3Qpd29yZHNpemUgLyA0VSksCiAgICAgICAgICAgICAgICAgICAgICAgICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwT3V0QnVmZlB0cikpOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogUHJvY2VzcyB1bkxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgYW5kIHBoYXNlICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAYnJpZWYgIEFFUyBDQ00gZW5jcnlwdGlvbi9kZWNyeXB0aW9uIHByb2Nlc3NpbmcgaW4gcG9sbGluZyBtb2RlCiAgKiAgICAgICAgIGZvciBUaW55QUVTIElQLCBubyBlbmNyeXB0L2RlY3J5cHQgcGVyZm9ybWVkLCBvbmx5IGF1dGhlbnRpY2F0aW9uIHByZXBhcmF0aW9uLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgVGltZW91dDogVGltZW91dCBkdXJhdGlvbgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTQ0NNX1Byb2Nlc3MoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IHRpY2tzdGFydDsKICB1aW50MzJfdCB3b3Jkc2l6ZSA9ICh1aW50MzJfdCkoaGNyeXAtPlNpemUpIC8gNFU7CiAgdWludDE2X3Qgb3V0Y291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlICovCiAgdWludDMyX3QgRG9LZXlJVkNvbmZpZyA9IDFVOyAvKiBCeSBkZWZhdWx0LCBjYXJyeSBvdXQgcGVyaXBoZXJhbCBLZXkgYW5kIElWIGNvbmZpZ3VyYXRpb24gKi8KI2lmIGRlZmluZWQoQUVTKQogIHVpbnQzMl90IGxvb3Bjb3VudGVyOwogIHVpbnQzMl90IG5wYmxiOwogIHVpbnQzMl90IGxhc3R3b3Jkc2l6ZTsKI2VuZGlmIC8qIEFFUyAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKCiAgICAvKiAgUmVzZXQgQ3J5cEhlYWRlckNvdW50ICovCiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50ID0gMFU7CgojaWYgZGVmaW5lZChDUllQKQoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqIEluaXQgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0lOSVQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgKElWKSB3aXRoIENUUjEgaW5mb3JtYXRpb24gKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAoaGNyeXAtPkluaXQuQjBbMF0pICYgQ1JZUF9DQ01fQ1RSMV8wOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9IGhjcnlwLT5Jbml0LkIwWzFdOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9IGhjcnlwLT5Jbml0LkIwWzJdOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9IChoY3J5cC0+SW5pdC5CMFszXSAmIENSWVBfQ0NNX0NUUjFfMSkgfCAgQ1JZUF9DQ01fQ1RSMV8yOwoKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIC8qV3JpdGUgIEIwIHBhY2tldCBpbnRvIENSWVBfRElOIFJlZ2lzdGVyKi8KICAgIGlmIChoY3J5cC0+SW5pdC5EYXRhVHlwZSA9PSBDUllQX0RBVEFUWVBFXzhCKQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpKTsKICAgIH0KICAgIGVsc2UgaWYgKGhjcnlwLT5Jbml0LkRhdGFUeXBlID09IENSWVBfREFUQVRZUEVfMTZCKQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApLCAxNik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19ST1IoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpLCAxNik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19ST1IoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDIpLCAxNik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19ST1IoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpLCAxNik7CiAgICB9CiAgICBlbHNlIGlmIChoY3J5cC0+SW5pdC5EYXRhVHlwZSA9PSBDUllQX0RBVEFUWVBFXzFCKQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SQklUKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SQklUKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SQklUKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSk7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMyk7CiAgICB9CiAgICAvKiBHZXQgdGljayAqLwogICAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKCiAgICAvKldhaXQgZm9yIHRoZSBDUllQRU4gYml0IHRvIGJlIGNsZWFyZWQqLwogICAgd2hpbGUgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9DUllQRU4pID09IENSWVBfQ1JfQ1JZUEVOKQogICAgewogICAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgICAgaWYgKFRpbWVvdXQgIT0gSEFMX01BWF9ERUxBWSkKICAgICAgewogICAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICAgIHsKICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgIH0KICAgICAgfQogICAgfQojZWxzZSAvKiBBRVMgKi8KICAgIC8qQUVTMnYxLjEuMSA6IENDTSBhdXRoZW50aWNhdGlvbiA6IG5vIGluaXQgcGhhc2UsIG9ubHkgaGVhZGVyIGFuZCBmaW5hbCBwaGFzZSAqLwogICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAvKiBjb25maWd1cmVkIGVuY3J5cHRpb24gbW9kZSAqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfTU9ERSwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3Igd2l0aCB6ZXJvIHZhbHVlcyovCiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjMgPSAwVTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMiA9IDBVOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIxID0gMFU7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjAgPSAwVTsKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIC8qV3JpdGUgdGhlIEIwIHBhY2tldCBpbnRvIENSWVBfRElOKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDIpOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwoKICAgIC8qICB3YWl0IHVudGlsIHRoZSBlbmQgb2YgY29tcHV0YXRpb24gKi8KICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgJiByZXR1cm4gZXJyb3IgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgLyogRnJvbSB0aGF0IHBvaW50IHRoZSB3aG9sZSBtZXNzYWdlIG11c3QgYmUgcHJvY2Vzc2VkLCBmaXJzdCB0aGUgSGVhZGVyIHRoZW4gdGhlIHBheWxvYWQuCiAgICBGaXJzdCB0aGUgIEhlYWRlciBibG9jayhCMSkgOiBhc3NvY2lhdGVkIGRhdGEgbGVuZ3RoIGV4cHJlc3NlZCBpbiBieXRlcyBjb25jYXRlbmF0ZWQgd2l0aCBBc3NvY2lhdGVkIERhdGEgKEEpKi8KCiAgICBpZiAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAhPSAwVSkKICAgIHsKICAgICAgaWYgKChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpID09IDBVKQogICAgICB7CiAgICAgICAgLyogSGVhZGVyU2l6ZSAlNCwgbm8gcGFkZGluZyAqLwogICAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgKGxvb3Bjb3VudGVyIDwgaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSk7IGxvb3Bjb3VudGVyICs9IDRVKQogICAgICAgIHsKICAgICAgICAgIC8qIFdyaXRlIHRoZSBJbnB1dCBibG9jayBpbiB0aGUgRGF0YSBJbnB1dCByZWdpc3RlciAqLwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwoKICAgICAgICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICAgICAgewogICAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgICAgfQogICAgICAgICAgLyogQ2xlYXIgQ0NGIEZsYWcgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwogICAgICAgIH0KICAgICAgfQogICAgICBlbHNlCiAgICAgIHsKICAgICAgICAvKldyaXRlIEhlYWRlciBibG9jayBpbiB0aGUgSU4gRklGTyB3aXRob3V0IGxhc3QgYmxvY2sgKi8KICAgICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8ICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSkgLSAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSkpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgICB7CiAgICAgICAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIGRhdGEgaW5wdXQgcmVnaXN0ZXIgKi8KICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKCiAgICAgICAgICBpZiAoQ1JZUF9XYWl0T25DQ0ZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgIH0KICAgICAgICAgIC8qIENsZWFyIENDRiBGbGFnICovCiAgICAgICAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKICAgICAgICB9CiAgICAgICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyAobG9vcGNvdW50ZXIgPCAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSk7IGxvb3Bjb3VudGVyKyspCiAgICAgICAgewogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgfQogICAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICAgIHsKICAgICAgICAgIC8qIFBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gMHgwVTsKICAgICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgICAgfQoKICAgICAgICBpZiAoQ1JZUF9XYWl0T25DQ0ZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgICB7CiAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICB9CiAgICAgICAgLyogQ2xlYXIgQ0NGIGZsYWcgKi8KICAgICAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKICAgICAgfQogICAgfQogIH0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCiAgLyogVGhlbiB0aGUgcGF5bG9hZDogY2xlYXJ0ZXh0IHBheWxvYWQgKG5vdCB0aGUgY2lwaGVydGV4dCBwYXlsb2FkKS4KICBXcml0ZSBpbnB1dCBEYXRhLCBubyBvdXRwdXQgRGF0YSB0byBnZXQgKi8KICBpZiAoaGNyeXAtPlNpemUgIT0gMFUpCiAgewogICAgaWYgKChoY3J5cC0+U2l6ZSAlIDE2VSkgIT0gMFUpCiAgICB7CiAgICAgIC8qIHJlY2FsY3VsYXRlICB3b3Jkc2l6ZSAqLwogICAgICB3b3Jkc2l6ZSA9ICgod29yZHNpemUgLyA0VSkgKiA0VSkgOwogICAgfQoKICAgIC8qIEdldCB0aWNrICovCiAgICB0aWNrc3RhcnQgPSBIQUxfR2V0VGljaygpOwogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKCiAgICB3aGlsZSAoKGhjcnlwLT5DcnlwSW5Db3VudCA8IHdvcmRzaXplKSAmJiAob3V0Y291bnQgPCB3b3Jkc2l6ZSkpCiAgICB7CiAgICAgIC8qIFdyaXRlIHBsYWluIGRhdGEgYW5kIGdldCBjaXBoZXIgZGF0YSAqLwogICAgICBDUllQX0FFU19Qcm9jZXNzRGF0YShoY3J5cCwgVGltZW91dCk7CgogICAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogICAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgICAgaWYgKFRpbWVvdXQgIT0gSEFMX01BWF9ERUxBWSkKICAgICAgewogICAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICAgIHsKICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgIH0KICAgICAgfQogICAgfQoKICAgIGlmICgoaGNyeXAtPlNpemUgJSAxNlUpICE9IDBVKQogICAgewogICAgICAvKiBDb21wdXRlIHRoZSBudW1iZXIgb2YgcGFkZGluZyBieXRlcyBpbiBsYXN0IGJsb2NrIG9mIHBheWxvYWQgKi8KICAgICAgbnBibGIgPSAoKCgodWludDMyX3QpKGhjcnlwLT5TaXplKSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogICAgICAvKiBOdW1iZXIgb2YgdmFsaWQgd29yZHMgKGxhc3R3b3Jkc2l6ZSkgaW4gbGFzdCBibG9jayAqLwogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDBVKQogICAgICB7CiAgICAgICAgbGFzdHdvcmRzaXplID0gKDE2VSAtIG5wYmxiKSAvIDRVOwogICAgICB9CiAgICAgIGVsc2UKICAgICAgewogICAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgoMTZVIC0gbnBibGIpIC8gNFUpICsgMVU7CiAgICAgIH0KICAgICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgbG9vcGNvdW50ZXIgPCBsYXN0d29yZHNpemU7IGxvb3Bjb3VudGVyICsrKQogICAgICB7CiAgICAgICAgLyogV3JpdGUgdGhlIGxhc3QgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9IDBVOwogICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgIH0KICAgICAgLyogV2FpdCBmb3IgQ0NGIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKICAgIH0KICB9CiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KCiNpZiBkZWZpbmVkKENSWVApCgogICAgLyoqKioqKioqKioqKioqKioqKioqKioqKiogSGVhZGVyIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCiAgICAvKiBIZWFkZXIgYmxvY2soQjEpIDogYXNzb2NpYXRlZCBkYXRhIGxlbmd0aCBleHByZXNzZWQgaW4gYnl0ZXMgY29uY2F0ZW5hdGVkCiAgICB3aXRoIEFzc29jaWF0ZWQgRGF0YSAoQSkqLwoKICAgIGlmIChDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZShoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgfQoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqIFBheWxvYWQgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfUEFZTE9BRCk7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgfSAvKiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KCiAgaWYgKChoY3J5cC0+U2l6ZSAlIDE2VSkgIT0gMFUpCiAgewogICAgLyogcmVjYWxjdWxhdGUgIHdvcmRzaXplICovCiAgICB3b3Jkc2l6ZSA9ICgod29yZHNpemUgLyA0VSkgKiA0VSkgOwogIH0KICAvKiBHZXQgdGljayAqLwogIHRpY2tzdGFydCA9IEhBTF9HZXRUaWNrKCk7CiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogIC8qIFdyaXRlIGlucHV0IGRhdGEgYW5kIGdldCBvdXRwdXQgZGF0YSAqLwogIHdoaWxlICgoaGNyeXAtPkNyeXBJbkNvdW50IDwgd29yZHNpemUpICYmIChvdXRjb3VudCA8IHdvcmRzaXplKSkKICB7CiAgICAvKiBXcml0ZSBwbGFpbiBkYXRhIGFuZCBnZXQgY2lwaGVyIGRhdGEgKi8KICAgIENSWVBfQUVTX1Byb2Nlc3NEYXRhKGhjcnlwLCBUaW1lb3V0KTsKCiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9CiAgfQoKICBpZiAoKGhjcnlwLT5TaXplICUgMTZVKSAhPSAwVSkKICB7CiAgICAvKiBDUllQIFdvcmthcm91bmQgOiAgQ1JZUDEgZ2VuZXJhdGVzIGNvcnJlY3QgVEFHICBkdXJpbmcgQ0NNIGRlY3J5cHRpb24gb25seSB3aGVuIGNpcGhlcnRleHQgYmxvY2tzIHNpemUgaXMgbXVsdGlwbGUgb2YKICAgIDEyOCBiaXRzLiBJZiBsdGhlIHNpemUgb2YgdGhlIGxhc3QgYmxvY2sgb2YgcGF5bG9hZCBpcyBpbmZlcmlvciB0byAxMjggYml0cywgd2hlbiBDQ00gZGVjcnlwdGlvbgogICAgaXMgc2VsZWN0ZWQsIHRoZW4gdGhlIFRBRyBtZXNzYWdlIHdpbGwgYmUgd3JvbmcuKi8KICAgIENSWVBfV29ya2Fyb3VuZChoY3J5cCwgVGltZW91dCk7CiAgfQojZW5kaWYgLyogQ1JZUCAqLwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIEFFUyBDQ00gZW5jcnlwdGlvbi9kZWNyeXB0aW9uIHByb2Nlc3MgaW4gaW50ZXJydXB0IG1vZGUKICAqICAgICAgICAgZm9yIFRpbnlBRVMgSVAsIG5vIGVuY3J5cHQvZGVjcnlwdCBwZXJmb3JtZWQsIG9ubHkgYXV0aGVudGljYXRpb24gcHJlcGFyYXRpb24uCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTQ0NNX1Byb2Nlc3NfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCiNpZiBkZWZpbmVkKENSWVApCiAgX19JTyB1aW50MzJfdCBjb3VudCA9IDBVOwojZW5kaWYgLyogQ1JZUCAqLwoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIC8qIENvbmZpZ3VyZSBLZXksIElWIGFuZCBwcm9jZXNzIG1lc3NhZ2UgKGhlYWRlciBhbmQgcGF5bG9hZCkgKi8KICBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkKICB7CiAgICAvKiAgUmVzZXQgQ3J5cEhlYWRlckNvdW50ICovCiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50ID0gMFU7CgojaWYgZGVmaW5lZChDUllQKQoKICAgIC8qKioqKioqKioqKiogSW5pdCBwaGFzZSAqKioqKioqKioqKiovCgogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSU5JVCk7CgogICAgLyogU2V0IHRoZSBrZXkgKi8KICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAvKiBTZXQgdGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciAoSVYpIHdpdGggQ1RSMSBpbmZvcm1hdGlvbiAqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9IChoY3J5cC0+SW5pdC5CMFswXSkgJiBDUllQX0NDTV9DVFIxXzA7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gaGNyeXAtPkluaXQuQjBbMV07CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gaGNyeXAtPkluaXQuQjBbMl07CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKGhjcnlwLT5Jbml0LkIwWzNdICYgQ1JZUF9DQ01fQ1RSMV8xKSB8ICBDUllQX0NDTV9DVFIxXzI7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAvKldyaXRlIHRoZSBCMCBwYWNrZXQgaW50byBDUllQX0RJTiBSZWdpc3RlciovCiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV84QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSk7CiAgICB9CiAgICBlbHNlIGlmIChoY3J5cC0+SW5pdC5EYXRhVHlwZSA9PSBDUllQX0RBVEFUWVBFXzE2QikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAyKSwgMTYpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUk9SKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKSwgMTYpOwogICAgfQogICAgZWxzZSBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV8xQikKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMikpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkJJVCgqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMykpOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpOwogICAgfQogICAgLypXYWl0IGZvciB0aGUgQ1JZUEVOIGJpdCB0byBiZSBjbGVhcmVkKi8KICAgIGNvdW50ID0gQ1JZUF9USU1FT1VUX0dDTUNDTUlOSVRQSEFTRTsKICAgIGRvCiAgICB7CiAgICAgIGNvdW50LS0gOwogICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0gd2hpbGUgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9DUllQRU4pID09IENSWVBfQ1JfQ1JZUEVOKTsKCiAgICAvKiBTZWxlY3QgaGVhZGVyIHBoYXNlICovCiAgICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9IRUFERVIpOwoKICB9IC8qIGVuZCBvZiBpZiAoRG9LZXlJVkNvbmZpZyA9PSAxVSkgKi8KCiAgLyogRW5hYmxlIGludGVycnVwdHMgKi8KICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwoKICAvKiBFbmFibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiNlbHNlIC8qIEFFUyAqLwoKICAgIC8qQUVTMnYxLjEuMSA6IENDTSBhdXRoZW50aWNhdGlvbiA6IG5vIGluaXQgcGhhc2UsIG9ubHkgaGVhZGVyIGFuZCBmaW5hbCBwaGFzZSAqLwogICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAvKiBjb25maWd1cmVkIG1vZGUgYW5kIGVuY3J5cHRpb24gbW9kZSAqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfTU9ERSwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3Igd2l0aCB6ZXJvIHZhbHVlcyovCiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjMgPSAwVTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMiA9IDBVOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIxID0gMFU7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjAgPSAwVTsKCiAgICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEVfSVQoaGNyeXAsIENSWVBfSVRfQ0NGSUUgfCBDUllQX0lUX0VSUklFKTsKICAgIC8qIEVuYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLypXcml0ZSB0aGUgQjAgcGFja2V0IGludG8gQ1JZUF9ESU4qLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMyk7CgogIH0gLyogZW5kIG9mIGlmIChEb0tleUlWQ29uZmlnID09IDFVKSAqLwojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9Ci8qKgogICogQGJyaWVmICBBRVMgQ0NNIGVuY3J5cHRpb24vZGVjcnlwdGlvbiBwcm9jZXNzIGluIERNQSBtb2RlCiAgKiAgICAgICAgIGZvciBUaW55QUVTIElQLCBubyBlbmNyeXB0L2RlY3J5cHQgcGVyZm9ybWVkLCBvbmx5IGF1dGhlbnRpY2F0aW9uIHByZXBhcmF0aW9uLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU0NDTV9Qcm9jZXNzX0RNQShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgdWludDMyX3Qgd29yZHNpemU7CiAgX19JTyB1aW50MzJfdCBjb3VudCA9IDBVOwogIHVpbnQzMl90IERvS2V5SVZDb25maWcgPSAxVTsgLyogQnkgZGVmYXVsdCwgY2Fycnkgb3V0IHBlcmlwaGVyYWwgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uICovCiNpZiBkZWZpbmVkKEFFUykKICB1aW50MzJfdCBsb29wY291bnRlcjsKICB1aW50MzJfdCBucGJsYjsKICB1aW50MzJfdCBsYXN0d29yZHNpemU7CiNlbmRpZgoKICBpZiAoaGNyeXAtPkluaXQuS2V5SVZDb25maWdTa2lwID09IENSWVBfS0VZSVZDT05GSUdfT05DRSkKICB7CiAgICBpZiAoaGNyeXAtPktleUlWQ29uZmlnID09IDFVKQogICAgewogICAgICAvKiBJZiB0aGUgS2V5IGFuZCBJViBjb25maWd1cmF0aW9uIGhhcyB0byBiZSBkb25lIG9ubHkgb25jZQogICAgICBhbmQgaWYgaXQgaGFzIGFscmVhZHkgYmVlbiBkb25lLCBza2lwIGl0ICovCiAgICAgIERvS2V5SVZDb25maWcgPSAwVTsKICAgICAgaGNyeXAtPlNpemVzU3VtICs9IGhjcnlwLT5TaXplOyAvKiBDb21wdXRlIG1lc3NhZ2UgdG90YWwgcGF5bG9hZCBsZW5ndGggKi8KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLyogSWYgdGhlIEtleSBhbmQgSVYgY29uZmlndXJhdGlvbiBoYXMgdG8gYmUgZG9uZSBvbmx5IG9uY2UKICAgICAgYW5kIGlmIGl0IGhhcyBub3QgYmVlbiBkb25lIGFscmVhZHksIGRvIGl0IGFuZCBzZXQgS2V5SVZDb25maWcKICAgICAgdG8ga2VlcCB0cmFjayBpdCB3b24ndCBoYXZlIHRvIGJlIGRvbmUgYWdhaW4gbmV4dCB0aW1lICovCiAgICAgIGhjcnlwLT5LZXlJVkNvbmZpZyA9IDFVOwogICAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsgLyogTWVyZWx5IHN0b3JlIHBheWxvYWQgbGVuZ3RoICovCiAgICB9CiAgfQogIGVsc2UKICB7CiAgICBoY3J5cC0+U2l6ZXNTdW0gPSBoY3J5cC0+U2l6ZTsKICB9CgogIGlmIChEb0tleUlWQ29uZmlnID09IDFVKQogIHsKCiAgICAvKiAgUmVzZXQgQ3J5cEhlYWRlckNvdW50ICovCiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50ID0gMFU7CgojaWYgZGVmaW5lZChDUllQKQoKICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKiBJbml0IHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0lOSVQpOwoKICAgIC8qIFNldCB0aGUga2V5ICovCiAgICBDUllQX1NldEtleShoY3J5cCwgaGNyeXAtPkluaXQuS2V5U2l6ZSk7CgogICAgLyogU2V0IHRoZSBpbml0aWFsaXphdGlvbiB2ZWN0b3IgKElWKSB3aXRoIENUUjEgaW5mb3JtYXRpb24gKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAoaGNyeXAtPkluaXQuQjBbMF0pICYgQ1JZUF9DQ01fQ1RSMV8wOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9IGhjcnlwLT5Jbml0LkIwWzFdOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9IGhjcnlwLT5Jbml0LkIwWzJdOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9IChoY3J5cC0+SW5pdC5CMFszXSAmIENSWVBfQ0NNX0NUUjFfMSkgfCAgQ1JZUF9DQ01fQ1RSMV8yOwoKICAgIC8qIEVuYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLypXcml0ZSB0aGUgQjAgcGFja2V0IGludG8gQ1JZUF9ESU4gUmVnaXN0ZXIqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFUeXBlID09IENSWVBfREFUQVRZUEVfOEIpCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IF9fUkVWKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAxKSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SRVYoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDIpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JFVigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMykpOwogICAgfQogICAgZWxzZSBpZiAoaGNyeXAtPkluaXQuRGF0YVR5cGUgPT0gQ1JZUF9EQVRBVFlQRV8xNkIpCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19ST1IoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCksIDE2KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSksIDE2KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMiksIDE2KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JPUigqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMyksIDE2KTsKICAgIH0KICAgIGVsc2UgaWYgKGhjcnlwLT5Jbml0LkRhdGFUeXBlID09IENSWVBfREFUQVRZUEVfMUIpCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gX19SQklUKCoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjApKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDEpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDIpKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBfX1JCSVQoKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDMpKTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwKTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCArIDIpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuQjAgKyAzKTsKICAgIH0KCiAgICAvKldhaXQgZm9yIHRoZSBDUllQRU4gYml0IHRvIGJlIGNsZWFyZWQqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfSB3aGlsZSAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0NSWVBFTikgPT0gQ1JZUF9DUl9DUllQRU4pOwoKI2Vsc2UgLyogQUVTICovCgogICAgLypBRVMydjEuMS4xIDogQ0NNIGF1dGhlbnRpY2F0aW9uIDogbm8gaW5pdCBwaGFzZSwgb25seSBoZWFkZXIgYW5kIGZpbmFsIHBoYXNlICovCiAgICAvKiBTZWxlY3QgaGVhZGVyIHBoYXNlICovCiAgICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9IRUFERVIpOwoKICAgIC8qIGNvbmZpZ3VyZWQgZW5jcnlwdGlvbiBtb2RlICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9NT0RFLCBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCk7CgogICAgLyogU2V0IHRoZSBrZXkgKi8KICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAvKiBTZXQgdGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciB3aXRoIHplcm8gdmFsdWVzKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMyA9IDBVOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVlIyID0gMFU7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWUjEgPSAwVTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVZSMCA9IDBVOwoKICAgIC8qIEVuYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLypXcml0ZSB0aGUgQjAgcGFja2V0IGludG8gQ1JZUF9ESU4qLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5CMCk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkIwICsgMyk7CgogICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQUVTX1NSX0NDRikpOwogICAgLyogQ2xlYXIgQ0NGIGZsYWcgKi8KICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAvKiBGcm9tIHRoYXQgcG9pbnQgdGhlIHdob2xlIG1lc3NhZ2UgbXVzdCBiZSBwcm9jZXNzZWQsIGZpcnN0IHRoZSBIZWFkZXIgdGhlbiB0aGUgcGF5bG9hZC4KICAgIEZpcnN0IHRoZSAgSGVhZGVyIGJsb2NrKEIxKSA6IGFzc29jaWF0ZWQgZGF0YSBsZW5ndGggZXhwcmVzc2VkIGluIGJ5dGVzIGNvbmNhdGVuYXRlZCB3aXRoIEFzc29jaWF0ZWQgRGF0YSAoQSkqLwoKICAgIGlmIChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICE9IDBVKQogICAgewogICAgICBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSkgPT0gMFUpCiAgICAgIHsKICAgICAgICAvKiBIZWFkZXJTaXplICU0LCBubyBwYWRkaW5nICovCiAgICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyAobG9vcGNvdW50ZXIgPCBoY3J5cC0+SW5pdC5IZWFkZXJTaXplKTsgbG9vcGNvdW50ZXIgKz0gNFUpCiAgICAgICAgewogICAgICAgICAgLyogV3JpdGUgdGhlIElucHV0IGJsb2NrIGluIHRoZSBEYXRhIElucHV0IHJlZ2lzdGVyICovCiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CgogICAgICAgICAgLyogIHdhaXQgdW50aWwgdGhlIGVuZCBvZiBjb21wdXRhdGlvbiAqLwogICAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgICAgICAgZG8KICAgICAgICAgIHsKICAgICAgICAgICAgY291bnQtLSA7CiAgICAgICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICAgICAgewogICAgICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgICAgfQogICAgICAgICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQUVTX1NSX0NDRikpOwogICAgICAgICAgLyogQ2xlYXIgQ0NGIGZsYWcgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwogICAgICAgIH0KICAgICAgfQogICAgICBlbHNlCiAgICAgIHsKICAgICAgICAvKldyaXRlIEhlYWRlciBibG9jayBpbiB0aGUgSU4gRklGTyB3aXRob3V0IGxhc3QgYmxvY2sgKi8KICAgICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8ICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSkgLSAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSkpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgICB7CiAgICAgICAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIGRhdGEgaW5wdXQgcmVnaXN0ZXIgKi8KICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKCiAgICAgICAgICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01JTklUUEhBU0U7CiAgICAgICAgICBkbwogICAgICAgICAgewogICAgICAgICAgICBjb3VudC0tIDsKICAgICAgICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgICAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgICAgICB9CiAgICAgICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBBRVNfU1JfQ0NGKSk7CiAgICAgICAgICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogICAgICAgICAgX19IQUxfQ1JZUF9DTEVBUl9GTEFHKGhjcnlwLCBDUllQX0NDRl9DTEVBUik7CiAgICAgICAgfQogICAgICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgKGxvb3Bjb3VudGVyIDwgKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSkpOyBsb29wY291bnRlcisrKQogICAgICAgIHsKICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIH0KICAgICAgICB3aGlsZSAobG9vcGNvdW50ZXIgPCA0VSkKICAgICAgICB7CiAgICAgICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9IDB4MFU7CiAgICAgICAgICBsb29wY291bnRlcisrOwogICAgICAgIH0KCiAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgICAgIGRvCiAgICAgICAgewogICAgICAgICAgY291bnQtLSA7CiAgICAgICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgICAgICB7CiAgICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgICB9CiAgICAgICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQUVTX1NSX0NDRikpOwogICAgICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICAgICAgX19IQUxfQ1JZUF9DTEVBUl9GTEFHKGhjcnlwLCBDUllQX0NDRl9DTEVBUik7CiAgICAgIH0KICAgIH0KICB9IC8qIGlmIChEb0tleUlWQ29uZmlnID09IDFVKSAqLwogIC8qIFRoZW4gdGhlIHBheWxvYWQ6IGNsZWFydGV4dCBwYXlsb2FkIChub3QgdGhlIGNpcGhlcnRleHQgcGF5bG9hZCkuCiAgV3JpdGUgaW5wdXQgRGF0YSwgbm8gb3V0cHV0IERhdGEgdG8gZ2V0ICovCiAgaWYgKGhjcnlwLT5TaXplICE9IDBVKQogIHsKICAgIGlmIChoY3J5cC0+U2l6ZSA+PSAxNlUpCiAgICB7CiAgICAgIGlmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKQogICAgICB7CiAgICAgICAgQ1JZUF9TZXRETUFDb25maWcoaGNyeXAsICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwSW5CdWZmUHRyKSwgKGhjcnlwLT5TaXplIC8gNFUpLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICAgICAgfQogICAgICBlbHNlIC8qdG8gY29tcHV0ZSBsYXN0IHdvcmQ8MTI4Yml0cywgb3RoZXJ3aXNlIGl0IHdpbGwgbm90IGJlIGVuY3J5cHRlZC9kZWNyeXB0ZWQgKi8KICAgICAgewogICAgICAgIHdvcmRzaXplID0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSkgKyAoMTZVIC0gKCh1aW50MzJfdCkoaGNyeXAtPlNpemUpICUgMTZVKSkgOwoKICAgICAgICAvKiBTZXQgdGhlIGlucHV0IGFuZCBvdXRwdXQgYWRkcmVzc2VzIGFuZCBzdGFydCBETUEgdHJhbnNmZXIsIHBDcnlwT3V0QnVmZlB0ciBzaXplIHNob3VsZCBiZSAlNCAqLwogICAgICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksICgodWludDE2X3Qpd29yZHNpemUgLyA0VSksCiAgICAgICAgICAgICAgICAgICAgICAgICAgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgICAgIH0KICAgIH0KICAgIGlmICgoaGNyeXAtPlNpemUgPCAxNlUpICE9IDBVKQogICAgewogICAgICAvKiBDb21wdXRlIHRoZSBudW1iZXIgb2YgcGFkZGluZyBieXRlcyBpbiBsYXN0IGJsb2NrIG9mIHBheWxvYWQgKi8KICAgICAgbnBibGIgPSAoKCgodWludDMyX3QpKGhjcnlwLT5TaXplKSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogICAgICAvKiBOdW1iZXIgb2YgdmFsaWQgd29yZHMgKGxhc3R3b3Jkc2l6ZSkgaW4gbGFzdCBibG9jayAqLwogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDBVKQogICAgICB7CiAgICAgICAgbGFzdHdvcmRzaXplID0gKDE2VSAtIG5wYmxiKSAvIDRVOwogICAgICB9CiAgICAgIGVsc2UKICAgICAgewogICAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgoMTZVIC0gbnBibGIpIC8gNFUpICsgMVU7CiAgICAgIH0KICAgICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgbG9vcGNvdW50ZXIgPCBsYXN0d29yZHNpemU7IGxvb3Bjb3VudGVyICsrKQogICAgICB7CiAgICAgICAgLyogV3JpdGUgdGhlIGxhc3QgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9IDBVOwogICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgIH0KICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSU5JVFBIQVNFOwogICAgICBkbwogICAgICB7CiAgICAgICAgY291bnQtLSA7CiAgICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICAgIHsKICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgIH0KICAgICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQUVTX1NSX0NDRikpOwogICAgICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogICAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSBhbmQgcGhhc2UgKi8KICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVuTG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSBhbmQgcGhhc2UgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogIH0KI2VuZGlmIC8qIEFFUyAqLwojaWYgZGVmaW5lZChDUllQKQogICAgLyoqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgogICAgaWYgKENSWVBfR0NNQ0NNX1NldEhlYWRlclBoYXNlX0RNQShoY3J5cCkgIT0gSEFMX09LKQogICAgewogICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgfQoKICAgIC8qKioqKioqKioqKioqKioqKioqKiBQYXlsb2FkIHBoYXNlICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfUEFZTE9BRCk7CgogIH0gLyogaWYgKERvS2V5SVZDb25maWcgPT0gMVUpICovCiAgaWYgKGhjcnlwLT5TaXplICE9IDBVKQogIHsKICAgIC8qIFNpemUgc2hvdWxkIGJlICU0ICBvdGhlcndpc2UgVGFnIHdpbGwgIGJlIGluY29ycmVjdGx5IGdlbmVyYXRlZCBmb3IgR0NNIEVuY3J5cHRpb24gJiBDQ00gRGVjcnlwdGlvbgogICAgV29ya2Fyb3VuZCBpcyBpbXBsZW1lbnRlZCBpbiBwb2xsaW5nIG1vZGUsIHNvIGlmIGxhc3QgYmxvY2sgb2YKICAgIHBheWxvYWQgPDEyOGJpdCBkb24ndCB1c2UgSEFMX0NSWVBfQUVTR0NNX0RNQSBvdGhlcndpc2UgVEFHIGlzIGluY29ycmVjdGx5IGdlbmVyYXRlZCBmb3IgR0NNIEVuY3J5cHRpb24uICovCiAgICAvKiBTZXQgdGhlIGlucHV0IGFuZCBvdXRwdXQgYWRkcmVzc2VzIGFuZCBzdGFydCBETUEgdHJhbnNmZXIgKi8KICAgIGlmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKQogICAgewogICAgICBDUllQX1NldERNQUNvbmZpZyhoY3J5cCwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCBoY3J5cC0+U2l6ZSAvIDRVLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgd29yZHNpemUgPSAodWludDMyX3QpKGhjcnlwLT5TaXplKSArIDE2VSAtICgodWludDMyX3QpKGhjcnlwLT5TaXplKSAlIDE2VSkgOwoKICAgICAgLyogU2V0IHRoZSBpbnB1dCBhbmQgb3V0cHV0IGFkZHJlc3NlcyBhbmQgc3RhcnQgRE1BIHRyYW5zZmVyLCBwQ3J5cE91dEJ1ZmZQdHIgc2l6ZSBzaG91bGQgYmUgJTQqLwogICAgICBDUllQX1NldERNQUNvbmZpZyhoY3J5cCwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCAodWludDE2X3Qpd29yZHNpemUgLyA0VSwKICAgICAgICAgICAgICAgICAgICAgICAgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgICB9CiAgfQogIGVsc2UgLypTaXplID0gMCovCiAgewogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgYW5kIHBoYXNlICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICB9CiNlbmRpZiAvKiBDUllQICovCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBTZXRzIHRoZSBwYXlsb2FkIHBoYXNlIGluIGl0ZXJydXB0IG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIHN0YXRlCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9HQ01DQ01fU2V0UGF5bG9hZFBoYXNlX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICB1aW50MzJfdCBsb29wY291bnRlcjsKICB1aW50MzJfdCB0ZW1wWzRdOyAgLyogVGVtcG9yYXJ5IENyeXBPdXRCdWZmICovCiAgdWludDMyX3QgbGFzdHdvcmRzaXplOwogIHVpbnQzMl90IG5wYmxiOwogIHVpbnQzMl90IGk7CiNpZiBkZWZpbmVkKEFFUykKICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KI2VuZGlmIC8qIEFFUyAqLwoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogUGF5bG9hZCBwaGFzZSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKI2lmIGRlZmluZWQoQ1JZUCkKICBpZiAoaGNyeXAtPlNpemUgPT0gMFUpCiAgewogICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JIHwgQ1JZUF9JVF9PVVRJKTsKCiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgfQoKICBlbHNlIGlmICgoKGhjcnlwLT5TaXplIC8gNFUpIC0gKGhjcnlwLT5DcnlwSW5Db3VudCkpID49IDRVKQogIHsKICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICBpZiAoKChoY3J5cC0+U2l6ZSAvIDRVKSA9PSBoY3J5cC0+Q3J5cEluQ291bnQpICYmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKSkKICAgIHsKICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwoKICAgICAgLyogQ2FsbCB0aGUgaW5wdXQgZGF0YSB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMVUpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQogICAgaWYgKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoaGNyeXAtPlNpemUgLyA0VSkpCiAgICB7CiAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBPdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgICovCiAgICAgIGZvciAoaSA9IDBVOyBpIDwgNFU7IGkrKykKICAgICAgewogICAgICAgIHRlbXBbaV0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgIH0KICAgICAgaSA9IDBVOwogICAgICB3aGlsZSAoKChoY3J5cC0+Q3J5cE91dENvdW50IDwgKChoY3J5cC0+U2l6ZSkgLyA0VSkpKSAmJiAoaSA8IDRVKSkKICAgICAgewogICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgICAgaSsrOwogICAgICB9CiAgICAgIGlmICgoKGhjcnlwLT5TaXplIC8gNFUpID09IGhjcnlwLT5DcnlwT3V0Q291bnQpICYmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIERpc2FibGUgQ1JZUCAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgICAgICAvKiBDYWxsIG91dHB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KICB9CiAgZWxzZSBpZiAoKGhjcnlwLT5TaXplICUgMTZVKSAhPSAwVSkKICB7CiAgICAvKiBTaXplIHNob3VsZCBiZSAlNCBpbiB3b3JkIGFuZCAlMTYgaW4gYnl0ZSAgb3RoZXJ3aXNlIFRBRyB3aWxsICBiZSBpbmNvcnJlY3RseSBnZW5lcmF0ZWQgZm9yIEdDTSBFbmNyeXB0aW9uICYgQ0NNIERlY3J5cHRpb24KICAgIFdvcmthcm91bmQgaXMgaW1wbGVtZW50ZWQgaW4gcG9sbGluZyBtb2RlLCBzbyBpZiBsYXN0IGJsb2NrIG9mCiAgICBwYXlsb2FkIDwxMjhiaXQgZG9uJ3QgdXNlIENSWVBfQUVTR0NNX0VuY3J5cHRfSVQgb3RoZXJ3aXNlIFRBRyBpcyBpbmNvcnJlY3RseSBnZW5lcmF0ZWQuICovCgogICAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgICBucGJsYiA9ICgoKCh1aW50MzJfdCkoaGNyeXAtPlNpemUpIC8gMTZVKSArIDFVKSAqIDE2VSkgLSAodWludDMyX3QpKGhjcnlwLT5TaXplKTsKCiAgICAvKiBOdW1iZXIgb2YgdmFsaWQgd29yZHMgKGxhc3R3b3Jkc2l6ZSkgaW4gbGFzdCBibG9jayAqLwogICAgaWYgKChucGJsYiAlIDRVKSA9PSAwVSkKICAgIHsKICAgICAgbGFzdHdvcmRzaXplID0gKDE2VSAtIG5wYmxiKSAvIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBsYXN0d29yZHNpemUgPSAoKDE2VSAtIG5wYmxiKSAvIDRVKSArIDFVOwogICAgfQoKICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgZm9yIChsb29wY291bnRlciA9IDBVOyBsb29wY291bnRlciA8IGxhc3R3b3Jkc2l6ZTsgbG9vcGNvdW50ZXIrKykKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgfQogICAgd2hpbGUgKGxvb3Bjb3VudGVyIDwgNFUpCiAgICB7CiAgICAgIC8qIFBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9IDB4MFU7CiAgICAgIGxvb3Bjb3VudGVyKys7CiAgICB9CiAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JKTsKCiAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSkgIT0gMHgwVSkKICAgIHsKICAgICAgZm9yIChpID0gMFU7IGkgPCA0VTsgaSsrKQogICAgICB7CiAgICAgICAgdGVtcFtpXSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICAgfQogICAgICBpZiAoKChoY3J5cC0+U2l6ZSkgLyA0VSkgPT0gMFUpCiAgICAgIHsKICAgICAgICBmb3IgKGkgPSAwVTsgaSA8ICgodWludDMyX3QpKGhjcnlwLT5TaXplKSAlIDRVKTsgaSsrKQogICAgICAgIHsKICAgICAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgICB9CiAgICAgIH0KICAgICAgaSA9IDB4MFU7CiAgICAgIHdoaWxlICgoKGhjcnlwLT5DcnlwT3V0Q291bnQgPCAoKGhjcnlwLT5TaXplKSAvIDRVKSkpICYmIChpIDwgNFUpKQogICAgICB7CiAgICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wW2ldOwogICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgICBpKys7CiAgICAgIH0KICAgIH0KICAgIGlmIChoY3J5cC0+Q3J5cE91dENvdW50ID49IChoY3J5cC0+U2l6ZSAvIDRVKSkKICAgIHsKICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJIHwgQ1JZUF9JVF9JTkkpOwoKICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgICAvKiBDYWxsIG91dHB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgT3V0cHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogTm90aGluZyB0byBkbyAqLwogIH0KI2Vsc2UgLyogQUVTICovCgogIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IGJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIqLwogIGZvciAoaSA9IDBVOyBpIDwgNFU7IGkrKykKICB7CiAgICB0ZW1wW2ldID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUUjsKICB9CiAgaSA9IDBVOwogIHdoaWxlICgoaGNyeXAtPkNyeXBPdXRDb3VudCA8ICgoaGNyeXAtPlNpemUgKyAzVSkgLyA0VSkpICYmIChpIDwgNFUpKQogIHsKICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcFtpXTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgIGkrKzsKICB9CiAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogIGlmICgoaGNyeXAtPkNyeXBPdXRDb3VudCA+PSAoaGNyeXAtPlNpemUgLyA0VSkpICYmICgob3V0Y291bnQgKiA0VSkgPj0gIGhjcnlwLT5TaXplKSkKICB7CiAgICAvKiBEaXNhYmxlIGNvbXB1dGF0aW9uIGNvbXBsZXRlIGZsYWcgYW5kIGVycm9ycyBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfQ0NGSUUgfCBDUllQX0lUX0VSUklFKTsKCiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQ2FsbCBvdXRwdXQgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgT3V0cHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQoKICBlbHNlIGlmICgoKGhjcnlwLT5TaXplIC8gNFUpIC0gKGhjcnlwLT5DcnlwSW5Db3VudCkpID49IDRVKQogIHsKICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgaWYgKChoY3J5cC0+Q3J5cEluQ291bnQgPT0gIGhjcnlwLT5TaXplKSAmJiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0dDTV9HTUFDKSkKICAgIHsKICAgICAgLyogQ2FsbCBJbnB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICB9CiAgfQogIGVsc2UgLyogTGFzdCBibG9jayBvZiBwYXlsb2FkIDwgMTI4Yml0Ki8KICB7CiAgICAvKiBXb3JrYXJvdW5kIG5vdCBpbXBsZW1lbnRlZCwgU2l6ZSBzaG91bGQgYmUgJTQgIG90aGVyd2lzZSBUYWcgd2lsbCAgYmUgaW5jb3JyZWN0bHkKICAgIGdlbmVyYXRlZCBmb3IgR0NNIEVuY3J5cHRpb24gJiBDQ00gRGVjcnlwdGlvbi4gV29ya2Fyb3VuZCBpcyBpbXBsZW1lbnRlZCBpbiBwb2xsaW5nIG1vZGUsIHNvIGlmIGxhc3QgYmxvY2sgb2YKICAgIHBheWxvYWQgPDEyOGJpdCBkb24ndCB1c2UgQ1JZUF9FbmNyeXB0X0lUIG90aGVyd2lzZSBUQUcgaXMgaW5jb3JyZWN0bHkgZ2VuZXJhdGVkIGZvciBHQ00gRW5jcnlwdGlvbiAmIENDTSBEZWNyeXB0aW9uLiAqLwoKICAgIC8qIENvbXB1dGUgdGhlIG51bWJlciBvZiBwYWRkaW5nIGJ5dGVzIGluIGxhc3QgYmxvY2sgb2YgcGF5bG9hZCAqLwogICAgbnBibGIgPSAoKCgodWludDMyX3QpKGhjcnlwLT5TaXplKSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogICAgLyogTnVtYmVyIG9mIHZhbGlkIHdvcmRzIChsYXN0d29yZHNpemUpIGluIGxhc3QgYmxvY2sgKi8KICAgIGlmICgobnBibGIgJSA0VSkgPT0gMFUpCiAgICB7CiAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgxNlUgLSBucGJsYikgLyA0VTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgbGFzdHdvcmRzaXplID0gKCgxNlUgLSBucGJsYikgLyA0VSkgKyAxVTsKICAgIH0KCiAgICAvKiAgTGFzdCBibG9jayBvcHRpb25hbGx5IHBhZCB0aGUgZGF0YSB3aXRoIHplcm9zKi8KICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgbG9vcGNvdW50ZXIgPCBsYXN0d29yZHNpemU7IGxvb3Bjb3VudGVyKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICB9CiAgICB3aGlsZSAobG9vcGNvdW50ZXIgPCA0VSkKICAgIHsKICAgICAgLyogcGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9IDB4MFU7CiAgICAgIGxvb3Bjb3VudGVyKys7CiAgICB9CiAgfQojZW5kaWYgLyogQUVTICovCgp9CgoKLyoqCiAgKiBAYnJpZWYgIFNldHMgdGhlIGhlYWRlciBwaGFzZSBpbiBwb2xsaW5nIG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlKEhlYWRlciAmIEhlYWRlclNpemUpCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgc3RhdGUKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9HQ01DQ01fU2V0SGVhZGVyUGhhc2UoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IGxvb3Bjb3VudGVyOwoKICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogSGVhZGVyIHBoYXNlIGZvciBHQ00vR01BQyBvciBDQ00gKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKICBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgIT0gMFUpKQogIHsKCiNpZiBkZWZpbmVkKENSWVApCgogICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfSEVBREVSKTsKCiAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIGlmICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSA9PSAwVSkKICAgIHsKICAgICAgLyogSGVhZGVyU2l6ZSAlNCwgbm8gcGFkZGluZyAqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IGhjcnlwLT5Jbml0LkhlYWRlclNpemUpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+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+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAmIHJldHVybiBlcnJvciAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgfQoKI2Vsc2UgLyogQUVTICovCgogICAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19HQ01fR01BQykKICAgIHsKICAgICAgLyogV29ya2Fyb3VuZCAxIDpvbmx5IEFFUyBiZWZvcmUgcmUtZW5hYmxpbmcgdGhlIElQLCBkYXRhdHlwZSBjYW4gYmUgY29uZmlndXJlZC4qLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9EQVRBVFlQRSwgaGNyeXAtPkluaXQuRGF0YVR5cGUpOwoKICAgICAgLyogU2VsZWN0IGhlYWRlciBwaGFzZSAqLwogICAgICBDUllQX1NFVF9QSEFTRShoY3J5cCwgQ1JZUF9QSEFTRV9IRUFERVIpOwoKICAgICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIH0KICAgIGlmICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSA9PSAwVSkKICAgIHsKICAgICAgLyogSGVhZGVyU2l6ZSAlNCwgbm8gcGFkZGluZyAqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IGhjcnlwLT5Jbml0LkhlYWRlclNpemUpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgewogICAgICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgZGF0YSBpbnB1dCByZWdpc3RlciAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CgogICAgICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICAgIHsKICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgIH0KICAgICAgICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogICAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwogICAgICB9CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qV3JpdGUgaGVhZGVyIGJsb2NrIGluIHRoZSBJTiBGSUZPIHdpdGhvdXQgbGFzdCBibG9jayAqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8ICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSkgLSAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSkpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgewogICAgICAgIC8qIFdyaXRlIHRoZSBpbnB1dCBibG9jayBpbiB0aGUgZGF0YSBpbnB1dCByZWdpc3RlciAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CgogICAgICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICAgIHsKICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgIH0KICAgICAgICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogICAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwogICAgICB9CiAgICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpKTsgbG9vcGNvdW50ZXIrKykKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgfQogICAgICB3aGlsZSAobG9vcGNvdW50ZXIgPCA0VSkKICAgICAgewogICAgICAgIC8qUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gMHgwVTsKICAgICAgICBsb29wY291bnRlcisrOwogICAgICB9CgogICAgICBpZiAoQ1JZUF9XYWl0T25DQ0ZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogICAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKICAgIH0KI2VuZGlmIC8qIEVuZCBBRVMgb3IgQ1JZUCAqLwogIH0KICBlbHNlCiAgewojaWYgZGVmaW5lZChBRVMpCiAgICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0dDTV9HTUFDKQogICAgewogICAgICAvKldvcmthcm91bmQgMTogb25seSBBRVMsIGJlZm9yZSByZS1lbmFibGluZyB0aGUgSVAsIGRhdGF0eXBlIGNhbiBiZSBjb25maWd1cmVkLiovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RBVEFUWVBFLCBoY3J5cC0+SW5pdC5EYXRhVHlwZSk7CgogICAgICAvKiBTZWxlY3QgaGVhZGVyIHBoYXNlICovCiAgICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0hFQURFUik7CgogICAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgICB9CiNlbmRpZiAvKiBBRVMgKi8KICB9CiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBTZXRzIHRoZSBoZWFkZXIgcGhhc2Ugd2hlbiB1c2luZyBETUEgaW4gcHJvY2VzcwogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUoSGVhZGVyICYgSGVhZGVyU2l6ZSkKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZV9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgID0gMFU7CiAgdWludDMyX3QgbG9vcGNvdW50ZXI7CgogIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgZm9yIEdDTS9HTUFDIG9yIENDTSAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCiAgaWYgKChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICE9IDBVKSkKICB7CgojaWYgZGVmaW5lZChDUllQKQoKICAgIC8qIFNlbGVjdCBoZWFkZXIgcGhhc2UgKi8KICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0hFQURFUik7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSkgPT0gMFUpCiAgICB7CiAgICAgIC8qIEhlYWRlclNpemUgJTQsIG5vIHBhZGRpbmcgKi8KICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyAobG9vcGNvdW50ZXIgPCBoY3J5cC0+SW5pdC5IZWFkZXJTaXplKTsgbG9vcGNvdW50ZXIgKz0gNFUpCiAgICAgIHsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKCiAgICAgICAgLyogV2FpdCBmb3IgSUZFTSB0byBiZSByYWlzZWQgKi8KICAgICAgICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01IRUFERVJQSEFTRTsKICAgICAgICBkbwogICAgICAgIHsKICAgICAgICAgIGNvdW50LS0gOwogICAgICAgICAgaWYgKGNvdW50ID09IDBVKQogICAgICAgICAgewogICAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICAgICAgfQogICAgICAgIH0gd2hpbGUgKEhBTF9JU19CSVRfQ0xSKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19JRkVNKSk7CiAgICAgIH0KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgLypXcml0ZSBoZWFkZXIgYmxvY2sgaW4gdGhlIElOIEZJRk8gd2l0aG91dCBsYXN0IGJsb2NrICovCiAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgKGxvb3Bjb3VudGVyIDwgKChoY3J5cC0+SW5pdC5IZWFkZXJTaXplKSAtIChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpKSk7IGxvb3Bjb3VudGVyICs9IDRVKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CgogICAgICAgIC8qIFdhaXQgZm9yIElGRU0gdG8gYmUgcmFpc2VkICovCiAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICBjb3VudC0tIDsKICAgICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgIH0KICAgICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfSUZFTSkpOwogICAgICB9CiAgICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8IChoY3J5cC0+SW5pdC5IZWFkZXJTaXplICUgNFUpKTsgbG9vcGNvdW50ZXIrKykKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAweDBVOwogICAgICAgIGxvb3Bjb3VudGVyKys7CiAgICAgIH0KICAgICAgLyogV2FpdCBmb3IgSUZFTSB0byBiZSByYWlzZWQgKi8KICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICAgIGRvCiAgICAgIHsKICAgICAgICBjb3VudC0tIDsKICAgICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgICAgewogICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgfQogICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfSUZFTSkpOwogICAgfQogICAgLyogV2FpdCB1bnRpbCB0aGUgY29tcGxldGUgbWVzc2FnZSBoYXMgYmVlbiBwcm9jZXNzZWQgKi8KICAgIGNvdW50ID0gQ1JZUF9USU1FT1VUX0dDTUNDTUhFQURFUlBIQVNFOwogICAgZG8KICAgIHsKICAgICAgY291bnQtLSA7CiAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9TRVQoaGNyeXAtPkluc3RhbmNlLT5TUiwgQ1JZUF9GTEFHX0JVU1kpKTsKCiNlbHNlIC8qIEFFUyAqLwoKICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfR0NNX0dNQUMpCiAgICB7CiAgICAgIC8qIFdvcmthcm91bmQgMTogb25seSBBRVMsIGJlZm9yZSByZS1lbmFibGluZyB0aGUgSVAsIGRhdGF0eXBlIGNhbiBiZSBjb25maWd1cmVkLiovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RBVEFUWVBFLCBoY3J5cC0+SW5pdC5EYXRhVHlwZSk7CgogICAgICAvKiBTZWxlY3QgaGVhZGVyIHBoYXNlICovCiAgICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0hFQURFUik7CgogICAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgICB9CiAgICBpZiAoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSkgPT0gMFUpCiAgICB7CiAgICAgIC8qIEhlYWRlclNpemUgJTQsIG5vIHBhZGRpbmcgKi8KICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyAobG9vcGNvdW50ZXIgPCBoY3J5cC0+SW5pdC5IZWFkZXJTaXplKTsgbG9vcGNvdW50ZXIgKz0gNFUpCiAgICAgIHsKICAgICAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIGRhdGEgaW5wdXQgcmVnaXN0ZXIgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwoKICAgICAgICAvKldhaXQgb24gQ0NGIGZsYWcqLwogICAgICAgIGNvdW50ID0gQ1JZUF9USU1FT1VUX0dDTUNDTUhFQURFUlBIQVNFOwogICAgICAgIGRvCiAgICAgICAgewogICAgICAgICAgY291bnQtLSA7CiAgICAgICAgICBpZiAoY291bnQgPT0gMFUpCiAgICAgICAgICB7CiAgICAgICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICAgICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgICAgICB9CiAgICAgICAgfSB3aGlsZSAoSEFMX0lTX0JJVF9DTFIoaGNyeXAtPkluc3RhbmNlLT5TUiwgQUVTX1NSX0NDRikpOwoKICAgICAgICAvKiBDbGVhciBDQ0YgZmxhZyAqLwogICAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwogICAgICB9CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qV3JpdGUgaGVhZGVyIGJsb2NrIGluIHRoZSBJTiBGSUZPIHdpdGhvdXQgbGFzdCBibG9jayAqLwogICAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IChsb29wY291bnRlciA8ICgoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSkgLSAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSAlIDRVKSkpOyBsb29wY291bnRlciArPSA0VSkKICAgICAgewogICAgICAgIC8qIFdyaXRlIHRoZSBJbnB1dCBibG9jayBpbiB0aGUgRGF0YSBJbnB1dCByZWdpc3RlciAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CgogICAgICAgIC8qV2FpdCBvbiBDQ0YgZmxhZyovCiAgICAgICAgY291bnQgPSBDUllQX1RJTUVPVVRfR0NNQ0NNSEVBREVSUEhBU0U7CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICBjb3VudC0tIDsKICAgICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICAgIHsKICAgICAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICAgIH0KICAgICAgICB9IHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBBRVNfU1JfQ0NGKSk7CgogICAgICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICAgICAgX19IQUxfQ1JZUF9DTEVBUl9GTEFHKGhjcnlwLCBDUllQX0NDRl9DTEVBUik7CiAgICAgIH0KICAgICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICAgIGZvciAobG9vcGNvdW50ZXIgPSAwVTsgKGxvb3Bjb3VudGVyIDwgKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSkpOyBsb29wY291bnRlcisrKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogICAgICB9CiAgICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgICB7CiAgICAgICAgLyogUGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gMHgwVTsKICAgICAgICBsb29wY291bnRlcisrOwogICAgICB9CgogICAgICAvKldhaXQgb24gQ0NGIGZsYWcqLwogICAgICBjb3VudCA9IENSWVBfVElNRU9VVF9HQ01DQ01IRUFERVJQSEFTRTsKICAgICAgZG8KICAgICAgewogICAgICAgIGNvdW50LS0gOwogICAgICAgIGlmIChjb3VudCA9PSAwVSkKICAgICAgICB7CiAgICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgICB9CiAgICAgIH0gd2hpbGUgKEhBTF9JU19CSVRfQ0xSKGhjcnlwLT5JbnN0YW5jZS0+U1IsIEFFU19TUl9DQ0YpKTsKCiAgICAgIC8qIENsZWFyIENDRiBmbGFnICovCiAgICAgIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwogICAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICAqLwogIH0KICBlbHNlCiAgewojaWYgZGVmaW5lZChBRVMpCiAgICBpZiAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0dDTV9HTUFDKQogICAgewogICAgICAvKldvcmthcm91bmQgMTogb25seSBBRVMsIGJlZm9yZSByZS1lbmFibGluZyB0aGUgSVAsIGRhdGF0eXBlIGNhbiBiZSBjb25maWd1cmVkLiovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0RBVEFUWVBFLCBoY3J5cC0+SW5pdC5EYXRhVHlwZSk7CgogICAgICAvKiBTZWxlY3QgaGVhZGVyIHBoYXNlICovCiAgICAgIENSWVBfU0VUX1BIQVNFKGhjcnlwLCBDUllQX1BIQVNFX0hFQURFUik7CgogICAgICAvKiBFbmFibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgICB9CiNlbmRpZiAvKiBBRVMgKi8KICB9CiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBTZXRzIHRoZSBoZWFkZXIgcGhhc2UgaW4gaW50ZXJydXB0IG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlKEhlYWRlciAmIEhlYWRlclNpemUpCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpzdGF0aWMgdm9pZCBDUllQX0dDTUNDTV9TZXRIZWFkZXJQaGFzZV9JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgdWludDMyX3QgbG9vcGNvdW50ZXI7CiNpZiBkZWZpbmVkKEFFUykKICB1aW50MzJfdCBsYXN0d29yZHNpemU7CiAgdWludDMyX3QgbnBibGI7CiNlbmRpZgogIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKiBIZWFkZXIgcGhhc2UgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwoKI2lmIGRlZmluZWQoQ1JZUCkKICBpZiAoaGNyeXAtPkluaXQuSGVhZGVyU2l6ZSA9PSAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCkKICB7CiAgICAvKiBEaXNhYmxlIGludGVycnVwdHMgKi8KICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwoKICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCAqLwogICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgQ1JZUF9TRVRfUEhBU0UoaGNyeXAsIENSWVBfUEhBU0VfUEFZTE9BRCk7CgogICAgLyogRW5hYmxlIEludGVycnVwdHMgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgLyogRW5hYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgKi8KICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICB9CiAgZWxzZSBpZiAoKChoY3J5cC0+SW5pdC5IZWFkZXJTaXplKSAtIChoY3J5cC0+Q3J5cEhlYWRlckNvdW50KSkgPj0gNFUpCgogIHsKICAgIC8qIEhlYWRlclNpemUgJTQsIG5vIHBhZGRpbmcgKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgIDsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KysgOwogIH0KICBlbHNlCiAgewogICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IGxvb3Bjb3VudGVyIDwgKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSk7IGxvb3Bjb3VudGVyKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrIDsKICAgIH0KICAgIHdoaWxlIChsb29wY291bnRlciA8IDRVKQogICAgewogICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSAweDBVOwogICAgICBsb29wY291bnRlcisrOwogICAgfQogIH0KI2Vsc2UgLyogQUVTICovCgogIGlmIChoY3J5cC0+SW5pdC5IZWFkZXJTaXplID09ICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KQogIHsKICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAvKiAgUGF5bG9hZCBwaGFzZSBub3Qgc3VwcG9ydGVkIGluIENDTSBBRVMyICAqLwogICAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19HQ01fR01BQykKICAgIHsKICAgICAgLyogU2VsZWN0IHBheWxvYWQgcGhhc2Ugb25jZSB0aGUgaGVhZGVyIHBoYXNlIGlzIHBlcmZvcm1lZCAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9HQ01QSCwgQ1JZUF9QSEFTRV9QQVlMT0FEKTsKICAgIH0KICAgIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfQ0NNKQogICAgewogICAgICAvKiBJbmNyZW1lbnQgQ3J5cEhlYWRlckNvdW50IHRvIHBhc3MgaW4gQ1JZUF9HQ01DQ01fU2V0UGF5bG9hZFBoYXNlX0lUICovCiAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKzsKICAgIH0KICAgIC8qIFdyaXRlIHRoZSBwYXlsb2FkIElucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICBpZiAoaGNyeXAtPlNpemUgPT0gMFUpCiAgICB7CiAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICBfX0hBTF9DUllQX0RJU0FCTEVfSVQoaGNyeXAsIENSWVBfSVRfQ0NGSUUgfCBDUllQX0lUX0VSUklFKTsKCiAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgIH0KICAgIGVsc2UgaWYgKGhjcnlwLT5TaXplID49IDE2VSkKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CgogICAgICBpZiAoKGhjcnlwLT5DcnlwSW5Db3VudCA9PSAoaGNyeXAtPlNpemUgLyA0VSkpICYmICgoaGNyeXAtPlNpemUgJSAxNlUpID09IDBVKSkKICAgICAgewogICAgICAgIC8qIENhbGwgdGhlIGlucHV0IGRhdGEgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sgKi8KI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgSW5wdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KICAgIGVsc2UgLyogU2l6ZSA8IDQgd29yZHMgIDogZmlyc3QgYmxvY2sgaXMgdGhlIGxhc3QgYmxvY2sqLwogICAgewogICAgICAvKiBXb3JrYXJvdW5kIG5vdCBpbXBsZW1lbnRlZCwgU2l6ZSBzaG91bGQgYmUgJTQgIG90aGVyd2lzZSBUYWcgd2lsbCAgYmUgaW5jb3JyZWN0bHkKICAgICAgZ2VuZXJhdGVkIGZvciBHQ00gRW5jcnlwdGlvbi4gV29ya2Fyb3VuZCBpcyBpbXBsZW1lbnRlZCBpbiBwb2xsaW5nIG1vZGUsIHNvIGlmIGxhc3QgYmxvY2sgb2YKICAgICAgcGF5bG9hZCA8MTI4Yml0IGRvbid0IHVzZSBDUllQX0VuY3J5cHRfSVQgb3RoZXJ3aXNlIFRBRyBpcyBpbmNvcnJlY3RseSBnZW5lcmF0ZWQgZm9yIEdDTSBFbmNyeXB0aW9uLiAqLwoKICAgICAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgICAgIG5wYmxiID0gKCgoKHVpbnQzMl90KShoY3J5cC0+U2l6ZSkgLyAxNlUpICsgMVUpICogMTZVKSAtICh1aW50MzJfdCkoaGNyeXAtPlNpemUpOwoKICAgICAgLyogTnVtYmVyIG9mIHZhbGlkIHdvcmRzIChsYXN0d29yZHNpemUpIGluIGxhc3QgYmxvY2sgKi8KICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAwVSkKICAgICAgewogICAgICAgIGxhc3R3b3Jkc2l6ZSA9ICgxNlUgLSBucGJsYikgLyA0VTsKICAgICAgfQogICAgICBlbHNlCiAgICAgIHsKICAgICAgICBsYXN0d29yZHNpemUgPSAoKDE2VSAtIG5wYmxiKSAvIDRVKSArIDFVOwogICAgICB9CgogICAgICAvKiAgTGFzdCBibG9jayBvcHRpb25hbGx5IHBhZCB0aGUgZGF0YSB3aXRoIHplcm9zKi8KICAgICAgZm9yIChsb29wY291bnRlciA9IDBVOyBsb29wY291bnRlciA8IGxhc3R3b3Jkc2l6ZTsgbG9vcGNvdW50ZXIrKykKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50KTsKICAgICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgfQogICAgICB3aGlsZSAobG9vcGNvdW50ZXIgPCA0VSkKICAgICAgewogICAgICAgIC8qIFBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9IDB4MFU7CiAgICAgICAgbG9vcGNvdW50ZXIrKzsKICAgICAgfQogICAgfQogIH0KICBlbHNlIGlmICgoKGhjcnlwLT5Jbml0LkhlYWRlclNpemUpIC0gKGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpKSA+PSA0VSkKICB7CiAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrOwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICBoY3J5cC0+Q3J5cEhlYWRlckNvdW50Kys7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+SW5pdC5IZWFkZXIgKyBoY3J5cC0+Q3J5cEhlYWRlckNvdW50KTsKICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5Jbml0LkhlYWRlciArIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQpOwogICAgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCsrOwogIH0KICBlbHNlIC8qSGVhZGVyU2l6ZSA8IDQgb3IgSGVhZGVyU2l6ZSA+NCAmIEhlYWRlclNpemUgJTQgIT0gMCovCiAgewogICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICBmb3IgKGxvb3Bjb3VudGVyID0gMFU7IGxvb3Bjb3VudGVyIDwgKGhjcnlwLT5Jbml0LkhlYWRlclNpemUgJSA0VSk7IGxvb3Bjb3VudGVyKyspCiAgICB7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9ICoodWludDMyX3QgKikoaGNyeXAtPkluaXQuSGVhZGVyICsgaGNyeXAtPkNyeXBIZWFkZXJDb3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSGVhZGVyQ291bnQrKyA7CiAgICB9CiAgICB3aGlsZSAobG9vcGNvdW50ZXIgPCA0VSkKICAgIHsKICAgICAgLyogcGFkIHRoZSBkYXRhIHdpdGggemVyb3MgdG8gaGF2ZSBhIGNvbXBsZXRlIGJsb2NrICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOUiA9IDB4MFU7CiAgICAgIGxvb3Bjb3VudGVyKys7CiAgICB9CiAgfQojZW5kaWYgLyogRW5kIEFFUyBvciBDUllQICovCn0KCgovKioKICAqIEBicmllZiAgV29ya2Fyb3VuZCB1c2VkIGZvciBHQ00vQ0NNIG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBUaW1lb3V0OiBzcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfV29ya2Fyb3VuZChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgbGFzdHdvcmRzaXplOwogIHVpbnQzMl90IG5wYmxiOwojaWYgZGVmaW5lZChDUllQKQogIHVpbnQzMl90ICBpdjF0ZW1wOwogIHVpbnQzMl90ICB0ZW1wWzRdID0gezB9OwogIHVpbnQzMl90ICB0ZW1wMls0XSA9IHswfTsKI2VuZGlmIC8qIENSWVAgKi8KICB1aW50MzJfdCBpbnRlcm1lZGlhdGVfZGF0YVs0XSA9IHswfTsKICB1aW50MzJfdCBpbmRleDsKCiAgLyogQ29tcHV0ZSB0aGUgbnVtYmVyIG9mIHBhZGRpbmcgYnl0ZXMgaW4gbGFzdCBibG9jayBvZiBwYXlsb2FkICovCiAgbnBibGIgPSAoKCgodWludDMyX3QpKGhjcnlwLT5TaXplKSAvIDE2VSkgKyAxVSkgKiAxNlUpIC0gKHVpbnQzMl90KShoY3J5cC0+U2l6ZSk7CgogIC8qIE51bWJlciBvZiB2YWxpZCB3b3JkcyAobGFzdHdvcmRzaXplKSBpbiBsYXN0IGJsb2NrICovCiAgaWYgKChucGJsYiAlIDRVKSA9PSAwVSkKICB7CiAgICBsYXN0d29yZHNpemUgPSAoMTZVIC0gbnBibGIpIC8gNFU7CiAgfQogIGVsc2UKICB7CiAgICBsYXN0d29yZHNpemUgPSAoKDE2VSAtIG5wYmxiKSAvIDRVKSArIDFVOwogIH0KCiNpZiBkZWZpbmVkKENSWVApCgogIC8qIFdvcmthcm91bmQgMiwgY2FzZSBHQ00gZW5jcnlwdGlvbiAqLwogIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfR0NNKQogIHsKICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpCiAgICB7CiAgICAgIC8qV29ya2Fyb3VuZCBpbiBvcmRlciB0byBwcm9wZXJseSBjb21wdXRlIGF1dGhlbnRpY2F0aW9uIHRhZ3Mgd2hpbGUgZG9pbmcKICAgICAgIGEgR0NNIGVuY3J5cHRpb24gd2l0aCB0aGUgbGFzdCBibG9jayBvZiBwYXlsb2FkIHNpemUgaW5mZXJpb3IgdG8gMTI4IGJpdHMqLwogICAgICAvKiBEaXNhYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKlVwZGF0ZSBDUllQX0lWMVIgcmVnaXN0ZXIgYW5kIEFMR09NT0RFKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICgoaGNyeXAtPkluc3RhbmNlLT5DU0dDTUNDTTdSKSAtIDFVKTsKICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0FFU19DVFIpOwoKICAgICAgLyogRW5hYmxlIENSWVAgdG8gc3RhcnQgdGhlIGZpbmFsIHBoYXNlICovCiAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKICAgIH0KICAgIC8qICBMYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogICAgZm9yIChpbmRleCA9IDA7IGluZGV4IDwgbGFzdHdvcmRzaXplOyBpbmRleCArKykKICAgIHsKICAgICAgLyogV3JpdGUgdGhlIGxhc3QgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIH0KICAgIHdoaWxlIChpbmRleCA8IDRVKQogICAgewogICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gMFU7CiAgICAgIGluZGV4Kys7CiAgICB9CiAgICAvKiBXYWl0IGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICBpZiAoQ1JZUF9XYWl0T25PRk5FRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUpICE9IDB4MFUpCiAgICB7CiAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCA0VTsgaW5kZXgrKykKICAgICAgewogICAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBvdXRwdXQgRklGTyAqLwogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKCiAgICAgICAgLyogSW50ZXJtZWRpYXRlIGRhdGEgYnVmZmVyIHRvIGJlIHVzZWQgaW4gZm9yIHRoZSB3b3JrYXJvdW5kKi8KICAgICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gaW50ZXJtZWRpYXRlX2RhdGFbaW5kZXhdOwogICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgfQogICAgfQoKICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpCiAgICB7CiAgICAgIC8qd29ya2Fyb3VuZCBpbiBvcmRlciB0byBwcm9wZXJseSBjb21wdXRlIGF1dGhlbnRpY2F0aW9uIHRhZ3Mgd2hpbGUgZG9pbmcKICAgICAgYSBHQ00gZW5jcnlwdGlvbiB3aXRoIHRoZSBsYXN0IGJsb2NrIG9mIHBheWxvYWQgc2l6ZSBpbmZlcmlvciB0byAxMjggYml0cyovCiAgICAgIC8qIENoYW5nZSB0aGUgQUVTIG1vZGUgdG8gR0NNIG1vZGUgYW5kIFNlbGVjdCBGaW5hbCBwaGFzZSAqLwogICAgICAvKiBjb25maWd1cmVkICBDSE1PRCBHQ00gICAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIENSWVBfQUVTX0dDTSk7CgogICAgICAvKiBjb25maWd1cmVkICBmaW5hbCBwaGFzZSAgKi8KICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0dDTV9DQ01QSCwgQ1JZUF9QSEFTRV9GSU5BTCk7CgogICAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0RBVEFUWVBFKSA9PSBDUllQX0RBVEFUWVBFXzMyQikKICAgICAgewogICAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gMVUpCiAgICAgICAgewogICAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IDB4RkZGRkZGMDBVOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICUgNFUpID09IDJVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSAweEZGRkYwMDAwVTsKICAgICAgICB9CiAgICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAzVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gMHhGRjAwMDAwMFU7CiAgICAgICAgfQogICAgICB9CiAgICAgIGVsc2UgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9EQVRBVFlQRSkgPT0gQ1JZUF9EQVRBVFlQRV84QikKICAgICAgewogICAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gMVUpCiAgICAgICAgewogICAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUkVWKDB4RkZGRkZGMDBVKTsKICAgICAgICB9CiAgICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAyVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gX19SRVYoMHhGRkZGMDAwMFUpOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICUgNFUpID09IDNVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSBfX1JFVigweEZGMDAwMDAwVSk7CiAgICAgICAgfQogICAgICB9CiAgICAgIGVsc2UgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9EQVRBVFlQRSkgPT0gQ1JZUF9EQVRBVFlQRV8xNkIpCiAgICAgIHsKICAgICAgICBpZiAoKG5wYmxiICUgNFUpID09IDFVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSBfX1JPUigoMHhGRkZGRkYwMFUpLCAxNik7CiAgICAgICAgfQogICAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gMlUpCiAgICAgICAgewogICAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUk9SKCgweEZGRkYwMDAwVSksIDE2KTsKICAgICAgICB9CiAgICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAzVSkKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gX19ST1IoKDB4RkYwMDAwMDBVKSwgMTYpOwogICAgICAgIH0KICAgICAgfQogICAgICBlbHNlIC8qQ1JZUF9EQVRBVFlQRV8xQiovCiAgICAgIHsKICAgICAgICBpZiAoKG5wYmxiICUgNFUpID09IDFVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSBfX1JCSVQoMHhGRkZGRkYwMFUpOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICUgNFUpID09IDJVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSBfX1JCSVQoMHhGRkZGMDAwMFUpOwogICAgICAgIH0KICAgICAgICBpZiAoKG5wYmxiICUgNFUpID09IDNVKQogICAgICAgIHsKICAgICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSBfX1JCSVQoMHhGRjAwMDAwMFUpOwogICAgICAgIH0KICAgICAgfQogICAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgbGFzdHdvcmRzaXplOyBpbmRleCArKykKICAgICAgewogICAgICAgIC8qV3JpdGUgdGhlIGludGVybWVkaWF0ZV9kYXRhIGluIHRoZSBJTiBGSUZPICovCiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gPSBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF07CiAgICAgIH0KICAgICAgd2hpbGUgKGluZGV4IDwgNFUpCiAgICAgIHsKICAgICAgICAvKiBQYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAweDBVOwogICAgICAgIGluZGV4Kys7CiAgICAgIH0KICAgICAgLyogV2FpdCBmb3IgT0ZORSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgICBpZiAoQ1JZUF9XYWl0T25PRk5FRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICB9CgogICAgICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSkgIT0gMHgwVSkKICAgICAgewogICAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCA0VTsgaW5kZXgrKykKICAgICAgICB7CiAgICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICAgfQogICAgICB9CiAgICB9CiAgfSAvKiBFbmQgb2YgR0NNIGVuY3J5cHRpb24gKi8KICBlbHNlCiAgewogICAgLyogV29ya2Fyb3VuZCAyLCBjYXNlIENDTSBkZWNyeXB0aW9uLCBpbiBvcmRlciB0byBwcm9wZXJseSBjb21wdXRlCiAgICAgIGF1dGhlbnRpY2F0aW9uIHRhZ3Mgd2hpbGUgZG9pbmcgYSBDQ00gZGVjcnlwdGlvbiB3aXRoIHRoZSBsYXN0IGJsb2NrCiAgICAgIG9mIHBheWxvYWQgc2l6ZSBpbmZlcmlvciB0byAxMjggYml0cyovCgogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPRElSKSA9PSBDUllQX09QRVJBVElOR01PREVfREVDUllQVCkKICAgIHsKICAgICAgaXYxdGVtcCA9IGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ003UjsKCiAgICAgIC8qIERpc2FibGUgQ1JZUCB0byBzdGFydCB0aGUgZmluYWwgcGhhc2UgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIHRlbXBbMF0gPSAgaGNyeXAtPkluc3RhbmNlLT5DU0dDTUNDTTBSOwogICAgICB0ZW1wWzFdID0gIGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ00xUjsKICAgICAgdGVtcFsyXSA9ICBoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNMlI7CiAgICAgIHRlbXBbM10gPSAgaGNyeXAtPkluc3RhbmNlLT5DU0dDTUNDTTNSOwoKICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9IGl2MXRlbXA7CgogICAgICAvKiBDb25maWd1cmVkICBDSE1PRCBDVFIgICAqLwogICAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIENSWVBfQUVTX0NUUik7CgogICAgICAvKiBFbmFibGUgQ1JZUCB0byBzdGFydCB0aGUgZmluYWwgcGhhc2UgKi8KICAgICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogICAgfQogICAgLyogIExhc3QgYmxvY2sgb3B0aW9uYWxseSBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyovCiAgICBmb3IgKGluZGV4ID0gMDsgaW5kZXggPCBsYXN0d29yZHNpemU7IGluZGV4ICsrKQogICAgewogICAgICAvKiBXcml0ZSB0aGUgbGFzdCBJbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgfQogICAgd2hpbGUgKGluZGV4IDwgNFUpCiAgICB7CiAgICAgIC8qIFBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAwVTsKICAgICAgaW5kZXgrKzsKICAgIH0KICAgIC8qIFdhaXQgZm9yIE9GTkUgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgIGlmIChDUllQX1dhaXRPbk9GTkVGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICB7CiAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgLyogQ2hhbmdlIHN0YXRlICovCiAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxVSkKICAgICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgfQoKICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19PRk5FKSAhPSAweDBVKQogICAgewogICAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgICAgIHsKICAgICAgICAvKiBSZWFkIHRoZSBPdXRwdXQgYmxvY2sgZnJvbSB0aGUgT3V0cHV0IEZJRk8gKi8KICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CgogICAgICAgIC8qaW50ZXJtZWRpYXRlIGRhdGEgYnVmZmVyIHRvIGJlIHVzZWQgaW4gZm9yIHRoZSB3b3JrYXJvdW5kKi8KICAgICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyAoaGNyeXAtPkNyeXBPdXRDb3VudCkpID0gaW50ZXJtZWRpYXRlX2RhdGFbaW5kZXhdOwogICAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgfQogICAgfQoKICAgIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIENSWVBfQ1JfQUxHT0RJUikgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpCiAgICB7CiAgICAgIHRlbXAyWzBdID0gIGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ00wUjsKICAgICAgdGVtcDJbMV0gPSAgaGNyeXAtPkluc3RhbmNlLT5DU0dDTUNDTTFSOwogICAgICB0ZW1wMlsyXSA9ICBoY3J5cC0+SW5zdGFuY2UtPkNTR0NNQ0NNMlI7CiAgICAgIHRlbXAyWzNdID0gIGhjcnlwLT5JbnN0YW5jZS0+Q1NHQ01DQ00zUjsKCiAgICAgIC8qIGNvbmZpZ3VyZWQgIENITU9EIENDTSAgICovCiAgICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPTU9ERSwgQ1JZUF9BRVNfQ0NNKTsKCiAgICAgIC8qIGNvbmZpZ3VyZWQgIEhlYWRlciBwaGFzZSAgKi8KICAgICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0dDTV9DQ01QSCwgQ1JZUF9QSEFTRV9IRUFERVIpOwoKICAgICAgLypzZXQgdG8gemVybyB0aGUgYml0cyBjb3JyZXNwb25kaW5nIHRvIHRoZSBwYWRkZWQgYml0cyovCiAgICAgIGZvciAoaW5kZXggPSBsYXN0d29yZHNpemU7IGluZGV4IDwgNFU7IGluZGV4ICsrKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbaW5kZXhdID0gMFU7CiAgICAgIH0KICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAxVSkKICAgICAgewogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSAweEZGRkZGRjAwVTsKICAgICAgfQogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDJVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IDB4RkZGRjAwMDBVOwogICAgICB9CiAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gM1UpCiAgICAgIHsKICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gMHhGRjAwMDAwMFU7CiAgICAgIH0KICAgICAgZm9yIChpbmRleCA9IDBVOyBpbmRleCA8IDRVIDsgaW5kZXggKyspCiAgICAgIHsKICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF0gXj0gIHRlbXBbaW5kZXhdOwogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSBePSAgdGVtcDJbaW5kZXhdOwogICAgICB9CiAgICAgIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCA0VTsgaW5kZXggKyspCiAgICAgIHsKICAgICAgICAvKiBXcml0ZSB0aGUgbGFzdCBJbnB1dCBibG9jayBpbiB0aGUgSU4gRklGTyAqLwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9IGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSA7CiAgICAgIH0KCiAgICAgIC8qIFdhaXQgZm9yIEJVU1kgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgICAgaWYgKENSWVBfV2FpdE9uQlVTWUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgICAgewogICAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfVElNRU9VVDsKICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxVSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgICAgIH0KICAgIH0KICB9IC8qIEVuZCBvZiBDQ00gV0tBKi8KCiAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgojZWxzZSAvKiBBRVMgKi8KCiAgLypXb3JrYXJvdW5kIDI6IGNhc2UgR0NNIGVuY3J5cHRpb24sIGR1cmluZyBwYXlsb2FkIHBoYXNlIGFuZCBiZWZvcmUgaW5zZXJ0aW5nCiAgdGhlIGxhc3QgYmxvY2sgb2YgcGF5bGFvZCwgd2hpY2ggc2l6ZSBpcyBpbmZlcmlvciB0byAgMTI4IGJpdHMgICovCgogIGlmICgoaGNyeXAtPkluc3RhbmNlLT5DUiAmIEFFU19DUl9NT0RFKSA9PSBDUllQX09QRVJBVElOR01PREVfRU5DUllQVCkKICB7CiAgICAvKiBjb25maWd1cmVkICBDSE1PRCBDVFIgICAqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBBRVNfQ1JfQ0hNT0QsIENSWVBfQUVTX0NUUik7CiAgfQogIC8qICBsYXN0IGJsb2NrIG9wdGlvbmFsbHkgcGFkIHRoZSBkYXRhIHdpdGggemVyb3MqLwogIGZvciAoaW5kZXggPSAwVTsgaW5kZXggPCBsYXN0d29yZHNpemU7IGluZGV4ICsrKQogIHsKICAgIC8qIFdyaXRlIHRoZSBsYXN0IElucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTlIgID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQpOwogICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgfQogIHdoaWxlIChpbmRleCA8IDRVKQogIHsKICAgIC8qIHBhZCB0aGUgZGF0YSB3aXRoIHplcm9zIHRvIGhhdmUgYSBjb21wbGV0ZSBibG9jayAqLwogICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9IDBVOwogICAgaW5kZXgrKzsKICB9CiAgLyogV2FpdCBmb3IgQ0NGIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgaWYgKENSWVBfV2FpdE9uQ0NGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgewogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMVUpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQoKICAvKiBDbGVhciBDQ0YgRmxhZyAqLwogIF9fSEFMX0NSWVBfQ0xFQVJfRkxBRyhoY3J5cCwgQ1JZUF9DQ0ZfQ0xFQVIpOwoKICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgewogICAgLyogUmVhZCB0aGUgT3V0cHV0IGJsb2NrIGZyb20gdGhlIE91dHB1dCBGSUZPICovCiAgICBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF0gPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVRSOwoKICAgIC8qaW50ZXJtZWRpYXRlIGRhdGEgYnVmZmVyIHRvIGJlIHVzZWQgaW4gIHRoZSB3b3JrYXJvdW5kKi8KICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIChoY3J5cC0+Q3J5cE91dENvdW50KSkgPSBpbnRlcm1lZGlhdGVfZGF0YVtpbmRleF07CiAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgfQoKICBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBBRVNfQ1JfTU9ERSkgPT0gQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpCiAgewogICAgLyogY29uZmlndXJlZCAgQ0hNT0QgR0NNICAgKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQUVTX0NSX0NITU9ELCBDUllQX0FFU19HQ01fR01BQyk7CgogICAgLyogU2VsZWN0IGZpbmFsIHBoYXNlICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIEFFU19DUl9HQ01QSCwgQ1JZUF9QSEFTRV9GSU5BTCk7CgogICAgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQUVTX0NSX0RBVEFUWVBFKSA9PSBDUllQX0RBVEFUWVBFXzMyQikKICAgIHsKICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAxVSkKICAgICAgewogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSAweEZGRkZGRjAwVTsKICAgICAgfQogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDJVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IDB4RkZGRjAwMDBVOwogICAgICB9CiAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gM1UpCiAgICAgIHsKICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gMHhGRjAwMDAwMFU7CiAgICAgIH0KICAgIH0KICAgIGVsc2UgaWYgKChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQUVTX0NSX0RBVEFUWVBFKSA9PSBDUllQX0RBVEFUWVBFXzhCKQogICAgewogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDFVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUkVWKDB4RkZGRkZGMDBVKTsKICAgICAgfQogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDJVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUkVWKDB4RkZGRjAwMDBVKTsKICAgICAgfQogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDNVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUkVWKDB4RkYwMDAwMDBVKTsKICAgICAgfQogICAgfQogICAgZWxzZSBpZiAoKGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBBRVNfQ1JfREFUQVRZUEUpID09IENSWVBfREFUQVRZUEVfMTZCKQogICAgewogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDFVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUk9SKCgweEZGRkZGRjAwVSksIDE2KTsKICAgICAgfQogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDJVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUk9SKCgweEZGRkYwMDAwVSksIDE2KTsKICAgICAgfQogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDNVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUk9SKCgweEZGMDAwMDAwVSksIDE2KTsKICAgICAgfQogICAgfQogICAgZWxzZSAvKkNSWVBfREFUQVRZUEVfMUIqLwogICAgewogICAgICBpZiAoKG5wYmxiICUgNFUpID09IDFVKQogICAgICB7CiAgICAgICAgaW50ZXJtZWRpYXRlX2RhdGFbbGFzdHdvcmRzaXplIC0gMVVdICY9IF9fUkJJVCgweEZGRkZGRjAwVSk7CiAgICAgIH0KICAgICAgaWYgKChucGJsYiAlIDRVKSA9PSAyVSkKICAgICAgewogICAgICAgIGludGVybWVkaWF0ZV9kYXRhW2xhc3R3b3Jkc2l6ZSAtIDFVXSAmPSBfX1JCSVQoMHhGRkZGMDAwMFUpOwogICAgICB9CiAgICAgIGlmICgobnBibGIgJSA0VSkgPT0gM1UpCiAgICAgIHsKICAgICAgICBpbnRlcm1lZGlhdGVfZGF0YVtsYXN0d29yZHNpemUgLSAxVV0gJj0gX19SQklUKDB4RkYwMDAwMDBVKTsKICAgICAgfQogICAgfQoKICAgIC8qV3JpdGUgdGhlIGludGVybWVkaWF0ZV9kYXRhIGluIHRoZSBJTiBGSUZPICovCiAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgbGFzdHdvcmRzaXplOyBpbmRleCArKykKICAgIHsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SICA9IGludGVybWVkaWF0ZV9kYXRhW2luZGV4XTsKICAgIH0KICAgIHdoaWxlIChpbmRleCA8IDRVKQogICAgewogICAgICAvKiBwYWQgdGhlIGRhdGEgd2l0aCB6ZXJvcyB0byBoYXZlIGEgY29tcGxldGUgYmxvY2sgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU5SID0gMFU7CiAgICAgIGluZGV4Kys7CiAgICB9CiAgICAvKiBXYWl0IGZvciBDQ0YgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgIGlmIChDUllQX1dhaXRPbkNDRmxhZyhoY3J5cCwgVGltZW91dCkgIT0gSEFMX09LKQogICAgewogICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgIC8qIENoYW5nZSBzdGF0ZSAqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMVUpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgIH0KICAgIC8qIENsZWFyIENDRiBGbGFnICovCiAgICBfX0hBTF9DUllQX0NMRUFSX0ZMQUcoaGNyeXAsIENSWVBfQ0NGX0NMRUFSKTsKCiAgICBmb3IgKGluZGV4ID0gMFU7IGluZGV4IDwgNFU7IGluZGV4KyspCiAgICB7CiAgICAgIGludGVybWVkaWF0ZV9kYXRhW2luZGV4XSA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVFI7CiAgICB9CiAgfS8qRW5kIG9mIFdvcmthcm91bmQgMiovCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgKi8KfQoKI2lmIGRlZmluZWQgKENSWVApCgovKioKICAqIEBicmllZiAgSGFuZGxlIENSWVAgaGFyZHdhcmUgYmxvY2sgVGltZW91dCB3aGVuIHdhaXRpbmcgZm9yIElGRU0gZmxhZyB0byBiZSByYWlzZWQuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEBwYXJhbSAgVGltZW91dDogVGltZW91dCBkdXJhdGlvbi4KICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1dhaXRPbklGRU1GbGFnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKICB1aW50MzJfdCB0aWNrc3RhcnQ7CgogIC8qIEdldCB0aW1lb3V0ICovCiAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKCiAgd2hpbGUgKEhBTF9JU19CSVRfQ0xSKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19JRkVNKSkKICB7CiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgSGFuZGxlIENSWVAgaGFyZHdhcmUgYmxvY2sgVGltZW91dCB3aGVuIHdhaXRpbmcgZm9yIEJVU1kgZmxhZyB0byBiZSByYWlzZWQuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEBwYXJhbSAgVGltZW91dDogVGltZW91dCBkdXJhdGlvbi4KICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1dhaXRPbkJVU1lGbGFnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKICB1aW50MzJfdCB0aWNrc3RhcnQ7CgogIC8qIEdldCB0aW1lb3V0ICovCiAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKCiAgd2hpbGUgKEhBTF9JU19CSVRfU0VUKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19CVVNZKSkKICB7CiAgICAvKiBDaGVjayBmb3IgdGhlIFRpbWVvdXQgKi8KICAgIGlmIChUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmICgoKEhBTF9HZXRUaWNrKCkgLSB0aWNrc3RhcnQpID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAYnJpZWYgIEhhbmRsZSBDUllQIGhhcmR3YXJlIGJsb2NrIFRpbWVvdXQgd2hlbiB3YWl0aW5nIGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgZHVyYXRpb24uCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9XYWl0T25PRk5FRmxhZyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgdWludDMyX3QgdGlja3N0YXJ0OwoKICAvKiBHZXQgdGltZW91dCAqLwogIHRpY2tzdGFydCA9IEhBTF9HZXRUaWNrKCk7CgogIHdoaWxlIChIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfT0ZORSkpCiAgewogICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICBpZiAoVGltZW91dCAhPSBIQUxfTUFYX0RFTEFZKQogICAgewogICAgICBpZiAoKChIQUxfR2V0VGljaygpIC0gdGlja3N0YXJ0KSA+IFRpbWVvdXQpIHx8IChUaW1lb3V0ID09IDBVKSkKICAgICAgewogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0KICB9CiAgcmV0dXJuIEhBTF9PSzsKfQoKI2Vsc2UgLyogQUVTICovCgovKioKICAqIEBicmllZiAgSGFuZGxlIENSWVAgaGFyZHdhcmUgYmxvY2sgVGltZW91dCB3aGVuIHdhaXRpbmcgZm9yIENDRiBmbGFnIHRvIGJlIHJhaXNlZC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHBhcmFtICBUaW1lb3V0OiBUaW1lb3V0IGR1cmF0aW9uLgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uQ0NGbGFnKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKICB1aW50MzJfdCB0aWNrc3RhcnQ7CgogIC8qIEdldCB0aW1lb3V0ICovCiAgdGlja3N0YXJ0ID0gSEFMX0dldFRpY2soKTsKCiAgd2hpbGUgKEhBTF9JU19CSVRfQ0xSKGhjcnlwLT5JbnN0YW5jZS0+U1IsIEFFU19TUl9DQ0YpKQogIHsKICAgIC8qIENoZWNrIGZvciB0aGUgVGltZW91dCAqLwogICAgaWYgKFRpbWVvdXQgIT0gSEFMX01BWF9ERUxBWSkKICAgIHsKICAgICAgaWYgKCgoSEFMX0dldFRpY2soKSAtIHRpY2tzdGFydCkgPiBUaW1lb3V0KSB8fCAoVGltZW91dCA9PSAwVSkpCiAgICAgIHsKICAgICAgICByZXR1cm4gSEFMX0VSUk9SOwogICAgICB9CiAgICB9CiAgfQogIHJldHVybiBIQUxfT0s7Cn0KCiNlbmRpZiAvKiBFbmQgQUVTIG9yIENSWVAgICovCgoKLyoqCiAgKiBAfQogICovCgoKCi8qKgogICogQH0KICAqLwoKLyoqCiAgKiBAfQogICovCgojZW5kaWYgLyogSEFMX0NSWVBfTU9EVUxFX0VOQUJMRUQgKi8KCgovKioKICAqIEB9CiAgKi8KI2VuZGlmIC8qIFRpbnlBRVMgb3IgQ1JZUCovCi8qKgogICogQH0KICAqLwoKLyoqKioqKioqKioqKioqKioqKioqKioqKiAoQykgQ09QWVJJR0hUIFNUTWljcm9lbGVjdHJvbmljcyAqKioqKkVORCBPRiBGSUxFKioqKi8K