LyoKICogUUVNVSBWTVBvcnQgZW11bGF0aW9uCiAqCiAqIENvcHlyaWdodCAoQykgMjAwNyBIZXJ26SBQb3Vzc2luZWF1CiAqCiAqIFBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhIGNvcHkKICogb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzICh0aGUgIlNvZnR3YXJlIiksIHRvIGRlYWwKICogaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0cwogKiB0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsCiAqIGNvcGllcyBvZiB0aGUgU29mdHdhcmUsIGFuZCB0byBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBTb2Z0d2FyZSBpcwogKiBmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgogKgogKiBUaGUgYWJvdmUgY29weXJpZ2h0IG5vdGljZSBhbmQgdGhpcyBwZXJtaXNzaW9uIG5vdGljZSBzaGFsbCBiZSBpbmNsdWRlZCBpbgogKiBhbGwgY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS4KICoKICogVEhFIFNPRlRXQVJFIElTIFBST1ZJREVEICJBUyBJUyIsIFdJVEhPVVQgV0FSUkFOVFkgT0YgQU5ZIEtJTkQsIEVYUFJFU1MgT1IKICogSU1QTElFRCwgSU5DTFVESU5HIEJVVCBOT1QgTElNSVRFRCBUTyBUSEUgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFksCiAqIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFORCBOT05JTkZSSU5HRU1FTlQuIElOIE5PIEVWRU5UIFNIQUxMCiAqIFRIRSBBVVRIT1JTIE9SIENPUFlSSUdIVCBIT0xERVJTIEJFIExJQUJMRSBGT1IgQU5ZIENMQUlNLCBEQU1BR0VTIE9SIE9USEVSCiAqIExJQUJJTElUWSwgV0hFVEhFUiBJTiBBTiBBQ1RJT04gT0YgQ09OVFJBQ1QsIFRPUlQgT1IgT1RIRVJXSVNFLCBBUklTSU5HIEZST00sCiAqIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFIFNPRlRXQVJFIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4KICogVEhFIFNPRlRXQVJFLgogKi8KI2luY2x1ZGUgImh3LmgiCiNpbmNsdWRlICJpc2EuaCIKI2luY2x1ZGUgInBjLmgiCiNpbmNsdWRlICJzeXNlbXUuaCIKI2luY2x1ZGUgImt2bS5oIgojaW5jbHVkZSAicWRldi5oIgoKLy8jZGVmaW5lIFZNUE9SVF9ERUJVRwoKI2RlZmluZSBWTVBPUlRfQ01EX0dFVFZFUlNJT04gMHgwYQojZGVmaW5lIFZNUE9SVF9DTURfR0VUUkFNU0laRSAweDE0CgojZGVmaW5lIFZNUE9SVF9FTlRSSUVTIDB4MmMKI2RlZmluZSBWTVBPUlRfTUFHSUMgICAweDU2NEQ1ODY4Cgp0eXBlZGVmIHN0cnVjdCBfVk1Qb3J0U3RhdGUKewogICAgSVNBRGV2aWNlIGRldjsKICAgIElPUG9ydFJlYWRGdW5jICpmdW5jW1ZNUE9SVF9FTlRSSUVTXTsKICAgIHZvaWQgKm9wYXF1ZVtWTVBPUlRfRU5UUklFU107Cn0gVk1Qb3J0U3RhdGU7CgpzdGF0aWMgVk1Qb3J0U3RhdGUgKnBvcnRfc3RhdGU7Cgp2b2lkIHZtcG9ydF9yZWdpc3Rlcih1bnNpZ25lZCBjaGFyIGNvbW1hbmQsIElPUG9ydFJlYWRGdW5jICpmdW5jLCB2b2lkICpvcGFxdWUpCnsKICAgIGlmIChjb21tYW5kID49IFZNUE9SVF9FTlRSSUVTKQogICAgICAgIHJldHVybjsKCiAgICBwb3J0X3N0YXRlLT5mdW5jW2NvbW1hbmRdID0gZnVuYzsKICAgIHBvcnRfc3RhdGUtPm9wYXF1ZVtjb21tYW5kXSA9IG9wYXF1ZTsKfQoKc3RhdGljIHVpbnQzMl90IHZtcG9ydF9pb3BvcnRfcmVhZCh2b2lkICpvcGFxdWUsIHVpbnQzMl90IGFkZHIpCnsKICAgIFZNUG9ydFN0YXRlICpzID0gb3BhcXVlOwogICAgQ1BVU3RhdGUgKmVudiA9IGNwdV9zaW5nbGVfZW52OwogICAgdW5zaWduZWQgY2hhciBjb21tYW5kOwogICAgdWludDMyX3QgZWF4OwoKICAgIGNwdV9zeW5jaHJvbml6ZV9zdGF0ZShlbnYpOwoKICAgIGVheCA9IGVudi0+cmVnc1tSX0VBWF07CiAgICBpZiAoZWF4ICE9IFZNUE9SVF9NQUdJQykKICAgICAgICByZXR1cm4gZWF4OwoKICAgIGNvbW1hbmQgPSBlbnYtPnJlZ3NbUl9FQ1hdOwogICAgaWYgKGNvbW1hbmQgPj0gVk1QT1JUX0VOVFJJRVMpCiAgICAgICAgcmV0dXJuIGVheDsKICAgIGlmICghcy0+ZnVuY1tjb21tYW5kXSkKICAgIHsKI2lmZGVmIFZNUE9SVF9ERUJVRwogICAgICAgIGZwcmludGYoc3RkZXJyLCAidm1wb3J0OiB1bmtub3duIGNvbW1hbmQgJXhcbiIsIGNvbW1hbmQpOwojZW5kaWYKICAgICAgICByZXR1cm4gZWF4OwogICAgfQoKICAgIHJldHVybiBzLT5mdW5jW2NvbW1hbmRdKHMtPm9wYXF1ZVtjb21tYW5kXSwgYWRkcik7Cn0KCnN0YXRpYyB2b2lkIHZtcG9ydF9pb3BvcnRfd3JpdGUodm9pZCAqb3BhcXVlLCB1aW50MzJfdCBhZGRyLCB1aW50MzJfdCB2YWwpCnsKICAgIENQVVN0YXRlICplbnYgPSBjcHVfc2luZ2xlX2VudjsKCiAgICBlbnYtPnJlZ3NbUl9FQVhdID0gdm1wb3J0X2lvcG9ydF9yZWFkKG9wYXF1ZSwgYWRkcik7Cn0KCnN0YXRpYyB1aW50MzJfdCB2bXBvcnRfY21kX2dldF92ZXJzaW9uKHZvaWQgKm9wYXF1ZSwgdWludDMyX3QgYWRkcikKewogICAgQ1BVU3RhdGUgKmVudiA9IGNwdV9zaW5nbGVfZW52OwogICAgZW52LT5yZWdzW1JfRUJYXSA9IFZNUE9SVF9NQUdJQzsKICAgIHJldHVybiA2Owp9CgpzdGF0aWMgdWludDMyX3Qgdm1wb3J0X2NtZF9yYW1fc2l6ZSh2b2lkICpvcGFxdWUsIHVpbnQzMl90IGFkZHIpCnsKICAgIENQVVN0YXRlICplbnYgPSBjcHVfc2luZ2xlX2VudjsKICAgIGVudi0+cmVnc1tSX0VCWF0gPSAweDExNzc7CiAgICByZXR1cm4gcmFtX3NpemU7Cn0KCi8qIHZtbW91c2UgaGVscGVycyAqLwp2b2lkIHZtbW91c2VfZ2V0X2RhdGEodWludDMyX3QgKmRhdGEpCnsKICAgIENQVVN0YXRlICplbnYgPSBjcHVfc2luZ2xlX2VudjsKCiAgICBkYXRhWzBdID0gZW52LT5yZWdzW1JfRUFYXTsgZGF0YVsxXSA9IGVudi0+cmVnc1tSX0VCWF07CiAgICBkYXRhWzJdID0gZW52LT5yZWdzW1JfRUNYXTsgZGF0YVszXSA9IGVudi0+cmVnc1tSX0VEWF07CiAgICBkYXRhWzRdID0gZW52LT5yZWdzW1JfRVNJXTsgZGF0YVs1XSA9IGVudi0+cmVnc1tSX0VESV07Cn0KCnZvaWQgdm1tb3VzZV9zZXRfZGF0YShjb25zdCB1aW50MzJfdCAqZGF0YSkKewogICAgQ1BVU3RhdGUgKmVudiA9IGNwdV9zaW5nbGVfZW52OwoKICAgIGVudi0+cmVnc1tSX0VBWF0gPSBkYXRhWzBdOyBlbnYtPnJlZ3NbUl9FQlhdID0gZGF0YVsxXTsKICAgIGVudi0+cmVnc1tSX0VDWF0gPSBkYXRhWzJdOyBlbnYtPnJlZ3NbUl9FRFhdID0gZGF0YVszXTsKICAgIGVudi0+cmVnc1tSX0VTSV0gPSBkYXRhWzRdOyBlbnYtPnJlZ3NbUl9FREldID0gZGF0YVs1XTsKfQoKc3RhdGljIGludCB2bXBvcnRfaW5pdGZuKElTQURldmljZSAqZGV2KQp7CiAgICBWTVBvcnRTdGF0ZSAqcyA9IERPX1VQQ0FTVChWTVBvcnRTdGF0ZSwgZGV2LCBkZXYpOwoKICAgIHJlZ2lzdGVyX2lvcG9ydF9yZWFkKDB4NTY1OCwgMSwgNCwgdm1wb3J0X2lvcG9ydF9yZWFkLCBzKTsKICAgIHJlZ2lzdGVyX2lvcG9ydF93cml0ZSgweDU2NTgsIDEsIDQsIHZtcG9ydF9pb3BvcnRfd3JpdGUsIHMpOwogICAgaXNhX2luaXRfaW9wb3J0KGRldiwgMHg1NjU4KTsKICAgIHBvcnRfc3RhdGUgPSBzOwogICAgLyogUmVnaXN0ZXIgc29tZSBnZW5lcmljIHBvcnQgY29tbWFuZHMgKi8KICAgIHZtcG9ydF9yZWdpc3RlcihWTVBPUlRfQ01EX0dFVFZFUlNJT04sIHZtcG9ydF9jbWRfZ2V0X3ZlcnNpb24sIE5VTEwpOwogICAgdm1wb3J0X3JlZ2lzdGVyKFZNUE9SVF9DTURfR0VUUkFNU0laRSwgdm1wb3J0X2NtZF9yYW1fc2l6ZSwgTlVMTCk7CiAgICByZXR1cm4gMDsKfQoKc3RhdGljIElTQURldmljZUluZm8gdm1wb3J0X2luZm8gPSB7CiAgICAucWRldi5uYW1lICAgICA9ICJ2bXBvcnQiLAogICAgLnFkZXYuc2l6ZSAgICAgPSBzaXplb2YoVk1Qb3J0U3RhdGUpLAogICAgLnFkZXYubm9fdXNlciAgPSAxLAogICAgLmluaXQgICAgICAgICAgPSB2bXBvcnRfaW5pdGZuLAp9OwoKc3RhdGljIHZvaWQgdm1wb3J0X2Rldl9yZWdpc3Rlcih2b2lkKQp7CiAgICBpc2FfcWRldl9yZWdpc3Rlcigmdm1wb3J0X2luZm8pOwp9CmRldmljZV9pbml0KHZtcG9ydF9kZXZfcmVnaXN0ZXIpCg==