LyoKICogbGludXgvYXJjaC9hcm0vcGxhdC1vbWFwL2RtYS5jCiAqCiAqIENvcHlyaWdodCAoQykgMjAwMyBOb2tpYSBDb3Jwb3JhdGlvbgogKiBBdXRob3I6IEp1aGEgWXJq9mzkIDxqdWhhLnlyam9sYUBub2tpYS5jb20+CiAqIERNQSBjaGFubmVsIGxpbmtpbmcgZm9yIDE2MTAgYnkgU2FtdWVsIE9ydGl6IDxzYW11ZWwub3J0aXpAbm9raWEuY29tPgogKiBHcmFwaGljcyBETUEgYW5kIExDRCBETUEgZ3JhcGhpY3MgdHJhbmZvcm1hdGlvbnMKICogYnkgSW1yZSBEZWFrIDxpbXJlLmRlYWtAbm9raWEuY29tPgogKiBTb21lIGZ1bmN0aW9ucyBiYXNlZCBvbiBlYXJsaWVyIGRtYS1vbWFwLmMgQ29weXJpZ2h0IChDKSAyMDAxIFJpZGdlUnVuLCBJbmMuCiAqCiAqIFN1cHBvcnQgZnVuY3Rpb25zIGZvciB0aGUgT01BUCBpbnRlcm5hbCBETUEgY2hhbm5lbHMuCiAqCiAqIFRoaXMgcHJvZ3JhbSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IgbW9kaWZ5CiAqIGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgdmVyc2lvbiAyIGFzCiAqIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uLgogKgogKi8KCiNpbmNsdWRlIDxsaW51eC9tb2R1bGUuaD4KI2luY2x1ZGUgPGxpbnV4L2luaXQuaD4KI2luY2x1ZGUgPGxpbnV4L3NjaGVkLmg+CiNpbmNsdWRlIDxsaW51eC9zcGlubG9jay5oPgojaW5jbHVkZSA8bGludXgvZXJybm8uaD4KI2luY2x1ZGUgPGxpbnV4L2ludGVycnVwdC5oPgoKI2luY2x1ZGUgPGFzbS9zeXN0ZW0uaD4KI2luY2x1ZGUgPGFzbS9pcnEuaD4KI2luY2x1ZGUgPGFzbS9oYXJkd2FyZS5oPgojaW5jbHVkZSA8YXNtL2RtYS5oPgojaW5jbHVkZSA8YXNtL2lvLmg+CgojaW5jbHVkZSA8YXNtL2FyY2gvdGMuaD4KCiNkZWZpbmUgT01BUF9ETUFfQUNUSVZFCQkweDAxCgojZGVmaW5lIE9NQVBfRE1BX0NDUl9FTgkJKDEgPDwgNykKCiNkZWZpbmUgT01BUF9GVU5DX01VWF9BUk1fQkFTRQkoMHhmZmZlMTAwMCArIDB4ZWMpCgpzdGF0aWMgaW50IGVuYWJsZV8xNTEwX21vZGUgPSAwOwoKc3RydWN0IG9tYXBfZG1hX2xjaCB7CglpbnQgbmV4dF9sY2g7CglpbnQgZGV2X2lkOwoJdTE2IHNhdmVkX2NzcjsKCXUxNiBlbmFibGVkX2lycXM7Cgljb25zdCBjaGFyICpkZXZfbmFtZTsKCXZvaWQgKCogY2FsbGJhY2spKGludCBsY2gsIHUxNiBjaF9zdGF0dXMsIHZvaWQgKmRhdGEpOwoJdm9pZCAqZGF0YTsKCWxvbmcgZmxhZ3M7Cn07CgpzdGF0aWMgaW50IGRtYV9jaGFuX2NvdW50OwoKc3RhdGljIHNwaW5sb2NrX3QgZG1hX2NoYW5fbG9jazsKc3RhdGljIHN0cnVjdCBvbWFwX2RtYV9sY2ggZG1hX2NoYW5bT01BUF9MT0dJQ0FMX0RNQV9DSF9DT1VOVF07Cgpjb25zdCBzdGF0aWMgdTggZG1hX2lycVtPTUFQX0xPR0lDQUxfRE1BX0NIX0NPVU5UXSA9IHsKCUlOVF9ETUFfQ0gwXzYsIElOVF9ETUFfQ0gxXzcsIElOVF9ETUFfQ0gyXzgsIElOVF9ETUFfQ0gzLAoJSU5UX0RNQV9DSDQsIElOVF9ETUFfQ0g1LCBJTlRfMTYxMF9ETUFfQ0g2LCBJTlRfMTYxMF9ETUFfQ0g3LAoJSU5UXzE2MTBfRE1BX0NIOCwgSU5UXzE2MTBfRE1BX0NIOSwgSU5UXzE2MTBfRE1BX0NIMTAsCglJTlRfMTYxMF9ETUFfQ0gxMSwgSU5UXzE2MTBfRE1BX0NIMTIsIElOVF8xNjEwX0RNQV9DSDEzLAoJSU5UXzE2MTBfRE1BX0NIMTQsIElOVF8xNjEwX0RNQV9DSDE1LCBJTlRfRE1BX0xDRAp9OwoKc3RhdGljIGlubGluZSBpbnQgZ2V0X2dkbWFfZGV2KGludCByZXEpCnsKCXUzMiByZWcgPSBPTUFQX0ZVTkNfTVVYX0FSTV9CQVNFICsgKChyZXEgLSAxKSAvIDUpICogNDsKCWludCBzaGlmdCA9ICgocmVxIC0gMSkgJSA1KSAqIDY7CgoJcmV0dXJuICgob21hcF9yZWFkbChyZWcpID4+IHNoaWZ0KSAmIDB4M2YpICsgMTsKfQoKc3RhdGljIGlubGluZSB2b2lkIHNldF9nZG1hX2RldihpbnQgcmVxLCBpbnQgZGV2KQp7Cgl1MzIgcmVnID0gT01BUF9GVU5DX01VWF9BUk1fQkFTRSArICgocmVxIC0gMSkgLyA1KSAqIDQ7CglpbnQgc2hpZnQgPSAoKHJlcSAtIDEpICUgNSkgKiA2OwoJdTMyIGw7CgoJbCA9IG9tYXBfcmVhZGwocmVnKTsKCWwgJj0gfigweDNmIDw8IHNoaWZ0KTsKCWwgfD0gKGRldiAtIDEpIDw8IHNoaWZ0OwoJb21hcF93cml0ZWwobCwgcmVnKTsKfQoKc3RhdGljIHZvaWQgY2xlYXJfbGNoX3JlZ3MoaW50IGxjaCkKewoJaW50IGk7Cgl1MzIgbGNoX2Jhc2UgPSBPTUFQX0RNQV9CQVNFICsgbGNoICogMHg0MDsKCglmb3IgKGkgPSAwOyBpIDwgMHgyYzsgaSArPSAyKQoJCW9tYXBfd3JpdGV3KDAsIGxjaF9iYXNlICsgaSk7Cn0KCnZvaWQgb21hcF9zZXRfZG1hX3ByaW9yaXR5KGludCBkc3RfcG9ydCwgaW50IHByaW9yaXR5KQp7Cgl1bnNpZ25lZCBsb25nIHJlZzsKCXUzMiBsOwoKCXN3aXRjaCAoZHN0X3BvcnQpIHsKCWNhc2UgT01BUF9ETUFfUE9SVF9PQ1BfVDE6CS8qIEZGRkVDQzAwICovCgkJcmVnID0gT01BUF9UQ19PQ1BUMV9QUklPUjsKCQlicmVhazsKCWNhc2UgT01BUF9ETUFfUE9SVF9PQ1BfVDI6CS8qIEZGRkVDQ0QwICovCgkJcmVnID0gT01BUF9UQ19PQ1BUMl9QUklPUjsKCQlicmVhazsKCWNhc2UgT01BUF9ETUFfUE9SVF9FTUlGRjoJLyogRkZGRUNDMDggKi8KCQlyZWcgPSBPTUFQX1RDX0VNSUZGX1BSSU9SOwoJCWJyZWFrOwoJY2FzZSBPTUFQX0RNQV9QT1JUX0VNSUZTOgkvKiBGRkZFQ0MwNCAqLwoJCXJlZyA9IE9NQVBfVENfRU1JRlNfUFJJT1I7CgkJYnJlYWs7CglkZWZhdWx0OgoJCUJVRygpOwoJCXJldHVybjsKCX0KCWwgPSBvbWFwX3JlYWRsKHJlZyk7CglsICY9IH4oMHhmIDw8IDgpOwoJbCB8PSAocHJpb3JpdHkgJiAweGYpIDw8IDg7CglvbWFwX3dyaXRlbChsLCByZWcpOwp9Cgp2b2lkIG9tYXBfc2V0X2RtYV90cmFuc2Zlcl9wYXJhbXMoaW50IGxjaCwgaW50IGRhdGFfdHlwZSwgaW50IGVsZW1fY291bnQsCgkJCQkgIGludCBmcmFtZV9jb3VudCwgaW50IHN5bmNfbW9kZSkKewoJdTE2IHc7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ1NEUChsY2gpKTsKCXcgJj0gfjB4MDM7Cgl3IHw9IGRhdGFfdHlwZTsKCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NTRFAobGNoKSk7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ0NSKGxjaCkpOwoJdyAmPSB+KDEgPDwgNSk7CglpZiAoc3luY19tb2RlID09IE9NQVBfRE1BX1NZTkNfRlJBTUUpCgkJdyB8PSAxIDw8IDU7CglvbWFwX3dyaXRldyh3LCBPTUFQX0RNQV9DQ1IobGNoKSk7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ0NSMihsY2gpKTsKCXcgJj0gfigxIDw8IDIpOwoJaWYgKHN5bmNfbW9kZSA9PSBPTUFQX0RNQV9TWU5DX0JMT0NLKQoJCXcgfD0gMSA8PCAyOwoJb21hcF93cml0ZXcodywgT01BUF9ETUFfQ0NSMihsY2gpKTsKCglvbWFwX3dyaXRldyhlbGVtX2NvdW50LCBPTUFQX0RNQV9DRU4obGNoKSk7CglvbWFwX3dyaXRldyhmcmFtZV9jb3VudCwgT01BUF9ETUFfQ0ZOKGxjaCkpOwoKfQp2b2lkIG9tYXBfc2V0X2RtYV9jb2xvcl9tb2RlKGludCBsY2gsIGVudW0gb21hcF9kbWFfY29sb3JfbW9kZSBtb2RlLCB1MzIgY29sb3IpCnsKCXUxNiB3OwoKCUJVR19PTihvbWFwX2RtYV9pbl8xNTEwX21vZGUoKSk7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ0NSMihsY2gpKSAmIH4weDAzOwoJc3dpdGNoIChtb2RlKSB7CgljYXNlIE9NQVBfRE1BX0NPTlNUQU5UX0ZJTEw6CgkJdyB8PSAweDAxOwoJCWJyZWFrOwoJY2FzZSBPTUFQX0RNQV9UUkFOU1BBUkVOVF9DT1BZOgoJCXcgfD0gMHgwMjsKCQlicmVhazsKCWNhc2UgT01BUF9ETUFfQ09MT1JfRElTOgoJCWJyZWFrOwoJZGVmYXVsdDoKCQlCVUcoKTsKCX0KCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NDUjIobGNoKSk7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfTENIX0NUUkwobGNoKSkgJiB+MHgwZjsKCS8qIERlZmF1bHQgaXMgY2hhbm5lbCB0eXBlIDJEICovCglpZiAobW9kZSkgewoJCW9tYXBfd3JpdGV3KCh1MTYpY29sb3IsIE9NQVBfRE1BX0NPTE9SX0wobGNoKSk7CgkJb21hcF93cml0ZXcoKHUxNikoY29sb3IgPj4gMTYpLCBPTUFQX0RNQV9DT0xPUl9VKGxjaCkpOwoJCXcgfD0gMTsJCS8qIENoYW5uZWwgdHlwZSBHICovCgl9CglvbWFwX3dyaXRldyh3LCBPTUFQX0RNQV9MQ0hfQ1RSTChsY2gpKTsKfQoKCnZvaWQgb21hcF9zZXRfZG1hX3NyY19wYXJhbXMoaW50IGxjaCwgaW50IHNyY19wb3J0LCBpbnQgc3JjX2Ftb2RlLAoJCQkgICAgIHVuc2lnbmVkIGxvbmcgc3JjX3N0YXJ0KQp7Cgl1MTYgdzsKCgl3ID0gb21hcF9yZWFkdyhPTUFQX0RNQV9DU0RQKGxjaCkpOwoJdyAmPSB+KDB4MWYgPDwgMik7Cgl3IHw9IHNyY19wb3J0IDw8IDI7CglvbWFwX3dyaXRldyh3LCBPTUFQX0RNQV9DU0RQKGxjaCkpOwoKCXcgPSBvbWFwX3JlYWR3KE9NQVBfRE1BX0NDUihsY2gpKTsKCXcgJj0gfigweDAzIDw8IDEyKTsKCXcgfD0gc3JjX2Ftb2RlIDw8IDEyOwoJb21hcF93cml0ZXcodywgT01BUF9ETUFfQ0NSKGxjaCkpOwoKCW9tYXBfd3JpdGV3KHNyY19zdGFydCA+PiAxNiwgT01BUF9ETUFfQ1NTQV9VKGxjaCkpOwoJb21hcF93cml0ZXcoc3JjX3N0YXJ0LCBPTUFQX0RNQV9DU1NBX0wobGNoKSk7Cn0KCnZvaWQgb21hcF9zZXRfZG1hX3NyY19pbmRleChpbnQgbGNoLCBpbnQgZWlkeCwgaW50IGZpZHgpCnsKCW9tYXBfd3JpdGV3KGVpZHgsIE9NQVBfRE1BX0NTRUkobGNoKSk7CglvbWFwX3dyaXRldyhmaWR4LCBPTUFQX0RNQV9DU0ZJKGxjaCkpOwp9Cgp2b2lkIG9tYXBfc2V0X2RtYV9zcmNfZGF0YV9wYWNrKGludCBsY2gsIGludCBlbmFibGUpCnsKCXUxNiB3OwoKCXcgPSBvbWFwX3JlYWR3KE9NQVBfRE1BX0NTRFAobGNoKSkgJiB+KDEgPDwgNik7Cgl3IHw9IGVuYWJsZSA/ICgxIDw8IDYpIDogMDsKCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NTRFAobGNoKSk7Cn0KCnZvaWQgb21hcF9zZXRfZG1hX3NyY19idXJzdF9tb2RlKGludCBsY2gsIGVudW0gb21hcF9kbWFfYnVyc3RfbW9kZSBidXJzdF9tb2RlKQp7Cgl1MTYgdzsKCgl3ID0gb21hcF9yZWFkdyhPTUFQX0RNQV9DU0RQKGxjaCkpICYgfigweDAzIDw8IDcpOwoJc3dpdGNoIChidXJzdF9tb2RlKSB7CgljYXNlIE9NQVBfRE1BX0RBVEFfQlVSU1RfRElTOgoJCWJyZWFrOwoJY2FzZSBPTUFQX0RNQV9EQVRBX0JVUlNUXzQ6CgkJdyB8PSAoMHgwMSA8PCA3KTsKCQlicmVhazsKCWNhc2UgT01BUF9ETUFfREFUQV9CVVJTVF84OgoJCS8qIG5vdCBzdXBwb3J0ZWQgYnkgY3VycmVudCBoYXJkd2FyZQoJCSAqIHcgfD0gKDB4MDMgPDwgNyk7CgkJICogZmFsbCB0aHJvdWdoCgkJICovCglkZWZhdWx0OgoJCUJVRygpOwoJfQoJb21hcF93cml0ZXcodywgT01BUF9ETUFfQ1NEUChsY2gpKTsKfQoKdm9pZCBvbWFwX3NldF9kbWFfZGVzdF9wYXJhbXMoaW50IGxjaCwgaW50IGRlc3RfcG9ydCwgaW50IGRlc3RfYW1vZGUsCgkJCSAgICAgIHVuc2lnbmVkIGxvbmcgZGVzdF9zdGFydCkKewoJdTE2IHc7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ1NEUChsY2gpKTsKCXcgJj0gfigweDFmIDw8IDkpOwoJdyB8PSBkZXN0X3BvcnQgPDwgOTsKCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NTRFAobGNoKSk7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ0NSKGxjaCkpOwoJdyAmPSB+KDB4MDMgPDwgMTQpOwoJdyB8PSBkZXN0X2Ftb2RlIDw8IDE0OwoJb21hcF93cml0ZXcodywgT01BUF9ETUFfQ0NSKGxjaCkpOwoKCW9tYXBfd3JpdGV3KGRlc3Rfc3RhcnQgPj4gMTYsIE9NQVBfRE1BX0NEU0FfVShsY2gpKTsKCW9tYXBfd3JpdGV3KGRlc3Rfc3RhcnQsIE9NQVBfRE1BX0NEU0FfTChsY2gpKTsKfQoKdm9pZCBvbWFwX3NldF9kbWFfZGVzdF9pbmRleChpbnQgbGNoLCBpbnQgZWlkeCwgaW50IGZpZHgpCnsKCW9tYXBfd3JpdGV3KGVpZHgsIE9NQVBfRE1BX0NERUkobGNoKSk7CglvbWFwX3dyaXRldyhmaWR4LCBPTUFQX0RNQV9DREZJKGxjaCkpOwp9Cgp2b2lkIG9tYXBfc2V0X2RtYV9kZXN0X2RhdGFfcGFjayhpbnQgbGNoLCBpbnQgZW5hYmxlKQp7Cgl1MTYgdzsKCgl3ID0gb21hcF9yZWFkdyhPTUFQX0RNQV9DU0RQKGxjaCkpICYgfigxIDw8IDEzKTsKCXcgfD0gZW5hYmxlID8gKDEgPDwgMTMpIDogMDsKCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NTRFAobGNoKSk7Cn0KCnZvaWQgb21hcF9zZXRfZG1hX2Rlc3RfYnVyc3RfbW9kZShpbnQgbGNoLCBlbnVtIG9tYXBfZG1hX2J1cnN0X21vZGUgYnVyc3RfbW9kZSkKewoJdTE2IHc7CgoJdyA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ1NEUChsY2gpKSAmIH4oMHgwMyA8PCAxNCk7Cglzd2l0Y2ggKGJ1cnN0X21vZGUpIHsKCWNhc2UgT01BUF9ETUFfREFUQV9CVVJTVF9ESVM6CgkJYnJlYWs7CgljYXNlIE9NQVBfRE1BX0RBVEFfQlVSU1RfNDoKCQl3IHw9ICgweDAxIDw8IDE0KTsKCQlicmVhazsKCWNhc2UgT01BUF9ETUFfREFUQV9CVVJTVF84OgoJCXcgfD0gKDB4MDMgPDwgMTQpOwoJCWJyZWFrOwoJZGVmYXVsdDoKCQlwcmludGsoS0VSTl9FUlIgIkludmFsaWQgRE1BIGJ1cnN0IG1vZGVcbiIpOwoJCUJVRygpOwoJCXJldHVybjsKCX0KCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NTRFAobGNoKSk7Cn0KCnN0YXRpYyBpbmxpbmUgdm9pZCBpbml0X2ludHIoaW50IGxjaCkKewoJdTE2IHc7CgoJLyogUmVhZCBDU1IgdG8gbWFrZSBzdXJlIGl0J3MgY2xlYXJlZC4gKi8KCXcgPSBvbWFwX3JlYWR3KE9NQVBfRE1BX0NTUihsY2gpKTsKCS8qIEVuYWJsZSBzb21lIG5pY2UgaW50ZXJydXB0cy4gKi8KCW9tYXBfd3JpdGV3KGRtYV9jaGFuW2xjaF0uZW5hYmxlZF9pcnFzLCBPTUFQX0RNQV9DSUNSKGxjaCkpOwoJZG1hX2NoYW5bbGNoXS5mbGFncyB8PSBPTUFQX0RNQV9BQ1RJVkU7Cn0KCnN0YXRpYyBpbmxpbmUgdm9pZCBlbmFibGVfbG5rKGludCBsY2gpCnsKCXUxNiB3OwoKCS8qIENsZWFyIHRoZSBTVE9QX0xOSyBiaXRzICovCgl3ID0gb21hcF9yZWFkdyhPTUFQX0RNQV9DTE5LX0NUUkwobGNoKSk7Cgl3ICY9IH4oMSA8PCAxNCk7CglvbWFwX3dyaXRldyh3LCBPTUFQX0RNQV9DTE5LX0NUUkwobGNoKSk7CgoJLyogQW5kIHNldCB0aGUgRU5BQkxFX0xOSyBiaXRzICovCglpZiAoZG1hX2NoYW5bbGNoXS5uZXh0X2xjaCAhPSAtMSkKCQlvbWFwX3dyaXRldyhkbWFfY2hhbltsY2hdLm5leHRfbGNoIHwgKDEgPDwgMTUpLAoJCQkgICAgT01BUF9ETUFfQ0xOS19DVFJMKGxjaCkpOwp9CgpzdGF0aWMgaW5saW5lIHZvaWQgZGlzYWJsZV9sbmsoaW50IGxjaCkKewoJdTE2IHc7CgoJLyogRGlzYWJsZSBpbnRlcnJ1cHRzICovCglvbWFwX3dyaXRldygwLCBPTUFQX0RNQV9DSUNSKGxjaCkpOwoKCS8qIFNldCB0aGUgU1RPUF9MTksgYml0ICovCgl3ID0gb21hcF9yZWFkdyhPTUFQX0RNQV9DTE5LX0NUUkwobGNoKSk7Cgl3IHw9ICgxIDw8IDE0KTsKCXcgPSBvbWFwX3dyaXRldyh3LCBPTUFQX0RNQV9DTE5LX0NUUkwobGNoKSk7CgoJZG1hX2NoYW5bbGNoXS5mbGFncyAmPSB+T01BUF9ETUFfQUNUSVZFOwp9Cgp2b2lkIG9tYXBfc3RhcnRfZG1hKGludCBsY2gpCnsKCXUxNiB3OwoKCWlmICghb21hcF9kbWFfaW5fMTUxMF9tb2RlKCkgJiYgZG1hX2NoYW5bbGNoXS5uZXh0X2xjaCAhPSAtMSkgewoJCWludCBuZXh0X2xjaCwgY3VyX2xjaDsKCQljaGFyIGRtYV9jaGFuX2xpbmtfbWFwW09NQVBfTE9HSUNBTF9ETUFfQ0hfQ09VTlRdOwoKCQlkbWFfY2hhbl9saW5rX21hcFtsY2hdID0gMTsKCQkvKiBTZXQgdGhlIGxpbmsgcmVnaXN0ZXIgb2YgdGhlIGZpcnN0IGNoYW5uZWwgKi8KCQllbmFibGVfbG5rKGxjaCk7CgoJCW1lbXNldChkbWFfY2hhbl9saW5rX21hcCwgMCwgc2l6ZW9mKGRtYV9jaGFuX2xpbmtfbWFwKSk7CgkJY3VyX2xjaCA9IGRtYV9jaGFuW2xjaF0ubmV4dF9sY2g7CgkJZG8gewoJCQluZXh0X2xjaCA9IGRtYV9jaGFuW2N1cl9sY2hdLm5leHRfbGNoOwoKICAgICAgICAgICAgICAgICAgICAgICAgLyogVGhlIGxvb3AgY2FzZTogd2UndmUgYmVlbiBoZXJlIGFscmVhZHkgKi8KCQkJaWYgKGRtYV9jaGFuX2xpbmtfbWFwW2N1cl9sY2hdKQoJCQkJYnJlYWs7CgkJCS8qIE1hcmsgdGhlIGN1cnJlbnQgY2hhbm5lbCAqLwoJCQlkbWFfY2hhbl9saW5rX21hcFtjdXJfbGNoXSA9IDE7CgoJCQllbmFibGVfbG5rKGN1cl9sY2gpOwoJCQlpbml0X2ludHIoY3VyX2xjaCk7CgoJCQljdXJfbGNoID0gbmV4dF9sY2g7CgkJfSB3aGlsZSAobmV4dF9sY2ggIT0gLTEpOwoJfQoKCWluaXRfaW50cihsY2gpOwoKCXcgPSBvbWFwX3JlYWR3KE9NQVBfRE1BX0NDUihsY2gpKTsKCXcgfD0gT01BUF9ETUFfQ0NSX0VOOwoJb21hcF93cml0ZXcodywgT01BUF9ETUFfQ0NSKGxjaCkpOwoJZG1hX2NoYW5bbGNoXS5mbGFncyB8PSBPTUFQX0RNQV9BQ1RJVkU7Cn0KCnZvaWQgb21hcF9zdG9wX2RtYShpbnQgbGNoKQp7Cgl1MTYgdzsKCglpZiAoIW9tYXBfZG1hX2luXzE1MTBfbW9kZSgpICYmIGRtYV9jaGFuW2xjaF0ubmV4dF9sY2ggIT0gLTEpIHsKCQlpbnQgbmV4dF9sY2gsIGN1cl9sY2ggPSBsY2g7CgkJY2hhciBkbWFfY2hhbl9saW5rX21hcFtPTUFQX0xPR0lDQUxfRE1BX0NIX0NPVU5UXTsKCgkJbWVtc2V0KGRtYV9jaGFuX2xpbmtfbWFwLCAwLCBzaXplb2YoZG1hX2NoYW5fbGlua19tYXApKTsKCQlkbyB7CgkJCS8qIFRoZSBsb29wIGNhc2U6IHdlJ3ZlIGJlZW4gaGVyZSBhbHJlYWR5ICovCgkJCWlmIChkbWFfY2hhbl9saW5rX21hcFtjdXJfbGNoXSkKCQkJCWJyZWFrOwoJCQkvKiBNYXJrIHRoZSBjdXJyZW50IGNoYW5uZWwgKi8KCQkJZG1hX2NoYW5fbGlua19tYXBbY3VyX2xjaF0gPSAxOwoKCQkJZGlzYWJsZV9sbmsoY3VyX2xjaCk7CgoJCQluZXh0X2xjaCA9IGRtYV9jaGFuW2N1cl9sY2hdLm5leHRfbGNoOwoJCQljdXJfbGNoID0gbmV4dF9sY2g7CgkJfSB3aGlsZSAobmV4dF9sY2ggIT0gLTEpOwoKCQlyZXR1cm47Cgl9CgkvKiBEaXNhYmxlIGFsbCBpbnRlcnJ1cHRzIG9uIHRoZSBjaGFubmVsICovCglvbWFwX3dyaXRldygwLCBPTUFQX0RNQV9DSUNSKGxjaCkpOwoKCXcgPSBvbWFwX3JlYWR3KE9NQVBfRE1BX0NDUihsY2gpKTsKCXcgJj0gfk9NQVBfRE1BX0NDUl9FTjsKCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0NDUihsY2gpKTsKCWRtYV9jaGFuW2xjaF0uZmxhZ3MgJj0gfk9NQVBfRE1BX0FDVElWRTsKfQoKdm9pZCBvbWFwX2VuYWJsZV9kbWFfaXJxKGludCBsY2gsIHUxNiBiaXRzKQp7CglkbWFfY2hhbltsY2hdLmVuYWJsZWRfaXJxcyB8PSBiaXRzOwp9Cgp2b2lkIG9tYXBfZGlzYWJsZV9kbWFfaXJxKGludCBsY2gsIHUxNiBiaXRzKQp7CglkbWFfY2hhbltsY2hdLmVuYWJsZWRfaXJxcyAmPSB+Yml0czsKfQoKc3RhdGljIGludCBkbWFfaGFuZGxlX2NoKGludCBjaCkKewoJdTE2IGNzcjsKCglpZiAoZW5hYmxlXzE1MTBfbW9kZSAmJiBjaCA+PSA2KSB7CgkJY3NyID0gZG1hX2NoYW5bY2hdLnNhdmVkX2NzcjsKCQlkbWFfY2hhbltjaF0uc2F2ZWRfY3NyID0gMDsKCX0gZWxzZQoJCWNzciA9IG9tYXBfcmVhZHcoT01BUF9ETUFfQ1NSKGNoKSk7CglpZiAoZW5hYmxlXzE1MTBfbW9kZSAmJiBjaCA8PSAyICYmIChjc3IgPj4gNykgIT0gMCkgewoJCWRtYV9jaGFuW2NoICsgNl0uc2F2ZWRfY3NyID0gY3NyID4+IDc7CgkJY3NyICY9IDB4N2Y7Cgl9CglpZiAoIWNzcikKCQlyZXR1cm4gMDsKCWlmICh1bmxpa2VseShkbWFfY2hhbltjaF0uZGV2X2lkID09IC0xKSkgewoJCXByaW50ayhLRVJOX1dBUk5JTkcgIlNwdXJpb3VzIGludGVycnVwdCBmcm9tIERNQSBjaGFubmVsICVkIChDU1IgJTA0eClcbiIsCgkJICAgICAgIGNoLCBjc3IpOwoJCXJldHVybiAwOwoJfQoJaWYgKHVubGlrZWx5KGNzciAmIE9NQVBfRE1BX1RPVVRfSVJRKSkKCQlwcmludGsoS0VSTl9XQVJOSU5HICJETUEgdGltZW91dCB3aXRoIGRldmljZSAlZFxuIiwgZG1hX2NoYW5bY2hdLmRldl9pZCk7CglpZiAodW5saWtlbHkoY3NyICYgT01BUF9ETUFfRFJPUF9JUlEpKQoJCXByaW50ayhLRVJOX1dBUk5JTkcgIkRNQSBzeW5jaHJvbml6YXRpb24gZXZlbnQgZHJvcCBvY2N1cnJlZCB3aXRoIGRldmljZSAlZFxuIiwKCQkgICAgICAgZG1hX2NoYW5bY2hdLmRldl9pZCk7CglpZiAobGlrZWx5KGNzciAmIE9NQVBfRE1BX0JMT0NLX0lSUSkpCgkJZG1hX2NoYW5bY2hdLmZsYWdzICY9IH5PTUFQX0RNQV9BQ1RJVkU7CglpZiAobGlrZWx5KGRtYV9jaGFuW2NoXS5jYWxsYmFjayAhPSBOVUxMKSkKCQlkbWFfY2hhbltjaF0uY2FsbGJhY2soY2gsIGNzciwgZG1hX2NoYW5bY2hdLmRhdGEpOwoJcmV0dXJuIDE7Cn0KCnN0YXRpYyBpcnFyZXR1cm5fdCBkbWFfaXJxX2hhbmRsZXIoaW50IGlycSwgdm9pZCAqZGV2X2lkLCBzdHJ1Y3QgcHRfcmVncyAqcmVncykKewoJaW50IGNoID0gKChpbnQpIGRldl9pZCkgLSAxOwoJaW50IGhhbmRsZWQgPSAwOwoKCWZvciAoOzspIHsKCQlpbnQgaGFuZGxlZF9ub3cgPSAwOwoKCQloYW5kbGVkX25vdyArPSBkbWFfaGFuZGxlX2NoKGNoKTsKCQlpZiAoZW5hYmxlXzE1MTBfbW9kZSAmJiBkbWFfY2hhbltjaCArIDZdLnNhdmVkX2NzcikKCQkJaGFuZGxlZF9ub3cgKz0gZG1hX2hhbmRsZV9jaChjaCArIDYpOwoJCWlmICghaGFuZGxlZF9ub3cpCgkJCWJyZWFrOwoJCWhhbmRsZWQgKz0gaGFuZGxlZF9ub3c7Cgl9CgoJcmV0dXJuIGhhbmRsZWQgPyBJUlFfSEFORExFRCA6IElSUV9OT05FOwp9CgppbnQgb21hcF9yZXF1ZXN0X2RtYShpbnQgZGV2X2lkLCBjb25zdCBjaGFyICpkZXZfbmFtZSwKCQkgICAgIHZvaWQgKCogY2FsbGJhY2spKGludCBsY2gsIHUxNiBjaF9zdGF0dXMsIHZvaWQgKmRhdGEpLAoJCSAgICAgdm9pZCAqZGF0YSwgaW50ICpkbWFfY2hfb3V0KQp7CglpbnQgY2gsIGZyZWVfY2ggPSAtMTsKCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CglzdHJ1Y3Qgb21hcF9kbWFfbGNoICpjaGFuOwoKCXNwaW5fbG9ja19pcnFzYXZlKCZkbWFfY2hhbl9sb2NrLCBmbGFncyk7Cglmb3IgKGNoID0gMDsgY2ggPCBkbWFfY2hhbl9jb3VudDsgY2grKykgewoJCWlmIChmcmVlX2NoID09IC0xICYmIGRtYV9jaGFuW2NoXS5kZXZfaWQgPT0gLTEpIHsKCQkJZnJlZV9jaCA9IGNoOwoJCQlpZiAoZGV2X2lkID09IDApCgkJCQlicmVhazsKCQl9Cgl9CglpZiAoZnJlZV9jaCA9PSAtMSkgewoJCXNwaW5fdW5sb2NrX2lycXJlc3RvcmUoJmRtYV9jaGFuX2xvY2ssIGZsYWdzKTsKCQlyZXR1cm4gLUVCVVNZOwoJfQoJY2hhbiA9IGRtYV9jaGFuICsgZnJlZV9jaDsKCWNoYW4tPmRldl9pZCA9IGRldl9pZDsKCWNsZWFyX2xjaF9yZWdzKGZyZWVfY2gpOwoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmZG1hX2NoYW5fbG9jaywgZmxhZ3MpOwoKCWNoYW4tPmRldl9pZCA9IGRldl9pZDsKCWNoYW4tPmRldl9uYW1lID0gZGV2X25hbWU7CgljaGFuLT5jYWxsYmFjayA9IGNhbGxiYWNrOwoJY2hhbi0+ZGF0YSA9IGRhdGE7CgljaGFuLT5lbmFibGVkX2lycXMgPSBPTUFQX0RNQV9UT1VUX0lSUSB8IE9NQVBfRE1BX0RST1BfSVJRIHwgT01BUF9ETUFfQkxPQ0tfSVJROwoKCWlmIChjcHVfaXNfb21hcDE2eHgoKSkgewoJCS8qIElmIHRoZSBzeW5jIGRldmljZSBpcyBzZXQsIGNvbmZpZ3VyZSBpdCBkeW5hbWljYWxseS4gKi8KCQlpZiAoZGV2X2lkICE9IDApIHsKCQkJc2V0X2dkbWFfZGV2KGZyZWVfY2ggKyAxLCBkZXZfaWQpOwoJCQlkZXZfaWQgPSBmcmVlX2NoICsgMTsKCQl9CgkJLyogRGlzYWJsZSB0aGUgMTUxMCBjb21wYXRpYmlsaXR5IG1vZGUgYW5kIHNldCB0aGUgc3luYyBkZXZpY2UKCQkgKiBpZC4gKi8KCQlvbWFwX3dyaXRldyhkZXZfaWQgfCAoMSA8PCAxMCksIE9NQVBfRE1BX0NDUihmcmVlX2NoKSk7Cgl9IGVsc2UgewoJCW9tYXBfd3JpdGV3KGRldl9pZCwgT01BUF9ETUFfQ0NSKGZyZWVfY2gpKTsKCX0KCSpkbWFfY2hfb3V0ID0gZnJlZV9jaDsKCglyZXR1cm4gMDsKfQoKdm9pZCBvbWFwX2ZyZWVfZG1hKGludCBjaCkKewoJdW5zaWduZWQgbG9uZyBmbGFnczsKCglzcGluX2xvY2tfaXJxc2F2ZSgmZG1hX2NoYW5fbG9jaywgZmxhZ3MpOwoJaWYgKGRtYV9jaGFuW2NoXS5kZXZfaWQgPT0gLTEpIHsKCQlwcmludGsoIm9tYXBfZG1hOiB0cnlpbmcgdG8gZnJlZSBub25hbGxvY2F0ZWQgRE1BIGNoYW5uZWwgJWRcbiIsIGNoKTsKCQlzcGluX3VubG9ja19pcnFyZXN0b3JlKCZkbWFfY2hhbl9sb2NrLCBmbGFncyk7CgkJcmV0dXJuOwoJfQoJZG1hX2NoYW5bY2hdLmRldl9pZCA9IC0xOwoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmZG1hX2NoYW5fbG9jaywgZmxhZ3MpOwoKCS8qIERpc2FibGUgYWxsIERNQSBpbnRlcnJ1cHRzIGZvciB0aGUgY2hhbm5lbC4gKi8KCW9tYXBfd3JpdGV3KDAsIE9NQVBfRE1BX0NJQ1IoY2gpKTsKCS8qIE1ha2Ugc3VyZSB0aGUgRE1BIHRyYW5zZmVyIGlzIHN0b3BwZWQuICovCglvbWFwX3dyaXRldygwLCBPTUFQX0RNQV9DQ1IoY2gpKTsKfQoKaW50IG9tYXBfZG1hX2luXzE1MTBfbW9kZSh2b2lkKQp7CglyZXR1cm4gZW5hYmxlXzE1MTBfbW9kZTsKfQoKLyoKICogbGNoX3F1ZXVlIERNQSB3aWxsIHN0YXJ0IHJpZ2h0IGFmdGVyIGxjaF9oZWFkIG9uZSBpcyBmaW5pc2hlZC4KICogRm9yIHRoaXMgRE1BIGxpbmsgdG8gc3RhcnQsIHlvdSBzdGlsbCBuZWVkIHRvIHN0YXJ0IChzZWUgb21hcF9zdGFydF9kbWEpCiAqIHRoZSBmaXJzdCBvbmUuIFRoYXQgd2lsbCBmaXJlIHVwIHRoZSBlbnRpcmUgcXVldWUuCiAqLwp2b2lkIG9tYXBfZG1hX2xpbmtfbGNoIChpbnQgbGNoX2hlYWQsIGludCBsY2hfcXVldWUpCnsKCWlmIChvbWFwX2RtYV9pbl8xNTEwX21vZGUoKSkgewoJCXByaW50ayhLRVJOX0VSUiAiRE1BIGxpbmtpbmcgaXMgbm90IHN1cHBvcnRlZCBpbiAxNTEwIG1vZGVcbiIpOwoJCUJVRygpOwoJCXJldHVybjsKCX0KCglpZiAoKGRtYV9jaGFuW2xjaF9oZWFkXS5kZXZfaWQgPT0gLTEpIHx8CgkgICAgKGRtYV9jaGFuW2xjaF9xdWV1ZV0uZGV2X2lkID09IC0xKSkgewoJCXByaW50ayhLRVJOX0VSUiAib21hcF9kbWE6IHRyeWluZyB0byBsaW5rIG5vbiByZXF1ZXN0ZWQgY2hhbm5lbHNcbiIpOwoJCWR1bXBfc3RhY2soKTsKCX0KCglkbWFfY2hhbltsY2hfaGVhZF0ubmV4dF9sY2ggPSBsY2hfcXVldWU7Cn0KCi8qCiAqIE9uY2UgdGhlIERNQSBxdWV1ZSBpcyBzdG9wcGVkLCB3ZSBjYW4gZGVzdHJveSBpdC4KICovCnZvaWQgb21hcF9kbWFfdW5saW5rX2xjaCAoaW50IGxjaF9oZWFkLCBpbnQgbGNoX3F1ZXVlKQp7CglpZiAob21hcF9kbWFfaW5fMTUxMF9tb2RlKCkpIHsKCQlwcmludGsoS0VSTl9FUlIgIkRNQSBsaW5raW5nIGlzIG5vdCBzdXBwb3J0ZWQgaW4gMTUxMCBtb2RlXG4iKTsKCQlCVUcoKTsKCQlyZXR1cm47Cgl9CgoJaWYgKGRtYV9jaGFuW2xjaF9oZWFkXS5uZXh0X2xjaCAhPSBsY2hfcXVldWUgfHwKCSAgICBkbWFfY2hhbltsY2hfaGVhZF0ubmV4dF9sY2ggPT0gLTEpIHsKCQlwcmludGsoS0VSTl9FUlIgIm9tYXBfZG1hOiB0cnlpbmcgdG8gdW5saW5rIG5vbiBsaW5rZWQgY2hhbm5lbHNcbiIpOwoJCWR1bXBfc3RhY2soKTsKCX0KCgoJaWYgKChkbWFfY2hhbltsY2hfaGVhZF0uZmxhZ3MgJiBPTUFQX0RNQV9BQ1RJVkUpIHx8CgkgICAgKGRtYV9jaGFuW2xjaF9oZWFkXS5mbGFncyAmIE9NQVBfRE1BX0FDVElWRSkpIHsKCQlwcmludGsoS0VSTl9FUlIgIm9tYXBfZG1hOiBZb3UgbmVlZCB0byBzdG9wIHRoZSBETUEgY2hhbm5lbHMgYmVmb3JlIHVubGlua2luZ1xuIik7CgkJZHVtcF9zdGFjaygpOwoJfQoKCWRtYV9jaGFuW2xjaF9oZWFkXS5uZXh0X2xjaCA9IC0xOwp9CgoKc3RhdGljIHN0cnVjdCBsY2RfZG1hX2luZm8gewoJc3BpbmxvY2tfdCBsb2NrOwoJaW50IHJlc2VydmVkOwoJdm9pZCAoKiBjYWxsYmFjaykodTE2IHN0YXR1cywgdm9pZCAqZGF0YSk7Cgl2b2lkICpjYl9kYXRhOwoKCWludCBhY3RpdmU7Cgl1bnNpZ25lZCBsb25nIGFkZHIsIHNpemU7CglpbnQgcm90YXRlLCBkYXRhX3R5cGUsIHhyZXMsIHlyZXM7CglpbnQgdnhyZXM7CglpbnQgbWlycm9yOwoJaW50IHhzY2FsZSwgeXNjYWxlOwoJaW50IGV4dF9jdHJsOwoJaW50IHNyY19wb3J0OwoJaW50IHNpbmdsZV90cmFuc2ZlcjsKfSBsY2RfZG1hOwoKdm9pZCBvbWFwX3NldF9sY2RfZG1hX2IxKHVuc2lnbmVkIGxvbmcgYWRkciwgdTE2IGZiX3hyZXMsIHUxNiBmYl95cmVzLAoJCQkgaW50IGRhdGFfdHlwZSkKewoJbGNkX2RtYS5hZGRyID0gYWRkcjsKCWxjZF9kbWEuZGF0YV90eXBlID0gZGF0YV90eXBlOwoJbGNkX2RtYS54cmVzID0gZmJfeHJlczsKCWxjZF9kbWEueXJlcyA9IGZiX3lyZXM7Cn0KCnZvaWQgb21hcF9zZXRfbGNkX2RtYV9zcmNfcG9ydChpbnQgcG9ydCkKewoJbGNkX2RtYS5zcmNfcG9ydCA9IHBvcnQ7Cn0KCnZvaWQgb21hcF9zZXRfbGNkX2RtYV9leHRfY29udHJvbGxlcihpbnQgZXh0ZXJuYWwpCnsKCWxjZF9kbWEuZXh0X2N0cmwgPSBleHRlcm5hbDsKfQoKdm9pZCBvbWFwX3NldF9sY2RfZG1hX3NpbmdsZV90cmFuc2ZlcihpbnQgc2luZ2xlKQp7CglsY2RfZG1hLnNpbmdsZV90cmFuc2ZlciA9IHNpbmdsZTsKfQoKCnZvaWQgb21hcF9zZXRfbGNkX2RtYV9iMV9yb3RhdGlvbihpbnQgcm90YXRlKQp7CglpZiAob21hcF9kbWFfaW5fMTUxMF9tb2RlKCkpIHsKCQlwcmludGsoS0VSTl9FUlIgIkRNQSByb3RhdGlvbiBpcyBub3Qgc3VwcG9ydGVkIGluIDE1MTAgbW9kZVxuIik7CgkJQlVHKCk7CgkJcmV0dXJuOwoJfQoJbGNkX2RtYS5yb3RhdGUgPSByb3RhdGU7Cn0KCnZvaWQgb21hcF9zZXRfbGNkX2RtYV9iMV9taXJyb3IoaW50IG1pcnJvcikKewoJaWYgKG9tYXBfZG1hX2luXzE1MTBfbW9kZSgpKSB7CgkJcHJpbnRrKEtFUk5fRVJSICJETUEgbWlycm9yIGlzIG5vdCBzdXBwb3J0ZWQgaW4gMTUxMCBtb2RlXG4iKTsKCQlCVUcoKTsKCX0KCWxjZF9kbWEubWlycm9yID0gbWlycm9yOwp9Cgp2b2lkIG9tYXBfc2V0X2xjZF9kbWFfYjFfdnhyZXModW5zaWduZWQgbG9uZyB2eHJlcykKewoJaWYgKG9tYXBfZG1hX2luXzE1MTBfbW9kZSgpKSB7CgkJcHJpbnRrKEtFUk5fRVJSICJETUEgdmlydHVhbCByZXN1bG90aW9uIGlzIG5vdCBzdXBwb3J0ZWQgIgoJCQkJImluIDE1MTAgbW9kZVxuIik7CgkJQlVHKCk7Cgl9CglsY2RfZG1hLnZ4cmVzID0gdnhyZXM7Cn0KCnZvaWQgb21hcF9zZXRfbGNkX2RtYV9iMV9zY2FsZSh1bnNpZ25lZCBpbnQgeHNjYWxlLCB1bnNpZ25lZCBpbnQgeXNjYWxlKQp7CglpZiAob21hcF9kbWFfaW5fMTUxMF9tb2RlKCkpIHsKCQlwcmludGsoS0VSTl9FUlIgIkRNQSBzY2FsZSBpcyBub3Qgc3VwcG9ydGVkIGluIDE1MTAgbW9kZVxuIik7CgkJQlVHKCk7Cgl9CglsY2RfZG1hLnhzY2FsZSA9IHhzY2FsZTsKCWxjZF9kbWEueXNjYWxlID0geXNjYWxlOwp9CgpzdGF0aWMgdm9pZCBzZXRfYjFfcmVncyh2b2lkKQp7Cgl1bnNpZ25lZCBsb25nIHRvcCwgYm90dG9tOwoJaW50IGVzOwoJdTE2IHc7Cgl1bnNpZ25lZCBsb25nIGVuLCBmbjsKCWxvbmcgZWksIGZpOwoJdW5zaWduZWQgbG9uZyB2eHJlczsKCXVuc2lnbmVkIGludCB4c2NhbGUsIHlzY2FsZTsKCglzd2l0Y2ggKGxjZF9kbWEuZGF0YV90eXBlKSB7CgljYXNlIE9NQVBfRE1BX0RBVEFfVFlQRV9TODoKCQllcyA9IDE7CgkJYnJlYWs7CgljYXNlIE9NQVBfRE1BX0RBVEFfVFlQRV9TMTY6CgkJZXMgPSAyOwoJCWJyZWFrOwoJY2FzZSBPTUFQX0RNQV9EQVRBX1RZUEVfUzMyOgoJCWVzID0gNDsKCQlicmVhazsKCWRlZmF1bHQ6CgkJQlVHKCk7CgkJcmV0dXJuOwoJfQoKCXZ4cmVzID0gbGNkX2RtYS52eHJlcyA/IGxjZF9kbWEudnhyZXMgOiBsY2RfZG1hLnhyZXM7Cgl4c2NhbGUgPSBsY2RfZG1hLnhzY2FsZSA/IGxjZF9kbWEueHNjYWxlIDogMTsKCXlzY2FsZSA9IGxjZF9kbWEueXNjYWxlID8gbGNkX2RtYS55c2NhbGUgOiAxOwoJQlVHX09OKHZ4cmVzIDwgbGNkX2RtYS54cmVzKTsKI2RlZmluZSBQSVhBRERSKHgseSkgKGxjZF9kbWEuYWRkciArICgoeSkgKiB2eHJlcyAqIHlzY2FsZSArICh4KSAqIHhzY2FsZSkgKiBlcykKI2RlZmluZSBQSVhTVEVQKHN4LCBzeSwgZHgsIGR5KSAoUElYQUREUihkeCwgZHkpIC0gUElYQUREUihzeCwgc3kpIC0gZXMgKyAxKQoJc3dpdGNoIChsY2RfZG1hLnJvdGF0ZSkgewoJY2FzZSAwOgoJCWlmICghbGNkX2RtYS5taXJyb3IpIHsKCQkJdG9wID0gUElYQUREUigwLCAwKTsKCQkJYm90dG9tID0gUElYQUREUihsY2RfZG1hLnhyZXMgLSAxLCBsY2RfZG1hLnlyZXMgLSAxKTsKCQkJLyogMTUxMCBETUEgcmVxdWlyZXMgdGhlIGJvdHRvbSBhZGRyZXNzIHRvIGJlIDIgbW9yZQoJCQkgKiB0aGFuIHRoZSBhY3R1YWwgbGFzdCBtZW1vcnkgYWNjZXNzIGxvY2F0aW9uLiAqLwoJCQlpZiAob21hcF9kbWFfaW5fMTUxMF9tb2RlKCkgJiYKCQkJICAgIGxjZF9kbWEuZGF0YV90eXBlID09IE9NQVBfRE1BX0RBVEFfVFlQRV9TMzIpCgkJCQlib3R0b20gKz0gMjsKCQkJZWkgPSBQSVhTVEVQKDAsIDAsIDEsIDApOwoJCQlmaSA9IFBJWFNURVAobGNkX2RtYS54cmVzIC0gMSwgMCwgMCwgMSk7CgkJfSBlbHNlIHsKCQkJdG9wID0gUElYQUREUihsY2RfZG1hLnhyZXMgLSAxLCAwKTsKCQkJYm90dG9tID0gUElYQUREUigwLCBsY2RfZG1hLnlyZXMgLSAxKTsKCQkJZWkgPSBQSVhTVEVQKDEsIDAsIDAsIDApOwoJCQlmaSA9IFBJWFNURVAoMCwgMCwgbGNkX2RtYS54cmVzIC0gMSwgMSk7CgkJfQoJCWVuID0gbGNkX2RtYS54cmVzOwoJCWZuID0gbGNkX2RtYS55cmVzOwoJCWJyZWFrOwoJY2FzZSA5MDoKCQlpZiAoIWxjZF9kbWEubWlycm9yKSB7CgkJCXRvcCA9IFBJWEFERFIoMCwgbGNkX2RtYS55cmVzIC0gMSk7CgkJCWJvdHRvbSA9IFBJWEFERFIobGNkX2RtYS54cmVzIC0gMSwgMCk7CgkJCWVpID0gUElYU1RFUCgwLCAxLCAwLCAwKTsKCQkJZmkgPSBQSVhTVEVQKDAsIDAsIDEsIGxjZF9kbWEueXJlcyAtIDEpOwoJCX0gZWxzZSB7CgkJCXRvcCA9IFBJWEFERFIobGNkX2RtYS54cmVzIC0gMSwgbGNkX2RtYS55cmVzIC0gMSk7CgkJCWJvdHRvbSA9IFBJWEFERFIoMCwgMCk7CgkJCWVpID0gUElYU1RFUCgwLCAxLCAwLCAwKTsKCQkJZmkgPSBQSVhTVEVQKDEsIDAsIDAsIGxjZF9kbWEueXJlcyAtIDEpOwoJCX0KCQllbiA9IGxjZF9kbWEueXJlczsKCQlmbiA9IGxjZF9kbWEueHJlczsKCQlicmVhazsKCWNhc2UgMTgwOgoJCWlmICghbGNkX2RtYS5taXJyb3IpIHsKCQkJdG9wID0gUElYQUREUihsY2RfZG1hLnhyZXMgLSAxLCBsY2RfZG1hLnlyZXMgLSAxKTsKCQkJYm90dG9tID0gUElYQUREUigwLCAwKTsKCQkJZWkgPSBQSVhTVEVQKDEsIDAsIDAsIDApOwoJCQlmaSA9IFBJWFNURVAoMCwgMSwgbGNkX2RtYS54cmVzIC0gMSwgMCk7CgkJfSBlbHNlIHsKCQkJdG9wID0gUElYQUREUigwLCBsY2RfZG1hLnlyZXMgLSAxKTsKCQkJYm90dG9tID0gUElYQUREUihsY2RfZG1hLnhyZXMgLSAxLCAwKTsKCQkJZWkgPSBQSVhTVEVQKDAsIDAsIDEsIDApOwoJCQlmaSA9IFBJWFNURVAobGNkX2RtYS54cmVzIC0gMSwgMSwgMCwgMCk7CgkJfQoJCWVuID0gbGNkX2RtYS54cmVzOwoJCWZuID0gbGNkX2RtYS55cmVzOwoJCWJyZWFrOwoJY2FzZSAyNzA6CgkJaWYgKCFsY2RfZG1hLm1pcnJvcikgewoJCQl0b3AgPSBQSVhBRERSKGxjZF9kbWEueHJlcyAtIDEsIDApOwoJCQlib3R0b20gPSBQSVhBRERSKDAsIGxjZF9kbWEueXJlcyAtIDEpOwoJCQllaSA9IFBJWFNURVAoMCwgMCwgMCwgMSk7CgkJCWZpID0gUElYU1RFUCgxLCBsY2RfZG1hLnlyZXMgLSAxLCAwLCAwKTsKCQl9IGVsc2UgewoJCQl0b3AgPSBQSVhBRERSKDAsIDApOwoJCQlib3R0b20gPSBQSVhBRERSKGxjZF9kbWEueHJlcyAtIDEsIGxjZF9kbWEueXJlcyAtIDEpOwoJCQllaSA9IFBJWFNURVAoMCwgMCwgMCwgMSk7CgkJCWZpID0gUElYU1RFUCgwLCBsY2RfZG1hLnlyZXMgLSAxLCAxLCAwKTsKCQl9CgkJZW4gPSBsY2RfZG1hLnlyZXM7CgkJZm4gPSBsY2RfZG1hLnhyZXM7CgkJYnJlYWs7CglkZWZhdWx0OgoJCUJVRygpOwoJCXJldHVybjsJLyogU3VwcmVzcyB3YXJuaW5nIGFib3V0IHVuaW5pdGlhbGl6ZWQgdmFycyAqLwoJfQoKCWlmIChvbWFwX2RtYV9pbl8xNTEwX21vZGUoKSkgewoJCW9tYXBfd3JpdGV3KHRvcCA+PiAxNiwgT01BUDE1MTBfRE1BX0xDRF9UT1BfRjFfVSk7CgkJb21hcF93cml0ZXcodG9wLCBPTUFQMTUxMF9ETUFfTENEX1RPUF9GMV9MKTsKCQlvbWFwX3dyaXRldyhib3R0b20gPj4gMTYsIE9NQVAxNTEwX0RNQV9MQ0RfQk9UX0YxX1UpOwoJCW9tYXBfd3JpdGV3KGJvdHRvbSwgT01BUDE1MTBfRE1BX0xDRF9CT1RfRjFfTCk7CgoJCXJldHVybjsKCX0KCgkvKiAxNjEwIHJlZ3MgKi8KCW9tYXBfd3JpdGV3KHRvcCA+PiAxNiwgT01BUDE2MTBfRE1BX0xDRF9UT1BfQjFfVSk7CglvbWFwX3dyaXRldyh0b3AsIE9NQVAxNjEwX0RNQV9MQ0RfVE9QX0IxX0wpOwoJb21hcF93cml0ZXcoYm90dG9tID4+IDE2LCBPTUFQMTYxMF9ETUFfTENEX0JPVF9CMV9VKTsKCW9tYXBfd3JpdGV3KGJvdHRvbSwgT01BUDE2MTBfRE1BX0xDRF9CT1RfQjFfTCk7CgoJb21hcF93cml0ZXcoZW4sIE9NQVAxNjEwX0RNQV9MQ0RfU1JDX0VOX0IxKTsKCW9tYXBfd3JpdGV3KGZuLCBPTUFQMTYxMF9ETUFfTENEX1NSQ19GTl9CMSk7CgoJdyA9IG9tYXBfcmVhZHcoT01BUDE2MTBfRE1BX0xDRF9DU0RQKTsKCXcgJj0gfjB4MDM7Cgl3IHw9IGxjZF9kbWEuZGF0YV90eXBlOwoJb21hcF93cml0ZXcodywgT01BUDE2MTBfRE1BX0xDRF9DU0RQKTsKCgl3ID0gb21hcF9yZWFkdyhPTUFQMTYxMF9ETUFfTENEX0NUUkwpOwoJLyogQWx3YXlzIHNldCB0aGUgc291cmNlIHBvcnQgYXMgU0RSQU0gZm9yIG5vdyovCgl3ICY9IH4oMHgwMyA8PCA2KTsKCWlmIChsY2RfZG1hLmNhbGxiYWNrICE9IE5VTEwpCgkJdyB8PSAxIDw8IDE7ICAgICAgICAgICAgLyogQmxvY2sgaW50ZXJydXB0IGVuYWJsZSAqLwoJZWxzZQoJCXcgJj0gfigxIDw8IDEpOwoJb21hcF93cml0ZXcodywgT01BUDE2MTBfRE1BX0xDRF9DVFJMKTsKCglpZiAoIShsY2RfZG1hLnJvdGF0ZSB8fCBsY2RfZG1hLm1pcnJvciB8fAoJICAgICAgbGNkX2RtYS52eHJlcyB8fCBsY2RfZG1hLnhzY2FsZSB8fCBsY2RfZG1hLnlzY2FsZSkpCgkJcmV0dXJuOwoKCXcgPSBvbWFwX3JlYWR3KE9NQVAxNjEwX0RNQV9MQ0RfQ0NSKTsKCS8qIFNldCB0aGUgZG91YmxlLWluZGV4ZWQgYWRkcmVzc2luZyBtb2RlICovCgl3IHw9ICgweDAzIDw8IDEyKTsKCW9tYXBfd3JpdGV3KHcsIE9NQVAxNjEwX0RNQV9MQ0RfQ0NSKTsKCglvbWFwX3dyaXRldyhlaSwgT01BUDE2MTBfRE1BX0xDRF9TUkNfRUlfQjEpOwoJb21hcF93cml0ZXcoZmkgPj4gMTYsIE9NQVAxNjEwX0RNQV9MQ0RfU1JDX0ZJX0IxX1UpOwoJb21hcF93cml0ZXcoZmksIE9NQVAxNjEwX0RNQV9MQ0RfU1JDX0ZJX0IxX0wpOwp9CgpzdGF0aWMgaXJxcmV0dXJuX3QgbGNkX2RtYV9pcnFfaGFuZGxlcihpbnQgaXJxLCB2b2lkICpkZXZfaWQsIHN0cnVjdCBwdF9yZWdzICpyZWdzKQp7Cgl1MTYgdzsKCgl3ID0gb21hcF9yZWFkdyhPTUFQMTYxMF9ETUFfTENEX0NUUkwpOwoJaWYgKHVubGlrZWx5KCEodyAmICgxIDw8IDMpKSkpIHsKCQlwcmludGsoS0VSTl9XQVJOSU5HICJTcHVyaW91cyBMQ0QgRE1BIElSUVxuIik7CgkJcmV0dXJuIElSUV9OT05FOwoJfQoJLyogQWNrIHRoZSBJUlEgKi8KCXcgfD0gKDEgPDwgMyk7CglvbWFwX3dyaXRldyh3LCBPTUFQMTYxMF9ETUFfTENEX0NUUkwpOwoJbGNkX2RtYS5hY3RpdmUgPSAwOwoJaWYgKGxjZF9kbWEuY2FsbGJhY2sgIT0gTlVMTCkKCQlsY2RfZG1hLmNhbGxiYWNrKHcsIGxjZF9kbWEuY2JfZGF0YSk7CgoJcmV0dXJuIElSUV9IQU5ETEVEOwp9CgppbnQgb21hcF9yZXF1ZXN0X2xjZF9kbWEodm9pZCAoKiBjYWxsYmFjaykodTE2IHN0YXR1cywgdm9pZCAqZGF0YSksCgkJCSB2b2lkICpkYXRhKQp7CglzcGluX2xvY2tfaXJxKCZsY2RfZG1hLmxvY2spOwoJaWYgKGxjZF9kbWEucmVzZXJ2ZWQpIHsKCQlzcGluX3VubG9ja19pcnEoJmxjZF9kbWEubG9jayk7CgkJcHJpbnRrKEtFUk5fRVJSICJMQ0QgRE1BIGNoYW5uZWwgYWxyZWFkeSByZXNlcnZlZFxuIik7CgkJQlVHKCk7CgkJcmV0dXJuIC1FQlVTWTsKCX0KCWxjZF9kbWEucmVzZXJ2ZWQgPSAxOwoJc3Bpbl91bmxvY2tfaXJxKCZsY2RfZG1hLmxvY2spOwoJbGNkX2RtYS5jYWxsYmFjayA9IGNhbGxiYWNrOwoJbGNkX2RtYS5jYl9kYXRhID0gZGF0YTsKCWxjZF9kbWEuYWN0aXZlID0gMDsKCWxjZF9kbWEuc2luZ2xlX3RyYW5zZmVyID0gMDsKCWxjZF9kbWEucm90YXRlID0gMDsKCWxjZF9kbWEudnhyZXMgPSAwOwoJbGNkX2RtYS5taXJyb3IgPSAwOwoJbGNkX2RtYS54c2NhbGUgPSAwOwoJbGNkX2RtYS55c2NhbGUgPSAwOwoJbGNkX2RtYS5leHRfY3RybCA9IDA7CglsY2RfZG1hLnNyY19wb3J0ID0gMDsKCglyZXR1cm4gMDsKfQoKdm9pZCBvbWFwX2ZyZWVfbGNkX2RtYSh2b2lkKQp7CglzcGluX2xvY2soJmxjZF9kbWEubG9jayk7CglpZiAoIWxjZF9kbWEucmVzZXJ2ZWQpIHsKCQlzcGluX3VubG9jaygmbGNkX2RtYS5sb2NrKTsKCQlwcmludGsoS0VSTl9FUlIgIkxDRCBETUEgaXMgbm90IHJlc2VydmVkXG4iKTsKCQlCVUcoKTsKCQlyZXR1cm47Cgl9CglpZiAoIWVuYWJsZV8xNTEwX21vZGUpCgkJb21hcF93cml0ZXcob21hcF9yZWFkdyhPTUFQMTYxMF9ETUFfTENEX0NDUikgJiB+MSwgT01BUDE2MTBfRE1BX0xDRF9DQ1IpOwoJbGNkX2RtYS5yZXNlcnZlZCA9IDA7CglzcGluX3VubG9jaygmbGNkX2RtYS5sb2NrKTsKfQoKdm9pZCBvbWFwX2VuYWJsZV9sY2RfZG1hKHZvaWQpCnsKCXUxNiB3OwoKCS8qIFNldCB0aGUgRW5hYmxlIGJpdCBvbmx5IGlmIGFuIGV4dGVybmFsIGNvbnRyb2xsZXIgaXMKCSAqIGNvbm5lY3RlZC4gT3RoZXJ3aXNlIHRoZSBPTUFQIGludGVybmFsIGNvbnRyb2xsZXIgd2lsbAoJICogc3RhcnQgdGhlIHRyYW5zZmVyIHdoZW4gaXQgZ2V0cyBlbmFibGVkLgoJICovCglpZiAoZW5hYmxlXzE1MTBfbW9kZSB8fCAhbGNkX2RtYS5leHRfY3RybCkKCQlyZXR1cm47CgoJdyA9IG9tYXBfcmVhZHcoT01BUDE2MTBfRE1BX0xDRF9DVFJMKTsKCXcgfD0gMSA8PCA4OwoJb21hcF93cml0ZXcodywgT01BUDE2MTBfRE1BX0xDRF9DVFJMKTsKCgl3ID0gb21hcF9yZWFkdyhPTUFQMTYxMF9ETUFfTENEX0NDUik7Cgl3IHw9IDEgPDwgNzsKCW9tYXBfd3JpdGV3KHcsIE9NQVAxNjEwX0RNQV9MQ0RfQ0NSKTsKCglsY2RfZG1hLmFjdGl2ZSA9IDE7Cn0KCnZvaWQgb21hcF9zZXR1cF9sY2RfZG1hKHZvaWQpCnsKCUJVR19PTihsY2RfZG1hLmFjdGl2ZSk7CglpZiAoIWVuYWJsZV8xNTEwX21vZGUpIHsKCQkvKiBTZXQgc29tZSByZWFzb25hYmxlIGRlZmF1bHRzICovCgkJb21hcF93cml0ZXcoMHg1NDQwLCBPTUFQMTYxMF9ETUFfTENEX0NDUik7CgkJb21hcF93cml0ZXcoMHg5MTAyLCBPTUFQMTYxMF9ETUFfTENEX0NTRFApOwoJCW9tYXBfd3JpdGV3KDB4MDAwNCwgT01BUDE2MTBfRE1BX0xDRF9MQ0hfQ1RSTCk7Cgl9CglzZXRfYjFfcmVncygpOwoJaWYgKCFlbmFibGVfMTUxMF9tb2RlKSB7CgkJdTE2IHc7CgoJCXcgPSBvbWFwX3JlYWR3KE9NQVAxNjEwX0RNQV9MQ0RfQ0NSKTsKCQkvKiBJZiBETUEgd2FzIGFscmVhZHkgYWN0aXZlIHNldCB0aGUgZW5kX3Byb2cgYml0IHRvIGhhdmUKCQkgKiB0aGUgcHJvZ3JhbW1lZCByZWdpc3RlciBzZXQgbG9hZGVkIGludG8gdGhlIGFjdGl2ZQoJCSAqIHJlZ2lzdGVyIHNldC4KCQkgKi8KCQl3IHw9IDEgPDwgMTE7CQkvKiBFbmRfcHJvZyAqLwoJCWlmICghbGNkX2RtYS5zaW5nbGVfdHJhbnNmZXIpCgkgICAgICAgIAl3IHw9ICgzIDw8IDgpOwkvKiBBdXRvX2luaXQsIHJlcGVhdCAqLwoJCW9tYXBfd3JpdGV3KHcsIE9NQVAxNjEwX0RNQV9MQ0RfQ0NSKTsKCX0KfQoKdm9pZCBvbWFwX3N0b3BfbGNkX2RtYSh2b2lkKQp7Cgl1MTYgdzsKCglsY2RfZG1hLmFjdGl2ZSA9IDA7CglpZiAoZW5hYmxlXzE1MTBfbW9kZSB8fCAhbGNkX2RtYS5leHRfY3RybCkKCQlyZXR1cm47CgoJdyA9IG9tYXBfcmVhZHcoT01BUDE2MTBfRE1BX0xDRF9DQ1IpOwoJdyAmPSB+KDEgPDwgNyk7CglvbWFwX3dyaXRldyh3LCBPTUFQMTYxMF9ETUFfTENEX0NDUik7CgoJdyA9IG9tYXBfcmVhZHcoT01BUDE2MTBfRE1BX0xDRF9DVFJMKTsKCXcgJj0gfigxIDw8IDgpOwoJb21hcF93cml0ZXcodywgT01BUDE2MTBfRE1BX0xDRF9DVFJMKTsKfQoKLyoKICogQ2xlYXJzIGFueSBETUEgc3RhdGUgc28gdGhlIERNQSBlbmdpbmUgaXMgcmVhZHkgdG8gcmVzdGFydCB3aXRoIG5ldyBidWZmZXJzCiAqIHRocm91Z2ggb21hcF9zdGFydF9kbWEoKS4gQW55IGJ1ZmZlcnMgaW4gZmxpZ2h0IGFyZSBkaXNjYXJkZWQuCiAqLwp2b2lkIG9tYXBfY2xlYXJfZG1hKGludCBsY2gpCnsKCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CglpbnQgc3RhdHVzOwoKCWxvY2FsX2lycV9zYXZlKGZsYWdzKTsKCW9tYXBfd3JpdGV3KG9tYXBfcmVhZHcoT01BUF9ETUFfQ0NSKGxjaCkpICYgfk9NQVBfRE1BX0NDUl9FTiwKCQkgICAgT01BUF9ETUFfQ0NSKGxjaCkpOwoJc3RhdHVzID0gT01BUF9ETUFfQ1NSKGxjaCk7CS8qIGNsZWFyIHBlbmRpbmcgaW50ZXJydXB0cyAqLwoJbG9jYWxfaXJxX3Jlc3RvcmUoZmxhZ3MpOwp9CgovKgogKiBSZXR1cm5zIGN1cnJlbnQgcGh5c2ljYWwgc291cmNlIGFkZHJlc3MgZm9yIHRoZSBnaXZlbiBETUEgY2hhbm5lbC4KICogSWYgdGhlIGNoYW5uZWwgaXMgcnVubmluZyB0aGUgY2FsbGVyIG11c3QgZGlzYWJsZSBpbnRlcnJ1cHRzIHByaW9yIGNhbGxpbmcKICogdGhpcyBmdW5jdGlvbiBhbmQgcHJvY2VzcyB0aGUgcmV0dXJuZWQgdmFsdWUgYmVmb3JlIHJlLWVuYWJsaW5nIGludGVycnVwdCB0bwogKiBwcmV2ZW50IHJhY2VzIHdpdGggdGhlIGludGVycnVwdCBoYW5kbGVyLiBOb3RlIHRoYXQgaW4gY29udGludW91cyBtb2RlIHRoZXJlCiAqIGlzIGEgY2hhbmNlIGZvciBDU1NBX0wgcmVnaXN0ZXIgb3ZlcmZsb3cgaW5iZXR3ZWVuIHRoZSB0d28gcmVhZHMgcmVzdWx0aW5nCiAqIGluIGluY29ycmVjdCByZXR1cm4gdmFsdWUuCiAqLwpkbWFfYWRkcl90IG9tYXBfZ2V0X2RtYV9zcmNfcG9zKGludCBsY2gpCnsKCXJldHVybiAoZG1hX2FkZHJfdCkgKE9NQVBfRE1BX0NTU0FfTChsY2gpIHwKCQkJICAgICAoT01BUF9ETUFfQ1NTQV9VKGxjaCkgPDwgMTYpKTsKfQoKLyoKICogUmV0dXJucyBjdXJyZW50IHBoeXNpY2FsIGRlc3RpbmF0aW9uIGFkZHJlc3MgZm9yIHRoZSBnaXZlbiBETUEgY2hhbm5lbC4KICogSWYgdGhlIGNoYW5uZWwgaXMgcnVubmluZyB0aGUgY2FsbGVyIG11c3QgZGlzYWJsZSBpbnRlcnJ1cHRzIHByaW9yIGNhbGxpbmcKICogdGhpcyBmdW5jdGlvbiBhbmQgcHJvY2VzcyB0aGUgcmV0dXJuZWQgdmFsdWUgYmVmb3JlIHJlLWVuYWJsaW5nIGludGVycnVwdCB0bwogKiBwcmV2ZW50IHJhY2VzIHdpdGggdGhlIGludGVycnVwdCBoYW5kbGVyLiBOb3RlIHRoYXQgaW4gY29udGludW91cyBtb2RlIHRoZXJlCiAqIGlzIGEgY2hhbmNlIGZvciBDRFNBX0wgcmVnaXN0ZXIgb3ZlcmZsb3cgaW5iZXR3ZWVuIHRoZSB0d28gcmVhZHMgcmVzdWx0aW5nCiAqIGluIGluY29ycmVjdCByZXR1cm4gdmFsdWUuCiAqLwpkbWFfYWRkcl90IG9tYXBfZ2V0X2RtYV9kc3RfcG9zKGludCBsY2gpCnsKCXJldHVybiAoZG1hX2FkZHJfdCkgKE9NQVBfRE1BX0NEU0FfTChsY2gpIHwKCQkJICAgICAoT01BUF9ETUFfQ0RTQV9VKGxjaCkgPDwgMTYpKTsKfQoKaW50IG9tYXBfZG1hX3J1bm5pbmcodm9pZCkKewoJaW50IGxjaDsKCgkvKiBDaGVjayBpZiBMQ0QgRE1BIGlzIHJ1bm5pbmcgKi8KCWlmIChjcHVfaXNfb21hcDE2eHgoKSkKCQlpZiAob21hcF9yZWFkdyhPTUFQMTYxMF9ETUFfTENEX0NDUikgJiBPTUFQX0RNQV9DQ1JfRU4pCgkJCXJldHVybiAxOwoKCWZvciAobGNoID0gMDsgbGNoIDwgZG1hX2NoYW5fY291bnQ7IGxjaCsrKSB7CgkJdTE2IHc7CgoJCXcgPSBvbWFwX3JlYWR3KE9NQVBfRE1BX0NDUihsY2gpKTsKCQlpZiAodyAmIE9NQVBfRE1BX0NDUl9FTikKCQkJcmV0dXJuIDE7Cgl9CglyZXR1cm4gMDsKfQoKc3RhdGljIGludCBfX2luaXQgb21hcF9pbml0X2RtYSh2b2lkKQp7CglpbnQgY2gsIHI7CgoJaWYgKGNwdV9pc19vbWFwMTUxMCgpKSB7CgkJcHJpbnRrKEtFUk5fSU5GTyAiRE1BIHN1cHBvcnQgZm9yIE9NQVAxNTEwIGluaXRpYWxpemVkXG4iKTsKCQlkbWFfY2hhbl9jb3VudCA9IDk7CgkJZW5hYmxlXzE1MTBfbW9kZSA9IDE7Cgl9IGVsc2UgaWYgKGNwdV9pc19vbWFwMTZ4eCgpIHx8IGNwdV9pc19vbWFwNzMwKCkpIHsKCQlwcmludGsoS0VSTl9JTkZPICJPTUFQIERNQSBoYXJkd2FyZSB2ZXJzaW9uICVkXG4iLAoJCSAgICAgICBvbWFwX3JlYWR3KE9NQVBfRE1BX0hXX0lEKSk7CgkJcHJpbnRrKEtFUk5fSU5GTyAiRE1BIGNhcGFiaWxpdGllczogJTA4eDolMDh4OiUwNHg6JTA0eDolMDR4XG4iLAoJCSAgICAgICAob21hcF9yZWFkdyhPTUFQX0RNQV9DQVBTXzBfVSkgPDwgMTYpIHwgb21hcF9yZWFkdyhPTUFQX0RNQV9DQVBTXzBfTCksCgkJICAgICAgIChvbWFwX3JlYWR3KE9NQVBfRE1BX0NBUFNfMV9VKSA8PCAxNikgfCBvbWFwX3JlYWR3KE9NQVBfRE1BX0NBUFNfMV9MKSwKCQkgICAgICAgb21hcF9yZWFkdyhPTUFQX0RNQV9DQVBTXzIpLCBvbWFwX3JlYWR3KE9NQVBfRE1BX0NBUFNfMyksCgkJICAgICAgIG9tYXBfcmVhZHcoT01BUF9ETUFfQ0FQU180KSk7CgkJaWYgKCFlbmFibGVfMTUxMF9tb2RlKSB7CgkJCXUxNiB3OwoKCQkJLyogRGlzYWJsZSBPTUFQIDMuMC8zLjEgY29tcGF0aWJpbGl0eSBtb2RlLiAqLwoJCQl3ID0gb21hcF9yZWFkdyhPTUFQX0RNQV9HU0NSKTsKCQkJdyB8PSAxIDw8IDM7CgkJCW9tYXBfd3JpdGV3KHcsIE9NQVBfRE1BX0dTQ1IpOwoJCQlkbWFfY2hhbl9jb3VudCA9IDE2OwoJCX0gZWxzZQoJCQlkbWFfY2hhbl9jb3VudCA9IDk7Cgl9IGVsc2UgewoJCWRtYV9jaGFuX2NvdW50ID0gMDsKCQlyZXR1cm4gMDsKCX0KCgltZW1zZXQoJmxjZF9kbWEsIDAsIHNpemVvZihsY2RfZG1hKSk7CglzcGluX2xvY2tfaW5pdCgmbGNkX2RtYS5sb2NrKTsKCXNwaW5fbG9ja19pbml0KCZkbWFfY2hhbl9sb2NrKTsKCW1lbXNldCgmZG1hX2NoYW4sIDAsIHNpemVvZihkbWFfY2hhbikpOwoKCWZvciAoY2ggPSAwOyBjaCA8IGRtYV9jaGFuX2NvdW50OyBjaCsrKSB7CgkJZG1hX2NoYW5bY2hdLmRldl9pZCA9IC0xOwoJCWRtYV9jaGFuW2NoXS5uZXh0X2xjaCA9IC0xOwoKCQlpZiAoY2ggPj0gNiAmJiBlbmFibGVfMTUxMF9tb2RlKQoJCQljb250aW51ZTsKCgkJLyogcmVxdWVzdF9pcnEoKSBkb2Vzbid0IGxpa2UgZGV2X2lkIChpZS4gY2gpIGJlaW5nIHplcm8sCgkJICogc28gd2UgaGF2ZSB0byBrbHVkZ2UgYXJvdW5kIHRoaXMuICovCgkJciA9IHJlcXVlc3RfaXJxKGRtYV9pcnFbY2hdLCBkbWFfaXJxX2hhbmRsZXIsIDAsICJETUEiLAoJCQkJKHZvaWQgKikgKGNoICsgMSkpOwoJCWlmIChyICE9IDApIHsKCQkJaW50IGk7CgoJCQlwcmludGsoS0VSTl9FUlIgInVuYWJsZSB0byByZXF1ZXN0IElSUSAlZCBmb3IgRE1BIChlcnJvciAlZClcbiIsCgkJCSAgICAgICBkbWFfaXJxW2NoXSwgcik7CgkJCWZvciAoaSA9IDA7IGkgPCBjaDsgaSsrKQoJCQkJZnJlZV9pcnEoZG1hX2lycVtpXSwgKHZvaWQgKikgKGkgKyAxKSk7CgkJCXJldHVybiByOwoJCX0KCX0KCXIgPSByZXF1ZXN0X2lycShJTlRfRE1BX0xDRCwgbGNkX2RtYV9pcnFfaGFuZGxlciwgMCwgIkxDRCBETUEiLCBOVUxMKTsKCWlmIChyICE9IDApIHsKCQlpbnQgaTsKCgkJcHJpbnRrKEtFUk5fRVJSICJ1bmFibGUgdG8gcmVxdWVzdCBJUlEgZm9yIExDRCBETUEgKGVycm9yICVkKVxuIiwgcik7CgkJZm9yIChpID0gMDsgaSA8IGRtYV9jaGFuX2NvdW50OyBpKyspCgkJCWZyZWVfaXJxKGRtYV9pcnFbaV0sICh2b2lkICopIChpICsgMSkpOwoJCXJldHVybiByOwoJfQoJcmV0dXJuIDA7Cn0KCmFyY2hfaW5pdGNhbGwob21hcF9pbml0X2RtYSk7CgoKRVhQT1JUX1NZTUJPTChvbWFwX2dldF9kbWFfc3JjX3Bvcyk7CkVYUE9SVF9TWU1CT0wob21hcF9nZXRfZG1hX2RzdF9wb3MpOwpFWFBPUlRfU1lNQk9MKG9tYXBfY2xlYXJfZG1hKTsKRVhQT1JUX1NZTUJPTChvbWFwX3NldF9kbWFfcHJpb3JpdHkpOwpFWFBPUlRfU1lNQk9MKG9tYXBfcmVxdWVzdF9kbWEpOwpFWFBPUlRfU1lNQk9MKG9tYXBfZnJlZV9kbWEpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc3RhcnRfZG1hKTsKRVhQT1JUX1NZTUJPTChvbWFwX3N0b3BfZG1hKTsKRVhQT1JUX1NZTUJPTChvbWFwX2VuYWJsZV9kbWFfaXJxKTsKRVhQT1JUX1NZTUJPTChvbWFwX2Rpc2FibGVfZG1hX2lycSk7CgpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2RtYV90cmFuc2Zlcl9wYXJhbXMpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2RtYV9jb2xvcl9tb2RlKTsKCkVYUE9SVF9TWU1CT0wob21hcF9zZXRfZG1hX3NyY19wYXJhbXMpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2RtYV9zcmNfaW5kZXgpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2RtYV9zcmNfZGF0YV9wYWNrKTsKRVhQT1JUX1NZTUJPTChvbWFwX3NldF9kbWFfc3JjX2J1cnN0X21vZGUpOwoKRVhQT1JUX1NZTUJPTChvbWFwX3NldF9kbWFfZGVzdF9wYXJhbXMpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2RtYV9kZXN0X2luZGV4KTsKRVhQT1JUX1NZTUJPTChvbWFwX3NldF9kbWFfZGVzdF9kYXRhX3BhY2spOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2RtYV9kZXN0X2J1cnN0X21vZGUpOwoKRVhQT1JUX1NZTUJPTChvbWFwX2RtYV9saW5rX2xjaCk7CkVYUE9SVF9TWU1CT0wob21hcF9kbWFfdW5saW5rX2xjaCk7CgpFWFBPUlRfU1lNQk9MKG9tYXBfcmVxdWVzdF9sY2RfZG1hKTsKRVhQT1JUX1NZTUJPTChvbWFwX2ZyZWVfbGNkX2RtYSk7CkVYUE9SVF9TWU1CT0wob21hcF9lbmFibGVfbGNkX2RtYSk7CkVYUE9SVF9TWU1CT0wob21hcF9zZXR1cF9sY2RfZG1hKTsKRVhQT1JUX1NZTUJPTChvbWFwX3N0b3BfbGNkX2RtYSk7CkVYUE9SVF9TWU1CT0wob21hcF9zZXRfbGNkX2RtYV9iMSk7CkVYUE9SVF9TWU1CT0wob21hcF9zZXRfbGNkX2RtYV9zaW5nbGVfdHJhbnNmZXIpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2xjZF9kbWFfZXh0X2NvbnRyb2xsZXIpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2xjZF9kbWFfYjFfcm90YXRpb24pOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2xjZF9kbWFfYjFfdnhyZXMpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2xjZF9kbWFfYjFfc2NhbGUpOwpFWFBPUlRfU1lNQk9MKG9tYXBfc2V0X2xjZF9kbWFfYjFfbWlycm9yKTsKCg==