LyoKICogcmFpZDEuYyA6IE11bHRpcGxlIERldmljZXMgZHJpdmVyIGZvciBMaW51eAogKgogKiBDb3B5cmlnaHQgKEMpIDE5OTksIDIwMDAsIDIwMDEgSW5nbyBNb2xuYXIsIFJlZCBIYXQKICoKICogQ29weXJpZ2h0IChDKSAxOTk2LCAxOTk3LCAxOTk4IEluZ28gTW9sbmFyLCBNaWd1ZWwgZGUgSWNhemEsIEdhZGkgT3htYW4KICoKICogUkFJRC0xIG1hbmFnZW1lbnQgZnVuY3Rpb25zLgogKgogKiBCZXR0ZXIgcmVhZC1iYWxhbmNpbmcgY29kZSB3cml0dGVuIGJ5IE1pa2EgS3VvcHBhbGEgPG1pa3VAaWtpLmZpPiwgMjAwMAogKgogKiBGaXhlcyB0byByZWNvbnN0cnVjdGlvbiBieSBKYWtvYiDYc3RlcmdhYXJkIiA8amFrb2JAb3N0ZW5mZWxkLmRrPgogKiBWYXJpb3VzIGZpeGVzIGJ5IE5laWwgQnJvd24gPG5laWxiQGNzZS51bnN3LmVkdS5hdT4KICoKICogQ2hhbmdlcyBieSBQZXRlciBULiBCcmV1ZXIgPHB0YkBpdC51YzNtLmVzPiAzMS8xLzIwMDMgdG8gc3VwcG9ydAogKiBiaXRtYXBwZWQgaW50ZWxsaWdlbmNlIGluIHJlc3luYzoKICoKICogICAgICAtIGJpdG1hcCBtYXJrZWQgZHVyaW5nIG5vcm1hbCBpL28KICogICAgICAtIGJpdG1hcCB1c2VkIHRvIHNraXAgbm9uZGlydHkgYmxvY2tzIGR1cmluZyBzeW5jCiAqCiAqIEFkZGl0aW9ucyB0byBiaXRtYXAgY29kZSwgKEMpIDIwMDMtMjAwNCBQYXVsIENsZW1lbnRzLCBTdGVlbEV5ZSBUZWNobm9sb2d5OgogKiAtIHBlcnNpc3RlbnQgYml0bWFwIGNvZGUKICoKICogVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKICogaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkKICogdGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbjsgZWl0aGVyIHZlcnNpb24gMiwgb3IgKGF0IHlvdXIgb3B0aW9uKQogKiBhbnkgbGF0ZXIgdmVyc2lvbi4KICoKICogWW91IHNob3VsZCBoYXZlIHJlY2VpdmVkIGEgY29weSBvZiB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UKICogKGZvciBleGFtcGxlIC91c3Ivc3JjL2xpbnV4L0NPUFlJTkcpOyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlCiAqIFNvZnR3YXJlIEZvdW5kYXRpb24sIEluYy4sIDY3NSBNYXNzIEF2ZSwgQ2FtYnJpZGdlLCBNQSAwMjEzOSwgVVNBLgogKi8KCiNpbmNsdWRlICJkbS1iaW8tbGlzdC5oIgojaW5jbHVkZSA8bGludXgvcmFpZC9yYWlkMS5oPgojaW5jbHVkZSA8bGludXgvcmFpZC9iaXRtYXAuaD4KCiNkZWZpbmUgREVCVUcgMAojaWYgREVCVUcKI2RlZmluZSBQUklOVEsoeC4uLikgcHJpbnRrKHgpCiNlbHNlCiNkZWZpbmUgUFJJTlRLKHguLi4pCiNlbmRpZgoKLyoKICogTnVtYmVyIG9mIGd1YXJhbnRlZWQgcjFiaW9zIGluIGNhc2Ugb2YgZXh0cmVtZSBWTSBsb2FkOgogKi8KI2RlZmluZQlOUl9SQUlEMV9CSU9TIDI1NgoKCnN0YXRpYyB2b2lkIHVucGx1Z19zbGF2ZXMobWRkZXZfdCAqbWRkZXYpOwoKc3RhdGljIHZvaWQgYWxsb3dfYmFycmllcihjb25mX3QgKmNvbmYpOwpzdGF0aWMgdm9pZCBsb3dlcl9iYXJyaWVyKGNvbmZfdCAqY29uZik7CgpzdGF0aWMgdm9pZCAqIHIxYmlvX3Bvb2xfYWxsb2MoZ2ZwX3QgZ2ZwX2ZsYWdzLCB2b2lkICpkYXRhKQp7CglzdHJ1Y3QgcG9vbF9pbmZvICpwaSA9IGRhdGE7CglyMWJpb190ICpyMV9iaW87CglpbnQgc2l6ZSA9IG9mZnNldG9mKHIxYmlvX3QsIGJpb3NbcGktPnJhaWRfZGlza3NdKTsKCgkvKiBhbGxvY2F0ZSBhIHIxYmlvIHdpdGggcm9vbSBmb3IgcmFpZF9kaXNrcyBlbnRyaWVzIGluIHRoZSBiaW9zIGFycmF5ICovCglyMV9iaW8gPSBremFsbG9jKHNpemUsIGdmcF9mbGFncyk7CglpZiAoIXIxX2JpbykKCQl1bnBsdWdfc2xhdmVzKHBpLT5tZGRldik7CgoJcmV0dXJuIHIxX2JpbzsKfQoKc3RhdGljIHZvaWQgcjFiaW9fcG9vbF9mcmVlKHZvaWQgKnIxX2Jpbywgdm9pZCAqZGF0YSkKewoJa2ZyZWUocjFfYmlvKTsKfQoKI2RlZmluZSBSRVNZTkNfQkxPQ0tfU0laRSAoNjQqMTAyNCkKLy8jZGVmaW5lIFJFU1lOQ19CTE9DS19TSVpFIFBBR0VfU0laRQojZGVmaW5lIFJFU1lOQ19TRUNUT1JTIChSRVNZTkNfQkxPQ0tfU0laRSA+PiA5KQojZGVmaW5lIFJFU1lOQ19QQUdFUyAoKFJFU1lOQ19CTE9DS19TSVpFICsgUEFHRV9TSVpFLTEpIC8gUEFHRV9TSVpFKQojZGVmaW5lIFJFU1lOQ19XSU5ET1cgKDIwNDgqMTAyNCkKCnN0YXRpYyB2b2lkICogcjFidWZfcG9vbF9hbGxvYyhnZnBfdCBnZnBfZmxhZ3MsIHZvaWQgKmRhdGEpCnsKCXN0cnVjdCBwb29sX2luZm8gKnBpID0gZGF0YTsKCXN0cnVjdCBwYWdlICpwYWdlOwoJcjFiaW9fdCAqcjFfYmlvOwoJc3RydWN0IGJpbyAqYmlvOwoJaW50IGksIGo7CgoJcjFfYmlvID0gcjFiaW9fcG9vbF9hbGxvYyhnZnBfZmxhZ3MsIHBpKTsKCWlmICghcjFfYmlvKSB7CgkJdW5wbHVnX3NsYXZlcyhwaS0+bWRkZXYpOwoJCXJldHVybiBOVUxMOwoJfQoKCS8qCgkgKiBBbGxvY2F0ZSBiaW9zIDogMSBmb3IgcmVhZGluZywgbi0xIGZvciB3cml0aW5nCgkgKi8KCWZvciAoaiA9IHBpLT5yYWlkX2Rpc2tzIDsgai0tIDsgKSB7CgkJYmlvID0gYmlvX2FsbG9jKGdmcF9mbGFncywgUkVTWU5DX1BBR0VTKTsKCQlpZiAoIWJpbykKCQkJZ290byBvdXRfZnJlZV9iaW87CgkJcjFfYmlvLT5iaW9zW2pdID0gYmlvOwoJfQoJLyoKCSAqIEFsbG9jYXRlIFJFU1lOQ19QQUdFUyBkYXRhIHBhZ2VzIGFuZCBhdHRhY2ggdGhlbSB0bwoJICogdGhlIGZpcnN0IGJpby4KCSAqIElmIHRoaXMgaXMgYSB1c2VyLXJlcXVlc3RlZCBjaGVjay9yZXBhaXIsIGFsbG9jYXRlCgkgKiBSRVNZTkNfUEFHRVMgZm9yIGVhY2ggYmlvLgoJICovCglpZiAodGVzdF9iaXQoTURfUkVDT1ZFUllfUkVRVUVTVEVELCAmcGktPm1kZGV2LT5yZWNvdmVyeSkpCgkJaiA9IHBpLT5yYWlkX2Rpc2tzOwoJZWxzZQoJCWogPSAxOwoJd2hpbGUoai0tKSB7CgkJYmlvID0gcjFfYmlvLT5iaW9zW2pdOwoJCWZvciAoaSA9IDA7IGkgPCBSRVNZTkNfUEFHRVM7IGkrKykgewoJCQlwYWdlID0gYWxsb2NfcGFnZShnZnBfZmxhZ3MpOwoJCQlpZiAodW5saWtlbHkoIXBhZ2UpKQoJCQkJZ290byBvdXRfZnJlZV9wYWdlczsKCgkJCWJpby0+YmlfaW9fdmVjW2ldLmJ2X3BhZ2UgPSBwYWdlOwoJCX0KCX0KCS8qIElmIG5vdCB1c2VyLXJlcXVlc3RzLCBjb3B5IHRoZSBwYWdlIHBvaW50ZXJzIHRvIGFsbCBiaW9zICovCglpZiAoIXRlc3RfYml0KE1EX1JFQ09WRVJZX1JFUVVFU1RFRCwgJnBpLT5tZGRldi0+cmVjb3ZlcnkpKSB7CgkJZm9yIChpPTA7IGk8UkVTWU5DX1BBR0VTIDsgaSsrKQoJCQlmb3IgKGo9MTsgajxwaS0+cmFpZF9kaXNrczsgaisrKQoJCQkJcjFfYmlvLT5iaW9zW2pdLT5iaV9pb192ZWNbaV0uYnZfcGFnZSA9CgkJCQkJcjFfYmlvLT5iaW9zWzBdLT5iaV9pb192ZWNbaV0uYnZfcGFnZTsKCX0KCglyMV9iaW8tPm1hc3Rlcl9iaW8gPSBOVUxMOwoKCXJldHVybiByMV9iaW87CgpvdXRfZnJlZV9wYWdlczoKCWZvciAoaT0wOyBpIDwgUkVTWU5DX1BBR0VTIDsgaSsrKQoJCWZvciAoaj0wIDsgaiA8IHBpLT5yYWlkX2Rpc2tzOyBqKyspCgkJCXNhZmVfcHV0X3BhZ2UocjFfYmlvLT5iaW9zW2pdLT5iaV9pb192ZWNbaV0uYnZfcGFnZSk7CglqID0gLTE7Cm91dF9mcmVlX2JpbzoKCXdoaWxlICggKytqIDwgcGktPnJhaWRfZGlza3MgKQoJCWJpb19wdXQocjFfYmlvLT5iaW9zW2pdKTsKCXIxYmlvX3Bvb2xfZnJlZShyMV9iaW8sIGRhdGEpOwoJcmV0dXJuIE5VTEw7Cn0KCnN0YXRpYyB2b2lkIHIxYnVmX3Bvb2xfZnJlZSh2b2lkICpfX3IxX2Jpbywgdm9pZCAqZGF0YSkKewoJc3RydWN0IHBvb2xfaW5mbyAqcGkgPSBkYXRhOwoJaW50IGksajsKCXIxYmlvX3QgKnIxYmlvID0gX19yMV9iaW87CgoJZm9yIChpID0gMDsgaSA8IFJFU1lOQ19QQUdFUzsgaSsrKQoJCWZvciAoaiA9IHBpLT5yYWlkX2Rpc2tzOyBqLS0gOykgewoJCQlpZiAoaiA9PSAwIHx8CgkJCSAgICByMWJpby0+Ymlvc1tqXS0+YmlfaW9fdmVjW2ldLmJ2X3BhZ2UgIT0KCQkJICAgIHIxYmlvLT5iaW9zWzBdLT5iaV9pb192ZWNbaV0uYnZfcGFnZSkKCQkJCXNhZmVfcHV0X3BhZ2UocjFiaW8tPmJpb3Nbal0tPmJpX2lvX3ZlY1tpXS5idl9wYWdlKTsKCQl9Cglmb3IgKGk9MCA7IGkgPCBwaS0+cmFpZF9kaXNrczsgaSsrKQoJCWJpb19wdXQocjFiaW8tPmJpb3NbaV0pOwoKCXIxYmlvX3Bvb2xfZnJlZShyMWJpbywgZGF0YSk7Cn0KCnN0YXRpYyB2b2lkIHB1dF9hbGxfYmlvcyhjb25mX3QgKmNvbmYsIHIxYmlvX3QgKnIxX2JpbykKewoJaW50IGk7CgoJZm9yIChpID0gMDsgaSA8IGNvbmYtPnJhaWRfZGlza3M7IGkrKykgewoJCXN0cnVjdCBiaW8gKipiaW8gPSByMV9iaW8tPmJpb3MgKyBpOwoJCWlmICgqYmlvICYmICpiaW8gIT0gSU9fQkxPQ0tFRCkKCQkJYmlvX3B1dCgqYmlvKTsKCQkqYmlvID0gTlVMTDsKCX0KfQoKc3RhdGljIHZvaWQgZnJlZV9yMWJpbyhyMWJpb190ICpyMV9iaW8pCnsKCWNvbmZfdCAqY29uZiA9IG1kZGV2X3RvX2NvbmYocjFfYmlvLT5tZGRldik7CgoJLyoKCSAqIFdha2UgdXAgYW55IHBvc3NpYmxlIHJlc3luYyB0aHJlYWQgdGhhdCB3YWl0cyBmb3IgdGhlIGRldmljZQoJICogdG8gZ28gaWRsZS4KCSAqLwoJYWxsb3dfYmFycmllcihjb25mKTsKCglwdXRfYWxsX2Jpb3MoY29uZiwgcjFfYmlvKTsKCW1lbXBvb2xfZnJlZShyMV9iaW8sIGNvbmYtPnIxYmlvX3Bvb2wpOwp9CgpzdGF0aWMgdm9pZCBwdXRfYnVmKHIxYmlvX3QgKnIxX2JpbykKewoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihyMV9iaW8tPm1kZGV2KTsKCWludCBpOwoKCWZvciAoaT0wOyBpPGNvbmYtPnJhaWRfZGlza3M7IGkrKykgewoJCXN0cnVjdCBiaW8gKmJpbyA9IHIxX2Jpby0+Ymlvc1tpXTsKCQlpZiAoYmlvLT5iaV9lbmRfaW8pCgkJCXJkZXZfZGVjX3BlbmRpbmcoY29uZi0+bWlycm9yc1tpXS5yZGV2LCByMV9iaW8tPm1kZGV2KTsKCX0KCgltZW1wb29sX2ZyZWUocjFfYmlvLCBjb25mLT5yMWJ1Zl9wb29sKTsKCglsb3dlcl9iYXJyaWVyKGNvbmYpOwp9CgpzdGF0aWMgdm9pZCByZXNjaGVkdWxlX3JldHJ5KHIxYmlvX3QgKnIxX2JpbykKewoJdW5zaWduZWQgbG9uZyBmbGFnczsKCW1kZGV2X3QgKm1kZGV2ID0gcjFfYmlvLT5tZGRldjsKCWNvbmZfdCAqY29uZiA9IG1kZGV2X3RvX2NvbmYobWRkZXYpOwoKCXNwaW5fbG9ja19pcnFzYXZlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoJbGlzdF9hZGQoJnIxX2Jpby0+cmV0cnlfbGlzdCwgJmNvbmYtPnJldHJ5X2xpc3QpOwoJY29uZi0+bnJfcXVldWVkICsrOwoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmY29uZi0+ZGV2aWNlX2xvY2ssIGZsYWdzKTsKCgl3YWtlX3VwKCZjb25mLT53YWl0X2JhcnJpZXIpOwoJbWRfd2FrZXVwX3RocmVhZChtZGRldi0+dGhyZWFkKTsKfQoKLyoKICogcmFpZF9lbmRfYmlvX2lvKCkgaXMgY2FsbGVkIHdoZW4gd2UgaGF2ZSBmaW5pc2hlZCBzZXJ2aWNpbmcgYSBtaXJyb3JlZAogKiBvcGVyYXRpb24gYW5kIGFyZSByZWFkeSB0byByZXR1cm4gYSBzdWNjZXNzL2ZhaWx1cmUgY29kZSB0byB0aGUgYnVmZmVyCiAqIGNhY2hlIGxheWVyLgogKi8Kc3RhdGljIHZvaWQgcmFpZF9lbmRfYmlvX2lvKHIxYmlvX3QgKnIxX2JpbykKewoJc3RydWN0IGJpbyAqYmlvID0gcjFfYmlvLT5tYXN0ZXJfYmlvOwoKCS8qIGlmIG5vYm9keSBoYXMgZG9uZSB0aGUgZmluYWwgZW5kaW8geWV0LCBkbyBpdCBub3cgKi8KCWlmICghdGVzdF9hbmRfc2V0X2JpdChSMUJJT19SZXR1cm5lZCwgJnIxX2Jpby0+c3RhdGUpKSB7CgkJUFJJTlRLKEtFUk5fREVCVUcgInJhaWQxOiBzeW5jIGVuZCAlcyBvbiBzZWN0b3JzICVsbHUtJWxsdVxuIiwKCQkJKGJpb19kYXRhX2RpcihiaW8pID09IFdSSVRFKSA/ICJ3cml0ZSIgOiAicmVhZCIsCgkJCSh1bnNpZ25lZCBsb25nIGxvbmcpIGJpby0+Ymlfc2VjdG9yLAoJCQkodW5zaWduZWQgbG9uZyBsb25nKSBiaW8tPmJpX3NlY3RvciArCgkJCQkoYmlvLT5iaV9zaXplID4+IDkpIC0gMSk7CgoJCWJpb19lbmRpbyhiaW8sIGJpby0+Ymlfc2l6ZSwKCQkJdGVzdF9iaXQoUjFCSU9fVXB0b2RhdGUsICZyMV9iaW8tPnN0YXRlKSA/IDAgOiAtRUlPKTsKCX0KCWZyZWVfcjFiaW8ocjFfYmlvKTsKfQoKLyoKICogVXBkYXRlIGRpc2sgaGVhZCBwb3NpdGlvbiBlc3RpbWF0b3IgYmFzZWQgb24gSVJRIGNvbXBsZXRpb24gaW5mby4KICovCnN0YXRpYyBpbmxpbmUgdm9pZCB1cGRhdGVfaGVhZF9wb3MoaW50IGRpc2ssIHIxYmlvX3QgKnIxX2JpbykKewoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihyMV9iaW8tPm1kZGV2KTsKCgljb25mLT5taXJyb3JzW2Rpc2tdLmhlYWRfcG9zaXRpb24gPQoJCXIxX2Jpby0+c2VjdG9yICsgKHIxX2Jpby0+c2VjdG9ycyk7Cn0KCnN0YXRpYyBpbnQgcmFpZDFfZW5kX3JlYWRfcmVxdWVzdChzdHJ1Y3QgYmlvICpiaW8sIHVuc2lnbmVkIGludCBieXRlc19kb25lLCBpbnQgZXJyb3IpCnsKCWludCB1cHRvZGF0ZSA9IHRlc3RfYml0KEJJT19VUFRPREFURSwgJmJpby0+YmlfZmxhZ3MpOwoJcjFiaW9fdCAqIHIxX2JpbyA9IChyMWJpb190ICopKGJpby0+YmlfcHJpdmF0ZSk7CglpbnQgbWlycm9yOwoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihyMV9iaW8tPm1kZGV2KTsKCglpZiAoYmlvLT5iaV9zaXplKQoJCXJldHVybiAxOwoJCgltaXJyb3IgPSByMV9iaW8tPnJlYWRfZGlzazsKCS8qCgkgKiB0aGlzIGJyYW5jaCBpcyBvdXIgJ29uZSBtaXJyb3IgSU8gaGFzIGZpbmlzaGVkJyBldmVudCBoYW5kbGVyOgoJICovCgl1cGRhdGVfaGVhZF9wb3MobWlycm9yLCByMV9iaW8pOwoKCWlmICh1cHRvZGF0ZSB8fCAoY29uZi0+cmFpZF9kaXNrcyAtIGNvbmYtPm1kZGV2LT5kZWdyYWRlZCkgPD0gMSkgewoJCS8qCgkJICogU2V0IFIxQklPX1VwdG9kYXRlIGluIG91ciBtYXN0ZXIgYmlvLCBzbyB0aGF0CgkJICogd2Ugd2lsbCByZXR1cm4gYSBnb29kIGVycm9yIGNvZGUgZm9yIHRvIHRoZSBoaWdoZXIKCQkgKiBsZXZlbHMgZXZlbiBpZiBJTyBvbiBzb21lIG90aGVyIG1pcnJvcmVkIGJ1ZmZlciBmYWlscy4KCQkgKgoJCSAqIFRoZSAnbWFzdGVyJyByZXByZXNlbnRzIHRoZSBjb21wb3NpdGUgSU8gb3BlcmF0aW9uIHRvCgkJICogdXNlci1zaWRlLiBTbyBpZiBzb21ldGhpbmcgd2FpdHMgZm9yIElPLCB0aGVuIGl0IHdpbGwKCQkgKiB3YWl0IGZvciB0aGUgJ21hc3RlcicgYmlvLgoJCSAqLwoJCWlmICh1cHRvZGF0ZSkKCQkJc2V0X2JpdChSMUJJT19VcHRvZGF0ZSwgJnIxX2Jpby0+c3RhdGUpOwoKCQlyYWlkX2VuZF9iaW9faW8ocjFfYmlvKTsKCX0gZWxzZSB7CgkJLyoKCQkgKiBvb3BzLCByZWFkIGVycm9yOgoJCSAqLwoJCWNoYXIgYltCREVWTkFNRV9TSVpFXTsKCQlpZiAocHJpbnRrX3JhdGVsaW1pdCgpKQoJCQlwcmludGsoS0VSTl9FUlIgInJhaWQxOiAlczogcmVzY2hlZHVsaW5nIHNlY3RvciAlbGx1XG4iLAoJCQkgICAgICAgYmRldm5hbWUoY29uZi0+bWlycm9yc1ttaXJyb3JdLnJkZXYtPmJkZXYsYiksICh1bnNpZ25lZCBsb25nIGxvbmcpcjFfYmlvLT5zZWN0b3IpOwoJCXJlc2NoZWR1bGVfcmV0cnkocjFfYmlvKTsKCX0KCglyZGV2X2RlY19wZW5kaW5nKGNvbmYtPm1pcnJvcnNbbWlycm9yXS5yZGV2LCBjb25mLT5tZGRldik7CglyZXR1cm4gMDsKfQoKc3RhdGljIGludCByYWlkMV9lbmRfd3JpdGVfcmVxdWVzdChzdHJ1Y3QgYmlvICpiaW8sIHVuc2lnbmVkIGludCBieXRlc19kb25lLCBpbnQgZXJyb3IpCnsKCWludCB1cHRvZGF0ZSA9IHRlc3RfYml0KEJJT19VUFRPREFURSwgJmJpby0+YmlfZmxhZ3MpOwoJcjFiaW9fdCAqIHIxX2JpbyA9IChyMWJpb190ICopKGJpby0+YmlfcHJpdmF0ZSk7CglpbnQgbWlycm9yLCBiZWhpbmQgPSB0ZXN0X2JpdChSMUJJT19CZWhpbmRJTywgJnIxX2Jpby0+c3RhdGUpOwoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihyMV9iaW8tPm1kZGV2KTsKCXN0cnVjdCBiaW8gKnRvX3B1dCA9IE5VTEw7CgoJaWYgKGJpby0+Ymlfc2l6ZSkKCQlyZXR1cm4gMTsKCglmb3IgKG1pcnJvciA9IDA7IG1pcnJvciA8IGNvbmYtPnJhaWRfZGlza3M7IG1pcnJvcisrKQoJCWlmIChyMV9iaW8tPmJpb3NbbWlycm9yXSA9PSBiaW8pCgkJCWJyZWFrOwoKCWlmIChlcnJvciA9PSAtRU9QTk9UU1VQUCAmJiB0ZXN0X2JpdChSMUJJT19CYXJyaWVyLCAmcjFfYmlvLT5zdGF0ZSkpIHsKCQlzZXRfYml0KEJhcnJpZXJzTm90c3VwcCwgJmNvbmYtPm1pcnJvcnNbbWlycm9yXS5yZGV2LT5mbGFncyk7CgkJc2V0X2JpdChSMUJJT19CYXJyaWVyUmV0cnksICZyMV9iaW8tPnN0YXRlKTsKCQlyMV9iaW8tPm1kZGV2LT5iYXJyaWVyc193b3JrID0gMDsKCQkvKiBEb24ndCByZGV2X2RlY19wZW5kaW5nIGluIHRoaXMgYnJhbmNoIC0ga2VlcCBpdCBmb3IgdGhlIHJldHJ5ICovCgl9IGVsc2UgewoJCS8qCgkJICogdGhpcyBicmFuY2ggaXMgb3VyICdvbmUgbWlycm9yIElPIGhhcyBmaW5pc2hlZCcgZXZlbnQgaGFuZGxlcjoKCQkgKi8KCQlyMV9iaW8tPmJpb3NbbWlycm9yXSA9IE5VTEw7CgkJdG9fcHV0ID0gYmlvOwoJCWlmICghdXB0b2RhdGUpIHsKCQkJbWRfZXJyb3IocjFfYmlvLT5tZGRldiwgY29uZi0+bWlycm9yc1ttaXJyb3JdLnJkZXYpOwoJCQkvKiBhbiBJL08gZmFpbGVkLCB3ZSBjYW4ndCBjbGVhciB0aGUgYml0bWFwICovCgkJCXNldF9iaXQoUjFCSU9fRGVncmFkZWQsICZyMV9iaW8tPnN0YXRlKTsKCQl9IGVsc2UKCQkJLyoKCQkJICogU2V0IFIxQklPX1VwdG9kYXRlIGluIG91ciBtYXN0ZXIgYmlvLCBzbyB0aGF0CgkJCSAqIHdlIHdpbGwgcmV0dXJuIGEgZ29vZCBlcnJvciBjb2RlIGZvciB0byB0aGUgaGlnaGVyCgkJCSAqIGxldmVscyBldmVuIGlmIElPIG9uIHNvbWUgb3RoZXIgbWlycm9yZWQgYnVmZmVyIGZhaWxzLgoJCQkgKgoJCQkgKiBUaGUgJ21hc3RlcicgcmVwcmVzZW50cyB0aGUgY29tcG9zaXRlIElPIG9wZXJhdGlvbiB0bwoJCQkgKiB1c2VyLXNpZGUuIFNvIGlmIHNvbWV0aGluZyB3YWl0cyBmb3IgSU8sIHRoZW4gaXQgd2lsbAoJCQkgKiB3YWl0IGZvciB0aGUgJ21hc3RlcicgYmlvLgoJCQkgKi8KCQkJc2V0X2JpdChSMUJJT19VcHRvZGF0ZSwgJnIxX2Jpby0+c3RhdGUpOwoKCQl1cGRhdGVfaGVhZF9wb3MobWlycm9yLCByMV9iaW8pOwoKCQlpZiAoYmVoaW5kKSB7CgkJCWlmICh0ZXN0X2JpdChXcml0ZU1vc3RseSwgJmNvbmYtPm1pcnJvcnNbbWlycm9yXS5yZGV2LT5mbGFncykpCgkJCQlhdG9taWNfZGVjKCZyMV9iaW8tPmJlaGluZF9yZW1haW5pbmcpOwoKCQkJLyogSW4gYmVoaW5kIG1vZGUsIHdlIEFDSyB0aGUgbWFzdGVyIGJpbyBvbmNlIHRoZSBJL08gaGFzIHNhZmVseQoJCQkgKiByZWFjaGVkIGFsbCBub24td3JpdGVtb3N0bHkgZGlza3MuIFNldHRpbmcgdGhlIFJldHVybmVkIGJpdAoJCQkgKiBlbnN1cmVzIHRoYXQgdGhpcyBnZXRzIGRvbmUgb25seSBvbmNlIC0tIHdlIGRvbid0IGV2ZXIgd2FudCB0bwoJCQkgKiByZXR1cm4gLUVJTyBoZXJlLCBpbnN0ZWFkIHdlJ2xsIHdhaXQgKi8KCgkJCWlmIChhdG9taWNfcmVhZCgmcjFfYmlvLT5iZWhpbmRfcmVtYWluaW5nKSA+PSAoYXRvbWljX3JlYWQoJnIxX2Jpby0+cmVtYWluaW5nKS0xKSAmJgoJCQkgICAgdGVzdF9iaXQoUjFCSU9fVXB0b2RhdGUsICZyMV9iaW8tPnN0YXRlKSkgewoJCQkJLyogTWF5YmUgd2UgY2FuIHJldHVybiBub3cgKi8KCQkJCWlmICghdGVzdF9hbmRfc2V0X2JpdChSMUJJT19SZXR1cm5lZCwgJnIxX2Jpby0+c3RhdGUpKSB7CgkJCQkJc3RydWN0IGJpbyAqbWJpbyA9IHIxX2Jpby0+bWFzdGVyX2JpbzsKCQkJCQlQUklOVEsoS0VSTl9ERUJVRyAicmFpZDE6IGJlaGluZCBlbmQgd3JpdGUgc2VjdG9ycyAlbGx1LSVsbHVcbiIsCgkJCQkJICAgICAgICh1bnNpZ25lZCBsb25nIGxvbmcpIG1iaW8tPmJpX3NlY3RvciwKCQkJCQkgICAgICAgKHVuc2lnbmVkIGxvbmcgbG9uZykgbWJpby0+Ymlfc2VjdG9yICsKCQkJCQkgICAgICAgKG1iaW8tPmJpX3NpemUgPj4gOSkgLSAxKTsKCQkJCQliaW9fZW5kaW8obWJpbywgbWJpby0+Ymlfc2l6ZSwgMCk7CgkJCQl9CgkJCX0KCQl9CgkJcmRldl9kZWNfcGVuZGluZyhjb25mLT5taXJyb3JzW21pcnJvcl0ucmRldiwgY29uZi0+bWRkZXYpOwoJfQoJLyoKCSAqCgkgKiBMZXQncyBzZWUgaWYgYWxsIG1pcnJvcmVkIHdyaXRlIG9wZXJhdGlvbnMgaGF2ZSBmaW5pc2hlZAoJICogYWxyZWFkeS4KCSAqLwoJaWYgKGF0b21pY19kZWNfYW5kX3Rlc3QoJnIxX2Jpby0+cmVtYWluaW5nKSkgewoJCWlmICh0ZXN0X2JpdChSMUJJT19CYXJyaWVyUmV0cnksICZyMV9iaW8tPnN0YXRlKSkKCQkJcmVzY2hlZHVsZV9yZXRyeShyMV9iaW8pOwoJCWVsc2UgewoJCQkvKiBpdCByZWFsbHkgaXMgdGhlIGVuZCBvZiB0aGlzIHJlcXVlc3QgKi8KCQkJaWYgKHRlc3RfYml0KFIxQklPX0JlaGluZElPLCAmcjFfYmlvLT5zdGF0ZSkpIHsKCQkJCS8qIGZyZWUgZXh0cmEgY29weSBvZiB0aGUgZGF0YSBwYWdlcyAqLwoJCQkJaW50IGkgPSBiaW8tPmJpX3ZjbnQ7CgkJCQl3aGlsZSAoaS0tKQoJCQkJCXNhZmVfcHV0X3BhZ2UoYmlvLT5iaV9pb192ZWNbaV0uYnZfcGFnZSk7CgkJCX0KCQkJLyogY2xlYXIgdGhlIGJpdG1hcCBpZiBhbGwgd3JpdGVzIGNvbXBsZXRlIHN1Y2Nlc3NmdWxseSAqLwoJCQliaXRtYXBfZW5kd3JpdGUocjFfYmlvLT5tZGRldi0+Yml0bWFwLCByMV9iaW8tPnNlY3RvciwKCQkJCQlyMV9iaW8tPnNlY3RvcnMsCgkJCQkJIXRlc3RfYml0KFIxQklPX0RlZ3JhZGVkLCAmcjFfYmlvLT5zdGF0ZSksCgkJCQkJYmVoaW5kKTsKCQkJbWRfd3JpdGVfZW5kKHIxX2Jpby0+bWRkZXYpOwoJCQlyYWlkX2VuZF9iaW9faW8ocjFfYmlvKTsKCQl9Cgl9CgoJaWYgKHRvX3B1dCkKCQliaW9fcHV0KHRvX3B1dCk7CgoJcmV0dXJuIDA7Cn0KCgovKgogKiBUaGlzIHJvdXRpbmUgcmV0dXJucyB0aGUgZGlzayBmcm9tIHdoaWNoIHRoZSByZXF1ZXN0ZWQgcmVhZCBzaG91bGQKICogYmUgZG9uZS4gVGhlcmUgaXMgYSBwZXItYXJyYXkgJ25leHQgZXhwZWN0ZWQgc2VxdWVudGlhbCBJTycgc2VjdG9yCiAqIG51bWJlciAtIGlmIHRoaXMgbWF0Y2hlcyBvbiB0aGUgbmV4dCBJTyB0aGVuIHdlIHVzZSB0aGUgbGFzdCBkaXNrLgogKiBUaGVyZSBpcyBhbHNvIGEgcGVyLWRpc2sgJ2xhc3Qga25vdyBoZWFkIHBvc2l0aW9uJyBzZWN0b3IgdGhhdCBpcwogKiBtYWludGFpbmVkIGZyb20gSVJRIGNvbnRleHRzLCBib3RoIHRoZSBub3JtYWwgYW5kIHRoZSByZXN5bmMgSU8KICogY29tcGxldGlvbiBoYW5kbGVycyB1cGRhdGUgdGhpcyBwb3NpdGlvbiBjb3JyZWN0bHkuIElmIHRoZXJlIGlzIG5vCiAqIHBlcmZlY3Qgc2VxdWVudGlhbCBtYXRjaCB0aGVuIHdlIHBpY2sgdGhlIGRpc2sgd2hvc2UgaGVhZCBpcyBjbG9zZXN0LgogKgogKiBJZiB0aGVyZSBhcmUgMiBtaXJyb3JzIGluIHRoZSBzYW1lIDIgZGV2aWNlcywgcGVyZm9ybWFuY2UgZGVncmFkZXMKICogYmVjYXVzZSBwb3NpdGlvbiBpcyBtaXJyb3IsIG5vdCBkZXZpY2UgYmFzZWQuCiAqCiAqIFRoZSByZGV2IGZvciB0aGUgZGV2aWNlIHNlbGVjdGVkIHdpbGwgaGF2ZSBucl9wZW5kaW5nIGluY3JlbWVudGVkLgogKi8Kc3RhdGljIGludCByZWFkX2JhbGFuY2UoY29uZl90ICpjb25mLCByMWJpb190ICpyMV9iaW8pCnsKCWNvbnN0IHVuc2lnbmVkIGxvbmcgdGhpc19zZWN0b3IgPSByMV9iaW8tPnNlY3RvcjsKCWludCBuZXdfZGlzayA9IGNvbmYtPmxhc3RfdXNlZCwgZGlzayA9IG5ld19kaXNrOwoJaW50IHdvbmx5X2Rpc2sgPSAtMTsKCWNvbnN0IGludCBzZWN0b3JzID0gcjFfYmlvLT5zZWN0b3JzOwoJc2VjdG9yX3QgbmV3X2Rpc3RhbmNlLCBjdXJyZW50X2Rpc3RhbmNlOwoJbWRrX3JkZXZfdCAqcmRldjsKCglyY3VfcmVhZF9sb2NrKCk7CgkvKgoJICogQ2hlY2sgaWYgd2UgY2FuIGJhbGFuY2UuIFdlIGNhbiBiYWxhbmNlIG9uIHRoZSB3aG9sZQoJICogZGV2aWNlIGlmIG5vIHJlc3luYyBpcyBnb2luZyBvbiwgb3IgYmVsb3cgdGhlIHJlc3luYyB3aW5kb3cuCgkgKiBXZSB0YWtlIHRoZSBmaXJzdCByZWFkYWJsZSBkaXNrIHdoZW4gYWJvdmUgdGhlIHJlc3luYyB3aW5kb3cuCgkgKi8KIHJldHJ5OgoJaWYgKGNvbmYtPm1kZGV2LT5yZWNvdmVyeV9jcCA8IE1heFNlY3RvciAmJgoJICAgICh0aGlzX3NlY3RvciArIHNlY3RvcnMgPj0gY29uZi0+bmV4dF9yZXN5bmMpKSB7CgkJLyogQ2hvb3NlIHRoZSBmaXJzdCBvcGVyYXRpb24gZGV2aWNlLCBmb3IgY29uc2lzdGFuY3kgKi8KCQluZXdfZGlzayA9IDA7CgoJCWZvciAocmRldiA9IHJjdV9kZXJlZmVyZW5jZShjb25mLT5taXJyb3JzW25ld19kaXNrXS5yZGV2KTsKCQkgICAgIHIxX2Jpby0+Ymlvc1tuZXdfZGlza10gPT0gSU9fQkxPQ0tFRCB8fAoJCSAgICAgIXJkZXYgfHwgIXRlc3RfYml0KEluX3N5bmMsICZyZGV2LT5mbGFncykKCQkJICAgICB8fCB0ZXN0X2JpdChXcml0ZU1vc3RseSwgJnJkZXYtPmZsYWdzKTsKCQkgICAgIHJkZXYgPSByY3VfZGVyZWZlcmVuY2UoY29uZi0+bWlycm9yc1srK25ld19kaXNrXS5yZGV2KSkgewoKCQkJaWYgKHJkZXYgJiYgdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSAmJgoJCQkJcjFfYmlvLT5iaW9zW25ld19kaXNrXSAhPSBJT19CTE9DS0VEKQoJCQkJd29ubHlfZGlzayA9IG5ld19kaXNrOwoKCQkJaWYgKG5ld19kaXNrID09IGNvbmYtPnJhaWRfZGlza3MgLSAxKSB7CgkJCQluZXdfZGlzayA9IHdvbmx5X2Rpc2s7CgkJCQlicmVhazsKCQkJfQoJCX0KCQlnb3RvIHJiX291dDsKCX0KCgoJLyogbWFrZSBzdXJlIHRoZSBkaXNrIGlzIG9wZXJhdGlvbmFsICovCglmb3IgKHJkZXYgPSByY3VfZGVyZWZlcmVuY2UoY29uZi0+bWlycm9yc1tuZXdfZGlza10ucmRldik7CgkgICAgIHIxX2Jpby0+Ymlvc1tuZXdfZGlza10gPT0gSU9fQkxPQ0tFRCB8fAoJICAgICAhcmRldiB8fCAhdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSB8fAoJCSAgICAgdGVzdF9iaXQoV3JpdGVNb3N0bHksICZyZGV2LT5mbGFncyk7CgkgICAgIHJkZXYgPSByY3VfZGVyZWZlcmVuY2UoY29uZi0+bWlycm9yc1tuZXdfZGlza10ucmRldikpIHsKCgkJaWYgKHJkZXYgJiYgdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSAmJgoJCSAgICByMV9iaW8tPmJpb3NbbmV3X2Rpc2tdICE9IElPX0JMT0NLRUQpCgkJCXdvbmx5X2Rpc2sgPSBuZXdfZGlzazsKCgkJaWYgKG5ld19kaXNrIDw9IDApCgkJCW5ld19kaXNrID0gY29uZi0+cmFpZF9kaXNrczsKCQluZXdfZGlzay0tOwoJCWlmIChuZXdfZGlzayA9PSBkaXNrKSB7CgkJCW5ld19kaXNrID0gd29ubHlfZGlzazsKCQkJYnJlYWs7CgkJfQoJfQoKCWlmIChuZXdfZGlzayA8IDApCgkJZ290byByYl9vdXQ7CgoJZGlzayA9IG5ld19kaXNrOwoJLyogbm93IGRpc2sgPT0gbmV3X2Rpc2sgPT0gc3RhcnRpbmcgcG9pbnQgZm9yIHNlYXJjaCAqLwoKCS8qCgkgKiBEb24ndCBjaGFuZ2UgdG8gYW5vdGhlciBkaXNrIGZvciBzZXF1ZW50aWFsIHJlYWRzOgoJICovCglpZiAoY29uZi0+bmV4dF9zZXFfc2VjdCA9PSB0aGlzX3NlY3RvcikKCQlnb3RvIHJiX291dDsKCWlmICh0aGlzX3NlY3RvciA9PSBjb25mLT5taXJyb3JzW25ld19kaXNrXS5oZWFkX3Bvc2l0aW9uKQoJCWdvdG8gcmJfb3V0OwoKCWN1cnJlbnRfZGlzdGFuY2UgPSBhYnModGhpc19zZWN0b3IgLSBjb25mLT5taXJyb3JzW2Rpc2tdLmhlYWRfcG9zaXRpb24pOwoKCS8qIEZpbmQgdGhlIGRpc2sgd2hvc2UgaGVhZCBpcyBjbG9zZXN0ICovCgoJZG8gewoJCWlmIChkaXNrIDw9IDApCgkJCWRpc2sgPSBjb25mLT5yYWlkX2Rpc2tzOwoJCWRpc2stLTsKCgkJcmRldiA9IHJjdV9kZXJlZmVyZW5jZShjb25mLT5taXJyb3JzW2Rpc2tdLnJkZXYpOwoKCQlpZiAoIXJkZXYgfHwgcjFfYmlvLT5iaW9zW2Rpc2tdID09IElPX0JMT0NLRUQgfHwKCQkgICAgIXRlc3RfYml0KEluX3N5bmMsICZyZGV2LT5mbGFncykgfHwKCQkgICAgdGVzdF9iaXQoV3JpdGVNb3N0bHksICZyZGV2LT5mbGFncykpCgkJCWNvbnRpbnVlOwoKCQlpZiAoIWF0b21pY19yZWFkKCZyZGV2LT5ucl9wZW5kaW5nKSkgewoJCQluZXdfZGlzayA9IGRpc2s7CgkJCWJyZWFrOwoJCX0KCQluZXdfZGlzdGFuY2UgPSBhYnModGhpc19zZWN0b3IgLSBjb25mLT5taXJyb3JzW2Rpc2tdLmhlYWRfcG9zaXRpb24pOwoJCWlmIChuZXdfZGlzdGFuY2UgPCBjdXJyZW50X2Rpc3RhbmNlKSB7CgkJCWN1cnJlbnRfZGlzdGFuY2UgPSBuZXdfZGlzdGFuY2U7CgkJCW5ld19kaXNrID0gZGlzazsKCQl9Cgl9IHdoaWxlIChkaXNrICE9IGNvbmYtPmxhc3RfdXNlZCk7CgogcmJfb3V0OgoKCglpZiAobmV3X2Rpc2sgPj0gMCkgewoJCXJkZXYgPSByY3VfZGVyZWZlcmVuY2UoY29uZi0+bWlycm9yc1tuZXdfZGlza10ucmRldik7CgkJaWYgKCFyZGV2KQoJCQlnb3RvIHJldHJ5OwoJCWF0b21pY19pbmMoJnJkZXYtPm5yX3BlbmRpbmcpOwoJCWlmICghdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSkgewoJCQkvKiBjYW5ub3QgcmlzayByZXR1cm5pbmcgYSBkZXZpY2UgdGhhdCBmYWlsZWQKCQkJICogYmVmb3JlIHdlIGluYydlZCBucl9wZW5kaW5nCgkJCSAqLwoJCQlyZGV2X2RlY19wZW5kaW5nKHJkZXYsIGNvbmYtPm1kZGV2KTsKCQkJZ290byByZXRyeTsKCQl9CgkJY29uZi0+bmV4dF9zZXFfc2VjdCA9IHRoaXNfc2VjdG9yICsgc2VjdG9yczsKCQljb25mLT5sYXN0X3VzZWQgPSBuZXdfZGlzazsKCX0KCXJjdV9yZWFkX3VubG9jaygpOwoKCXJldHVybiBuZXdfZGlzazsKfQoKc3RhdGljIHZvaWQgdW5wbHVnX3NsYXZlcyhtZGRldl90ICptZGRldikKewoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CglpbnQgaTsKCglyY3VfcmVhZF9sb2NrKCk7Cglmb3IgKGk9MDsgaTxtZGRldi0+cmFpZF9kaXNrczsgaSsrKSB7CgkJbWRrX3JkZXZfdCAqcmRldiA9IHJjdV9kZXJlZmVyZW5jZShjb25mLT5taXJyb3JzW2ldLnJkZXYpOwoJCWlmIChyZGV2ICYmICF0ZXN0X2JpdChGYXVsdHksICZyZGV2LT5mbGFncykgJiYgYXRvbWljX3JlYWQoJnJkZXYtPm5yX3BlbmRpbmcpKSB7CgkJCXJlcXVlc3RfcXVldWVfdCAqcl9xdWV1ZSA9IGJkZXZfZ2V0X3F1ZXVlKHJkZXYtPmJkZXYpOwoKCQkJYXRvbWljX2luYygmcmRldi0+bnJfcGVuZGluZyk7CgkJCXJjdV9yZWFkX3VubG9jaygpOwoKCQkJaWYgKHJfcXVldWUtPnVucGx1Z19mbikKCQkJCXJfcXVldWUtPnVucGx1Z19mbihyX3F1ZXVlKTsKCgkJCXJkZXZfZGVjX3BlbmRpbmcocmRldiwgbWRkZXYpOwoJCQlyY3VfcmVhZF9sb2NrKCk7CgkJfQoJfQoJcmN1X3JlYWRfdW5sb2NrKCk7Cn0KCnN0YXRpYyB2b2lkIHJhaWQxX3VucGx1ZyhyZXF1ZXN0X3F1ZXVlX3QgKnEpCnsKCW1kZGV2X3QgKm1kZGV2ID0gcS0+cXVldWVkYXRhOwoKCXVucGx1Z19zbGF2ZXMobWRkZXYpOwoJbWRfd2FrZXVwX3RocmVhZChtZGRldi0+dGhyZWFkKTsKfQoKc3RhdGljIGludCByYWlkMV9pc3N1ZV9mbHVzaChyZXF1ZXN0X3F1ZXVlX3QgKnEsIHN0cnVjdCBnZW5kaXNrICpkaXNrLAoJCQkgICAgIHNlY3Rvcl90ICplcnJvcl9zZWN0b3IpCnsKCW1kZGV2X3QgKm1kZGV2ID0gcS0+cXVldWVkYXRhOwoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CglpbnQgaSwgcmV0ID0gMDsKCglyY3VfcmVhZF9sb2NrKCk7Cglmb3IgKGk9MDsgaTxtZGRldi0+cmFpZF9kaXNrcyAmJiByZXQgPT0gMDsgaSsrKSB7CgkJbWRrX3JkZXZfdCAqcmRldiA9IHJjdV9kZXJlZmVyZW5jZShjb25mLT5taXJyb3JzW2ldLnJkZXYpOwoJCWlmIChyZGV2ICYmICF0ZXN0X2JpdChGYXVsdHksICZyZGV2LT5mbGFncykpIHsKCQkJc3RydWN0IGJsb2NrX2RldmljZSAqYmRldiA9IHJkZXYtPmJkZXY7CgkJCXJlcXVlc3RfcXVldWVfdCAqcl9xdWV1ZSA9IGJkZXZfZ2V0X3F1ZXVlKGJkZXYpOwoKCQkJaWYgKCFyX3F1ZXVlLT5pc3N1ZV9mbHVzaF9mbikKCQkJCXJldCA9IC1FT1BOT1RTVVBQOwoJCQllbHNlIHsKCQkJCWF0b21pY19pbmMoJnJkZXYtPm5yX3BlbmRpbmcpOwoJCQkJcmN1X3JlYWRfdW5sb2NrKCk7CgkJCQlyZXQgPSByX3F1ZXVlLT5pc3N1ZV9mbHVzaF9mbihyX3F1ZXVlLCBiZGV2LT5iZF9kaXNrLAoJCQkJCQkJICAgICAgZXJyb3Jfc2VjdG9yKTsKCQkJCXJkZXZfZGVjX3BlbmRpbmcocmRldiwgbWRkZXYpOwoJCQkJcmN1X3JlYWRfbG9jaygpOwoJCQl9CgkJfQoJfQoJcmN1X3JlYWRfdW5sb2NrKCk7CglyZXR1cm4gcmV0Owp9CgpzdGF0aWMgaW50IHJhaWQxX2Nvbmdlc3RlZCh2b2lkICpkYXRhLCBpbnQgYml0cykKewoJbWRkZXZfdCAqbWRkZXYgPSBkYXRhOwoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CglpbnQgaSwgcmV0ID0gMDsKCglyY3VfcmVhZF9sb2NrKCk7Cglmb3IgKGkgPSAwOyBpIDwgbWRkZXYtPnJhaWRfZGlza3M7IGkrKykgewoJCW1ka19yZGV2X3QgKnJkZXYgPSByY3VfZGVyZWZlcmVuY2UoY29uZi0+bWlycm9yc1tpXS5yZGV2KTsKCQlpZiAocmRldiAmJiAhdGVzdF9iaXQoRmF1bHR5LCAmcmRldi0+ZmxhZ3MpKSB7CgkJCXJlcXVlc3RfcXVldWVfdCAqcSA9IGJkZXZfZ2V0X3F1ZXVlKHJkZXYtPmJkZXYpOwoKCQkJLyogTm90ZSB0aGUgJ3x8IDEnIC0gd2hlbiByZWFkX2JhbGFuY2UgcHJlZmVycwoJCQkgKiBub24tY29uZ2VzdGVkIHRhcmdldHMsIGl0IGNhbiBiZSByZW1vdmVkCgkJCSAqLwoJCQlpZiAoKGJpdHMgJiAoMTw8QkRJX3dyaXRlX2Nvbmdlc3RlZCkpIHx8IDEpCgkJCQlyZXQgfD0gYmRpX2Nvbmdlc3RlZCgmcS0+YmFja2luZ19kZXZfaW5mbywgYml0cyk7CgkJCWVsc2UKCQkJCXJldCAmPSBiZGlfY29uZ2VzdGVkKCZxLT5iYWNraW5nX2Rldl9pbmZvLCBiaXRzKTsKCQl9Cgl9CglyY3VfcmVhZF91bmxvY2soKTsKCXJldHVybiByZXQ7Cn0KCgovKiBCYXJyaWVycy4uLi4KICogU29tZXRpbWVzIHdlIG5lZWQgdG8gc3VzcGVuZCBJTyB3aGlsZSB3ZSBkbyBzb21ldGhpbmcgZWxzZSwKICogZWl0aGVyIHNvbWUgcmVzeW5jL3JlY292ZXJ5LCBvciByZWNvbmZpZ3VyZSB0aGUgYXJyYXkuCiAqIFRvIGRvIHRoaXMgd2UgcmFpc2UgYSAnYmFycmllcicuCiAqIFRoZSAnYmFycmllcicgaXMgYSBjb3VudGVyIHRoYXQgY2FuIGJlIHJhaXNlZCBtdWx0aXBsZSB0aW1lcwogKiB0byBjb3VudCBob3cgbWFueSBhY3Rpdml0aWVzIGFyZSBoYXBwZW5pbmcgd2hpY2ggcHJlY2x1ZGUKICogbm9ybWFsIElPLgogKiBXZSBjYW4gb25seSByYWlzZSB0aGUgYmFycmllciBpZiB0aGVyZSBpcyBubyBwZW5kaW5nIElPLgogKiBpLmUuIGlmIG5yX3BlbmRpbmcgPT0gMC4KICogV2UgY2hvb3NlIG9ubHkgdG8gcmFpc2UgdGhlIGJhcnJpZXIgaWYgbm8tb25lIGlzIHdhaXRpbmcgZm9yIHRoZQogKiBiYXJyaWVyIHRvIGdvIGRvd24uICBUaGlzIG1lYW5zIHRoYXQgYXMgc29vbiBhcyBhbiBJTyByZXF1ZXN0CiAqIGlzIHJlYWR5LCBubyBvdGhlciBvcGVyYXRpb25zIHdoaWNoIHJlcXVpcmUgYSBiYXJyaWVyIHdpbGwgc3RhcnQKICogdW50aWwgdGhlIElPIHJlcXVlc3QgaGFzIGhhZCBhIGNoYW5jZS4KICoKICogU286IHJlZ3VsYXIgSU8gY2FsbHMgJ3dhaXRfYmFycmllcicuICBXaGVuIHRoYXQgcmV0dXJucyB0aGVyZQogKiAgICBpcyBubyBiYWNrZ3JvdXAgSU8gaGFwcGVuaW5nLCAgSXQgbXVzdCBhcnJhbmdlIHRvIGNhbGwKICogICAgYWxsb3dfYmFycmllciB3aGVuIGl0IGhhcyBmaW5pc2hlZCBpdHMgSU8uCiAqIGJhY2tncm91cCBJTyBjYWxscyBtdXN0IGNhbGwgcmFpc2VfYmFycmllci4gIE9uY2UgdGhhdCByZXR1cm5zCiAqICAgIHRoZXJlIGlzIG5vIG5vcm1hbCBJTyBoYXBwZWluZy4gIEl0IG11c3QgYXJyYW5nZSB0byBjYWxsCiAqICAgIGxvd2VyX2JhcnJpZXIgd2hlbiB0aGUgcGFydGljdWxhciBiYWNrZ3JvdW5kIElPIGNvbXBsZXRlcy4KICovCiNkZWZpbmUgUkVTWU5DX0RFUFRIIDMyCgpzdGF0aWMgdm9pZCByYWlzZV9iYXJyaWVyKGNvbmZfdCAqY29uZikKewoJc3Bpbl9sb2NrX2lycSgmY29uZi0+cmVzeW5jX2xvY2spOwoKCS8qIFdhaXQgdW50aWwgbm8gYmxvY2sgSU8gaXMgd2FpdGluZyAqLwoJd2FpdF9ldmVudF9sb2NrX2lycShjb25mLT53YWl0X2JhcnJpZXIsICFjb25mLT5ucl93YWl0aW5nLAoJCQkgICAgY29uZi0+cmVzeW5jX2xvY2ssCgkJCSAgICByYWlkMV91bnBsdWcoY29uZi0+bWRkZXYtPnF1ZXVlKSk7CgoJLyogYmxvY2sgYW55IG5ldyBJTyBmcm9tIHN0YXJ0aW5nICovCgljb25mLT5iYXJyaWVyKys7CgoJLyogTm8gd2FpdCBmb3IgYWxsIHBlbmRpbmcgSU8gdG8gY29tcGxldGUgKi8KCXdhaXRfZXZlbnRfbG9ja19pcnEoY29uZi0+d2FpdF9iYXJyaWVyLAoJCQkgICAgIWNvbmYtPm5yX3BlbmRpbmcgJiYgY29uZi0+YmFycmllciA8IFJFU1lOQ19ERVBUSCwKCQkJICAgIGNvbmYtPnJlc3luY19sb2NrLAoJCQkgICAgcmFpZDFfdW5wbHVnKGNvbmYtPm1kZGV2LT5xdWV1ZSkpOwoKCXNwaW5fdW5sb2NrX2lycSgmY29uZi0+cmVzeW5jX2xvY2spOwp9CgpzdGF0aWMgdm9pZCBsb3dlcl9iYXJyaWVyKGNvbmZfdCAqY29uZikKewoJdW5zaWduZWQgbG9uZyBmbGFnczsKCXNwaW5fbG9ja19pcnFzYXZlKCZjb25mLT5yZXN5bmNfbG9jaywgZmxhZ3MpOwoJY29uZi0+YmFycmllci0tOwoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmY29uZi0+cmVzeW5jX2xvY2ssIGZsYWdzKTsKCXdha2VfdXAoJmNvbmYtPndhaXRfYmFycmllcik7Cn0KCnN0YXRpYyB2b2lkIHdhaXRfYmFycmllcihjb25mX3QgKmNvbmYpCnsKCXNwaW5fbG9ja19pcnEoJmNvbmYtPnJlc3luY19sb2NrKTsKCWlmIChjb25mLT5iYXJyaWVyKSB7CgkJY29uZi0+bnJfd2FpdGluZysrOwoJCXdhaXRfZXZlbnRfbG9ja19pcnEoY29uZi0+d2FpdF9iYXJyaWVyLCAhY29uZi0+YmFycmllciwKCQkJCSAgICBjb25mLT5yZXN5bmNfbG9jaywKCQkJCSAgICByYWlkMV91bnBsdWcoY29uZi0+bWRkZXYtPnF1ZXVlKSk7CgkJY29uZi0+bnJfd2FpdGluZy0tOwoJfQoJY29uZi0+bnJfcGVuZGluZysrOwoJc3Bpbl91bmxvY2tfaXJxKCZjb25mLT5yZXN5bmNfbG9jayk7Cn0KCnN0YXRpYyB2b2lkIGFsbG93X2JhcnJpZXIoY29uZl90ICpjb25mKQp7Cgl1bnNpZ25lZCBsb25nIGZsYWdzOwoJc3Bpbl9sb2NrX2lycXNhdmUoJmNvbmYtPnJlc3luY19sb2NrLCBmbGFncyk7Cgljb25mLT5ucl9wZW5kaW5nLS07CglzcGluX3VubG9ja19pcnFyZXN0b3JlKCZjb25mLT5yZXN5bmNfbG9jaywgZmxhZ3MpOwoJd2FrZV91cCgmY29uZi0+d2FpdF9iYXJyaWVyKTsKfQoKc3RhdGljIHZvaWQgZnJlZXplX2FycmF5KGNvbmZfdCAqY29uZikKewoJLyogc3RvcCBzeW5jaW8gYW5kIG5vcm1hbCBJTyBhbmQgd2FpdCBmb3IgZXZlcnl0aGluZyB0bwoJICogZ28gcXVpdGUuCgkgKiBXZSBpbmNyZW1lbnQgYmFycmllciBhbmQgbnJfd2FpdGluZywgYW5kIHRoZW4KCSAqIHdhaXQgdW50aWwgYmFycmllcitucl9wZW5kaW5nIG1hdGNoIG5yX3F1ZXVlZCsyCgkgKi8KCXNwaW5fbG9ja19pcnEoJmNvbmYtPnJlc3luY19sb2NrKTsKCWNvbmYtPmJhcnJpZXIrKzsKCWNvbmYtPm5yX3dhaXRpbmcrKzsKCXdhaXRfZXZlbnRfbG9ja19pcnEoY29uZi0+d2FpdF9iYXJyaWVyLAoJCQkgICAgY29uZi0+YmFycmllcitjb25mLT5ucl9wZW5kaW5nID09IGNvbmYtPm5yX3F1ZXVlZCsyLAoJCQkgICAgY29uZi0+cmVzeW5jX2xvY2ssCgkJCSAgICByYWlkMV91bnBsdWcoY29uZi0+bWRkZXYtPnF1ZXVlKSk7CglzcGluX3VubG9ja19pcnEoJmNvbmYtPnJlc3luY19sb2NrKTsKfQpzdGF0aWMgdm9pZCB1bmZyZWV6ZV9hcnJheShjb25mX3QgKmNvbmYpCnsKCS8qIHJldmVyc2UgdGhlIGVmZmVjdCBvZiB0aGUgZnJlZXplICovCglzcGluX2xvY2tfaXJxKCZjb25mLT5yZXN5bmNfbG9jayk7Cgljb25mLT5iYXJyaWVyLS07Cgljb25mLT5ucl93YWl0aW5nLS07Cgl3YWtlX3VwKCZjb25mLT53YWl0X2JhcnJpZXIpOwoJc3Bpbl91bmxvY2tfaXJxKCZjb25mLT5yZXN5bmNfbG9jayk7Cn0KCgovKiBkdXBsaWNhdGUgdGhlIGRhdGEgcGFnZXMgZm9yIGJlaGluZCBJL08gKi8Kc3RhdGljIHN0cnVjdCBwYWdlICoqYWxsb2NfYmVoaW5kX3BhZ2VzKHN0cnVjdCBiaW8gKmJpbykKewoJaW50IGk7CglzdHJ1Y3QgYmlvX3ZlYyAqYnZlYzsKCXN0cnVjdCBwYWdlICoqcGFnZXMgPSBremFsbG9jKGJpby0+YmlfdmNudCAqIHNpemVvZihzdHJ1Y3QgcGFnZSAqKSwKCQkJCQlHRlBfTk9JTyk7CglpZiAodW5saWtlbHkoIXBhZ2VzKSkKCQlnb3RvIGRvX3N5bmNfaW87CgoJYmlvX2Zvcl9lYWNoX3NlZ21lbnQoYnZlYywgYmlvLCBpKSB7CgkJcGFnZXNbaV0gPSBhbGxvY19wYWdlKEdGUF9OT0lPKTsKCQlpZiAodW5saWtlbHkoIXBhZ2VzW2ldKSkKCQkJZ290byBkb19zeW5jX2lvOwoJCW1lbWNweShrbWFwKHBhZ2VzW2ldKSArIGJ2ZWMtPmJ2X29mZnNldCwKCQkJa21hcChidmVjLT5idl9wYWdlKSArIGJ2ZWMtPmJ2X29mZnNldCwgYnZlYy0+YnZfbGVuKTsKCQlrdW5tYXAocGFnZXNbaV0pOwoJCWt1bm1hcChidmVjLT5idl9wYWdlKTsKCX0KCglyZXR1cm4gcGFnZXM7Cgpkb19zeW5jX2lvOgoJaWYgKHBhZ2VzKQoJCWZvciAoaSA9IDA7IGkgPCBiaW8tPmJpX3ZjbnQgJiYgcGFnZXNbaV07IGkrKykKCQkJcHV0X3BhZ2UocGFnZXNbaV0pOwoJa2ZyZWUocGFnZXMpOwoJUFJJTlRLKCIlZEIgYmVoaW5kIGFsbG9jIGZhaWxlZCwgZG9pbmcgc3luYyBJL09cbiIsIGJpby0+Ymlfc2l6ZSk7CglyZXR1cm4gTlVMTDsKfQoKc3RhdGljIGludCBtYWtlX3JlcXVlc3QocmVxdWVzdF9xdWV1ZV90ICpxLCBzdHJ1Y3QgYmlvICogYmlvKQp7CgltZGRldl90ICptZGRldiA9IHEtPnF1ZXVlZGF0YTsKCWNvbmZfdCAqY29uZiA9IG1kZGV2X3RvX2NvbmYobWRkZXYpOwoJbWlycm9yX2luZm9fdCAqbWlycm9yOwoJcjFiaW9fdCAqcjFfYmlvOwoJc3RydWN0IGJpbyAqcmVhZF9iaW87CglpbnQgaSwgdGFyZ2V0cyA9IDAsIGRpc2tzOwoJbWRrX3JkZXZfdCAqcmRldjsKCXN0cnVjdCBiaXRtYXAgKmJpdG1hcCA9IG1kZGV2LT5iaXRtYXA7Cgl1bnNpZ25lZCBsb25nIGZsYWdzOwoJc3RydWN0IGJpb19saXN0IGJsOwoJc3RydWN0IHBhZ2UgKipiZWhpbmRfcGFnZXMgPSBOVUxMOwoJY29uc3QgaW50IHJ3ID0gYmlvX2RhdGFfZGlyKGJpbyk7Cgljb25zdCBpbnQgZG9fc3luYyA9IGJpb19zeW5jKGJpbyk7CglpbnQgZG9fYmFycmllcnM7CgoJLyoKCSAqIFJlZ2lzdGVyIHRoZSBuZXcgcmVxdWVzdCBhbmQgd2FpdCBpZiB0aGUgcmVjb25zdHJ1Y3Rpb24KCSAqIHRocmVhZCBoYXMgcHV0IHVwIGEgYmFyIGZvciBuZXcgcmVxdWVzdHMuCgkgKiBDb250aW51ZSBpbW1lZGlhdGVseSBpZiBubyByZXN5bmMgaXMgYWN0aXZlIGN1cnJlbnRseS4KCSAqIFdlIHRlc3QgYmFycmllcnNfd29yayAqYWZ0ZXIqIG1kX3dyaXRlX3N0YXJ0IGFzIG1kX3dyaXRlX3N0YXJ0CgkgKiBtYXkgY2F1c2UgdGhlIGZpcnN0IHN1cGVyYmxvY2sgd3JpdGUsIGFuZCB0aGF0IHdpbGwgY2hlY2sgb3V0CgkgKiBpZiBiYXJyaWVycyB3b3JrLgoJICovCgoJbWRfd3JpdGVfc3RhcnQobWRkZXYsIGJpbyk7IC8qIHdhaXQgb24gc3VwZXJibG9jayB1cGRhdGUgZWFybHkgKi8KCglpZiAodW5saWtlbHkoIW1kZGV2LT5iYXJyaWVyc193b3JrICYmIGJpb19iYXJyaWVyKGJpbykpKSB7CgkJaWYgKHJ3ID09IFdSSVRFKQoJCQltZF93cml0ZV9lbmQobWRkZXYpOwoJCWJpb19lbmRpbyhiaW8sIGJpby0+Ymlfc2l6ZSwgLUVPUE5PVFNVUFApOwoJCXJldHVybiAwOwoJfQoKCXdhaXRfYmFycmllcihjb25mKTsKCglkaXNrX3N0YXRfaW5jKG1kZGV2LT5nZW5kaXNrLCBpb3NbcnddKTsKCWRpc2tfc3RhdF9hZGQobWRkZXYtPmdlbmRpc2ssIHNlY3RvcnNbcnddLCBiaW9fc2VjdG9ycyhiaW8pKTsKCgkvKgoJICogbWFrZV9yZXF1ZXN0KCkgY2FuIGFib3J0IHRoZSBvcGVyYXRpb24gd2hlbiBSRUFEQSBpcyBiZWluZwoJICogdXNlZCBhbmQgbm8gZW1wdHkgcmVxdWVzdCBpcyBhdmFpbGFibGUuCgkgKgoJICovCglyMV9iaW8gPSBtZW1wb29sX2FsbG9jKGNvbmYtPnIxYmlvX3Bvb2wsIEdGUF9OT0lPKTsKCglyMV9iaW8tPm1hc3Rlcl9iaW8gPSBiaW87CglyMV9iaW8tPnNlY3RvcnMgPSBiaW8tPmJpX3NpemUgPj4gOTsKCXIxX2Jpby0+c3RhdGUgPSAwOwoJcjFfYmlvLT5tZGRldiA9IG1kZGV2OwoJcjFfYmlvLT5zZWN0b3IgPSBiaW8tPmJpX3NlY3RvcjsKCglpZiAocncgPT0gUkVBRCkgewoJCS8qCgkJICogcmVhZCBiYWxhbmNpbmcgbG9naWM6CgkJICovCgkJaW50IHJkaXNrID0gcmVhZF9iYWxhbmNlKGNvbmYsIHIxX2Jpbyk7CgoJCWlmIChyZGlzayA8IDApIHsKCQkJLyogY291bGRuJ3QgZmluZCBhbnl3aGVyZSB0byByZWFkIGZyb20gKi8KCQkJcmFpZF9lbmRfYmlvX2lvKHIxX2Jpbyk7CgkJCXJldHVybiAwOwoJCX0KCQltaXJyb3IgPSBjb25mLT5taXJyb3JzICsgcmRpc2s7CgoJCXIxX2Jpby0+cmVhZF9kaXNrID0gcmRpc2s7CgoJCXJlYWRfYmlvID0gYmlvX2Nsb25lKGJpbywgR0ZQX05PSU8pOwoKCQlyMV9iaW8tPmJpb3NbcmRpc2tdID0gcmVhZF9iaW87CgoJCXJlYWRfYmlvLT5iaV9zZWN0b3IgPSByMV9iaW8tPnNlY3RvciArIG1pcnJvci0+cmRldi0+ZGF0YV9vZmZzZXQ7CgkJcmVhZF9iaW8tPmJpX2JkZXYgPSBtaXJyb3ItPnJkZXYtPmJkZXY7CgkJcmVhZF9iaW8tPmJpX2VuZF9pbyA9IHJhaWQxX2VuZF9yZWFkX3JlcXVlc3Q7CgkJcmVhZF9iaW8tPmJpX3J3ID0gUkVBRCB8IGRvX3N5bmM7CgkJcmVhZF9iaW8tPmJpX3ByaXZhdGUgPSByMV9iaW87CgoJCWdlbmVyaWNfbWFrZV9yZXF1ZXN0KHJlYWRfYmlvKTsKCQlyZXR1cm4gMDsKCX0KCgkvKgoJICogV1JJVEU6CgkgKi8KCS8qIGZpcnN0IHNlbGVjdCB0YXJnZXQgZGV2aWNlcyB1bmRlciBzcGlubG9jayBhbmQKCSAqIGluYyByZWZjb3VudCBvbiB0aGVpciByZGV2LiAgUmVjb3JkIHRoZW0gYnkgc2V0dGluZwoJICogYmlvc1t4XSB0byBiaW8KCSAqLwoJZGlza3MgPSBjb25mLT5yYWlkX2Rpc2tzOwojaWYgMAoJeyBzdGF0aWMgaW50IGZpcnN0PTE7CglpZiAoZmlyc3QpIHByaW50aygiRmlyc3QgV3JpdGUgc2VjdG9yICVsbHUgZGlza3MgJWRcbiIsCgkJCSAgKHVuc2lnbmVkIGxvbmcgbG9uZylyMV9iaW8tPnNlY3RvciwgZGlza3MpOwoJZmlyc3QgPSAwOwoJfQojZW5kaWYKCXJjdV9yZWFkX2xvY2soKTsKCWZvciAoaSA9IDA7ICBpIDwgZGlza3M7IGkrKykgewoJCWlmICgocmRldj1yY3VfZGVyZWZlcmVuY2UoY29uZi0+bWlycm9yc1tpXS5yZGV2KSkgIT0gTlVMTCAmJgoJCSAgICAhdGVzdF9iaXQoRmF1bHR5LCAmcmRldi0+ZmxhZ3MpKSB7CgkJCWF0b21pY19pbmMoJnJkZXYtPm5yX3BlbmRpbmcpOwoJCQlpZiAodGVzdF9iaXQoRmF1bHR5LCAmcmRldi0+ZmxhZ3MpKSB7CgkJCQlyZGV2X2RlY19wZW5kaW5nKHJkZXYsIG1kZGV2KTsKCQkJCXIxX2Jpby0+Ymlvc1tpXSA9IE5VTEw7CgkJCX0gZWxzZQoJCQkJcjFfYmlvLT5iaW9zW2ldID0gYmlvOwoJCQl0YXJnZXRzKys7CgkJfSBlbHNlCgkJCXIxX2Jpby0+Ymlvc1tpXSA9IE5VTEw7Cgl9CglyY3VfcmVhZF91bmxvY2soKTsKCglCVUdfT04odGFyZ2V0cyA9PSAwKTsgLyogd2UgbmV2ZXIgZmFpbCB0aGUgbGFzdCBkZXZpY2UgKi8KCglpZiAodGFyZ2V0cyA8IGNvbmYtPnJhaWRfZGlza3MpIHsKCQkvKiBhcnJheSBpcyBkZWdyYWRlZCwgd2Ugd2lsbCBub3QgY2xlYXIgdGhlIGJpdG1hcAoJCSAqIG9uIEkvTyBjb21wbGV0aW9uIChzZWUgcmFpZDFfZW5kX3dyaXRlX3JlcXVlc3QpICovCgkJc2V0X2JpdChSMUJJT19EZWdyYWRlZCwgJnIxX2Jpby0+c3RhdGUpOwoJfQoKCS8qIGRvIGJlaGluZCBJL08gPyAqLwoJaWYgKGJpdG1hcCAmJgoJICAgIGF0b21pY19yZWFkKCZiaXRtYXAtPmJlaGluZF93cml0ZXMpIDwgYml0bWFwLT5tYXhfd3JpdGVfYmVoaW5kICYmCgkgICAgKGJlaGluZF9wYWdlcyA9IGFsbG9jX2JlaGluZF9wYWdlcyhiaW8pKSAhPSBOVUxMKQoJCXNldF9iaXQoUjFCSU9fQmVoaW5kSU8sICZyMV9iaW8tPnN0YXRlKTsKCglhdG9taWNfc2V0KCZyMV9iaW8tPnJlbWFpbmluZywgMCk7CglhdG9taWNfc2V0KCZyMV9iaW8tPmJlaGluZF9yZW1haW5pbmcsIDApOwoKCWRvX2JhcnJpZXJzID0gYmlvX2JhcnJpZXIoYmlvKTsKCWlmIChkb19iYXJyaWVycykKCQlzZXRfYml0KFIxQklPX0JhcnJpZXIsICZyMV9iaW8tPnN0YXRlKTsKCgliaW9fbGlzdF9pbml0KCZibCk7Cglmb3IgKGkgPSAwOyBpIDwgZGlza3M7IGkrKykgewoJCXN0cnVjdCBiaW8gKm1iaW87CgkJaWYgKCFyMV9iaW8tPmJpb3NbaV0pCgkJCWNvbnRpbnVlOwoKCQltYmlvID0gYmlvX2Nsb25lKGJpbywgR0ZQX05PSU8pOwoJCXIxX2Jpby0+Ymlvc1tpXSA9IG1iaW87CgoJCW1iaW8tPmJpX3NlY3Rvcgk9IHIxX2Jpby0+c2VjdG9yICsgY29uZi0+bWlycm9yc1tpXS5yZGV2LT5kYXRhX29mZnNldDsKCQltYmlvLT5iaV9iZGV2ID0gY29uZi0+bWlycm9yc1tpXS5yZGV2LT5iZGV2OwoJCW1iaW8tPmJpX2VuZF9pbwk9IHJhaWQxX2VuZF93cml0ZV9yZXF1ZXN0OwoJCW1iaW8tPmJpX3J3ID0gV1JJVEUgfCBkb19iYXJyaWVycyB8IGRvX3N5bmM7CgkJbWJpby0+YmlfcHJpdmF0ZSA9IHIxX2JpbzsKCgkJaWYgKGJlaGluZF9wYWdlcykgewoJCQlzdHJ1Y3QgYmlvX3ZlYyAqYnZlYzsKCQkJaW50IGo7CgoJCQkvKiBZZXMsIEkgcmVhbGx5IHdhbnQgdGhlICdfXycgdmVyc2lvbiBzbyB0aGF0CgkJCSAqIHdlIGNsZWFyIGFueSB1bnVzZWQgcG9pbnRlciBpbiB0aGUgaW9fdmVjLCByYXRoZXIKCQkJICogdGhhbiBsZWF2ZSB0aGVtIHVuY2hhbmdlZC4gIFRoaXMgaXMgaW1wb3J0YW50CgkJCSAqIGJlY2F1c2Ugd2hlbiB3ZSBjb21lIHRvIGZyZWUgdGhlIHBhZ2VzLCB3ZSB3b24ndAoJCQkgKiBrbm93IHRoZSBvcmlnaW5pYWwgYmlfaWR4LCBzbyB3ZSBqdXN0IGZyZWUKCQkJICogdGhlbSBhbGwKCQkJICovCgkJCV9fYmlvX2Zvcl9lYWNoX3NlZ21lbnQoYnZlYywgbWJpbywgaiwgMCkKCQkJCWJ2ZWMtPmJ2X3BhZ2UgPSBiZWhpbmRfcGFnZXNbal07CgkJCWlmICh0ZXN0X2JpdChXcml0ZU1vc3RseSwgJmNvbmYtPm1pcnJvcnNbaV0ucmRldi0+ZmxhZ3MpKQoJCQkJYXRvbWljX2luYygmcjFfYmlvLT5iZWhpbmRfcmVtYWluaW5nKTsKCQl9CgoJCWF0b21pY19pbmMoJnIxX2Jpby0+cmVtYWluaW5nKTsKCgkJYmlvX2xpc3RfYWRkKCZibCwgbWJpbyk7Cgl9CglrZnJlZShiZWhpbmRfcGFnZXMpOyAvKiB0aGUgYmVoaW5kIHBhZ2VzIGFyZSBhdHRhY2hlZCB0byB0aGUgYmlvcyBub3cgKi8KCgliaXRtYXBfc3RhcnR3cml0ZShiaXRtYXAsIGJpby0+Ymlfc2VjdG9yLCByMV9iaW8tPnNlY3RvcnMsCgkJCQl0ZXN0X2JpdChSMUJJT19CZWhpbmRJTywgJnIxX2Jpby0+c3RhdGUpKTsKCXNwaW5fbG9ja19pcnFzYXZlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoJYmlvX2xpc3RfbWVyZ2UoJmNvbmYtPnBlbmRpbmdfYmlvX2xpc3QsICZibCk7CgliaW9fbGlzdF9pbml0KCZibCk7CgoJYmxrX3BsdWdfZGV2aWNlKG1kZGV2LT5xdWV1ZSk7CglzcGluX3VubG9ja19pcnFyZXN0b3JlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoKCWlmIChkb19zeW5jKQoJCW1kX3dha2V1cF90aHJlYWQobWRkZXYtPnRocmVhZCk7CiNpZiAwCgl3aGlsZSAoKGJpbyA9IGJpb19saXN0X3BvcCgmYmwpKSAhPSBOVUxMKQoJCWdlbmVyaWNfbWFrZV9yZXF1ZXN0KGJpbyk7CiNlbmRpZgoKCXJldHVybiAwOwp9CgpzdGF0aWMgdm9pZCBzdGF0dXMoc3RydWN0IHNlcV9maWxlICpzZXEsIG1kZGV2X3QgKm1kZGV2KQp7Cgljb25mX3QgKmNvbmYgPSBtZGRldl90b19jb25mKG1kZGV2KTsKCWludCBpOwoKCXNlcV9wcmludGYoc2VxLCAiIFslZC8lZF0gWyIsIGNvbmYtPnJhaWRfZGlza3MsCgkJICAgY29uZi0+cmFpZF9kaXNrcyAtIG1kZGV2LT5kZWdyYWRlZCk7CglyY3VfcmVhZF9sb2NrKCk7Cglmb3IgKGkgPSAwOyBpIDwgY29uZi0+cmFpZF9kaXNrczsgaSsrKSB7CgkJbWRrX3JkZXZfdCAqcmRldiA9IHJjdV9kZXJlZmVyZW5jZShjb25mLT5taXJyb3JzW2ldLnJkZXYpOwoJCXNlcV9wcmludGYoc2VxLCAiJXMiLAoJCQkgICByZGV2ICYmIHRlc3RfYml0KEluX3N5bmMsICZyZGV2LT5mbGFncykgPyAiVSIgOiAiXyIpOwoJfQoJcmN1X3JlYWRfdW5sb2NrKCk7CglzZXFfcHJpbnRmKHNlcSwgIl0iKTsKfQoKCnN0YXRpYyB2b2lkIGVycm9yKG1kZGV2X3QgKm1kZGV2LCBtZGtfcmRldl90ICpyZGV2KQp7CgljaGFyIGJbQkRFVk5BTUVfU0laRV07Cgljb25mX3QgKmNvbmYgPSBtZGRldl90b19jb25mKG1kZGV2KTsKCgkvKgoJICogSWYgaXQgaXMgbm90IG9wZXJhdGlvbmFsLCB0aGVuIHdlIGhhdmUgYWxyZWFkeSBtYXJrZWQgaXQgYXMgZGVhZAoJICogZWxzZSBpZiBpdCBpcyB0aGUgbGFzdCB3b3JraW5nIGRpc2tzLCBpZ25vcmUgdGhlIGVycm9yLCBsZXQgdGhlCgkgKiBuZXh0IGxldmVsIHVwIGtub3cuCgkgKiBlbHNlIG1hcmsgdGhlIGRyaXZlIGFzIGZhaWxlZAoJICovCglpZiAodGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKQoJICAgICYmIChjb25mLT5yYWlkX2Rpc2tzIC0gbWRkZXYtPmRlZ3JhZGVkKSA9PSAxKQoJCS8qCgkJICogRG9uJ3QgZmFpbCB0aGUgZHJpdmUsIGFjdCBhcyB0aG91Z2ggd2Ugd2VyZSBqdXN0IGEKCQkgKiBub3JtYWwgc2luZ2xlIGRyaXZlCgkJICovCgkJcmV0dXJuOwoJaWYgKHRlc3RfYW5kX2NsZWFyX2JpdChJbl9zeW5jLCAmcmRldi0+ZmxhZ3MpKSB7CgkJdW5zaWduZWQgbG9uZyBmbGFnczsKCQlzcGluX2xvY2tfaXJxc2F2ZSgmY29uZi0+ZGV2aWNlX2xvY2ssIGZsYWdzKTsKCQltZGRldi0+ZGVncmFkZWQrKzsKCQlzcGluX3VubG9ja19pcnFyZXN0b3JlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoJCS8qCgkJICogaWYgcmVjb3ZlcnkgaXMgcnVubmluZywgbWFrZSBzdXJlIGl0IGFib3J0cy4KCQkgKi8KCQlzZXRfYml0KE1EX1JFQ09WRVJZX0VSUiwgJm1kZGV2LT5yZWNvdmVyeSk7Cgl9CglzZXRfYml0KEZhdWx0eSwgJnJkZXYtPmZsYWdzKTsKCXNldF9iaXQoTURfQ0hBTkdFX0RFVlMsICZtZGRldi0+ZmxhZ3MpOwoJcHJpbnRrKEtFUk5fQUxFUlQgInJhaWQxOiBEaXNrIGZhaWx1cmUgb24gJXMsIGRpc2FibGluZyBkZXZpY2UuIFxuIgoJCSIJT3BlcmF0aW9uIGNvbnRpbnVpbmcgb24gJWQgZGV2aWNlc1xuIiwKCQliZGV2bmFtZShyZGV2LT5iZGV2LGIpLCBjb25mLT5yYWlkX2Rpc2tzIC0gbWRkZXYtPmRlZ3JhZGVkKTsKfQoKc3RhdGljIHZvaWQgcHJpbnRfY29uZihjb25mX3QgKmNvbmYpCnsKCWludCBpOwoKCXByaW50aygiUkFJRDEgY29uZiBwcmludG91dDpcbiIpOwoJaWYgKCFjb25mKSB7CgkJcHJpbnRrKCIoIWNvbmYpXG4iKTsKCQlyZXR1cm47Cgl9CglwcmludGsoIiAtLS0gd2Q6JWQgcmQ6JWRcbiIsIGNvbmYtPnJhaWRfZGlza3MgLSBjb25mLT5tZGRldi0+ZGVncmFkZWQsCgkJY29uZi0+cmFpZF9kaXNrcyk7CgoJcmN1X3JlYWRfbG9jaygpOwoJZm9yIChpID0gMDsgaSA8IGNvbmYtPnJhaWRfZGlza3M7IGkrKykgewoJCWNoYXIgYltCREVWTkFNRV9TSVpFXTsKCQltZGtfcmRldl90ICpyZGV2ID0gcmN1X2RlcmVmZXJlbmNlKGNvbmYtPm1pcnJvcnNbaV0ucmRldik7CgkJaWYgKHJkZXYpCgkJCXByaW50aygiIGRpc2sgJWQsIHdvOiVkLCBvOiVkLCBkZXY6JXNcbiIsCgkJCSAgICAgICBpLCAhdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSwKCQkJICAgICAgICF0ZXN0X2JpdChGYXVsdHksICZyZGV2LT5mbGFncyksCgkJCSAgICAgICBiZGV2bmFtZShyZGV2LT5iZGV2LGIpKTsKCX0KCXJjdV9yZWFkX3VubG9jaygpOwp9CgpzdGF0aWMgdm9pZCBjbG9zZV9zeW5jKGNvbmZfdCAqY29uZikKewoJd2FpdF9iYXJyaWVyKGNvbmYpOwoJYWxsb3dfYmFycmllcihjb25mKTsKCgltZW1wb29sX2Rlc3Ryb3koY29uZi0+cjFidWZfcG9vbCk7Cgljb25mLT5yMWJ1Zl9wb29sID0gTlVMTDsKfQoKc3RhdGljIGludCByYWlkMV9zcGFyZV9hY3RpdmUobWRkZXZfdCAqbWRkZXYpCnsKCWludCBpOwoJY29uZl90ICpjb25mID0gbWRkZXYtPnByaXZhdGU7CgoJLyoKCSAqIEZpbmQgYWxsIGZhaWxlZCBkaXNrcyB3aXRoaW4gdGhlIFJBSUQxIGNvbmZpZ3VyYXRpb24gCgkgKiBhbmQgbWFyayB0aGVtIHJlYWRhYmxlLgoJICogQ2FsbGVkIHVuZGVyIG1kZGV2IGxvY2ssIHNvIHJjdSBwcm90ZWN0aW9uIG5vdCBuZWVkZWQuCgkgKi8KCWZvciAoaSA9IDA7IGkgPCBjb25mLT5yYWlkX2Rpc2tzOyBpKyspIHsKCQltZGtfcmRldl90ICpyZGV2ID0gY29uZi0+bWlycm9yc1tpXS5yZGV2OwoJCWlmIChyZGV2CgkJICAgICYmICF0ZXN0X2JpdChGYXVsdHksICZyZGV2LT5mbGFncykKCQkgICAgJiYgIXRlc3RfYW5kX3NldF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSkgewoJCQl1bnNpZ25lZCBsb25nIGZsYWdzOwoJCQlzcGluX2xvY2tfaXJxc2F2ZSgmY29uZi0+ZGV2aWNlX2xvY2ssIGZsYWdzKTsKCQkJbWRkZXYtPmRlZ3JhZGVkLS07CgkJCXNwaW5fdW5sb2NrX2lycXJlc3RvcmUoJmNvbmYtPmRldmljZV9sb2NrLCBmbGFncyk7CgkJfQoJfQoKCXByaW50X2NvbmYoY29uZik7CglyZXR1cm4gMDsKfQoKCnN0YXRpYyBpbnQgcmFpZDFfYWRkX2Rpc2sobWRkZXZfdCAqbWRkZXYsIG1ka19yZGV2X3QgKnJkZXYpCnsKCWNvbmZfdCAqY29uZiA9IG1kZGV2LT5wcml2YXRlOwoJaW50IGZvdW5kID0gMDsKCWludCBtaXJyb3IgPSAwOwoJbWlycm9yX2luZm9fdCAqcDsKCglmb3IgKG1pcnJvcj0wOyBtaXJyb3IgPCBtZGRldi0+cmFpZF9kaXNrczsgbWlycm9yKyspCgkJaWYgKCAhKHA9Y29uZi0+bWlycm9ycyttaXJyb3IpLT5yZGV2KSB7CgoJCQlibGtfcXVldWVfc3RhY2tfbGltaXRzKG1kZGV2LT5xdWV1ZSwKCQkJCQkgICAgICAgcmRldi0+YmRldi0+YmRfZGlzay0+cXVldWUpOwoJCQkvKiBhcyB3ZSBkb24ndCBob25vdXIgbWVyZ2VfYnZlY19mbiwgd2UgbXVzdCBuZXZlciByaXNrCgkJCSAqIHZpb2xhdGluZyBpdCwgc28gbGltaXQgLT5tYXhfc2VjdG9yIHRvIG9uZSBQQUdFLCBhcwoJCQkgKiBhIG9uZSBwYWdlIHJlcXVlc3QgaXMgbmV2ZXIgaW4gdmlvbGF0aW9uLgoJCQkgKi8KCQkJaWYgKHJkZXYtPmJkZXYtPmJkX2Rpc2stPnF1ZXVlLT5tZXJnZV9idmVjX2ZuICYmCgkJCSAgICBtZGRldi0+cXVldWUtPm1heF9zZWN0b3JzID4gKFBBR0VfU0laRT4+OSkpCgkJCQlibGtfcXVldWVfbWF4X3NlY3RvcnMobWRkZXYtPnF1ZXVlLCBQQUdFX1NJWkU+PjkpOwoKCQkJcC0+aGVhZF9wb3NpdGlvbiA9IDA7CgkJCXJkZXYtPnJhaWRfZGlzayA9IG1pcnJvcjsKCQkJZm91bmQgPSAxOwoJCQkvKiBBcyBhbGwgZGV2aWNlcyBhcmUgZXF1aXZhbGVudCwgd2UgZG9uJ3QgbmVlZCBhIGZ1bGwgcmVjb3ZlcnkKCQkJICogaWYgdGhpcyB3YXMgcmVjZW50bHkgYW55IGRyaXZlIG9mIHRoZSBhcnJheQoJCQkgKi8KCQkJaWYgKHJkZXYtPnNhdmVkX3JhaWRfZGlzayA8IDApCgkJCQljb25mLT5mdWxsc3luYyA9IDE7CgkJCXJjdV9hc3NpZ25fcG9pbnRlcihwLT5yZGV2LCByZGV2KTsKCQkJYnJlYWs7CgkJfQoKCXByaW50X2NvbmYoY29uZik7CglyZXR1cm4gZm91bmQ7Cn0KCnN0YXRpYyBpbnQgcmFpZDFfcmVtb3ZlX2Rpc2sobWRkZXZfdCAqbWRkZXYsIGludCBudW1iZXIpCnsKCWNvbmZfdCAqY29uZiA9IG1kZGV2LT5wcml2YXRlOwoJaW50IGVyciA9IDA7CgltZGtfcmRldl90ICpyZGV2OwoJbWlycm9yX2luZm9fdCAqcCA9IGNvbmYtPm1pcnJvcnMrIG51bWJlcjsKCglwcmludF9jb25mKGNvbmYpOwoJcmRldiA9IHAtPnJkZXY7CglpZiAocmRldikgewoJCWlmICh0ZXN0X2JpdChJbl9zeW5jLCAmcmRldi0+ZmxhZ3MpIHx8CgkJICAgIGF0b21pY19yZWFkKCZyZGV2LT5ucl9wZW5kaW5nKSkgewoJCQllcnIgPSAtRUJVU1k7CgkJCWdvdG8gYWJvcnQ7CgkJfQoJCXAtPnJkZXYgPSBOVUxMOwoJCXN5bmNocm9uaXplX3JjdSgpOwoJCWlmIChhdG9taWNfcmVhZCgmcmRldi0+bnJfcGVuZGluZykpIHsKCQkJLyogbG9zdCB0aGUgcmFjZSwgdHJ5IGxhdGVyICovCgkJCWVyciA9IC1FQlVTWTsKCQkJcC0+cmRldiA9IHJkZXY7CgkJfQoJfQphYm9ydDoKCglwcmludF9jb25mKGNvbmYpOwoJcmV0dXJuIGVycjsKfQoKCnN0YXRpYyBpbnQgZW5kX3N5bmNfcmVhZChzdHJ1Y3QgYmlvICpiaW8sIHVuc2lnbmVkIGludCBieXRlc19kb25lLCBpbnQgZXJyb3IpCnsKCXIxYmlvX3QgKiByMV9iaW8gPSAocjFiaW9fdCAqKShiaW8tPmJpX3ByaXZhdGUpOwoJaW50IGk7CgoJaWYgKGJpby0+Ymlfc2l6ZSkKCQlyZXR1cm4gMTsKCglmb3IgKGk9cjFfYmlvLT5tZGRldi0+cmFpZF9kaXNrczsgaS0tOyApCgkJaWYgKHIxX2Jpby0+Ymlvc1tpXSA9PSBiaW8pCgkJCWJyZWFrOwoJQlVHX09OKGkgPCAwKTsKCXVwZGF0ZV9oZWFkX3BvcyhpLCByMV9iaW8pOwoJLyoKCSAqIHdlIGhhdmUgcmVhZCBhIGJsb2NrLCBub3cgaXQgbmVlZHMgdG8gYmUgcmUtd3JpdHRlbiwKCSAqIG9yIHJlLXJlYWQgaWYgdGhlIHJlYWQgZmFpbGVkLgoJICogV2UgZG9uJ3QgZG8gbXVjaCBoZXJlLCBqdXN0IHNjaGVkdWxlIGhhbmRsaW5nIGJ5IHJhaWQxZAoJICovCglpZiAodGVzdF9iaXQoQklPX1VQVE9EQVRFLCAmYmlvLT5iaV9mbGFncykpCgkJc2V0X2JpdChSMUJJT19VcHRvZGF0ZSwgJnIxX2Jpby0+c3RhdGUpOwoKCWlmIChhdG9taWNfZGVjX2FuZF90ZXN0KCZyMV9iaW8tPnJlbWFpbmluZykpCgkJcmVzY2hlZHVsZV9yZXRyeShyMV9iaW8pOwoJcmV0dXJuIDA7Cn0KCnN0YXRpYyBpbnQgZW5kX3N5bmNfd3JpdGUoc3RydWN0IGJpbyAqYmlvLCB1bnNpZ25lZCBpbnQgYnl0ZXNfZG9uZSwgaW50IGVycm9yKQp7CglpbnQgdXB0b2RhdGUgPSB0ZXN0X2JpdChCSU9fVVBUT0RBVEUsICZiaW8tPmJpX2ZsYWdzKTsKCXIxYmlvX3QgKiByMV9iaW8gPSAocjFiaW9fdCAqKShiaW8tPmJpX3ByaXZhdGUpOwoJbWRkZXZfdCAqbWRkZXYgPSByMV9iaW8tPm1kZGV2OwoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CglpbnQgaTsKCWludCBtaXJyb3I9MDsKCglpZiAoYmlvLT5iaV9zaXplKQoJCXJldHVybiAxOwoKCWZvciAoaSA9IDA7IGkgPCBjb25mLT5yYWlkX2Rpc2tzOyBpKyspCgkJaWYgKHIxX2Jpby0+Ymlvc1tpXSA9PSBiaW8pIHsKCQkJbWlycm9yID0gaTsKCQkJYnJlYWs7CgkJfQoJaWYgKCF1cHRvZGF0ZSkgewoJCWludCBzeW5jX2Jsb2NrcyA9IDA7CgkJc2VjdG9yX3QgcyA9IHIxX2Jpby0+c2VjdG9yOwoJCWxvbmcgc2VjdG9yc190b19nbyA9IHIxX2Jpby0+c2VjdG9yczsKCQkvKiBtYWtlIHN1cmUgdGhlc2UgYml0cyBkb2Vzbid0IGdldCBjbGVhcmVkLiAqLwoJCWRvIHsKCQkJYml0bWFwX2VuZF9zeW5jKG1kZGV2LT5iaXRtYXAsIHMsCgkJCQkJJnN5bmNfYmxvY2tzLCAxKTsKCQkJcyArPSBzeW5jX2Jsb2NrczsKCQkJc2VjdG9yc190b19nbyAtPSBzeW5jX2Jsb2NrczsKCQl9IHdoaWxlIChzZWN0b3JzX3RvX2dvID4gMCk7CgkJbWRfZXJyb3IobWRkZXYsIGNvbmYtPm1pcnJvcnNbbWlycm9yXS5yZGV2KTsKCX0KCgl1cGRhdGVfaGVhZF9wb3MobWlycm9yLCByMV9iaW8pOwoKCWlmIChhdG9taWNfZGVjX2FuZF90ZXN0KCZyMV9iaW8tPnJlbWFpbmluZykpIHsKCQltZF9kb25lX3N5bmMobWRkZXYsIHIxX2Jpby0+c2VjdG9ycywgdXB0b2RhdGUpOwoJCXB1dF9idWYocjFfYmlvKTsKCX0KCXJldHVybiAwOwp9CgpzdGF0aWMgdm9pZCBzeW5jX3JlcXVlc3Rfd3JpdGUobWRkZXZfdCAqbWRkZXYsIHIxYmlvX3QgKnIxX2JpbykKewoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CglpbnQgaTsKCWludCBkaXNrcyA9IGNvbmYtPnJhaWRfZGlza3M7CglzdHJ1Y3QgYmlvICpiaW8sICp3YmlvOwoKCWJpbyA9IHIxX2Jpby0+Ymlvc1tyMV9iaW8tPnJlYWRfZGlza107CgoKCWlmICh0ZXN0X2JpdChNRF9SRUNPVkVSWV9SRVFVRVNURUQsICZtZGRldi0+cmVjb3ZlcnkpKSB7CgkJLyogV2UgaGF2ZSByZWFkIGFsbCByZWFkYWJsZSBkZXZpY2VzLiAgSWYgd2UgaGF2ZW4ndAoJCSAqIGdvdCB0aGUgYmxvY2ssIHRoZW4gdGhlcmUgaXMgbm8gaG9wZSBsZWZ0LgoJCSAqIElmIHdlIGhhdmUsIHRoZW4gd2Ugd2FudCB0byBkbyBhIGNvbXBhcmlzb24KCQkgKiBhbmQgc2tpcCB0aGUgd3JpdGUgaWYgZXZlcnl0aGluZyBpcyB0aGUgc2FtZS4KCQkgKiBJZiBhbnkgYmxvY2tzIGZhaWxlZCB0byByZWFkLCB0aGVuIHdlIG5lZWQgdG8KCQkgKiBhdHRlbXB0IGFuIG92ZXItd3JpdGUKCQkgKi8KCQlpbnQgcHJpbWFyeTsKCQlpZiAoIXRlc3RfYml0KFIxQklPX1VwdG9kYXRlLCAmcjFfYmlvLT5zdGF0ZSkpIHsKCQkJZm9yIChpPTA7IGk8bWRkZXYtPnJhaWRfZGlza3M7IGkrKykKCQkJCWlmIChyMV9iaW8tPmJpb3NbaV0tPmJpX2VuZF9pbyA9PSBlbmRfc3luY19yZWFkKQoJCQkJCW1kX2Vycm9yKG1kZGV2LCBjb25mLT5taXJyb3JzW2ldLnJkZXYpOwoKCQkJbWRfZG9uZV9zeW5jKG1kZGV2LCByMV9iaW8tPnNlY3RvcnMsIDEpOwoJCQlwdXRfYnVmKHIxX2Jpbyk7CgkJCXJldHVybjsKCQl9CgkJZm9yIChwcmltYXJ5PTA7IHByaW1hcnk8bWRkZXYtPnJhaWRfZGlza3M7IHByaW1hcnkrKykKCQkJaWYgKHIxX2Jpby0+Ymlvc1twcmltYXJ5XS0+YmlfZW5kX2lvID09IGVuZF9zeW5jX3JlYWQgJiYKCQkJICAgIHRlc3RfYml0KEJJT19VUFRPREFURSwgJnIxX2Jpby0+Ymlvc1twcmltYXJ5XS0+YmlfZmxhZ3MpKSB7CgkJCQlyMV9iaW8tPmJpb3NbcHJpbWFyeV0tPmJpX2VuZF9pbyA9IE5VTEw7CgkJCQlyZGV2X2RlY19wZW5kaW5nKGNvbmYtPm1pcnJvcnNbcHJpbWFyeV0ucmRldiwgbWRkZXYpOwoJCQkJYnJlYWs7CgkJCX0KCQlyMV9iaW8tPnJlYWRfZGlzayA9IHByaW1hcnk7CgkJZm9yIChpPTA7IGk8bWRkZXYtPnJhaWRfZGlza3M7IGkrKykKCQkJaWYgKHIxX2Jpby0+Ymlvc1tpXS0+YmlfZW5kX2lvID09IGVuZF9zeW5jX3JlYWQgJiYKCQkJICAgIHRlc3RfYml0KEJJT19VUFRPREFURSwgJnIxX2Jpby0+Ymlvc1tpXS0+YmlfZmxhZ3MpKSB7CgkJCQlpbnQgajsKCQkJCWludCB2Y250ID0gcjFfYmlvLT5zZWN0b3JzID4+IChQQUdFX1NISUZULSA5KTsKCQkJCXN0cnVjdCBiaW8gKnBiaW8gPSByMV9iaW8tPmJpb3NbcHJpbWFyeV07CgkJCQlzdHJ1Y3QgYmlvICpzYmlvID0gcjFfYmlvLT5iaW9zW2ldOwoJCQkJZm9yIChqID0gdmNudDsgai0tIDsgKQoJCQkJCWlmIChtZW1jbXAocGFnZV9hZGRyZXNzKHBiaW8tPmJpX2lvX3ZlY1tqXS5idl9wYWdlKSwKCQkJCQkJICAgcGFnZV9hZGRyZXNzKHNiaW8tPmJpX2lvX3ZlY1tqXS5idl9wYWdlKSwKCQkJCQkJICAgUEFHRV9TSVpFKSkKCQkJCQkJYnJlYWs7CgkJCQlpZiAoaiA+PSAwKQoJCQkJCW1kZGV2LT5yZXN5bmNfbWlzbWF0Y2hlcyArPSByMV9iaW8tPnNlY3RvcnM7CgkJCQlpZiAoaiA8IDAgfHwgdGVzdF9iaXQoTURfUkVDT1ZFUllfQ0hFQ0ssICZtZGRldi0+cmVjb3ZlcnkpKSB7CgkJCQkJc2Jpby0+YmlfZW5kX2lvID0gTlVMTDsKCQkJCQlyZGV2X2RlY19wZW5kaW5nKGNvbmYtPm1pcnJvcnNbaV0ucmRldiwgbWRkZXYpOwoJCQkJfSBlbHNlIHsKCQkJCQkvKiBmaXh1cCB0aGUgYmlvIGZvciByZXVzZSAqLwoJCQkJCXNiaW8tPmJpX3ZjbnQgPSB2Y250OwoJCQkJCXNiaW8tPmJpX3NpemUgPSByMV9iaW8tPnNlY3RvcnMgPDwgOTsKCQkJCQlzYmlvLT5iaV9pZHggPSAwOwoJCQkJCXNiaW8tPmJpX3BoeXNfc2VnbWVudHMgPSAwOwoJCQkJCXNiaW8tPmJpX2h3X3NlZ21lbnRzID0gMDsKCQkJCQlzYmlvLT5iaV9od19mcm9udF9zaXplID0gMDsKCQkJCQlzYmlvLT5iaV9od19iYWNrX3NpemUgPSAwOwoJCQkJCXNiaW8tPmJpX2ZsYWdzICY9IH4oQklPX1BPT0xfTUFTSyAtIDEpOwoJCQkJCXNiaW8tPmJpX2ZsYWdzIHw9IDEgPDwgQklPX1VQVE9EQVRFOwoJCQkJCXNiaW8tPmJpX25leHQgPSBOVUxMOwoJCQkJCXNiaW8tPmJpX3NlY3RvciA9IHIxX2Jpby0+c2VjdG9yICsKCQkJCQkJY29uZi0+bWlycm9yc1tpXS5yZGV2LT5kYXRhX29mZnNldDsKCQkJCQlzYmlvLT5iaV9iZGV2ID0gY29uZi0+bWlycm9yc1tpXS5yZGV2LT5iZGV2OwoJCQkJfQoJCQl9Cgl9CglpZiAoIXRlc3RfYml0KFIxQklPX1VwdG9kYXRlLCAmcjFfYmlvLT5zdGF0ZSkpIHsKCQkvKiBvdWNoIC0gZmFpbGVkIHRvIHJlYWQgYWxsIG9mIHRoYXQuCgkJICogVHJ5IHNvbWUgc3luY2hyb25vdXMgcmVhZHMgb2Ygb3RoZXIgZGV2aWNlcyB0byBnZXQKCQkgKiBnb29kIGRhdGEsIG11Y2ggbGlrZSB3aXRoIG5vcm1hbCByZWFkIGVycm9ycy4gIE9ubHkKCQkgKiByZWFkIGludG8gdGhlIHBhZ2VzIHdlIGFscmVhZHkgaGF2ZSBzbyB3ZSBkb24ndAoJCSAqIG5lZWQgdG8gcmUtaXNzdWUgdGhlIHJlYWQgcmVxdWVzdC4KCQkgKiBXZSBkb24ndCBuZWVkIHRvIGZyZWV6ZSB0aGUgYXJyYXksIGJlY2F1c2UgYmVpbmcgaW4gYW4KCQkgKiBhY3RpdmUgc3luYyByZXF1ZXN0LCB0aGVyZSBpcyBubyBub3JtYWwgSU8sIGFuZAoJCSAqIG5vIG92ZXJsYXBwaW5nIHN5bmNzLgoJCSAqLwoJCXNlY3Rvcl90IHNlY3QgPSByMV9iaW8tPnNlY3RvcjsKCQlpbnQgc2VjdG9ycyA9IHIxX2Jpby0+c2VjdG9yczsKCQlpbnQgaWR4ID0gMDsKCgkJd2hpbGUoc2VjdG9ycykgewoJCQlpbnQgcyA9IHNlY3RvcnM7CgkJCWludCBkID0gcjFfYmlvLT5yZWFkX2Rpc2s7CgkJCWludCBzdWNjZXNzID0gMDsKCQkJbWRrX3JkZXZfdCAqcmRldjsKCgkJCWlmIChzID4gKFBBR0VfU0laRT4+OSkpCgkJCQlzID0gUEFHRV9TSVpFID4+IDk7CgkJCWRvIHsKCQkJCWlmIChyMV9iaW8tPmJpb3NbZF0tPmJpX2VuZF9pbyA9PSBlbmRfc3luY19yZWFkKSB7CgkJCQkJLyogTm8gcmN1IHByb3RlY3Rpb24gbmVlZGVkIGhlcmUgZGV2aWNlcwoJCQkJCSAqIGNhbiBvbmx5IGJlIHJlbW92ZWQgd2hlbiBubyByZXN5bmMgaXMKCQkJCQkgKiBhY3RpdmUsIGFuZCByZXN5bmMgaXMgY3VycmVudGx5IGFjdGl2ZQoJCQkJCSAqLwoJCQkJCXJkZXYgPSBjb25mLT5taXJyb3JzW2RdLnJkZXY7CgkJCQkJaWYgKHN5bmNfcGFnZV9pbyhyZGV2LT5iZGV2LAoJCQkJCQkJIHNlY3QgKyByZGV2LT5kYXRhX29mZnNldCwKCQkJCQkJCSBzPDw5LAoJCQkJCQkJIGJpby0+YmlfaW9fdmVjW2lkeF0uYnZfcGFnZSwKCQkJCQkJCSBSRUFEKSkgewoJCQkJCQlzdWNjZXNzID0gMTsKCQkJCQkJYnJlYWs7CgkJCQkJfQoJCQkJfQoJCQkJZCsrOwoJCQkJaWYgKGQgPT0gY29uZi0+cmFpZF9kaXNrcykKCQkJCQlkID0gMDsKCQkJfSB3aGlsZSAoIXN1Y2Nlc3MgJiYgZCAhPSByMV9iaW8tPnJlYWRfZGlzayk7CgoJCQlpZiAoc3VjY2VzcykgewoJCQkJaW50IHN0YXJ0ID0gZDsKCQkJCS8qIHdyaXRlIGl0IGJhY2sgYW5kIHJlLXJlYWQgKi8KCQkJCXNldF9iaXQoUjFCSU9fVXB0b2RhdGUsICZyMV9iaW8tPnN0YXRlKTsKCQkJCXdoaWxlIChkICE9IHIxX2Jpby0+cmVhZF9kaXNrKSB7CgkJCQkJaWYgKGQgPT0gMCkKCQkJCQkJZCA9IGNvbmYtPnJhaWRfZGlza3M7CgkJCQkJZC0tOwoJCQkJCWlmIChyMV9iaW8tPmJpb3NbZF0tPmJpX2VuZF9pbyAhPSBlbmRfc3luY19yZWFkKQoJCQkJCQljb250aW51ZTsKCQkJCQlyZGV2ID0gY29uZi0+bWlycm9yc1tkXS5yZGV2OwoJCQkJCWF0b21pY19hZGQocywgJnJkZXYtPmNvcnJlY3RlZF9lcnJvcnMpOwoJCQkJCWlmIChzeW5jX3BhZ2VfaW8ocmRldi0+YmRldiwKCQkJCQkJCSBzZWN0ICsgcmRldi0+ZGF0YV9vZmZzZXQsCgkJCQkJCQkgczw8OSwKCQkJCQkJCSBiaW8tPmJpX2lvX3ZlY1tpZHhdLmJ2X3BhZ2UsCgkJCQkJCQkgV1JJVEUpID09IDApCgkJCQkJCW1kX2Vycm9yKG1kZGV2LCByZGV2KTsKCQkJCX0KCQkJCWQgPSBzdGFydDsKCQkJCXdoaWxlIChkICE9IHIxX2Jpby0+cmVhZF9kaXNrKSB7CgkJCQkJaWYgKGQgPT0gMCkKCQkJCQkJZCA9IGNvbmYtPnJhaWRfZGlza3M7CgkJCQkJZC0tOwoJCQkJCWlmIChyMV9iaW8tPmJpb3NbZF0tPmJpX2VuZF9pbyAhPSBlbmRfc3luY19yZWFkKQoJCQkJCQljb250aW51ZTsKCQkJCQlyZGV2ID0gY29uZi0+bWlycm9yc1tkXS5yZGV2OwoJCQkJCWlmIChzeW5jX3BhZ2VfaW8ocmRldi0+YmRldiwKCQkJCQkJCSBzZWN0ICsgcmRldi0+ZGF0YV9vZmZzZXQsCgkJCQkJCQkgczw8OSwKCQkJCQkJCSBiaW8tPmJpX2lvX3ZlY1tpZHhdLmJ2X3BhZ2UsCgkJCQkJCQkgUkVBRCkgPT0gMCkKCQkJCQkJbWRfZXJyb3IobWRkZXYsIHJkZXYpOwoJCQkJfQoJCQl9IGVsc2UgewoJCQkJY2hhciBiW0JERVZOQU1FX1NJWkVdOwoJCQkJLyogQ2Fubm90IHJlYWQgZnJvbSBhbnl3aGVyZSwgYXJyYXkgaXMgdG9hc3QgKi8KCQkJCW1kX2Vycm9yKG1kZGV2LCBjb25mLT5taXJyb3JzW3IxX2Jpby0+cmVhZF9kaXNrXS5yZGV2KTsKCQkJCXByaW50ayhLRVJOX0FMRVJUICJyYWlkMTogJXM6IHVucmVjb3ZlcmFibGUgSS9PIHJlYWQgZXJyb3IiCgkJCQkgICAgICAgIiBmb3IgYmxvY2sgJWxsdVxuIiwKCQkJCSAgICAgICBiZGV2bmFtZShiaW8tPmJpX2JkZXYsYiksCgkJCQkgICAgICAgKHVuc2lnbmVkIGxvbmcgbG9uZylyMV9iaW8tPnNlY3Rvcik7CgkJCQltZF9kb25lX3N5bmMobWRkZXYsIHIxX2Jpby0+c2VjdG9ycywgMCk7CgkJCQlwdXRfYnVmKHIxX2Jpbyk7CgkJCQlyZXR1cm47CgkJCX0KCQkJc2VjdG9ycyAtPSBzOwoJCQlzZWN0ICs9IHM7CgkJCWlkeCArKzsKCQl9Cgl9CgoJLyoKCSAqIHNjaGVkdWxlIHdyaXRlcwoJICovCglhdG9taWNfc2V0KCZyMV9iaW8tPnJlbWFpbmluZywgMSk7Cglmb3IgKGkgPSAwOyBpIDwgZGlza3MgOyBpKyspIHsKCQl3YmlvID0gcjFfYmlvLT5iaW9zW2ldOwoJCWlmICh3YmlvLT5iaV9lbmRfaW8gPT0gTlVMTCB8fAoJCSAgICAod2Jpby0+YmlfZW5kX2lvID09IGVuZF9zeW5jX3JlYWQgJiYKCQkgICAgIChpID09IHIxX2Jpby0+cmVhZF9kaXNrIHx8CgkJICAgICAgIXRlc3RfYml0KE1EX1JFQ09WRVJZX1NZTkMsICZtZGRldi0+cmVjb3ZlcnkpKSkpCgkJCWNvbnRpbnVlOwoKCQl3YmlvLT5iaV9ydyA9IFdSSVRFOwoJCXdiaW8tPmJpX2VuZF9pbyA9IGVuZF9zeW5jX3dyaXRlOwoJCWF0b21pY19pbmMoJnIxX2Jpby0+cmVtYWluaW5nKTsKCQltZF9zeW5jX2FjY3QoY29uZi0+bWlycm9yc1tpXS5yZGV2LT5iZGV2LCB3YmlvLT5iaV9zaXplID4+IDkpOwoKCQlnZW5lcmljX21ha2VfcmVxdWVzdCh3YmlvKTsKCX0KCglpZiAoYXRvbWljX2RlY19hbmRfdGVzdCgmcjFfYmlvLT5yZW1haW5pbmcpKSB7CgkJLyogaWYgd2UncmUgaGVyZSwgYWxsIHdyaXRlKHMpIGhhdmUgY29tcGxldGVkLCBzbyBjbGVhbiB1cCAqLwoJCW1kX2RvbmVfc3luYyhtZGRldiwgcjFfYmlvLT5zZWN0b3JzLCAxKTsKCQlwdXRfYnVmKHIxX2Jpbyk7Cgl9Cn0KCi8qCiAqIFRoaXMgaXMgYSBrZXJuZWwgdGhyZWFkIHdoaWNoOgogKgogKgkxLglSZXRyaWVzIGZhaWxlZCByZWFkIG9wZXJhdGlvbnMgb24gd29ya2luZyBtaXJyb3JzLgogKgkyLglVcGRhdGVzIHRoZSByYWlkIHN1cGVyYmxvY2sgd2hlbiBwcm9ibGVtcyBlbmNvdW50ZXIuCiAqCTMuCVBlcmZvcm1zIHdyaXRlcyBmb2xsb3dpbmcgcmVhZHMgZm9yIGFycmF5IHN5bmNyb25pc2luZy4KICovCgpzdGF0aWMgdm9pZCBmaXhfcmVhZF9lcnJvcihjb25mX3QgKmNvbmYsIGludCByZWFkX2Rpc2ssCgkJCSAgIHNlY3Rvcl90IHNlY3QsIGludCBzZWN0b3JzKQp7CgltZGRldl90ICptZGRldiA9IGNvbmYtPm1kZGV2OwoJd2hpbGUoc2VjdG9ycykgewoJCWludCBzID0gc2VjdG9yczsKCQlpbnQgZCA9IHJlYWRfZGlzazsKCQlpbnQgc3VjY2VzcyA9IDA7CgkJaW50IHN0YXJ0OwoJCW1ka19yZGV2X3QgKnJkZXY7CgoJCWlmIChzID4gKFBBR0VfU0laRT4+OSkpCgkJCXMgPSBQQUdFX1NJWkUgPj4gOTsKCgkJZG8gewoJCQkvKiBOb3RlOiBubyByY3UgcHJvdGVjdGlvbiBuZWVkZWQgaGVyZQoJCQkgKiBhcyB0aGlzIGlzIHN5bmNocm9ub3VzIGluIHRoZSByYWlkMWQgdGhyZWFkCgkJCSAqIHdoaWNoIGlzIHRoZSB0aHJlYWQgdGhhdCBtaWdodCByZW1vdmUKCQkJICogYSBkZXZpY2UuICBJZiByYWlkMWQgZXZlciBiZWNvbWVzIG11bHRpLXRocmVhZGVkLi4uLgoJCQkgKi8KCQkJcmRldiA9IGNvbmYtPm1pcnJvcnNbZF0ucmRldjsKCQkJaWYgKHJkZXYgJiYKCQkJICAgIHRlc3RfYml0KEluX3N5bmMsICZyZGV2LT5mbGFncykgJiYKCQkJICAgIHN5bmNfcGFnZV9pbyhyZGV2LT5iZGV2LAoJCQkJCSBzZWN0ICsgcmRldi0+ZGF0YV9vZmZzZXQsCgkJCQkJIHM8PDksCgkJCQkJIGNvbmYtPnRtcHBhZ2UsIFJFQUQpKQoJCQkJc3VjY2VzcyA9IDE7CgkJCWVsc2UgewoJCQkJZCsrOwoJCQkJaWYgKGQgPT0gY29uZi0+cmFpZF9kaXNrcykKCQkJCQlkID0gMDsKCQkJfQoJCX0gd2hpbGUgKCFzdWNjZXNzICYmIGQgIT0gcmVhZF9kaXNrKTsKCgkJaWYgKCFzdWNjZXNzKSB7CgkJCS8qIENhbm5vdCByZWFkIGZyb20gYW55d2hlcmUgLS0gYnllIGJ5ZSBhcnJheSAqLwoJCQltZF9lcnJvcihtZGRldiwgY29uZi0+bWlycm9yc1tyZWFkX2Rpc2tdLnJkZXYpOwoJCQlicmVhazsKCQl9CgkJLyogd3JpdGUgaXQgYmFjayBhbmQgcmUtcmVhZCAqLwoJCXN0YXJ0ID0gZDsKCQl3aGlsZSAoZCAhPSByZWFkX2Rpc2spIHsKCQkJaWYgKGQ9PTApCgkJCQlkID0gY29uZi0+cmFpZF9kaXNrczsKCQkJZC0tOwoJCQlyZGV2ID0gY29uZi0+bWlycm9yc1tkXS5yZGV2OwoJCQlpZiAocmRldiAmJgoJCQkgICAgdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSkgewoJCQkJaWYgKHN5bmNfcGFnZV9pbyhyZGV2LT5iZGV2LAoJCQkJCQkgc2VjdCArIHJkZXYtPmRhdGFfb2Zmc2V0LAoJCQkJCQkgczw8OSwgY29uZi0+dG1wcGFnZSwgV1JJVEUpCgkJCQkgICAgPT0gMCkKCQkJCQkvKiBXZWxsLCB0aGlzIGRldmljZSBpcyBkZWFkICovCgkJCQkJbWRfZXJyb3IobWRkZXYsIHJkZXYpOwoJCQl9CgkJfQoJCWQgPSBzdGFydDsKCQl3aGlsZSAoZCAhPSByZWFkX2Rpc2spIHsKCQkJY2hhciBiW0JERVZOQU1FX1NJWkVdOwoJCQlpZiAoZD09MCkKCQkJCWQgPSBjb25mLT5yYWlkX2Rpc2tzOwoJCQlkLS07CgkJCXJkZXYgPSBjb25mLT5taXJyb3JzW2RdLnJkZXY7CgkJCWlmIChyZGV2ICYmCgkJCSAgICB0ZXN0X2JpdChJbl9zeW5jLCAmcmRldi0+ZmxhZ3MpKSB7CgkJCQlpZiAoc3luY19wYWdlX2lvKHJkZXYtPmJkZXYsCgkJCQkJCSBzZWN0ICsgcmRldi0+ZGF0YV9vZmZzZXQsCgkJCQkJCSBzPDw5LCBjb25mLT50bXBwYWdlLCBSRUFEKQoJCQkJICAgID09IDApCgkJCQkJLyogV2VsbCwgdGhpcyBkZXZpY2UgaXMgZGVhZCAqLwoJCQkJCW1kX2Vycm9yKG1kZGV2LCByZGV2KTsKCQkJCWVsc2UgewoJCQkJCWF0b21pY19hZGQocywgJnJkZXYtPmNvcnJlY3RlZF9lcnJvcnMpOwoJCQkJCXByaW50ayhLRVJOX0lORk8KCQkJCQkgICAgICAgInJhaWQxOiVzOiByZWFkIGVycm9yIGNvcnJlY3RlZCAiCgkJCQkJICAgICAgICIoJWQgc2VjdG9ycyBhdCAlbGx1IG9uICVzKVxuIiwKCQkJCQkgICAgICAgbWRuYW1lKG1kZGV2KSwgcywKCQkJCQkgICAgICAgKHVuc2lnbmVkIGxvbmcgbG9uZykoc2VjdCArCgkJCQkJICAgICAgICAgICByZGV2LT5kYXRhX29mZnNldCksCgkJCQkJICAgICAgIGJkZXZuYW1lKHJkZXYtPmJkZXYsIGIpKTsKCQkJCX0KCQkJfQoJCX0KCQlzZWN0b3JzIC09IHM7CgkJc2VjdCArPSBzOwoJfQp9CgpzdGF0aWMgdm9pZCByYWlkMWQobWRkZXZfdCAqbWRkZXYpCnsKCXIxYmlvX3QgKnIxX2JpbzsKCXN0cnVjdCBiaW8gKmJpbzsKCXVuc2lnbmVkIGxvbmcgZmxhZ3M7Cgljb25mX3QgKmNvbmYgPSBtZGRldl90b19jb25mKG1kZGV2KTsKCXN0cnVjdCBsaXN0X2hlYWQgKmhlYWQgPSAmY29uZi0+cmV0cnlfbGlzdDsKCWludCB1bnBsdWc9MDsKCW1ka19yZGV2X3QgKnJkZXY7CgoJbWRfY2hlY2tfcmVjb3ZlcnkobWRkZXYpOwoJCglmb3IgKDs7KSB7CgkJY2hhciBiW0JERVZOQU1FX1NJWkVdOwoJCXNwaW5fbG9ja19pcnFzYXZlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoKCQlpZiAoY29uZi0+cGVuZGluZ19iaW9fbGlzdC5oZWFkKSB7CgkJCWJpbyA9IGJpb19saXN0X2dldCgmY29uZi0+cGVuZGluZ19iaW9fbGlzdCk7CgkJCWJsa19yZW1vdmVfcGx1ZyhtZGRldi0+cXVldWUpOwoJCQlzcGluX3VubG9ja19pcnFyZXN0b3JlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoJCQkvKiBmbHVzaCBhbnkgcGVuZGluZyBiaXRtYXAgd3JpdGVzIHRvIGRpc2sgYmVmb3JlIHByb2NlZWRpbmcgdy8gSS9PICovCgkJCWlmIChiaXRtYXBfdW5wbHVnKG1kZGV2LT5iaXRtYXApICE9IDApCgkJCQlwcmludGsoIiVzOiBiaXRtYXAgZmlsZSB3cml0ZSBmYWlsZWQhXG4iLCBtZG5hbWUobWRkZXYpKTsKCgkJCXdoaWxlIChiaW8pIHsgLyogc3VibWl0IHBlbmRpbmcgd3JpdGVzICovCgkJCQlzdHJ1Y3QgYmlvICpuZXh0ID0gYmlvLT5iaV9uZXh0OwoJCQkJYmlvLT5iaV9uZXh0ID0gTlVMTDsKCQkJCWdlbmVyaWNfbWFrZV9yZXF1ZXN0KGJpbyk7CgkJCQliaW8gPSBuZXh0OwoJCQl9CgkJCXVucGx1ZyA9IDE7CgoJCQljb250aW51ZTsKCQl9CgoJCWlmIChsaXN0X2VtcHR5KGhlYWQpKQoJCQlicmVhazsKCQlyMV9iaW8gPSBsaXN0X2VudHJ5KGhlYWQtPnByZXYsIHIxYmlvX3QsIHJldHJ5X2xpc3QpOwoJCWxpc3RfZGVsKGhlYWQtPnByZXYpOwoJCWNvbmYtPm5yX3F1ZXVlZC0tOwoJCXNwaW5fdW5sb2NrX2lycXJlc3RvcmUoJmNvbmYtPmRldmljZV9sb2NrLCBmbGFncyk7CgoJCW1kZGV2ID0gcjFfYmlvLT5tZGRldjsKCQljb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CgkJaWYgKHRlc3RfYml0KFIxQklPX0lzU3luYywgJnIxX2Jpby0+c3RhdGUpKSB7CgkJCXN5bmNfcmVxdWVzdF93cml0ZShtZGRldiwgcjFfYmlvKTsKCQkJdW5wbHVnID0gMTsKCQl9IGVsc2UgaWYgKHRlc3RfYml0KFIxQklPX0JhcnJpZXJSZXRyeSwgJnIxX2Jpby0+c3RhdGUpKSB7CgkJCS8qIHNvbWUgcmVxdWVzdHMgaW4gdGhlIHIxYmlvIHdlcmUgQklPX1JXX0JBUlJJRVIKCQkJICogcmVxdWVzdHMgd2hpY2ggZmFpbGVkIHdpdGggLUVPUE5PVFNVUFAuICBIb2h1bW0uLgoJCQkgKiBCZXR0ZXIgcmVzdWJtaXQgd2l0aG91dCB0aGUgYmFycmllci4KCQkJICogV2Uga25vdyB3aGljaCBkZXZpY2VzIHRvIHJlc3VibWl0IGZvciwgYmVjYXVzZQoJCQkgKiBhbGwgb3RoZXJzIGhhdmUgaGFkIHRoZWlyIGJpb3NbXSBlbnRyeSBjbGVhcmVkLgoJCQkgKiBXZSBhbHJlYWR5IGhhdmUgYSBucl9wZW5kaW5nIHJlZmVyZW5jZSBvbiB0aGVzZSByZGV2cy4KCQkJICovCgkJCWludCBpOwoJCQljb25zdCBpbnQgZG9fc3luYyA9IGJpb19zeW5jKHIxX2Jpby0+bWFzdGVyX2Jpbyk7CgkJCWNsZWFyX2JpdChSMUJJT19CYXJyaWVyUmV0cnksICZyMV9iaW8tPnN0YXRlKTsKCQkJY2xlYXJfYml0KFIxQklPX0JhcnJpZXIsICZyMV9iaW8tPnN0YXRlKTsKCQkJZm9yIChpPTA7IGkgPCBjb25mLT5yYWlkX2Rpc2tzOyBpKyspCgkJCQlpZiAocjFfYmlvLT5iaW9zW2ldKQoJCQkJCWF0b21pY19pbmMoJnIxX2Jpby0+cmVtYWluaW5nKTsKCQkJZm9yIChpPTA7IGkgPCBjb25mLT5yYWlkX2Rpc2tzOyBpKyspCgkJCQlpZiAocjFfYmlvLT5iaW9zW2ldKSB7CgkJCQkJc3RydWN0IGJpb192ZWMgKmJ2ZWM7CgkJCQkJaW50IGo7CgoJCQkJCWJpbyA9IGJpb19jbG9uZShyMV9iaW8tPm1hc3Rlcl9iaW8sIEdGUF9OT0lPKTsKCQkJCQkvKiBjb3B5IHBhZ2VzIGZyb20gdGhlIGZhaWxlZCBiaW8sIGFzCgkJCQkJICogdGhpcyBtaWdodCBiZSBhIHdyaXRlLWJlaGluZCBkZXZpY2UgKi8KCQkJCQlfX2Jpb19mb3JfZWFjaF9zZWdtZW50KGJ2ZWMsIGJpbywgaiwgMCkKCQkJCQkJYnZlYy0+YnZfcGFnZSA9IGJpb19pb3ZlY19pZHgocjFfYmlvLT5iaW9zW2ldLCBqKS0+YnZfcGFnZTsKCQkJCQliaW9fcHV0KHIxX2Jpby0+Ymlvc1tpXSk7CgkJCQkJYmlvLT5iaV9zZWN0b3IgPSByMV9iaW8tPnNlY3RvciArCgkJCQkJCWNvbmYtPm1pcnJvcnNbaV0ucmRldi0+ZGF0YV9vZmZzZXQ7CgkJCQkJYmlvLT5iaV9iZGV2ID0gY29uZi0+bWlycm9yc1tpXS5yZGV2LT5iZGV2OwoJCQkJCWJpby0+YmlfZW5kX2lvID0gcmFpZDFfZW5kX3dyaXRlX3JlcXVlc3Q7CgkJCQkJYmlvLT5iaV9ydyA9IFdSSVRFIHwgZG9fc3luYzsKCQkJCQliaW8tPmJpX3ByaXZhdGUgPSByMV9iaW87CgkJCQkJcjFfYmlvLT5iaW9zW2ldID0gYmlvOwoJCQkJCWdlbmVyaWNfbWFrZV9yZXF1ZXN0KGJpbyk7CgkJCQl9CgkJfSBlbHNlIHsKCQkJaW50IGRpc2s7CgoJCQkvKiB3ZSBnb3QgYSByZWFkIGVycm9yLiBNYXliZSB0aGUgZHJpdmUgaXMgYmFkLiAgTWF5YmUganVzdAoJCQkgKiB0aGUgYmxvY2sgYW5kIHdlIGNhbiBmaXggaXQuCgkJCSAqIFdlIGZyZWV6ZSBhbGwgb3RoZXIgSU8sIGFuZCB0cnkgcmVhZGluZyB0aGUgYmxvY2sgZnJvbQoJCQkgKiBvdGhlciBkZXZpY2VzLiAgV2hlbiB3ZSBmaW5kIG9uZSwgd2UgcmUtd3JpdGUKCQkJICogYW5kIGNoZWNrIGl0IHRoYXQgZml4ZXMgdGhlIHJlYWQgZXJyb3IuCgkJCSAqIFRoaXMgaXMgYWxsIGRvbmUgc3luY2hyb25vdXNseSB3aGlsZSB0aGUgYXJyYXkgaXMKCQkJICogZnJvemVuCgkJCSAqLwoJCQlpZiAobWRkZXYtPnJvID09IDApIHsKCQkJCWZyZWV6ZV9hcnJheShjb25mKTsKCQkJCWZpeF9yZWFkX2Vycm9yKGNvbmYsIHIxX2Jpby0+cmVhZF9kaXNrLAoJCQkJCSAgICAgICByMV9iaW8tPnNlY3RvciwKCQkJCQkgICAgICAgcjFfYmlvLT5zZWN0b3JzKTsKCQkJCXVuZnJlZXplX2FycmF5KGNvbmYpOwoJCQl9CgoJCQliaW8gPSByMV9iaW8tPmJpb3NbcjFfYmlvLT5yZWFkX2Rpc2tdOwoJCQlpZiAoKGRpc2s9cmVhZF9iYWxhbmNlKGNvbmYsIHIxX2JpbykpID09IC0xKSB7CgkJCQlwcmludGsoS0VSTl9BTEVSVCAicmFpZDE6ICVzOiB1bnJlY292ZXJhYmxlIEkvTyIKCQkJCSAgICAgICAiIHJlYWQgZXJyb3IgZm9yIGJsb2NrICVsbHVcbiIsCgkJCQkgICAgICAgYmRldm5hbWUoYmlvLT5iaV9iZGV2LGIpLAoJCQkJICAgICAgICh1bnNpZ25lZCBsb25nIGxvbmcpcjFfYmlvLT5zZWN0b3IpOwoJCQkJcmFpZF9lbmRfYmlvX2lvKHIxX2Jpbyk7CgkJCX0gZWxzZSB7CgkJCQljb25zdCBpbnQgZG9fc3luYyA9IGJpb19zeW5jKHIxX2Jpby0+bWFzdGVyX2Jpbyk7CgkJCQlyMV9iaW8tPmJpb3NbcjFfYmlvLT5yZWFkX2Rpc2tdID0KCQkJCQltZGRldi0+cm8gPyBJT19CTE9DS0VEIDogTlVMTDsKCQkJCXIxX2Jpby0+cmVhZF9kaXNrID0gZGlzazsKCQkJCWJpb19wdXQoYmlvKTsKCQkJCWJpbyA9IGJpb19jbG9uZShyMV9iaW8tPm1hc3Rlcl9iaW8sIEdGUF9OT0lPKTsKCQkJCXIxX2Jpby0+Ymlvc1tyMV9iaW8tPnJlYWRfZGlza10gPSBiaW87CgkJCQlyZGV2ID0gY29uZi0+bWlycm9yc1tkaXNrXS5yZGV2OwoJCQkJaWYgKHByaW50a19yYXRlbGltaXQoKSkKCQkJCQlwcmludGsoS0VSTl9FUlIgInJhaWQxOiAlczogcmVkaXJlY3Rpbmcgc2VjdG9yICVsbHUgdG8iCgkJCQkJICAgICAgICIgYW5vdGhlciBtaXJyb3JcbiIsCgkJCQkJICAgICAgIGJkZXZuYW1lKHJkZXYtPmJkZXYsYiksCgkJCQkJICAgICAgICh1bnNpZ25lZCBsb25nIGxvbmcpcjFfYmlvLT5zZWN0b3IpOwoJCQkJYmlvLT5iaV9zZWN0b3IgPSByMV9iaW8tPnNlY3RvciArIHJkZXYtPmRhdGFfb2Zmc2V0OwoJCQkJYmlvLT5iaV9iZGV2ID0gcmRldi0+YmRldjsKCQkJCWJpby0+YmlfZW5kX2lvID0gcmFpZDFfZW5kX3JlYWRfcmVxdWVzdDsKCQkJCWJpby0+YmlfcncgPSBSRUFEIHwgZG9fc3luYzsKCQkJCWJpby0+YmlfcHJpdmF0ZSA9IHIxX2JpbzsKCQkJCXVucGx1ZyA9IDE7CgkJCQlnZW5lcmljX21ha2VfcmVxdWVzdChiaW8pOwoJCQl9CgkJfQoJfQoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmY29uZi0+ZGV2aWNlX2xvY2ssIGZsYWdzKTsKCWlmICh1bnBsdWcpCgkJdW5wbHVnX3NsYXZlcyhtZGRldik7Cn0KCgpzdGF0aWMgaW50IGluaXRfcmVzeW5jKGNvbmZfdCAqY29uZikKewoJaW50IGJ1ZmZzOwoKCWJ1ZmZzID0gUkVTWU5DX1dJTkRPVyAvIFJFU1lOQ19CTE9DS19TSVpFOwoJQlVHX09OKGNvbmYtPnIxYnVmX3Bvb2wpOwoJY29uZi0+cjFidWZfcG9vbCA9IG1lbXBvb2xfY3JlYXRlKGJ1ZmZzLCByMWJ1Zl9wb29sX2FsbG9jLCByMWJ1Zl9wb29sX2ZyZWUsCgkJCQkJICBjb25mLT5wb29saW5mbyk7CglpZiAoIWNvbmYtPnIxYnVmX3Bvb2wpCgkJcmV0dXJuIC1FTk9NRU07Cgljb25mLT5uZXh0X3Jlc3luYyA9IDA7CglyZXR1cm4gMDsKfQoKLyoKICogcGVyZm9ybSBhICJzeW5jIiBvbiBvbmUgImJsb2NrIgogKgogKiBXZSBuZWVkIHRvIG1ha2Ugc3VyZSB0aGF0IG5vIG5vcm1hbCBJL08gcmVxdWVzdCAtIHBhcnRpY3VsYXJseSB3cml0ZQogKiByZXF1ZXN0cyAtIGNvbmZsaWN0IHdpdGggYWN0aXZlIHN5bmMgcmVxdWVzdHMuCiAqCiAqIFRoaXMgaXMgYWNoaWV2ZWQgYnkgdHJhY2tpbmcgcGVuZGluZyByZXF1ZXN0cyBhbmQgYSAnYmFycmllcicgY29uY2VwdAogKiB0aGF0IGNhbiBiZSBpbnN0YWxsZWQgdG8gZXhjbHVkZSBub3JtYWwgSU8gcmVxdWVzdHMuCiAqLwoKc3RhdGljIHNlY3Rvcl90IHN5bmNfcmVxdWVzdChtZGRldl90ICptZGRldiwgc2VjdG9yX3Qgc2VjdG9yX25yLCBpbnQgKnNraXBwZWQsIGludCBnb19mYXN0ZXIpCnsKCWNvbmZfdCAqY29uZiA9IG1kZGV2X3RvX2NvbmYobWRkZXYpOwoJcjFiaW9fdCAqcjFfYmlvOwoJc3RydWN0IGJpbyAqYmlvOwoJc2VjdG9yX3QgbWF4X3NlY3RvciwgbnJfc2VjdG9yczsKCWludCBkaXNrID0gLTE7CglpbnQgaTsKCWludCB3b25seSA9IC0xOwoJaW50IHdyaXRlX3RhcmdldHMgPSAwLCByZWFkX3RhcmdldHMgPSAwOwoJaW50IHN5bmNfYmxvY2tzOwoJaW50IHN0aWxsX2RlZ3JhZGVkID0gMDsKCglpZiAoIWNvbmYtPnIxYnVmX3Bvb2wpCgl7Ci8qCgkJcHJpbnRrKCJzeW5jIHN0YXJ0IC0gYml0bWFwICVwXG4iLCBtZGRldi0+Yml0bWFwKTsKKi8KCQlpZiAoaW5pdF9yZXN5bmMoY29uZikpCgkJCXJldHVybiAwOwoJfQoKCW1heF9zZWN0b3IgPSBtZGRldi0+c2l6ZSA8PCAxOwoJaWYgKHNlY3Rvcl9uciA+PSBtYXhfc2VjdG9yKSB7CgkJLyogSWYgd2UgYWJvcnRlZCwgd2UgbmVlZCB0byBhYm9ydCB0aGUKCQkgKiBzeW5jIG9uIHRoZSAnY3VycmVudCcgYml0bWFwIGNodW5rICh0aGVyZSB3aWxsCgkJICogb25seSBiZSBvbmUgaW4gcmFpZDEgcmVzeW5jLgoJCSAqIFdlIGNhbiBmaW5kIHRoZSBjdXJyZW50IGFkZGVzcyBpbiBtZGRldi0+Y3Vycl9yZXN5bmMKCQkgKi8KCQlpZiAobWRkZXYtPmN1cnJfcmVzeW5jIDwgbWF4X3NlY3RvcikgLyogYWJvcnRlZCAqLwoJCQliaXRtYXBfZW5kX3N5bmMobWRkZXYtPmJpdG1hcCwgbWRkZXYtPmN1cnJfcmVzeW5jLAoJCQkJCQkmc3luY19ibG9ja3MsIDEpOwoJCWVsc2UgLyogY29tcGxldGVkIHN5bmMgKi8KCQkJY29uZi0+ZnVsbHN5bmMgPSAwOwoKCQliaXRtYXBfY2xvc2Vfc3luYyhtZGRldi0+Yml0bWFwKTsKCQljbG9zZV9zeW5jKGNvbmYpOwoJCXJldHVybiAwOwoJfQoKCWlmIChtZGRldi0+Yml0bWFwID09IE5VTEwgJiYKCSAgICBtZGRldi0+cmVjb3ZlcnlfY3AgPT0gTWF4U2VjdG9yICYmCgkgICAgIXRlc3RfYml0KE1EX1JFQ09WRVJZX1JFUVVFU1RFRCwgJm1kZGV2LT5yZWNvdmVyeSkgJiYKCSAgICBjb25mLT5mdWxsc3luYyA9PSAwKSB7CgkJKnNraXBwZWQgPSAxOwoJCXJldHVybiBtYXhfc2VjdG9yIC0gc2VjdG9yX25yOwoJfQoJLyogYmVmb3JlIGJ1aWxkaW5nIGEgcmVxdWVzdCwgY2hlY2sgaWYgd2UgY2FuIHNraXAgdGhlc2UgYmxvY2tzLi4KCSAqIFRoaXMgY2FsbCB0aGUgYml0bWFwX3N0YXJ0X3N5bmMgZG9lc24ndCBhY3R1YWxseSByZWNvcmQgYW55dGhpbmcKCSAqLwoJaWYgKCFiaXRtYXBfc3RhcnRfc3luYyhtZGRldi0+Yml0bWFwLCBzZWN0b3JfbnIsICZzeW5jX2Jsb2NrcywgMSkgJiYKCSAgICAhY29uZi0+ZnVsbHN5bmMgJiYgIXRlc3RfYml0KE1EX1JFQ09WRVJZX1JFUVVFU1RFRCwgJm1kZGV2LT5yZWNvdmVyeSkpIHsKCQkvKiBXZSBjYW4gc2tpcCB0aGlzIGJsb2NrLCBhbmQgcHJvYmFibHkgc2V2ZXJhbCBtb3JlICovCgkJKnNraXBwZWQgPSAxOwoJCXJldHVybiBzeW5jX2Jsb2NrczsKCX0KCS8qCgkgKiBJZiB0aGVyZSBpcyBub24tcmVzeW5jIGFjdGl2aXR5IHdhaXRpbmcgZm9yIGEgdHVybiwKCSAqIGFuZCByZXN5bmMgaXMgZ29pbmcgZmFzdCBlbm91Z2gsCgkgKiB0aGVuIGxldCBpdCB0aG91Z2ggYmVmb3JlIHN0YXJ0aW5nIG9uIHRoaXMgbmV3IHN5bmMgcmVxdWVzdC4KCSAqLwoJaWYgKCFnb19mYXN0ZXIgJiYgY29uZi0+bnJfd2FpdGluZykKCQltc2xlZXBfaW50ZXJydXB0aWJsZSgxMDAwKTsKCglyYWlzZV9iYXJyaWVyKGNvbmYpOwoKCWNvbmYtPm5leHRfcmVzeW5jID0gc2VjdG9yX25yOwoKCXIxX2JpbyA9IG1lbXBvb2xfYWxsb2MoY29uZi0+cjFidWZfcG9vbCwgR0ZQX05PSU8pOwoJcmN1X3JlYWRfbG9jaygpOwoJLyoKCSAqIElmIHdlIGdldCBhIGNvcnJlY3RhYmx5IHJlYWQgZXJyb3IgZHVyaW5nIHJlc3luYyBvciByZWNvdmVyeSwKCSAqIHdlIG1pZ2h0IHdhbnQgdG8gcmVhZCBmcm9tIGEgZGlmZmVyZW50IGRldmljZS4gIFNvIHdlCgkgKiBmbGFnIGFsbCBkcml2ZXMgdGhhdCBjb3VsZCBjb25jZWl2YWJseSBiZSByZWFkIGZyb20gZm9yIFJFQUQsCgkgKiBhbmQgYW55IG90aGVycyAod2hpY2ggd2lsbCBiZSBub24tSW5fc3luYyBkZXZpY2VzKSBmb3IgV1JJVEUuCgkgKiBJZiBhIHJlYWQgZmFpbHMsIHdlIHRyeSByZWFkaW5nIGZyb20gc29tZXRoaW5nIGVsc2UgZm9yIHdoaWNoIFJFQUQKCSAqIGlzIE9LLgoJICovCgoJcjFfYmlvLT5tZGRldiA9IG1kZGV2OwoJcjFfYmlvLT5zZWN0b3IgPSBzZWN0b3JfbnI7CglyMV9iaW8tPnN0YXRlID0gMDsKCXNldF9iaXQoUjFCSU9fSXNTeW5jLCAmcjFfYmlvLT5zdGF0ZSk7CgoJZm9yIChpPTA7IGkgPCBjb25mLT5yYWlkX2Rpc2tzOyBpKyspIHsKCQltZGtfcmRldl90ICpyZGV2OwoJCWJpbyA9IHIxX2Jpby0+Ymlvc1tpXTsKCgkJLyogdGFrZSBmcm9tIGJpb19pbml0ICovCgkJYmlvLT5iaV9uZXh0ID0gTlVMTDsKCQliaW8tPmJpX2ZsYWdzIHw9IDEgPDwgQklPX1VQVE9EQVRFOwoJCWJpby0+YmlfcncgPSBSRUFEOwoJCWJpby0+YmlfdmNudCA9IDA7CgkJYmlvLT5iaV9pZHggPSAwOwoJCWJpby0+YmlfcGh5c19zZWdtZW50cyA9IDA7CgkJYmlvLT5iaV9od19zZWdtZW50cyA9IDA7CgkJYmlvLT5iaV9zaXplID0gMDsKCQliaW8tPmJpX2VuZF9pbyA9IE5VTEw7CgkJYmlvLT5iaV9wcml2YXRlID0gTlVMTDsKCgkJcmRldiA9IHJjdV9kZXJlZmVyZW5jZShjb25mLT5taXJyb3JzW2ldLnJkZXYpOwoJCWlmIChyZGV2ID09IE5VTEwgfHwKCQkJICAgdGVzdF9iaXQoRmF1bHR5LCAmcmRldi0+ZmxhZ3MpKSB7CgkJCXN0aWxsX2RlZ3JhZGVkID0gMTsKCQkJY29udGludWU7CgkJfSBlbHNlIGlmICghdGVzdF9iaXQoSW5fc3luYywgJnJkZXYtPmZsYWdzKSkgewoJCQliaW8tPmJpX3J3ID0gV1JJVEU7CgkJCWJpby0+YmlfZW5kX2lvID0gZW5kX3N5bmNfd3JpdGU7CgkJCXdyaXRlX3RhcmdldHMgKys7CgkJfSBlbHNlIHsKCQkJLyogbWF5IG5lZWQgdG8gcmVhZCBmcm9tIGhlcmUgKi8KCQkJYmlvLT5iaV9ydyA9IFJFQUQ7CgkJCWJpby0+YmlfZW5kX2lvID0gZW5kX3N5bmNfcmVhZDsKCQkJaWYgKHRlc3RfYml0KFdyaXRlTW9zdGx5LCAmcmRldi0+ZmxhZ3MpKSB7CgkJCQlpZiAod29ubHkgPCAwKQoJCQkJCXdvbmx5ID0gaTsKCQkJfSBlbHNlIHsKCQkJCWlmIChkaXNrIDwgMCkKCQkJCQlkaXNrID0gaTsKCQkJfQoJCQlyZWFkX3RhcmdldHMrKzsKCQl9CgkJYXRvbWljX2luYygmcmRldi0+bnJfcGVuZGluZyk7CgkJYmlvLT5iaV9zZWN0b3IgPSBzZWN0b3JfbnIgKyByZGV2LT5kYXRhX29mZnNldDsKCQliaW8tPmJpX2JkZXYgPSByZGV2LT5iZGV2OwoJCWJpby0+YmlfcHJpdmF0ZSA9IHIxX2JpbzsKCX0KCXJjdV9yZWFkX3VubG9jaygpOwoJaWYgKGRpc2sgPCAwKQoJCWRpc2sgPSB3b25seTsKCXIxX2Jpby0+cmVhZF9kaXNrID0gZGlzazsKCglpZiAodGVzdF9iaXQoTURfUkVDT1ZFUllfU1lOQywgJm1kZGV2LT5yZWNvdmVyeSkgJiYgcmVhZF90YXJnZXRzID4gMCkKCQkvKiBleHRyYSByZWFkIHRhcmdldHMgYXJlIGFsc28gd3JpdGUgdGFyZ2V0cyAqLwoJCXdyaXRlX3RhcmdldHMgKz0gcmVhZF90YXJnZXRzLTE7CgoJaWYgKHdyaXRlX3RhcmdldHMgPT0gMCB8fCByZWFkX3RhcmdldHMgPT0gMCkgewoJCS8qIFRoZXJlIGlzIG5vd2hlcmUgdG8gd3JpdGUsIHNvIGFsbCBub24tc3luYwoJCSAqIGRyaXZlcyBtdXN0IGJlIGZhaWxlZCAtIHNvIHdlIGFyZSBmaW5pc2hlZAoJCSAqLwoJCXNlY3Rvcl90IHJ2ID0gbWF4X3NlY3RvciAtIHNlY3Rvcl9ucjsKCQkqc2tpcHBlZCA9IDE7CgkJcHV0X2J1ZihyMV9iaW8pOwoJCXJldHVybiBydjsKCX0KCglucl9zZWN0b3JzID0gMDsKCXN5bmNfYmxvY2tzID0gMDsKCWRvIHsKCQlzdHJ1Y3QgcGFnZSAqcGFnZTsKCQlpbnQgbGVuID0gUEFHRV9TSVpFOwoJCWlmIChzZWN0b3JfbnIgKyAobGVuPj45KSA+IG1heF9zZWN0b3IpCgkJCWxlbiA9IChtYXhfc2VjdG9yIC0gc2VjdG9yX25yKSA8PCA5OwoJCWlmIChsZW4gPT0gMCkKCQkJYnJlYWs7CgkJaWYgKHN5bmNfYmxvY2tzID09IDApIHsKCQkJaWYgKCFiaXRtYXBfc3RhcnRfc3luYyhtZGRldi0+Yml0bWFwLCBzZWN0b3JfbnIsCgkJCQkJICAgICAgICZzeW5jX2Jsb2Nrcywgc3RpbGxfZGVncmFkZWQpICYmCgkJCSAgICAhY29uZi0+ZnVsbHN5bmMgJiYKCQkJICAgICF0ZXN0X2JpdChNRF9SRUNPVkVSWV9SRVFVRVNURUQsICZtZGRldi0+cmVjb3ZlcnkpKQoJCQkJYnJlYWs7CgkJCUJVR19PTihzeW5jX2Jsb2NrcyA8IChQQUdFX1NJWkU+PjkpKTsKCQkJaWYgKGxlbiA+IChzeW5jX2Jsb2Nrczw8OSkpCgkJCQlsZW4gPSBzeW5jX2Jsb2Nrczw8OTsKCQl9CgoJCWZvciAoaT0wIDsgaSA8IGNvbmYtPnJhaWRfZGlza3M7IGkrKykgewoJCQliaW8gPSByMV9iaW8tPmJpb3NbaV07CgkJCWlmIChiaW8tPmJpX2VuZF9pbykgewoJCQkJcGFnZSA9IGJpby0+YmlfaW9fdmVjW2Jpby0+YmlfdmNudF0uYnZfcGFnZTsKCQkJCWlmIChiaW9fYWRkX3BhZ2UoYmlvLCBwYWdlLCBsZW4sIDApID09IDApIHsKCQkJCQkvKiBzdG9wIGhlcmUgKi8KCQkJCQliaW8tPmJpX2lvX3ZlY1tiaW8tPmJpX3ZjbnRdLmJ2X3BhZ2UgPSBwYWdlOwoJCQkJCXdoaWxlIChpID4gMCkgewoJCQkJCQlpLS07CgkJCQkJCWJpbyA9IHIxX2Jpby0+Ymlvc1tpXTsKCQkJCQkJaWYgKGJpby0+YmlfZW5kX2lvPT1OVUxMKQoJCQkJCQkJY29udGludWU7CgkJCQkJCS8qIHJlbW92ZSBsYXN0IHBhZ2UgZnJvbSB0aGlzIGJpbyAqLwoJCQkJCQliaW8tPmJpX3ZjbnQtLTsKCQkJCQkJYmlvLT5iaV9zaXplIC09IGxlbjsKCQkJCQkJYmlvLT5iaV9mbGFncyAmPSB+KDE8PCBCSU9fU0VHX1ZBTElEKTsKCQkJCQl9CgkJCQkJZ290byBiaW9fZnVsbDsKCQkJCX0KCQkJfQoJCX0KCQlucl9zZWN0b3JzICs9IGxlbj4+OTsKCQlzZWN0b3JfbnIgKz0gbGVuPj45OwoJCXN5bmNfYmxvY2tzIC09IChsZW4+PjkpOwoJfSB3aGlsZSAocjFfYmlvLT5iaW9zW2Rpc2tdLT5iaV92Y250IDwgUkVTWU5DX1BBR0VTKTsKIGJpb19mdWxsOgoJcjFfYmlvLT5zZWN0b3JzID0gbnJfc2VjdG9yczsKCgkvKiBGb3IgYSB1c2VyLXJlcXVlc3RlZCBzeW5jLCB3ZSByZWFkIGFsbCByZWFkYWJsZSBkZXZpY2VzIGFuZCBkbyBhCgkgKiBjb21wYXJlCgkgKi8KCWlmICh0ZXN0X2JpdChNRF9SRUNPVkVSWV9SRVFVRVNURUQsICZtZGRldi0+cmVjb3ZlcnkpKSB7CgkJYXRvbWljX3NldCgmcjFfYmlvLT5yZW1haW5pbmcsIHJlYWRfdGFyZ2V0cyk7CgkJZm9yIChpPTA7IGk8Y29uZi0+cmFpZF9kaXNrczsgaSsrKSB7CgkJCWJpbyA9IHIxX2Jpby0+Ymlvc1tpXTsKCQkJaWYgKGJpby0+YmlfZW5kX2lvID09IGVuZF9zeW5jX3JlYWQpIHsKCQkJCW1kX3N5bmNfYWNjdChiaW8tPmJpX2JkZXYsIG5yX3NlY3RvcnMpOwoJCQkJZ2VuZXJpY19tYWtlX3JlcXVlc3QoYmlvKTsKCQkJfQoJCX0KCX0gZWxzZSB7CgkJYXRvbWljX3NldCgmcjFfYmlvLT5yZW1haW5pbmcsIDEpOwoJCWJpbyA9IHIxX2Jpby0+Ymlvc1tyMV9iaW8tPnJlYWRfZGlza107CgkJbWRfc3luY19hY2N0KGJpby0+YmlfYmRldiwgbnJfc2VjdG9ycyk7CgkJZ2VuZXJpY19tYWtlX3JlcXVlc3QoYmlvKTsKCgl9CglyZXR1cm4gbnJfc2VjdG9yczsKfQoKc3RhdGljIGludCBydW4obWRkZXZfdCAqbWRkZXYpCnsKCWNvbmZfdCAqY29uZjsKCWludCBpLCBqLCBkaXNrX2lkeDsKCW1pcnJvcl9pbmZvX3QgKmRpc2s7CgltZGtfcmRldl90ICpyZGV2OwoJc3RydWN0IGxpc3RfaGVhZCAqdG1wOwoKCWlmIChtZGRldi0+bGV2ZWwgIT0gMSkgewoJCXByaW50aygicmFpZDE6ICVzOiByYWlkIGxldmVsIG5vdCBzZXQgdG8gbWlycm9yaW5nICglZClcbiIsCgkJICAgICAgIG1kbmFtZShtZGRldiksIG1kZGV2LT5sZXZlbCk7CgkJZ290byBvdXQ7Cgl9CglpZiAobWRkZXYtPnJlc2hhcGVfcG9zaXRpb24gIT0gTWF4U2VjdG9yKSB7CgkJcHJpbnRrKCJyYWlkMTogJXM6IHJlc2hhcGVfcG9zaXRpb24gc2V0IGJ1dCBub3Qgc3VwcG9ydGVkXG4iLAoJCSAgICAgICBtZG5hbWUobWRkZXYpKTsKCQlnb3RvIG91dDsKCX0KCS8qCgkgKiBjb3B5IHRoZSBhbHJlYWR5IHZlcmlmaWVkIGRldmljZXMgaW50byBvdXIgcHJpdmF0ZSBSQUlEMQoJICogYm9va2tlZXBpbmcgYXJlYS4gW3doYXRldmVyIHdlIGFsbG9jYXRlIGluIHJ1bigpLAoJICogc2hvdWxkIGJlIGZyZWVkIGluIHN0b3AoKV0KCSAqLwoJY29uZiA9IGt6YWxsb2Moc2l6ZW9mKGNvbmZfdCksIEdGUF9LRVJORUwpOwoJbWRkZXYtPnByaXZhdGUgPSBjb25mOwoJaWYgKCFjb25mKQoJCWdvdG8gb3V0X25vX21lbTsKCgljb25mLT5taXJyb3JzID0ga3phbGxvYyhzaXplb2Yoc3RydWN0IG1pcnJvcl9pbmZvKSptZGRldi0+cmFpZF9kaXNrcywKCQkJCSBHRlBfS0VSTkVMKTsKCWlmICghY29uZi0+bWlycm9ycykKCQlnb3RvIG91dF9ub19tZW07CgoJY29uZi0+dG1wcGFnZSA9IGFsbG9jX3BhZ2UoR0ZQX0tFUk5FTCk7CglpZiAoIWNvbmYtPnRtcHBhZ2UpCgkJZ290byBvdXRfbm9fbWVtOwoKCWNvbmYtPnBvb2xpbmZvID0ga21hbGxvYyhzaXplb2YoKmNvbmYtPnBvb2xpbmZvKSwgR0ZQX0tFUk5FTCk7CglpZiAoIWNvbmYtPnBvb2xpbmZvKQoJCWdvdG8gb3V0X25vX21lbTsKCWNvbmYtPnBvb2xpbmZvLT5tZGRldiA9IG1kZGV2OwoJY29uZi0+cG9vbGluZm8tPnJhaWRfZGlza3MgPSBtZGRldi0+cmFpZF9kaXNrczsKCWNvbmYtPnIxYmlvX3Bvb2wgPSBtZW1wb29sX2NyZWF0ZShOUl9SQUlEMV9CSU9TLCByMWJpb19wb29sX2FsbG9jLAoJCQkJCSAgcjFiaW9fcG9vbF9mcmVlLAoJCQkJCSAgY29uZi0+cG9vbGluZm8pOwoJaWYgKCFjb25mLT5yMWJpb19wb29sKQoJCWdvdG8gb3V0X25vX21lbTsKCglJVEVSQVRFX1JERVYobWRkZXYsIHJkZXYsIHRtcCkgewoJCWRpc2tfaWR4ID0gcmRldi0+cmFpZF9kaXNrOwoJCWlmIChkaXNrX2lkeCA+PSBtZGRldi0+cmFpZF9kaXNrcwoJCSAgICB8fCBkaXNrX2lkeCA8IDApCgkJCWNvbnRpbnVlOwoJCWRpc2sgPSBjb25mLT5taXJyb3JzICsgZGlza19pZHg7CgoJCWRpc2stPnJkZXYgPSByZGV2OwoKCQlibGtfcXVldWVfc3RhY2tfbGltaXRzKG1kZGV2LT5xdWV1ZSwKCQkJCSAgICAgICByZGV2LT5iZGV2LT5iZF9kaXNrLT5xdWV1ZSk7CgkJLyogYXMgd2UgZG9uJ3QgaG9ub3VyIG1lcmdlX2J2ZWNfZm4sIHdlIG11c3QgbmV2ZXIgcmlzawoJCSAqIHZpb2xhdGluZyBpdCwgc28gbGltaXQgLT5tYXhfc2VjdG9yIHRvIG9uZSBQQUdFLCBhcwoJCSAqIGEgb25lIHBhZ2UgcmVxdWVzdCBpcyBuZXZlciBpbiB2aW9sYXRpb24uCgkJICovCgkJaWYgKHJkZXYtPmJkZXYtPmJkX2Rpc2stPnF1ZXVlLT5tZXJnZV9idmVjX2ZuICYmCgkJICAgIG1kZGV2LT5xdWV1ZS0+bWF4X3NlY3RvcnMgPiAoUEFHRV9TSVpFPj45KSkKCQkJYmxrX3F1ZXVlX21heF9zZWN0b3JzKG1kZGV2LT5xdWV1ZSwgUEFHRV9TSVpFPj45KTsKCgkJZGlzay0+aGVhZF9wb3NpdGlvbiA9IDA7Cgl9Cgljb25mLT5yYWlkX2Rpc2tzID0gbWRkZXYtPnJhaWRfZGlza3M7Cgljb25mLT5tZGRldiA9IG1kZGV2OwoJc3Bpbl9sb2NrX2luaXQoJmNvbmYtPmRldmljZV9sb2NrKTsKCUlOSVRfTElTVF9IRUFEKCZjb25mLT5yZXRyeV9saXN0KTsKCglzcGluX2xvY2tfaW5pdCgmY29uZi0+cmVzeW5jX2xvY2spOwoJaW5pdF93YWl0cXVldWVfaGVhZCgmY29uZi0+d2FpdF9iYXJyaWVyKTsKCgliaW9fbGlzdF9pbml0KCZjb25mLT5wZW5kaW5nX2Jpb19saXN0KTsKCWJpb19saXN0X2luaXQoJmNvbmYtPmZsdXNoaW5nX2Jpb19saXN0KTsKCgoJbWRkZXYtPmRlZ3JhZGVkID0gMDsKCWZvciAoaSA9IDA7IGkgPCBjb25mLT5yYWlkX2Rpc2tzOyBpKyspIHsKCgkJZGlzayA9IGNvbmYtPm1pcnJvcnMgKyBpOwoKCQlpZiAoIWRpc2stPnJkZXYgfHwKCQkgICAgIXRlc3RfYml0KEluX3N5bmMsICZkaXNrLT5yZGV2LT5mbGFncykpIHsKCQkJZGlzay0+aGVhZF9wb3NpdGlvbiA9IDA7CgkJCW1kZGV2LT5kZWdyYWRlZCsrOwoJCQljb25mLT5mdWxsc3luYyA9IDE7CgkJfQoJfQoJaWYgKG1kZGV2LT5kZWdyYWRlZCA9PSBjb25mLT5yYWlkX2Rpc2tzKSB7CgkJcHJpbnRrKEtFUk5fRVJSICJyYWlkMTogbm8gb3BlcmF0aW9uYWwgbWlycm9ycyBmb3IgJXNcbiIsCgkJCW1kbmFtZShtZGRldikpOwoJCWdvdG8gb3V0X2ZyZWVfY29uZjsKCX0KCWlmIChjb25mLT5yYWlkX2Rpc2tzIC0gbWRkZXYtPmRlZ3JhZGVkID09IDEpCgkJbWRkZXYtPnJlY292ZXJ5X2NwID0gTWF4U2VjdG9yOwoKCS8qCgkgKiBmaW5kIHRoZSBmaXJzdCB3b3JraW5nIG9uZSBhbmQgdXNlIGl0IGFzIGEgc3RhcnRpbmcgcG9pbnQKCSAqIHRvIHJlYWQgYmFsYW5jaW5nLgoJICovCglmb3IgKGogPSAwOyBqIDwgY29uZi0+cmFpZF9kaXNrcyAmJgoJCSAgICAgKCFjb25mLT5taXJyb3JzW2pdLnJkZXYgfHwKCQkgICAgICAhdGVzdF9iaXQoSW5fc3luYywgJmNvbmYtPm1pcnJvcnNbal0ucmRldi0+ZmxhZ3MpKSA7IGorKykKCQkvKiBub3RoaW5nICovOwoJY29uZi0+bGFzdF91c2VkID0gajsKCgoJbWRkZXYtPnRocmVhZCA9IG1kX3JlZ2lzdGVyX3RocmVhZChyYWlkMWQsIG1kZGV2LCAiJXNfcmFpZDEiKTsKCWlmICghbWRkZXYtPnRocmVhZCkgewoJCXByaW50ayhLRVJOX0VSUgoJCSAgICAgICAicmFpZDE6IGNvdWxkbid0IGFsbG9jYXRlIHRocmVhZCBmb3IgJXNcbiIsCgkJICAgICAgIG1kbmFtZShtZGRldikpOwoJCWdvdG8gb3V0X2ZyZWVfY29uZjsKCX0KCglwcmludGsoS0VSTl9JTkZPIAoJCSJyYWlkMTogcmFpZCBzZXQgJXMgYWN0aXZlIHdpdGggJWQgb3V0IG9mICVkIG1pcnJvcnNcbiIsCgkJbWRuYW1lKG1kZGV2KSwgbWRkZXYtPnJhaWRfZGlza3MgLSBtZGRldi0+ZGVncmFkZWQsIAoJCW1kZGV2LT5yYWlkX2Rpc2tzKTsKCS8qCgkgKiBPaywgZXZlcnl0aGluZyBpcyBqdXN0IGZpbmUgbm93CgkgKi8KCW1kZGV2LT5hcnJheV9zaXplID0gbWRkZXYtPnNpemU7CgoJbWRkZXYtPnF1ZXVlLT51bnBsdWdfZm4gPSByYWlkMV91bnBsdWc7CgltZGRldi0+cXVldWUtPmlzc3VlX2ZsdXNoX2ZuID0gcmFpZDFfaXNzdWVfZmx1c2g7CgltZGRldi0+cXVldWUtPmJhY2tpbmdfZGV2X2luZm8uY29uZ2VzdGVkX2ZuID0gcmFpZDFfY29uZ2VzdGVkOwoJbWRkZXYtPnF1ZXVlLT5iYWNraW5nX2Rldl9pbmZvLmNvbmdlc3RlZF9kYXRhID0gbWRkZXY7CgoJcmV0dXJuIDA7CgpvdXRfbm9fbWVtOgoJcHJpbnRrKEtFUk5fRVJSICJyYWlkMTogY291bGRuJ3QgYWxsb2NhdGUgbWVtb3J5IGZvciAlc1xuIiwKCSAgICAgICBtZG5hbWUobWRkZXYpKTsKCm91dF9mcmVlX2NvbmY6CglpZiAoY29uZikgewoJCWlmIChjb25mLT5yMWJpb19wb29sKQoJCQltZW1wb29sX2Rlc3Ryb3koY29uZi0+cjFiaW9fcG9vbCk7CgkJa2ZyZWUoY29uZi0+bWlycm9ycyk7CgkJc2FmZV9wdXRfcGFnZShjb25mLT50bXBwYWdlKTsKCQlrZnJlZShjb25mLT5wb29saW5mbyk7CgkJa2ZyZWUoY29uZik7CgkJbWRkZXYtPnByaXZhdGUgPSBOVUxMOwoJfQpvdXQ6CglyZXR1cm4gLUVJTzsKfQoKc3RhdGljIGludCBzdG9wKG1kZGV2X3QgKm1kZGV2KQp7Cgljb25mX3QgKmNvbmYgPSBtZGRldl90b19jb25mKG1kZGV2KTsKCXN0cnVjdCBiaXRtYXAgKmJpdG1hcCA9IG1kZGV2LT5iaXRtYXA7CglpbnQgYmVoaW5kX3dhaXQgPSAwOwoKCS8qIHdhaXQgZm9yIGJlaGluZCB3cml0ZXMgdG8gY29tcGxldGUgKi8KCXdoaWxlIChiaXRtYXAgJiYgYXRvbWljX3JlYWQoJmJpdG1hcC0+YmVoaW5kX3dyaXRlcykgPiAwKSB7CgkJYmVoaW5kX3dhaXQrKzsKCQlwcmludGsoS0VSTl9JTkZPICJyYWlkMTogYmVoaW5kIHdyaXRlcyBpbiBwcm9ncmVzcyBvbiBkZXZpY2UgJXMsIHdhaXRpbmcgdG8gc3RvcCAoJWQpXG4iLCBtZG5hbWUobWRkZXYpLCBiZWhpbmRfd2FpdCk7CgkJc2V0X2N1cnJlbnRfc3RhdGUoVEFTS19VTklOVEVSUlVQVElCTEUpOwoJCXNjaGVkdWxlX3RpbWVvdXQoSFopOyAvKiB3YWl0IGEgc2Vjb25kICovCgkJLyogbmVlZCB0byBraWNrIHNvbWV0aGluZyBoZXJlIHRvIG1ha2Ugc3VyZSBJL08gZ29lcz8gKi8KCX0KCgltZF91bnJlZ2lzdGVyX3RocmVhZChtZGRldi0+dGhyZWFkKTsKCW1kZGV2LT50aHJlYWQgPSBOVUxMOwoJYmxrX3N5bmNfcXVldWUobWRkZXYtPnF1ZXVlKTsgLyogdGhlIHVucGx1ZyBmbiByZWZlcmVuY2VzICdjb25mJyovCglpZiAoY29uZi0+cjFiaW9fcG9vbCkKCQltZW1wb29sX2Rlc3Ryb3koY29uZi0+cjFiaW9fcG9vbCk7CglrZnJlZShjb25mLT5taXJyb3JzKTsKCWtmcmVlKGNvbmYtPnBvb2xpbmZvKTsKCWtmcmVlKGNvbmYpOwoJbWRkZXYtPnByaXZhdGUgPSBOVUxMOwoJcmV0dXJuIDA7Cn0KCnN0YXRpYyBpbnQgcmFpZDFfcmVzaXplKG1kZGV2X3QgKm1kZGV2LCBzZWN0b3JfdCBzZWN0b3JzKQp7CgkvKiBubyByZXN5bmMgaXMgaGFwcGVuaW5nLCBhbmQgdGhlcmUgaXMgZW5vdWdoIHNwYWNlCgkgKiBvbiBhbGwgZGV2aWNlcywgc28gd2UgY2FuIHJlc2l6ZS4KCSAqIFdlIG5lZWQgdG8gbWFrZSBzdXJlIHJlc3luYyBjb3ZlcnMgYW55IG5ldyBzcGFjZS4KCSAqIElmIHRoZSBhcnJheSBpcyBzaHJpbmtpbmcgd2Ugc2hvdWxkIHBvc3NpYmx5IHdhaXQgdW50aWwKCSAqIGFueSBpbyBpbiB0aGUgcmVtb3ZlZCBzcGFjZSBjb21wbGV0ZXMsIGJ1dCBpdCBoYXJkbHkgc2VlbXMKCSAqIHdvcnRoIGl0LgoJICovCgltZGRldi0+YXJyYXlfc2l6ZSA9IHNlY3RvcnM+PjE7CglzZXRfY2FwYWNpdHkobWRkZXYtPmdlbmRpc2ssIG1kZGV2LT5hcnJheV9zaXplIDw8IDEpOwoJbWRkZXYtPmNoYW5nZWQgPSAxOwoJaWYgKG1kZGV2LT5hcnJheV9zaXplID4gbWRkZXYtPnNpemUgJiYgbWRkZXYtPnJlY292ZXJ5X2NwID09IE1heFNlY3RvcikgewoJCW1kZGV2LT5yZWNvdmVyeV9jcCA9IG1kZGV2LT5zaXplIDw8IDE7CgkJc2V0X2JpdChNRF9SRUNPVkVSWV9ORUVERUQsICZtZGRldi0+cmVjb3ZlcnkpOwoJfQoJbWRkZXYtPnNpemUgPSBtZGRldi0+YXJyYXlfc2l6ZTsKCW1kZGV2LT5yZXN5bmNfbWF4X3NlY3RvcnMgPSBzZWN0b3JzOwoJcmV0dXJuIDA7Cn0KCnN0YXRpYyBpbnQgcmFpZDFfcmVzaGFwZShtZGRldl90ICptZGRldikKewoJLyogV2UgbmVlZCB0bzoKCSAqIDEvIHJlc2l6ZSB0aGUgcjFiaW9fcG9vbAoJICogMi8gcmVzaXplIGNvbmYtPm1pcnJvcnMKCSAqCgkgKiBXZSBhbGxvY2F0ZSBhIG5ldyByMWJpb19wb29sIGlmIHdlIGNhbi4KCSAqIFRoZW4gcmFpc2UgYSBkZXZpY2UgYmFycmllciBhbmQgd2FpdCB1bnRpbCBhbGwgSU8gc3RvcHMuCgkgKiBUaGVuIHJlc2l6ZSBjb25mLT5taXJyb3JzIGFuZCBzd2FwIGluIHRoZSBuZXcgcjFiaW8gcG9vbC4KCSAqCgkgKiBBdCB0aGUgc2FtZSB0aW1lLCB3ZSAicGFjayIgdGhlIGRldmljZXMgc28gdGhhdCBhbGwgdGhlIG1pc3NpbmcKCSAqIGRldmljZXMgaGF2ZSB0aGUgaGlnaGVyIHJhaWRfZGlzayBudW1iZXJzLgoJICovCgltZW1wb29sX3QgKm5ld3Bvb2wsICpvbGRwb29sOwoJc3RydWN0IHBvb2xfaW5mbyAqbmV3cG9vbGluZm87CgltaXJyb3JfaW5mb190ICpuZXdtaXJyb3JzOwoJY29uZl90ICpjb25mID0gbWRkZXZfdG9fY29uZihtZGRldik7CglpbnQgY250LCByYWlkX2Rpc2tzOwoJdW5zaWduZWQgbG9uZyBmbGFnczsKCWludCBkLCBkMjsKCgkvKiBDYW5ub3QgY2hhbmdlIGNodW5rX3NpemUsIGxheW91dCwgb3IgbGV2ZWwgKi8KCWlmIChtZGRldi0+Y2h1bmtfc2l6ZSAhPSBtZGRldi0+bmV3X2NodW5rIHx8CgkgICAgbWRkZXYtPmxheW91dCAhPSBtZGRldi0+bmV3X2xheW91dCB8fAoJICAgIG1kZGV2LT5sZXZlbCAhPSBtZGRldi0+bmV3X2xldmVsKSB7CgkJbWRkZXYtPm5ld19jaHVuayA9IG1kZGV2LT5jaHVua19zaXplOwoJCW1kZGV2LT5uZXdfbGF5b3V0ID0gbWRkZXYtPmxheW91dDsKCQltZGRldi0+bmV3X2xldmVsID0gbWRkZXYtPmxldmVsOwoJCXJldHVybiAtRUlOVkFMOwoJfQoKCXJhaWRfZGlza3MgPSBtZGRldi0+cmFpZF9kaXNrcyArIG1kZGV2LT5kZWx0YV9kaXNrczsKCglpZiAocmFpZF9kaXNrcyA8IGNvbmYtPnJhaWRfZGlza3MpIHsKCQljbnQ9MDsKCQlmb3IgKGQ9IDA7IGQgPCBjb25mLT5yYWlkX2Rpc2tzOyBkKyspCgkJCWlmIChjb25mLT5taXJyb3JzW2RdLnJkZXYpCgkJCQljbnQrKzsKCQlpZiAoY250ID4gcmFpZF9kaXNrcykKCQkJcmV0dXJuIC1FQlVTWTsKCX0KCgluZXdwb29saW5mbyA9IGttYWxsb2Moc2l6ZW9mKCpuZXdwb29saW5mbyksIEdGUF9LRVJORUwpOwoJaWYgKCFuZXdwb29saW5mbykKCQlyZXR1cm4gLUVOT01FTTsKCW5ld3Bvb2xpbmZvLT5tZGRldiA9IG1kZGV2OwoJbmV3cG9vbGluZm8tPnJhaWRfZGlza3MgPSByYWlkX2Rpc2tzOwoKCW5ld3Bvb2wgPSBtZW1wb29sX2NyZWF0ZShOUl9SQUlEMV9CSU9TLCByMWJpb19wb29sX2FsbG9jLAoJCQkJIHIxYmlvX3Bvb2xfZnJlZSwgbmV3cG9vbGluZm8pOwoJaWYgKCFuZXdwb29sKSB7CgkJa2ZyZWUobmV3cG9vbGluZm8pOwoJCXJldHVybiAtRU5PTUVNOwoJfQoJbmV3bWlycm9ycyA9IGt6YWxsb2Moc2l6ZW9mKHN0cnVjdCBtaXJyb3JfaW5mbykgKiByYWlkX2Rpc2tzLCBHRlBfS0VSTkVMKTsKCWlmICghbmV3bWlycm9ycykgewoJCWtmcmVlKG5ld3Bvb2xpbmZvKTsKCQltZW1wb29sX2Rlc3Ryb3kobmV3cG9vbCk7CgkJcmV0dXJuIC1FTk9NRU07Cgl9CgoJcmFpc2VfYmFycmllcihjb25mKTsKCgkvKiBvaywgZXZlcnl0aGluZyBpcyBzdG9wcGVkICovCglvbGRwb29sID0gY29uZi0+cjFiaW9fcG9vbDsKCWNvbmYtPnIxYmlvX3Bvb2wgPSBuZXdwb29sOwoKCWZvciAoZD1kMj0wOyBkIDwgY29uZi0+cmFpZF9kaXNrczsgZCsrKQoJCWlmIChjb25mLT5taXJyb3JzW2RdLnJkZXYpIHsKCQkJY29uZi0+bWlycm9yc1tkXS5yZGV2LT5yYWlkX2Rpc2sgPSBkMjsKCQkJbmV3bWlycm9yc1tkMisrXS5yZGV2ID0gY29uZi0+bWlycm9yc1tkXS5yZGV2OwoJCX0KCWtmcmVlKGNvbmYtPm1pcnJvcnMpOwoJY29uZi0+bWlycm9ycyA9IG5ld21pcnJvcnM7CglrZnJlZShjb25mLT5wb29saW5mbyk7Cgljb25mLT5wb29saW5mbyA9IG5ld3Bvb2xpbmZvOwoKCXNwaW5fbG9ja19pcnFzYXZlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoJbWRkZXYtPmRlZ3JhZGVkICs9IChyYWlkX2Rpc2tzIC0gY29uZi0+cmFpZF9kaXNrcyk7CglzcGluX3VubG9ja19pcnFyZXN0b3JlKCZjb25mLT5kZXZpY2VfbG9jaywgZmxhZ3MpOwoJY29uZi0+cmFpZF9kaXNrcyA9IG1kZGV2LT5yYWlkX2Rpc2tzID0gcmFpZF9kaXNrczsKCW1kZGV2LT5kZWx0YV9kaXNrcyA9IDA7CgoJY29uZi0+bGFzdF91c2VkID0gMDsgLyoganVzdCBtYWtlIHN1cmUgaXQgaXMgaW4tcmFuZ2UgKi8KCWxvd2VyX2JhcnJpZXIoY29uZik7CgoJc2V0X2JpdChNRF9SRUNPVkVSWV9ORUVERUQsICZtZGRldi0+cmVjb3ZlcnkpOwoJbWRfd2FrZXVwX3RocmVhZChtZGRldi0+dGhyZWFkKTsKCgltZW1wb29sX2Rlc3Ryb3kob2xkcG9vbCk7CglyZXR1cm4gMDsKfQoKc3RhdGljIHZvaWQgcmFpZDFfcXVpZXNjZShtZGRldl90ICptZGRldiwgaW50IHN0YXRlKQp7Cgljb25mX3QgKmNvbmYgPSBtZGRldl90b19jb25mKG1kZGV2KTsKCglzd2l0Y2goc3RhdGUpIHsKCWNhc2UgMToKCQlyYWlzZV9iYXJyaWVyKGNvbmYpOwoJCWJyZWFrOwoJY2FzZSAwOgoJCWxvd2VyX2JhcnJpZXIoY29uZik7CgkJYnJlYWs7Cgl9Cn0KCgpzdGF0aWMgc3RydWN0IG1ka19wZXJzb25hbGl0eSByYWlkMV9wZXJzb25hbGl0eSA9CnsKCS5uYW1lCQk9ICJyYWlkMSIsCgkubGV2ZWwJCT0gMSwKCS5vd25lcgkJPSBUSElTX01PRFVMRSwKCS5tYWtlX3JlcXVlc3QJPSBtYWtlX3JlcXVlc3QsCgkucnVuCQk9IHJ1biwKCS5zdG9wCQk9IHN0b3AsCgkuc3RhdHVzCQk9IHN0YXR1cywKCS5lcnJvcl9oYW5kbGVyCT0gZXJyb3IsCgkuaG90X2FkZF9kaXNrCT0gcmFpZDFfYWRkX2Rpc2ssCgkuaG90X3JlbW92ZV9kaXNrPSByYWlkMV9yZW1vdmVfZGlzaywKCS5zcGFyZV9hY3RpdmUJPSByYWlkMV9zcGFyZV9hY3RpdmUsCgkuc3luY19yZXF1ZXN0CT0gc3luY19yZXF1ZXN0LAoJLnJlc2l6ZQkJPSByYWlkMV9yZXNpemUsCgkuY2hlY2tfcmVzaGFwZQk9IHJhaWQxX3Jlc2hhcGUsCgkucXVpZXNjZQk9IHJhaWQxX3F1aWVzY2UsCn07CgpzdGF0aWMgaW50IF9faW5pdCByYWlkX2luaXQodm9pZCkKewoJcmV0dXJuIHJlZ2lzdGVyX21kX3BlcnNvbmFsaXR5KCZyYWlkMV9wZXJzb25hbGl0eSk7Cn0KCnN0YXRpYyB2b2lkIHJhaWRfZXhpdCh2b2lkKQp7Cgl1bnJlZ2lzdGVyX21kX3BlcnNvbmFsaXR5KCZyYWlkMV9wZXJzb25hbGl0eSk7Cn0KCm1vZHVsZV9pbml0KHJhaWRfaW5pdCk7Cm1vZHVsZV9leGl0KHJhaWRfZXhpdCk7Ck1PRFVMRV9MSUNFTlNFKCJHUEwiKTsKTU9EVUxFX0FMSUFTKCJtZC1wZXJzb25hbGl0eS0zIik7IC8qIFJBSUQxICovCk1PRFVMRV9BTElBUygibWQtcmFpZDEiKTsKTU9EVUxFX0FMSUFTKCJtZC1sZXZlbC0xIik7Cg==