LyogCiAgIEJORVAgaW1wbGVtZW50YXRpb24gZm9yIExpbnV4IEJsdWV0b290aCBzdGFjayAoQmx1ZVopLgogICBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwMiBJbnZlbnRlbCBTeXN0ZW1lcwogICBXcml0dGVuIDIwMDEtMjAwMiBieQoJQ2zpbWVudCBNb3JlYXUgPGNsZW1lbnQubW9yZWF1QGludmVudGVsLmZyPgoJRGF2aWQgTGliYXVsdCAgPGRhdmlkLmxpYmF1bHRAaW52ZW50ZWwuZnI+CgogICBDb3B5cmlnaHQgKEMpIDIwMDIgTWF4aW0gS3Jhc255YW5za3kgPG1heGtAcXVhbGNvbW0uY29tPgoKICAgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKICAgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSB2ZXJzaW9uIDIgYXMKICAgcHVibGlzaGVkIGJ5IHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb247CgogICBUSEUgU09GVFdBUkUgSVMgUFJPVklERUQgIkFTIElTIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwgRVhQUkVTUwogICBPUiBJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GIE1FUkNIQU5UQUJJTElUWSwKICAgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UgQU5EIE5PTklORlJJTkdFTUVOVCBPRiBUSElSRCBQQVJUWSBSSUdIVFMuCiAgIElOIE5PIEVWRU5UIFNIQUxMIFRIRSBDT1BZUklHSFQgSE9MREVSKFMpIEFORCBBVVRIT1IoUykgQkUgTElBQkxFIEZPUiBBTlkKICAgQ0xBSU0sIE9SIEFOWSBTUEVDSUFMIElORElSRUNUIE9SIENPTlNFUVVFTlRJQUwgREFNQUdFUywgT1IgQU5ZIERBTUFHRVMgCiAgIFdIQVRTT0VWRVIgUkVTVUxUSU5HIEZST00gTE9TUyBPRiBVU0UsIERBVEEgT1IgUFJPRklUUywgV0hFVEhFUiBJTiBBTiAKICAgQUNUSU9OIE9GIENPTlRSQUNULCBORUdMSUdFTkNFIE9SIE9USEVSIFRPUlRJT1VTIEFDVElPTiwgQVJJU0lORyBPVVQgT0YgCiAgIE9SIElOIENPTk5FQ1RJT04gV0lUSCBUSEUgVVNFIE9SIFBFUkZPUk1BTkNFIE9GIFRISVMgU09GVFdBUkUuCgogICBBTEwgTElBQklMSVRZLCBJTkNMVURJTkcgTElBQklMSVRZIEZPUiBJTkZSSU5HRU1FTlQgT0YgQU5ZIFBBVEVOVFMsIAogICBDT1BZUklHSFRTLCBUUkFERU1BUktTIE9SIE9USEVSIFJJR0hUUywgUkVMQVRJTkcgVE8gVVNFIE9GIFRISVMgCiAgIFNPRlRXQVJFIElTIERJU0NMQUlNRUQuCiovCgovKgogKiAkSWQ6IG5ldGRldi5jLHYgMS44IDIwMDIvMDgvMDQgMjE6MjM6NTggbWF4ayBFeHAgJAogKi8gCgojaW5jbHVkZSA8bGludXgvbW9kdWxlLmg+CgojaW5jbHVkZSA8bGludXgvc29ja2V0Lmg+CiNpbmNsdWRlIDxsaW51eC9uZXRkZXZpY2UuaD4KI2luY2x1ZGUgPGxpbnV4L2V0aGVyZGV2aWNlLmg+CiNpbmNsdWRlIDxsaW51eC9za2J1ZmYuaD4KI2luY2x1ZGUgPGxpbnV4L3dhaXQuaD4KCiNpbmNsdWRlIDxhc20vdW5hbGlnbmVkLmg+CgojaW5jbHVkZSA8bmV0L2JsdWV0b290aC9ibHVldG9vdGguaD4KI2luY2x1ZGUgPG5ldC9ibHVldG9vdGgvaGNpX2NvcmUuaD4KI2luY2x1ZGUgPG5ldC9ibHVldG9vdGgvbDJjYXAuaD4KCiNpbmNsdWRlICJibmVwLmgiCgojaWZuZGVmIENPTkZJR19CVF9CTkVQX0RFQlVHCiN1bmRlZiAgQlRfREJHCiNkZWZpbmUgQlRfREJHKCBBLi4uICkKI2VuZGlmCgojZGVmaW5lIEJORVBfVFhfUVVFVUVfTEVOIDIwCgpzdGF0aWMgaW50IGJuZXBfbmV0X29wZW4oc3RydWN0IG5ldF9kZXZpY2UgKmRldikKewoJbmV0aWZfc3RhcnRfcXVldWUoZGV2KTsKCXJldHVybiAwOwp9CgpzdGF0aWMgaW50IGJuZXBfbmV0X2Nsb3NlKHN0cnVjdCBuZXRfZGV2aWNlICpkZXYpCnsKCW5ldGlmX3N0b3BfcXVldWUoZGV2KTsKCXJldHVybiAwOwp9CgpzdGF0aWMgc3RydWN0IG5ldF9kZXZpY2Vfc3RhdHMgKmJuZXBfbmV0X2dldF9zdGF0cyhzdHJ1Y3QgbmV0X2RldmljZSAqZGV2KQp7CglzdHJ1Y3QgYm5lcF9zZXNzaW9uICpzID0gZGV2LT5wcml2OwoJcmV0dXJuICZzLT5zdGF0czsKfQoKc3RhdGljIHZvaWQgYm5lcF9uZXRfc2V0X21jX2xpc3Qoc3RydWN0IG5ldF9kZXZpY2UgKmRldikKewojaWZkZWYgQ09ORklHX0JUX0JORVBfTUNfRklMVEVSCglzdHJ1Y3QgYm5lcF9zZXNzaW9uICpzID0gZGV2LT5wcml2OwoJc3RydWN0IHNvY2sgKnNrID0gcy0+c29jay0+c2s7CglzdHJ1Y3QgYm5lcF9zZXRfZmlsdGVyX3JlcSAqcjsKCXN0cnVjdCBza19idWZmICpza2I7CglpbnQgc2l6ZTsKCglCVF9EQkcoIiVzIG1jX2NvdW50ICVkIiwgZGV2LT5uYW1lLCBkZXYtPm1jX2NvdW50KTsKCglzaXplID0gc2l6ZW9mKCpyKSArIChCTkVQX01BWF9NVUxUSUNBU1RfRklMVEVSUyArIDEpICogRVRIX0FMRU4gKiAyOwoJc2tiICA9IGFsbG9jX3NrYihzaXplLCBHRlBfQVRPTUlDKTsKCWlmICghc2tiKSB7CgkJQlRfRVJSKCIlcyBNdWx0aWNhc3QgbGlzdCBhbGxvY2F0aW9uIGZhaWxlZCIsIGRldi0+bmFtZSk7CgkJcmV0dXJuOwoJfQoKCXIgPSAodm9pZCAqKSBza2ItPmRhdGE7CglfX3NrYl9wdXQoc2tiLCBzaXplb2YoKnIpKTsKCglyLT50eXBlID0gQk5FUF9DT05UUk9MOwoJci0+Y3RybCA9IEJORVBfRklMVEVSX01VTFRJX0FERFJfU0VUOwoKICAgICAgICBpZiAoZGV2LT5mbGFncyAmIChJRkZfUFJPTUlTQyB8IElGRl9BTExNVUxUSSkpIHsKCQl1OCBzdGFydFtFVEhfQUxFTl0gPSB7IDB4MDEgfTsKCgkJLyogUmVxdWVzdCBhbGwgYWRkcmVzc2VzICovCgkJbWVtY3B5KF9fc2tiX3B1dChza2IsIEVUSF9BTEVOKSwgc3RhcnQsIEVUSF9BTEVOKTsKCQltZW1jcHkoX19za2JfcHV0KHNrYiwgRVRIX0FMRU4pLCBkZXYtPmJyb2FkY2FzdCwgRVRIX0FMRU4pOwoJCXItPmxlbiA9IGh0b25zKEVUSF9BTEVOICogMik7Cgl9IGVsc2UgewogICAgICAgICAgICAgICAgc3RydWN0IGRldl9tY19saXN0ICpkbWkgPSBkZXYtPm1jX2xpc3Q7CgkJaW50IGksIGxlbiA9IHNrYi0+bGVuOwoKCQlpZiAoZGV2LT5mbGFncyAmIElGRl9CUk9BRENBU1QpIHsKCQkJbWVtY3B5KF9fc2tiX3B1dChza2IsIEVUSF9BTEVOKSwgZGV2LT5icm9hZGNhc3QsIEVUSF9BTEVOKTsKCQkJbWVtY3B5KF9fc2tiX3B1dChza2IsIEVUSF9BTEVOKSwgZGV2LT5icm9hZGNhc3QsIEVUSF9BTEVOKTsKCQl9CQoJCQoJCS8qIEZJWE1FOiBXZSBzaG91bGQgZ3JvdXAgYWRkcmVzc2VzIGhlcmUuICovCgoJCWZvciAoaSA9IDA7IGkgPCBkZXYtPm1jX2NvdW50ICYmIGkgPCBCTkVQX01BWF9NVUxUSUNBU1RfRklMVEVSUzsgaSsrKSB7CgkJCW1lbWNweShfX3NrYl9wdXQoc2tiLCBFVEhfQUxFTiksIGRtaS0+ZG1pX2FkZHIsIEVUSF9BTEVOKTsKCQkJbWVtY3B5KF9fc2tiX3B1dChza2IsIEVUSF9BTEVOKSwgZG1pLT5kbWlfYWRkciwgRVRIX0FMRU4pOwoJCQlkbWkgPSBkbWktPm5leHQ7CgkJfQoJCXItPmxlbiA9IGh0b25zKHNrYi0+bGVuIC0gbGVuKTsKCX0KCglza2JfcXVldWVfdGFpbCgmc2stPnNrX3dyaXRlX3F1ZXVlLCBza2IpOwoJd2FrZV91cF9pbnRlcnJ1cHRpYmxlKHNrLT5za19zbGVlcCk7CiNlbmRpZgp9CgpzdGF0aWMgaW50IGJuZXBfbmV0X3NldF9tYWNfYWRkcihzdHJ1Y3QgbmV0X2RldmljZSAqZGV2LCB2b2lkICphcmcpCnsKCUJUX0RCRygiJXMiLCBkZXYtPm5hbWUpOwoJcmV0dXJuIDA7Cn0KCnN0YXRpYyB2b2lkIGJuZXBfbmV0X3RpbWVvdXQoc3RydWN0IG5ldF9kZXZpY2UgKmRldikKewoJQlRfREJHKCJuZXRfdGltZW91dCIpOwoJbmV0aWZfd2FrZV9xdWV1ZShkZXYpOwp9CgpzdGF0aWMgaW50IGJuZXBfbmV0X2lvY3RsKHN0cnVjdCBuZXRfZGV2aWNlICpkZXYsIHN0cnVjdCBpZnJlcSAqaWZyLCBpbnQgY21kKQp7CglyZXR1cm4gLUVJTlZBTDsKfQoKI2lmZGVmIENPTkZJR19CVF9CTkVQX01DX0ZJTFRFUgpzdGF0aWMgaW5saW5lIGludCBibmVwX25ldF9tY19maWx0ZXIoc3RydWN0IHNrX2J1ZmYgKnNrYiwgc3RydWN0IGJuZXBfc2Vzc2lvbiAqcykKewoJc3RydWN0IGV0aGhkciAqZWggPSAodm9pZCAqKSBza2ItPmRhdGE7CgoJaWYgKChlaC0+aF9kZXN0WzBdICYgMSkgJiYgIXRlc3RfYml0KGJuZXBfbWNfaGFzaChlaC0+aF9kZXN0KSwgKHVsb25nICopICZzLT5tY19maWx0ZXIpKQoJCXJldHVybiAxOwoJcmV0dXJuIDA7Cn0KI2VuZGlmCgojaWZkZWYgQ09ORklHX0JUX0JORVBfUFJPVE9fRklMVEVSCi8qIERldGVybWluZSBldGhlciBwcm90b2NvbC4gQmFzZWQgb24gZXRoX3R5cGVfdHJhbnMuICovCnN0YXRpYyBpbmxpbmUgdTE2IGJuZXBfbmV0X2V0aF9wcm90byhzdHJ1Y3Qgc2tfYnVmZiAqc2tiKQp7CglzdHJ1Y3QgZXRoaGRyICplaCA9ICh2b2lkICopIHNrYi0+ZGF0YTsKCQoJaWYgKG50b2hzKGVoLT5oX3Byb3RvKSA+PSAxNTM2KQoJCXJldHVybiBlaC0+aF9wcm90bzsKCQkKCWlmIChnZXRfdW5hbGlnbmVkKCh1MTYgKikgc2tiLT5kYXRhKSA9PSAweEZGRkYpCgkJcmV0dXJuIGh0b25zKEVUSF9QXzgwMl8zKTsKCQkKCXJldHVybiBodG9ucyhFVEhfUF84MDJfMik7Cn0KCnN0YXRpYyBpbmxpbmUgaW50IGJuZXBfbmV0X3Byb3RvX2ZpbHRlcihzdHJ1Y3Qgc2tfYnVmZiAqc2tiLCBzdHJ1Y3QgYm5lcF9zZXNzaW9uICpzKQp7Cgl1MTYgcHJvdG8gPSBibmVwX25ldF9ldGhfcHJvdG8oc2tiKTsKCXN0cnVjdCBibmVwX3Byb3RvX2ZpbHRlciAqZiA9IHMtPnByb3RvX2ZpbHRlcjsKCWludCBpOwoJCglmb3IgKGkgPSAwOyBpIDwgQk5FUF9NQVhfUFJPVE9fRklMVEVSUyAmJiBmW2ldLmVuZDsgaSsrKSB7CgkJaWYgKHByb3RvID49IGZbaV0uc3RhcnQgJiYgcHJvdG8gPD0gZltpXS5lbmQpCgkJCXJldHVybiAwOwoJfQoKCUJUX0RCRygiQk5FUDogZmlsdGVyZWQgc2tiICVwLCBwcm90byAweCUuNHgiLCBza2IsIHByb3RvKTsKCXJldHVybiAxOwp9CiNlbmRpZgoKc3RhdGljIGludCBibmVwX25ldF94bWl0KHN0cnVjdCBza19idWZmICpza2IsIHN0cnVjdCBuZXRfZGV2aWNlICpkZXYpCnsKCXN0cnVjdCBibmVwX3Nlc3Npb24gKnMgPSBkZXYtPnByaXY7CglzdHJ1Y3Qgc29jayAqc2sgPSBzLT5zb2NrLT5zazsKCglCVF9EQkcoInNrYiAlcCwgZGV2ICVwIiwgc2tiLCBkZXYpOwoKI2lmZGVmIENPTkZJR19CVF9CTkVQX01DX0ZJTFRFUgoJaWYgKGJuZXBfbmV0X21jX2ZpbHRlcihza2IsIHMpKSB7CgkJa2ZyZWVfc2tiKHNrYik7CgkJcmV0dXJuIDA7Cgl9CiNlbmRpZgoJCiNpZmRlZiBDT05GSUdfQlRfQk5FUF9QUk9UT19GSUxURVIKCWlmIChibmVwX25ldF9wcm90b19maWx0ZXIoc2tiLCBzKSkgewoJCWtmcmVlX3NrYihza2IpOwoJCXJldHVybiAwOwoJfQojZW5kaWYKCQoJLyoKCSAqIFdlIGNhbm5vdCBzZW5kIEwyQ0FQIHBhY2tldHMgZnJvbSBoZXJlIGFzIHdlIGFyZSBwb3RlbnRpYWxseSBpbiBhIGJoLgoJICogU28gd2UgaGF2ZSB0byBxdWV1ZSB0aGVtIGFuZCB3YWtlIHVwIHNlc3Npb24gdGhyZWFkIHdoaWNoIGlzIHNsZWVwaW5nCgkgKiBvbiB0aGUgc2stPnNrX3NsZWVwLgoJICovCglkZXYtPnRyYW5zX3N0YXJ0ID0gamlmZmllczsKCXNrYl9xdWV1ZV90YWlsKCZzay0+c2tfd3JpdGVfcXVldWUsIHNrYik7Cgl3YWtlX3VwX2ludGVycnVwdGlibGUoc2stPnNrX3NsZWVwKTsKCglpZiAoc2tiX3F1ZXVlX2xlbigmc2stPnNrX3dyaXRlX3F1ZXVlKSA+PSBCTkVQX1RYX1FVRVVFX0xFTikgewoJCUJUX0RCRygidHggcXVldWUgaXMgZnVsbCIpOwoKCQkvKiBTdG9wIHF1ZXVpbmcuCgkJICogU2Vzc2lvbiB0aHJlYWQgd2lsbCBkbyBuZXRpZl93YWtlX3F1ZXVlKCkgKi8KCQluZXRpZl9zdG9wX3F1ZXVlKGRldik7Cgl9CgoJcmV0dXJuIDA7Cn0KCnZvaWQgYm5lcF9uZXRfc2V0dXAoc3RydWN0IG5ldF9kZXZpY2UgKmRldikKewoKCW1lbXNldChkZXYtPmJyb2FkY2FzdCwgMHhmZiwgRVRIX0FMRU4pOwoJZGV2LT5hZGRyX2xlbiA9IEVUSF9BTEVOOwoKCWV0aGVyX3NldHVwKGRldik7CgoJZGV2LT5vcGVuICAgICAgICAgICAgPSBibmVwX25ldF9vcGVuOwoJZGV2LT5zdG9wICAgICAgICAgICAgPSBibmVwX25ldF9jbG9zZTsKCWRldi0+aGFyZF9zdGFydF94bWl0ID0gYm5lcF9uZXRfeG1pdDsKCWRldi0+Z2V0X3N0YXRzICAgICAgID0gYm5lcF9uZXRfZ2V0X3N0YXRzOwoJZGV2LT5kb19pb2N0bCAgICAgICAgPSBibmVwX25ldF9pb2N0bDsKCWRldi0+c2V0X21hY19hZGRyZXNzID0gYm5lcF9uZXRfc2V0X21hY19hZGRyOwoJZGV2LT5zZXRfbXVsdGljYXN0X2xpc3QgPSBibmVwX25ldF9zZXRfbWNfbGlzdDsKCglkZXYtPndhdGNoZG9nX3RpbWVvICA9IEhaICogMjsKCWRldi0+dHhfdGltZW91dCAgICAgID0gYm5lcF9uZXRfdGltZW91dDsKfQo=