LyogCiAqIEVtYWdpYyBFTUkgMnw2IHVzYiBhdWRpbyBpbnRlcmZhY2UgZmlybXdhcmUgbG9hZGVyLgogKiBDb3B5cmlnaHQgKEMpIDIwMDIKICogCVRhcGlvIExheHN0cvZtICh0YXBpby5sYXhzdHJvbUBpcHRpbWUuZmkpCiAqCiAqIFRoaXMgcHJvZ3JhbSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IgbW9kaWZ5CiAqIGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UsIGFzIHB1Ymxpc2hlZCBieQogKiB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uLCB2ZXJzaW9uIDIuCiAqIAogKiBlbWkyNi5jLHYgMS4xMyAyMDAyLzAzLzA4IDEzOjEwOjI2IHRhcGlvIEV4cAogKi8KI2luY2x1ZGUgPGxpbnV4L2tlcm5lbC5oPgojaW5jbHVkZSA8bGludXgvZXJybm8uaD4KI2luY2x1ZGUgPGxpbnV4L3NsYWIuaD4KI2luY2x1ZGUgPGxpbnV4L21vZHVsZS5oPgojaW5jbHVkZSA8bGludXgvaW5pdC5oPgojaW5jbHVkZSA8bGludXgvdXNiLmg+CgojZGVmaW5lIE1BWF9JTlRFTF9IRVhfUkVDT1JEX0xFTkdUSCAxNgp0eXBlZGVmIHN0cnVjdCBfSU5URUxfSEVYX1JFQ09SRAp7CglfX3UzMglsZW5ndGg7CglfX3UzMglhZGRyZXNzOwoJX191MzIJdHlwZTsKCV9fdTgJZGF0YVtNQVhfSU5URUxfSEVYX1JFQ09SRF9MRU5HVEhdOwp9IElOVEVMX0hFWF9SRUNPUkQsICpQSU5URUxfSEVYX1JFQ09SRDsKCi8qIGluY2x1ZGUgZmlybXdhcmUgKHZhcmlhYmxlcykgKi8KI2luY2x1ZGUgImVtaTI2X2Z3LmgiCgojZGVmaW5lIEVNSTI2X1ZFTkRPUl9JRCAJCTB4MDg2YSAgLyogRW1hZ2ljIFNvZnQtdW5kIEhhcmR3YXJlIEdtQkggKi8KI2RlZmluZSBFTUkyNl9QUk9EVUNUX0lECQkweDAxMDAJLyogRU1JIDJ8NiB3aXRob3V0IGZpcm13YXJlICovCiNkZWZpbmUgRU1JMjZCX1BST0RVQ1RfSUQJCTB4MDEwMgkvKiBFTUkgMnw2IHdpdGhvdXQgZmlybXdhcmUgKi8KCiNkZWZpbmUgQU5DSE9SX0xPQURfSU5URVJOQUwJMHhBMAkvKiBWZW5kb3Igc3BlY2lmaWMgcmVxdWVzdCBjb2RlIGZvciBBbmNob3IgVXBsb2FkL0Rvd25sb2FkIChUaGlzIG9uZSBpcyBpbXBsZW1lbnRlZCBpbiB0aGUgY29yZSkgKi8KI2RlZmluZSBBTkNIT1JfTE9BRF9FWFRFUk5BTAkweEEzCS8qIFRoaXMgY29tbWFuZCBpcyBub3QgaW1wbGVtZW50ZWQgaW4gdGhlIGNvcmUuIFJlcXVpcmVzIGZpcm13YXJlICovCiNkZWZpbmUgQU5DSE9SX0xPQURfRlBHQQkweEE1CS8qIFRoaXMgY29tbWFuZCBpcyBub3QgaW1wbGVtZW50ZWQgaW4gdGhlIGNvcmUuIFJlcXVpcmVzIGZpcm13YXJlLiBFbWFnaWMgZXh0ZW5zaW9uICovCiNkZWZpbmUgTUFYX0lOVEVSTkFMX0FERFJFU1MJMHgxQjNGCS8qIFRoaXMgaXMgdGhlIGhpZ2hlc3QgaW50ZXJuYWwgUkFNIGFkZHJlc3MgZm9yIHRoZSBBTjIxMzFRICovCiNkZWZpbmUgQ1BVQ1NfUkVHCQkweDdGOTIgIC8qIEVaLVVTQiBDb250cm9sIGFuZCBTdGF0dXMgUmVnaXN0ZXIuICBCaXQgMCBjb250cm9scyA4MDUxIHJlc2V0ICovIAojZGVmaW5lIElOVEVSTkFMX1JBTShhZGRyZXNzKSAgIChhZGRyZXNzIDw9IE1BWF9JTlRFUk5BTF9BRERSRVNTKQoKc3RhdGljIGludCBlbWkyNl93cml0ZW1lbW9yeSggc3RydWN0IHVzYl9kZXZpY2UgKmRldiwgaW50IGFkZHJlc3MsIHVuc2lnbmVkIGNoYXIgKmRhdGEsIGludCBsZW5ndGgsIF9fdTggYlJlcXVlc3QpOwpzdGF0aWMgaW50IGVtaTI2X3NldF9yZXNldChzdHJ1Y3QgdXNiX2RldmljZSAqZGV2LCB1bnNpZ25lZCBjaGFyIHJlc2V0X2JpdCk7CnN0YXRpYyBpbnQgZW1pMjZfbG9hZF9maXJtd2FyZSAoc3RydWN0IHVzYl9kZXZpY2UgKmRldik7CnN0YXRpYyBpbnQgZW1pMjZfcHJvYmUoc3RydWN0IHVzYl9pbnRlcmZhY2UgKmludGYsIGNvbnN0IHN0cnVjdCB1c2JfZGV2aWNlX2lkICppZCk7CnN0YXRpYyB2b2lkIGVtaTI2X2Rpc2Nvbm5lY3Qoc3RydWN0IHVzYl9pbnRlcmZhY2UgKmludGYpOwpzdGF0aWMgaW50IF9faW5pdCBlbWkyNl9pbml0ICh2b2lkKTsKc3RhdGljIHZvaWQgX19leGl0IGVtaTI2X2V4aXQgKHZvaWQpOwoKCi8qIHRoYW5rcyB0byBkcml2ZXJzL3VzYi9zZXJpYWwva2V5c3Bhbl9wZGEuYyBjb2RlICovCnN0YXRpYyBpbnQgZW1pMjZfd3JpdGVtZW1vcnkgKHN0cnVjdCB1c2JfZGV2aWNlICpkZXYsIGludCBhZGRyZXNzLCB1bnNpZ25lZCBjaGFyICpkYXRhLCBpbnQgbGVuZ3RoLCBfX3U4IHJlcXVlc3QpCnsKCWludCByZXN1bHQ7Cgl1bnNpZ25lZCBjaGFyICpidWZmZXIgPSAga21hbGxvYyAobGVuZ3RoLCBHRlBfS0VSTkVMKTsKCglpZiAoIWJ1ZmZlcikgewoJCWVycigiZW1pMjY6IGttYWxsb2MoJWQpIGZhaWxlZC4iLCBsZW5ndGgpOwoJCXJldHVybiAtRU5PTUVNOwoJfQoJbWVtY3B5IChidWZmZXIsIGRhdGEsIGxlbmd0aCk7CgkvKiBOb3RlOiB1c2JfY29udHJvbF9tc2cgcmV0dXJucyBuZWdhdGl2ZSB2YWx1ZSBvbiBlcnJvciBvciBsZW5ndGggb2YgdGhlCgkgKiAJCSBkYXRhIHRoYXQgd2FzIHdyaXR0ZW4hICovCglyZXN1bHQgPSB1c2JfY29udHJvbF9tc2cgKGRldiwgdXNiX3NuZGN0cmxwaXBlKGRldiwgMCksIHJlcXVlc3QsIDB4NDAsIGFkZHJlc3MsIDAsIGJ1ZmZlciwgbGVuZ3RoLCAzMDApOwoJa2ZyZWUgKGJ1ZmZlcik7CglyZXR1cm4gcmVzdWx0Owp9CgovKiB0aGFua3MgdG8gZHJpdmVycy91c2Ivc2VyaWFsL2tleXNwYW5fcGRhLmMgY29kZSAqLwpzdGF0aWMgaW50IGVtaTI2X3NldF9yZXNldCAoc3RydWN0IHVzYl9kZXZpY2UgKmRldiwgdW5zaWduZWQgY2hhciByZXNldF9iaXQpCnsKCWludCByZXNwb25zZTsKCWluZm8oIiVzIC0gJWQiLCBfX0ZVTkNUSU9OX18sIHJlc2V0X2JpdCk7CgkvKiBwcmludGsoS0VSTl9ERUJVRyAiJXMgLSAlZCIsIF9fRlVOQ1RJT05fXywgcmVzZXRfYml0KTsgKi8KCXJlc3BvbnNlID0gZW1pMjZfd3JpdGVtZW1vcnkgKGRldiwgQ1BVQ1NfUkVHLCAmcmVzZXRfYml0LCAxLCAweGEwKTsKCWlmIChyZXNwb25zZSA8IDApIHsKCQllcnIoImVtaTI2OiBzZXRfcmVzZXQgKCVkKSBmYWlsZWQiLCByZXNldF9iaXQpOwoJfQoJcmV0dXJuIHJlc3BvbnNlOwp9CgojZGVmaW5lIEZXX0xPQURfU0laRQkJMTAyMwoKc3RhdGljIGludCBlbWkyNl9sb2FkX2Zpcm13YXJlIChzdHJ1Y3QgdXNiX2RldmljZSAqZGV2KQp7CglpbnQgZXJyOwoJaW50IGk7CglpbnQgcG9zID0gMDsJLyogUG9zaXRpb24gaW4gaGV4IHJlY29yZCAqLwoJX191MzIgYWRkcjsJLyogQWRkcmVzcyB0byB3cml0ZSAqLwoJX191OCAqYnVmOwoKCWJ1ZiA9IGttYWxsb2MoRldfTE9BRF9TSVpFLCBHRlBfS0VSTkVMKTsKCWlmICghYnVmKSB7CgkJZXJyKCAiJXMgLSBlcnJvciBsb2FkaW5nIGZpcm13YXJlOiBlcnJvciA9ICVkIiwgX19GVU5DVElPTl9fLCAtRU5PTUVNKTsKCQllcnIgPSAtRU5PTUVNOwoJCWdvdG8gd3JhcGVycjsKCX0KCgkvKiBBc3NlcnQgcmVzZXQgKHN0b3AgdGhlIENQVSBpbiB0aGUgRU1JKSAqLwoJZXJyID0gZW1pMjZfc2V0X3Jlc2V0KGRldiwxKTsKCWlmIChlcnIgPCAwKSB7CgkJZXJyKCAiJXMgLSBlcnJvciBsb2FkaW5nIGZpcm13YXJlOiBlcnJvciA9ICVkIiwgX19GVU5DVElPTl9fLCBlcnIpOwoJCWdvdG8gd3JhcGVycjsKCX0KCgkvKiAxLiBXZSBuZWVkIHRvIHB1dCB0aGUgbG9hZGVyIGZvciB0aGUgRlBHQSBpbnRvIHRoZSBFWi1VU0IgKi8KCWZvciAoaT0wOyBnX0xvYWRlcltpXS50eXBlID09IDA7IGkrKykgewoJCWVyciA9IGVtaTI2X3dyaXRlbWVtb3J5KGRldiwgZ19Mb2FkZXJbaV0uYWRkcmVzcywgZ19Mb2FkZXJbaV0uZGF0YSwgZ19Mb2FkZXJbaV0ubGVuZ3RoLCBBTkNIT1JfTE9BRF9JTlRFUk5BTCk7CgkJaWYgKGVyciA8IDApIHsKCQkJZXJyKCIlcyAtIGVycm9yIGxvYWRpbmcgZmlybXdhcmU6IGVycm9yID0gJWQiLCBfX0ZVTkNUSU9OX18sIGVycik7CgkJCWdvdG8gd3JhcGVycjsKCQl9Cgl9CgoJLyogRGUtYXNzZXJ0IHJlc2V0IChsZXQgdGhlIENQVSBydW4pICovCgllcnIgPSBlbWkyNl9zZXRfcmVzZXQoZGV2LDApOwoKCS8qIDIuIFdlIHVwbG9hZCB0aGUgRlBHQSBmaXJtd2FyZSBpbnRvIHRoZSBFTUkKCSAqIE5vdGU6IGNvbGxlY3QgdXAgdG8gMTAyMyAoeWVzISkgYnl0ZXMgYW5kIHNlbmQgdGhlbSB3aXRoCgkgKiBhIHNpbmdsZSByZXF1ZXN0LiBUaGlzIGlzIF9tdWNoXyBmYXN0ZXIhICovCglkbyB7CgkJaSA9IDA7CgkJYWRkciA9IGdfYml0c3RyZWFtW3Bvc10uYWRkcmVzczsKCgkJLyogaW50ZWwgaGV4IHJlY29yZHMgYXJlIHRlcm1pbmF0ZWQgd2l0aCB0eXBlIDAgZWxlbWVudCAqLwoJCXdoaWxlICgoZ19iaXRzdHJlYW1bcG9zXS50eXBlID09IDApICYmIChpICsgZ19iaXRzdHJlYW1bcG9zXS5sZW5ndGggPCBGV19MT0FEX1NJWkUpKSB7CgkJCW1lbWNweShidWYgKyBpLCBnX2JpdHN0cmVhbVtwb3NdLmRhdGEsIGdfYml0c3RyZWFtW3Bvc10ubGVuZ3RoKTsKCQkJaSArPSBnX2JpdHN0cmVhbVtwb3NdLmxlbmd0aDsKCQkJcG9zKys7CgkJfQoJCWVyciA9IGVtaTI2X3dyaXRlbWVtb3J5KGRldiwgYWRkciwgYnVmLCBpLCBBTkNIT1JfTE9BRF9GUEdBKTsKCQlpZiAoZXJyIDwgMCkgewoJCQllcnIoIiVzIC0gZXJyb3IgbG9hZGluZyBmaXJtd2FyZTogZXJyb3IgPSAlZCIsIF9fRlVOQ1RJT05fXywgZXJyKTsKCQkJZ290byB3cmFwZXJyOwoJCX0KCX0gd2hpbGUgKGkgPiAwKTsKCgkvKiBBc3NlcnQgcmVzZXQgKHN0b3AgdGhlIENQVSBpbiB0aGUgRU1JKSAqLwoJZXJyID0gZW1pMjZfc2V0X3Jlc2V0KGRldiwxKTsKCWlmIChlcnIgPCAwKSB7CgkJZXJyKCIlcyAtIGVycm9yIGxvYWRpbmcgZmlybXdhcmU6IGVycm9yID0gJWQiLCBfX0ZVTkNUSU9OX18sIGVycik7CgkJZ290byB3cmFwZXJyOwoJfQoKCS8qIDMuIFdlIG5lZWQgdG8gcHV0IHRoZSBsb2FkZXIgZm9yIHRoZSBmaXJtd2FyZSBpbnRvIHRoZSBFWi1VU0IgKGFnYWluLi4uKSAqLwoJZm9yIChpPTA7IGdfTG9hZGVyW2ldLnR5cGUgPT0gMDsgaSsrKSB7CgkJZXJyID0gZW1pMjZfd3JpdGVtZW1vcnkoZGV2LCBnX0xvYWRlcltpXS5hZGRyZXNzLCBnX0xvYWRlcltpXS5kYXRhLCBnX0xvYWRlcltpXS5sZW5ndGgsIEFOQ0hPUl9MT0FEX0lOVEVSTkFMKTsKCQlpZiAoZXJyIDwgMCkgewoJCQllcnIoIiVzIC0gZXJyb3IgbG9hZGluZyBmaXJtd2FyZTogZXJyb3IgPSAlZCIsIF9fRlVOQ1RJT05fXywgZXJyKTsKCQkJZ290byB3cmFwZXJyOwoJCX0KCX0KCgkvKiBEZS1hc3NlcnQgcmVzZXQgKGxldCB0aGUgQ1BVIHJ1bikgKi8KCWVyciA9IGVtaTI2X3NldF9yZXNldChkZXYsMCk7CglpZiAoZXJyIDwgMCkgewoJCWVycigiJXMgLSBlcnJvciBsb2FkaW5nIGZpcm13YXJlOiBlcnJvciA9ICVkIiwgX19GVU5DVElPTl9fLCBlcnIpOwoJCWdvdG8gd3JhcGVycjsKCX0KCgkvKiA0LiBXZSBwdXQgdGhlIHBhcnQgb2YgdGhlIGZpcm13YXJlIHRoYXQgbGllcyBpbiB0aGUgZXh0ZXJuYWwgUkFNIGludG8gdGhlIEVaLVVTQiAqLwoJZm9yIChpPTA7IGdfRmlybXdhcmVbaV0udHlwZSA9PSAwOyBpKyspIHsKCQlpZiAoIUlOVEVSTkFMX1JBTShnX0Zpcm13YXJlW2ldLmFkZHJlc3MpKSB7CgkJCWVyciA9IGVtaTI2X3dyaXRlbWVtb3J5KGRldiwgZ19GaXJtd2FyZVtpXS5hZGRyZXNzLCBnX0Zpcm13YXJlW2ldLmRhdGEsIGdfRmlybXdhcmVbaV0ubGVuZ3RoLCBBTkNIT1JfTE9BRF9FWFRFUk5BTCk7CgkJCWlmIChlcnIgPCAwKSB7CgkJCQllcnIoIiVzIC0gZXJyb3IgbG9hZGluZyBmaXJtd2FyZTogZXJyb3IgPSAlZCIsIF9fRlVOQ1RJT05fXywgZXJyKTsKCQkJCWdvdG8gd3JhcGVycjsKCQkJfQoJCX0KCX0KCQoJLyogQXNzZXJ0IHJlc2V0IChzdG9wIHRoZSBDUFUgaW4gdGhlIEVNSSkgKi8KCWVyciA9IGVtaTI2X3NldF9yZXNldChkZXYsMSk7CglpZiAoZXJyIDwgMCkgewoJCWVycigiJXMgLSBlcnJvciBsb2FkaW5nIGZpcm13YXJlOiBlcnJvciA9ICVkIiwgX19GVU5DVElPTl9fLCBlcnIpOwoJCWdvdG8gd3JhcGVycjsKCX0KCglmb3IgKGk9MDsgZ19GaXJtd2FyZVtpXS50eXBlID09IDA7IGkrKykgewoJCWlmIChJTlRFUk5BTF9SQU0oZ19GaXJtd2FyZVtpXS5hZGRyZXNzKSkgewoJCQllcnIgPSBlbWkyNl93cml0ZW1lbW9yeShkZXYsIGdfRmlybXdhcmVbaV0uYWRkcmVzcywgZ19GaXJtd2FyZVtpXS5kYXRhLCBnX0Zpcm13YXJlW2ldLmxlbmd0aCwgQU5DSE9SX0xPQURfSU5URVJOQUwpOwoJCQlpZiAoZXJyIDwgMCkgewoJCQkJZXJyKCIlcyAtIGVycm9yIGxvYWRpbmcgZmlybXdhcmU6IGVycm9yID0gJWQiLCBfX0ZVTkNUSU9OX18sIGVycik7CgkJCQlnb3RvIHdyYXBlcnI7CgkJCX0KCQl9Cgl9CgoJLyogRGUtYXNzZXJ0IHJlc2V0IChsZXQgdGhlIENQVSBydW4pICovCgllcnIgPSBlbWkyNl9zZXRfcmVzZXQoZGV2LDApOwoJaWYgKGVyciA8IDApIHsKCQllcnIoIiVzIC0gZXJyb3IgbG9hZGluZyBmaXJtd2FyZTogZXJyb3IgPSAlZCIsIF9fRlVOQ1RJT05fXywgZXJyKTsKCQlnb3RvIHdyYXBlcnI7Cgl9CgoJLyogcmV0dXJuIDEgdG8gZmFpbCB0aGUgZHJpdmVyIGluaWFsaXphdGlvbgoJICogYW5kIGdpdmUgcmVhbCBkcml2ZXIgY2hhbmdlIHRvIGxvYWQgKi8KCWVyciA9IDE7Cgp3cmFwZXJyOgoJa2ZyZWUoYnVmKTsKCXJldHVybiBlcnI7Cn0KCnN0YXRpYyBzdHJ1Y3QgdXNiX2RldmljZV9pZCBpZF90YWJsZSBbXSA9IHsKCXsgVVNCX0RFVklDRShFTUkyNl9WRU5ET1JfSUQsIEVNSTI2X1BST0RVQ1RfSUQpIH0sCgl7IFVTQl9ERVZJQ0UoRU1JMjZfVkVORE9SX0lELCBFTUkyNkJfUFJPRFVDVF9JRCkgfSwKCXsgfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8qIFRlcm1pbmF0aW5nIGVudHJ5ICovCn07CgpNT0RVTEVfREVWSUNFX1RBQkxFICh1c2IsIGlkX3RhYmxlKTsKCnN0YXRpYyBpbnQgZW1pMjZfcHJvYmUoc3RydWN0IHVzYl9pbnRlcmZhY2UgKmludGYsIGNvbnN0IHN0cnVjdCB1c2JfZGV2aWNlX2lkICppZCkKewoJc3RydWN0IHVzYl9kZXZpY2UgKmRldiA9IGludGVyZmFjZV90b191c2JkZXYoaW50Zik7CgoJaW5mbygiJXMgc3RhcnQiLCBfX0ZVTkNUSU9OX18pOyAKCgllbWkyNl9sb2FkX2Zpcm13YXJlKGRldik7CgoJLyogZG8gbm90IHJldHVybiB0aGUgZHJpdmVyIGNvbnRleHQsIGxldCByZWFsIGF1ZGlvIGRyaXZlciBkbyB0aGF0ICovCglyZXR1cm4gLUVJTzsKfQoKc3RhdGljIHZvaWQgZW1pMjZfZGlzY29ubmVjdChzdHJ1Y3QgdXNiX2ludGVyZmFjZSAqaW50ZikKewp9CgpzdGF0aWMgc3RydWN0IHVzYl9kcml2ZXIgZW1pMjZfZHJpdmVyID0gewoJLm93bmVyCQk9IFRISVNfTU9EVUxFLAoJLm5hbWUJCT0gImVtaTI2IC0gZmlybXdhcmUgbG9hZGVyIiwKCS5wcm9iZQkJPSBlbWkyNl9wcm9iZSwKCS5kaXNjb25uZWN0CT0gZW1pMjZfZGlzY29ubmVjdCwKCS5pZF90YWJsZQk9IGlkX3RhYmxlLAp9OwoKc3RhdGljIGludCBfX2luaXQgZW1pMjZfaW5pdCAodm9pZCkKewoJcmV0dXJuIHVzYl9yZWdpc3RlcigmZW1pMjZfZHJpdmVyKTsKfQoKc3RhdGljIHZvaWQgX19leGl0IGVtaTI2X2V4aXQgKHZvaWQpCnsKCXVzYl9kZXJlZ2lzdGVyICgmZW1pMjZfZHJpdmVyKTsKfQoKbW9kdWxlX2luaXQoZW1pMjZfaW5pdCk7Cm1vZHVsZV9leGl0KGVtaTI2X2V4aXQpOwoKTU9EVUxFX0FVVEhPUigidGFwaW8gbGF4c3Ry9m0iKTsKTU9EVUxFX0RFU0NSSVBUSU9OKCJFbWFnaWMgRU1JIDJ8NiBmaXJtd2FyZSBsb2FkZXIuIik7Ck1PRFVMRV9MSUNFTlNFKCJHUEwiKTsKCi8qIHZpOmFpOnN5bnRheD1jOnN3PTg6dHM9ODp0dz04MAogKi8K