LyoKICogUUVNVSBWTVBvcnQgZW11bGF0aW9uCiAqCiAqIENvcHlyaWdodCAoQykgMjAwNyBIZXJ26SBQb3Vzc2luZWF1CiAqCiAqIFBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhIGNvcHkKICogb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzICh0aGUgIlNvZnR3YXJlIiksIHRvIGRlYWwKICogaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0cwogKiB0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsCiAqIGNvcGllcyBvZiB0aGUgU29mdHdhcmUsIGFuZCB0byBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBTb2Z0d2FyZSBpcwogKiBmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgogKgogKiBUaGUgYWJvdmUgY29weXJpZ2h0IG5vdGljZSBhbmQgdGhpcyBwZXJtaXNzaW9uIG5vdGljZSBzaGFsbCBiZSBpbmNsdWRlZCBpbgogKiBhbGwgY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS4KICoKICogVEhFIFNPRlRXQVJFIElTIFBST1ZJREVEICJBUyBJUyIsIFdJVEhPVVQgV0FSUkFOVFkgT0YgQU5ZIEtJTkQsIEVYUFJFU1MgT1IKICogSU1QTElFRCwgSU5DTFVESU5HIEJVVCBOT1QgTElNSVRFRCBUTyBUSEUgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFksCiAqIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFORCBOT05JTkZSSU5HRU1FTlQuIElOIE5PIEVWRU5UIFNIQUxMCiAqIFRIRSBBVVRIT1JTIE9SIENPUFlSSUdIVCBIT0xERVJTIEJFIExJQUJMRSBGT1IgQU5ZIENMQUlNLCBEQU1BR0VTIE9SIE9USEVSCiAqIExJQUJJTElUWSwgV0hFVEhFUiBJTiBBTiBBQ1RJT04gT0YgQ09OVFJBQ1QsIFRPUlQgT1IgT1RIRVJXSVNFLCBBUklTSU5HIEZST00sCiAqIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFIFNPRlRXQVJFIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4KICogVEhFIFNPRlRXQVJFLgogKi8KI2luY2x1ZGUgImh3LmgiCiNpbmNsdWRlICJpc2EuaCIKI2luY2x1ZGUgInBjLmgiCiNpbmNsdWRlICJrdm0uaCIKI2luY2x1ZGUgInFkZXYuaCIKCi8vI2RlZmluZSBWTVBPUlRfREVCVUcKCiNkZWZpbmUgVk1QT1JUX0NNRF9HRVRWRVJTSU9OIDB4MGEKI2RlZmluZSBWTVBPUlRfQ01EX0dFVFJBTVNJWkUgMHgxNAoKI2RlZmluZSBWTVBPUlRfRU5UUklFUyAweDJjCiNkZWZpbmUgVk1QT1JUX01BR0lDICAgMHg1NjRENTg2OAoKdHlwZWRlZiBzdHJ1Y3QgX1ZNUG9ydFN0YXRlCnsKICAgIElTQURldmljZSBkZXY7CiAgICBJT1BvcnRSZWFkRnVuYyAqZnVuY1tWTVBPUlRfRU5UUklFU107CiAgICB2b2lkICpvcGFxdWVbVk1QT1JUX0VOVFJJRVNdOwp9IFZNUG9ydFN0YXRlOwoKc3RhdGljIFZNUG9ydFN0YXRlICpwb3J0X3N0YXRlOwoKdm9pZCB2bXBvcnRfcmVnaXN0ZXIodW5zaWduZWQgY2hhciBjb21tYW5kLCBJT1BvcnRSZWFkRnVuYyAqZnVuYywgdm9pZCAqb3BhcXVlKQp7CiAgICBpZiAoY29tbWFuZCA+PSBWTVBPUlRfRU5UUklFUykKICAgICAgICByZXR1cm47CgogICAgcG9ydF9zdGF0ZS0+ZnVuY1tjb21tYW5kXSA9IGZ1bmM7CiAgICBwb3J0X3N0YXRlLT5vcGFxdWVbY29tbWFuZF0gPSBvcGFxdWU7Cn0KCnN0YXRpYyB1aW50MzJfdCB2bXBvcnRfaW9wb3J0X3JlYWQodm9pZCAqb3BhcXVlLCB1aW50MzJfdCBhZGRyKQp7CiAgICBWTVBvcnRTdGF0ZSAqcyA9IG9wYXF1ZTsKICAgIENQVVN0YXRlICplbnYgPSBjcHVfc2luZ2xlX2VudjsKICAgIHVuc2lnbmVkIGNoYXIgY29tbWFuZDsKICAgIHVpbnQzMl90IGVheDsKCiAgICBjcHVfc3luY2hyb25pemVfc3RhdGUoZW52KTsKCiAgICBlYXggPSBlbnYtPnJlZ3NbUl9FQVhdOwogICAgaWYgKGVheCAhPSBWTVBPUlRfTUFHSUMpCiAgICAgICAgcmV0dXJuIGVheDsKCiAgICBjb21tYW5kID0gZW52LT5yZWdzW1JfRUNYXTsKICAgIGlmIChjb21tYW5kID49IFZNUE9SVF9FTlRSSUVTKQogICAgICAgIHJldHVybiBlYXg7CiAgICBpZiAoIXMtPmZ1bmNbY29tbWFuZF0pCiAgICB7CiNpZmRlZiBWTVBPUlRfREVCVUcKICAgICAgICBmcHJpbnRmKHN0ZGVyciwgInZtcG9ydDogdW5rbm93biBjb21tYW5kICV4XG4iLCBjb21tYW5kKTsKI2VuZGlmCiAgICAgICAgcmV0dXJuIGVheDsKICAgIH0KCiAgICByZXR1cm4gcy0+ZnVuY1tjb21tYW5kXShzLT5vcGFxdWVbY29tbWFuZF0sIGFkZHIpOwp9CgpzdGF0aWMgdm9pZCB2bXBvcnRfaW9wb3J0X3dyaXRlKHZvaWQgKm9wYXF1ZSwgdWludDMyX3QgYWRkciwgdWludDMyX3QgdmFsKQp7CiAgICBDUFVTdGF0ZSAqZW52ID0gY3B1X3NpbmdsZV9lbnY7CgogICAgZW52LT5yZWdzW1JfRUFYXSA9IHZtcG9ydF9pb3BvcnRfcmVhZChvcGFxdWUsIGFkZHIpOwp9CgpzdGF0aWMgdWludDMyX3Qgdm1wb3J0X2NtZF9nZXRfdmVyc2lvbih2b2lkICpvcGFxdWUsIHVpbnQzMl90IGFkZHIpCnsKICAgIENQVVN0YXRlICplbnYgPSBjcHVfc2luZ2xlX2VudjsKICAgIGVudi0+cmVnc1tSX0VCWF0gPSBWTVBPUlRfTUFHSUM7CiAgICByZXR1cm4gNjsKfQoKc3RhdGljIHVpbnQzMl90IHZtcG9ydF9jbWRfcmFtX3NpemUodm9pZCAqb3BhcXVlLCB1aW50MzJfdCBhZGRyKQp7CiAgICBDUFVTdGF0ZSAqZW52ID0gY3B1X3NpbmdsZV9lbnY7CiAgICBlbnYtPnJlZ3NbUl9FQlhdID0gMHgxMTc3OwogICAgcmV0dXJuIHJhbV9zaXplOwp9CgovKiB2bW1vdXNlIGhlbHBlcnMgKi8Kdm9pZCB2bW1vdXNlX2dldF9kYXRhKHVpbnQzMl90ICpkYXRhKQp7CiAgICBDUFVTdGF0ZSAqZW52ID0gY3B1X3NpbmdsZV9lbnY7CgogICAgZGF0YVswXSA9IGVudi0+cmVnc1tSX0VBWF07IGRhdGFbMV0gPSBlbnYtPnJlZ3NbUl9FQlhdOwogICAgZGF0YVsyXSA9IGVudi0+cmVnc1tSX0VDWF07IGRhdGFbM10gPSBlbnYtPnJlZ3NbUl9FRFhdOwogICAgZGF0YVs0XSA9IGVudi0+cmVnc1tSX0VTSV07IGRhdGFbNV0gPSBlbnYtPnJlZ3NbUl9FREldOwp9Cgp2b2lkIHZtbW91c2Vfc2V0X2RhdGEoY29uc3QgdWludDMyX3QgKmRhdGEpCnsKICAgIENQVVN0YXRlICplbnYgPSBjcHVfc2luZ2xlX2VudjsKCiAgICBlbnYtPnJlZ3NbUl9FQVhdID0gZGF0YVswXTsgZW52LT5yZWdzW1JfRUJYXSA9IGRhdGFbMV07CiAgICBlbnYtPnJlZ3NbUl9FQ1hdID0gZGF0YVsyXTsgZW52LT5yZWdzW1JfRURYXSA9IGRhdGFbM107CiAgICBlbnYtPnJlZ3NbUl9FU0ldID0gZGF0YVs0XTsgZW52LT5yZWdzW1JfRURJXSA9IGRhdGFbNV07Cn0KCnN0YXRpYyBpbnQgdm1wb3J0X2luaXRmbihJU0FEZXZpY2UgKmRldikKewogICAgVk1Qb3J0U3RhdGUgKnMgPSBET19VUENBU1QoVk1Qb3J0U3RhdGUsIGRldiwgZGV2KTsKCiAgICByZWdpc3Rlcl9pb3BvcnRfcmVhZCgweDU2NTgsIDEsIDQsIHZtcG9ydF9pb3BvcnRfcmVhZCwgcyk7CiAgICByZWdpc3Rlcl9pb3BvcnRfd3JpdGUoMHg1NjU4LCAxLCA0LCB2bXBvcnRfaW9wb3J0X3dyaXRlLCBzKTsKICAgIGlzYV9pbml0X2lvcG9ydChkZXYsIDB4NTY1OCk7CiAgICBwb3J0X3N0YXRlID0gczsKICAgIC8qIFJlZ2lzdGVyIHNvbWUgZ2VuZXJpYyBwb3J0IGNvbW1hbmRzICovCiAgICB2bXBvcnRfcmVnaXN0ZXIoVk1QT1JUX0NNRF9HRVRWRVJTSU9OLCB2bXBvcnRfY21kX2dldF92ZXJzaW9uLCBOVUxMKTsKICAgIHZtcG9ydF9yZWdpc3RlcihWTVBPUlRfQ01EX0dFVFJBTVNJWkUsIHZtcG9ydF9jbWRfcmFtX3NpemUsIE5VTEwpOwogICAgcmV0dXJuIDA7Cn0KCnN0YXRpYyBJU0FEZXZpY2VJbmZvIHZtcG9ydF9pbmZvID0gewogICAgLnFkZXYubmFtZSAgICAgPSAidm1wb3J0IiwKICAgIC5xZGV2LnNpemUgICAgID0gc2l6ZW9mKFZNUG9ydFN0YXRlKSwKICAgIC5xZGV2Lm5vX3VzZXIgID0gMSwKICAgIC5pbml0ICAgICAgICAgID0gdm1wb3J0X2luaXRmbiwKfTsKCnN0YXRpYyB2b2lkIHZtcG9ydF9kZXZfcmVnaXN0ZXIodm9pZCkKewogICAgaXNhX3FkZXZfcmVnaXN0ZXIoJnZtcG9ydF9pbmZvKTsKfQpkZXZpY2VfaW5pdCh2bXBvcnRfZGV2X3JlZ2lzdGVyKQo=