LyoqCiAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAgKiBAZmlsZSAgICBzdG0zMmYyeHhfaGFsX2NyeXAuYwogICogQGF1dGhvciAgTUNEIEFwcGxpY2F0aW9uIFRlYW0KICAqIEBicmllZiAgIENSWVAgSEFMIG1vZHVsZSBkcml2ZXIuCiAgKiAgICAgICAgICBUaGlzIGZpbGUgcHJvdmlkZXMgZmlybXdhcmUgZnVuY3Rpb25zIHRvIG1hbmFnZSB0aGUgZm9sbG93aW5nCiAgKiAgICAgICAgICBmdW5jdGlvbmFsaXRpZXMgb2YgdGhlIENyeXB0b2dyYXBoeSAoQ1JZUCkgcGVyaXBoZXJhbDoKICAqICAgICAgICAgICArIEluaXRpYWxpemF0aW9uIGFuZCBkZS1pbml0aWFsaXphdGlvbiBmdW5jdGlvbnMKICAqICAgICAgICAgICArIEFFUyBwcm9jZXNzaW5nIGZ1bmN0aW9ucwogICogICAgICAgICAgICsgREVTIHByb2Nlc3NpbmcgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBUREVTIHByb2Nlc3NpbmcgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBETUEgY2FsbGJhY2sgZnVuY3Rpb25zCiAgKiAgICAgICAgICAgKyBDUllQIElSUSBoYW5kbGVyIG1hbmFnZW1lbnQKICAqICAgICAgICAgICArIFBlcmlwaGVyYWwgU3RhdGUgZnVuY3Rpb25zCiAgKgogIEB2ZXJiYXRpbQogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgICAgICAgICAgICAgICAgICAjIyMjIyBIb3cgdG8gdXNlIHRoaXMgZHJpdmVyICMjIyMjCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICBbLi5dCiAgICAgIFRoZSBDUllQIEhBTCBkcml2ZXIgY2FuIGJlIHVzZWQgaW4gQ1JZUCBJUCBhcyBmb2xsb3dzOgoKICAgICAgKCMpSW5pdGlhbGl6ZSB0aGUgQ1JZUCBsb3cgbGV2ZWwgcmVzb3VyY2VzIGJ5IGltcGxlbWVudGluZyB0aGUgSEFMX0NSWVBfTXNwSW5pdCgpOgogICAgICAgICAoIyMpIEVuYWJsZSB0aGUgQ1JZUCBpbnRlcmZhY2UgY2xvY2sgdXNpbmcgX19IQUxfUkNDX0NSWVBfQ0xLX0VOQUJMRSgpCiAgICAgICAgICgjIykgSW4gY2FzZSBvZiB1c2luZyBpbnRlcnJ1cHRzIChlLmcuIEhBTF9DUllQX0VuY3J5cHRfSVQoKSkKICAgICAgICAgICAgICgrKyspIENvbmZpZ3VyZSB0aGUgQ1JZUCBpbnRlcnJ1cHQgcHJpb3JpdHkgdXNpbmcgSEFMX05WSUNfU2V0UHJpb3JpdHkoKQogICAgICAgICAgICAgKCsrKykgRW5hYmxlIHRoZSBDUllQIElSUSBoYW5kbGVyIHVzaW5nIEhBTF9OVklDX0VuYWJsZUlSUSgpCiAgICAgICAgICAgICAoKysrKSBJbiBDUllQIElSUSBoYW5kbGVyLCBjYWxsIEhBTF9DUllQX0lSUUhhbmRsZXIoKQogICAgICAgICAoIyMpIEluIGNhc2Ugb2YgdXNpbmcgRE1BIHRvIGNvbnRyb2wgZGF0YSB0cmFuc2ZlciAoZS5nLiBIQUxfQ1JZUF9FbmNyeXB0X0RNQSgpKQogICAgICAgICAgICAgKCsrKykgRW5hYmxlIHRoZSBETUF4IGludGVyZmFjZSBjbG9jayB1c2luZyBfX1JDQ19ETUF4X0NMS19FTkFCTEUoKQogICAgICAgICAgICAgKCsrKykgQ29uZmlndXJlIGFuZCBlbmFibGUgdHdvIERNQSBzdHJlYW1zIG9uZSBmb3IgbWFuYWdpbmcgZGF0YSB0cmFuc2ZlciBmcm9tCiAgICAgICAgICAgICAgICAgbWVtb3J5IHRvIHBlcmlwaGVyYWwgKGlucHV0IHN0cmVhbSkgYW5kIGFub3RoZXIgc3RyZWFtIGZvciBtYW5hZ2luZyBkYXRhCiAgICAgICAgICAgICAgICAgdHJhbnNmZXIgZnJvbSBwZXJpcGhlcmFsIHRvIG1lbW9yeSAob3V0cHV0IHN0cmVhbSkKICAgICAgICAgICAgICgrKyspIEFzc29jaWF0ZSB0aGUgaW5pdGlhbGl6ZWQgRE1BIGhhbmRsZSB0byB0aGUgQ1JZUCBETUEgaGFuZGxlCiAgICAgICAgICAgICAgICAgdXNpbmcgIF9fSEFMX0xJTktETUEoKQogICAgICAgICAgICAgKCsrKykgQ29uZmlndXJlIHRoZSBwcmlvcml0eSBhbmQgZW5hYmxlIHRoZSBOVklDIGZvciB0aGUgdHJhbnNmZXIgY29tcGxldGUKICAgICAgICAgICAgICAgICBpbnRlcnJ1cHQgb24gdGhlIHR3byBETUEgU3RyZWFtcy4gVGhlIG91dHB1dCBzdHJlYW0gc2hvdWxkIGhhdmUgaGlnaGVyCiAgICAgICAgICAgICAgICAgcHJpb3JpdHkgdGhhbiB0aGUgaW5wdXQgc3RyZWFtIEhBTF9OVklDX1NldFByaW9yaXR5KCkgYW5kIEhBTF9OVklDX0VuYWJsZUlSUSgpCgogICAgICAoIylJbml0aWFsaXplIHRoZSBDUllQIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkIHBhcmFtZXRlcnMgOgogICAgICAgICAoIyMpIFRoZSBkYXRhIHR5cGU6IDEtYml0LCA4LWJpdCwgMTYtYml0IG9yIDMyLWJpdC4KICAgICAgICAgKCMjKSBUaGUga2V5IHNpemU6IDEyOCwgMTkyIG9yIDI1Ni4KICAgICAgICAgKCMjKSBUaGUgQWxnb01vZGUgREVTLyBUREVTIEFsZ29yaXRobSBFQ0IvQ0JDIG9yIEFFUyBBbGdvcml0aG0gRUNCL0NCQy9DVFIuCiAgICAgICAgICgjIykgVGhlIGluaXRpYWxpemF0aW9uIHZlY3RvciAoY291bnRlcikuIEl0IGlzIG5vdCB1c2VkIGluIEVDQiBtb2RlLgogICAgICAgICAoIyMpIFRoZSBrZXkgYnVmZmVyIHVzZWQgZm9yIGVuY3J5cHRpb24vZGVjcnlwdGlvbi4KCiAgICAgICgjKVRocmVlIHByb2Nlc3NpbmcgKGVuY3J5cHRpb24vZGVjcnlwdGlvbikgZnVuY3Rpb25zIGFyZSBhdmFpbGFibGU6CiAgICAgICAgICgjIykgUG9sbGluZyBtb2RlOiBlbmNyeXB0aW9uIGFuZCBkZWNyeXB0aW9uIEFQSXMgYXJlIGJsb2NraW5nIGZ1bmN0aW9ucwogICAgICAgICAgICAgIGkuZS4gdGhleSBwcm9jZXNzIHRoZSBkYXRhIGFuZCB3YWl0IHRpbGwgdGhlIHByb2Nlc3NpbmcgaXMgZmluaXNoZWQsCiAgICAgICAgICAgICAgZS5nLiBIQUxfQ1JZUF9FbmNyeXB0ICYgSEFMX0NSWVBfRGVjcnlwdAogICAgICAgICAoIyMpIEludGVycnVwdCBtb2RlOiBlbmNyeXB0aW9uIGFuZCBkZWNyeXB0aW9uIEFQSXMgYXJlIG5vdCBibG9ja2luZyBmdW5jdGlvbnMKICAgICAgICAgICAgICBpLmUuIHRoZXkgcHJvY2VzcyB0aGUgZGF0YSB1bmRlciBpbnRlcnJ1cHQsCiAgICAgICAgICAgICAgZS5nLiBIQUxfQ1JZUF9FbmNyeXB0X0lUICYgSEFMX0NSWVBfRGVjcnlwdF9JVAogICAgICAgICAoIyMpIERNQSBtb2RlOiBlbmNyeXB0aW9uIGFuZCBkZWNyeXB0aW9uIEFQSXMgYXJlIG5vdCBibG9ja2luZyBmdW5jdGlvbnMKICAgICAgICAgICAgICBpLmUuIHRoZSBkYXRhIHRyYW5zZmVyIGlzIGVuc3VyZWQgYnkgRE1BLAogICAgICAgICAgICAgIGUuZy4gSEFMX0NSWVBfRW5jcnlwdF9ETUEgJiBIQUxfQ1JZUF9EZWNyeXB0X0RNQQoKICAgICAgKCMpV2hlbiB0aGUgcHJvY2Vzc2luZyBmdW5jdGlvbiBpcyBjYWxsZWQgYXQgZmlyc3QgdGltZSBhZnRlciBIQUxfQ1JZUF9Jbml0KCkKICAgICAgICAgdGhlIENSWVAgcGVyaXBoZXJhbCBpcyBjb25maWd1cmVkIGFuZCBwcm9jZXNzZXMgdGhlIGJ1ZmZlciBpbiBpbnB1dC4KICAgICAgICAgQXQgc2Vjb25kIGNhbGwsIG5vIG5lZWQgdG8gSW5pdGlhbGl6ZSB0aGUgQ1JZUCwgdXNlciBoYXZlIHRvIGdldCBjdXJyZW50IGNvbmZpZ3VyYXRpb24gdmlhCiAgICAgICAgIEhBTF9DUllQX0dldENvbmZpZygpIEFQSSwgdGhlbiBvbmx5ICBIQUxfQ1JZUF9TZXRDb25maWcoKSBpcyByZXF1ZXN0ZWQgdG8gc2V0CiAgICAgICAgIG5ldyBwYXJhbWV0cmVzLCBmaW5hbGx5IHVzZXIgY2FuICBzdGFydCBlbmNyeXB0aW9uL2RlY3J5cHRpb24uCgogICAgICAgKCMpQ2FsbCBIQUxfQ1JZUF9EZUluaXQoKSB0byBkZWluaXRpYWxpemUgdGhlIENSWVAgcGVyaXBoZXJhbC4KCiAgICBbLi5dCiAgICAgIFRoZSBjcnlwdG9ncmFwaGljIHByb2Nlc3NvciBzdXBwb3J0cyBmb2xsb3dpbmcgc3RhbmRhcmRzOgogICAgICAoIykgVGhlIGRhdGEgZW5jcnlwdGlvbiBzdGFuZGFyZCAoREVTKSBhbmQgVHJpcGxlLURFUyAoVERFUykgc3VwcG9ydGVkIG9ubHkgYnkgQ1JZUDEgSVA6CiAgICAgICAgICgjIyk2NC1iaXQgZGF0YSBibG9jayBwcm9jZXNzaW5nCiAgICAgICAgICgjIykgY2hhaW5pbmcgbW9kZXMgc3VwcG9ydGVkIDoKICAgICAgICAgICAgICgrKyspICBFbGVjdHJvbmljIENvZGUgQm9vayhFQ0IpCiAgICAgICAgICAgICAoKysrKSAgQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICAgICgjIykga2V5cyBsZW5ndGggc3VwcG9ydGVkIDo2NC1iaXQsIDEyOC1iaXQgYW5kIDE5Mi1iaXQuCiAgICAgICgjKSBUaGUgYWR2YW5jZWQgZW5jcnlwdGlvbiBzdGFuZGFyZCAoQUVTKSBzdXBwb3J0ZWQgIGJ5IENSWVAxOgogICAgICAgICAoIyMpMTI4LWJpdCBkYXRhIGJsb2NrIHByb2Nlc3NpbmcKICAgICAgICAgKCMjKSBjaGFpbmluZyBtb2RlcyBzdXBwb3J0ZWQgOgogICAgICAgICAgICAgKCsrKykgIEVsZWN0cm9uaWMgQ29kZSBCb29rKEVDQikKICAgICAgICAgICAgICgrKyspICBDaXBoZXIgQmxvY2sgQ2hhaW5pbmcgKENCQykKICAgICAgICAgICAgICgrKyspICBDb3VudGVyIG1vZGUgKENUUikKICAgICAgICAgKCMjKSBrZXlzIGxlbmd0aCBTdXBwb3J0ZWQgOgogICAgICAgICAgICAgKCsrKykgZm9yIENSWVAxIElQOiAxMjgtYml0LCAxOTItYml0IGFuZCAyNTYtYml0LgoKICAqKiogQ2FsbGJhY2sgcmVnaXN0cmF0aW9uICoqKgogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogIFsuLl0KICBUaGUgY29tcGlsYXRpb24gZGVmaW5lICBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTIHdoZW4gc2V0IHRvIDEKICBhbGxvd3MgdGhlIHVzZXIgdG8gY29uZmlndXJlIGR5bmFtaWNhbGx5IHRoZSBkcml2ZXIgY2FsbGJhY2tzLgogIFVzZSBGdW5jdGlvbnMgQHJlZiBIQUxfQ1JZUF9SZWdpc3RlckNhbGxiYWNrKCkgb3IgSEFMX0NSWVBfUmVnaXN0ZXJYWFhDYWxsYmFjaygpCiAgdG8gcmVnaXN0ZXIgYW4gaW50ZXJydXB0IGNhbGxiYWNrLgogIFsuLl0KICBGdW5jdGlvbiBAcmVmIEhBTF9DUllQX1JlZ2lzdGVyQ2FsbGJhY2soKSBhbGxvd3MgdG8gcmVnaXN0ZXIgZm9sbG93aW5nIGNhbGxiYWNrczoKICAgICgrKSBJbkNwbHRDYWxsYmFjayAgICAgOiAgSW5wdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2suCiAgICAoKykgT3V0Q3BsdENhbGxiYWNrICAgIDogT3V0cHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrLgogICAgKCspIEVycm9yQ2FsbGJhY2sgICAgICA6IGNhbGxiYWNrIGZvciBlcnJvciBkZXRlY3Rpb24uCiAgICAoKykgTXNwSW5pdENhbGxiYWNrICAgIDogQ1JZUCBNc3BJbml0LgogICAgKCspIE1zcERlSW5pdENhbGxiYWNrICA6IENSWVAgTXNwRGVJbml0LgogIFRoaXMgZnVuY3Rpb24gdGFrZXMgYXMgcGFyYW1ldGVycyB0aGUgSEFMIHBlcmlwaGVyYWwgaGFuZGxlLCB0aGUgQ2FsbGJhY2sgSUQKICBhbmQgYSBwb2ludGVyIHRvIHRoZSB1c2VyIGNhbGxiYWNrIGZ1bmN0aW9uLgogIFsuLl0KICBVc2UgZnVuY3Rpb24gQHJlZiBIQUxfQ1JZUF9VblJlZ2lzdGVyQ2FsbGJhY2soKSB0byByZXNldCBhIGNhbGxiYWNrIHRvIHRoZSBkZWZhdWx0CiAgd2VhayBmdW5jdGlvbi4KICBAcmVmIEhBTF9DUllQX1VuUmVnaXN0ZXJDYWxsYmFjaygpIHRha2VzIGFzIHBhcmFtZXRlcnMgdGhlIEhBTCBwZXJpcGhlcmFsIGhhbmRsZSwKICBhbmQgdGhlIENhbGxiYWNrIElELgogIFRoaXMgZnVuY3Rpb24gYWxsb3dzIHRvIHJlc2V0IGZvbGxvd2luZyBjYWxsYmFja3M6CiAgICAoKykgSW5DcGx0Q2FsbGJhY2sgICAgIDogIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrLgogICAgKCspIE91dENwbHRDYWxsYmFjayAgICA6IE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAgICgrKSBFcnJvckNhbGxiYWNrICAgICAgOiBjYWxsYmFjayBmb3IgZXJyb3IgZGV0ZWN0aW9uLgogICAgKCspIE1zcEluaXRDYWxsYmFjayAgICA6IENSWVAgTXNwSW5pdC4KICAgICgrKSBNc3BEZUluaXRDYWxsYmFjayAgOiBDUllQIE1zcERlSW5pdC4KICBbLi5dCiAgQnkgZGVmYXVsdCwgYWZ0ZXIgdGhlIEByZWYgSEFMX0NSWVBfSW5pdCgpIGFuZCB3aGVuIHRoZSBzdGF0ZSBpcyBIQUxfQ1JZUF9TVEFURV9SRVNFVAogIGFsbCBjYWxsYmFja3MgYXJlIHNldCB0byB0aGUgY29ycmVzcG9uZGluZyB3ZWFrIGZ1bmN0aW9ucyA6CiAgZXhhbXBsZXMgQHJlZiBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjaygpICwgQHJlZiBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2soKS4KICBFeGNlcHRpb24gZG9uZSBmb3IgTXNwSW5pdCBhbmQgTXNwRGVJbml0IGZ1bmN0aW9ucyB0aGF0IGFyZQogIHJlc2V0IHRvIHRoZSBsZWdhY3kgd2VhayBmdW5jdGlvbiBpbiB0aGUgQHJlZiBIQUxfQ1JZUF9Jbml0KCkvIEByZWYgSEFMX0NSWVBfRGVJbml0KCkgb25seSB3aGVuCiAgdGhlc2UgY2FsbGJhY2tzIGFyZSBudWxsIChub3QgcmVnaXN0ZXJlZCBiZWZvcmVoYW5kKS4KICBpZiBub3QsIE1zcEluaXQgb3IgTXNwRGVJbml0IGFyZSBub3QgbnVsbCwgdGhlIEByZWYgSEFMX0NSWVBfSW5pdCgpIC8gQHJlZiBIQUxfQ1JZUF9EZUluaXQoKQogIGtlZXAgYW5kIHVzZSB0aGUgdXNlciBNc3BJbml0L01zcERlSW5pdCBmdW5jdGlvbnMgKHJlZ2lzdGVyZWQgYmVmb3JlaGFuZCkKCiAgQ2FsbGJhY2tzIGNhbiBiZSByZWdpc3RlcmVkL3VucmVnaXN0ZXJlZCBpbiBIQUxfQ1JZUF9TVEFURV9SRUFEWSBzdGF0ZSBvbmx5LgogIEV4Y2VwdGlvbiBkb25lIE1zcEluaXQvTXNwRGVJbml0IGNhbGxiYWNrcyB0aGF0IGNhbiBiZSByZWdpc3RlcmVkL3VucmVnaXN0ZXJlZAogIGluIEhBTF9DUllQX1NUQVRFX1JFQURZIG9yIEhBTF9DUllQX1NUQVRFX1JFU0VUIHN0YXRlLAogIHRodXMgcmVnaXN0ZXJlZCAodXNlcikgTXNwSW5pdC9EZUluaXQgY2FsbGJhY2tzIGNhbiBiZSB1c2VkIGR1cmluZyB0aGUgSW5pdC9EZUluaXQuCiAgSW4gdGhhdCBjYXNlIGZpcnN0IHJlZ2lzdGVyIHRoZSBNc3BJbml0L01zcERlSW5pdCB1c2VyIGNhbGxiYWNrcwogIHVzaW5nIEByZWYgSEFMX0NSWVBfUmVnaXN0ZXJDYWxsYmFjaygpIGJlZm9yZSBjYWxsaW5nIEByZWYgSEFMX0NSWVBfRGVJbml0KCkKICBvciBAcmVmIEhBTF9DUllQX0luaXQoKSBmdW5jdGlvbi4KICBbLi5dCiAgV2hlbiBUaGUgY29tcGlsYXRpb24gZGVmaW5lIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgaXMgc2V0IHRvIDAgb3IKICBub3QgZGVmaW5lZCwgdGhlIGNhbGxiYWNrIHJlZ2lzdHJhdGlvbiBmZWF0dXJlIGlzIG5vdCBhdmFpbGFibGUgYW5kIGFsbCBjYWxsYmFja3MKICBhcmUgc2V0IHRvIHRoZSBjb3JyZXNwb25kaW5nIHdlYWsgZnVuY3Rpb25zLgogIAogIEBlbmR2ZXJiYXRpbQogICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogICogQGF0dGVudGlvbgogICoKICAqIDxoMj48Y2VudGVyPiZjb3B5OyBDb3B5cmlnaHQgKGMpIDIwMTYgU1RNaWNyb2VsZWN0cm9uaWNzLgogICogQWxsIHJpZ2h0cyByZXNlcnZlZC48L2NlbnRlcj48L2gyPgogICoKICAqIFRoaXMgc29mdHdhcmUgY29tcG9uZW50IGlzIGxpY2Vuc2VkIGJ5IFNUIHVuZGVyIEJTRCAzLUNsYXVzZSBsaWNlbnNlLAogICogdGhlICJMaWNlbnNlIjsgWW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZQogICogTGljZW5zZS4gWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0OgogICogICAgICAgICAgICAgICAgICAgICAgICBvcGVuc291cmNlLm9yZy9saWNlbnNlcy9CU0QtMy1DbGF1c2UKICAqCiAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAgKi8KCi8qIEluY2x1ZGVzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCiNpbmNsdWRlICJzdG0zMmYyeHhfaGFsLmgiCgojaWYgZGVmaW5lZChDUllQKQojaWZkZWYgSEFMX0NSWVBfTU9EVUxFX0VOQUJMRUQKLyoqIEBhZGR0b2dyb3VwIFNUTTMyRjJ4eF9IQUxfRHJpdmVyCiAgKiBAewogICovCgovKiogQGFkZHRvZ3JvdXAgQ1JZUAogICogQHsKICAqLwoKLyogUHJpdmF0ZSB0eXBlZGVmIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyogUHJpdmF0ZSBkZWZpbmUgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KLyoqIEBhZGR0b2dyb3VwIENSWVBfUHJpdmF0ZV9EZWZpbmVzCiAgKiBAewogICovCgojZGVmaW5lIENSWVBfVElNRU9VVF9LRVlQUkVQQVJBVElPTiAgICAgIDgyVSAgICAgICAgIC8qVGhlIGxhdGVuY3kgb2Yga2V5IHByZXBhcmF0aW9uIG9wZXJhdGlvbiBpcyA4MiBjbG9jayBjeWNsZXMuKi8KCiNkZWZpbmUgIENSWVBfUEhBU0VfUkVBRFkgICAgICAgICAgICAgICAgMHgwMDAwMDAwMVUgLyohPCBDUllQIHBlcmlwaGVyYWwgaXMgcmVhZHkgZm9yIGluaXRpYWxpemF0aW9uLiAqLwojZGVmaW5lICBDUllQX1BIQVNFX1BST0NFU1MgICAgICAgICAgICAgIDB4MDAwMDAwMDJVIC8qITwgQ1JZUCBwZXJpcGhlcmFsIGlzIGluIHByb2Nlc3NpbmcgcGhhc2UgKi8KCiNkZWZpbmUgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQgICAgICAgMHgwMDAwMDAwMFUgICAgICAgICAgICAgLyohPCBFbmNyeXB0aW9uIG1vZGUgICAqLwojZGVmaW5lIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUICAgICAgIENSWVBfQ1JfQUxHT0RJUiAgICAgICAgIC8qITwgRGVjcnlwdGlvbiAgICAgICAgKi8KCi8qKgogICogQH0KICAqLwoKCi8qIFByaXZhdGUgbWFjcm8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCi8qKiBAYWRkdG9ncm91cCBDUllQX1ByaXZhdGVfTWFjcm9zCiAgKiBAewogICovCgojZGVmaW5lIEhBTF9DUllQX0ZJRk9fRkxVU0goX19IQU5ETEVfXykgKChfX0hBTkRMRV9fKS0+SW5zdGFuY2UtPkNSIHw9ICBDUllQX0NSX0ZGTFVTSCkKCgovKioKICAqIEB9CiAgKi8KCi8qIFByaXZhdGUgc3RydWN0IC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiBQcml2YXRlIHZhcmlhYmxlcyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiBQcml2YXRlIGZ1bmN0aW9uIHByb3RvdHlwZXMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiogQGFkZHRvZ3JvdXAgQ1JZUF9Qcml2YXRlX0Z1bmN0aW9uc19wcm90b3R5cGVzCiAgKiBAewogICovCgpzdGF0aWMgdm9pZCBDUllQX1NldERNQUNvbmZpZyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBpbnB1dGFkZHIsIHVpbnQxNl90IFNpemUsIHVpbnQzMl90IG91dHB1dGFkZHIpOwpzdGF0aWMgdm9pZCBDUllQX0RNQUluQ3BsdChETUFfSGFuZGxlVHlwZURlZiAqaGRtYSk7CnN0YXRpYyB2b2lkIENSWVBfRE1BT3V0Q3BsdChETUFfSGFuZGxlVHlwZURlZiAqaGRtYSk7CnN0YXRpYyB2b2lkIENSWVBfRE1BRXJyb3IoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpOwpzdGF0aWMgdm9pZCBDUllQX1NldEtleSggQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgS2V5U2l6ZSk7CnN0YXRpYyB2b2lkIENSWVBfQUVTX0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgdm9pZCBDUllQX0FFU19Qcm9jZXNzRGF0YShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwdCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19FbmNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0RlY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX0FFU19FbmNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCk7CnN0YXRpYyB2b2lkIENSWVBfVERFU19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKTsKc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uQlVTWUZsYWcoY29uc3QgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1dhaXRPbk9GTkVGbGFnKGNvbnN0IENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpOwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9UREVTX1Byb2Nlc3MoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCk7CgovKioKICAqIEB9CiAgKi8KCi8qIEV4cG9ydGVkIGZ1bmN0aW9ucyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwoKLyoqIEBkZWZncm91cCBDUllQX0V4cG9ydGVkX0Z1bmN0aW9ucyBDUllQIEV4cG9ydGVkIEZ1bmN0aW9ucwogICogQHsKICAqLwoKCi8qKiBAZGVmZ3JvdXAgQ1JZUF9FeHBvcnRlZF9GdW5jdGlvbnNfR3JvdXAxIEluaXRpYWxpemF0aW9uIGFuZCBkZS1pbml0aWFsaXphdGlvbiBmdW5jdGlvbnMKICogIEBicmllZiAgICBDUllQICBJbml0aWFsaXphdGlvbiBhbmQgQ29uZmlndXJhdGlvbiBmdW5jdGlvbnMuCiAqCkB2ZXJiYXRpbQogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICAgICAjIyMjIyBJbml0aWFsaXphdGlvbiwgZGUtaW5pdGlhbGl6YXRpb24gYW5kIFNldCBhbmQgR2V0IGNvbmZpZ3VyYXRpb24gZnVuY3Rpb25zICMjIyMjCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgWy4uXSAgVGhpcyBzZWN0aW9uIHByb3ZpZGVzIGZ1bmN0aW9ucyBhbGxvd2luZyB0bzoKICAgICAgKCspIEluaXRpYWxpemUgdGhlIENSWVAKICAgICAgKCspIERlSW5pdGlhbGl6ZSB0aGUgQ1JZUAogICAgICAoKykgSW5pdGlhbGl6ZSB0aGUgQ1JZUCBNU1AKICAgICAgKCspIERlSW5pdGlhbGl6ZSB0aGUgQ1JZUCBNU1AKICAgICAgKCspIGNvbmZpZ3VyZSBDUllQIChIQUxfQ1JZUF9TZXRDb25maWcpIHdpdGggdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzIGluIHRoZSBDUllQX0NvbmZpZ1R5cGVEZWYKICAgICAgICAgIFBhcmFtZXRlcnMgd2hpY2ggYXJlIGNvbmZpZ3VyZWQgaW4gVGhpcyBzZWN0aW9uIGFyZSA6CiAgICAgICAgICAoKyspIEtleSBzaXplCiAgICAgICAgICAoKyspIERhdGEgVHlwZSA6IDMyLDE2LCA4IG9yIDFiaXQKICAgICAgICAgICgrKykgQWxnb01vZGUgOiBmb3IgQ1JZUDEgSVAKICAgICAgICAgICAgICAgICBFQ0IgYW5kIENCQyBpbiBERVMvVERFUyBTdGFuZGFyZAogICAgICAgICAgICAgICAgIEVDQixDQkMgYW5kIENUUiBpbiBBRVMgU3RhbmRhcmQuCiAgICAgICgrKSBHZXQgQ1JZUCBjb25maWd1cmF0aW9uIChIQUxfQ1JZUF9HZXRDb25maWcpIGZyb20gdGhlIHNwZWNpZmllZCBwYXJhbWV0ZXJzIGluIHRoZSBDUllQX0hhbmRsZVR5cGVEZWYKCgpAZW5kdmVyYmF0aW0KICAqIEB7CiAgKi8KCgovKioKICAqIEBicmllZiAgSW5pdGlhbGl6ZXMgdGhlIENSWVAgYWNjb3JkaW5nIHRvIHRoZSBzcGVjaWZpZWQKICAqICAgICAgICAgcGFyYW1ldGVycyBpbiB0aGUgQ1JZUF9Db25maWdUeXBlRGVmIGFuZCBjcmVhdGVzIHRoZSBhc3NvY2lhdGVkIGhhbmRsZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9Jbml0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBDaGVjayB0aGUgQ1JZUCBoYW5kbGUgYWxsb2NhdGlvbiAqLwogIGlmKGhjcnlwID09IE5VTEwpCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIENoZWNrIHBhcmFtZXRlcnMgKi8KICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9LRVlTSVpFKGhjcnlwLT5Jbml0LktleVNpemUpKTsKICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9EQVRBVFlQRShoY3J5cC0+SW5pdC5EYXRhVHlwZSkpOwogIGFzc2VydF9wYXJhbShJU19DUllQX0FMR09SSVRITShoY3J5cC0+SW5pdC5BbGdvcml0aG0pKTsKCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRVNFVCkKICB7CiAgICAvKiBBbGxvY2F0ZSBsb2NrIHJlc291cmNlIGFuZCBpbml0aWFsaXplIGl0ICovCiAgICBoY3J5cC0+TG9jayA9IEhBTF9VTkxPQ0tFRDsKCiAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2sgID0gSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2s7ICAvKiBMZWdhY3kgd2VhayAgSW5DcGx0Q2FsbGJhY2sgICovCiAgICBoY3J5cC0+T3V0Q3BsdENhbGxiYWNrID0gSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrOyAvKiBMZWdhY3kgd2VhayBPdXRDcGx0Q2FsbGJhY2sgICovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayAgID0gSEFMX0NSWVBfRXJyb3JDYWxsYmFjazsgICAvKiBMZWdhY3kgd2VhayBFcnJvckNhbGxiYWNrICAgICovCgogICAgaWYoaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9PSBOVUxMKQogICAgewogICAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwSW5pdDsgLyogTGVnYWN5IHdlYWsgTXNwSW5pdCAgKi8KICAgIH0KCiAgICAvKiBJbml0IHRoZSBsb3cgbGV2ZWwgaGFyZHdhcmUgKi8KICAgIGhjcnlwLT5Nc3BJbml0Q2FsbGJhY2soaGNyeXApOwogIH0KI2Vsc2UKICBpZihoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVTRVQpCiAgewogICAgLyogQWxsb2NhdGUgbG9jayByZXNvdXJjZSBhbmQgaW5pdGlhbGl6ZSBpdCAqLwogICAgaGNyeXAtPkxvY2sgPSBIQUxfVU5MT0NLRUQ7CgogICAgLyogSW5pdCB0aGUgbG93IGxldmVsIGhhcmR3YXJlICovCiAgICBIQUxfQ1JZUF9Nc3BJbml0KGhjcnlwKTsKICB9CiNlbmRpZiAvKiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUykgKi8KCiAgLyogU2V0IHRoZSBrZXkgc2l6ZShUaGlzIGJpdCBmaWVsZCBpcyCRZG9uknQgY2FyZZIgaW4gdGhlIERFUyBvciBUREVTIG1vZGVzKSBkYXRhIHR5cGUgYW5kIEFsZ29yaXRobSAqLwogIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9EQVRBVFlQRXxDUllQX0NSX0tFWVNJWkV8Q1JZUF9DUl9BTEdPTU9ERSwgaGNyeXAtPkluaXQuRGF0YVR5cGUgfCBoY3J5cC0+SW5pdC5LZXlTaXplIHwgaGNyeXAtPkluaXQuQWxnb3JpdGhtKTsKCiAgLyogUmVzZXQgRXJyb3IgQ29kZSBmaWVsZCAqLwogIGhjcnlwLT5FcnJvckNvZGUgPSBIQUxfQ1JZUF9FUlJPUl9OT05FOwoKICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgLyogU2V0IHRoZSBkZWZhdWx0IENSWVAgcGhhc2UgKi8KICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1JFQURZOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIERlLUluaXRpYWxpemVzIHRoZSBDUllQIHBlcmlwaGVyYWwuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0RlSW5pdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgLyogQ2hlY2sgdGhlIENSWVAgaGFuZGxlIGFsbG9jYXRpb24gKi8KICBpZihoY3J5cCA9PSBOVUxMKQogIHsKICAgIHJldHVybiBIQUxfRVJST1I7CiAgfQoKICAvKiBTZXQgdGhlIGRlZmF1bHQgQ1JZUCBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUkVBRFk7CgogIC8qIFJlc2V0IENyeXBJbkNvdW50IGFuZCBDcnlwT3V0Q291bnQgKi8KICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwOwogIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwOwoKICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgaWYoaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID09IE5VTEwpCiAgewogICAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwRGVJbml0OyAvKiBMZWdhY3kgd2VhayBNc3BEZUluaXQgICovCiAgfQogIC8qIERlSW5pdCB0aGUgbG93IGxldmVsIGhhcmR3YXJlICovCiAgaGNyeXAtPk1zcERlSW5pdENhbGxiYWNrKGhjcnlwKTsKCiNlbHNlCiAgLyogRGVJbml0IHRoZSBsb3cgbGV2ZWwgaGFyZHdhcmU6IENMT0NLLCBOVklDLiovCiAgSEFMX0NSWVBfTXNwRGVJbml0KGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KCiAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVTRVQ7CgogIC8qIFJlbGVhc2UgTG9jayAqLwogIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgovKioKICAqIEBicmllZiAgQ29uZmlndXJlIHRoZSBDUllQIGFjY29yZGluZyB0byB0aGUgc3BlY2lmaWVkCiAgKiAgICAgICAgIHBhcmFtZXRlcnMgaW4gdGhlIENSWVBfQ29uZmlnVHlwZURlZgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUKICAqIEBwYXJhbSAgcENvbmY6IHBvaW50ZXIgdG8gYSBDUllQX0NvbmZpZ1R5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9TZXRDb25maWcoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgQ1JZUF9Db25maWdUeXBlRGVmICpwQ29uZiApCnsKICAvKiBDaGVjayB0aGUgQ1JZUCBoYW5kbGUgYWxsb2NhdGlvbiAqLwogIGlmKChoY3J5cCA9PSBOVUxMKXx8IChwQ29uZiA9PSBOVUxMKSApCiAgewogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9CgogIC8qIENoZWNrIHBhcmFtZXRlcnMgKi8KICBhc3NlcnRfcGFyYW0oSVNfQ1JZUF9LRVlTSVpFKHBDb25mLT5LZXlTaXplKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfREFUQVRZUEUocENvbmYtPkRhdGFUeXBlKSk7CiAgYXNzZXJ0X3BhcmFtKElTX0NSWVBfQUxHT1JJVEhNKHBDb25mLT5BbGdvcml0aG0pKTsKCiAgaWYoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogU2V0ICBDUllQIHBhcmFtZXRlcnMgICovCiAgICBoY3J5cC0+SW5pdC5EYXRhVHlwZSAgICAgPSBwQ29uZi0+RGF0YVR5cGU7CiAgICBoY3J5cC0+SW5pdC5wS2V5ICAgICAgICAgPSBwQ29uZi0+cEtleTsKICAgIGhjcnlwLT5Jbml0LkFsZ29yaXRobSAgICA9IHBDb25mLT5BbGdvcml0aG07CiAgICBoY3J5cC0+SW5pdC5LZXlTaXplICAgICAgPSBwQ29uZi0+S2V5U2l6ZTsKICAgIGhjcnlwLT5Jbml0LnBJbml0VmVjdCAgICA9IHBDb25mLT5wSW5pdFZlY3Q7CgogICAgLyogU2V0IHRoZSBrZXkgc2l6ZShUaGlzIGJpdCBmaWVsZCBpcyCRZG9uknQgY2FyZZIgaW4gdGhlIERFUyBvciBUREVTIG1vZGVzKSBkYXRhIHR5cGUsIEFsZ29Nb2RlIGFuZCBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfREFUQVRZUEV8Q1JZUF9DUl9LRVlTSVpFfENSWVBfQ1JfQUxHT01PREUsIGhjcnlwLT5Jbml0LkRhdGFUeXBlIHwgaGNyeXAtPkluaXQuS2V5U2l6ZSB8IGhjcnlwLT5Jbml0LkFsZ29yaXRobSk7CgogICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBSZXNldCBFcnJvciBDb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlID0gSEFMX0NSWVBfRVJST1JfTk9ORTsKCiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgIC8qIFNldCB0aGUgZGVmYXVsdCBDUllQIHBoYXNlICovCiAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1JFQURZOwoKICAgIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICAgIHJldHVybiBIQUxfT0s7CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIFVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgcmV0dXJuIEhBTF9FUlJPUjsKICB9Cn0KCi8qKgogICogQGJyaWVmICBHZXQgQ1JZUCBDb25maWd1cmF0aW9uIHBhcmFtZXRlcnMgaW4gYXNzb2NpYXRlZCBoYW5kbGUuCiAgKiBAcGFyYW0gIHBDb25mOiBwb2ludGVyIHRvIGEgQ1JZUF9Db25maWdUeXBlRGVmIHN0cnVjdHVyZQogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0dldENvbmZpZyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCBDUllQX0NvbmZpZ1R5cGVEZWYgKnBDb25mICkKewogIC8qIENoZWNrIHRoZSBDUllQIGhhbmRsZSBhbGxvY2F0aW9uICovCiAgaWYoKGhjcnlwID09IE5VTEwpfHwgKHBDb25mID09IE5VTEwpICkKICB7CiAgICByZXR1cm4gSEFMX0VSUk9SOwogIH0KCiAgaWYoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX1JFQURZKQogIHsKICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogR2V0ICBDUllQIHBhcmFtZXRlcnMgICovCiAgICBwQ29uZi0+RGF0YVR5cGUgICAgICAgID0gaGNyeXAtPkluaXQuRGF0YVR5cGU7CiAgICBwQ29uZi0+cEtleSAgICAgICAgICAgID0gaGNyeXAtPkluaXQucEtleTsKICAgIHBDb25mLT5BbGdvcml0aG0gICAgICAgPSBoY3J5cC0+SW5pdC5BbGdvcml0aG07CiAgICBwQ29uZi0+S2V5U2l6ZSAgICAgICAgID0gaGNyeXAtPkluaXQuS2V5U2l6ZSA7CiAgICBwQ29uZi0+cEluaXRWZWN0ICAgICAgID0gaGNyeXAtPkluaXQucEluaXRWZWN0OwoKICAgIC8qIFByb2Nlc3MgVW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgICByZXR1cm4gSEFMX09LOwogIH0KICBlbHNlCiAgewogICAgLyogUHJvY2VzcyBVbmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHJldHVybiBIQUxfRVJST1I7CiAgfQp9Ci8qKgogICogQGJyaWVmICBJbml0aWFsaXplcyB0aGUgQ1JZUCBNU1AuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBOb25lCiAgKi8KX193ZWFrIHZvaWQgSEFMX0NSWVBfTXNwSW5pdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgLyogUHJldmVudCB1bnVzZWQgYXJndW1lbnQocykgY29tcGlsYXRpb24gd2FybmluZyAqLwogIFVOVVNFRChoY3J5cCk7CgogIC8qIE5PVEUgOiBUaGlzIGZ1bmN0aW9uIFNob3VsZCBub3QgYmUgbW9kaWZpZWQsIHdoZW4gdGhlIGNhbGxiYWNrIGlzIG5lZWRlZCwKICAgICAgICAgICAgdGhlIEhBTF9DUllQX01zcEluaXQgY291bGQgYmUgaW1wbGVtZW50ZWQgaW4gdGhlIHVzZXIgZmlsZQogICAqLwp9CgovKioKICAqIEBicmllZiAgRGVJbml0aWFsaXplcyBDUllQIE1TUC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpfX3dlYWsgdm9pZCBIQUxfQ1JZUF9Nc3BEZUluaXQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIC8qIFByZXZlbnQgdW51c2VkIGFyZ3VtZW50KHMpIGNvbXBpbGF0aW9uIHdhcm5pbmcgKi8KICBVTlVTRUQoaGNyeXApOwoKICAvKiBOT1RFIDogVGhpcyBmdW5jdGlvbiBTaG91bGQgbm90IGJlIG1vZGlmaWVkLCB3aGVuIHRoZSBjYWxsYmFjayBpcyBuZWVkZWQsCiAgICAgICAgICAgIHRoZSBIQUxfQ1JZUF9Nc3BEZUluaXQgY291bGQgYmUgaW1wbGVtZW50ZWQgaW4gdGhlIHVzZXIgZmlsZQogICAqLwp9CgojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKLyoqCiAgKiBAYnJpZWYgIFJlZ2lzdGVyIGEgVXNlciBDUllQIENhbGxiYWNrCiAgKiAgICAgICAgIFRvIGJlIHVzZWQgaW5zdGVhZCBvZiB0aGUgd2VhayBwcmVkZWZpbmVkIGNhbGxiYWNrCiAgKiBAcGFyYW0gaGNyeXAgY3J5cCBoYW5kbGUKICAqIEBwYXJhbSBDYWxsYmFja0lEIElEIG9mIHRoZSBjYWxsYmFjayB0byBiZSByZWdpc3RlcmVkCiAgKiAgICAgICAgVGhpcyBwYXJhbWV0ZXIgY2FuIGJlIG9uZSBvZiB0aGUgZm9sbG93aW5nIHZhbHVlczoKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9JTlBVVF9DT01QTEVURV9DQl9JRCBJbnB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX09VVFBVVF9DT01QTEVURV9DQl9JRCBPdXRwdXQgRklGTyB0cmFuc2ZlciBjb21wbGV0ZWQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9FUlJPUl9DQl9JRCBSeCBIYWxmIEVycm9yIGNhbGxiYWNrIElECiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfTVNQSU5JVF9DQl9JRCBNc3BJbml0IGNhbGxiYWNrIElECiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfTVNQREVJTklUX0NCX0lEIE1zcERlSW5pdCBjYWxsYmFjayBJRAogICogQHBhcmFtIHBDYWxsYmFjayBwb2ludGVyIHRvIHRoZSBDYWxsYmFjayBmdW5jdGlvbgogICogQHJldHZhbCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9SZWdpc3RlckNhbGxiYWNrKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIEhBTF9DUllQX0NhbGxiYWNrSURUeXBlRGVmIENhbGxiYWNrSUQsIHBDUllQX0NhbGxiYWNrVHlwZURlZiBwQ2FsbGJhY2spCnsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CgogIGlmKHBDYWxsYmFjayA9PSBOVUxMKQogIHsKICAgIC8qIFVwZGF0ZSB0aGUgZXJyb3IgY29kZSAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwoKICAgIHJldHVybiBIQUxfRVJST1I7CiAgfQogIC8qIFByb2Nlc3MgbG9ja2VkICovCiAgX19IQUxfTE9DSyhoY3J5cCk7CgogIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICBzd2l0Y2ggKENhbGxiYWNrSUQpCiAgICB7CiAgICBjYXNlIEhBTF9DUllQX0lOUFVUX0NPTVBMRVRFX0NCX0lEIDoKICAgICAgaGNyeXAtPkluQ3BsdENhbGxiYWNrID0gcENhbGxiYWNrOwogICAgICBicmVhazsKCiAgICBjYXNlIEhBTF9DUllQX09VVFBVVF9DT01QTEVURV9DQl9JRCA6CiAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgIGJyZWFrOwoKICAgIGNhc2UgSEFMX0NSWVBfRVJST1JfQ0JfSUQgOgogICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgYnJlYWs7CgogICAgY2FzZSBIQUxfQ1JZUF9NU1BJTklUX0NCX0lEIDoKICAgICAgaGNyeXAtPk1zcEluaXRDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgYnJlYWs7CgogICAgY2FzZSBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgOgogICAgICBoY3J5cC0+TXNwRGVJbml0Q2FsbGJhY2sgPSBwQ2FsbGJhY2s7CiAgICAgIGJyZWFrOwoKICAgIGRlZmF1bHQgOgogICAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRVNFVCkKICB7CiAgICBzd2l0Y2ggKENhbGxiYWNrSUQpCiAgICB7CiAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrID0gcENhbGxiYWNrOwogICAgICBicmVhazsKCiAgICBjYXNlIEhBTF9DUllQX01TUERFSU5JVF9DQl9JRCA6CiAgICAgIGhjcnlwLT5Nc3BEZUluaXRDYWxsYmFjayA9IHBDYWxsYmFjazsKICAgICAgYnJlYWs7CgogICAgZGVmYXVsdCA6CiAgICAgIC8qIFVwZGF0ZSB0aGUgZXJyb3IgY29kZSAqLwogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CiAgICAgIC8qIFJldHVybiBlcnJvciBzdGF0dXMgKi8KICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgYnJlYWs7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfSU5WQUxJRF9DQUxMQkFDSzsKICAgIC8qIFJldHVybiBlcnJvciBzdGF0dXMgKi8KICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgfQoKICAvKiBSZWxlYXNlIExvY2sgKi8KICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICByZXR1cm4gc3RhdHVzOwp9CgovKioKICAqIEBicmllZiAgVW5yZWdpc3RlciBhbiBDUllQIENhbGxiYWNrCiAgKiAgICAgICAgIENSWVAgY2FsbGFiY2sgaXMgcmVkaXJlY3RlZCB0byB0aGUgd2VhayBwcmVkZWZpbmVkIGNhbGxiYWNrCiAgKiBAcGFyYW0gaGNyeXAgY3J5cCBoYW5kbGUKICAqIEBwYXJhbSBDYWxsYmFja0lEIElEIG9mIHRoZSBjYWxsYmFjayB0byBiZSB1bnJlZ2lzdGVyZWQKICAqICAgICAgICBUaGlzIHBhcmFtZXRlciBjYW4gYmUgb25lIG9mIHRoZSBmb2xsb3dpbmcgdmFsdWVzOgogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX0lOUFVUX0NPTVBMRVRFX0NCX0lEIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrIElECiAgKiAgICAgICAgICBAYXJnIEByZWYgSEFMX0NSWVBfT1VUUFVUX0NPTVBMRVRFX0NCX0lEIE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjayBJRAogICogICAgICAgICAgQGFyZyBAcmVmIEhBTF9DUllQX0VSUk9SX0NCX0lEIFJ4IEhhbGYgRXJyb3IgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9NU1BJTklUX0NCX0lEIE1zcEluaXQgY2FsbGJhY2sgSUQKICAqICAgICAgICAgIEBhcmcgQHJlZiBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgTXNwRGVJbml0IGNhbGxiYWNrIElECiAgKiBAcmV0dmFsIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX1VuUmVnaXN0ZXJDYWxsYmFjayhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCBIQUxfQ1JZUF9DYWxsYmFja0lEVHlwZURlZiBDYWxsYmFja0lEKQp7CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogIF9fSEFMX0xPQ0soaGNyeXApOwoKICBpZihoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgc3dpdGNoIChDYWxsYmFja0lEKQogICAgewogICAgY2FzZSBIQUxfQ1JZUF9JTlBVVF9DT01QTEVURV9DQl9JRCA6CiAgICAgIGhjcnlwLT5JbkNwbHRDYWxsYmFjayA9IEhBTF9DUllQX0luQ3BsdENhbGxiYWNrOyAgLyogTGVnYWN5IHdlYWsgIEluQ3BsdENhbGxiYWNrICAqLwogICAgICBicmVhazsKCiAgICBjYXNlIEhBTF9DUllQX09VVFBVVF9DT01QTEVURV9DQl9JRCA6CiAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2s7ICAgICAgICAgLyogTGVnYWN5IHdlYWsgT3V0Q3BsdENhbGxiYWNrICAgICAgICovCiAgICAgIGJyZWFrOwoKICAgIGNhc2UgSEFMX0NSWVBfRVJST1JfQ0JfSUQgOgogICAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayA9IEhBTF9DUllQX0Vycm9yQ2FsbGJhY2s7ICAgICAgICAgICAvKiBMZWdhY3kgd2VhayBFcnJvckNhbGxiYWNrICAgICAgICAqLwogICAgICBicmVhazsKCiAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwSW5pdDsKICAgICAgYnJlYWs7CgogICAgY2FzZSBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgOgogICAgICBoY3J5cC0+TXNwRGVJbml0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9Nc3BEZUluaXQ7CiAgICAgIGJyZWFrOwoKICAgIGRlZmF1bHQgOgogICAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRVNFVCkKICB7CiAgICBzd2l0Y2ggKENhbGxiYWNrSUQpCiAgICB7CiAgICBjYXNlIEhBTF9DUllQX01TUElOSVRfQ0JfSUQgOgogICAgICBoY3J5cC0+TXNwSW5pdENhbGxiYWNrID0gSEFMX0NSWVBfTXNwSW5pdDsKICAgICAgYnJlYWs7CgogICAgY2FzZSBIQUxfQ1JZUF9NU1BERUlOSVRfQ0JfSUQgOgogICAgICBoY3J5cC0+TXNwRGVJbml0Q2FsbGJhY2sgPSBIQUxfQ1JZUF9Nc3BEZUluaXQ7CiAgICAgIGJyZWFrOwoKICAgIGRlZmF1bHQgOgogICAgICAvKiBVcGRhdGUgdGhlIGVycm9yIGNvZGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9JTlZBTElEX0NBTExCQUNLOwogICAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogVXBkYXRlIHRoZSBlcnJvciBjb2RlICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0lOVkFMSURfQ0FMTEJBQ0s7CiAgICAvKiBSZXR1cm4gZXJyb3Igc3RhdHVzICovCiAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogIH0KCiAgLyogUmVsZWFzZSBMb2NrICovCiAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgcmV0dXJuIHN0YXR1czsKfQojZW5kaWYgLyogVVNFX0hBTF9VQVJUX1JFR0lTVEVSX0NBTExCQUNLUyAqLwoKLyoqCiAgKiBAfQogICovCgovKiogQGRlZmdyb3VwIENSWVBfRXhwb3J0ZWRfRnVuY3Rpb25zX0dyb3VwMiAgRW5jcnlwdCBEZWNyeXB0IGZ1bmN0aW9ucwogKiAgQGJyaWVmICAgQ1JZUCBwcm9jZXNzaW5nIGZ1bmN0aW9ucy4KICoKQHZlcmJhdGltCiAgPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICAgICAgICAgICAgICAgICAgICAjIyMjIyBFbmNyeXB0IERlY3J5cHQgIGZ1bmN0aW9ucyAjIyMjIwogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQogICAgWy4uXSAgVGhpcyBzZWN0aW9uIHByb3ZpZGVzIEFQSSBhbGxvd2luZyB0byBFbmNyeXB0L0RlY3J5cHQgRGF0YSBmb2xsb3dpbmcKICAgICAgICAgIFN0YW5kYXJkIERFUy9UREVTIG9yIEFFUywgYW5kIEFsZ29yaXRobSBjb25maWd1cmVkIGJ5IHRoZSB1c2VyOgogICAgICAoKykgU3RhbmRhcmQgREVTL1RERVMgb25seSBzdXBwb3J0ZWQgYnkgQ1JZUDEgSVAsIGJlbG93IGxpc3Qgb2YgQWxnb3JpdGhtIHN1cHBvcnRlZCA6CiAgICAgICAgICAgKCsrKSAgRWxlY3Ryb25pYyBDb2RlIEJvb2soRUNCKQogICAgICAgICAgICgrKykgQ2lwaGVyIEJsb2NrIENoYWluaW5nIChDQkMpCiAgICAgICgrKSBTdGFuZGFyZCBBRVMgIHN1cHBvcnRlZCBieSBDUllQMSBJUCAsIGxpc3Qgb2YgQWxnb3JpdGhtIHN1cHBvcnRlZDoKICAgICAgICAgICAoKyspIEVsZWN0cm9uaWMgQ29kZSBCb29rKEVDQikKICAgICAgICAgICAoKyspIENpcGhlciBCbG9jayBDaGFpbmluZyAoQ0JDKQogICAgICAgICAgICgrKykgQ291bnRlciBtb2RlIChDVFIpCiAgICAgICAgICAgKCsrKSBDaXBoZXIgQmxvY2sgQ2hhaW5pbmcgKENCQykKICAgICAgICAgICAoKyspIENvdW50ZXIgbW9kZSAoQ1RSKQogICAgWy4uXSAgVGhyZWUgcHJvY2Vzc2luZyBmdW5jdGlvbnMgYXJlIGF2YWlsYWJsZToKICAgICAgKCspIFBvbGxpbmcgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHQgJiBIQUxfQ1JZUF9EZWNyeXB0CiAgICAgICgrKSBJbnRlcnJ1cHQgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHRfSVQgJiBIQUxfQ1JZUF9EZWNyeXB0X0lUCiAgICAgICgrKSBETUEgbW9kZSA6IEhBTF9DUllQX0VuY3J5cHRfRE1BICYgSEFMX0NSWVBfRGVjcnlwdF9ETUEKCkBlbmR2ZXJiYXRpbQogICogQHsKICAqLwoKCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChwbGFpbnRleHQpCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBpbiB3b3JkLgogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIoY2lwaGVydGV4dCkKICAqIEBwYXJhbSAgVGltZW91dDogU3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9FbmNyeXB0KENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzOwoKICBpZihoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyLCBwQ3J5cE91dEJ1ZmZQdHIgYW5kIFNpemUgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgogICAgLyogU2V0IEVuY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9FTkNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERTsKCiAgICBzd2l0Y2goYWxnbykKICAgIHsKICAgIGNhc2UgQ1JZUF9ERVNfRUNCOgogICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICBjYXNlIENSWVBfVERFU19FQ0I6CiAgICBjYXNlIENSWVBfVERFU19DQkM6CgogICAgICAvKlNldCBLZXkgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSsxKTsKICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzIpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzMpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzQpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzUpOwogICAgICB9CgogICAgICAvKlNldCBJbml0aWFsaXphdGlvbiBWZWN0b3IgKElWKSovCiAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfREVTX0NCQykgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsxKTsKICAgICAgfQoKICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgLyogU3RhdHJ0IERFUy9UREVTIGVuY3J5cHRpb24gcHJvY2VzcyAqLwogICAgICBzdGF0dXMgPSBDUllQX1RERVNfUHJvY2VzcyhoY3J5cCxUaW1lb3V0KTsKICAgICAgYnJlYWs7CgogICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgLyogQUVTIGVuY3J5cHRpb24gKi8KICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRW5jcnlwdChoY3J5cCwgVGltZW91dCk7CiAgICAgIGJyZWFrOwoKICAgIGRlZmF1bHQ6CiAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgc3RhdHVzID0gSEFMX0VSUk9SOwogICAgICBicmVhazsKICAgIH0KCiAgICBpZiAoc3RhdHVzID09IEhBTF9PSykKICAgIHsKICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgc3RhdHVzID0gSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXMgOwp9CgovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBtb2RlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgSW5wdXQ6IFBvaW50ZXIgdG8gdGhlIGlucHV0IGJ1ZmZlciAoY2lwaGVydGV4dCApCiAgKiBAcGFyYW0gIFNpemU6IExlbmd0aCBvZiB0aGUgcGxhaW50ZXh0IGJ1ZmZlciBpbiB3b3JkLgogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIocGxhaW50ZXh0KQogICogQHBhcmFtICBUaW1lb3V0OiBTcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0RlY3J5cHQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgKklucHV0LCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCAqT3V0cHV0LCB1aW50MzJfdCBUaW1lb3V0KQp7CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzOwogIHVpbnQzMl90IGFsZ287CgogIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2Ugc3RhdGUgQnVzeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogIFJlc2V0IENyeXBJbkNvdW50LCBDcnlwT3V0Q291bnQgYW5kIEluaXRpYWxpemUgcENyeXBJbkJ1ZmZQdHIsIHBDcnlwT3V0QnVmZlB0ciBhbmQgU2l6ZSBwYXJhbWV0ZXJzKi8KICAgIGhjcnlwLT5DcnlwSW5Db3VudCA9IDBVOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IDBVOwogICAgaGNyeXAtPnBDcnlwSW5CdWZmUHRyID0gSW5wdXQ7CiAgICBoY3J5cC0+cENyeXBPdXRCdWZmUHRyID0gT3V0cHV0OwoKICAgIC8qICBDYWxjdWxhdGUgU2l6ZSBwYXJhbWV0ZXIgaW4gQnl0ZSovCiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdCA9PSBDUllQX0RBVEFXSURUSFVOSVRfV09SRCkKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplICogNFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZTsKICAgIH0KCiAgICAvKiBTZXQgRGVjcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT0RJUiwgQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09NT0RFOwoKICAgIHN3aXRjaChhbGdvKQogICAgewogICAgY2FzZSBDUllQX0RFU19FQ0I6CiAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgIGNhc2UgQ1JZUF9UREVTX0NCQzoKCiAgICAgIC8qU2V0IEtleSAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzEpOwogICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfRUNCKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMik7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMyk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNCk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNSk7CiAgICAgIH0KCiAgICAgIC8qU2V0IEluaXRpYWxpemF0aW9uIFZlY3RvciAoSVYpKi8KICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzEpOwogICAgICB9CgogICAgICAvKiBGbHVzaCBGSUZPICovCiAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgICAvKiBTdGFydCBERVMvVERFUyBkZWNyeXB0aW9uIHByb2Nlc3MgKi8KICAgICAgc3RhdHVzID0gQ1JZUF9UREVTX1Byb2Nlc3MoaGNyeXAsIFRpbWVvdXQpOwoKICAgICAgYnJlYWs7CgogICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgLyogQUVTIGRlY3J5cHRpb24gKi8KICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRGVjcnlwdChoY3J5cCwgVGltZW91dCk7CiAgICAgIGJyZWFrOwoKICAgIGRlZmF1bHQ6CiAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgc3RhdHVzID0gSEFMX0VSUk9SOwogICAgICBicmVhazsKICAgIH0KCiAgICBpZiAoc3RhdHVzID09IEhBTF9PSykKICAgIHsKICAgICAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwoKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwogICAgc3RhdHVzID0gSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXM7Cn0KCi8qKgogICogQGJyaWVmICBFbmNyeXB0aW9uIGluIGludGVycnVwdCBtb2RlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUKICAqIEBwYXJhbSAgSW5wdXQ6IFBvaW50ZXIgdG8gdGhlIGlucHV0IGJ1ZmZlciAocGxhaW50ZXh0KQogICogQHBhcmFtICBTaXplOiBMZW5ndGggb2YgdGhlIHBsYWludGV4dCBidWZmZXIgaW4gd29yZAogICogQHBhcmFtICBPdXRwdXQ6IFBvaW50ZXIgdG8gdGhlIG91dHB1dCBidWZmZXIoY2lwaGVydGV4dCkKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0VuY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgKklucHV0LCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCAqT3V0cHV0KQp7CiAgdWludDMyX3QgYWxnbzsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXM7CgogIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2Ugc3RhdGUgQnVzeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogIFJlc2V0IENyeXBJbkNvdW50LCBDcnlwT3V0Q291bnQgYW5kIEluaXRpYWxpemUgcENyeXBJbkJ1ZmZQdHIsIHBDcnlwT3V0QnVmZlB0ciBhbmQgU2l6ZSBwYXJhbWV0ZXJzKi8KICAgIGhjcnlwLT5DcnlwSW5Db3VudCA9IDBVOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IDBVOwogICAgaGNyeXAtPnBDcnlwSW5CdWZmUHRyID0gSW5wdXQ7CiAgICBoY3J5cC0+cENyeXBPdXRCdWZmUHRyID0gT3V0cHV0OwoKICAgIC8qICBDYWxjdWxhdGUgU2l6ZSBwYXJhbWV0ZXIgaW4gQnl0ZSovCiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdCA9PSBDUllQX0RBVEFXSURUSFVOSVRfV09SRCkKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplICogNFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZTsKICAgIH0KCiAgICAvKiBTZXQgZW5jcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT0RJUiwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IChoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERSk7CgogICAgc3dpdGNoKGFsZ28pCiAgICB7CiAgICBjYXNlIENSWVBfREVTX0VDQjoKICAgIGNhc2UgQ1JZUF9ERVNfQ0JDOgogICAgY2FzZSBDUllQX1RERVNfRUNCOgogICAgY2FzZSBDUllQX1RERVNfQ0JDOgoKICAgICAgLypTZXQgS2V5ICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMVJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMSk7CiAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19FQ0IpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0NCQykpCiAgICAgIHsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSsyKTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSszKTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSs0KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPkszUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSs1KTsKICAgICAgfQogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfREVTX0NCQykgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsxKTsKICAgICAgfQoKICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgLyogRW5hYmxlIGludGVycnVwdHMgKi8KICAgICAgX19IQUxfQ1JZUF9FTkFCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JIHwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgIC8qIEVuYWJsZSBDUllQIHRvIHN0YXJ0IERFUy9UREVTIHByb2Nlc3MqLwogICAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgICBzdGF0dXMgPSBIQUxfT0s7CiAgICAgIGJyZWFrOwoKICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICBjYXNlIENSWVBfQUVTX0NUUjoKCiAgICAgIHN0YXR1cyA9IENSWVBfQUVTX0VuY3J5cHRfSVQoaGNyeXApOwogICAgICBicmVhazsKCiAgICBkZWZhdWx0OgogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICBzdGF0dXMgPSAgSEFMX0VSUk9SOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBzdGF0dXMgOwp9CgovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBpdG50ZXJydXB0IG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChjaXBoZXJ0ZXh0ICkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGluIHdvcmQuCiAgKiBAcGFyYW0gIE91dHB1dDogUG9pbnRlciB0byB0aGUgb3V0cHV0IGJ1ZmZlcihwbGFpbnRleHQpCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpIQUxfU3RhdHVzVHlwZURlZiBIQUxfQ1JZUF9EZWNyeXB0X0lUKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICBpZihoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyLCBwQ3J5cE91dEJ1ZmZQdHIgYW5kIFNpemUgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgogICAgLyogU2V0IGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsQ1JZUF9PUEVSQVRJTkdNT0RFX0RFQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09NT0RFOwoKICAgIHN3aXRjaChhbGdvKQogICAgewogICAgY2FzZSBDUllQX0RFU19FQ0I6CiAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgIGNhc2UgQ1JZUF9UREVTX0NCQzoKCiAgICAgIC8qU2V0IEtleSAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzEpOwogICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfRUNCKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMik7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMyk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNCk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNSk7CiAgICAgIH0KCiAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzEpOwogICAgICB9CiAgICAgIC8qIEZsdXNoIEZJRk8gKi8KICAgICAgSEFMX0NSWVBfRklGT19GTFVTSChoY3J5cCk7CgogICAgICAvKiBTZXQgdGhlIHBoYXNlICovCiAgICAgIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgICAgIC8qIEVuYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgIF9fSEFMX0NSWVBfRU5BQkxFX0lUKGhjcnlwLCBDUllQX0lUX0lOSSB8IENSWVBfSVRfT1VUSSk7CgogICAgICAvKiBFbmFibGUgQ1JZUCBhbmQgc3RhcnQgREVTL1RERVMgcHJvY2VzcyovCiAgICAgIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgICAgIGJyZWFrOwoKICAgIGNhc2UgQ1JZUF9BRVNfRUNCOgogICAgY2FzZSBDUllQX0FFU19DQkM6CiAgICBjYXNlIENSWVBfQUVTX0NUUjoKCiAgICAgIC8qIEFFUyBkZWNyeXB0aW9uICovCiAgICAgIHN0YXR1cyA9IENSWVBfQUVTX0RlY3J5cHRfSVQoaGNyeXApOwogICAgICBicmVhazsKCiAgICBkZWZhdWx0OgogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgIHN0YXR1cyA9IEhBTF9FUlJPUjsKICAgICAgYnJlYWs7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHN0YXR1cyA9IEhBTF9FUlJPUjsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gc3RhdHVzOwp9CgovKioKICAqIEBicmllZiAgRW5jcnlwdGlvbiBpbiBETUEgbW9kZS4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIElucHV0OiBQb2ludGVyIHRvIHRoZSBpbnB1dCBidWZmZXIgKHBsYWludGV4dCkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGluIHdvcmQuCiAgKiBAcGFyYW0gIE91dHB1dDogUG9pbnRlciB0byB0aGUgb3V0cHV0IGJ1ZmZlcihjaXBoZXJ0ZXh0KQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8KSEFMX1N0YXR1c1R5cGVEZWYgSEFMX0NSWVBfRW5jcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgKklucHV0LCB1aW50MTZfdCBTaXplLCB1aW50MzJfdCAqT3V0cHV0KQp7CiAgdWludDMyX3QgYWxnbzsKICBIQUxfU3RhdHVzVHlwZURlZiBzdGF0dXMgPSBIQUxfT0s7CgogIGlmKGhjcnlwLT5TdGF0ZSA9PSBIQUxfQ1JZUF9TVEFURV9SRUFEWSkKICB7CiAgICAvKiBDaGFuZ2Ugc3RhdGUgQnVzeSAqLwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfQlVTWTsKCiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfTE9DSyhoY3J5cCk7CgogICAgLyogIFJlc2V0IENyeXBJbkNvdW50LCBDcnlwT3V0Q291bnQgYW5kIEluaXRpYWxpemUgcENyeXBJbkJ1ZmZQdHIsIHBDcnlwT3V0QnVmZlB0ciBhbmQgU2l6ZSBwYXJhbWV0ZXJzKi8KICAgIGhjcnlwLT5DcnlwSW5Db3VudCA9IDBVOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCA9IDBVOwogICAgaGNyeXAtPnBDcnlwSW5CdWZmUHRyID0gSW5wdXQ7CiAgICBoY3J5cC0+cENyeXBPdXRCdWZmUHRyID0gT3V0cHV0OwoKICAgIC8qICBDYWxjdWxhdGUgU2l6ZSBwYXJhbWV0ZXIgaW4gQnl0ZSovCiAgICBpZiAoaGNyeXAtPkluaXQuRGF0YVdpZHRoVW5pdCA9PSBDUllQX0RBVEFXSURUSFVOSVRfV09SRCkKICAgIHsKICAgICAgaGNyeXAtPlNpemUgPSBTaXplICogNFU7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZTsKICAgIH0KCiAgICAvKiBTZXQgZW5jcnlwdGlvbiBvcGVyYXRpbmcgbW9kZSovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT0RJUiwgQ1JZUF9PUEVSQVRJTkdNT0RFX0VOQ1JZUFQpOwoKICAgIC8qIGFsZ28gZ2V0IGFsZ29yaXRobSBzZWxlY3RlZCAqLwogICAgYWxnbyA9IGhjcnlwLT5JbnN0YW5jZS0+Q1IgJiBDUllQX0NSX0FMR09NT0RFOwoKICAgIHN3aXRjaChhbGdvKQogICAgewogICAgY2FzZSBDUllQX0RFU19FQ0I6CiAgICBjYXNlIENSWVBfREVTX0NCQzoKICAgIGNhc2UgQ1JZUF9UREVTX0VDQjoKICAgIGNhc2UgQ1JZUF9UREVTX0NCQzoKCiAgICAgIC8qU2V0IEtleSAqLwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSk7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzEpOwogICAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfRUNCKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMik7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMyk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM0xSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNCk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNSk7CiAgICAgIH0KCiAgICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9ERVNfQ0JDKSB8fCAoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfVERFU19DQkMpKQogICAgICB7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzEpOwogICAgICB9CgogICAgICAvKiBGbHVzaCBGSUZPICovCiAgICAgIEhBTF9DUllQX0ZJRk9fRkxVU0goaGNyeXApOwoKICAgICAgLyogU2V0IHRoZSBwaGFzZSAqLwogICAgICBoY3J5cC0+UGhhc2UgPSBDUllQX1BIQVNFX1BST0NFU1M7CgogICAgICAvKiBTdGFydCBETUEgcHJvY2VzcyB0cmFuc2ZlciBmb3IgREVTL1RERVMgKi8KICAgICAgQ1JZUF9TZXRETUFDb25maWcoaGNyeXAsICh1aW50MzJfdCkoIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciksIChoY3J5cC0+U2l6ZS80VSksICh1aW50MzJfdCkoaGNyeXAtPnBDcnlwT3V0QnVmZlB0cikpOwoKICAgICAgYnJlYWs7CgogICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgLyogIFNldCB0aGUgS2V5Ki8KICAgICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IgSVYgKi8KICAgICAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgICAgIHsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMExSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMSk7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMik7CiAgICAgICAgaGNyeXAtPkluc3RhbmNlLT5JVjFSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMyk7CiAgICAgIH0KCiAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgLyogU3RhcnQgRE1BIHByb2Nlc3MgdHJhbnNmZXIgZm9yIEFFUyAqLwogICAgICBDUllQX1NldERNQUNvbmZpZyhoY3J5cCwgKHVpbnQzMl90KSggaGNyeXAtPnBDcnlwSW5CdWZmUHRyKSwgKGhjcnlwLT5TaXplLzRVKSwgKHVpbnQzMl90KShoY3J5cC0+cENyeXBPdXRCdWZmUHRyKSk7CiAgICAgIGJyZWFrOwoKICAgIGRlZmF1bHQ6CiAgICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfTk9UX1NVUFBPUlRFRDsKICAgICAgc3RhdHVzID0gIEhBTF9FUlJPUjsKICAgICAgYnJlYWs7CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBCdXN5IGVycm9yIGNvZGUgZmllbGQgKi8KICAgIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfQlVTWTsKICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIHN0YXR1czsKfQoKLyoqCiAgKiBAYnJpZWYgIERlY3J5cHRpb24gaW4gRE1BIG1vZGUuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBJbnB1dDogUG9pbnRlciB0byB0aGUgaW5wdXQgYnVmZmVyIChjaXBoZXJ0ZXh0ICkKICAqIEBwYXJhbSAgU2l6ZTogTGVuZ3RoIG9mIHRoZSBwbGFpbnRleHQgYnVmZmVyIGluIHdvcmQKICAqIEBwYXJhbSAgT3V0cHV0OiBQb2ludGVyIHRvIHRoZSBvdXRwdXQgYnVmZmVyKHBsYWludGV4dCkKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCkhBTF9TdGF0dXNUeXBlRGVmIEhBTF9DUllQX0RlY3J5cHRfRE1BKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90ICpJbnB1dCwgdWludDE2X3QgU2l6ZSwgdWludDMyX3QgKk91dHB1dCkKewogIHVpbnQzMl90IGFsZ287CiAgSEFMX1N0YXR1c1R5cGVEZWYgc3RhdHVzID0gSEFMX09LOwoKICBpZihoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfUkVBRFkpCiAgewogICAgLyogQ2hhbmdlIHN0YXRlIEJ1c3kgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX0JVU1k7CgogICAgLyogUHJvY2VzcyBsb2NrZWQgKi8KICAgIF9fSEFMX0xPQ0soaGNyeXApOwoKICAgIC8qICBSZXNldCBDcnlwSW5Db3VudCwgQ3J5cE91dENvdW50IGFuZCBJbml0aWFsaXplIHBDcnlwSW5CdWZmUHRyLCBwQ3J5cE91dEJ1ZmZQdHIgYW5kIFNpemUgcGFyYW1ldGVycyovCiAgICBoY3J5cC0+Q3J5cEluQ291bnQgPSAwVTsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQgPSAwVTsKICAgIGhjcnlwLT5wQ3J5cEluQnVmZlB0ciA9IElucHV0OwogICAgaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciA9IE91dHB1dDsKCiAgICAvKiAgQ2FsY3VsYXRlIFNpemUgcGFyYW1ldGVyIGluIEJ5dGUqLwogICAgaWYgKGhjcnlwLT5Jbml0LkRhdGFXaWR0aFVuaXQgPT0gQ1JZUF9EQVRBV0lEVEhVTklUX1dPUkQpCiAgICB7CiAgICAgIGhjcnlwLT5TaXplID0gU2l6ZSAqIDRVOwogICAgfQogICAgZWxzZQogICAgewogICAgICBoY3J5cC0+U2l6ZSA9IFNpemU7CiAgICB9CgogICAgLyogU2V0IGRlY3J5cHRpb24gb3BlcmF0aW5nIG1vZGUqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09ESVIsIENSWVBfT1BFUkFUSU5HTU9ERV9ERUNSWVBUKTsKCiAgICAvKiBhbGdvIGdldCBhbGdvcml0aG0gc2VsZWN0ZWQgKi8KICAgIGFsZ28gPSBoY3J5cC0+SW5zdGFuY2UtPkNSICYgQ1JZUF9DUl9BTEdPTU9ERTsKCiAgICBzd2l0Y2goYWxnbykKICAgIHsKICAgIGNhc2UgQ1JZUF9ERVNfRUNCOgogICAgY2FzZSBDUllQX0RFU19DQkM6CiAgICBjYXNlIENSWVBfVERFU19FQ0I6CiAgICBjYXNlIENSWVBfVERFU19DQkM6CgogICAgICAvKlNldCBLZXkgKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSsxKTsKICAgICAgaWYgKChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzIpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzMpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzQpOwogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzUpOwogICAgICB9CgogICAgICAvKiBTZXQgdGhlIEluaXRpYWxpemF0aW9uIFZlY3RvciovCiAgICAgIGlmICgoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfREVTX0NCQykgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgICAgewogICAgICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgICAgICBoY3J5cC0+SW5zdGFuY2UtPklWMFJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsxKTsKICAgICAgfQoKICAgICAgLyogRmx1c2ggRklGTyAqLwogICAgICBIQUxfQ1JZUF9GSUZPX0ZMVVNIKGhjcnlwKTsKCiAgICAgIC8qIFNldCB0aGUgcGhhc2UgKi8KICAgICAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICAgICAgLyogU3RhcnQgRE1BIHByb2Nlc3MgdHJhbnNmZXIgZm9yIERFUy9UREVTICovCiAgICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKCBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCAoaGNyeXAtPlNpemUvNFUpLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICAgICAgYnJlYWs7CgogICAgY2FzZSBDUllQX0FFU19FQ0I6CiAgICBjYXNlIENSWVBfQUVTX0NCQzoKICAgIGNhc2UgQ1JZUF9BRVNfQ1RSOgoKICAgICAgLyogQUVTIGRlY3J5cHRpb24gKi8KICAgICAgc3RhdHVzID0gQ1JZUF9BRVNfRGVjcnlwdF9ETUEoaGNyeXApOwogICAgICBicmVhazsKCiAgICBkZWZhdWx0OgogICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX05PVF9TVVBQT1JURUQ7CiAgICAgIHN0YXR1cyA9ICBIQUxfRVJST1I7CiAgICAgIGJyZWFrOwogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiAgICBzdGF0dXMgPSBIQUxfRVJST1I7CiAgfQoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIHN0YXR1czsKfQoKLyoqCiAgKiBAfQogICovCgovKiogQGRlZmdyb3VwIENSWVBfRXhwb3J0ZWRfRnVuY3Rpb25zX0dyb3VwMyBDUllQIElSUSBoYW5kbGVyIG1hbmFnZW1lbnQKICogIEBicmllZiAgICBDUllQIElSUSBoYW5kbGVyLgogKgpAdmVyYmF0aW0KICA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICAgICAgICAgICAgICAgICMjIyMjIENSWVAgSVJRIGhhbmRsZXIgbWFuYWdlbWVudCAjIyMjIwogID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQpbLi5dICBUaGlzIHNlY3Rpb24gcHJvdmlkZXMgQ1JZUCBJUlEgaGFuZGxlciBhbmQgY2FsbGJhY2sgZnVuY3Rpb25zLgogICAgICAoKykgSEFMX0NSWVBfSVJRSGFuZGxlciBDUllQIGludGVycnVwdCByZXF1ZXN0CiAgICAgICgrKSBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrCiAgICAgICgrKSBIQUxfQ1JZUF9PdXRDcGx0Q2FsbGJhY2sgb3V0cHV0IGRhdGEgdHJhbnNmZXIgY29tcGxldGUgY2FsbGJhY2sKICAgICAgKCspIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2sgIENSWVAgZXJyb3IgY2FsbGJhY2sKICAgICAgKCspIEhBTF9DUllQX0dldFN0YXRlIHJldHVybiB0aGUgQ1JZUCBzdGF0ZQogICAgICAoKykgSEFMX0NSWVBfR2V0RXJyb3IgcmV0dXJuIHRoZSBDUllQIGVycm9yIGNvZGUKQGVuZHZlcmJhdGltCiAgKiBAewogICovCgovKioKICAqIEBicmllZiAgVGhpcyBmdW5jdGlvbiBoYW5kbGVzIGNyeXB0b2dyYXBoaWMgaW50ZXJydXB0IHJlcXVlc3QuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBOb25lCiAgKi8Kdm9pZCBIQUxfQ1JZUF9JUlFIYW5kbGVyKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAKICBpZigoX19IQUxfQ1JZUF9HRVRfSVQoaGNyeXAsIENSWVBfSVRfSU5JKSAhPSAwVSkgfHwgKF9fSEFMX0NSWVBfR0VUX0lUKGhjcnlwLCBDUllQX0lUX09VVEkpICE9IDBVKSkKICB7CiAgICBpZiAoKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19FQ0IpfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0RFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9UREVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX1RERVNfQ0JDKSkKICAgIHsKICAgICAgQ1JZUF9UREVTX0lUKGhjcnlwKTsgLyogREVTIG9yIFRERVMqLwogICAgfQogICAgZWxzZSBpZigoaGNyeXAtPkluaXQuQWxnb3JpdGhtID09IENSWVBfQUVTX0VDQikgfHwgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSA9PSBDUllQX0FFU19DQkMpIHx8IChoY3J5cC0+SW5pdC5BbGdvcml0aG0gPT0gQ1JZUF9BRVNfQ1RSKSkKICAgIHsKICAgICAgQ1JZUF9BRVNfSVQoaGNyeXApOyAvKkFFUyovCiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgIC8qIE5vdGhpbmcgdG8gZG8gKi8KICAgIH0KICB9Cn0KCi8qKgogICogQGJyaWVmICBSZXR1cm4gdGhlIENSWVAgZXJyb3IgY29kZS4KICAqIEBwYXJhbSAgaGNyeXAgOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIHRoZSAgQ1JZUCBJUAogICogQHJldHZhbCBDUllQIGVycm9yIGNvZGUKICAqLwp1aW50MzJfdCBIQUxfQ1JZUF9HZXRFcnJvcihDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgcmV0dXJuIGhjcnlwLT5FcnJvckNvZGU7Cn0KCi8qKgogICogQGJyaWVmICBSZXR1cm5zIHRoZSBDUllQIHN0YXRlLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIEhBTCBzdGF0ZQogICovCkhBTF9DUllQX1NUQVRFVHlwZURlZiBIQUxfQ1JZUF9HZXRTdGF0ZShDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgcmV0dXJuIGhjcnlwLT5TdGF0ZTsKfQoKLyoqCiAgKiBAYnJpZWYgIElucHV0IEZJRk8gdHJhbnNmZXIgY29tcGxldGVkIGNhbGxiYWNrLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpfX3dlYWsgdm9pZCBIQUxfQ1JZUF9JbkNwbHRDYWxsYmFjayhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgLyogUHJldmVudCB1bnVzZWQgYXJndW1lbnQocykgY29tcGlsYXRpb24gd2FybmluZyAqLwogIFVOVVNFRChoY3J5cCk7CgogIC8qIE5PVEUgOiBUaGlzIGZ1bmN0aW9uIFNob3VsZCBub3QgYmUgbW9kaWZpZWQsIHdoZW4gdGhlIGNhbGxiYWNrIGlzIG5lZWRlZCwKICAgICAgICAgICAgdGhlIEhBTF9DUllQX0luQ3BsdENhbGxiYWNrIGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIE91dHB1dCBGSUZPIHRyYW5zZmVyIGNvbXBsZXRlZCBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHJldHZhbCBOb25lCiAgKi8KX193ZWFrIHZvaWQgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBQcmV2ZW50IHVudXNlZCBhcmd1bWVudChzKSBjb21waWxhdGlvbiB3YXJuaW5nICovCiAgVU5VU0VEKGhjcnlwKTsKCiAgLyogTk9URSA6IFRoaXMgZnVuY3Rpb24gU2hvdWxkIG5vdCBiZSBtb2RpZmllZCwgd2hlbiB0aGUgY2FsbGJhY2sgaXMgbmVlZGVkLAogICAgICAgICAgICB0aGUgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrIGNvdWxkIGJlIGltcGxlbWVudGVkIGluIHRoZSB1c2VyIGZpbGUKICAgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIENSWVAgZXJyb3IgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEByZXR2YWwgTm9uZQogICovCiBfX3dlYWsgdm9pZCBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXApCnsKICAvKiBQcmV2ZW50IHVudXNlZCBhcmd1bWVudChzKSBjb21waWxhdGlvbiB3YXJuaW5nICovCiAgVU5VU0VEKGhjcnlwKTsKCiAgLyogTk9URSA6IFRoaXMgZnVuY3Rpb24gU2hvdWxkIG5vdCBiZSBtb2RpZmllZCwgd2hlbiB0aGUgY2FsbGJhY2sgaXMgbmVlZGVkLAogICAgICAgICAgICB0aGUgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayBjb3VsZCBiZSBpbXBsZW1lbnRlZCBpbiB0aGUgdXNlciBmaWxlCiAgICovCn0KLyoqCiAgKiBAfQogICovCgovKiBQcml2YXRlIGZ1bmN0aW9ucyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwovKiogQGFkZHRvZ3JvdXAgQ1JZUF9Qcml2YXRlX0Z1bmN0aW9ucwogICogQHsKICAqLwoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRUNCL0NCQyBBbGdvcml0aG0gd2l0aCBERVMvVERFUyBzdGFuZGFyZC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyBIQUxfU3RhdHVzVHlwZURlZiBDUllQX1RERVNfUHJvY2VzcyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0KQp7CgogIHVpbnQzMl90IHRlbXA7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MTZfdCBpbmNvdW50OyAvKiBUZW1wb3JhcnkgQ3J5cEluQ291bnQgVmFsdWUgKi8gCiAgdWludDE2X3Qgb3V0Y291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlICovCgogIC8qIEVuYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAvKlN0YXJ0IHByb2Nlc3NpbmcqLwogIHdoaWxlKChoY3J5cC0+Q3J5cEluQ291bnQgPCAoaGNyeXAtPlNpemUvNFUpKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUvNFUpKSkKICB7ICAKICAgIC8qIFRlbXBvcmFyeSBDcnlwSW5Db3VudCBWYWx1ZSAqLyAKICAgIGluY291bnQgPSBoY3J5cC0+Q3J5cEluQ291bnQ7ICAgCiAgICAvKiBXcml0ZSBwbGFpbiBkYXRhIGFuZCBnZXQgY2lwaGVyIGRhdGEgKi8KICAgIGlmKCgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19JRk5GICkgIT0gMHgwVSkgJiYgKGluY291bnQgPCAoaGNyeXAtPlNpemUvNFUpKSkKICAgIHsKICAgICAgLyogV3JpdGUgdGhlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50ICk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCApOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIH0KCiAgICAvKiBXYWl0IGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkICovCiAgICBpZihDUllQX1dhaXRPbk9GTkVGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgICB7CiAgICAgIC8qIERpc2FibGUgdGhlIENSWVAgcGVyaXBoZXJhbCBjbG9jayAqLwogICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgLyogQ2hhbmdlIHN0YXRlICYgZXJyb3JDb2RlKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgIH0KCiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICAgIGlmKCgoaGNyeXAtPkluc3RhbmNlLT5TUiAmIENSWVBfRkxBR19PRk5FICkgIT0gMHgwVSkgJiYgKG91dGNvdW50IDwgKGhjcnlwLT5TaXplLzRVKSkpCiAgICB7CiAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBPdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IEJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgICovCiAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIChoY3J5cC0+Q3J5cE91dENvdW50KSkgPSB0ZW1wOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIChoY3J5cC0+Q3J5cE91dENvdW50KSkgPSB0ZW1wOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICB9CiAgICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogICAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwogIH0KICAvKiBEaXNhYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIENSWVAgYmxvY2sgaW5wdXQvb3V0cHV0IGRhdGEgaGFuZGxpbmcgdW5kZXIgaW50ZXJydXB0aW9uIHdpdGggREVTL1RERVMgc3RhbmRhcmQuCiAgKiBAbm90ZSAgIFRoZSBmdW5jdGlvbiBpcyBjYWxsZWQgdW5kZXIgaW50ZXJydXB0aW9uIG9ubHksIG9uY2UKICAqICAgICAgICAgaW50ZXJydXB0aW9ucyBoYXZlIGJlZW4gZW5hYmxlZCBieSBDUllQX0RlY3J5cHRfSVQoKSBhbmQgQ1JZUF9FbmNyeXB0X0lUKCkuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZS4KICAqIEByZXR2YWwgSEFMIHN0YXR1cwogICovCnN0YXRpYyB2b2lkIENSWVBfVERFU19JVChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwKQp7CiAgdWludDMyX3QgdGVtcDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwoKICBpZihoY3J5cC0+U3RhdGUgPT0gSEFMX0NSWVBfU1RBVEVfQlVTWSkKICB7CiAgICBpZigoX19IQUxfQ1JZUF9HRVRfSVQoaGNyeXAsIENSWVBfSVRfSU5JKSAhPSAweDBVKSAmJiAoX19IQUxfQ1JZUF9HRVRfRkxBRyhoY3J5cCwgQ1JZUF9GTEFHX0lOUklTKSAhPSAweDBVKSkKCiAgICB7CiAgICAgIC8qIFdyaXRlIGlucHV0IGJsb2NrIGluIHRoZSBJTiBGSUZPICovCiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQgKTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQgKTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CgogICAgICBpZihoY3J5cC0+Q3J5cEluQ291bnQgPT0gIChoY3J5cC0+U2l6ZS80VSkpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIGludGVycnVwdGlvbiAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwoKICAgICAgICAvKiBDYWxsIHRoZSBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICB9CiAgICB9CiAgICBpZigoX19IQUxfQ1JZUF9HRVRfSVQoaGNyeXAsIENSWVBfSVRfT1VUSSkgIT0gMHgwVSkmJiAoX19IQUxfQ1JZUF9HRVRfRkxBRyhoY3J5cCwgQ1JZUF9GTEFHX09VVFJJUykgIT0gMHgwVSkpCiAgICB7CiAgICAgIC8qIFJlYWQgdGhlIG91dHB1dCBibG9jayBmcm9tIHRoZSBPdXRwdXQgRklGTyBhbmQgcHV0IHRoZW0gaW4gdGVtcG9yYXJ5IEJ1ZmZlciB0aGVuIGdldCBDcnlwT3V0QnVmZiBmcm9tIHRlbXBvcmFyeSBidWZmZXIgICovCiAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIChoY3J5cC0+Q3J5cE91dENvdW50KSkgPSB0ZW1wOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIHRlbXAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIChoY3J5cC0+Q3J5cE91dENvdW50KSkgPSB0ZW1wOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIGlmKGhjcnlwLT5DcnlwT3V0Q291bnQgPT0gIChoY3J5cC0+U2l6ZS80VSkpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIGludGVycnVwdGlvbiAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9PVVRJKTsKCiAgICAgICAgLyogRGlzYWJsZSBDUllQICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgICAgIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogICAgICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAgICAgICAvKiBDYWxsIG91dHB1dCB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgICAgICAvKkNhbGwgcmVnaXN0ZXJlZCBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIGhjcnlwLT5PdXRDcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBPdXRwdXQgY29tcGxldGUgY2FsbGJhY2sqLwogICAgICAgIEhBTF9DUllQX091dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCgogICAgICB9CiAgICB9CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgLyogQnVzeSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0JVU1k7CiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogIH0KfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRUNCL0NCQyAmIENUUiBBbGdvcml0aG0gd2l0aCBBRVMgU3RhbmRhcmQKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IHNwZWNpZnkgVGltZW91dCB2YWx1ZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0VuY3J5cHQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwoKICAvKiAgU2V0IHRoZSBLZXkqLwogIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgewogICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzMpOwogIH0KCiAgLyogU2V0IHRoZSBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgLyogRW5hYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwogIAogIHdoaWxlKChoY3J5cC0+Q3J5cEluQ291bnQgPCAoaGNyeXAtPlNpemUvNFUpKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUvNFUpKSkKICB7CiAgICAvKiBXcml0ZSBwbGFpbiBEZHRhIGFuZCBnZXQgY2lwaGVyIGRhdGEgKi8KICAgIENSWVBfQUVTX1Byb2Nlc3NEYXRhKGhjcnlwLFRpbWVvdXQpOwogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIG91dGNvdW50ID0gaGNyeXAtPkNyeXBPdXRDb3VudDsKICB9CgogIC8qIERpc2FibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogIC8qIENoYW5nZSB0aGUgQ1JZUCBzdGF0ZSAqLwogIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQoKLyoqCiAgKiBAYnJpZWYgIEVuY3J5cHRpb24gaW4gRUNCL0NCQyAmIENUUiBtb2RlIHdpdGggQUVTIFN0YW5kYXJkIHVzaW5nIGludGVycnVwdCBtb2RlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0VuY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewoKICAvKiAgU2V0IHRoZSBLZXkqLwogIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgewogICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzMpOwogIH0KICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICBpZihoY3J5cC0+U2l6ZSAhPSAwVSkKICB7CiAgICAvKiBFbmFibGUgaW50ZXJydXB0cyAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEVfSVQoaGNyeXAsIENSWVBfSVRfSU5JIHwgQ1JZUF9JVF9PVVRJKTsKCiAgICAvKiBFbmFibGUgQ1JZUCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwogIH0KICBlbHNlCiAgewogICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAvKiBQcm9jZXNzIHVubG9ja2VkICovCiAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogIH0KCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KCi8qKgogICogQGJyaWVmICBEZWNyeXB0aW9uIGluIEVDQi9DQkMgJiBDVFIgbW9kZSB3aXRoIEFFUyBTdGFuZGFyZAogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUKICAqIEBwYXJhbSAgVGltZW91dDogU3BlY2lmeSBUaW1lb3V0IHZhbHVlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdChDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBUaW1lb3V0ICkKewogIHVpbnQxNl90IG91dGNvdW50OyAgLyogVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSAqLwoKICAvKiAgS2V5IHByZXBhcmF0aW9uIGZvciBFQ0IvQ0JDICovCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19DVFIpICAgLypFQ0Igb3IgQ0JDKi8KICB7CiAgICAvKiBjaGFuZ2UgQUxHT01PREUgdG8ga2V5IHByZXBhcmF0aW9uIGZvciBkZWNyeXB0aW9uKi8KICAgIE1PRElGWV9SRUcoaGNyeXAtPkluc3RhbmNlLT5DUiwgQ1JZUF9DUl9BTEdPTU9ERSwgQ1JZUF9DUl9BTEdPTU9ERV9BRVNfS0VZICk7CgogICAgLyogIFNldCB0aGUgS2V5Ki8KICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKCiAgICAvKiBFbmFibGUgQ1JZUCAqLwogICAgX19IQUxfQ1JZUF9FTkFCTEUoaGNyeXApOwoKICAgIC8qIFdhaXQgZm9yIEJVU1kgZmxhZyB0byBiZSByYWlzZWQgKi8KICAgIGlmKENSWVBfV2FpdE9uQlVTWUZsYWcoaGNyeXAsIFRpbWVvdXQpICE9IEhBTF9PSykKICAgIHsKICAgICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICAgIF9fSEFMX0NSWVBfRElTQUJMRShoY3J5cCk7CgogICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgIH0KICAgIC8qIFR1cm4gYmFjayB0byBBTEdPTU9ERSBvZiB0aGUgY29uZmlndXJhdGlvbiAqLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBoY3J5cC0+SW5pdC5BbGdvcml0aG0gKTsKICB9CiAgZWxzZSAgLypBbGdvcml0aG0gQ1RSICovCiAgewogICAgLyogIFNldCB0aGUgS2V5Ki8KICAgIENSWVBfU2V0S2V5KGhjcnlwLCBoY3J5cC0+SW5pdC5LZXlTaXplKTsKICB9CgogIC8qIFNldCBJViAqLwogIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfRUNCKQogIHsKICAgIC8qIFNldCB0aGUgSW5pdGlhbGl6YXRpb24gVmVjdG9yKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYwUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjFMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMVJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCszKTsKICB9CiAgLyogU2V0IHRoZSBwaGFzZSAqLwogIGhjcnlwLT5QaGFzZSA9IENSWVBfUEhBU0VfUFJPQ0VTUzsKCiAgLyogRW5hYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICB3aGlsZSgoaGNyeXAtPkNyeXBJbkNvdW50IDwgKGhjcnlwLT5TaXplLzRVKSkgJiYgKG91dGNvdW50IDwgKGhjcnlwLT5TaXplLzRVKSkpCiAgewogICAgLyogV3JpdGUgcGxhaW4gZGF0YSBhbmQgZ2V0IGNpcGhlciBkYXRhICovCiAgICBDUllQX0FFU19Qcm9jZXNzRGF0YShoY3J5cCxUaW1lb3V0KTsKICAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CiAgfQoKICAvKiBEaXNhYmxlIENSWVAgKi8KICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgLyogUmV0dXJuIGZ1bmN0aW9uIHN0YXR1cyAqLwogIHJldHVybiBIQUxfT0s7Cn0KLyoqCiAgKiBAYnJpZWYgIERlY3J5cHRpb24gaW4gRUNCL0NCQyAmIENUUiBtb2RlIHdpdGggQUVTIFN0YW5kYXJkIHVzaW5nIGludGVycnVwdCBtb2RlCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfQUVTX0RlY3J5cHRfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKCiAgLyogIEtleSBwcmVwYXJhdGlvbiBmb3IgRUNCL0NCQyAqLwogIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfQ1RSKQogIHsKICAgIC8qIGNoYW5nZSBBTEdPTU9ERSB0byBrZXkgcHJlcGFyYXRpb24gZm9yIGRlY3J5cHRpb24qLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0NSX0FMR09NT0RFX0FFU19LRVkgKTsKCiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogV2FpdCBmb3IgQlVTWSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT047CiAgICBkbwogICAgewogICAgICBjb3VudC0tIDsKICAgICAgaWYoY291bnQgPT0gMFUpCiAgICAgIHsKICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0KICAgIHdoaWxlKEhBTF9JU19CSVRfU0VUKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19CVVNZKSk7CgogICAgLyogVHVybiBiYWNrIHRvIEFMR09NT0RFIG9mIHRoZSBjb25maWd1cmF0aW9uICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIGhjcnlwLT5Jbml0LkFsZ29yaXRobSApOwogIH0KICBlbHNlICAvKkFsZ29yaXRobSBDVFIgKi8KICB7CiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwogIH0KCiAgLyogU2V0IElWICovCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgewogICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzMpOwogIH0KICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwogIGlmKGhjcnlwLT5TaXplICE9IDBVKQogIHsKICAgIC8qIEVuYWJsZSBpbnRlcnJ1cHRzICovCiAgICBfX0hBTF9DUllQX0VOQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkgfCBDUllQX0lUX09VVEkpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CiAgfQogIGVsc2UKICB7CiAgICAvKiBQcm9jZXNzIGxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKCiAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgIGhjcnlwLT5TdGF0ZSA9IEhBTF9DUllQX1NUQVRFX1JFQURZOwogIH0KICAvKiBSZXR1cm4gZnVuY3Rpb24gc3RhdHVzICovCiAgcmV0dXJuIEhBTF9PSzsKfQovKioKICAqIEBicmllZiAgRGVjcnlwdGlvbiBpbiBFQ0IvQ0JDICYgQ1RSIG1vZGUgd2l0aCBBRVMgU3RhbmRhcmQgdXNpbmcgRE1BIG1vZGUKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9BRVNfRGVjcnlwdF9ETUEoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIF9fSU8gdWludDMyX3QgY291bnQgPSAwVTsKCiAgLyogIEtleSBwcmVwYXJhdGlvbiBmb3IgRUNCL0NCQyAqLwogIGlmIChoY3J5cC0+SW5pdC5BbGdvcml0aG0gIT0gQ1JZUF9BRVNfQ1RSKQogIHsKICAgIC8qIGNoYW5nZSBBTEdPTU9ERSB0byBrZXkgcHJlcGFyYXRpb24gZm9yIGRlY3J5cHRpb24qLwogICAgTU9ESUZZX1JFRyhoY3J5cC0+SW5zdGFuY2UtPkNSLCBDUllQX0NSX0FMR09NT0RFLCBDUllQX0NSX0FMR09NT0RFX0FFU19LRVkgKTsKCiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwoKICAgIC8qIEVuYWJsZSBDUllQICovCiAgICBfX0hBTF9DUllQX0VOQUJMRShoY3J5cCk7CgogICAgLyogV2FpdCBmb3IgQlVTWSBmbGFnIHRvIGJlIHJhaXNlZCAqLwogICAgY291bnQgPSBDUllQX1RJTUVPVVRfS0VZUFJFUEFSQVRJT047CiAgICBkbwogICAgewogICAgICBjb3VudC0tIDsKICAgICAgaWYoY291bnQgPT0gMFUpCiAgICAgIHsKICAgICAgICAvKiBEaXNhYmxlIHRoZSBDUllQIHBlcmlwaGVyYWwgY2xvY2sgKi8KICAgICAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgICAgICAvKiBDaGFuZ2Ugc3RhdGUgKi8KICAgICAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX1RJTUVPVVQ7CiAgICAgICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgICAgICBfX0hBTF9VTkxPQ0soaGNyeXApOwogICAgICAgIHJldHVybiBIQUxfRVJST1I7CiAgICAgIH0KICAgIH0KICAgIHdoaWxlKEhBTF9JU19CSVRfU0VUKGhjcnlwLT5JbnN0YW5jZS0+U1IsIENSWVBfRkxBR19CVVNZKSk7CgogICAgLyogVHVybiBiYWNrIHRvIEFMR09NT0RFIG9mIHRoZSBjb25maWd1cmF0aW9uICovCiAgICBNT0RJRllfUkVHKGhjcnlwLT5JbnN0YW5jZS0+Q1IsIENSWVBfQ1JfQUxHT01PREUsIGhjcnlwLT5Jbml0LkFsZ29yaXRobSApOwogIH0KICBlbHNlICAvKkFsZ29yaXRobSBDVFIgKi8KICB7CiAgICAvKiAgU2V0IHRoZSBLZXkqLwogICAgQ1JZUF9TZXRLZXkoaGNyeXAsIGhjcnlwLT5Jbml0LktleVNpemUpOwogIH0KCiAgaWYgKGhjcnlwLT5Jbml0LkFsZ29yaXRobSAhPSBDUllQX0FFU19FQ0IpCiAgewogICAgLyogU2V0IHRoZSBJbml0aWFsaXphdGlvbiBWZWN0b3IqLwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QpOwogICAgaGNyeXAtPkluc3RhbmNlLT5JVjBSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wSW5pdFZlY3QrMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPklWMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBJbml0VmVjdCsyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SVYxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEluaXRWZWN0KzMpOwogIH0KICAvKiBTZXQgdGhlIHBoYXNlICovCiAgaGNyeXAtPlBoYXNlID0gQ1JZUF9QSEFTRV9QUk9DRVNTOwoKICBpZihoY3J5cC0+U2l6ZSAhPSAwVSkKICB7CiAgICAvKiBTZXQgdGhlIGlucHV0IGFuZCBvdXRwdXQgYWRkcmVzc2VzIGFuZCBzdGFydCBETUEgdHJhbnNmZXIgKi8KICAgIENSWVBfU2V0RE1BQ29uZmlnKGhjcnlwLCAodWludDMyX3QpKCBoY3J5cC0+cENyeXBJbkJ1ZmZQdHIpLCAoaGNyeXAtPlNpemUvNFUpLCAodWludDMyX3QpKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIpKTsKICB9CiAgZWxzZQogIHsKICAgIC8qIFByb2Nlc3MgdW5sb2NrZWQgKi8KICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogQ2hhbmdlIHRoZSBDUllQIHN0YXRlICovCiAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKICB9CgogIC8qIFJldHVybiBmdW5jdGlvbiBzdGF0dXMgKi8KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAYnJpZWYgIERNQSBDUllQIGlucHV0IGRhdGEgcHJvY2VzcyBjb21wbGV0ZSBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGRtYTogRE1BIGhhbmRsZQogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9ETUFJbkNwbHQoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpCnsKICBDUllQX0hhbmRsZVR5cGVEZWYqIGhjcnlwID0gKENSWVBfSGFuZGxlVHlwZURlZiopKChETUFfSGFuZGxlVHlwZURlZiopaGRtYSktPlBhcmVudDsKCiAgLyogRGlzYWJsZSB0aGUgRE1BIHRyYW5zZmVyIGZvciBpbnB1dCBGSUZPIHJlcXVlc3QgYnkgcmVzZXR0aW5nIHRoZSBESUVOIGJpdAogIGluIHRoZSBETUFDUiByZWdpc3RlciAqLwogIGhjcnlwLT5JbnN0YW5jZS0+RE1BQ1IgJj0gKHVpbnQzMl90KSh+Q1JZUF9ETUFDUl9ESUVOKTsKCiAgLyogQ2FsbCBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwp9CgovKioKICAqIEBicmllZiAgRE1BIENSWVAgb3V0cHV0IGRhdGEgcHJvY2VzcyBjb21wbGV0ZSBjYWxsYmFjay4KICAqIEBwYXJhbSAgaGRtYTogRE1BIGhhbmRsZQogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9ETUFPdXRDcGx0KERNQV9IYW5kbGVUeXBlRGVmICpoZG1hKQp7CiAgQ1JZUF9IYW5kbGVUeXBlRGVmKiBoY3J5cCA9IChDUllQX0hhbmRsZVR5cGVEZWYqKSgoRE1BX0hhbmRsZVR5cGVEZWYqKWhkbWEpLT5QYXJlbnQ7CgoKICAvKiBEaXNhYmxlIHRoZSBETUEgdHJhbnNmZXIgZm9yIG91dHB1dCBGSUZPICovCiAgaGNyeXAtPkluc3RhbmNlLT5ETUFDUiAmPSAodWludDMyX3QpKH5DUllQX0RNQUNSX0RPRU4pOwoKICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgdG8gcmVhZHkgKi8KICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgLyogRGlzYWJsZSBDUllQICovCiAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKICAKICAKICAvKiBDYWxsIG91dHB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIERNQSBDUllQIGNvbW11bmljYXRpb24gZXJyb3IgY2FsbGJhY2suCiAgKiBAcGFyYW0gIGhkbWE6IERNQSBoYW5kbGUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfRE1BRXJyb3IoRE1BX0hhbmRsZVR5cGVEZWYgKmhkbWEpCnsKICBDUllQX0hhbmRsZVR5cGVEZWYqIGhjcnlwID0gKENSWVBfSGFuZGxlVHlwZURlZiopKChETUFfSGFuZGxlVHlwZURlZiopaGRtYSktPlBhcmVudDsKCiAgLyogQ2hhbmdlIHRoZSBDUllQIHBlcmlwaGVyYWwgc3RhdGUgKi8KICBoY3J5cC0+U3RhdGU9IEhBTF9DUllQX1NUQVRFX1JFQURZOwoKICAvKiBETUEgZXJyb3IgY29kZSBmaWVsZCAqLwogIGhjcnlwLT5FcnJvckNvZGUgfD0gSEFMX0NSWVBfRVJST1JfRE1BOwoKICAvKiBDYWxsIGVycm9yIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KfQoKLyoqCiAgKiBAYnJpZWYgIFNldCB0aGUgRE1BIGNvbmZpZ3VyYXRpb24gYW5kIHN0YXJ0IHRoZSBETUEgdHJhbnNmZXIKICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlCiAgKiBAcGFyYW0gIGlucHV0YWRkcjogYWRkcmVzcyBvZiB0aGUgaW5wdXQgYnVmZmVyCiAgKiBAcGFyYW0gIFNpemU6IHNpemUgb2YgdGhlIGlucHV0IGJ1ZmZlciwgbXVzdCBiZSBhIG11bHRpcGxlIG9mIDE2LgogICogQHBhcmFtICBvdXRwdXRhZGRyOiBhZGRyZXNzIG9mIHRoZSBvdXRwdXQgYnVmZmVyCiAgKiBAcmV0dmFsIE5vbmUKICAqLwpzdGF0aWMgdm9pZCBDUllQX1NldERNQUNvbmZpZyhDUllQX0hhbmRsZVR5cGVEZWYgKmhjcnlwLCB1aW50MzJfdCBpbnB1dGFkZHIsIHVpbnQxNl90IFNpemUsIHVpbnQzMl90IG91dHB1dGFkZHIpCnsKICAvKiBTZXQgdGhlIENSWVAgRE1BIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiAgaGNyeXAtPmhkbWFpbi0+WGZlckNwbHRDYWxsYmFjayA9IENSWVBfRE1BSW5DcGx0OwoKICAvKiBTZXQgdGhlIERNQSBpbnB1dCBlcnJvciBjYWxsYmFjayAqLwogIGhjcnlwLT5oZG1haW4tPlhmZXJFcnJvckNhbGxiYWNrID0gQ1JZUF9ETUFFcnJvcjsKCiAgLyogU2V0IHRoZSBDUllQIERNQSB0cmFuc2ZlciBjb21wbGV0ZSBjYWxsYmFjayAqLwogIGhjcnlwLT5oZG1hb3V0LT5YZmVyQ3BsdENhbGxiYWNrID0gQ1JZUF9ETUFPdXRDcGx0OwoKICAvKiBTZXQgdGhlIERNQSBvdXRwdXQgZXJyb3IgY2FsbGJhY2sgKi8KICBoY3J5cC0+aGRtYW91dC0+WGZlckVycm9yQ2FsbGJhY2sgPSBDUllQX0RNQUVycm9yOwoKICAvKiBFbmFibGUgQ1JZUCAqLwogIF9fSEFMX0NSWVBfRU5BQkxFKGhjcnlwKTsKCiAgLyogRW5hYmxlIHRoZSBpbnB1dCBETUEgU3RyZWFtICovCiAgaWYgKEhBTF9ETUFfU3RhcnRfSVQoaGNyeXAtPmhkbWFpbiwgaW5wdXRhZGRyLCAodWludDMyX3QpJmhjcnlwLT5JbnN0YW5jZS0+RElOLCBTaXplKSE9SEFMX09LKQogIHsKICAgIC8qIERNQSBlcnJvciBjb2RlIGZpZWxkICovCiAgICBoY3J5cC0+RXJyb3JDb2RlIHw9IEhBTF9DUllQX0VSUk9SX0RNQTsKCiAgICAvKiBDYWxsIGVycm9yIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgLypDYWxsIHJlZ2lzdGVyZWQgZXJyb3IgY2FsbGJhY2sqLwogICAgaGNyeXAtPkVycm9yQ2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgLypDYWxsIGxlZ2FjeSB3ZWFrIGVycm9yIGNhbGxiYWNrKi8KICAgIEhBTF9DUllQX0Vycm9yQ2FsbGJhY2soaGNyeXApOyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogIH0KCiAgLyogRW5hYmxlIHRoZSBvdXRwdXQgRE1BIFN0cmVhbSAqLwogIGlmIChIQUxfRE1BX1N0YXJ0X0lUKGhjcnlwLT5oZG1hb3V0LCAodWludDMyX3QpJmhjcnlwLT5JbnN0YW5jZS0+RE9VVCwgb3V0cHV0YWRkciwgU2l6ZSkhPUhBTF9PSykKICB7CiAgICAvKiBETUEgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9ETUE7CgogICAgLyogQ2FsbCBlcnJvciBjYWxsYmFjayAqLwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICB9CiAgLyogRW5hYmxlIEluL091dCBETUEgcmVxdWVzdCAqLwogIGhjcnlwLT5JbnN0YW5jZS0+RE1BQ1IgPSBDUllQX0RNQUNSX0RPRU4gfCBDUllQX0RNQUNSX0RJRU47Cn0KCi8qKgogICogQGJyaWVmICBQcm9jZXNzIERhdGE6IFdyaXRlIElucHV0IGRhdGEgaW4gcG9sbGluZyBtb2RlIGFuZCB1c2VkIGluIEFFUyBmdW5jdGlvbnMuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBUaW1lb3V0OiBTcGVjaWZ5IFRpbWVvdXQgdmFsdWUKICAqIEByZXR2YWwgTm9uZQogICovCnN0YXRpYyB2b2lkIENSWVBfQUVTX1Byb2Nlc3NEYXRhKENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IFRpbWVvdXQpCnsKCiAgdWludDMyX3QgdGVtcDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0QnVmZiAqLwogIHVpbnQxNl90IGluY291bnQ7ICAvKiBUZW1wb3JhcnkgQ3J5cEluQ291bnQgVmFsdWUgKi8KICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KICAKICAvKlRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUqLwogIGluY291bnQgPSBoY3J5cC0+Q3J5cEluQ291bnQ7IAoKICBpZigoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfSUZORiApICE9IDB4MFUpICYmIChpbmNvdW50IDwgKChoY3J5cC0+U2l6ZSkvNFUpKSkKICB7CiAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50ICk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50ICk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50ICk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50ICk7CiAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICB9CgogIC8qIFdhaXQgZm9yIE9GTkUgZmxhZyB0byBiZSByYWlzZWQgKi8KICBpZihDUllQX1dhaXRPbk9GTkVGbGFnKGhjcnlwLCBUaW1lb3V0KSAhPSBIQUxfT0spCiAgewogICAgLyogRGlzYWJsZSB0aGUgQ1JZUCBwZXJpcGhlcmFsIGNsb2NrICovCiAgICBfX0hBTF9DUllQX0RJU0FCTEUoaGNyeXApOwoKICAgIC8qIENoYW5nZSBzdGF0ZSAmIGVycm9yIGNvZGUqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9USU1FT1VUOwogICAgaGNyeXAtPlN0YXRlID0gSEFMX0NSWVBfU1RBVEVfUkVBRFk7CgogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKI2lmIChVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTID09IDEpCiAgICAvKkNhbGwgcmVnaXN0ZXJlZCBlcnJvciBjYWxsYmFjayovCiAgICBoY3J5cC0+RXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAvKkNhbGwgbGVnYWN5IHdlYWsgZXJyb3IgY2FsbGJhY2sqLwogICAgSEFMX0NSWVBfRXJyb3JDYWxsYmFjayhoY3J5cCk7CiNlbmRpZiAvKiBVU0VfSEFMX0NSWVBfUkVHSVNURVJfQ0FMTEJBQ0tTICovCiAgfQogIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgb3V0Y291bnQgPSBoY3J5cC0+Q3J5cE91dENvdW50OwoKICBpZigoKGhjcnlwLT5JbnN0YW5jZS0+U1IgJiBDUllQX0ZMQUdfT0ZORSApICE9IDB4MFUpICYmIChvdXRjb3VudCA8ICgoaGNyeXAtPlNpemUpLzRVKSkpCiAgewogICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIE91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgYnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAgKi8KICAgIHRlbXAgID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgdGVtcCAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cE91dENvdW50KSA9IHRlbXA7CiAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICB0ZW1wICA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcDsKICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgIHRlbXAgID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wOwogICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogIH0KfQoKLyoqCiAgKiBAYnJpZWYgIEhhbmRsZSBDUllQIGJsb2NrIGlucHV0L291dHB1dCBkYXRhIGhhbmRsaW5nIHVuZGVyIGludGVycnVwdGlvbi4KICAqIEBub3RlICAgVGhlIGZ1bmN0aW9uIGlzIGNhbGxlZCB1bmRlciBpbnRlcnJ1cHRpb24gb25seSwgb25jZQogICogICAgICAgICBpbnRlcnJ1cHRpb25zIGhhdmUgYmVlbiBlbmFibGVkIGJ5IEhBTF9DUllQX0VuY3J5cHRfSVQgb3IgSEFMX0NSWVBfRGVjcnlwdF9JVC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9BRVNfSVQoQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCkKewogIHVpbnQzMl90IHRlbXA7ICAvKiBUZW1wb3JhcnkgQ3J5cE91dEJ1ZmYgKi8KICB1aW50MTZfdCBpbmNvdW50OyAvKiBUZW1wb3JhcnkgQ3J5cEluQ291bnQgVmFsdWUgKi8KICB1aW50MTZfdCBvdXRjb3VudDsgIC8qIFRlbXBvcmFyeSBDcnlwT3V0Q291bnQgVmFsdWUgKi8KCiAgaWYoaGNyeXAtPlN0YXRlID09IEhBTF9DUllQX1NUQVRFX0JVU1kpCiAgewogICAgLypUZW1wb3JhcnkgQ3J5cE91dENvdW50IFZhbHVlKi8KICAgIGluY291bnQgPSBoY3J5cC0+Q3J5cEluQ291bnQ7CgogICAgaWYoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX0lGTkYgKSAhPSAweDBVKSAmJiAoaW5jb3VudCA8IChoY3J5cC0+U2l6ZS80VSkpKQogICAgewogICAgICAvKiBXcml0ZSB0aGUgaW5wdXQgYmxvY2sgaW4gdGhlIElOIEZJRk8gKi8KICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQgKTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGhjcnlwLT5JbnN0YW5jZS0+RElOICA9ICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwSW5CdWZmUHRyICsgaGNyeXAtPkNyeXBJbkNvdW50ICk7CiAgICAgIGhjcnlwLT5DcnlwSW5Db3VudCsrOwogICAgICBoY3J5cC0+SW5zdGFuY2UtPkRJTiAgPSAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cEluQnVmZlB0ciArIGhjcnlwLT5DcnlwSW5Db3VudCApOwogICAgICBoY3J5cC0+Q3J5cEluQ291bnQrKzsKICAgICAgaGNyeXAtPkluc3RhbmNlLT5ESU4gID0gKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBJbkJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cEluQ291bnQgKTsKICAgICAgaGNyeXAtPkNyeXBJbkNvdW50Kys7CiAgICAgIGlmKGhjcnlwLT5DcnlwSW5Db3VudCA9PSAgKGhjcnlwLT5TaXplLzRVKSkKICAgICAgewogICAgICAgIC8qIERpc2FibGUgaW50ZXJydXB0cyAqLwogICAgICAgIF9fSEFMX0NSWVBfRElTQUJMRV9JVChoY3J5cCwgQ1JZUF9JVF9JTkkpOwoKICAgICAgICAvKiBDYWxsIHRoZSBpbnB1dCBkYXRhIHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIElucHV0IGNvbXBsZXRlIGNhbGxiYWNrKi8KICAgICAgICBoY3J5cC0+SW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZWxzZQogICAgICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBJbnB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfSW5DcGx0Q2FsbGJhY2soaGNyeXApOwojZW5kaWYgLyogVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyAqLwogICAgICB9CiAgICB9CiAgICAKICAgIC8qVGVtcG9yYXJ5IENyeXBPdXRDb3VudCBWYWx1ZSovCiAgICBvdXRjb3VudCA9IGhjcnlwLT5DcnlwT3V0Q291bnQ7CgogICAgaWYoKChoY3J5cC0+SW5zdGFuY2UtPlNSICYgQ1JZUF9GTEFHX09GTkUgKSAhPSAweDBVKSAmJiAob3V0Y291bnQgPCAoaGNyeXAtPlNpemUvNFUpKSkKICAgIHsKICAgICAgLyogUmVhZCB0aGUgb3V0cHV0IGJsb2NrIGZyb20gdGhlIG91dHB1dCBGSUZPIGFuZCBwdXQgdGhlbSBpbiB0ZW1wb3JhcnkgYnVmZmVyIHRoZW4gZ2V0IENyeXBPdXRCdWZmIGZyb20gdGVtcG9yYXJ5IGJ1ZmZlciAgKi8KICAgICAgdGVtcCAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcDsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgICB0ZW1wICA9IGhjcnlwLT5JbnN0YW5jZS0+RE9VVDsKICAgICAgKih1aW50MzJfdCAqKShoY3J5cC0+cENyeXBPdXRCdWZmUHRyICsgaGNyeXAtPkNyeXBPdXRDb3VudCkgPSB0ZW1wOwogICAgICBoY3J5cC0+Q3J5cE91dENvdW50Kys7CiAgICAgIHRlbXAgID0gaGNyeXAtPkluc3RhbmNlLT5ET1VUOwogICAgICAqKHVpbnQzMl90ICopKGhjcnlwLT5wQ3J5cE91dEJ1ZmZQdHIgKyBoY3J5cC0+Q3J5cE91dENvdW50KSA9IHRlbXA7CiAgICAgIGhjcnlwLT5DcnlwT3V0Q291bnQrKzsKICAgICAgdGVtcCAgPSBoY3J5cC0+SW5zdGFuY2UtPkRPVVQ7CiAgICAgICoodWludDMyX3QgKikoaGNyeXAtPnBDcnlwT3V0QnVmZlB0ciArIGhjcnlwLT5DcnlwT3V0Q291bnQpID0gdGVtcDsKICAgICAgaGNyeXAtPkNyeXBPdXRDb3VudCsrOwogICAgICBpZihoY3J5cC0+Q3J5cE91dENvdW50ID09ICAoaGNyeXAtPlNpemUvNFUpKQogICAgICB7CiAgICAgICAgLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFX0lUKGhjcnlwLCBDUllQX0lUX09VVEkpOwoKICAgICAgICAvKiBDaGFuZ2UgdGhlIENSWVAgc3RhdGUgKi8KICAgICAgICBoY3J5cC0+U3RhdGUgPSBIQUxfQ1JZUF9TVEFURV9SRUFEWTsKCiAgICAgICAgLyogRGlzYWJsZSBDUllQICovCiAgICAgICAgX19IQUxfQ1JZUF9ESVNBQkxFKGhjcnlwKTsKCiAgICAgICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgICAgIF9fSEFMX1VOTE9DSyhoY3J5cCk7CgogICAgICAgIC8qIENhbGwgb3V0cHV0IHRyYW5zZmVyIGNvbXBsZXRlIGNhbGxiYWNrICovCiNpZiAoVVNFX0hBTF9DUllQX1JFR0lTVEVSX0NBTExCQUNLUyA9PSAxKQogICAgICAgIC8qQ2FsbCByZWdpc3RlcmVkIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgaGNyeXAtPk91dENwbHRDYWxsYmFjayhoY3J5cCk7CiNlbHNlCiAgICAgICAgLypDYWxsIGxlZ2FjeSB3ZWFrIE91dHB1dCBjb21wbGV0ZSBjYWxsYmFjayovCiAgICAgICAgSEFMX0NSWVBfT3V0Q3BsdENhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICAgICAgfQogICAgfQogIH0KICBlbHNlCiAgewogICAgLyogUHJvY2VzcyB1bmxvY2tlZCAqLwogICAgX19IQUxfVU5MT0NLKGhjcnlwKTsKICAgIC8qIEJ1c3kgZXJyb3IgY29kZSBmaWVsZCAqLwogICAgaGNyeXAtPkVycm9yQ29kZSB8PSBIQUxfQ1JZUF9FUlJPUl9CVVNZOwojaWYgKFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgPT0gMSkKICAgIC8qQ2FsbCByZWdpc3RlcmVkIGVycm9yIGNhbGxiYWNrKi8KICAgIGhjcnlwLT5FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2Vsc2UKICAgIC8qQ2FsbCBsZWdhY3kgd2VhayBlcnJvciBjYWxsYmFjayovCiAgICBIQUxfQ1JZUF9FcnJvckNhbGxiYWNrKGhjcnlwKTsKI2VuZGlmIC8qIFVTRV9IQUxfQ1JZUF9SRUdJU1RFUl9DQUxMQkFDS1MgKi8KICB9Cn0KCi8qKgogICogQGJyaWVmICBXcml0ZXMgS2V5IGluIEtleSByZWdpc3RlcnMuCiAgKiBAcGFyYW0gIGhjcnlwOiBwb2ludGVyIHRvIGEgQ1JZUF9IYW5kbGVUeXBlRGVmIHN0cnVjdHVyZSB0aGF0IGNvbnRhaW5zCiAgKiAgICAgICAgIHRoZSBjb25maWd1cmF0aW9uIGluZm9ybWF0aW9uIGZvciBDUllQIG1vZHVsZQogICogQHBhcmFtICBLZXlTaXplOiBTaXplIG9mIEtleQogICogQHJldHZhbCBOb25lCiAgKi8Kc3RhdGljIHZvaWQgQ1JZUF9TZXRLZXkoIENSWVBfSGFuZGxlVHlwZURlZiAqaGNyeXAsIHVpbnQzMl90IEtleVNpemUpCnsKICBzd2l0Y2goS2V5U2l6ZSkKICB7CiAgY2FzZSBDUllQX0tFWVNJWkVfMjU2QjoKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzBMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzBSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5LMUxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPksxUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSszKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzJMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzQpOwogICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSs2KTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzcpOwogICAgYnJlYWs7CiAgY2FzZSBDUllQX0tFWVNJWkVfMTkyQjoKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzFSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzEpOwogICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMik7CiAgICBoY3J5cC0+SW5zdGFuY2UtPksyUlIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSszKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNMUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzQpOwogICAgaGNyeXAtPkluc3RhbmNlLT5LM1JSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrNSk7CiAgICBicmVhazsKICBjYXNlIENSWVBfS0VZU0laRV8xMjhCOgogICAgaGNyeXAtPkluc3RhbmNlLT5LMkxSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkpOwogICAgaGNyeXAtPkluc3RhbmNlLT5LMlJSID0gKih1aW50MzJfdCopKGhjcnlwLT5Jbml0LnBLZXkrMSk7CiAgICBoY3J5cC0+SW5zdGFuY2UtPkszTFIgPSAqKHVpbnQzMl90KikoaGNyeXAtPkluaXQucEtleSsyKTsKICAgIGhjcnlwLT5JbnN0YW5jZS0+SzNSUiA9ICoodWludDMyX3QqKShoY3J5cC0+SW5pdC5wS2V5KzMpOwoKICAgIGJyZWFrOwogIGRlZmF1bHQ6CiAgICBicmVhazsKICB9Cn0KCi8qKgogICogQGJyaWVmICBIYW5kbGUgQ1JZUCBoYXJkd2FyZSBibG9jayBUaW1lb3V0IHdoZW4gd2FpdGluZyBmb3IgQlVTWSBmbGFnIHRvIGJlIHJhaXNlZC4KICAqIEBwYXJhbSAgaGNyeXA6IHBvaW50ZXIgdG8gYSBDUllQX0hhbmRsZVR5cGVEZWYgc3RydWN0dXJlIHRoYXQgY29udGFpbnMKICAqICAgICAgICAgdGhlIGNvbmZpZ3VyYXRpb24gaW5mb3JtYXRpb24gZm9yIENSWVAgbW9kdWxlLgogICogQHBhcmFtICBUaW1lb3V0OiBUaW1lb3V0IGR1cmF0aW9uLgogICogQHJldHZhbCBIQUwgc3RhdHVzCiAgKi8Kc3RhdGljIEhBTF9TdGF0dXNUeXBlRGVmIENSWVBfV2FpdE9uQlVTWUZsYWcoY29uc3QgQ1JZUF9IYW5kbGVUeXBlRGVmICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IHRpY2tzdGFydDsKCiAgLyogR2V0IHRpbWVvdXQgKi8KICB0aWNrc3RhcnQgPSBIQUxfR2V0VGljaygpOwoKICB3aGlsZShIQUxfSVNfQklUX1NFVChoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfQlVTWSkpCiAgewogICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICBpZihUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmKCgoSEFMX0dldFRpY2soKSAtIHRpY2tzdGFydCApID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAYnJpZWYgIEhhbmRsZSBDUllQIGhhcmR3YXJlIGJsb2NrIFRpbWVvdXQgd2hlbiB3YWl0aW5nIGZvciBPRk5FIGZsYWcgdG8gYmUgcmFpc2VkLgogICogQHBhcmFtICBoY3J5cDogcG9pbnRlciB0byBhIENSWVBfSGFuZGxlVHlwZURlZiBzdHJ1Y3R1cmUgdGhhdCBjb250YWlucwogICogICAgICAgICB0aGUgY29uZmlndXJhdGlvbiBpbmZvcm1hdGlvbiBmb3IgQ1JZUCBtb2R1bGUuCiAgKiBAcGFyYW0gIFRpbWVvdXQ6IFRpbWVvdXQgZHVyYXRpb24uCiAgKiBAcmV0dmFsIEhBTCBzdGF0dXMKICAqLwpzdGF0aWMgSEFMX1N0YXR1c1R5cGVEZWYgQ1JZUF9XYWl0T25PRk5FRmxhZyhjb25zdCBDUllQX0hhbmRsZVR5cGVEZWYgICpoY3J5cCwgdWludDMyX3QgVGltZW91dCkKewogIHVpbnQzMl90IHRpY2tzdGFydDsKCiAgLyogR2V0IHRpbWVvdXQgKi8KICB0aWNrc3RhcnQgPSBIQUxfR2V0VGljaygpOwoKICB3aGlsZShIQUxfSVNfQklUX0NMUihoY3J5cC0+SW5zdGFuY2UtPlNSLCBDUllQX0ZMQUdfT0ZORSkpCiAgewogICAgLyogQ2hlY2sgZm9yIHRoZSBUaW1lb3V0ICovCiAgICBpZihUaW1lb3V0ICE9IEhBTF9NQVhfREVMQVkpCiAgICB7CiAgICAgIGlmKCgoSEFMX0dldFRpY2soKSAtIHRpY2tzdGFydCApID4gVGltZW91dCkgfHwgKFRpbWVvdXQgPT0gMFUpKQogICAgICB7CiAgICAgICAgcmV0dXJuIEhBTF9FUlJPUjsKICAgICAgfQogICAgfQogIH0KICByZXR1cm4gSEFMX09LOwp9CgoKLyoqCiAgKiBAfQogICovCgoKCi8qKgogICogQH0KICAqLwoKLyoqCiAgKiBAfQogICovCgojZW5kaWYgLyogSEFMX0NSWVBfTU9EVUxFX0VOQUJMRUQgKi8KCgovKioKICAqIEB9CiAgKi8KI2VuZGlmIC8qICBDUllQKi8KLyoqCiAgKiBAfQogICovCgovKioqKioqKioqKioqKioqKioqKioqKioqIChDKSBDT1BZUklHSFQgU1RNaWNyb2VsZWN0cm9uaWNzICoqKioqRU5EIE9GIEZJTEUqKioqLwo=