LyoKICogV2luZWZpbGUKICoKICogQ29weXJpZ2h0IDIwMDAsIDIwMDMsIDIwMDQsIDIwMDUgTWFydGluIEZ1Y2hzCiAqIENvcHlyaWdodCAyMDA2IEphc29uIEdyZWVuCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKICogbW9kaWZ5IGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIKICogdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAogKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgogKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlIEdOVQogKiBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgogKgogKiBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCiAqIExpY2Vuc2UgYWxvbmcgd2l0aCB0aGlzIGxpYnJhcnk7IGlmIG5vdCwgd3JpdGUgdG8gdGhlIEZyZWUgU29mdHdhcmUKICogRm91bmRhdGlvbiwgSW5jLiwgNTEgRnJhbmtsaW4gU3QsIEZpZnRoIEZsb29yLCBCb3N0b24sIE1BIDAyMTEwLTEzMDEsIFVTQQogKi8KCiNpZmRlZiBfX1dJTkVfXwojaW5jbHVkZSAiY29uZmlnLmgiCiNpbmNsdWRlICJ3aW5lL3BvcnQuaCIKCi8qIGZvciB1bml4IGZpbGVzeXN0ZW0gZnVuY3Rpb24gY2FsbHMgKi8KI2luY2x1ZGUgPHN5cy9zdGF0Lmg+CiNpbmNsdWRlIDxzeXMvdHlwZXMuaD4KI2luY2x1ZGUgPGRpcmVudC5oPgojZW5kaWYKCiNkZWZpbmUgQ09CSk1BQ1JPUwoKI2luY2x1ZGUgIndpbmVmaWxlLmgiCiNpbmNsdWRlICJyZXNvdXJjZS5oIgoKI2lmZGVmIF9OT19FWFRFTlNJT05TCiN1bmRlZiBfTEVGVF9GSUxFUwojZW5kaWYKCiNpZm5kZWYgX01BWF9QQVRICiNkZWZpbmUgX01BWF9EUklWRSAgICAgICAgICAzCiNkZWZpbmUgX01BWF9GTkFNRSAgICAgICAgICAyNTYKI2RlZmluZSBfTUFYX0RJUiAgICAgICAgICAgIF9NQVhfRk5BTUUKI2RlZmluZSBfTUFYX0VYVCAgICAgICAgICAgIF9NQVhfRk5BTUUKI2RlZmluZSBfTUFYX1BBVEggICAgICAgICAgIDI2MAojZW5kaWYKCiNpZmRlZiBOT05BTUVMRVNTVU5JT04KI2RlZmluZQlVTklPTl9NRU1CRVIoeCkgRFVNTVlVTklPTk5BTUUueAojZWxzZQojZGVmaW5lCVVOSU9OX01FTUJFUih4KSB4CiNlbmRpZgoKCiNpZmRlZiBfU0hFTExfRk9MREVSUwojZGVmaW5lCURFRkFVTFRfU1BMSVRfUE9TCTMwMAojZWxzZQojZGVmaW5lCURFRkFVTFRfU1BMSVRfUE9TCTIwMAojZW5kaWYKCnN0YXRpYyBjb25zdCBXQ0hBUiByZWdpc3RyeV9rZXlbXSA9IHsgJ1MnLCdvJywnZicsJ3QnLCd3JywnYScsJ3InLCdlJywnXFwnLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdXJywnaScsJ24nLCdlJywnXFwnLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdXJywnaScsJ24nLCdlJywnRicsJ2knLCdsJywnZScsJ1wwJ307CnN0YXRpYyBjb25zdCBXQ0hBUiByZWdfc3RhcnRfeFtdID0geyAncycsJ3QnLCdhJywncicsJ3QnLCdYJywnXDAnfTsKc3RhdGljIGNvbnN0IFdDSEFSIHJlZ19zdGFydF95W10gPSB7ICdzJywndCcsJ2EnLCdyJywndCcsJ1knLCdcMCd9OwpzdGF0aWMgY29uc3QgV0NIQVIgcmVnX3dpZHRoW10gPSB7ICd3JywnaScsJ2QnLCd0JywnaCcsJ1wwJ307CnN0YXRpYyBjb25zdCBXQ0hBUiByZWdfaGVpZ2h0W10gPSB7ICdoJywnZScsJ2knLCdnJywnaCcsJ3QnLCdcMCd9OwpzdGF0aWMgY29uc3QgV0NIQVIgcmVnX2xvZ2ZvbnRbXSA9IHsgJ2wnLCdvJywnZycsJ2YnLCdvJywnbicsJ3QnLCdcMCd9OwoKZW51bSBFTlRSWV9UWVBFIHsKCUVUX1dJTkRPV1MsCglFVF9VTklYLAojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCUVUX1NIRUxMCiNlbmRpZgp9OwoKdHlwZWRlZiBzdHJ1Y3QgX0VudHJ5IHsKCXN0cnVjdCBfRW50cnkqCW5leHQ7CglzdHJ1Y3QgX0VudHJ5Kglkb3duOwoJc3RydWN0IF9FbnRyeSoJdXA7CgoJQk9PTAkJCWV4cGFuZGVkOwoJQk9PTAkJCXNjYW5uZWQ7CglpbnQJCQkJbGV2ZWw7CgoJV0lOMzJfRklORF9EQVRBCWRhdGE7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCglCWV9IQU5ETEVfRklMRV9JTkZPUk1BVElPTiBiaGZpOwoJQk9PTAkJCWJoZmlfdmFsaWQ7CgllbnVtIEVOVFJZX1RZUEUJZXR5cGU7CiNlbmRpZgojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCUxQSVRFTUlETElTVAlwaWRsOwoJSVNoZWxsRm9sZGVyKglmb2xkZXI7CglISUNPTgkJCWhpY29uOwojZW5kaWYKfSBFbnRyeTsKCnR5cGVkZWYgc3RydWN0IHsKCUVudHJ5CWVudHJ5OwoJVENIQVIJcGF0aFtNQVhfUEFUSF07CglUQ0hBUgl2b2xuYW1lW19NQVhfRk5BTUVdOwoJVENIQVIJZnNbX01BWF9ESVJdOwoJRFdPUkQJZHJpdmVfdHlwZTsKCURXT1JECWZzX2ZsYWdzOwp9IFJvb3Q7CgplbnVtIENPTFVNTl9GTEFHUyB7CglDT0xfU0laRQkJPSAweDAxLAoJQ09MX0RBVEUJCT0gMHgwMiwKCUNPTF9USU1FCQk9IDB4MDQsCglDT0xfQVRUUklCVVRFUwk9IDB4MDgsCglDT0xfRE9TTkFNRVMJPSAweDEwLAojaWZkZWYgX05PX0VYVEVOU0lPTlMKCUNPTF9BTEwgPSBDT0xfU0laRXxDT0xfREFURXxDT0xfVElNRXxDT0xfQVRUUklCVVRFU3xDT0xfRE9TTkFNRVMKI2Vsc2UKCUNPTF9JTkRFWAkJPSAweDIwLAoJQ09MX0xJTktTCQk9IDB4NDAsCglDT0xfQUxMID0gQ09MX1NJWkV8Q09MX0RBVEV8Q09MX1RJTUV8Q09MX0FUVFJJQlVURVN8Q09MX0RPU05BTUVTfENPTF9JTkRFWHxDT0xfTElOS1MKI2VuZGlmCn07Cgp0eXBlZGVmIGVudW0gewoJU09SVF9OQU1FLAoJU09SVF9FWFQsCglTT1JUX1NJWkUsCglTT1JUX0RBVEUKfSBTT1JUX09SREVSOwoKdHlwZWRlZiBzdHJ1Y3QgewoJSFdORAlod25kOwojaWZuZGVmIF9OT19FWFRFTlNJT05TCglIV05ECWh3bmRIZWFkZXI7CiNlbmRpZgoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwojZGVmaW5lCUNPTFVNTlMJMTAKI2Vsc2UKI2RlZmluZQlDT0xVTU5TCTUKI2VuZGlmCglpbnQJCXdpZHRoc1tDT0xVTU5TXTsKCWludAkJcG9zaXRpb25zW0NPTFVNTlMrMV07CgoJQk9PTAl0cmVlUGFuZTsKCWludAkJdmlzaWJsZV9jb2xzOwoJRW50cnkqCXJvb3Q7CglFbnRyeSoJY3VyOwp9IFBhbmU7Cgp0eXBlZGVmIHN0cnVjdCB7CglIV05ECWh3bmQ7CglQYW5lCWxlZnQ7CglQYW5lCXJpZ2h0OwoJaW50CQlmb2N1c19wYW5lOwkJLyogMDogbGVmdCAgMTogcmlnaHQgKi8KCVdJTkRPV1BMQUNFTUVOVCBwb3M7CglpbnQJCXNwbGl0X3BvczsKCUJPT0wJaGVhZGVyX3dkdGhzX29rOwoKCVRDSEFSCXBhdGhbTUFYX1BBVEhdOwoJVENIQVIJZmlsdGVyX3BhdHRlcm5bTUFYX1BBVEhdOwoJaW50CQlmaWx0ZXJfZmxhZ3M7CglSb290CXJvb3Q7CgoJU09SVF9PUkRFUiBzb3J0T3JkZXI7Cn0gQ2hpbGRXbmQ7CgoKCnN0YXRpYyB2b2lkIHJlYWRfZGlyZWN0b3J5KEVudHJ5KiBkaXIsIExQQ1RTVFIgcGF0aCwgU09SVF9PUkRFUiBzb3J0T3JkZXIsIEhXTkQgaHduZCk7CnN0YXRpYyB2b2lkIHNldF9jdXJkaXIoQ2hpbGRXbmQqIGNoaWxkLCBFbnRyeSogZW50cnksIGludCBpZHgsIEhXTkQgaHduZCk7CnN0YXRpYyB2b2lkIHJlZnJlc2hfY2hpbGQoQ2hpbGRXbmQqIGNoaWxkKTsKc3RhdGljIHZvaWQgcmVmcmVzaF9kcml2ZXModm9pZCk7CnN0YXRpYyB2b2lkIGdldF9wYXRoKEVudHJ5KiBkaXIsIFBUU1RSIHBhdGgpOwpzdGF0aWMgdm9pZCBmb3JtYXRfZGF0ZShjb25zdCBGSUxFVElNRSogZnQsIFRDSEFSKiBidWZmZXIsIGludCB2aXNpYmxlX2NvbHMpOwoKc3RhdGljIExSRVNVTFQgQ0FMTEJBQ0sgRnJhbWVXbmRQcm9jKEhXTkQgaHduZCwgVUlOVCBubXNnLCBXUEFSQU0gd3BhcmFtLCBMUEFSQU0gbHBhcmFtKTsKc3RhdGljIExSRVNVTFQgQ0FMTEJBQ0sgQ2hpbGRXbmRQcm9jKEhXTkQgaHduZCwgVUlOVCBubXNnLCBXUEFSQU0gd3BhcmFtLCBMUEFSQU0gbHBhcmFtKTsKc3RhdGljIExSRVNVTFQgQ0FMTEJBQ0sgVHJlZVduZFByb2MoSFdORCBod25kLCBVSU5UIG5tc2csIFdQQVJBTSB3cGFyYW0sIExQQVJBTSBscGFyYW0pOwoKCi8qIGdsb2JhbHMgKi8KV0lORUZJTEVfR0xPQkFMUyBHbG9iYWxzOwoKc3RhdGljIGludCBsYXN0X3NwbGl0OwoKLyogc29tZSBjb21tb24gc3RyaW5nIGNvbnN0YW50cyAqLwpzdGF0aWMgY29uc3QgVENIQVIgc0VtcHR5W10gPSB7J1wwJ307CnN0YXRpYyBjb25zdCBXQ0hBUiBzU3BhY2VbXSA9IHsnICcsICdcMCd9OwpzdGF0aWMgY29uc3QgVENIQVIgc051bUZtdFtdID0geyclJywnZCcsJ1wwJ307CnN0YXRpYyBjb25zdCBUQ0hBUiBzUU1hcmtzW10gPSB7Jz8nLCc/JywnPycsJ1wwJ307CgovKiB3aW5kb3cgY2xhc3MgbmFtZXMgKi8Kc3RhdGljIGNvbnN0IFRDSEFSIHNXSU5FRklMRUZSQU1FW10gPSB7J1cnLCdGJywnUycsJ18nLCdGJywncicsJ2EnLCdtJywnZScsJ1wwJ307CnN0YXRpYyBjb25zdCBUQ0hBUiBzV0lORUZJTEVUUkVFW10gPSB7J1cnLCdGJywnUycsJ18nLCdUJywncicsJ2UnLCdlJywnXDAnfTsKCnN0YXRpYyBjb25zdCBUQ0hBUiBzTG9uZ0hleEZtdFtdID0geyclJywnSScsJzYnLCc0JywnWCcsJ1wwJ307CnN0YXRpYyBjb25zdCBUQ0hBUiBzTG9uZ051bUZtdFtdID0geyclJywnSScsJzYnLCc0JywndScsJ1wwJ307CgoKLyogbG9hZCByZXNvdXJjZSBzdHJpbmcgKi8Kc3RhdGljIExQVFNUUiBsb2FkX3N0cmluZyhMUFRTVFIgYnVmZmVyLCBVSU5UIGlkKQp7CglMb2FkU3RyaW5nKEdsb2JhbHMuaEluc3RhbmNlLCBpZCwgYnVmZmVyLCBCVUZGRVJfTEVOKTsKCglyZXR1cm4gYnVmZmVyOwp9CgojZGVmaW5lIFJTKGIsIGkpIGxvYWRfc3RyaW5nKGIsIGkpCgoKLyogZGlzcGxheSBlcnJvciBtZXNzYWdlIGZvciB0aGUgc3BlY2lmaWVkIFdJTjMyIGVycm9yIGNvZGUgKi8Kc3RhdGljIHZvaWQgZGlzcGxheV9lcnJvcihIV05EIGh3bmQsIERXT1JEIGVycm9yKQp7CglUQ0hBUiBiMVtCVUZGRVJfTEVOXSwgYjJbQlVGRkVSX0xFTl07CglQVFNUUiBtc2c7CgoJaWYgKEZvcm1hdE1lc3NhZ2UoRk9STUFUX01FU1NBR0VfQUxMT0NBVEVfQlVGRkVSfEZPUk1BVF9NRVNTQUdFX0ZST01fU1lTVEVNLAoJCTAsIGVycm9yLCBNQUtFTEFOR0lEKExBTkdfTkVVVFJBTCxTVUJMQU5HX0RFRkFVTFQpLCAoUFRTVFIpJm1zZywgMCwgTlVMTCkpCgkJTWVzc2FnZUJveChod25kLCBtc2csIFJTKGIyLElEU19XSU5FRklMRSksIE1CX09LKTsKCWVsc2UKCQlNZXNzYWdlQm94KGh3bmQsIFJTKGIxLElEU19FUlJPUiksIFJTKGIyLElEU19XSU5FRklMRSksIE1CX09LKTsKCglMb2NhbEZyZWUobXNnKTsKfQoKCi8qIGRpc3BsYXkgbmV0d29yayBlcnJvciBtZXNzYWdlIHVzaW5nIFdOZXRHZXRMYXN0RXJyb3IoKSAqLwpzdGF0aWMgdm9pZCBkaXNwbGF5X25ldHdvcmtfZXJyb3IoSFdORCBod25kKQp7CglUQ0hBUiBtc2dbQlVGRkVSX0xFTl0sIHByb3ZpZGVyW0JVRkZFUl9MRU5dLCBiMltCVUZGRVJfTEVOXTsKCURXT1JEIGVycm9yOwoKCWlmIChXTmV0R2V0TGFzdEVycm9yKCZlcnJvciwgbXNnLCBCVUZGRVJfTEVOLCBwcm92aWRlciwgQlVGRkVSX0xFTikgPT0gTk9fRVJST1IpCgkJTWVzc2FnZUJveChod25kLCBtc2csIFJTKGIyLElEU19XSU5FRklMRSksIE1CX09LKTsKfQoKc3RhdGljIGlubGluZSBCT09MIGdldF9jaGVjayhIV05EIGh3bmQsIElOVCBpZCkKewoJcmV0dXJuIEJTVF9DSEVDS0VEJlNlbmRNZXNzYWdlVyhHZXREbGdJdGVtKGh3bmQsIGlkKSwgQk1fR0VUU1RBVEUsIDAsIDApOwp9CgpzdGF0aWMgaW5saW5lIElOVCBzZXRfY2hlY2soSFdORCBod25kLCBJTlQgaWQsIEJPT0wgb24pCnsKCXJldHVybiBTZW5kTWVzc2FnZVcoR2V0RGxnSXRlbShod25kLCBpZCksIEJNX1NFVENIRUNLLCBvbj9CU1RfQ0hFQ0tFRDpCU1RfVU5DSEVDS0VELCAwKTsKfQoKc3RhdGljIGlubGluZSB2b2lkIGNob29zZV9mb250KEhXTkQgaHduZCkKewogICAgICAgIFdDSEFSIGRsZ19uYW1lW0JVRkZFUl9MRU5dLCBkbGdfaW5mb1tCVUZGRVJfTEVOXTsKICAgICAgICBDSE9PU0VGT05UVyBjaEZvbnQ7CiAgICAgICAgTE9HRk9OVFcgbEZvbnQ7CgogICAgICAgIEhEQyBoZGMgPSBHZXREQyhod25kKTsKICAgICAgICBjaEZvbnQubFN0cnVjdFNpemUgPSBzaXplb2YoQ0hPT1NFRk9OVCk7CiAgICAgICAgY2hGb250Lmh3bmRPd25lciA9IGh3bmQ7CiAgICAgICAgY2hGb250LmhEQyA9IE5VTEw7CiAgICAgICAgY2hGb250LmxwTG9nRm9udCA9ICZsRm9udDsKICAgICAgICBjaEZvbnQuRmxhZ3MgPSBDRl9TQ1JFRU5GT05UUyB8IENGX0ZPUkNFRk9OVEVYSVNUIHwgQ0ZfTElNSVRTSVpFIHwgQ0ZfTk9TQ1JJUFRTRUw7CiAgICAgICAgY2hGb250LnJnYkNvbG9ycyA9IFJHQigwLDAsMCk7CiAgICAgICAgY2hGb250LmxDdXN0RGF0YSA9IDA7CiAgICAgICAgY2hGb250LmxwZm5Ib29rID0gTlVMTDsKICAgICAgICBjaEZvbnQubHBUZW1wbGF0ZU5hbWUgPSBOVUxMOwogICAgICAgIGNoRm9udC5oSW5zdGFuY2UgPSBHbG9iYWxzLmhJbnN0YW5jZTsKICAgICAgICBjaEZvbnQubHBzelN0eWxlID0gTlVMTDsKICAgICAgICBjaEZvbnQubkZvbnRUeXBlID0gU0lNVUxBVEVEX0ZPTlRUWVBFOwogICAgICAgIGNoRm9udC5uU2l6ZU1pbiA9IDA7CiAgICAgICAgY2hGb250Lm5TaXplTWF4ID0gMjQ7CgogICAgICAgIGlmIChDaG9vc2VGb250VygmY2hGb250KSkgewogICAgICAgICAgICAgICAgSFdORCBjaGlsZFduZDsKICAgICAgICAgICAgICAgIEhGT05UIGhGb250T2xkOwoKICAgICAgICAgICAgICAgIERlbGV0ZU9iamVjdChHbG9iYWxzLmhmb250KTsKICAgICAgICAgICAgICAgIEdsb2JhbHMuaGZvbnQgPSBDcmVhdGVGb250SW5kaXJlY3RXKCZsRm9udCk7CiAgICAgICAgICAgICAgICBoRm9udE9sZCA9IFNlbGVjdE9iamVjdChoZGMsIEdsb2JhbHMuaGZvbnQpOwogICAgICAgICAgICAgICAgR2V0VGV4dEV4dGVudFBvaW50MzJXKGhkYywgc1NwYWNlLCAxLCAmR2xvYmFscy5zcGFjZVNpemUpOwoKICAgICAgICAgICAgICAgIC8qIGNoYW5nZSBmb250IGluIGFsbCBvcGVuIGNoaWxkIHdpbmRvd3MgKi8KICAgICAgICAgICAgICAgIGZvcihjaGlsZFduZD1HZXRXaW5kb3coR2xvYmFscy5obWRpY2xpZW50LEdXX0NISUxEKTsgY2hpbGRXbmQ7IGNoaWxkV25kPUdldE5leHRXaW5kb3coY2hpbGRXbmQsR1dfSFdORE5FWFQpKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIENoaWxkV25kKiBjaGlsZCA9IChDaGlsZFduZCopIEdldFdpbmRvd0xvbmdQdHJXKGNoaWxkV25kLCBHV0xQX1VTRVJEQVRBKTsKICAgICAgICAgICAgICAgICAgICAgICAgU2VuZE1lc3NhZ2VXKGNoaWxkLT5sZWZ0Lmh3bmQsIFdNX1NFVEZPTlQsIChXUEFSQU0pR2xvYmFscy5oZm9udCwgVFJVRSk7CiAgICAgICAgICAgICAgICAgICAgICAgIFNlbmRNZXNzYWdlVyhjaGlsZC0+cmlnaHQuaHduZCwgV01fU0VURk9OVCwgKFdQQVJBTSlHbG9iYWxzLmhmb250LCBUUlVFKTsKICAgICAgICAgICAgICAgICAgICAgICAgU2VuZE1lc3NhZ2VXKGNoaWxkLT5sZWZ0Lmh3bmQsIExCX1NFVElURU1IRUlHSFQsIDEsIG1heChHbG9iYWxzLnNwYWNlU2l6ZS5jeSxJTUFHRV9IRUlHSFQrMykpOwogICAgICAgICAgICAgICAgICAgICAgICBTZW5kTWVzc2FnZVcoY2hpbGQtPnJpZ2h0Lmh3bmQsIExCX1NFVElURU1IRUlHSFQsIDEsIG1heChHbG9iYWxzLnNwYWNlU2l6ZS5jeSxJTUFHRV9IRUlHSFQrMykpOwogICAgICAgICAgICAgICAgICAgICAgICBJbnZhbGlkYXRlUmVjdChjaGlsZC0+bGVmdC5od25kLCBOVUxMLCBUUlVFKTsKICAgICAgICAgICAgICAgICAgICAgICAgSW52YWxpZGF0ZVJlY3QoY2hpbGQtPnJpZ2h0Lmh3bmQsIE5VTEwsIFRSVUUpOwogICAgICAgICAgICAgICAgfQoKICAgICAgICAgICAgICAgIFNlbGVjdE9iamVjdChoZGMsIGhGb250T2xkKTsKICAgICAgICB9CiAgICAgICAgZWxzZSBpZiAoQ29tbURsZ0V4dGVuZGVkRXJyb3IoKSkgewogICAgICAgICAgICAgICAgTG9hZFN0cmluZ1coR2xvYmFscy5oSW5zdGFuY2UsIElEU19GT05UX1NFTF9ETEdfTkFNRSwgZGxnX25hbWUsIEJVRkZFUl9MRU4pOwogICAgICAgICAgICAgICAgTG9hZFN0cmluZ1coR2xvYmFscy5oSW5zdGFuY2UsIElEU19GT05UX1NFTF9FUlJPUiwgZGxnX2luZm8sIEJVRkZFUl9MRU4pOwogICAgICAgICAgICAgICAgTWVzc2FnZUJveFcoaHduZCwgZGxnX2luZm8sIGRsZ19uYW1lLCBNQl9PSyk7CiAgICAgICAgfQoKICAgICAgICBSZWxlYXNlREMoaHduZCwgaGRjKTsKfQoKI2lmZGVmIF9fV0lORV9fCgojaWZkZWYgVU5JQ09ERQoKLyogY2FsbCB2c3dwcmludGYoKSBpbiBtc3ZjcnQuZGxsICovCi8qVE9ETzogZml4IHN3cHJpbnRmKCkgaW4gbm9uLW1zdmNydCBtb2RlLCBzbyB0aGF0IHRoaXMgZHluYW1pYyBsaW5raW5nIGZ1bmN0aW9uIGNhbiBiZSByZW1vdmVkICovCnN0YXRpYyBpbnQgbXN2Y3J0X3N3cHJpbnRmKFdDSEFSKiBidWZmZXIsIGNvbnN0IFdDSEFSKiBmbXQsIC4uLikKewoJc3RhdGljIGludCAoX19jZGVjbCAqcHZzd3ByaW50ZikoV0NIQVIqLCBjb25zdCBXQ0hBUiosIHZhX2xpc3QpID0gTlVMTDsKCXZhX2xpc3QgYXA7CglpbnQgcmV0OwoKCWlmICghcHZzd3ByaW50ZikgewoJCUhNT0RVTEUgaE1vZE1zdmNydCA9IExvYWRMaWJyYXJ5QSgibXN2Y3J0Iik7CgkJcHZzd3ByaW50ZiA9IChpbnQoX19jZGVjbCopKFdDSEFSKixjb25zdCBXQ0hBUiosdmFfbGlzdCkpIEdldFByb2NBZGRyZXNzKGhNb2RNc3ZjcnQsICJ2c3dwcmludGYiKTsKCX0KCgl2YV9zdGFydChhcCwgZm10KTsKCXJldCA9ICgqcHZzd3ByaW50ZikoYnVmZmVyLCBmbXQsIGFwKTsKCXZhX2VuZChhcCk7CgoJcmV0dXJuIHJldDsKfQoKc3RhdGljIExQQ1dTVFIgbXlfd2NzcmNocihMUENXU1RSIHN0ciwgV0NIQVIgYykKewoJTFBDV1NUUiBwID0gc3RyOwoKCXdoaWxlKCpwKQoJCSsrcDsKCglkbyB7CgkJaWYgKC0tcCA8IHN0cikKCQkJcmV0dXJuIE5VTEw7Cgl9IHdoaWxlKCpwICE9IGMpOwoKCXJldHVybiBwOwp9CgojZGVmaW5lIF90Y3NyY2hyIG15X3djc3JjaHIKI2Vsc2UJLyogVU5JQ09ERSAqLwojZGVmaW5lIF90Y3NyY2hyIHN0cnJjaHIKI2VuZGlmCS8qIFVOSUNPREUgKi8KCiNlbmRpZgkvKiBfX1dJTkVfXyAqLwoKCi8qIGFsbG9jYXRlIGFuZCBpbml0aWFsaXNlIGEgZGlyZWN0b3J5IGVudHJ5ICovCnN0YXRpYyBFbnRyeSogYWxsb2NfZW50cnkodm9pZCkKewoJRW50cnkqIGVudHJ5ID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIHNpemVvZihFbnRyeSkpOwoKI2lmZGVmIF9TSEVMTF9GT0xERVJTCgllbnRyeS0+cGlkbCA9IE5VTEw7CgllbnRyeS0+Zm9sZGVyID0gTlVMTDsKCWVudHJ5LT5oaWNvbiA9IDA7CiNlbmRpZgoKCXJldHVybiBlbnRyeTsKfQoKLyogZnJlZSBhIGRpcmVjdG9yeSBlbnRyeSAqLwpzdGF0aWMgdm9pZCBmcmVlX2VudHJ5KEVudHJ5KiBlbnRyeSkKewojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCWlmIChlbnRyeS0+aGljb24gJiYgZW50cnktPmhpY29uIT0oSElDT04pLTEpCgkJRGVzdHJveUljb24oZW50cnktPmhpY29uKTsKCglpZiAoZW50cnktPmZvbGRlciAmJiBlbnRyeS0+Zm9sZGVyIT1HbG9iYWxzLmlEZXNrdG9wKQoJCUlTaGVsbEZvbGRlcl9SZWxlYXNlKGVudHJ5LT5mb2xkZXIpOwoKCWlmIChlbnRyeS0+cGlkbCkKCQlJTWFsbG9jX0ZyZWUoR2xvYmFscy5pTWFsbG9jLCBlbnRyeS0+cGlkbCk7CiNlbmRpZgoKCUhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGVudHJ5KTsKfQoKLyogcmVjdXJzaXZlbHkgZnJlZSBhbGwgY2hpbGQgZW50cmllcyAqLwpzdGF0aWMgdm9pZCBmcmVlX2VudHJpZXMoRW50cnkqIGRpcikKewoJRW50cnkgKmVudHJ5LCAqbmV4dD1kaXItPmRvd247CgoJaWYgKG5leHQpIHsKCQlkaXItPmRvd24gPSAwOwoKCQlkbyB7CgkJCWVudHJ5ID0gbmV4dDsKCQkJbmV4dCA9IGVudHJ5LT5uZXh0OwoKCQkJZnJlZV9lbnRyaWVzKGVudHJ5KTsKCQkJZnJlZV9lbnRyeShlbnRyeSk7CgkJfSB3aGlsZShuZXh0KTsKCX0KfQoKCnN0YXRpYyB2b2lkIHJlYWRfZGlyZWN0b3J5X3dpbihFbnRyeSogZGlyLCBMUENUU1RSIHBhdGgpCnsKCUVudHJ5KiBmaXJzdF9lbnRyeSA9IE5VTEw7CglFbnRyeSogbGFzdCA9IE5VTEw7CglFbnRyeSogZW50cnk7CgoJaW50IGxldmVsID0gZGlyLT5sZXZlbCArIDE7CglXSU4zMl9GSU5EX0RBVEEgdzMyZmQ7CglIQU5ETEUgaEZpbmQ7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCUhBTkRMRSBoRmlsZTsKI2VuZGlmCgoJVENIQVIgYnVmZmVyW01BWF9QQVRIXSwgKnA7Cglmb3IocD1idWZmZXI7ICpwYXRoOyApCgkJKnArKyA9ICpwYXRoKys7CgoJKnArKyA9ICdcXCc7CglwWzBdID0gJyonOwoJcFsxXSA9ICdcMCc7CgoJaEZpbmQgPSBGaW5kRmlyc3RGaWxlKGJ1ZmZlciwgJnczMmZkKTsKCglpZiAoaEZpbmQgIT0gSU5WQUxJRF9IQU5ETEVfVkFMVUUpIHsKCQlkbyB7CiNpZmRlZiBfTk9fRVhURU5TSU9OUwoJCQkvKiBoaWRlIGRpcmVjdG9yeSBlbnRyeSAiLiIgKi8KCQkJaWYgKHczMmZkLmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpIHsKCQkJCUxQQ1RTVFIgbmFtZSA9IHczMmZkLmNGaWxlTmFtZTsKCgkJCQlpZiAobmFtZVswXT09Jy4nICYmIG5hbWVbMV09PSdcMCcpCgkJCQkJY29udGludWU7CgkJCX0KI2VuZGlmCgkJCWVudHJ5ID0gYWxsb2NfZW50cnkoKTsKCgkJCWlmICghZmlyc3RfZW50cnkpCgkJCQlmaXJzdF9lbnRyeSA9IGVudHJ5OwoKCQkJaWYgKGxhc3QpCgkJCQlsYXN0LT5uZXh0ID0gZW50cnk7CgoJCQltZW1jcHkoJmVudHJ5LT5kYXRhLCAmdzMyZmQsIHNpemVvZihXSU4zMl9GSU5EX0RBVEEpKTsKCQkJZW50cnktPmRvd24gPSBOVUxMOwoJCQllbnRyeS0+dXAgPSBkaXI7CgkJCWVudHJ5LT5leHBhbmRlZCA9IEZBTFNFOwoJCQllbnRyeS0+c2Nhbm5lZCA9IEZBTFNFOwoJCQllbnRyeS0+bGV2ZWwgPSBsZXZlbDsKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJZW50cnktPmV0eXBlID0gRVRfV0lORE9XUzsKCQkJZW50cnktPmJoZmlfdmFsaWQgPSBGQUxTRTsKCgkJCWxzdHJjcHkocCwgZW50cnktPmRhdGEuY0ZpbGVOYW1lKTsKCgkJCWhGaWxlID0gQ3JlYXRlRmlsZShidWZmZXIsIEdFTkVSSUNfUkVBRCwgRklMRV9TSEFSRV9SRUFEfEZJTEVfU0hBUkVfV1JJVEV8RklMRV9TSEFSRV9ERUxFVEUsCgkJCQkJCQkJMCwgT1BFTl9FWElTVElORywgRklMRV9GTEFHX0JBQ0tVUF9TRU1BTlRJQ1MsIDApOwoKCQkJaWYgKGhGaWxlICE9IElOVkFMSURfSEFORExFX1ZBTFVFKSB7CgkJCQlpZiAoR2V0RmlsZUluZm9ybWF0aW9uQnlIYW5kbGUoaEZpbGUsICZlbnRyeS0+YmhmaSkpCgkJCQkJZW50cnktPmJoZmlfdmFsaWQgPSBUUlVFOwoKCQkJCUNsb3NlSGFuZGxlKGhGaWxlKTsKCQkJfQojZW5kaWYKCgkJCWxhc3QgPSBlbnRyeTsKCQl9IHdoaWxlKEZpbmROZXh0RmlsZShoRmluZCwgJnczMmZkKSk7CgoJCWlmIChsYXN0KQoJCQlsYXN0LT5uZXh0ID0gTlVMTDsKCgkJRmluZENsb3NlKGhGaW5kKTsKCX0KCglkaXItPmRvd24gPSBmaXJzdF9lbnRyeTsKCWRpci0+c2Nhbm5lZCA9IFRSVUU7Cn0KCgpzdGF0aWMgRW50cnkqIGZpbmRfZW50cnlfd2luKEVudHJ5KiBkaXIsIExQQ1RTVFIgbmFtZSkKewoJRW50cnkqIGVudHJ5OwoKCWZvcihlbnRyeT1kaXItPmRvd247IGVudHJ5OyBlbnRyeT1lbnRyeS0+bmV4dCkgewoJCUxQQ1RTVFIgcCA9IG5hbWU7CgkJTFBDVFNUUiBxID0gZW50cnktPmRhdGEuY0ZpbGVOYW1lOwoKCQlkbyB7CgkJCWlmICghKnAgfHwgKnAgPT0gJ1xcJyB8fCAqcCA9PSAnLycpCgkJCQlyZXR1cm4gZW50cnk7CgkJfSB3aGlsZSh0b2xvd2VyKCpwKyspID09IHRvbG93ZXIoKnErKykpOwoKCQlwID0gbmFtZTsKCQlxID0gZW50cnktPmRhdGEuY0FsdGVybmF0ZUZpbGVOYW1lOwoKCQlkbyB7CgkJCWlmICghKnAgfHwgKnAgPT0gJ1xcJyB8fCAqcCA9PSAnLycpCgkJCQlyZXR1cm4gZW50cnk7CgkJfSB3aGlsZSh0b2xvd2VyKCpwKyspID09IHRvbG93ZXIoKnErKykpOwoJfQoKCXJldHVybiAwOwp9CgoKc3RhdGljIEVudHJ5KiByZWFkX3RyZWVfd2luKFJvb3QqIHJvb3QsIExQQ1RTVFIgcGF0aCwgU09SVF9PUkRFUiBzb3J0T3JkZXIsIEhXTkQgaHduZCkKewoJVENIQVIgYnVmZmVyW01BWF9QQVRIXTsKCUVudHJ5KiBlbnRyeSA9ICZyb290LT5lbnRyeTsKCUxQQ1RTVFIgcyA9IHBhdGg7CglQVFNUUiBkID0gYnVmZmVyOwoKCUhDVVJTT1Igb2xkX2N1cnNvciA9IFNldEN1cnNvcihMb2FkQ3Vyc29yKDAsIElEQ19XQUlUKSk7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCgllbnRyeS0+ZXR5cGUgPSBFVF9XSU5ET1dTOwojZW5kaWYKCgl3aGlsZShlbnRyeSkgewoJCXdoaWxlKCpzICYmICpzICE9ICdcXCcgJiYgKnMgIT0gJy8nKQoJCQkqZCsrID0gKnMrKzsKCgkJd2hpbGUoKnMgPT0gJ1xcJyB8fCAqcyA9PSAnLycpCgkJCXMrKzsKCgkJKmQrKyA9ICdcXCc7CgkJKmQgPSAnXDAnOwoKCQlyZWFkX2RpcmVjdG9yeShlbnRyeSwgYnVmZmVyLCBzb3J0T3JkZXIsIGh3bmQpOwoKCQlpZiAoZW50cnktPmRvd24pCgkJCWVudHJ5LT5leHBhbmRlZCA9IFRSVUU7CgoJCWlmICghKnMpCgkJCWJyZWFrOwoKCQllbnRyeSA9IGZpbmRfZW50cnlfd2luKGVudHJ5LCBzKTsKCX0KCglTZXRDdXJzb3Iob2xkX2N1cnNvcik7CgoJcmV0dXJuIGVudHJ5Owp9CgoKI2lmICFkZWZpbmVkKF9OT19FWFRFTlNJT05TKSAmJiBkZWZpbmVkKF9fV0lORV9fKQoKc3RhdGljIEJPT0wgdGltZV90b19maWxldGltZShjb25zdCB0aW1lX3QqIHQsIEZJTEVUSU1FKiBmdGltZSkKewoJc3RydWN0IHRtKiB0bSA9IGdtdGltZSh0KTsKCVNZU1RFTVRJTUUgc3RpbWU7CgoJaWYgKCF0bSkKCQlyZXR1cm4gRkFMU0U7CgoJc3RpbWUud1llYXIgPSB0bS0+dG1feWVhcisxOTAwOwoJc3RpbWUud01vbnRoID0gdG0tPnRtX21vbisxOwoJLyoJc3RpbWUud0RheU9mV2VlayAqLwoJc3RpbWUud0RheSA9IHRtLT50bV9tZGF5OwoJc3RpbWUud0hvdXIgPSB0bS0+dG1faG91cjsKCXN0aW1lLndNaW51dGUgPSB0bS0+dG1fbWluOwoJc3RpbWUud1NlY29uZCA9IHRtLT50bV9zZWM7CgoJcmV0dXJuIFN5c3RlbVRpbWVUb0ZpbGVUaW1lKCZzdGltZSwgZnRpbWUpOwp9CgpzdGF0aWMgdm9pZCByZWFkX2RpcmVjdG9yeV91bml4KEVudHJ5KiBkaXIsIExQQ1RTVFIgcGF0aCkKewoJRW50cnkqIGZpcnN0X2VudHJ5ID0gTlVMTDsKCUVudHJ5KiBsYXN0ID0gTlVMTDsKCUVudHJ5KiBlbnRyeTsKCURJUiogcGRpcjsKCglpbnQgbGV2ZWwgPSBkaXItPmxldmVsICsgMTsKI2lmZGVmIFVOSUNPREUKCWNoYXIgY3BhdGhbTUFYX1BBVEhdOwoKCVdpZGVDaGFyVG9NdWx0aUJ5dGUoQ1BfVU5JWENQLCAwLCBwYXRoLCAtMSwgY3BhdGgsIE1BWF9QQVRILCBOVUxMLCBOVUxMKTsKI2Vsc2UKCWNvbnN0IGNoYXIqIGNwYXRoID0gcGF0aDsKI2VuZGlmCgoJcGRpciA9IG9wZW5kaXIoY3BhdGgpOwoKCWlmIChwZGlyKSB7CgkJc3RydWN0IHN0YXQgc3Q7CgkJc3RydWN0IGRpcmVudCogZW50OwoJCWNoYXIgYnVmZmVyW01BWF9QQVRIXSwgKnA7CgkJY29uc3QgY2hhciogczsKCgkJZm9yKHA9YnVmZmVyLHM9Y3BhdGg7ICpzOyApCgkJCSpwKysgPSAqcysrOwoKCQlpZiAocD09YnVmZmVyIHx8IHBbLTFdIT0nLycpCgkJCSpwKysgPSAnLyc7CgoJCXdoaWxlKChlbnQ9cmVhZGRpcihwZGlyKSkpIHsKCQkJZW50cnkgPSBhbGxvY19lbnRyeSgpOwoKCQkJaWYgKCFmaXJzdF9lbnRyeSkKCQkJCWZpcnN0X2VudHJ5ID0gZW50cnk7CgoJCQlpZiAobGFzdCkKCQkJCWxhc3QtPm5leHQgPSBlbnRyeTsKCgkJCWVudHJ5LT5ldHlwZSA9IEVUX1VOSVg7CgoJCQlzdHJjcHkocCwgZW50LT5kX25hbWUpOwojaWZkZWYgVU5JQ09ERQoJCQlNdWx0aUJ5dGVUb1dpZGVDaGFyKENQX1VOSVhDUCwgMCwgcCwgLTEsIGVudHJ5LT5kYXRhLmNGaWxlTmFtZSwgTUFYX1BBVEgpOwojZWxzZQoJCQlsc3RyY3B5KGVudHJ5LT5kYXRhLmNGaWxlTmFtZSwgcCk7CiNlbmRpZgoKCQkJaWYgKCFzdGF0KGJ1ZmZlciwgJnN0KSkgewoJCQkJZW50cnktPmRhdGEuZHdGaWxlQXR0cmlidXRlcyA9IHBbMF09PScuJz8gRklMRV9BVFRSSUJVVEVfSElEREVOOiAwOwoKCQkJCWlmIChTX0lTRElSKHN0LnN0X21vZGUpKQoJCQkJCWVudHJ5LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgfD0gRklMRV9BVFRSSUJVVEVfRElSRUNUT1JZOwoKCQkJCWVudHJ5LT5kYXRhLm5GaWxlU2l6ZUxvdyA9IHN0LnN0X3NpemUgJiAweEZGRkZGRkZGOwoJCQkJZW50cnktPmRhdGEubkZpbGVTaXplSGlnaCA9IHN0LnN0X3NpemUgPj4gMzI7CgoJCQkJbWVtc2V0KCZlbnRyeS0+ZGF0YS5mdENyZWF0aW9uVGltZSwgMCwgc2l6ZW9mKEZJTEVUSU1FKSk7CgkJCQl0aW1lX3RvX2ZpbGV0aW1lKCZzdC5zdF9hdGltZSwgJmVudHJ5LT5kYXRhLmZ0TGFzdEFjY2Vzc1RpbWUpOwoJCQkJdGltZV90b19maWxldGltZSgmc3Quc3RfbXRpbWUsICZlbnRyeS0+ZGF0YS5mdExhc3RXcml0ZVRpbWUpOwoKCQkJCWVudHJ5LT5iaGZpLm5GaWxlSW5kZXhMb3cgPSBlbnQtPmRfaW5vOwoJCQkJZW50cnktPmJoZmkubkZpbGVJbmRleEhpZ2ggPSAwOwoKCQkJCWVudHJ5LT5iaGZpLm5OdW1iZXJPZkxpbmtzID0gc3Quc3Rfbmxpbms7CgoJCQkJZW50cnktPmJoZmlfdmFsaWQgPSBUUlVFOwoJCQl9IGVsc2UgewoJCQkJZW50cnktPmRhdGEubkZpbGVTaXplTG93ID0gMDsKCQkJCWVudHJ5LT5kYXRhLm5GaWxlU2l6ZUhpZ2ggPSAwOwoJCQkJZW50cnktPmJoZmlfdmFsaWQgPSBGQUxTRTsKCQkJfQoKCQkJZW50cnktPmRvd24gPSBOVUxMOwoJCQllbnRyeS0+dXAgPSBkaXI7CgkJCWVudHJ5LT5leHBhbmRlZCA9IEZBTFNFOwoJCQllbnRyeS0+c2Nhbm5lZCA9IEZBTFNFOwoJCQllbnRyeS0+bGV2ZWwgPSBsZXZlbDsKCgkJCWxhc3QgPSBlbnRyeTsKCQl9CgoJCWlmIChsYXN0KQoJCQlsYXN0LT5uZXh0ID0gTlVMTDsKCgkJY2xvc2VkaXIocGRpcik7Cgl9CgoJZGlyLT5kb3duID0gZmlyc3RfZW50cnk7CglkaXItPnNjYW5uZWQgPSBUUlVFOwp9CgpzdGF0aWMgRW50cnkqIGZpbmRfZW50cnlfdW5peChFbnRyeSogZGlyLCBMUENUU1RSIG5hbWUpCnsKCUVudHJ5KiBlbnRyeTsKCglmb3IoZW50cnk9ZGlyLT5kb3duOyBlbnRyeTsgZW50cnk9ZW50cnktPm5leHQpIHsKCQlMUENUU1RSIHAgPSBuYW1lOwoJCUxQQ1RTVFIgcSA9IGVudHJ5LT5kYXRhLmNGaWxlTmFtZTsKCgkJZG8gewoJCQlpZiAoISpwIHx8ICpwID09ICcvJykKCQkJCXJldHVybiBlbnRyeTsKCQl9IHdoaWxlKCpwKysgPT0gKnErKyk7Cgl9CgoJcmV0dXJuIDA7Cn0KCnN0YXRpYyBFbnRyeSogcmVhZF90cmVlX3VuaXgoUm9vdCogcm9vdCwgTFBDVFNUUiBwYXRoLCBTT1JUX09SREVSIHNvcnRPcmRlciwgSFdORCBod25kKQp7CglUQ0hBUiBidWZmZXJbTUFYX1BBVEhdOwoJRW50cnkqIGVudHJ5ID0gJnJvb3QtPmVudHJ5OwoJTFBDVFNUUiBzID0gcGF0aDsKCVBUU1RSIGQgPSBidWZmZXI7CgoJSENVUlNPUiBvbGRfY3Vyc29yID0gU2V0Q3Vyc29yKExvYWRDdXJzb3IoMCwgSURDX1dBSVQpKTsKCgllbnRyeS0+ZXR5cGUgPSBFVF9VTklYOwoKCXdoaWxlKGVudHJ5KSB7CgkJd2hpbGUoKnMgJiYgKnMgIT0gJy8nKQoJCQkqZCsrID0gKnMrKzsKCgkJd2hpbGUoKnMgPT0gJy8nKQoJCQlzKys7CgoJCSpkKysgPSAnLyc7CgkJKmQgPSAnXDAnOwoKCQlyZWFkX2RpcmVjdG9yeShlbnRyeSwgYnVmZmVyLCBzb3J0T3JkZXIsIGh3bmQpOwoKCQlpZiAoZW50cnktPmRvd24pCgkJCWVudHJ5LT5leHBhbmRlZCA9IFRSVUU7CgoJCWlmICghKnMpCgkJCWJyZWFrOwoKCQllbnRyeSA9IGZpbmRfZW50cnlfdW5peChlbnRyeSwgcyk7Cgl9CgoJU2V0Q3Vyc29yKG9sZF9jdXJzb3IpOwoKCXJldHVybiBlbnRyeTsKfQoKI2VuZGlmIC8qICFkZWZpbmVkKF9OT19FWFRFTlNJT05TKSAmJiBkZWZpbmVkKF9fV0lORV9fKSAqLwoKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoKI2lmZGVmIFVOSUNPREUKI2RlZmluZQlnZXRfc3RycmV0IGdldF9zdHJyZXRXCiNkZWZpbmUJcGF0aF9mcm9tX3BpZGwgcGF0aF9mcm9tX3BpZGxXCiNlbHNlCiNkZWZpbmUJZ2V0X3N0cnJldCBnZXRfc3RycmV0QQojZGVmaW5lCXBhdGhfZnJvbV9waWRsIHBhdGhfZnJvbV9waWRsQQojZW5kaWYKCgpzdGF0aWMgdm9pZCBmcmVlX3N0cnJldChTVFJSRVQqIHN0cikKewoJaWYgKHN0ci0+dVR5cGUgPT0gU1RSUkVUX1dTVFIpCgkJSU1hbGxvY19GcmVlKEdsb2JhbHMuaU1hbGxvYywgc3RyLT5VTklPTl9NRU1CRVIocE9sZVN0cikpOwp9CgoKI2lmbmRlZiBVTklDT0RFCgpzdGF0aWMgTFBTVFIgc3RyY3B5bihMUFNUUiBkZXN0LCBMUENTVFIgc291cmNlLCBzaXplX3QgY291bnQpCnsKIExQQ1NUUiBzOwogTFBTVFIgZCA9IGRlc3Q7CgogZm9yKHM9c291cmNlOyBjb3VudCYmKCpkKys9KnMrKyk7ICkKICBjb3VudC0tOwoKIHJldHVybiBkZXN0Owp9CgpzdGF0aWMgdm9pZCBnZXRfc3RycmV0QShTVFJSRVQqIHN0ciwgY29uc3QgU0hJVEVNSUQqIHNoaWlkLCBMUFNUUiBidWZmZXIsIGludCBsZW4pCnsKIHN3aXRjaChzdHItPnVUeXBlKSB7CiAgY2FzZSBTVFJSRVRfV1NUUjoKCVdpZGVDaGFyVG9NdWx0aUJ5dGUoQ1BfQUNQLCAwLCBzdHItPlVOSU9OX01FTUJFUihwT2xlU3RyKSwgLTEsIGJ1ZmZlciwgbGVuLCBOVUxMLCBOVUxMKTsKCWJyZWFrOwoKICBjYXNlIFNUUlJFVF9PRkZTRVQ6CglzdHJjcHluKGJ1ZmZlciwgKExQQ1NUUilzaGlpZCtzdHItPlVOSU9OX01FTUJFUih1T2Zmc2V0KSwgbGVuKTsKCWJyZWFrOwoKICBjYXNlIFNUUlJFVF9DU1RSOgoJc3RyY3B5bihidWZmZXIsIHN0ci0+VU5JT05fTUVNQkVSKGNTdHIpLCBsZW4pOwogfQp9CgpzdGF0aWMgSFJFU1VMVCBwYXRoX2Zyb21fcGlkbEEoSVNoZWxsRm9sZGVyKiBmb2xkZXIsIExQSVRFTUlETElTVCBwaWRsLCBMUFNUUiBidWZmZXIsIGludCBsZW4pCnsKCVNUUlJFVCBzdHI7CgoJIC8qIFNIR0ROX0ZPUlBBUlNJTkc6IGdldCBmdWxsIHBhdGggb2YgaWQgbGlzdCAqLwoJSFJFU1VMVCBociA9IElTaGVsbEZvbGRlcl9HZXREaXNwbGF5TmFtZU9mKGZvbGRlciwgcGlkbCwgU0hHRE5fRk9SUEFSU0lORywgJnN0cik7CgoJaWYgKFNVQ0NFRURFRChocikpIHsKCQlnZXRfc3RycmV0QSgmc3RyLCAmcGlkbC0+bWtpZCwgYnVmZmVyLCBsZW4pOwoJCWZyZWVfc3RycmV0KCZzdHIpOwoJfSBlbHNlCgkJYnVmZmVyWzBdID0gJ1wwJzsKCglyZXR1cm4gaHI7Cn0KCiNlbmRpZgoKc3RhdGljIExQV1NUUiB3Y3NjcHluKExQV1NUUiBkZXN0LCBMUENXU1RSIHNvdXJjZSwgc2l6ZV90IGNvdW50KQp7CiBMUENXU1RSIHM7CiBMUFdTVFIgZCA9IGRlc3Q7CgogZm9yKHM9c291cmNlOyBjb3VudCYmKCpkKys9KnMrKyk7ICkKICBjb3VudC0tOwoKIHJldHVybiBkZXN0Owp9CgpzdGF0aWMgdm9pZCBnZXRfc3RycmV0VyhTVFJSRVQqIHN0ciwgY29uc3QgU0hJVEVNSUQqIHNoaWlkLCBMUFdTVFIgYnVmZmVyLCBpbnQgbGVuKQp7CiBzd2l0Y2goc3RyLT51VHlwZSkgewogIGNhc2UgU1RSUkVUX1dTVFI6Cgl3Y3NjcHluKGJ1ZmZlciwgc3RyLT5VTklPTl9NRU1CRVIocE9sZVN0ciksIGxlbik7CglicmVhazsKCiAgY2FzZSBTVFJSRVRfT0ZGU0VUOgoJTXVsdGlCeXRlVG9XaWRlQ2hhcihDUF9BQ1AsIDAsIChMUENTVFIpc2hpaWQrc3RyLT5VTklPTl9NRU1CRVIodU9mZnNldCksIC0xLCBidWZmZXIsIGxlbik7CglicmVhazsKCiAgY2FzZSBTVFJSRVRfQ1NUUjoKCU11bHRpQnl0ZVRvV2lkZUNoYXIoQ1BfQUNQLCAwLCBzdHItPlVOSU9OX01FTUJFUihjU3RyKSwgLTEsIGJ1ZmZlciwgbGVuKTsKIH0KfQoKCnN0YXRpYyBIUkVTVUxUIG5hbWVfZnJvbV9waWRsKElTaGVsbEZvbGRlciogZm9sZGVyLCBMUElURU1JRExJU1QgcGlkbCwgTFBUU1RSIGJ1ZmZlciwgaW50IGxlbiwgU0hHRE5GIGZsYWdzKQp7CglTVFJSRVQgc3RyOwoKCUhSRVNVTFQgaHIgPSBJU2hlbGxGb2xkZXJfR2V0RGlzcGxheU5hbWVPZihmb2xkZXIsIHBpZGwsIGZsYWdzLCAmc3RyKTsKCglpZiAoU1VDQ0VFREVEKGhyKSkgewoJCWdldF9zdHJyZXQoJnN0ciwgJnBpZGwtPm1raWQsIGJ1ZmZlciwgbGVuKTsKCQlmcmVlX3N0cnJldCgmc3RyKTsKCX0gZWxzZQoJCWJ1ZmZlclswXSA9ICdcMCc7CgoJcmV0dXJuIGhyOwp9CgoKc3RhdGljIEhSRVNVTFQgcGF0aF9mcm9tX3BpZGxXKElTaGVsbEZvbGRlciogZm9sZGVyLCBMUElURU1JRExJU1QgcGlkbCwgTFBXU1RSIGJ1ZmZlciwgaW50IGxlbikKewoJU1RSUkVUIHN0cjsKCgkgLyogU0hHRE5fRk9SUEFSU0lORzogZ2V0IGZ1bGwgcGF0aCBvZiBpZCBsaXN0ICovCglIUkVTVUxUIGhyID0gSVNoZWxsRm9sZGVyX0dldERpc3BsYXlOYW1lT2YoZm9sZGVyLCBwaWRsLCBTSEdETl9GT1JQQVJTSU5HLCAmc3RyKTsKCglpZiAoU1VDQ0VFREVEKGhyKSkgewoJCWdldF9zdHJyZXRXKCZzdHIsICZwaWRsLT5ta2lkLCBidWZmZXIsIGxlbik7CgkJZnJlZV9zdHJyZXQoJnN0cik7Cgl9IGVsc2UKCQlidWZmZXJbMF0gPSAnXDAnOwoKCXJldHVybiBocjsKfQoKCiAvKiBjcmVhdGUgYW4gaXRlbSBpZCBsaXN0IGZyb20gYSBmaWxlIHN5c3RlbSBwYXRoICovCgpzdGF0aWMgTFBJVEVNSURMSVNUIGdldF9wYXRoX3BpZGwoTFBUU1RSIHBhdGgsIEhXTkQgaHduZCkKewoJTFBJVEVNSURMSVNUIHBpZGw7CglIUkVTVUxUIGhyOwoJVUxPTkcgbGVuOwoKI2lmZGVmIFVOSUNPREUKCUxQV1NUUiBidWZmZXIgPSBwYXRoOwojZWxzZQoJV0NIQVIgYnVmZmVyW01BWF9QQVRIXTsKCU11bHRpQnl0ZVRvV2lkZUNoYXIoQ1BfQUNQLCAwLCBwYXRoLCAtMSwgYnVmZmVyLCBNQVhfUEFUSCk7CiNlbmRpZgoKCWhyID0gSVNoZWxsRm9sZGVyX1BhcnNlRGlzcGxheU5hbWUoR2xvYmFscy5pRGVza3RvcCwgaHduZCwgTlVMTCwgYnVmZmVyLCAmbGVuLCAmcGlkbCwgTlVMTCk7CglpZiAoRkFJTEVEKGhyKSkKCQlyZXR1cm4gTlVMTDsKCglyZXR1cm4gcGlkbDsKfQoKCiAvKiBjb252ZXJ0IGFuIGl0ZW0gaWQgbGlzdCBmcm9tIHJlbGF0aXZlIHRvIGFic29sdXRlICg9cmVsYXRpdmUgdG8gdGhlIGRlc2t0b3ApIGZvcm1hdCAqLwoKc3RhdGljIExQSVRFTUlETElTVCBnZXRfdG9fYWJzb2x1dGVfcGlkbChFbnRyeSogZW50cnksIEhXTkQgaHduZCkKewoJaWYgKGVudHJ5LT51cCAmJiBlbnRyeS0+dXAtPmV0eXBlPT1FVF9TSEVMTCkgewoJCUxQSVRFTUlETElTVCBpZGwgPSBOVUxMOwoKCQl3aGlsZSAoZW50cnktPnVwKSB7CgkJCWlkbCA9IElMQ29tYmluZShJTENsb25lKGVudHJ5LT5waWRsKSwgaWRsKTsKCQkJZW50cnkgPSBlbnRyeS0+dXA7CgkJfQoKCQlyZXR1cm4gaWRsOwoJfSBlbHNlIGlmIChlbnRyeS0+ZXR5cGUgPT0gRVRfV0lORE9XUykgewoJCVRDSEFSIHBhdGhbTUFYX1BBVEhdOwoKCQlnZXRfcGF0aChlbnRyeSwgcGF0aCk7CgoJCXJldHVybiBnZXRfcGF0aF9waWRsKHBhdGgsIGh3bmQpOwoJfSBlbHNlIGlmIChlbnRyeS0+cGlkbCkKCQlyZXR1cm4gSUxDbG9uZShlbnRyeS0+cGlkbCk7CgoJcmV0dXJuIE5VTEw7Cn0KCgpzdGF0aWMgSElDT04gZXh0cmFjdF9pY29uKElTaGVsbEZvbGRlciogZm9sZGVyLCBMUENJVEVNSURMSVNUIHBpZGwpCnsKCUlFeHRyYWN0SWNvbiogcEV4dHJhY3Q7CgoJaWYgKFNVQ0NFRURFRChJU2hlbGxGb2xkZXJfR2V0VUlPYmplY3RPZihmb2xkZXIsIDAsIDEsIChMUENJVEVNSURMSVNUKikmcGlkbCwgJklJRF9JRXh0cmFjdEljb24sIDAsIChMUFZPSUQqKSZwRXh0cmFjdCkpKSB7CgkJVENIQVIgcGF0aFtfTUFYX1BBVEhdOwoJCXVuc2lnbmVkIGZsYWdzOwoJCUhJQ09OIGhpY29uOwoJCWludCBpZHg7CgoJCWlmIChTVUNDRUVERUQoSUV4dHJhY3RJY29uV19HZXRJY29uTG9jYXRpb24ocEV4dHJhY3QsIEdJTF9GT1JTSEVMTCwgcGF0aCwgX01BWF9QQVRILCAmaWR4LCAmZmxhZ3MpKSkgewoJCQlpZiAoIShmbGFncyAmIEdJTF9OT1RGSUxFTkFNRSkpIHsKCQkJCWlmIChpZHggPT0gLTEpCgkJCQkJaWR4ID0gMDsJLyogc3BlY2lhbCBjYXNlIGZvciBzb21lIGNvbnRyb2wgcGFuZWwgYXBwbGljYXRpb25zICovCgoJCQkJaWYgKChpbnQpRXh0cmFjdEljb25FeChwYXRoLCBpZHgsIDAsICZoaWNvbiwgMSkgPiAwKQoJCQkJCWZsYWdzICY9IH5HSUxfRE9OVENBQ0hFOwoJCQl9IGVsc2UgewoJCQkJSElDT04gaEljb25MYXJnZSA9IDA7CgoJCQkJSFJFU1VMVCBociA9IElFeHRyYWN0SWNvbldfRXh0cmFjdChwRXh0cmFjdCwgcGF0aCwgaWR4LCAmaEljb25MYXJnZSwgJmhpY29uLCBNQUtFTE9ORygwLypHZXRTeXN0ZW1NZXRyaWNzKFNNX0NYSUNPTikqLyxHZXRTeXN0ZW1NZXRyaWNzKFNNX0NYU01JQ09OKSkpOwoKCQkJCWlmIChTVUNDRUVERUQoaHIpKQoJCQkJCURlc3Ryb3lJY29uKGhJY29uTGFyZ2UpOwoJCQl9CgoJCQlyZXR1cm4gaGljb247CgkJfQoJfQoKCXJldHVybiAwOwp9CgoKc3RhdGljIEVudHJ5KiBmaW5kX2VudHJ5X3NoZWxsKEVudHJ5KiBkaXIsIExQQ0lURU1JRExJU1QgcGlkbCkKewoJRW50cnkqIGVudHJ5OwoKCWZvcihlbnRyeT1kaXItPmRvd247IGVudHJ5OyBlbnRyeT1lbnRyeS0+bmV4dCkgewoJCWlmIChlbnRyeS0+cGlkbC0+bWtpZC5jYiA9PSBwaWRsLT5ta2lkLmNiICYmCgkJCSFtZW1jbXAoZW50cnktPnBpZGwsIHBpZGwsIGVudHJ5LT5waWRsLT5ta2lkLmNiKSkKCQkJcmV0dXJuIGVudHJ5OwoJfQoKCXJldHVybiAwOwp9CgpzdGF0aWMgRW50cnkqIHJlYWRfdHJlZV9zaGVsbChSb290KiByb290LCBMUElURU1JRExJU1QgcGlkbCwgU09SVF9PUkRFUiBzb3J0T3JkZXIsIEhXTkQgaHduZCkKewoJRW50cnkqIGVudHJ5ID0gJnJvb3QtPmVudHJ5OwoJRW50cnkqIG5leHQ7CglMUElURU1JRExJU1QgbmV4dF9waWRsID0gcGlkbDsKCUlTaGVsbEZvbGRlciogZm9sZGVyOwoJSVNoZWxsRm9sZGVyKiBjaGlsZCA9IE5VTEw7CglIUkVTVUxUIGhyOwoKCUhDVVJTT1Igb2xkX2N1cnNvciA9IFNldEN1cnNvcihMb2FkQ3Vyc29yKDAsIElEQ19XQUlUKSk7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCgllbnRyeS0+ZXR5cGUgPSBFVF9TSEVMTDsKI2VuZGlmCgoJZm9sZGVyID0gR2xvYmFscy5pRGVza3RvcDsKCgl3aGlsZShlbnRyeSkgewoJCWVudHJ5LT5waWRsID0gbmV4dF9waWRsOwoJCWVudHJ5LT5mb2xkZXIgPSBmb2xkZXI7CgoJCWlmICghcGlkbC0+bWtpZC5jYikKCQkJYnJlYWs7CgoJCSAvKiBjb3B5IGZpcnN0IGVsZW1lbnQgb2YgaXRlbSBpZGxpc3QgKi8KCQluZXh0X3BpZGwgPSBJTWFsbG9jX0FsbG9jKEdsb2JhbHMuaU1hbGxvYywgcGlkbC0+bWtpZC5jYitzaXplb2YoVVNIT1JUKSk7CgkJbWVtY3B5KG5leHRfcGlkbCwgcGlkbCwgcGlkbC0+bWtpZC5jYik7CgkJKChMUElURU1JRExJU1QpKChMUEJZVEUpbmV4dF9waWRsK3BpZGwtPm1raWQuY2IpKS0+bWtpZC5jYiA9IDA7CgoJCWhyID0gSVNoZWxsRm9sZGVyX0JpbmRUb09iamVjdChmb2xkZXIsIG5leHRfcGlkbCwgMCwgJklJRF9JU2hlbGxGb2xkZXIsICh2b2lkKiopJmNoaWxkKTsKCQlpZiAoIVNVQ0NFRURFRChocikpCgkJCWJyZWFrOwoKCQlyZWFkX2RpcmVjdG9yeShlbnRyeSwgTlVMTCwgc29ydE9yZGVyLCBod25kKTsKCgkJaWYgKGVudHJ5LT5kb3duKQoJCQllbnRyeS0+ZXhwYW5kZWQgPSBUUlVFOwoKCQluZXh0ID0gZmluZF9lbnRyeV9zaGVsbChlbnRyeSwgbmV4dF9waWRsKTsKCQlpZiAoIW5leHQpCgkJCWJyZWFrOwoKCQlmb2xkZXIgPSBjaGlsZDsKCQllbnRyeSA9IG5leHQ7CgoJCSAvKiBnbyB0byBuZXh0IGVsZW1lbnQgKi8KCQlwaWRsID0gKExQSVRFTUlETElTVCkgKChMUEJZVEUpcGlkbCtwaWRsLT5ta2lkLmNiKTsKCX0KCglTZXRDdXJzb3Iob2xkX2N1cnNvcik7CgoJcmV0dXJuIGVudHJ5Owp9CgoKc3RhdGljIHZvaWQgZmlsbF93MzJmZGF0YV9zaGVsbChJU2hlbGxGb2xkZXIqIGZvbGRlciwgTFBDSVRFTUlETElTVCBwaWRsLCBTRkdBT0YgYXR0cmlicywgV0lOMzJfRklORF9EQVRBKiB3MzJmZGF0YSkKewoJaWYgKCEoYXR0cmlicyAmIFNGR0FPX0ZJTEVTWVNURU0pIHx8CgkJICBGQUlMRUQoU0hHZXREYXRhRnJvbUlETGlzdChmb2xkZXIsIHBpZGwsIFNIR0RGSUxfRklORERBVEEsIHczMmZkYXRhLCBzaXplb2YoV0lOMzJfRklORF9EQVRBKSkpKSB7CgkJV0lOMzJfRklMRV9BVFRSSUJVVEVfREFUQSBmYWQ7CgkJSURhdGFPYmplY3QqIHBEYXRhT2JqOwoKCQlTVEdNRURJVU0gbWVkaXVtID0gezAsIHswfSwgMH07CgkJRk9STUFURVRDIGZtdCA9IHtHbG9iYWxzLmNmU3RyRk5hbWUsIDAsIERWQVNQRUNUX0NPTlRFTlQsIC0xLCBUWU1FRF9IR0xPQkFMfTsKCgkJSFJFU1VMVCBociA9IElTaGVsbEZvbGRlcl9HZXRVSU9iamVjdE9mKGZvbGRlciwgMCwgMSwgJnBpZGwsICZJSURfSURhdGFPYmplY3QsIDAsIChMUFZPSUQqKSZwRGF0YU9iaik7CgoJCWlmIChTVUNDRUVERUQoaHIpKSB7CgkJCWhyID0gSURhdGFPYmplY3RfR2V0RGF0YShwRGF0YU9iaiwgJmZtdCwgJm1lZGl1bSk7CgoJCQlJRGF0YU9iamVjdF9SZWxlYXNlKHBEYXRhT2JqKTsKCgkJCWlmIChTVUNDRUVERUQoaHIpKSB7CgkJCQlMUENUU1RSIHBhdGggPSAoTFBDVFNUUilHbG9iYWxMb2NrKG1lZGl1bS5VTklPTl9NRU1CRVIoaEdsb2JhbCkpOwoJCQkJVUlOVCBzZW1fb3JnID0gU2V0RXJyb3JNb2RlKFNFTV9GQUlMQ1JJVElDQUxFUlJPUlMpOwoKCQkJCWlmIChHZXRGaWxlQXR0cmlidXRlc0V4KHBhdGgsIEdldEZpbGVFeEluZm9TdGFuZGFyZCwgJmZhZCkpIHsKCQkJCQl3MzJmZGF0YS0+ZHdGaWxlQXR0cmlidXRlcyA9IGZhZC5kd0ZpbGVBdHRyaWJ1dGVzOwoJCQkJCXczMmZkYXRhLT5mdENyZWF0aW9uVGltZSA9IGZhZC5mdENyZWF0aW9uVGltZTsKCQkJCQl3MzJmZGF0YS0+ZnRMYXN0QWNjZXNzVGltZSA9IGZhZC5mdExhc3RBY2Nlc3NUaW1lOwoJCQkJCXczMmZkYXRhLT5mdExhc3RXcml0ZVRpbWUgPSBmYWQuZnRMYXN0V3JpdGVUaW1lOwoKCQkJCQlpZiAoIShmYWQuZHdGaWxlQXR0cmlidXRlcyAmIEZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWSkpIHsKCQkJCQkJdzMyZmRhdGEtPm5GaWxlU2l6ZUxvdyA9IGZhZC5uRmlsZVNpemVMb3c7CgkJCQkJCXczMmZkYXRhLT5uRmlsZVNpemVIaWdoID0gZmFkLm5GaWxlU2l6ZUhpZ2g7CgkJCQkJfQoJCQkJfQoKCQkJCVNldEVycm9yTW9kZShzZW1fb3JnKTsKCgkJCQlHbG9iYWxVbmxvY2sobWVkaXVtLlVOSU9OX01FTUJFUihoR2xvYmFsKSk7CgkJCQlHbG9iYWxGcmVlKG1lZGl1bS5VTklPTl9NRU1CRVIoaEdsb2JhbCkpOwoJCQl9CgkJfQoJfQoKCWlmIChhdHRyaWJzICYgKFNGR0FPX0ZPTERFUnxTRkdBT19IQVNTVUJGT0xERVIpKQoJCXczMmZkYXRhLT5kd0ZpbGVBdHRyaWJ1dGVzIHw9IEZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWTsKCglpZiAoYXR0cmlicyAmIFNGR0FPX1JFQURPTkxZKQoJCXczMmZkYXRhLT5kd0ZpbGVBdHRyaWJ1dGVzIHw9IEZJTEVfQVRUUklCVVRFX1JFQURPTkxZOwoKCWlmIChhdHRyaWJzICYgU0ZHQU9fQ09NUFJFU1NFRCkKCQl3MzJmZGF0YS0+ZHdGaWxlQXR0cmlidXRlcyB8PSBGSUxFX0FUVFJJQlVURV9DT01QUkVTU0VEOwp9CgoKc3RhdGljIHZvaWQgcmVhZF9kaXJlY3Rvcnlfc2hlbGwoRW50cnkqIGRpciwgSFdORCBod25kKQp7CglJU2hlbGxGb2xkZXIqIGZvbGRlciA9IGRpci0+Zm9sZGVyOwoJaW50IGxldmVsID0gZGlyLT5sZXZlbCArIDE7CglIUkVTVUxUIGhyOwoKCUlTaGVsbEZvbGRlciogY2hpbGQ7CglJRW51bUlETGlzdCogaWRsaXN0OwoKCUVudHJ5KiBmaXJzdF9lbnRyeSA9IE5VTEw7CglFbnRyeSogbGFzdCA9IE5VTEw7CglFbnRyeSogZW50cnk7CgoJaWYgKCFmb2xkZXIpCgkJcmV0dXJuOwoKCWhyID0gSVNoZWxsRm9sZGVyX0VudW1PYmplY3RzKGZvbGRlciwgaHduZCwgU0hDT05URl9GT0xERVJTfFNIQ09OVEZfTk9ORk9MREVSU3xTSENPTlRGX0lOQ0xVREVISURERU58U0hDT05URl9TSEFSRUFCTEV8U0hDT05URl9TVE9SQUdFLCAmaWRsaXN0KTsKCglpZiAoU1VDQ0VFREVEKGhyKSkgewoJCWZvcig7OykgewojZGVmaW5lCUZFVENIX0lURU1fQ09VTlQJMzIKCQkJTFBJVEVNSURMSVNUIHBpZGxzW0ZFVENIX0lURU1fQ09VTlRdOwoJCQlTRkdBT0YgYXR0cmliczsKCQkJVUxPTkcgY250ID0gMDsKCQkJVUxPTkcgbjsKCgkJCW1lbXNldChwaWRscywgMCwgc2l6ZW9mKHBpZGxzKSk7CgoJCQlociA9IElFbnVtSURMaXN0X05leHQoaWRsaXN0LCBGRVRDSF9JVEVNX0NPVU5ULCBwaWRscywgJmNudCk7CgkJCWlmICghU1VDQ0VFREVEKGhyKSkKCQkJCWJyZWFrOwoKCQkJaWYgKGhyID09IFNfRkFMU0UpCgkJCQlicmVhazsKCgkJCWZvcihuPTA7IG48Y250OyArK24pIHsKCQkJCWVudHJ5ID0gYWxsb2NfZW50cnkoKTsKCgkJCQlpZiAoIWZpcnN0X2VudHJ5KQoJCQkJCWZpcnN0X2VudHJ5ID0gZW50cnk7CgoJCQkJaWYgKGxhc3QpCgkJCQkJbGFzdC0+bmV4dCA9IGVudHJ5OwoKCQkJCW1lbXNldCgmZW50cnktPmRhdGEsIDAsIHNpemVvZihXSU4zMl9GSU5EX0RBVEEpKTsKCQkJCWVudHJ5LT5iaGZpX3ZhbGlkID0gRkFMU0U7CgoJCQkJYXR0cmlicyA9IH5TRkdBT19GSUxFU1lTVEVNOwkvKlNGR0FPX0hBU1NVQkZPTERFUnxTRkdBT19GT0xERVI7IFNGR0FPX0ZJTEVTWVNURU0gc29yZ3QgZGFm/HIsIGRh3yAiTXkgRG9jdW1lbnRzIiBhbnN0YXR0IHZvbiAiTWFydGluJ3MgRG9jdW1lbnRzIiBhbmdlemVpZ3Qgd2lyZCAqLwoKCQkJCWhyID0gSVNoZWxsRm9sZGVyX0dldEF0dHJpYnV0ZXNPZihmb2xkZXIsIDEsIChMUENJVEVNSURMSVNUKikmcGlkbHNbbl0sICZhdHRyaWJzKTsKCgkJCQlpZiAoU1VDQ0VFREVEKGhyKSkgewoJCQkJCWlmIChhdHRyaWJzICE9IChTRkdBT0YpflNGR0FPX0ZJTEVTWVNURU0pIHsKCQkJCQkJZmlsbF93MzJmZGF0YV9zaGVsbChmb2xkZXIsIHBpZGxzW25dLCBhdHRyaWJzLCAmZW50cnktPmRhdGEpOwoKCQkJCQkJZW50cnktPmJoZmlfdmFsaWQgPSBUUlVFOwoJCQkJCX0gZWxzZQoJCQkJCQlhdHRyaWJzID0gMDsKCQkJCX0gZWxzZQoJCQkJCWF0dHJpYnMgPSAwOwoKCQkJCWVudHJ5LT5waWRsID0gcGlkbHNbbl07CgoJCQkJaWYgKGVudHJ5LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpIHsKCQkJCQlociA9IElTaGVsbEZvbGRlcl9CaW5kVG9PYmplY3QoZm9sZGVyLCBwaWRsc1tuXSwgMCwgJklJRF9JU2hlbGxGb2xkZXIsICh2b2lkKiopJmNoaWxkKTsKCgkJCQkJaWYgKFNVQ0NFRURFRChocikpCgkJCQkJCWVudHJ5LT5mb2xkZXIgPSBjaGlsZDsKCQkJCQllbHNlCgkJCQkJCWVudHJ5LT5mb2xkZXIgPSBOVUxMOwoJCQkJfQoJCQkJZWxzZQoJCQkJCWVudHJ5LT5mb2xkZXIgPSBOVUxMOwoKCQkJCWlmICghZW50cnktPmRhdGEuY0ZpbGVOYW1lWzBdKQoJCQkJCS8qaHIgPSAqL25hbWVfZnJvbV9waWRsKGZvbGRlciwgcGlkbHNbbl0sIGVudHJ5LT5kYXRhLmNGaWxlTmFtZSwgTUFYX1BBVEgsIC8qU0hHRE5fSU5GT0xERVIqLzB4MjAwMC8qMHgyMDAwPVNIR0ROX0lOQ0xVREVfTk9ORklMRVNZUyovKTsKCgkJCQkgLyogZ2V0IGRpc3BsYXkgaWNvbnMgZm9yIGZpbGVzIGFuZCB2aXJ0dWFsIG9iamVjdHMgKi8KCQkJCWlmICghKGVudHJ5LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpIHx8CgkJCQkJIShhdHRyaWJzICYgU0ZHQU9fRklMRVNZU1RFTSkpIHsKCQkJCQllbnRyeS0+aGljb24gPSBleHRyYWN0X2ljb24oZm9sZGVyLCBwaWRsc1tuXSk7CgoJCQkJCWlmICghZW50cnktPmhpY29uKQoJCQkJCQllbnRyeS0+aGljb24gPSAoSElDT04pLTE7CS8qIGRvbid0IHRyeSBhZ2FpbiBsYXRlciAqLwoJCQkJfQoKCQkJCWVudHJ5LT5kb3duID0gTlVMTDsKCQkJCWVudHJ5LT51cCA9IGRpcjsKCQkJCWVudHJ5LT5leHBhbmRlZCA9IEZBTFNFOwoJCQkJZW50cnktPnNjYW5uZWQgPSBGQUxTRTsKCQkJCWVudHJ5LT5sZXZlbCA9IGxldmVsOwoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJCQkJZW50cnktPmV0eXBlID0gRVRfU0hFTEw7CgkJCQllbnRyeS0+YmhmaV92YWxpZCA9IEZBTFNFOwojZW5kaWYKCgkJCQlsYXN0ID0gZW50cnk7CgkJCX0KCQl9CgoJCUlFbnVtSURMaXN0X1JlbGVhc2UoaWRsaXN0KTsKCX0KCglpZiAobGFzdCkKCQlsYXN0LT5uZXh0ID0gTlVMTDsKCglkaXItPmRvd24gPSBmaXJzdF9lbnRyeTsKCWRpci0+c2Nhbm5lZCA9IFRSVUU7Cn0KCiNlbmRpZiAvKiBfU0hFTExfRk9MREVSUyAqLwoKCi8qIHNvcnQgb3JkZXIgZm9yIGRpZmZlcmVudCBkaXJlY3RvcnkvZmlsZSB0eXBlcyAqLwplbnVtIFRZUEVfT1JERVIgewoJVE9fRElSID0gMCwKCVRPX0RPVCA9IDEsCglUT19ET1RET1QgPSAyLAoJVE9fT1RIRVJfRElSID0gMywKCVRPX0ZJTEUgPSA0Cn07CgovKiBkaXN0aW5ndWlzaCBiZXR3ZWVuICIuIiwgIi4uIiBhbmQgYW55IG90aGVyIGRpcmVjdG9yeSBuYW1lcyAqLwpzdGF0aWMgaW50IFR5cGVPcmRlckZyb21EaXJuYW1lKExQQ1RTVFIgbmFtZSkKewoJaWYgKG5hbWVbMF0gPT0gJy4nKSB7CgkJaWYgKG5hbWVbMV0gPT0gJ1wwJykKCQkJcmV0dXJuIFRPX0RPVDsJLyogIi4iICovCgoJCWlmIChuYW1lWzFdPT0nLicgJiYgbmFtZVsyXT09J1wwJykKCQkJcmV0dXJuIFRPX0RPVERPVDsJLyogIi4uIiAqLwoJfQoKCXJldHVybiBUT19PVEhFUl9ESVI7CS8qIGFueXRoaW5nIGVsc2UgKi8KfQoKLyogZGlyZWN0b3JpZXMgZmlyc3QuLi4gKi8Kc3RhdGljIGludCBjb21wYXJlVHlwZShjb25zdCBXSU4zMl9GSU5EX0RBVEEqIGZkMSwgY29uc3QgV0lOMzJfRklORF9EQVRBKiBmZDIpCnsKCWludCBvcmRlcjEgPSBmZDEtPmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlk/IFRPX0RJUjogVE9fRklMRTsKCWludCBvcmRlcjIgPSBmZDItPmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlk/IFRPX0RJUjogVE9fRklMRTsKCgkvKiBIYW5kbGUgIi4iIGFuZCAiLi4iIGFzIHNwZWNpYWwgY2FzZSBhbmQgbW92ZSB0aGVtIGF0IHRoZSB2ZXJ5IGZpcnN0IGJlZ2lubmluZy4gKi8KCWlmIChvcmRlcjE9PVRPX0RJUiAmJiBvcmRlcjI9PVRPX0RJUikgewoJCW9yZGVyMSA9IFR5cGVPcmRlckZyb21EaXJuYW1lKGZkMS0+Y0ZpbGVOYW1lKTsKCQlvcmRlcjIgPSBUeXBlT3JkZXJGcm9tRGlybmFtZShmZDItPmNGaWxlTmFtZSk7Cgl9CgoJcmV0dXJuIG9yZGVyMj09b3JkZXIxPyAwOiBvcmRlcjE8b3JkZXIyPyAtMTogMTsKfQoKCnN0YXRpYyBpbnQgY29tcGFyZU5hbWUoY29uc3Qgdm9pZCogYXJnMSwgY29uc3Qgdm9pZCogYXJnMikKewoJY29uc3QgV0lOMzJfRklORF9EQVRBKiBmZDEgPSAmKCooY29uc3QgRW50cnkqIGNvbnN0KilhcmcxKS0+ZGF0YTsKCWNvbnN0IFdJTjMyX0ZJTkRfREFUQSogZmQyID0gJigqKGNvbnN0IEVudHJ5KiBjb25zdCopYXJnMiktPmRhdGE7CgoJaW50IGNtcCA9IGNvbXBhcmVUeXBlKGZkMSwgZmQyKTsKCWlmIChjbXApCgkJcmV0dXJuIGNtcDsKCglyZXR1cm4gbHN0cmNtcGkoZmQxLT5jRmlsZU5hbWUsIGZkMi0+Y0ZpbGVOYW1lKTsKfQoKc3RhdGljIGludCBjb21wYXJlRXh0KGNvbnN0IHZvaWQqIGFyZzEsIGNvbnN0IHZvaWQqIGFyZzIpCnsKCWNvbnN0IFdJTjMyX0ZJTkRfREFUQSogZmQxID0gJigqKGNvbnN0IEVudHJ5KiBjb25zdCopYXJnMSktPmRhdGE7Cgljb25zdCBXSU4zMl9GSU5EX0RBVEEqIGZkMiA9ICYoKihjb25zdCBFbnRyeSogY29uc3QqKWFyZzIpLT5kYXRhOwoJY29uc3QgVENIQVIgKm5hbWUxLCAqbmFtZTIsICpleHQxLCAqZXh0MjsKCglpbnQgY21wID0gY29tcGFyZVR5cGUoZmQxLCBmZDIpOwoJaWYgKGNtcCkKCQlyZXR1cm4gY21wOwoKCW5hbWUxID0gZmQxLT5jRmlsZU5hbWU7CgluYW1lMiA9IGZkMi0+Y0ZpbGVOYW1lOwoKCWV4dDEgPSBfdGNzcmNocihuYW1lMSwgJy4nKTsKCWV4dDIgPSBfdGNzcmNocihuYW1lMiwgJy4nKTsKCglpZiAoZXh0MSkKCQlleHQxKys7CgllbHNlCgkJZXh0MSA9IHNFbXB0eTsKCglpZiAoZXh0MikKCQlleHQyKys7CgllbHNlCgkJZXh0MiA9IHNFbXB0eTsKCgljbXAgPSBsc3RyY21waShleHQxLCBleHQyKTsKCWlmIChjbXApCgkJcmV0dXJuIGNtcDsKCglyZXR1cm4gbHN0cmNtcGkobmFtZTEsIG5hbWUyKTsKfQoKc3RhdGljIGludCBjb21wYXJlU2l6ZShjb25zdCB2b2lkKiBhcmcxLCBjb25zdCB2b2lkKiBhcmcyKQp7Cgljb25zdCBXSU4zMl9GSU5EX0RBVEEqIGZkMSA9ICYoKihjb25zdCBFbnRyeSogY29uc3QqKWFyZzEpLT5kYXRhOwoJY29uc3QgV0lOMzJfRklORF9EQVRBKiBmZDIgPSAmKCooY29uc3QgRW50cnkqIGNvbnN0KilhcmcyKS0+ZGF0YTsKCglpbnQgY21wID0gY29tcGFyZVR5cGUoZmQxLCBmZDIpOwoJaWYgKGNtcCkKCQlyZXR1cm4gY21wOwoKCWNtcCA9IGZkMi0+bkZpbGVTaXplSGlnaCAtIGZkMS0+bkZpbGVTaXplSGlnaDsKCglpZiAoY21wIDwgMCkKCQlyZXR1cm4gLTE7CgllbHNlIGlmIChjbXAgPiAwKQoJCXJldHVybiAxOwoKCWNtcCA9IGZkMi0+bkZpbGVTaXplTG93IC0gZmQxLT5uRmlsZVNpemVMb3c7CgoJcmV0dXJuIGNtcDwwPyAtMTogY21wPjA/IDE6IDA7Cn0KCnN0YXRpYyBpbnQgY29tcGFyZURhdGUoY29uc3Qgdm9pZCogYXJnMSwgY29uc3Qgdm9pZCogYXJnMikKewoJY29uc3QgV0lOMzJfRklORF9EQVRBKiBmZDEgPSAmKCooY29uc3QgRW50cnkqIGNvbnN0KilhcmcxKS0+ZGF0YTsKCWNvbnN0IFdJTjMyX0ZJTkRfREFUQSogZmQyID0gJigqKGNvbnN0IEVudHJ5KiBjb25zdCopYXJnMiktPmRhdGE7CgoJaW50IGNtcCA9IGNvbXBhcmVUeXBlKGZkMSwgZmQyKTsKCWlmIChjbXApCgkJcmV0dXJuIGNtcDsKCglyZXR1cm4gQ29tcGFyZUZpbGVUaW1lKCZmZDItPmZ0TGFzdFdyaXRlVGltZSwgJmZkMS0+ZnRMYXN0V3JpdGVUaW1lKTsKfQoKCnN0YXRpYyBpbnQgKCpzb3J0RnVuY3Rpb25zW10pKGNvbnN0IHZvaWQqIGFyZzEsIGNvbnN0IHZvaWQqIGFyZzIpID0gewoJY29tcGFyZU5hbWUsCS8qIFNPUlRfTkFNRSAqLwoJY29tcGFyZUV4dCwJCS8qIFNPUlRfRVhUICovCgljb21wYXJlU2l6ZSwJLyogU09SVF9TSVpFICovCgljb21wYXJlRGF0ZQkJLyogU09SVF9EQVRFICovCn07CgoKc3RhdGljIHZvaWQgU29ydERpcmVjdG9yeShFbnRyeSogZGlyLCBTT1JUX09SREVSIHNvcnRPcmRlcikKewoJRW50cnkqIGVudHJ5ID0gZGlyLT5kb3duOwoJRW50cnkqKiBhcnJheSwgKipwOwoJaW50IGxlbjsKCglsZW4gPSAwOwoJZm9yKGVudHJ5PWRpci0+ZG93bjsgZW50cnk7IGVudHJ5PWVudHJ5LT5uZXh0KQoJCWxlbisrOwoKCWlmIChsZW4pIHsKCQlhcnJheSA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBsZW4qc2l6ZW9mKEVudHJ5KikpOwoKCQlwID0gYXJyYXk7CgkJZm9yKGVudHJ5PWRpci0+ZG93bjsgZW50cnk7IGVudHJ5PWVudHJ5LT5uZXh0KQoJCQkqcCsrID0gZW50cnk7CgoJCS8qIGNhbGwgcXNvcnQgd2l0aCB0aGUgYXBwcm9wcmlhdGUgY29tcGFyZSBmdW5jdGlvbiAqLwoJCXFzb3J0KGFycmF5LCBsZW4sIHNpemVvZihhcnJheVswXSksIHNvcnRGdW5jdGlvbnNbc29ydE9yZGVyXSk7CgoJCWRpci0+ZG93biA9IGFycmF5WzBdOwoKCQlmb3IocD1hcnJheTsgLS1sZW47IHArKykKCQkJcFswXS0+bmV4dCA9IHBbMV07CgoJCSgqcCktPm5leHQgPSAwOwoKICAgICAgICAgICAgICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGFycmF5KTsKCX0KfQoKCnN0YXRpYyB2b2lkIHJlYWRfZGlyZWN0b3J5KEVudHJ5KiBkaXIsIExQQ1RTVFIgcGF0aCwgU09SVF9PUkRFUiBzb3J0T3JkZXIsIEhXTkQgaHduZCkKewoJVENIQVIgYnVmZmVyW01BWF9QQVRIXTsKCUVudHJ5KiBlbnRyeTsKCUxQQ1RTVFIgczsKCVBUU1RSIGQ7CgojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCWlmIChkaXItPmV0eXBlID09IEVUX1NIRUxMKQoJewoJCXJlYWRfZGlyZWN0b3J5X3NoZWxsKGRpciwgaHduZCk7CgoJCWlmIChHbG9iYWxzLnByZXNjYW5fbm9kZSkgewoJCQlzID0gcGF0aDsKCQkJZCA9IGJ1ZmZlcjsKCgkJCXdoaWxlKCpzKQoJCQkJKmQrKyA9ICpzKys7CgoJCQkqZCsrID0gJ1xcJzsKCgkJCWZvcihlbnRyeT1kaXItPmRvd247IGVudHJ5OyBlbnRyeT1lbnRyeS0+bmV4dCkKCQkJCWlmIChlbnRyeS0+ZGF0YS5kd0ZpbGVBdHRyaWJ1dGVzICYgRklMRV9BVFRSSUJVVEVfRElSRUNUT1JZKSB7CgkJCQkJcmVhZF9kaXJlY3Rvcnlfc2hlbGwoZW50cnksIGh3bmQpOwoJCQkJCVNvcnREaXJlY3RvcnkoZW50cnksIHNvcnRPcmRlcik7CgkJCQl9CgkJfQoJfQoJZWxzZQojZW5kaWYKI2lmICFkZWZpbmVkKF9OT19FWFRFTlNJT05TKSAmJiBkZWZpbmVkKF9fV0lORV9fKQoJaWYgKGRpci0+ZXR5cGUgPT0gRVRfVU5JWCkKCXsKCQlyZWFkX2RpcmVjdG9yeV91bml4KGRpciwgcGF0aCk7CgoJCWlmIChHbG9iYWxzLnByZXNjYW5fbm9kZSkgewoJCQlzID0gcGF0aDsKCQkJZCA9IGJ1ZmZlcjsKCgkJCXdoaWxlKCpzKQoJCQkJKmQrKyA9ICpzKys7CgoJCQkqZCsrID0gJy8nOwoKCQkJZm9yKGVudHJ5PWRpci0+ZG93bjsgZW50cnk7IGVudHJ5PWVudHJ5LT5uZXh0KQoJCQkJaWYgKGVudHJ5LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpIHsKCQkJCQlsc3RyY3B5KGQsIGVudHJ5LT5kYXRhLmNGaWxlTmFtZSk7CgkJCQkJcmVhZF9kaXJlY3RvcnlfdW5peChlbnRyeSwgYnVmZmVyKTsKCQkJCQlTb3J0RGlyZWN0b3J5KGVudHJ5LCBzb3J0T3JkZXIpOwoJCQkJfQoJCX0KCX0KCWVsc2UKI2VuZGlmCgl7CgkJcmVhZF9kaXJlY3Rvcnlfd2luKGRpciwgcGF0aCk7CgoJCWlmIChHbG9iYWxzLnByZXNjYW5fbm9kZSkgewoJCQlzID0gcGF0aDsKCQkJZCA9IGJ1ZmZlcjsKCgkJCXdoaWxlKCpzKQoJCQkJKmQrKyA9ICpzKys7CgoJCQkqZCsrID0gJ1xcJzsKCgkJCWZvcihlbnRyeT1kaXItPmRvd247IGVudHJ5OyBlbnRyeT1lbnRyeS0+bmV4dCkKCQkJCWlmIChlbnRyeS0+ZGF0YS5kd0ZpbGVBdHRyaWJ1dGVzICYgRklMRV9BVFRSSUJVVEVfRElSRUNUT1JZKSB7CgkJCQkJbHN0cmNweShkLCBlbnRyeS0+ZGF0YS5jRmlsZU5hbWUpOwoJCQkJCXJlYWRfZGlyZWN0b3J5X3dpbihlbnRyeSwgYnVmZmVyKTsKCQkJCQlTb3J0RGlyZWN0b3J5KGVudHJ5LCBzb3J0T3JkZXIpOwoJCQkJfQoJCX0KCX0KCglTb3J0RGlyZWN0b3J5KGRpciwgc29ydE9yZGVyKTsKfQoKCnN0YXRpYyBFbnRyeSogcmVhZF90cmVlKFJvb3QqIHJvb3QsIExQQ1RTVFIgcGF0aCwgTFBJVEVNSURMSVNUIHBpZGwsIExQVFNUUiBkcnYsIFNPUlRfT1JERVIgc29ydE9yZGVyLCBIV05EIGh3bmQpCnsKI2lmICFkZWZpbmVkKF9OT19FWFRFTlNJT05TKSAmJiBkZWZpbmVkKF9fV0lORV9fKQoJc3RhdGljIGNvbnN0IFRDSEFSIHNTbGFzaFtdID0geycvJywgJ1wwJ307CiNlbmRpZgoJc3RhdGljIGNvbnN0IFRDSEFSIHNCYWNrc2xhc2hbXSA9IHsnXFwnLCAnXDAnfTsKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJaWYgKHBpZGwpCgl7CgkJIC8qIHJlYWQgc2hlbGwgbmFtZXNwYWNlIHRyZWUgKi8KCQlyb290LT5kcml2ZV90eXBlID0gRFJJVkVfVU5LTk9XTjsKCQlkcnZbMF0gPSAnXFwnOwoJCWRydlsxXSA9ICdcMCc7CgkJbG9hZF9zdHJpbmcocm9vdC0+dm9sbmFtZSwgSURTX0RFU0tUT1ApOwoJCXJvb3QtPmZzX2ZsYWdzID0gMDsKCQlsb2FkX3N0cmluZyhyb290LT5mcywgSURTX1NIRUxMKTsKCgkJcmV0dXJuIHJlYWRfdHJlZV9zaGVsbChyb290LCBwaWRsLCBzb3J0T3JkZXIsIGh3bmQpOwoJfQoJZWxzZQojZW5kaWYKI2lmICFkZWZpbmVkKF9OT19FWFRFTlNJT05TKSAmJiBkZWZpbmVkKF9fV0lORV9fKQoJaWYgKCpwYXRoID09ICcvJykKCXsKCQkgLyogcmVhZCB1bml4IGZpbGUgc3lzdGVtIHRyZWUgKi8KCQlyb290LT5kcml2ZV90eXBlID0gR2V0RHJpdmVUeXBlKHBhdGgpOwoKCQlsc3RyY2F0KGRydiwgc1NsYXNoKTsKCQlsb2FkX3N0cmluZyhyb290LT52b2xuYW1lLCBJRFNfUk9PVF9GUyk7CgkJcm9vdC0+ZnNfZmxhZ3MgPSAwOwoJCWxvYWRfc3RyaW5nKHJvb3QtPmZzLCBJRFNfVU5JWEZTKTsKCgkJbHN0cmNweShyb290LT5wYXRoLCBzU2xhc2gpOwoKCQlyZXR1cm4gcmVhZF90cmVlX3VuaXgocm9vdCwgcGF0aCwgc29ydE9yZGVyLCBod25kKTsKCX0KI2VuZGlmCgoJIC8qIHJlYWQgV0lOMzIgZmlsZSBzeXN0ZW0gdHJlZSAqLwoJcm9vdC0+ZHJpdmVfdHlwZSA9IEdldERyaXZlVHlwZShwYXRoKTsKCglsc3RyY2F0KGRydiwgc0JhY2tzbGFzaCk7CglHZXRWb2x1bWVJbmZvcm1hdGlvbihkcnYsIHJvb3QtPnZvbG5hbWUsIF9NQVhfRk5BTUUsIDAsIDAsICZyb290LT5mc19mbGFncywgcm9vdC0+ZnMsIF9NQVhfRElSKTsKCglsc3RyY3B5KHJvb3QtPnBhdGgsIGRydik7CgoJcmV0dXJuIHJlYWRfdHJlZV93aW4ocm9vdCwgcGF0aCwgc29ydE9yZGVyLCBod25kKTsKfQoKCi8qIGZsYWdzIHRvIGZpbHRlciBkaWZmZXJlbnQgZmlsZSB0eXBlcyAqLwplbnVtIFRZUEVfRklMVEVSIHsKCVRGX0RJUkVDVE9SSUVTCT0gMHgwMSwKCVRGX1BST0dSQU1TCQk9IDB4MDIsCglURl9ET0NVTUVOVFMJPSAweDA0LAoJVEZfT1RIRVJTCQk9IDB4MDgsCglURl9ISURERU4JCT0gMHgxMCwKCVRGX0FMTAkJCT0gMHgxRgp9OwoKCnN0YXRpYyBDaGlsZFduZCogYWxsb2NfY2hpbGRfd2luZG93KExQQ1RTVFIgcGF0aCwgTFBJVEVNSURMSVNUIHBpZGwsIEhXTkQgaHduZCkKewoJVENIQVIgZHJ2W19NQVhfRFJJVkUrMV0sIGRpcltfTUFYX0RJUl0sIG5hbWVbX01BWF9GTkFNRV0sIGV4dFtfTUFYX0VYVF07CglUQ0hBUiBkaXJfcGF0aFtNQVhfUEFUSF07CglUQ0hBUiBiMVtCVUZGRVJfTEVOXTsKCXN0YXRpYyBjb25zdCBUQ0hBUiBzQXN0ZXJpY3NbXSA9IHsnKicsICdcMCd9OwoKCUNoaWxkV25kKiBjaGlsZCA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBzaXplb2YoQ2hpbGRXbmQpKTsKCVJvb3QqIHJvb3QgPSAmY2hpbGQtPnJvb3Q7CglFbnRyeSogZW50cnk7CgoJbWVtc2V0KGNoaWxkLCAwLCBzaXplb2YoQ2hpbGRXbmQpKTsKCgljaGlsZC0+bGVmdC50cmVlUGFuZSA9IFRSVUU7CgljaGlsZC0+bGVmdC52aXNpYmxlX2NvbHMgPSAwOwoKCWNoaWxkLT5yaWdodC50cmVlUGFuZSA9IEZBTFNFOwojaWZuZGVmIF9OT19FWFRFTlNJT05TCgljaGlsZC0+cmlnaHQudmlzaWJsZV9jb2xzID0gQ09MX1NJWkV8Q09MX0RBVEV8Q09MX1RJTUV8Q09MX0FUVFJJQlVURVN8Q09MX0lOREVYfENPTF9MSU5LUzsKI2Vsc2UKCWNoaWxkLT5yaWdodC52aXNpYmxlX2NvbHMgPSBDT0xfU0laRXxDT0xfREFURXxDT0xfVElNRXxDT0xfQVRUUklCVVRFUzsKI2VuZGlmCgoJY2hpbGQtPnBvcy5sZW5ndGggPSBzaXplb2YoV0lORE9XUExBQ0VNRU5UKTsKCWNoaWxkLT5wb3MuZmxhZ3MgPSAwOwoJY2hpbGQtPnBvcy5zaG93Q21kID0gU1dfU0hPV05PUk1BTDsKCWNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi5sZWZ0ID0gQ1dfVVNFREVGQVVMVDsKCWNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi50b3AgPSBDV19VU0VERUZBVUxUOwoJY2hpbGQtPnBvcy5yY05vcm1hbFBvc2l0aW9uLnJpZ2h0ID0gQ1dfVVNFREVGQVVMVDsKCWNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi5ib3R0b20gPSBDV19VU0VERUZBVUxUOwoKCWNoaWxkLT5mb2N1c19wYW5lID0gMDsKCWNoaWxkLT5zcGxpdF9wb3MgPSBERUZBVUxUX1NQTElUX1BPUzsKCWNoaWxkLT5zb3J0T3JkZXIgPSBTT1JUX05BTUU7CgljaGlsZC0+aGVhZGVyX3dkdGhzX29rID0gRkFMU0U7CgoJaWYgKHBhdGgpCgl7CgkJbHN0cmNweShjaGlsZC0+cGF0aCwgcGF0aCk7CgoJCV90c3BsaXRwYXRoKHBhdGgsIGRydiwgZGlyLCBuYW1lLCBleHQpOwoJfQoKCWxzdHJjcHkoY2hpbGQtPmZpbHRlcl9wYXR0ZXJuLCBzQXN0ZXJpY3MpOwoJY2hpbGQtPmZpbHRlcl9mbGFncyA9IFRGX0FMTDsKCglyb290LT5lbnRyeS5sZXZlbCA9IDA7CgoJbHN0cmNweShkaXJfcGF0aCwgZHJ2KTsKCWxzdHJjYXQoZGlyX3BhdGgsIGRpcik7CgllbnRyeSA9IHJlYWRfdHJlZShyb290LCBkaXJfcGF0aCwgcGlkbCwgZHJ2LCBjaGlsZC0+c29ydE9yZGVyLCBod25kKTsKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJaWYgKHJvb3QtPmVudHJ5LmV0eXBlID09IEVUX1NIRUxMKQoJCWxvYWRfc3RyaW5nKHJvb3QtPmVudHJ5LmRhdGEuY0ZpbGVOYW1lLCBJRFNfREVTS1RPUCk7CgllbHNlCiNlbmRpZgoJCXdzcHJpbnRmKHJvb3QtPmVudHJ5LmRhdGEuY0ZpbGVOYW1lLCBSUyhiMSxJRFNfVElUTEVGTVQpLCBkcnYsIHJvb3QtPmZzKTsKCglyb290LT5lbnRyeS5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgPSBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlk7CgoJY2hpbGQtPmxlZnQucm9vdCA9ICZyb290LT5lbnRyeTsKCWNoaWxkLT5yaWdodC5yb290ID0gTlVMTDsKCglzZXRfY3VyZGlyKGNoaWxkLCBlbnRyeSwgMCwgaHduZCk7CgoJcmV0dXJuIGNoaWxkOwp9CgoKLyogZnJlZSBhbGwgbWVtb3J5IGFzc29jaWF0ZWQgd2l0aCBhIGNoaWxkIHdpbmRvdyAqLwpzdGF0aWMgdm9pZCBmcmVlX2NoaWxkX3dpbmRvdyhDaGlsZFduZCogY2hpbGQpCnsKCWZyZWVfZW50cmllcygmY2hpbGQtPnJvb3QuZW50cnkpOwoJSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgY2hpbGQpOwp9CgoKLyogZ2V0IGZ1bGwgcGF0aCBvZiBzcGVjaWZpZWQgZGlyZWN0b3J5IGVudHJ5ICovCnN0YXRpYyB2b2lkIGdldF9wYXRoKEVudHJ5KiBkaXIsIFBUU1RSIHBhdGgpCnsKCUVudHJ5KiBlbnRyeTsKCWludCBsZW4gPSAwOwoJaW50IGxldmVsID0gMDsKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJaWYgKGRpci0+ZXR5cGUgPT0gRVRfU0hFTEwpCgl7CgkJU0ZHQU9GIGF0dHJpYnM7CgkJSFJFU1VMVCBociA9IFNfT0s7CgoJCXBhdGhbMF0gPSAnXDAnOwoKCQlhdHRyaWJzID0gMDsKCgkJaWYgKGRpci0+Zm9sZGVyKQoJCQlociA9IElTaGVsbEZvbGRlcl9HZXRBdHRyaWJ1dGVzT2YoZGlyLT5mb2xkZXIsIDEsIChMUENJVEVNSURMSVNUKikmZGlyLT5waWRsLCAmYXR0cmlicyk7CgoJCWlmIChTVUNDRUVERUQoaHIpICYmIChhdHRyaWJzJlNGR0FPX0ZJTEVTWVNURU0pKSB7CgkJCUlTaGVsbEZvbGRlciogcGFyZW50ID0gZGlyLT51cD8gZGlyLT51cC0+Zm9sZGVyOiBHbG9iYWxzLmlEZXNrdG9wOwoKCQkJaHIgPSBwYXRoX2Zyb21fcGlkbChwYXJlbnQsIGRpci0+cGlkbCwgcGF0aCwgTUFYX1BBVEgpOwoJCX0KCX0KCWVsc2UKI2VuZGlmCgl7CgkJZm9yKGVudHJ5PWRpcjsgZW50cnk7IGxldmVsKyspIHsKCQkJTFBDVFNUUiBuYW1lOwoJCQlpbnQgbDsKCgkJCXsKCQkJCUxQQ1RTVFIgczsKCQkJCW5hbWUgPSBlbnRyeS0+ZGF0YS5jRmlsZU5hbWU7CgkJCQlzID0gbmFtZTsKCgkJCQlmb3IobD0wOyAqcyAmJiAqcyAhPSAnLycgJiYgKnMgIT0gJ1xcJzsgcysrKQoJCQkJCWwrKzsKCQkJfQoKCQkJaWYgKGVudHJ5LT51cCkgewoJCQkJaWYgKGwgPiAwKSB7CgkJCQkJbWVtbW92ZShwYXRoK2wrMSwgcGF0aCwgbGVuKnNpemVvZihUQ0hBUikpOwoJCQkJCW1lbWNweShwYXRoKzEsIG5hbWUsIGwqc2l6ZW9mKFRDSEFSKSk7CgkJCQkJbGVuICs9IGwrMTsKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJCQlpZiAoZW50cnktPmV0eXBlID09IEVUX1VOSVgpCgkJCQkJCXBhdGhbMF0gPSAnLyc7CgkJCQkJZWxzZQojZW5kaWYKCQkJCQlwYXRoWzBdID0gJ1xcJzsKCQkJCX0KCgkJCQllbnRyeSA9IGVudHJ5LT51cDsKCQkJfSBlbHNlIHsKCQkJCW1lbW1vdmUocGF0aCtsLCBwYXRoLCBsZW4qc2l6ZW9mKFRDSEFSKSk7CgkJCQltZW1jcHkocGF0aCwgbmFtZSwgbCpzaXplb2YoVENIQVIpKTsKCQkJCWxlbiArPSBsOwoJCQkJYnJlYWs7CgkJCX0KCQl9CgoJCWlmICghbGV2ZWwpIHsKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJCQlpZiAoZW50cnktPmV0eXBlID09IEVUX1VOSVgpCgkJCQlwYXRoW2xlbisrXSA9ICcvJzsKCQkJZWxzZQojZW5kaWYKCQkJCXBhdGhbbGVuKytdID0gJ1xcJzsKCQl9CgoJCXBhdGhbbGVuXSA9ICdcMCc7Cgl9Cn0KCnN0YXRpYyB3aW5kb3dPcHRpb25zIGxvYWRfcmVnaXN0cnlfc2V0dGluZ3Modm9pZCkKewoJRFdPUkQgc2l6ZTsKCURXT1JEIHR5cGU7CglIS0VZIGhLZXk7Cgl3aW5kb3dPcHRpb25zIG9wdHM7CglMT0dGT05UIGxvZ2ZvbnQ7CgogICAgICAgIFJlZ09wZW5LZXlFeFcoIEhLRVlfQ1VSUkVOVF9VU0VSLCByZWdpc3RyeV9rZXksCiAgICAgICAgICAgICAgICAgICAgICAgMCwgS0VZX1FVRVJZX1ZBTFVFLCAmaEtleSApOwoKCXNpemUgPSBzaXplb2YoRFdPUkQpOwoKICAgICAgICBpZiggUmVnUXVlcnlWYWx1ZUV4VyggaEtleSwgcmVnX3N0YXJ0X3gsIE5VTEwsICZ0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTFBCWVRFKSAmb3B0cy5zdGFydF94LCAmc2l6ZSApICE9IEVSUk9SX1NVQ0NFU1MgKQoJCW9wdHMuc3RhcnRfeCA9IENXX1VTRURFRkFVTFQ7CgogICAgICAgIGlmKCBSZWdRdWVyeVZhbHVlRXhXKCBoS2V5LCByZWdfc3RhcnRfeSwgTlVMTCwgJnR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUEJZVEUpICZvcHRzLnN0YXJ0X3ksICZzaXplICkgIT0gRVJST1JfU1VDQ0VTUyApCgkJb3B0cy5zdGFydF95ID0gQ1dfVVNFREVGQVVMVDsKCiAgICAgICAgaWYoIFJlZ1F1ZXJ5VmFsdWVFeFcoIGhLZXksIHJlZ193aWR0aCwgTlVMTCwgJnR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUEJZVEUpICZvcHRzLndpZHRoLCAmc2l6ZSApICE9IEVSUk9SX1NVQ0NFU1MgKQoJCW9wdHMud2lkdGggPSBDV19VU0VERUZBVUxUOwoKICAgICAgICBpZiggUmVnUXVlcnlWYWx1ZUV4VyggaEtleSwgcmVnX2hlaWdodCwgTlVMTCwgJnR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUEJZVEUpICZvcHRzLmhlaWdodCwgJnNpemUgKSAhPSBFUlJPUl9TVUNDRVNTICkKCQlvcHRzLmhlaWdodCA9IENXX1VTRURFRkFVTFQ7CglzaXplPXNpemVvZihsb2dmb250KTsKCWlmKCBSZWdRdWVyeVZhbHVlRXhXKCBoS2V5LCByZWdfbG9nZm9udCwgTlVMTCwgJnR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUEJZVEUpICZsb2dmb250LCAmc2l6ZSApICE9IEVSUk9SX1NVQ0NFU1MgKQoJCUdldE9iamVjdChHZXRTdG9ja09iamVjdChERUZBVUxUX0dVSV9GT05UKSxzaXplb2YobG9nZm9udCksJmxvZ2ZvbnQpOwoKCVJlZ0Nsb3NlS2V5KCBoS2V5ICk7CgoJR2xvYmFscy5oZm9udCA9IENyZWF0ZUZvbnRJbmRpcmVjdCgmbG9nZm9udCk7CglyZXR1cm4gb3B0czsKfQoKc3RhdGljIHZvaWQgc2F2ZV9yZWdpc3RyeV9zZXR0aW5ncyh2b2lkKQp7CglXSU5ET1dJTkZPIHdpOwoJSEtFWSBoS2V5OwoJSU5UIHdpZHRoLCBoZWlnaHQ7CglMT0dGT05UIGxvZ2ZvbnQ7CgoJd2kuY2JTaXplID0gc2l6ZW9mKCBXSU5ET1dJTkZPICk7CglHZXRXaW5kb3dJbmZvKEdsb2JhbHMuaE1haW5XbmQsICZ3aSk7Cgl3aWR0aCA9IHdpLnJjV2luZG93LnJpZ2h0IC0gd2kucmNXaW5kb3cubGVmdDsKCWhlaWdodCA9IHdpLnJjV2luZG93LmJvdHRvbSAtIHdpLnJjV2luZG93LnRvcDsKCglpZiAoIFJlZ09wZW5LZXlFeFcoIEhLRVlfQ1VSUkVOVF9VU0VSLCByZWdpc3RyeV9rZXksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAwLCBLRVlfU0VUX1ZBTFVFLCAmaEtleSApICE9IEVSUk9SX1NVQ0NFU1MgKQoJewoJCS8qIFVuYWJsZSB0byBzYXZlIHJlZ2lzdHJ5IHNldHRpbmdzIC0gdHJ5IHRvIGNyZWF0ZSBrZXkgKi8KICAgICAgICAgICAgICAgIGlmICggUmVnQ3JlYXRlS2V5RXhXKCBIS0VZX0NVUlJFTlRfVVNFUiwgcmVnaXN0cnlfa2V5LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsIE5VTEwsIFJFR19PUFRJT05fTk9OX1ZPTEFUSUxFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEtFWV9TRVRfVkFMVUUsIE5VTEwsICZoS2V5LCBOVUxMICkgIT0gRVJST1JfU1VDQ0VTUyApCgkJewoJCQkvKiBGSVhNRTogQ2Fubm90IGNyZWF0ZSBrZXkgKi8KCQkJcmV0dXJuOwoJCX0KCX0KCS8qIFNhdmUgYWxsIG9mIHRoZSBzZXR0aW5ncyAqLwogICAgICAgIFJlZ1NldFZhbHVlRXhXKCBoS2V5LCByZWdfc3RhcnRfeCwgMCwgUkVHX0RXT1JELAogICAgICAgICAgICAgICAgICAgICAgICAoTFBCWVRFKSAmd2kucmNXaW5kb3cubGVmdCwgc2l6ZW9mKERXT1JEKSApOwogICAgICAgIFJlZ1NldFZhbHVlRXhXKCBoS2V5LCByZWdfc3RhcnRfeSwgMCwgUkVHX0RXT1JELAogICAgICAgICAgICAgICAgICAgICAgICAoTFBCWVRFKSAmd2kucmNXaW5kb3cudG9wLCBzaXplb2YoRFdPUkQpICk7CiAgICAgICAgUmVnU2V0VmFsdWVFeFcoIGhLZXksIHJlZ193aWR0aCwgMCwgUkVHX0RXT1JELAogICAgICAgICAgICAgICAgICAgICAgICAoTFBCWVRFKSAmd2lkdGgsIHNpemVvZihEV09SRCkgKTsKICAgICAgICBSZWdTZXRWYWx1ZUV4VyggaEtleSwgcmVnX2hlaWdodCwgMCwgUkVHX0RXT1JELAogICAgICAgICAgICAgICAgICAgICAgICAoTFBCWVRFKSAmaGVpZ2h0LCBzaXplb2YoRFdPUkQpICk7CiAgICAgICAgR2V0T2JqZWN0KEdsb2JhbHMuaGZvbnQsIHNpemVvZihsb2dmb250KSwgJmxvZ2ZvbnQpOwogICAgICAgIFJlZ1NldFZhbHVlRXhXKCBoS2V5LCByZWdfbG9nZm9udCwgMCwgUkVHX0JJTkFSWSwKICAgICAgICAgICAgICAgICAgICAgICAgKExQQllURSkgJmxvZ2ZvbnQsIHNpemVvZihMT0dGT05UKSApOwoKCS8qIFRPRE86IFNhdmUgbW9yZSBzZXR0aW5ncyBoZXJlIChMaXN0IHZzLiBEZXRhaWxlZCBWaWV3LCBldGMuKSAqLwoJUmVnQ2xvc2VLZXkoIGhLZXkgKTsKfQoKc3RhdGljIHZvaWQgcmVzaXplX2ZyYW1lX3JlY3QoSFdORCBod25kLCBQUkVDVCBwcmVjdCkKewoJaW50IG5ld190b3A7CglSRUNUIHJ0OwoKCWlmIChJc1dpbmRvd1Zpc2libGUoR2xvYmFscy5odG9vbGJhcikpIHsKCQlTZW5kTWVzc2FnZShHbG9iYWxzLmh0b29sYmFyLCBXTV9TSVpFLCAwLCAwKTsKCQlHZXRDbGllbnRSZWN0KEdsb2JhbHMuaHRvb2xiYXIsICZydCk7CgkJcHJlY3QtPnRvcCA9IHJ0LmJvdHRvbSszOwoJCXByZWN0LT5ib3R0b20gLT0gcnQuYm90dG9tKzM7Cgl9CgoJaWYgKElzV2luZG93VmlzaWJsZShHbG9iYWxzLmhkcml2ZWJhcikpIHsKCQlTZW5kTWVzc2FnZShHbG9iYWxzLmhkcml2ZWJhciwgV01fU0laRSwgMCwgMCk7CgkJR2V0Q2xpZW50UmVjdChHbG9iYWxzLmhkcml2ZWJhciwgJnJ0KTsKCQluZXdfdG9wID0gLS1wcmVjdC0+dG9wICsgcnQuYm90dG9tKzM7CgkJTW92ZVdpbmRvdyhHbG9iYWxzLmhkcml2ZWJhciwgMCwgcHJlY3QtPnRvcCwgcnQucmlnaHQsIG5ld190b3AsIFRSVUUpOwoJCXByZWN0LT50b3AgPSBuZXdfdG9wOwoJCXByZWN0LT5ib3R0b20gLT0gcnQuYm90dG9tKzI7Cgl9CgoJaWYgKElzV2luZG93VmlzaWJsZShHbG9iYWxzLmhzdGF0dXNiYXIpKSB7CgkJaW50IHBhcnRzW10gPSB7MzAwLCA1MDB9OwoKCQlTZW5kTWVzc2FnZShHbG9iYWxzLmhzdGF0dXNiYXIsIFdNX1NJWkUsIDAsIDApOwoJCVNlbmRNZXNzYWdlKEdsb2JhbHMuaHN0YXR1c2JhciwgU0JfU0VUUEFSVFMsIDIsIChMUEFSQU0pJnBhcnRzKTsKCQlHZXRDbGllbnRSZWN0KEdsb2JhbHMuaHN0YXR1c2JhciwgJnJ0KTsKCQlwcmVjdC0+Ym90dG9tIC09IHJ0LmJvdHRvbTsKCX0KCglNb3ZlV2luZG93KEdsb2JhbHMuaG1kaWNsaWVudCwgcHJlY3QtPmxlZnQtMSxwcmVjdC0+dG9wLTEscHJlY3QtPnJpZ2h0KzIscHJlY3QtPmJvdHRvbSsxLCBUUlVFKTsKfQoKc3RhdGljIHZvaWQgcmVzaXplX2ZyYW1lKEhXTkQgaHduZCwgaW50IGN4LCBpbnQgY3kpCnsKCVJFQ1QgcmVjdDsKCglyZWN0LmxlZnQgICA9IDA7CglyZWN0LnRvcCAgICA9IDA7CglyZWN0LnJpZ2h0ICA9IGN4OwoJcmVjdC5ib3R0b20gPSBjeTsKCglyZXNpemVfZnJhbWVfcmVjdChod25kLCAmcmVjdCk7Cn0KCnN0YXRpYyB2b2lkIHJlc2l6ZV9mcmFtZV9jbGllbnQoSFdORCBod25kKQp7CglSRUNUIHJlY3Q7CgoJR2V0Q2xpZW50UmVjdChod25kLCAmcmVjdCk7CgoJcmVzaXplX2ZyYW1lX3JlY3QoaHduZCwgJnJlY3QpOwp9CgoKc3RhdGljIEhIT09LIGhjYnRob29rOwpzdGF0aWMgQ2hpbGRXbmQqIG5ld2NoaWxkID0gTlVMTDsKCnN0YXRpYyBMUkVTVUxUIENBTExCQUNLIENCVFByb2MoaW50IGNvZGUsIFdQQVJBTSB3cGFyYW0sIExQQVJBTSBscGFyYW0pCnsKCWlmIChjb2RlPT1IQ0JUX0NSRUFURVdORCAmJiBuZXdjaGlsZCkgewoJCUNoaWxkV25kKiBjaGlsZCA9IG5ld2NoaWxkOwoJCW5ld2NoaWxkID0gTlVMTDsKCgkJY2hpbGQtPmh3bmQgPSAoSFdORCkgd3BhcmFtOwoJCVNldFdpbmRvd0xvbmdQdHIoY2hpbGQtPmh3bmQsIEdXTFBfVVNFUkRBVEEsIChMUEFSQU0pY2hpbGQpOwoJfQoKCXJldHVybiBDYWxsTmV4dEhvb2tFeChoY2J0aG9vaywgY29kZSwgd3BhcmFtLCBscGFyYW0pOwp9CgpzdGF0aWMgSFdORCBjcmVhdGVfY2hpbGRfd2luZG93KENoaWxkV25kKiBjaGlsZCkKewoJTURJQ1JFQVRFU1RSVUNUIG1jczsKCWludCBpZHg7CgoJbWNzLnN6Q2xhc3MgPSBzV0lORUZJTEVUUkVFOwoJbWNzLnN6VGl0bGUgPSAoTFBUU1RSKWNoaWxkLT5wYXRoOwoJbWNzLmhPd25lciAgPSBHbG9iYWxzLmhJbnN0YW5jZTsKCW1jcy54ICAgICAgID0gY2hpbGQtPnBvcy5yY05vcm1hbFBvc2l0aW9uLmxlZnQ7CgltY3MueSAgICAgICA9IGNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi50b3A7CgltY3MuY3ggICAgICA9IGNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi5yaWdodC1jaGlsZC0+cG9zLnJjTm9ybWFsUG9zaXRpb24ubGVmdDsKCW1jcy5jeSAgICAgID0gY2hpbGQtPnBvcy5yY05vcm1hbFBvc2l0aW9uLmJvdHRvbS1jaGlsZC0+cG9zLnJjTm9ybWFsUG9zaXRpb24udG9wOwoJbWNzLnN0eWxlICAgPSAwOwoJbWNzLmxQYXJhbSAgPSAwOwoKCWhjYnRob29rID0gU2V0V2luZG93c0hvb2tFeChXSF9DQlQsIENCVFByb2MsIDAsIEdldEN1cnJlbnRUaHJlYWRJZCgpKTsKCgluZXdjaGlsZCA9IGNoaWxkOwoJY2hpbGQtPmh3bmQgPSAoSFdORCkgU2VuZE1lc3NhZ2UoR2xvYmFscy5obWRpY2xpZW50LCBXTV9NRElDUkVBVEUsIDAsIChMUEFSQU0pJm1jcyk7CglpZiAoIWNoaWxkLT5od25kKSB7CgkJVW5ob29rV2luZG93c0hvb2tFeChoY2J0aG9vayk7CgkJcmV0dXJuIDA7Cgl9CgoJVW5ob29rV2luZG93c0hvb2tFeChoY2J0aG9vayk7CgoJU2VuZE1lc3NhZ2UoY2hpbGQtPmxlZnQuaHduZCwgTEJfU0VUSVRFTUhFSUdIVCwgMSwgbWF4KEdsb2JhbHMuc3BhY2VTaXplLmN5LElNQUdFX0hFSUdIVCszKSk7CglTZW5kTWVzc2FnZShjaGlsZC0+cmlnaHQuaHduZCwgTEJfU0VUSVRFTUhFSUdIVCwgMSwgbWF4KEdsb2JhbHMuc3BhY2VTaXplLmN5LElNQUdFX0hFSUdIVCszKSk7CgoJaWR4ID0gU2VuZE1lc3NhZ2UoY2hpbGQtPmxlZnQuaHduZCwgTEJfRklORFNUUklORywgMCwgKExQQVJBTSljaGlsZC0+bGVmdC5jdXIpOwoJU2VuZE1lc3NhZ2UoY2hpbGQtPmxlZnQuaHduZCwgTEJfU0VUQ1VSU0VMLCBpZHgsIDApOwoKCXJldHVybiBjaGlsZC0+aHduZDsKfQoKCnN0cnVjdCBFeGVjdXRlRGlhbG9nIHsKCVRDSEFSCWNtZFtNQVhfUEFUSF07CglpbnQJCWNtZHNob3c7Cn07CgpzdGF0aWMgSU5UX1BUUiBDQUxMQkFDSyBFeGVjdXRlRGlhbG9nRGxnUHJvYyhIV05EIGh3bmQsIFVJTlQgbm1zZywgV1BBUkFNIHdwYXJhbSwgTFBBUkFNIGxwYXJhbSkKewoJc3RhdGljIHN0cnVjdCBFeGVjdXRlRGlhbG9nKiBkbGc7CgoJc3dpdGNoKG5tc2cpIHsKCQljYXNlIFdNX0lOSVRESUFMT0c6CgkJCWRsZyA9IChzdHJ1Y3QgRXhlY3V0ZURpYWxvZyopIGxwYXJhbTsKCQkJcmV0dXJuIDE7CgoJCWNhc2UgV01fQ09NTUFORDogewoJCQlpbnQgaWQgPSAoaW50KXdwYXJhbTsKCgkJCWlmIChpZCA9PSBJRE9LKSB7CgkJCQlHZXRXaW5kb3dUZXh0KEdldERsZ0l0ZW0oaHduZCwgMjAxKSwgZGxnLT5jbWQsIE1BWF9QQVRIKTsKCQkJCWRsZy0+Y21kc2hvdyA9IGdldF9jaGVjayhod25kLDIxNCkgPyBTV19TSE9XTUlOSU1JWkVEIDogU1dfU0hPV05PUk1BTDsKCQkJCUVuZERpYWxvZyhod25kLCBpZCk7CgkJCX0gZWxzZSBpZiAoaWQgPT0gSURDQU5DRUwpCgkJCQlFbmREaWFsb2coaHduZCwgaWQpOwoKCQkJcmV0dXJuIDE7fQoJfQoKCXJldHVybiAwOwp9CgoKc3RhdGljIElOVF9QVFIgQ0FMTEJBQ0sgRGVzdGluYXRpb25EbGdQcm9jKEhXTkQgaHduZCwgVUlOVCBubXNnLCBXUEFSQU0gd3BhcmFtLCBMUEFSQU0gbHBhcmFtKQp7CglUQ0hBUiBiMVtCVUZGRVJfTEVOXSwgYjJbQlVGRkVSX0xFTl07CgoJc3dpdGNoKG5tc2cpIHsKCQljYXNlIFdNX0lOSVRESUFMT0c6CgkJCVNldFdpbmRvd0xvbmdQdHIoaHduZCwgR1dMUF9VU0VSREFUQSwgbHBhcmFtKTsKCQkJU2V0V2luZG93VGV4dChHZXREbGdJdGVtKGh3bmQsIDIwMSksIChMUENUU1RSKWxwYXJhbSk7CgkJCXJldHVybiAxOwoKCQljYXNlIFdNX0NPTU1BTkQ6IHsKCQkJaW50IGlkID0gKGludCl3cGFyYW07CgoJCQlzd2l0Y2goaWQpIHsKCQkJICBjYXNlIElET0s6IHsKCQkJCUxQVFNUUiBkZXN0ID0gKExQVFNUUikgR2V0V2luZG93TG9uZ1B0cihod25kLCBHV0xQX1VTRVJEQVRBKTsKCQkJCUdldFdpbmRvd1RleHQoR2V0RGxnSXRlbShod25kLCAyMDEpLCBkZXN0LCBNQVhfUEFUSCk7CgkJCQlFbmREaWFsb2coaHduZCwgaWQpOwoJCQkJYnJlYWs7fQoKCQkJICBjYXNlIElEQ0FOQ0VMOgoJCQkJRW5kRGlhbG9nKGh3bmQsIGlkKTsKCQkJCWJyZWFrOwoKCQkJICBjYXNlIDI1NDoKCQkJCU1lc3NhZ2VCb3goaHduZCwgUlMoYjEsSURTX05PX0lNUEwpLCBSUyhiMixJRFNfV0lORUZJTEUpLCBNQl9PSyk7CgkJCQlicmVhazsKCQkJfQoKCQkJcmV0dXJuIDE7CgkJfQoJfQoKCXJldHVybiAwOwp9CgoKc3RydWN0IEZpbHRlckRpYWxvZyB7CglUQ0hBUglwYXR0ZXJuW01BWF9QQVRIXTsKCWludAkJZmxhZ3M7Cn07CgpzdGF0aWMgSU5UX1BUUiBDQUxMQkFDSyBGaWx0ZXJEaWFsb2dEbGdQcm9jKEhXTkQgaHduZCwgVUlOVCBubXNnLCBXUEFSQU0gd3BhcmFtLCBMUEFSQU0gbHBhcmFtKQp7CglzdGF0aWMgc3RydWN0IEZpbHRlckRpYWxvZyogZGxnOwoKCXN3aXRjaChubXNnKSB7CgkJY2FzZSBXTV9JTklURElBTE9HOgoJCQlkbGcgPSAoc3RydWN0IEZpbHRlckRpYWxvZyopIGxwYXJhbTsKCQkJU2V0V2luZG93VGV4dChHZXREbGdJdGVtKGh3bmQsIElEQ19WSUVXX1BBVFRFUk4pLCBkbGctPnBhdHRlcm4pOwoJCQlzZXRfY2hlY2soaHduZCwgSURDX1ZJRVdfVFlQRV9ESVJFQ1RPUklFUywgZGxnLT5mbGFncyZURl9ESVJFQ1RPUklFUyk7CgkJCXNldF9jaGVjayhod25kLCBJRENfVklFV19UWVBFX1BST0dSQU1TLCBkbGctPmZsYWdzJlRGX1BST0dSQU1TKTsKCQkJc2V0X2NoZWNrKGh3bmQsIElEQ19WSUVXX1RZUEVfRE9DVU1FTlRTLCBkbGctPmZsYWdzJlRGX0RPQ1VNRU5UUyk7CgkJCXNldF9jaGVjayhod25kLCBJRENfVklFV19UWVBFX09USEVSUywgZGxnLT5mbGFncyZURl9PVEhFUlMpOwoJCQlzZXRfY2hlY2soaHduZCwgSURDX1ZJRVdfVFlQRV9ISURERU4sIGRsZy0+ZmxhZ3MmVEZfSElEREVOKTsKCQkJcmV0dXJuIDE7CgoJCWNhc2UgV01fQ09NTUFORDogewoJCQlpbnQgaWQgPSAoaW50KXdwYXJhbTsKCgkJCWlmIChpZCA9PSBJRE9LKSB7CgkJCQlpbnQgZmxhZ3MgPSAwOwoKCQkJCUdldFdpbmRvd1RleHQoR2V0RGxnSXRlbShod25kLCBJRENfVklFV19QQVRURVJOKSwgZGxnLT5wYXR0ZXJuLCBNQVhfUEFUSCk7CgoJCQkJZmxhZ3MgfD0gZ2V0X2NoZWNrKGh3bmQsIElEQ19WSUVXX1RZUEVfRElSRUNUT1JJRVMpID8gVEZfRElSRUNUT1JJRVMgOiAwOwoJCQkJZmxhZ3MgfD0gZ2V0X2NoZWNrKGh3bmQsIElEQ19WSUVXX1RZUEVfUFJPR1JBTVMpID8gVEZfUFJPR1JBTVMgOiAwOwoJCQkJZmxhZ3MgfD0gZ2V0X2NoZWNrKGh3bmQsIElEQ19WSUVXX1RZUEVfRE9DVU1FTlRTKSA/IFRGX0RPQ1VNRU5UUyA6IDA7CgkJCQlmbGFncyB8PSBnZXRfY2hlY2soaHduZCwgSURDX1ZJRVdfVFlQRV9PVEhFUlMpID8gVEZfT1RIRVJTIDogMDsKCQkJCWZsYWdzIHw9IGdldF9jaGVjayhod25kLCBJRENfVklFV19UWVBFX0hJRERFTikgPyBURl9ISURERU4gOiAwOwoKCQkJCWRsZy0+ZmxhZ3MgPSBmbGFnczsKCgkJCQlFbmREaWFsb2coaHduZCwgaWQpOwoJCQl9IGVsc2UgaWYgKGlkID09IElEQ0FOQ0VMKQoJCQkJRW5kRGlhbG9nKGh3bmQsIGlkKTsKCgkJCXJldHVybiAxO30KCX0KCglyZXR1cm4gMDsKfQoKCnN0cnVjdCBQcm9wZXJ0aWVzRGlhbG9nIHsKCVRDSEFSCXBhdGhbTUFYX1BBVEhdOwoJRW50cnkJZW50cnk7Cgl2b2lkKglwVmVyc2lvbkRhdGE7Cn07CgovKiBTdHJ1Y3R1cmUgdXNlZCB0byBzdG9yZSBlbnVtZXJhdGVkIGxhbmd1YWdlcyBhbmQgY29kZSBwYWdlcy4gKi8Kc3RydWN0IExBTkdBTkRDT0RFUEFHRSB7CglXT1JEIHdMYW5ndWFnZTsKCVdPUkQgd0NvZGVQYWdlOwp9ICpscFRyYW5zbGF0ZTsKCnN0YXRpYyBMUENTVFIgSW5mb1N0cmluZ3NbXSA9IHsKCSJDb21tZW50cyIsCgkiQ29tcGFueU5hbWUiLAoJIkZpbGVEZXNjcmlwdGlvbiIsCgkiRmlsZVZlcnNpb24iLAoJIkludGVybmFsTmFtZSIsCgkiTGVnYWxDb3B5cmlnaHQiLAoJIkxlZ2FsVHJhZGVtYXJrcyIsCgkiT3JpZ2luYWxGaWxlbmFtZSIsCgkiUHJpdmF0ZUJ1aWxkIiwKCSJQcm9kdWN0TmFtZSIsCgkiUHJvZHVjdFZlcnNpb24iLAoJIlNwZWNpYWxCdWlsZCIsCglOVUxMCn07CgpzdGF0aWMgdm9pZCBQcm9wRGxnX0Rpc3BsYXlWYWx1ZShIV05EIGhsYm94LCBIV05EIGhlZGl0KQp7CglpbnQgaWR4ID0gU2VuZE1lc3NhZ2UoaGxib3gsIExCX0dFVENVUlNFTCwgMCwgMCk7CgoJaWYgKGlkeCAhPSBMQl9FUlIpIHsKCQlMUENUU1RSIHBWYWx1ZSA9IChMUENUU1RSKSBTZW5kTWVzc2FnZShobGJveCwgTEJfR0VUSVRFTURBVEEsIGlkeCwgMCk7CgoJCWlmIChwVmFsdWUpCgkJCVNldFdpbmRvd1RleHQoaGVkaXQsIHBWYWx1ZSk7Cgl9Cn0KCnN0YXRpYyB2b2lkIENoZWNrRm9yRmlsZUluZm8oc3RydWN0IFByb3BlcnRpZXNEaWFsb2cqIGRsZywgSFdORCBod25kLCBMUENUU1RSIHN0ckZpbGVuYW1lKQp7CglzdGF0aWMgVENIQVIgc0JhY2tTbGFzaFtdID0geydcXCcsJ1wwJ307CglzdGF0aWMgVENIQVIgc1RyYW5zbGF0aW9uW10gPSB7J1xcJywnVicsJ2EnLCdyJywnRicsJ2knLCdsJywnZScsJ0knLCduJywnZicsJ28nLCdcXCcsJ1QnLCdyJywnYScsJ24nLCdzJywnbCcsJ2EnLCd0JywnaScsJ28nLCduJywnXDAnfTsKCXN0YXRpYyBUQ0hBUiBzU3RyaW5nRmlsZUluZm9bXSA9IHsnXFwnLCdTJywndCcsJ3InLCdpJywnbicsJ2cnLCdGJywnaScsJ2wnLCdlJywnSScsJ24nLCdmJywnbycsJ1xcJywKCQkJCQkJCQkJCSclJywnMCcsJzQnLCd4JywnJScsJzAnLCc0JywneCcsJ1xcJywnJScsJ3MnLCdcMCd9OwoJRFdPUkQgZHdWZXJzaW9uRGF0YUxlbiA9IEdldEZpbGVWZXJzaW9uSW5mb1NpemUoc3RyRmlsZW5hbWUsIE5VTEwpOwoKCWlmIChkd1ZlcnNpb25EYXRhTGVuKSB7CgkJZGxnLT5wVmVyc2lvbkRhdGEgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgZHdWZXJzaW9uRGF0YUxlbik7CgoJCWlmIChHZXRGaWxlVmVyc2lvbkluZm8oc3RyRmlsZW5hbWUsIDAsIGR3VmVyc2lvbkRhdGFMZW4sIGRsZy0+cFZlcnNpb25EYXRhKSkgewoJCQlMUFZPSUQgcFZhbDsKCQkJVUlOVCBuVmFsTGVuOwoKCQkJaWYgKFZlclF1ZXJ5VmFsdWUoZGxnLT5wVmVyc2lvbkRhdGEsIHNCYWNrU2xhc2gsICZwVmFsLCAmblZhbExlbikpIHsKCQkJCWlmIChuVmFsTGVuID09IHNpemVvZihWU19GSVhFREZJTEVJTkZPKSkgewoJCQkJCVZTX0ZJWEVERklMRUlORk8qIHBGaXhlZEZpbGVJbmZvID0gKFZTX0ZJWEVERklMRUlORk8qKXBWYWw7CgkJCQkJY2hhciBidWZmZXJbQlVGRkVSX0xFTl07CgoJCQkJCXNwcmludGYoYnVmZmVyLCAiJWQuJWQuJWQuJWQiLAoJCQkJCQlISVdPUkQocEZpeGVkRmlsZUluZm8tPmR3RmlsZVZlcnNpb25NUyksIExPV09SRChwRml4ZWRGaWxlSW5mby0+ZHdGaWxlVmVyc2lvbk1TKSwKCQkJCQkJSElXT1JEKHBGaXhlZEZpbGVJbmZvLT5kd0ZpbGVWZXJzaW9uTFMpLCBMT1dPUkQocEZpeGVkRmlsZUluZm8tPmR3RmlsZVZlcnNpb25MUykpOwoKCQkJCQlTZXREbGdJdGVtVGV4dEEoaHduZCwgSURDX1NUQVRJQ19QUk9QX1ZFUlNJT04sIGJ1ZmZlcik7CgkJCQl9CgkJCX0KCgkJCS8qIFJlYWQgdGhlIGxpc3Qgb2YgbGFuZ3VhZ2VzIGFuZCBjb2RlIHBhZ2VzLiAqLwoJCQlpZiAoVmVyUXVlcnlWYWx1ZShkbGctPnBWZXJzaW9uRGF0YSwgc1RyYW5zbGF0aW9uLCAmcFZhbCwgJm5WYWxMZW4pKSB7CgkJCQlzdHJ1Y3QgTEFOR0FORENPREVQQUdFKiBwVHJhbnNsYXRlID0gKHN0cnVjdCBMQU5HQU5EQ09ERVBBR0UqKXBWYWw7CgkJCQlzdHJ1Y3QgTEFOR0FORENPREVQQUdFKiBwRW5kID0gKHN0cnVjdCBMQU5HQU5EQ09ERVBBR0UqKSgoTFBCWVRFKXBWYWwrblZhbExlbik7CgoJCQkJSFdORCBobGJveCA9IEdldERsZ0l0ZW0oaHduZCwgSURDX0xJU1RfUFJPUF9WRVJTSU9OX1RZUEVTKTsKCgkJCQkvKiBSZWFkIHRoZSBmaWxlIGRlc2NyaXB0aW9uIGZvciBlYWNoIGxhbmd1YWdlIGFuZCBjb2RlIHBhZ2UuICovCgkJCQlmb3IoOyBwVHJhbnNsYXRlPHBFbmQ7ICsrcFRyYW5zbGF0ZSkgewoJCQkJCUxQQ1NUUiogcDsKCgkJCQkJZm9yKHA9SW5mb1N0cmluZ3M7ICpwOyArK3ApIHsKCQkJCQkJVENIQVIgc3ViYmxvY2tbMjAwXTsKI2lmZGVmIFVOSUNPREUKCQkJCQkJVENIQVIgaW5mb1N0clsxMDBdOwojZW5kaWYKCQkJCQkJTFBDVFNUUiBwVHh0OwoJCQkJCQlVSU5UIG5WYWxMZW47CgoJCQkJCQlMUENTVFIgcEluZm9TdHJpbmcgPSAqcDsKI2lmZGVmIFVOSUNPREUKCQkJCQkJTXVsdGlCeXRlVG9XaWRlQ2hhcihDUF9BQ1AsIDAsIHBJbmZvU3RyaW5nLCAtMSwgaW5mb1N0ciwgMTAwKTsKI2Vsc2UKI2RlZmluZQlpbmZvU3RyIHBJbmZvU3RyaW5nCiNlbmRpZgoJCQkJCQl3c3ByaW50ZihzdWJibG9jaywgc1N0cmluZ0ZpbGVJbmZvLCBwVHJhbnNsYXRlLT53TGFuZ3VhZ2UsIHBUcmFuc2xhdGUtPndDb2RlUGFnZSwgaW5mb1N0cik7CgoJCQkJCQkvKiBSZXRyaWV2ZSBmaWxlIGRlc2NyaXB0aW9uIGZvciBsYW5ndWFnZSBhbmQgY29kZSBwYWdlICovCgkJCQkJCWlmIChWZXJRdWVyeVZhbHVlKGRsZy0+cFZlcnNpb25EYXRhLCBzdWJibG9jaywgKFBWT0lEKSZwVHh0LCAmblZhbExlbikpIHsKCQkJCQkJCWludCBpZHggPSBTZW5kTWVzc2FnZShobGJveCwgTEJfQUREU1RSSU5HLCAwTCwgKExQQVJBTSlpbmZvU3RyKTsKCQkJCQkJCVNlbmRNZXNzYWdlKGhsYm94LCBMQl9TRVRJVEVNREFUQSwgaWR4LCAoTFBBUkFNKSBwVHh0KTsKCQkJCQkJfQoJCQkJCX0KCQkJCX0KCgkJCQlTZW5kTWVzc2FnZShobGJveCwgTEJfU0VUQ1VSU0VMLCAwLCAwKTsKCgkJCQlQcm9wRGxnX0Rpc3BsYXlWYWx1ZShobGJveCwgR2V0RGxnSXRlbShod25kLElEQ19MSVNUX1BST1BfVkVSU0lPTl9WQUxVRVMpKTsKCQkJfQoJCX0KCX0KfQoKc3RhdGljIElOVF9QVFIgQ0FMTEJBQ0sgUHJvcGVydGllc0RpYWxvZ0RsZ1Byb2MoSFdORCBod25kLCBVSU5UIG5tc2csIFdQQVJBTSB3cGFyYW0sIExQQVJBTSBscGFyYW0pCnsKCXN0YXRpYyBzdHJ1Y3QgUHJvcGVydGllc0RpYWxvZyogZGxnOwoKCXN3aXRjaChubXNnKSB7CgkJY2FzZSBXTV9JTklURElBTE9HOiB7CgkJCXN0YXRpYyBjb25zdCBUQ0hBUiBzQnl0ZUZtdFtdID0geyclJywncycsJyAnLCdCJywneScsJ3QnLCdlJywncycsJ1wwJ307CgkJCVRDSEFSIGIxW0JVRkZFUl9MRU5dLCBiMltCVUZGRVJfTEVOXTsKCQkJTFBXSU4zMl9GSU5EX0RBVEEgcFdGRDsKCQkJVUxPTkdMT05HIHNpemU7CgoJCQlkbGcgPSAoc3RydWN0IFByb3BlcnRpZXNEaWFsb2cqKSBscGFyYW07CgkJCXBXRkQgPSAoTFBXSU4zMl9GSU5EX0RBVEEpICZkbGctPmVudHJ5LmRhdGE7CgoJCQlHZXRXaW5kb3dUZXh0KGh3bmQsIGIxLCBNQVhfUEFUSCk7CgkJCXdzcHJpbnRmKGIyLCBiMSwgcFdGRC0+Y0ZpbGVOYW1lKTsKCQkJU2V0V2luZG93VGV4dChod25kLCBiMik7CgoJCQlmb3JtYXRfZGF0ZSgmcFdGRC0+ZnRMYXN0V3JpdGVUaW1lLCBiMSwgQ09MX0RBVEV8Q09MX1RJTUUpOwoJCQlTZXRXaW5kb3dUZXh0KEdldERsZ0l0ZW0oaHduZCwgSURDX1NUQVRJQ19QUk9QX0xBU1RDSEFOR0UpLCBiMSk7CgoJCQlzaXplID0gKChVTE9OR0xPTkcpcFdGRC0+bkZpbGVTaXplSGlnaCA8PCAzMikgfCBwV0ZELT5uRmlsZVNpemVMb3c7CgkJCV9zdHByaW50ZihiMSwgc0xvbmdOdW1GbXQsIHNpemUpOwoJCQl3c3ByaW50ZihiMiwgc0J5dGVGbXQsIGIxKTsKCQkJU2V0V2luZG93VGV4dChHZXREbGdJdGVtKGh3bmQsIElEQ19TVEFUSUNfUFJPUF9TSVpFKSwgYjIpOwoKCQkJU2V0V2luZG93VGV4dChHZXREbGdJdGVtKGh3bmQsIElEQ19TVEFUSUNfUFJPUF9GSUxFTkFNRSksIHBXRkQtPmNGaWxlTmFtZSk7CgkJCVNldFdpbmRvd1RleHQoR2V0RGxnSXRlbShod25kLCBJRENfU1RBVElDX1BST1BfUEFUSCksIGRsZy0+cGF0aCk7CgoJCQlzZXRfY2hlY2soaHduZCwgSURDX0NIRUNLX1JFQURPTkxZLCBwV0ZELT5kd0ZpbGVBdHRyaWJ1dGVzJkZJTEVfQVRUUklCVVRFX1JFQURPTkxZKTsKCQkJc2V0X2NoZWNrKGh3bmQsIElEQ19DSEVDS19BUkNISVZFLCBwV0ZELT5kd0ZpbGVBdHRyaWJ1dGVzJkZJTEVfQVRUUklCVVRFX0FSQ0hJVkUpOwoJCQlzZXRfY2hlY2soaHduZCwgSURDX0NIRUNLX0NPTVBSRVNTRUQsIHBXRkQtPmR3RmlsZUF0dHJpYnV0ZXMmRklMRV9BVFRSSUJVVEVfQ09NUFJFU1NFRCk7CgkJCXNldF9jaGVjayhod25kLCBJRENfQ0hFQ0tfSElEREVOLCBwV0ZELT5kd0ZpbGVBdHRyaWJ1dGVzJkZJTEVfQVRUUklCVVRFX0hJRERFTik7CgkJCXNldF9jaGVjayhod25kLCBJRENfQ0hFQ0tfU1lTVEVNLCBwV0ZELT5kd0ZpbGVBdHRyaWJ1dGVzJkZJTEVfQVRUUklCVVRFX1NZU1RFTSk7CgoJCQlDaGVja0ZvckZpbGVJbmZvKGRsZywgaHduZCwgZGxnLT5wYXRoKTsKCQkJcmV0dXJuIDE7fQoKCQljYXNlIFdNX0NPTU1BTkQ6IHsKCQkJaW50IGlkID0gKGludCl3cGFyYW07CgoJCQlzd2l0Y2goSElXT1JEKHdwYXJhbSkpIHsKCQkJICBjYXNlIExCTl9TRUxDSEFOR0U6IHsKCQkJCUhXTkQgaGxib3ggPSBHZXREbGdJdGVtKGh3bmQsIElEQ19MSVNUX1BST1BfVkVSU0lPTl9UWVBFUyk7CgkJCQlQcm9wRGxnX0Rpc3BsYXlWYWx1ZShobGJveCwgR2V0RGxnSXRlbShod25kLElEQ19MSVNUX1BST1BfVkVSU0lPTl9WQUxVRVMpKTsKCQkJCWJyZWFrOwoJCQkgIH0KCgkJCSAgY2FzZSBCTl9DTElDS0VEOgoJCQkJaWYgKGlkPT1JRE9LIHx8IGlkPT1JRENBTkNFTCkKCQkJCQlFbmREaWFsb2coaHduZCwgaWQpOwoJCQl9CgoJCQlyZXR1cm4gMTt9CgoJCWNhc2UgV01fTkNERVNUUk9ZOgoJCQlIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBkbGctPnBWZXJzaW9uRGF0YSk7CgkJCWRsZy0+cFZlcnNpb25EYXRhID0gTlVMTDsKCQkJYnJlYWs7Cgl9CgoJcmV0dXJuIDA7Cn0KCnN0YXRpYyB2b2lkIHNob3dfcHJvcGVydGllc19kbGcoRW50cnkqIGVudHJ5LCBIV05EIGh3bmQpCnsKCXN0cnVjdCBQcm9wZXJ0aWVzRGlhbG9nIGRsZzsKCgltZW1zZXQoJmRsZywgMCwgc2l6ZW9mKHN0cnVjdCBQcm9wZXJ0aWVzRGlhbG9nKSk7CglnZXRfcGF0aChlbnRyeSwgZGxnLnBhdGgpOwoJbWVtY3B5KCZkbGcuZW50cnksIGVudHJ5LCBzaXplb2YoRW50cnkpKTsKCglEaWFsb2dCb3hQYXJhbShHbG9iYWxzLmhJbnN0YW5jZSwgTUFLRUlOVFJFU09VUkNFKElERF9ESUFMT0dfUFJPUEVSVElFUyksIGh3bmQsIFByb3BlcnRpZXNEaWFsb2dEbGdQcm9jLCAoTFBBUkFNKSZkbGcpOwp9CgoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoKc3RhdGljIHN0cnVjdCBGdWxsU2NyZWVuUGFyYW1ldGVycyB7CglCT09MCW1vZGU7CglSRUNUCW9yZ1BvczsKCUJPT0wJd2FzWm9vbWVkOwp9IGdfZnVsbHNjcmVlbiA9IHsKICAgIEZBTFNFLAkvKiBtb2RlICovCgl7MCwgMCwgMCwgMH0sCglGQUxTRQp9OwoKc3RhdGljIHZvaWQgZnJhbWVfZ2V0X2NsaWVudHNwYWNlKEhXTkQgaHduZCwgUFJFQ1QgcHJlY3QpCnsKCVJFQ1QgcnQ7CgoJaWYgKCFJc0ljb25pYyhod25kKSkKCQlHZXRDbGllbnRSZWN0KGh3bmQsIHByZWN0KTsKCWVsc2UgewoJCVdJTkRPV1BMQUNFTUVOVCB3cDsKCgkJR2V0V2luZG93UGxhY2VtZW50KGh3bmQsICZ3cCk7CgoJCXByZWN0LT5sZWZ0ID0gcHJlY3QtPnRvcCA9IDA7CgkJcHJlY3QtPnJpZ2h0ID0gd3AucmNOb3JtYWxQb3NpdGlvbi5yaWdodC13cC5yY05vcm1hbFBvc2l0aW9uLmxlZnQtCgkJCQkJCTIqKEdldFN5c3RlbU1ldHJpY3MoU01fQ1hTSVpFRlJBTUUpK0dldFN5c3RlbU1ldHJpY3MoU01fQ1hFREdFKSk7CgkJcHJlY3QtPmJvdHRvbSA9IHdwLnJjTm9ybWFsUG9zaXRpb24uYm90dG9tLXdwLnJjTm9ybWFsUG9zaXRpb24udG9wLQoJCQkJCQkyKihHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZU0laRUZSQU1FKStHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZRURHRSkpLQoJCQkJCQlHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZQ0FQVElPTiktR2V0U3lzdGVtTWV0cmljcyhTTV9DWU1FTlVTSVpFKTsKCX0KCglpZiAoSXNXaW5kb3dWaXNpYmxlKEdsb2JhbHMuaHRvb2xiYXIpKSB7CgkJR2V0Q2xpZW50UmVjdChHbG9iYWxzLmh0b29sYmFyLCAmcnQpOwoJCXByZWN0LT50b3AgKz0gcnQuYm90dG9tKzI7Cgl9CgoJaWYgKElzV2luZG93VmlzaWJsZShHbG9iYWxzLmhkcml2ZWJhcikpIHsKCQlHZXRDbGllbnRSZWN0KEdsb2JhbHMuaGRyaXZlYmFyLCAmcnQpOwoJCXByZWN0LT50b3AgKz0gcnQuYm90dG9tKzI7Cgl9CgoJaWYgKElzV2luZG93VmlzaWJsZShHbG9iYWxzLmhzdGF0dXNiYXIpKSB7CgkJR2V0Q2xpZW50UmVjdChHbG9iYWxzLmhzdGF0dXNiYXIsICZydCk7CgkJcHJlY3QtPmJvdHRvbSAtPSBydC5ib3R0b207Cgl9Cn0KCnN0YXRpYyBCT09MIHRvZ2dsZV9mdWxsc2NyZWVuKEhXTkQgaHduZCkKewoJUkVDVCBydDsKCglpZiAoKGdfZnVsbHNjcmVlbi5tb2RlPSFnX2Z1bGxzY3JlZW4ubW9kZSkpIHsKCQlHZXRXaW5kb3dSZWN0KGh3bmQsICZnX2Z1bGxzY3JlZW4ub3JnUG9zKTsKCQlnX2Z1bGxzY3JlZW4ud2FzWm9vbWVkID0gSXNab29tZWQoaHduZCk7CgoJCUZyYW1lX0NhbGNGcmFtZUNsaWVudChod25kLCAmcnQpOwoJCUNsaWVudFRvU2NyZWVuKGh3bmQsIChMUFBPSU5UKSZydC5sZWZ0KTsKCQlDbGllbnRUb1NjcmVlbihod25kLCAoTFBQT0lOVCkmcnQucmlnaHQpOwoKCQlydC5sZWZ0ID0gZ19mdWxsc2NyZWVuLm9yZ1Bvcy5sZWZ0LXJ0LmxlZnQ7CgkJcnQudG9wID0gZ19mdWxsc2NyZWVuLm9yZ1Bvcy50b3AtcnQudG9wOwoJCXJ0LnJpZ2h0ID0gR2V0U3lzdGVtTWV0cmljcyhTTV9DWFNDUkVFTikrZ19mdWxsc2NyZWVuLm9yZ1Bvcy5yaWdodC1ydC5yaWdodDsKCQlydC5ib3R0b20gPSBHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZU0NSRUVOKStnX2Z1bGxzY3JlZW4ub3JnUG9zLmJvdHRvbS1ydC5ib3R0b207CgoJCU1vdmVXaW5kb3coaHduZCwgcnQubGVmdCwgcnQudG9wLCBydC5yaWdodC1ydC5sZWZ0LCBydC5ib3R0b20tcnQudG9wLCBUUlVFKTsKCX0gZWxzZSB7CgkJTW92ZVdpbmRvdyhod25kLCBnX2Z1bGxzY3JlZW4ub3JnUG9zLmxlZnQsIGdfZnVsbHNjcmVlbi5vcmdQb3MudG9wLAoJCQkJCQkJZ19mdWxsc2NyZWVuLm9yZ1Bvcy5yaWdodC1nX2Z1bGxzY3JlZW4ub3JnUG9zLmxlZnQsCgkJCQkJCQlnX2Z1bGxzY3JlZW4ub3JnUG9zLmJvdHRvbS1nX2Z1bGxzY3JlZW4ub3JnUG9zLnRvcCwgVFJVRSk7CgoJCWlmIChnX2Z1bGxzY3JlZW4ud2FzWm9vbWVkKQoJCQlTaG93V2luZG93KGh3bmQsIFdTX01BWElNSVpFKTsKCX0KCglyZXR1cm4gZ19mdWxsc2NyZWVuLm1vZGU7Cn0KCnN0YXRpYyB2b2lkIGZ1bGxzY3JlZW5fbW92ZShIV05EIGh3bmQpCnsKCVJFQ1QgcnQsIHBvczsKCUdldFdpbmRvd1JlY3QoaHduZCwgJnBvcyk7CgoJRnJhbWVfQ2FsY0ZyYW1lQ2xpZW50KGh3bmQsICZydCk7CglDbGllbnRUb1NjcmVlbihod25kLCAoTFBQT0lOVCkmcnQubGVmdCk7CglDbGllbnRUb1NjcmVlbihod25kLCAoTFBQT0lOVCkmcnQucmlnaHQpOwoKCXJ0LmxlZnQgPSBwb3MubGVmdC1ydC5sZWZ0OwoJcnQudG9wID0gcG9zLnRvcC1ydC50b3A7CglydC5yaWdodCA9IEdldFN5c3RlbU1ldHJpY3MoU01fQ1hTQ1JFRU4pK3Bvcy5yaWdodC1ydC5yaWdodDsKCXJ0LmJvdHRvbSA9IEdldFN5c3RlbU1ldHJpY3MoU01fQ1lTQ1JFRU4pK3Bvcy5ib3R0b20tcnQuYm90dG9tOwoKCU1vdmVXaW5kb3coaHduZCwgcnQubGVmdCwgcnQudG9wLCBydC5yaWdodC1ydC5sZWZ0LCBydC5ib3R0b20tcnQudG9wLCBUUlVFKTsKfQoKI2VuZGlmCgoKc3RhdGljIHZvaWQgdG9nZ2xlX2NoaWxkKEhXTkQgaHduZCwgVUlOVCBjbWQsIEhXTkQgaGNoaWxkKQp7CglCT09MIHZpcyA9IElzV2luZG93VmlzaWJsZShoY2hpbGQpOwoKCUNoZWNrTWVudUl0ZW0oR2xvYmFscy5oTWVudU9wdGlvbnMsIGNtZCwgdmlzP01GX0JZQ09NTUFORDpNRl9CWUNPTU1BTkR8TUZfQ0hFQ0tFRCk7CgoJU2hvd1dpbmRvdyhoY2hpbGQsIHZpcz9TV19ISURFOlNXX1NIT1cpOwoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJaWYgKGdfZnVsbHNjcmVlbi5tb2RlKQoJCWZ1bGxzY3JlZW5fbW92ZShod25kKTsKI2VuZGlmCgoJcmVzaXplX2ZyYW1lX2NsaWVudChod25kKTsKfQoKc3RhdGljIEJPT0wgYWN0aXZhdGVfZHJpdmVfd2luZG93KExQQ1RTVFIgcGF0aCkKewoJVENIQVIgZHJ2MVtfTUFYX0RSSVZFXSwgZHJ2MltfTUFYX0RSSVZFXTsKCUhXTkQgY2hpbGRfd25kOwoKCV90c3BsaXRwYXRoKHBhdGgsIGRydjEsIDAsIDAsIDApOwoKCS8qIHNlYXJjaCBmb3IgYSBhbHJlYWR5IG9wZW4gd2luZG93IGZvciB0aGUgc2FtZSBkcml2ZSAqLwoJZm9yKGNoaWxkX3duZD1HZXROZXh0V2luZG93KEdsb2JhbHMuaG1kaWNsaWVudCxHV19DSElMRCk7IGNoaWxkX3duZDsgY2hpbGRfd25kPUdldE5leHRXaW5kb3coY2hpbGRfd25kLCBHV19IV05ETkVYVCkpIHsKCQlDaGlsZFduZCogY2hpbGQgPSAoQ2hpbGRXbmQqKSBHZXRXaW5kb3dMb25nUHRyKGNoaWxkX3duZCwgR1dMUF9VU0VSREFUQSk7CgoJCWlmIChjaGlsZCkgewoJCQlfdHNwbGl0cGF0aChjaGlsZC0+cm9vdC5wYXRoLCBkcnYyLCAwLCAwLCAwKTsKCgkJCWlmICghbHN0cmNtcGkoZHJ2MiwgZHJ2MSkpIHsKCQkJCVNlbmRNZXNzYWdlKEdsb2JhbHMuaG1kaWNsaWVudCwgV01fTURJQUNUSVZBVEUsIChXUEFSQU0pY2hpbGRfd25kLCAwKTsKCgkJCQlpZiAoSXNJY29uaWMoY2hpbGRfd25kKSkKCQkJCQlTaG93V2luZG93KGNoaWxkX3duZCwgU1dfU0hPV05PUk1BTCk7CgoJCQkJcmV0dXJuIFRSVUU7CgkJCX0KCQl9Cgl9CgoJcmV0dXJuIEZBTFNFOwp9CgpzdGF0aWMgQk9PTCBhY3RpdmF0ZV9mc193aW5kb3coTFBDVFNUUiBmaWxlc3lzKQp7CglIV05EIGNoaWxkX3duZDsKCgkvKiBzZWFyY2ggZm9yIGEgYWxyZWFkeSBvcGVuIHdpbmRvdyBvZiB0aGUgZ2l2ZW4gZmlsZSBzeXN0ZW0gbmFtZSAqLwoJZm9yKGNoaWxkX3duZD1HZXROZXh0V2luZG93KEdsb2JhbHMuaG1kaWNsaWVudCxHV19DSElMRCk7IGNoaWxkX3duZDsgY2hpbGRfd25kPUdldE5leHRXaW5kb3coY2hpbGRfd25kLCBHV19IV05ETkVYVCkpIHsKCQlDaGlsZFduZCogY2hpbGQgPSAoQ2hpbGRXbmQqKSBHZXRXaW5kb3dMb25nUHRyKGNoaWxkX3duZCwgR1dMUF9VU0VSREFUQSk7CgoJCWlmIChjaGlsZCkgewoJCQlpZiAoIWxzdHJjbXBpKGNoaWxkLT5yb290LmZzLCBmaWxlc3lzKSkgewoJCQkJU2VuZE1lc3NhZ2UoR2xvYmFscy5obWRpY2xpZW50LCBXTV9NRElBQ1RJVkFURSwgKFdQQVJBTSljaGlsZF93bmQsIDApOwoKCQkJCWlmIChJc0ljb25pYyhjaGlsZF93bmQpKQoJCQkJCVNob3dXaW5kb3coY2hpbGRfd25kLCBTV19TSE9XTk9STUFMKTsKCgkJCQlyZXR1cm4gVFJVRTsKCQkJfQoJCX0KCX0KCglyZXR1cm4gRkFMU0U7Cn0KCnN0YXRpYyBMUkVTVUxUIENBTExCQUNLIEZyYW1lV25kUHJvYyhIV05EIGh3bmQsIFVJTlQgbm1zZywgV1BBUkFNIHdwYXJhbSwgTFBBUkFNIGxwYXJhbSkKewoJVENIQVIgYjFbQlVGRkVSX0xFTl0sIGIyW0JVRkZFUl9MRU5dOwoKCXN3aXRjaChubXNnKSB7CgkJY2FzZSBXTV9DTE9TRToKCQkJaWYgKEdsb2JhbHMuc2F2ZVNldHRpbmdzKQoJCQkJc2F2ZV9yZWdpc3RyeV9zZXR0aW5ncygpOyAgCgkJCQoJCQlEZXN0cm95V2luZG93KGh3bmQpOwoKCQkJIC8qIGNsZWFyIGhhbmRsZSB2YXJpYWJsZXMgKi8KCQkJR2xvYmFscy5oTWVudUZyYW1lID0gMDsKCQkJR2xvYmFscy5oTWVudVZpZXcgPSAwOwoJCQlHbG9iYWxzLmhNZW51T3B0aW9ucyA9IDA7CgkJCUdsb2JhbHMuaE1haW5XbmQgPSAwOwoJCQlHbG9iYWxzLmhtZGljbGllbnQgPSAwOwoJCQlHbG9iYWxzLmhkcml2ZWJhciA9IDA7CgkJCWJyZWFrOwoKCQljYXNlIFdNX0RFU1RST1k6CgkJCVBvc3RRdWl0TWVzc2FnZSgwKTsKCQkJYnJlYWs7CgoJCWNhc2UgV01fSU5JVE1FTlVQT1BVUDogewoJCQlIV05EIGh3bmRDbGllbnQgPSAoSFdORCkgU2VuZE1lc3NhZ2UoR2xvYmFscy5obWRpY2xpZW50LCBXTV9NRElHRVRBQ1RJVkUsIDAsIDApOwoKCQkJaWYgKCFTZW5kTWVzc2FnZShod25kQ2xpZW50LCBXTV9JTklUTUVOVVBPUFVQLCB3cGFyYW0sIGxwYXJhbSkpCgkJCQlyZXR1cm4gMDsKCQkJYnJlYWs7fQoKCQljYXNlIFdNX0NPTU1BTkQ6IHsKCQkJVUlOVCBjbWQgPSBMT1dPUkQod3BhcmFtKTsKCQkJSFdORCBod25kQ2xpZW50ID0gKEhXTkQpIFNlbmRNZXNzYWdlKEdsb2JhbHMuaG1kaWNsaWVudCwgV01fTURJR0VUQUNUSVZFLCAwLCAwKTsKCgkJCWlmIChTZW5kTWVzc2FnZShod25kQ2xpZW50LCBXTV9ESVNQQVRDSF9DT01NQU5ELCB3cGFyYW0sIGxwYXJhbSkpCgkJCQlicmVhazsKCgkJCWlmIChjbWQ+PUlEX0RSSVZFX0ZJUlNUICYmIGNtZDw9SURfRFJJVkVfRklSU1QrMHhGRikgewoJCQkJVENIQVIgZHJ2W19NQVhfRFJJVkVdLCBwYXRoW01BWF9QQVRIXTsKCQkJCUNoaWxkV25kKiBjaGlsZDsKCQkJCUxQQ1RTVFIgcm9vdCA9IEdsb2JhbHMuZHJpdmVzOwoJCQkJaW50IGk7CgoJCQkJZm9yKGk9Y21kLUlEX0RSSVZFX0ZJUlNUOyBpLS07IHJvb3QrKykKCQkJCQl3aGlsZSgqcm9vdCkKCQkJCQkJcm9vdCsrOwoKCQkJCWlmIChhY3RpdmF0ZV9kcml2ZV93aW5kb3cocm9vdCkpCgkJCQkJcmV0dXJuIDA7CgoJCQkJX3RzcGxpdHBhdGgocm9vdCwgZHJ2LCAwLCAwLCAwKTsKCgkJCQlpZiAoIVNldEN1cnJlbnREaXJlY3RvcnkoZHJ2KSkgewoJCQkJCWRpc3BsYXlfZXJyb3IoaHduZCwgR2V0TGFzdEVycm9yKCkpOwoJCQkJCXJldHVybiAwOwoJCQkJfQoKCQkJCUdldEN1cnJlbnREaXJlY3RvcnkoTUFYX1BBVEgsIHBhdGgpOyAvKlRPRE86IHN0b3JlIGxhc3QgZGlyZWN0b3J5IHBlciBkcml2ZSAqLwoJCQkJY2hpbGQgPSBhbGxvY19jaGlsZF93aW5kb3cocGF0aCwgTlVMTCwgaHduZCk7CgoJCQkJaWYgKCFjcmVhdGVfY2hpbGRfd2luZG93KGNoaWxkKSkKCQkJCQlIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBjaGlsZCk7CgkJCX0gZWxzZSBzd2l0Y2goY21kKSB7CgkJCQljYXNlIElEX0ZJTEVfRVhJVDoKCQkJCQlTZW5kTWVzc2FnZShod25kLCBXTV9DTE9TRSwgMCwgMCk7CgkJCQkJYnJlYWs7CgoJCQkJY2FzZSBJRF9XSU5ET1dfTkVXOiB7CgkJCQkJVENIQVIgcGF0aFtNQVhfUEFUSF07CgkJCQkJQ2hpbGRXbmQqIGNoaWxkOwoKCQkJCQlHZXRDdXJyZW50RGlyZWN0b3J5KE1BWF9QQVRILCBwYXRoKTsKCQkJCQljaGlsZCA9IGFsbG9jX2NoaWxkX3dpbmRvdyhwYXRoLCBOVUxMLCBod25kKTsKCgkJCQkJaWYgKCFjcmVhdGVfY2hpbGRfd2luZG93KGNoaWxkKSkKCQkJCQkJSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgY2hpbGQpOwoJCQkJCWJyZWFrO30KCgkJCQljYXNlIElEX1JFRlJFU0g6CgkJCQkJcmVmcmVzaF9kcml2ZXMoKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX1dJTkRPV19DQVNDQURFOgoJCQkJCVNlbmRNZXNzYWdlKEdsb2JhbHMuaG1kaWNsaWVudCwgV01fTURJQ0FTQ0FERSwgMCwgMCk7CgkJCQkJYnJlYWs7CgoJCQkJY2FzZSBJRF9XSU5ET1dfVElMRV9IT1JaOgoJCQkJCVNlbmRNZXNzYWdlKEdsb2JhbHMuaG1kaWNsaWVudCwgV01fTURJVElMRSwgTURJVElMRV9IT1JJWk9OVEFMLCAwKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX1dJTkRPV19USUxFX1ZFUlQ6CgkJCQkJU2VuZE1lc3NhZ2UoR2xvYmFscy5obWRpY2xpZW50LCBXTV9NRElUSUxFLCBNRElUSUxFX1ZFUlRJQ0FMLCAwKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX1dJTkRPV19BUlJBTkdFOgoJCQkJCVNlbmRNZXNzYWdlKEdsb2JhbHMuaG1kaWNsaWVudCwgV01fTURJSUNPTkFSUkFOR0UsIDAsIDApOwoJCQkJCWJyZWFrOwoKCQkJCWNhc2UgSURfU0VMRUNUX0ZPTlQ6CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjaG9vc2VfZm9udChod25kKTsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrOwoKCQkJCWNhc2UgSURfVklFV19UT09MX0JBUjoKCQkJCQl0b2dnbGVfY2hpbGQoaHduZCwgY21kLCBHbG9iYWxzLmh0b29sYmFyKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX1ZJRVdfRFJJVkVfQkFSOgoJCQkJCXRvZ2dsZV9jaGlsZChod25kLCBjbWQsIEdsb2JhbHMuaGRyaXZlYmFyKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX1ZJRVdfU1RBVFVTQkFSOgoJCQkJCXRvZ2dsZV9jaGlsZChod25kLCBjbWQsIEdsb2JhbHMuaHN0YXR1c2Jhcik7CgkJCQkJYnJlYWs7CgoJCQkJY2FzZSBJRF9WSUVXX1NBVkVTRVRUSU5HUzoKCQkJCQlHbG9iYWxzLnNhdmVTZXR0aW5ncyA9ICFHbG9iYWxzLnNhdmVTZXR0aW5nczsKCQkJCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVPcHRpb25zLCBJRF9WSUVXX1NBVkVTRVRUSU5HUywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR2xvYmFscy5zYXZlU2V0dGluZ3MgPyBNRl9DSEVDS0VEIDogTUZfVU5DSEVDS0VEICk7CgkJCQkJYnJlYWs7CgoJCQkJY2FzZSBJRF9FWEVDVVRFOiB7CgkJCQkJc3RydWN0IEV4ZWN1dGVEaWFsb2cgZGxnOwoKCQkJCQltZW1zZXQoJmRsZywgMCwgc2l6ZW9mKHN0cnVjdCBFeGVjdXRlRGlhbG9nKSk7CgoJCQkJCWlmIChEaWFsb2dCb3hQYXJhbShHbG9iYWxzLmhJbnN0YW5jZSwgTUFLRUlOVFJFU09VUkNFKElERF9FWEVDVVRFKSwgaHduZCwgRXhlY3V0ZURpYWxvZ0RsZ1Byb2MsIChMUEFSQU0pJmRsZykgPT0gSURPSykgewoJCQkJCQlISU5TVEFOQ0UgaGluc3QgPSBTaGVsbEV4ZWN1dGUoaHduZCwgTlVMTC8qb3BlcmF0aW9uKi8sIGRsZy5jbWQvKmZpbGUqLywgTlVMTC8qcGFyYW1ldGVycyovLCBOVUxMLypkaXIqLywgZGxnLmNtZHNob3cpOwoKCQkJCQkJaWYgKFB0clRvVWxvbmcoaGluc3QpIDw9IDMyKQoJCQkJCQkJZGlzcGxheV9lcnJvcihod25kLCBHZXRMYXN0RXJyb3IoKSk7CgkJCQkJfQoJCQkJCWJyZWFrO30KCgkJCQljYXNlIElEX0NPTk5FQ1RfTkVUV09SS19EUklWRTogewoJCQkJCURXT1JEIHJldCA9IFdOZXRDb25uZWN0aW9uRGlhbG9nKGh3bmQsIFJFU09VUkNFVFlQRV9ESVNLKTsKCQkJCQlpZiAocmV0ID09IE5PX0VSUk9SKQoJCQkJCQlyZWZyZXNoX2RyaXZlcygpOwoJCQkJCWVsc2UgaWYgKHJldCAhPSAoRFdPUkQpLTEpIHsKCQkJCQkJaWYgKHJldCA9PSBFUlJPUl9FWFRFTkRFRF9FUlJPUikKCQkJCQkJCWRpc3BsYXlfbmV0d29ya19lcnJvcihod25kKTsKCQkJCQkJZWxzZQoJCQkJCQkJZGlzcGxheV9lcnJvcihod25kLCByZXQpOwoJCQkJCX0KCQkJCQlicmVhazt9CgoJCQkJY2FzZSBJRF9ESVNDT05ORUNUX05FVFdPUktfRFJJVkU6IHsKCQkJCQlEV09SRCByZXQgPSBXTmV0RGlzY29ubmVjdERpYWxvZyhod25kLCBSRVNPVVJDRVRZUEVfRElTSyk7CgkJCQkJaWYgKHJldCA9PSBOT19FUlJPUikKCQkJCQkJcmVmcmVzaF9kcml2ZXMoKTsKCQkJCQllbHNlIGlmIChyZXQgIT0gKERXT1JEKS0xKSB7CgkJCQkJCWlmIChyZXQgPT0gRVJST1JfRVhURU5ERURfRVJST1IpCgkJCQkJCQlkaXNwbGF5X25ldHdvcmtfZXJyb3IoaHduZCk7CgkJCQkJCWVsc2UKCQkJCQkJCWRpc3BsYXlfZXJyb3IoaHduZCwgcmV0KTsKCQkJCQl9CgkJCQkJYnJlYWs7fQoKCQkJCWNhc2UgSURfRk9STUFUX0RJU0s6IHsKCQkJCQlVSU5UIHNlbV9vcmcgPSBTZXRFcnJvck1vZGUoMCk7IC8qIEdldCB0aGUgY3VycmVudCBFcnJvciBNb2RlIHNldHRpbmdzLiAqLwoJCQkJCVNldEVycm9yTW9kZShzZW1fb3JnICYgflNFTV9GQUlMQ1JJVElDQUxFUlJPUlMpOyAvKiBGb3JjZSBPL1MgdG8gaGFuZGxlICovCgkJCQkJU0hGb3JtYXREcml2ZShod25kLCAwIC8qIEE6ICovLCBTSEZNVF9JRF9ERUZBVUxULCAwKTsKCQkJCQlTZXRFcnJvck1vZGUoc2VtX29yZyk7IC8qIFB1dCBpdCBiYWNrIHRoZSB3YXkgaXQgd2FzLiAqLwoJCQkJCWJyZWFrO30KCgkJCQljYXNlIElEX0hFTFA6CgkJCQkJV2luSGVscChod25kLCBSUyhiMSxJRFNfV0lORUZJTEUpLCBIRUxQX0lOREVYLCAwKTsKCQkJCQlicmVhazsKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJCWNhc2UgSURfVklFV19GVUxMU0NSRUVOOgoJCQkJCUNoZWNrTWVudUl0ZW0oR2xvYmFscy5oTWVudU9wdGlvbnMsIGNtZCwgdG9nZ2xlX2Z1bGxzY3JlZW4oaHduZCk/TUZfQ0hFQ0tFRDowKTsKCQkJCQlicmVhazsKCiNpZmRlZiBfX1dJTkVfXwoJCQkJY2FzZSBJRF9EUklWRV9VTklYX0ZTOiB7CgkJCQkJVENIQVIgcGF0aFtNQVhfUEFUSF07CiNpZmRlZiBVTklDT0RFCgkJCQkJY2hhciBjcGF0aFtNQVhfUEFUSF07CiNlbmRpZgoJCQkJCUNoaWxkV25kKiBjaGlsZDsKCgkJCQkJaWYgKGFjdGl2YXRlX2ZzX3dpbmRvdyhSUyhiMSxJRFNfVU5JWEZTKSkpCgkJCQkJCWJyZWFrOwoKI2lmZGVmIFVOSUNPREUKCQkJCQlnZXRjd2QoY3BhdGgsIE1BWF9QQVRIKTsKCQkJCQlNdWx0aUJ5dGVUb1dpZGVDaGFyKENQX1VOSVhDUCwgMCwgY3BhdGgsIC0xLCBwYXRoLCBNQVhfUEFUSCk7CiNlbHNlCgkJCQkJZ2V0Y3dkKHBhdGgsIE1BWF9QQVRIKTsKI2VuZGlmCgkJCQkJY2hpbGQgPSBhbGxvY19jaGlsZF93aW5kb3cocGF0aCwgTlVMTCwgaHduZCk7CgoJCQkJCWlmICghY3JlYXRlX2NoaWxkX3dpbmRvdyhjaGlsZCkpCgkJCQkJCUhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGNoaWxkKTsKCQkJCQlicmVhazt9CiNlbmRpZgojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCQkJCWNhc2UgSURfRFJJVkVfU0hFTExfTlM6IHsKCQkJCQlUQ0hBUiBwYXRoW01BWF9QQVRIXTsKCQkJCQlDaGlsZFduZCogY2hpbGQ7CgoJCQkJCWlmIChhY3RpdmF0ZV9mc193aW5kb3coUlMoYjEsSURTX1NIRUxMKSkpCgkJCQkJCWJyZWFrOwoKCQkJCQlHZXRDdXJyZW50RGlyZWN0b3J5KE1BWF9QQVRILCBwYXRoKTsKCQkJCQljaGlsZCA9IGFsbG9jX2NoaWxkX3dpbmRvdyhwYXRoLCBnZXRfcGF0aF9waWRsKHBhdGgsaHduZCksIGh3bmQpOwoKCQkJCQlpZiAoIWNyZWF0ZV9jaGlsZF93aW5kb3coY2hpbGQpKQoJCQkJCQlIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBjaGlsZCk7CgkJCQkJYnJlYWs7fQojZW5kaWYKI2VuZGlmCgoJCQkJLypUT0RPOiBUaGVyZSBhcmUgZXZlbiBtb3JlIG1lbnUgaXRlbXMhICovCgoJCQkJY2FzZSBJRF9BQk9VVDoKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFNoZWxsQWJvdXQoaHduZCwgUlMoYjEsSURTX1dJTkVGSUxFKSwgTlVMTCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTG9hZEltYWdlKCBHbG9iYWxzLmhJbnN0YW5jZSwgTUFLRUlOVFJFU09VUkNFKElESV9XSU5FRklMRSksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSU1BR0VfSUNPTiwgNDgsIDQ4LCBMUl9TSEFSRUQgKSk7CgkJCQkJYnJlYWs7CgoJCQkJZGVmYXVsdDoKCQkJCQkvKlRPRE86IGlmICh3UGFyYW0gPj0gUE1fRklSU1RfTEFOR1VBR0UgJiYgd1BhcmFtIDw9IFBNX0xBU1RfTEFOR1VBR0UpCgkJCQkJCVNUUklOR19TZWxlY3RMYW5ndWFnZUJ5TnVtYmVyKHdQYXJhbSAtIFBNX0ZJUlNUX0xBTkdVQUdFKTsKCQkJCQllbHNlICovaWYgKChjbWQ8SURXX0ZJUlNUX0NISUxEIHx8IGNtZD49SURXX0ZJUlNUX0NISUxEKzB4MTAwKSAmJgoJCQkJCQkoY21kPFNDX1NJWkUgfHwgY21kPlNDX1JFU1RPUkUpKQoJCQkJCQlNZXNzYWdlQm94KGh3bmQsIFJTKGIyLElEU19OT19JTVBMKSwgUlMoYjEsSURTX1dJTkVGSUxFKSwgTUJfT0spOwoKCQkJCQlyZXR1cm4gRGVmRnJhbWVQcm9jKGh3bmQsIEdsb2JhbHMuaG1kaWNsaWVudCwgbm1zZywgd3BhcmFtLCBscGFyYW0pOwoJCQl9CgkJCWJyZWFrO30KCgkJY2FzZSBXTV9TSVpFOgoJCQlyZXNpemVfZnJhbWUoaHduZCwgTE9XT1JEKGxwYXJhbSksIEhJV09SRChscGFyYW0pKTsKCQkJYnJlYWs7CS8qIGRvIG5vdCBwYXNzIG1lc3NhZ2UgdG8gRGVmRnJhbWVQcm9jICovCgoJCWNhc2UgV01fREVWSUNFQ0hBTkdFOgoJCQlTZW5kTWVzc2FnZShod25kLCBXTV9DT01NQU5ELCBNQUtFTE9ORyhJRF9SRUZSRVNILDApLCAwKTsKCQkJYnJlYWs7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCgkJY2FzZSBXTV9HRVRNSU5NQVhJTkZPOiB7CgkJCUxQTUlOTUFYSU5GTyBscG1taSA9IChMUE1JTk1BWElORk8pbHBhcmFtOwoKCQkJbHBtbWktPnB0TWF4VHJhY2tTaXplLnggPDw9IDE7LyoyKkdldFN5c3RlbU1ldHJpY3MoU01fQ1hTQ1JFRU4pIC8gU01fQ1hWSVJUVUFMU0NSRUVOICovCgkJCWxwbW1pLT5wdE1heFRyYWNrU2l6ZS55IDw8PSAxOy8qMipHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZU0NSRUVOKSAvIFNNX0NZVklSVFVBTFNDUkVFTiAqLwoJCQlicmVhazt9CgoJCWNhc2UgRlJNX0NBTENfQ0xJRU5UOgoJCQlmcmFtZV9nZXRfY2xpZW50c3BhY2UoaHduZCwgKFBSRUNUKWxwYXJhbSk7CgkJCXJldHVybiBUUlVFOwojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KCgkJZGVmYXVsdDoKCQkJcmV0dXJuIERlZkZyYW1lUHJvYyhod25kLCBHbG9iYWxzLmhtZGljbGllbnQsIG5tc2csIHdwYXJhbSwgbHBhcmFtKTsKCX0KCglyZXR1cm4gMDsKfQoKCnN0YXRpYyBUQ0hBUiBnX3Bvc19uYW1lc1tDT0xVTU5TXVsyMF0gPSB7Cgl7J1wwJ30JLyogc3ltYm9sICovCn07CgpzdGF0aWMgY29uc3QgaW50IGdfcG9zX2FsaWduW10gPSB7CgkwLAoJSERGX0xFRlQsCS8qIE5hbWUgKi8KCUhERl9SSUdIVCwJLyogU2l6ZSAqLwoJSERGX0xFRlQsCS8qIENEYXRlICovCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCUhERl9MRUZULAkvKiBBRGF0ZSAqLwoJSERGX0xFRlQsCS8qIE1EYXRlICovCglIREZfTEVGVCwJLyogSW5kZXggKi8KCUhERl9DRU5URVIsCS8qIExpbmtzICovCiNlbmRpZgoJSERGX0NFTlRFUiwJLyogQXR0cmlidXRlcyAqLwojaWZuZGVmIF9OT19FWFRFTlNJT05TCglIREZfTEVGVAkvKiBTZWN1cml0eSAqLwojZW5kaWYKfTsKCnN0YXRpYyB2b2lkIHJlc2l6ZV90cmVlKENoaWxkV25kKiBjaGlsZCwgaW50IGN4LCBpbnQgY3kpCnsKCUhEV1AgaGR3cCA9IEJlZ2luRGVmZXJXaW5kb3dQb3MoNCk7CglSRUNUIHJ0OwoKCXJ0LmxlZnQgICA9IDA7CglydC50b3AgICAgPSAwOwoJcnQucmlnaHQgID0gY3g7CglydC5ib3R0b20gPSBjeTsKCgljeCA9IGNoaWxkLT5zcGxpdF9wb3MgKyBTUExJVF9XSURUSC8yOwoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJewoJCVdJTkRPV1BPUyB3cDsKCQlIRF9MQVlPVVQgaGRsOwoKCQloZGwucHJjICAgPSAmcnQ7CgkJaGRsLnB3cG9zID0gJndwOwoKCQlTZW5kTWVzc2FnZShjaGlsZC0+bGVmdC5od25kSGVhZGVyLCBIRE1fTEFZT1VULCAwLCAoTFBBUkFNKSZoZGwpOwoKCQlEZWZlcldpbmRvd1BvcyhoZHdwLCBjaGlsZC0+bGVmdC5od25kSGVhZGVyLCB3cC5od25kSW5zZXJ0QWZ0ZXIsCgkJCQkJCXdwLngtMSwgd3AueSwgY2hpbGQtPnNwbGl0X3Bvcy1TUExJVF9XSURUSC8yKzEsIHdwLmN5LCB3cC5mbGFncyk7CgkJRGVmZXJXaW5kb3dQb3MoaGR3cCwgY2hpbGQtPnJpZ2h0Lmh3bmRIZWFkZXIsIHdwLmh3bmRJbnNlcnRBZnRlciwKCQkJCQkJcnQubGVmdCtjeCsxLCB3cC55LCB3cC5jeC1jeCsyLCB3cC5jeSwgd3AuZmxhZ3MpOwoJfQojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KCglEZWZlcldpbmRvd1BvcyhoZHdwLCBjaGlsZC0+bGVmdC5od25kLCAwLCBydC5sZWZ0LCBydC50b3AsIGNoaWxkLT5zcGxpdF9wb3MtU1BMSVRfV0lEVEgvMi1ydC5sZWZ0LCBydC5ib3R0b20tcnQudG9wLCBTV1BfTk9aT1JERVJ8U1dQX05PQUNUSVZBVEUpOwoJRGVmZXJXaW5kb3dQb3MoaGR3cCwgY2hpbGQtPnJpZ2h0Lmh3bmQsIDAsIHJ0LmxlZnQrY3grMSwgcnQudG9wLCBydC5yaWdodC1jeCwgcnQuYm90dG9tLXJ0LnRvcCwgU1dQX05PWk9SREVSfFNXUF9OT0FDVElWQVRFKTsKCglFbmREZWZlcldpbmRvd1BvcyhoZHdwKTsKfQoKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCnN0YXRpYyBIV05EIGNyZWF0ZV9oZWFkZXIoSFdORCBwYXJlbnQsIFBhbmUqIHBhbmUsIFVJTlQgaWQpCnsKCUhEX0lURU0gaGRpOwoJaW50IGlkeDsKCglIV05EIGh3bmQgPSBDcmVhdGVXaW5kb3coV0NfSEVBREVSLCAwLCBXU19DSElMRHxXU19WSVNJQkxFfEhEU19IT1JafEhEU19GVUxMRFJBRy8qVE9ETzogfEhEU19CVVRUT05TICsgc29ydCBvcmRlcnMqLywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwgMCwgMCwgMCwgcGFyZW50LCAoSE1FTlUpVUxvbmdUb0hhbmRsZShpZCksIEdsb2JhbHMuaEluc3RhbmNlLCAwKTsKCWlmICghaHduZCkKCQlyZXR1cm4gMDsKCglTZW5kTWVzc2FnZShod25kLCBXTV9TRVRGT05ULCAoV1BBUkFNKUdldFN0b2NrT2JqZWN0KERFRkFVTFRfR1VJX0ZPTlQpLCBGQUxTRSk7CgoJaGRpLm1hc2sgPSBIRElfVEVYVHxIRElfV0lEVEh8SERJX0ZPUk1BVDsKCglmb3IoaWR4PTA7IGlkeDxDT0xVTU5TOyBpZHgrKykgewoJCWhkaS5wc3pUZXh0ID0gZ19wb3NfbmFtZXNbaWR4XTsKCQloZGkuZm10ID0gSERGX1NUUklORyB8IGdfcG9zX2FsaWduW2lkeF07CgkJaGRpLmN4eSA9IHBhbmUtPndpZHRoc1tpZHhdOwoJCVNlbmRNZXNzYWdlKGh3bmQsIEhETV9JTlNFUlRJVEVNLCBpZHgsIChMUEFSQU0pICZoZGkpOwoJfQoKCXJldHVybiBod25kOwp9CgojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KCgpzdGF0aWMgdm9pZCBpbml0X291dHB1dChIV05EIGh3bmQpCnsKCXN0YXRpYyBjb25zdCBXQ0hBUiBzMTAwMFtdID0geycxJywnMCcsJzAnLCcwJywnXDAnfTsKCVdDSEFSIGJbMTZdOwoJSEZPTlQgb2xkX2ZvbnQ7CglIREMgaGRjID0gR2V0REMoaHduZCk7CgoJaWYgKEdldE51bWJlckZvcm1hdFcoTE9DQUxFX1VTRVJfREVGQVVMVCwgMCwgczEwMDAsIDAsIGIsIDE2KSA+IDQpCgkJR2xvYmFscy5udW1fc2VwID0gYlsxXTsKCWVsc2UKCQlHbG9iYWxzLm51bV9zZXAgPSAnLic7CgoJb2xkX2ZvbnQgPSBTZWxlY3RPYmplY3QoaGRjLCBHbG9iYWxzLmhmb250KTsKCUdldFRleHRFeHRlbnRQb2ludDMyVyhoZGMsIHNTcGFjZSwgMSwgJkdsb2JhbHMuc3BhY2VTaXplKTsKCVNlbGVjdE9iamVjdChoZGMsIG9sZF9mb250KTsKCVJlbGVhc2VEQyhod25kLCBoZGMpOwp9CgpzdGF0aWMgdm9pZCBkcmF3X2l0ZW0oUGFuZSogcGFuZSwgTFBEUkFXSVRFTVNUUlVDVCBkaXMsIEVudHJ5KiBlbnRyeSwgaW50IGNhbGNXaWR0aENvbCk7CgoKLyogY2FsY3VsYXRlIHByZWZlcnJlZCB3aWR0aCBmb3IgYWxsIHZpc2libGUgY29sdW1ucyAqLwoKc3RhdGljIEJPT0wgY2FsY193aWR0aHMoUGFuZSogcGFuZSwgQk9PTCBhbnl3YXkpCnsKCWludCBjb2wsIHgsIGN4LCBzcGM9MypHbG9iYWxzLnNwYWNlU2l6ZS5jeDsKCWludCBlbnRyaWVzID0gU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgTEJfR0VUQ09VTlQsIDAsIDApOwoJaW50IG9yZ1dpZHRoc1tDT0xVTU5TXTsKCWludCBvcmdQb3NpdGlvbnNbQ09MVU1OUysxXTsKCUhGT05UIGhmb250T2xkOwoJSERDIGhkYzsKCWludCBjbnQ7CgoJaWYgKCFhbnl3YXkpIHsKCQltZW1jcHkob3JnV2lkdGhzLCBwYW5lLT53aWR0aHMsIHNpemVvZihvcmdXaWR0aHMpKTsKCQltZW1jcHkob3JnUG9zaXRpb25zLCBwYW5lLT5wb3NpdGlvbnMsIHNpemVvZihvcmdQb3NpdGlvbnMpKTsKCX0KCglmb3IoY29sPTA7IGNvbDxDT0xVTU5TOyBjb2wrKykKCQlwYW5lLT53aWR0aHNbY29sXSA9IDA7CgoJaGRjID0gR2V0REMocGFuZS0+aHduZCk7CgloZm9udE9sZCA9IFNlbGVjdE9iamVjdChoZGMsIEdsb2JhbHMuaGZvbnQpOwoKCWZvcihjbnQ9MDsgY250PGVudHJpZXM7IGNudCsrKSB7CgkJRW50cnkqIGVudHJ5ID0gKEVudHJ5KikgU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgTEJfR0VUSVRFTURBVEEsIGNudCwgMCk7CgoJCURSQVdJVEVNU1RSVUNUIGRpczsKCgkJZGlzLkN0bFR5cGUJCSAgPSAwOwoJCWRpcy5DdGxJRAkJICA9IDA7CgkJZGlzLml0ZW1JRAkJICA9IDA7CgkJZGlzLml0ZW1BY3Rpb24JICA9IDA7CgkJZGlzLml0ZW1TdGF0ZQkgID0gMDsKCQlkaXMuaHduZEl0ZW0JICA9IHBhbmUtPmh3bmQ7CgkJZGlzLmhEQwkJCSAgPSBoZGM7CgkJZGlzLnJjSXRlbS5sZWZ0CSAgPSAwOwoJCWRpcy5yY0l0ZW0udG9wICAgID0gMDsKCQlkaXMucmNJdGVtLnJpZ2h0ICA9IDA7CgkJZGlzLnJjSXRlbS5ib3R0b20gPSAwOwoJCS8qZGlzLml0ZW1EYXRhCSAgPSAwOyAqLwoKCQlkcmF3X2l0ZW0ocGFuZSwgJmRpcywgZW50cnksIENPTFVNTlMpOwoJfQoKCVNlbGVjdE9iamVjdChoZGMsIGhmb250T2xkKTsKCVJlbGVhc2VEQyhwYW5lLT5od25kLCBoZGMpOwoKCXggPSAwOwoJZm9yKGNvbD0wOyBjb2w8Q09MVU1OUzsgY29sKyspIHsKCQlwYW5lLT5wb3NpdGlvbnNbY29sXSA9IHg7CgkJY3ggPSBwYW5lLT53aWR0aHNbY29sXTsKCgkJaWYgKGN4KSB7CgkJCWN4ICs9IHNwYzsKCgkJCWlmIChjeCA8IElNQUdFX1dJRFRIKQoJCQkJY3ggPSBJTUFHRV9XSURUSDsKCgkJCXBhbmUtPndpZHRoc1tjb2xdID0gY3g7CgkJfQoKCQl4ICs9IGN4OwoJfQoKCXBhbmUtPnBvc2l0aW9uc1tDT0xVTU5TXSA9IHg7CgoJU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgTEJfU0VUSE9SSVpPTlRBTEVYVEVOVCwgeCwgMCk7CgoJLyogbm8gY2hhbmdlPyAqLwoJaWYgKCFhbnl3YXkgJiYgIW1lbWNtcChvcmdXaWR0aHMsIHBhbmUtPndpZHRocywgc2l6ZW9mKG9yZ1dpZHRocykpKQoJCXJldHVybiBGQUxTRTsKCgkvKiBkb24ndCBtb3ZlLCBpZiBvbmx5IGNvbGxhcHNpbmcgYW4gZW50cnkgKi8KCWlmICghYW55d2F5ICYmIHBhbmUtPndpZHRoc1swXTxvcmdXaWR0aHNbMF0gJiYKCQkhbWVtY21wKG9yZ1dpZHRocysxLCBwYW5lLT53aWR0aHMrMSwgc2l6ZW9mKG9yZ1dpZHRocyktc2l6ZW9mKGludCkpKSB7CgkJcGFuZS0+d2lkdGhzWzBdID0gb3JnV2lkdGhzWzBdOwoJCW1lbWNweShwYW5lLT5wb3NpdGlvbnMsIG9yZ1Bvc2l0aW9ucywgc2l6ZW9mKG9yZ1Bvc2l0aW9ucykpOwoKCQlyZXR1cm4gRkFMU0U7Cgl9CgoJSW52YWxpZGF0ZVJlY3QocGFuZS0+aHduZCwgMCwgVFJVRSk7CgoJcmV0dXJuIFRSVUU7Cn0KCgovKiBjYWxjdWxhdGUgb25lIHByZWZlcnJlZCBjb2x1bW4gd2lkdGggKi8KCnN0YXRpYyB2b2lkIGNhbGNfc2luZ2xlX3dpZHRoKFBhbmUqIHBhbmUsIGludCBjb2wpCnsKCUhGT05UIGhmb250T2xkOwoJaW50IHgsIGN4OwoJaW50IGVudHJpZXMgPSBTZW5kTWVzc2FnZShwYW5lLT5od25kLCBMQl9HRVRDT1VOVCwgMCwgMCk7CglpbnQgY250OwoJSERDIGhkYzsKCglwYW5lLT53aWR0aHNbY29sXSA9IDA7CgoJaGRjID0gR2V0REMocGFuZS0+aHduZCk7CgloZm9udE9sZCA9IFNlbGVjdE9iamVjdChoZGMsIEdsb2JhbHMuaGZvbnQpOwoKCWZvcihjbnQ9MDsgY250PGVudHJpZXM7IGNudCsrKSB7CgkJRW50cnkqIGVudHJ5ID0gKEVudHJ5KikgU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgTEJfR0VUSVRFTURBVEEsIGNudCwgMCk7CgkJRFJBV0lURU1TVFJVQ1QgZGlzOwoKCQlkaXMuQ3RsVHlwZQkJICA9IDA7CgkJZGlzLkN0bElECQkgID0gMDsKCQlkaXMuaXRlbUlECQkgID0gMDsKCQlkaXMuaXRlbUFjdGlvbgkgID0gMDsKCQlkaXMuaXRlbVN0YXRlCSAgPSAwOwoJCWRpcy5od25kSXRlbQkgID0gcGFuZS0+aHduZDsKCQlkaXMuaERDCQkJICA9IGhkYzsKCQlkaXMucmNJdGVtLmxlZnQJICA9IDA7CgkJZGlzLnJjSXRlbS50b3AgICAgPSAwOwoJCWRpcy5yY0l0ZW0ucmlnaHQgID0gMDsKCQlkaXMucmNJdGVtLmJvdHRvbSA9IDA7CgkJLypkaXMuaXRlbURhdGEJICA9IDA7ICovCgoJCWRyYXdfaXRlbShwYW5lLCAmZGlzLCBlbnRyeSwgY29sKTsKCX0KCglTZWxlY3RPYmplY3QoaGRjLCBoZm9udE9sZCk7CglSZWxlYXNlREMocGFuZS0+aHduZCwgaGRjKTsKCgljeCA9IHBhbmUtPndpZHRoc1tjb2xdOwoKCWlmIChjeCkgewoJCWN4ICs9IDMqR2xvYmFscy5zcGFjZVNpemUuY3g7CgoJCWlmIChjeCA8IElNQUdFX1dJRFRIKQoJCQljeCA9IElNQUdFX1dJRFRIOwoJfQoKCXBhbmUtPndpZHRoc1tjb2xdID0gY3g7CgoJeCA9IHBhbmUtPnBvc2l0aW9uc1tjb2xdICsgY3g7CgoJZm9yKDsgY29sPENPTFVNTlM7ICkgewoJCXBhbmUtPnBvc2l0aW9uc1srK2NvbF0gPSB4OwoJCXggKz0gcGFuZS0+d2lkdGhzW2NvbF07Cgl9CgoJU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgTEJfU0VUSE9SSVpPTlRBTEVYVEVOVCwgeCwgMCk7Cn0KCgpzdGF0aWMgQk9PTCBwYXR0ZXJuX21hdGNoKExQQ1RTVFIgc3RyLCBMUENUU1RSIHBhdHRlcm4pCnsKCWZvciggOyAqc3RyJiYqcGF0dGVybjsgc3RyKysscGF0dGVybisrKSB7CgkJaWYgKCpwYXR0ZXJuID09ICcqJykgewoJCQlkbyBwYXR0ZXJuKys7CgkJCXdoaWxlKCpwYXR0ZXJuID09ICcqJyk7CgoJCQlpZiAoISpwYXR0ZXJuKQoJCQkJcmV0dXJuIFRSVUU7CgoJCQlmb3IoOyAqc3RyOyBzdHIrKykKCQkJCWlmICgqc3RyPT0qcGF0dGVybiAmJiBwYXR0ZXJuX21hdGNoKHN0ciwgcGF0dGVybikpCgkJCQkJcmV0dXJuIFRSVUU7CgoJCQlyZXR1cm4gRkFMU0U7CgkJfQoJCWVsc2UgaWYgKCpzdHIhPSpwYXR0ZXJuICYmICpwYXR0ZXJuIT0nPycpCgkJCXJldHVybiBGQUxTRTsKCX0KCglpZiAoKnN0ciB8fCAqcGF0dGVybikKCQlpZiAoKnBhdHRlcm4hPScqJyB8fCBwYXR0ZXJuWzFdIT0nXDAnKQoJCQlyZXR1cm4gRkFMU0U7CgoJcmV0dXJuIFRSVUU7Cn0KCnN0YXRpYyBCT09MIHBhdHRlcm5faW1hdGNoKExQQ1RTVFIgc3RyLCBMUENUU1RSIHBhdHRlcm4pCnsKCVRDSEFSIGIxW0JVRkZFUl9MRU5dLCBiMltCVUZGRVJfTEVOXTsKCglsc3RyY3B5KGIxLCBzdHIpOwoJbHN0cmNweShiMiwgcGF0dGVybik7CglDaGFyVXBwZXIoYjEpOwoJQ2hhclVwcGVyKGIyKTsKCglyZXR1cm4gcGF0dGVybl9tYXRjaChiMSwgYjIpOwp9CgoKZW51bSBGSUxFX1RZUEUgewoJRlRfT1RIRVIJCT0gMCwKCUZUX0VYRUNVVEFCTEUJPSAxLAoJRlRfRE9DVU1FTlQJCT0gMgp9OwoKc3RhdGljIGVudW0gRklMRV9UWVBFIGdldF9maWxlX3R5cGUoTFBDVFNUUiBmaWxlbmFtZSk7CgoKLyogaW5zZXJ0IGxpc3Rib3ggZW50cmllcyBhZnRlciBpbmRleCBpZHggKi8KCnN0YXRpYyBpbnQgaW5zZXJ0X2VudHJpZXMoUGFuZSogcGFuZSwgRW50cnkqIGRpciwgTFBDVFNUUiBwYXR0ZXJuLCBpbnQgZmlsdGVyX2ZsYWdzLCBpbnQgaWR4KQp7CglFbnRyeSogZW50cnkgPSBkaXI7CgoJaWYgKCFlbnRyeSkKCQlyZXR1cm4gaWR4OwoKCVNob3dXaW5kb3cocGFuZS0+aHduZCwgU1dfSElERSk7CgoJZm9yKDsgZW50cnk7IGVudHJ5PWVudHJ5LT5uZXh0KSB7CiNpZm5kZWYgX0xFRlRfRklMRVMKCQlpZiAocGFuZS0+dHJlZVBhbmUgJiYgIShlbnRyeS0+ZGF0YS5kd0ZpbGVBdHRyaWJ1dGVzJkZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWSkpCgkJCWNvbnRpbnVlOwojZW5kaWYKCgkJaWYgKGVudHJ5LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpIHsKCQkJLyogZG9uJ3QgZGlzcGxheSBlbnRyaWVzICIuIiBhbmQgIi4uIiBpbiB0aGUgbGVmdCBwYW5lICovCgkJCWlmIChwYW5lLT50cmVlUGFuZSAmJiBlbnRyeS0+ZGF0YS5jRmlsZU5hbWVbMF0gPT0gJy4nKQoJCQkJaWYgKAojaWZuZGVmIF9OT19FWFRFTlNJT05TCgkJCQkJZW50cnktPmRhdGEuY0ZpbGVOYW1lWzFdID09ICdcMCcgfHwKI2VuZGlmCgkJCQkJKGVudHJ5LT5kYXRhLmNGaWxlTmFtZVsxXSA9PSAnLicgJiYgZW50cnktPmRhdGEuY0ZpbGVOYW1lWzJdID09ICdcMCcpKQoJCQkJCWNvbnRpbnVlOwoKCQkJLyogZmlsdGVyIGRpcmVjdG9yaWVzIGluIHJpZ2h0IHBhbmUgKi8KCQkJaWYgKCFwYW5lLT50cmVlUGFuZSAmJiAhKGZpbHRlcl9mbGFncyZURl9ESVJFQ1RPUklFUykpCgkJCQljb250aW51ZTsKCQl9CgoJCS8qIGZpbHRlciB1c2luZyB0aGUgZmlsZSBuYW1lIHBhdHRlcm4gKi8KCQlpZiAocGF0dGVybikKCQkJaWYgKCFwYXR0ZXJuX2ltYXRjaChlbnRyeS0+ZGF0YS5jRmlsZU5hbWUsIHBhdHRlcm4pKQoJCQkJY29udGludWU7CgoJCS8qIGZpbHRlciBzeXN0ZW0gYW5kIGhpZGRlbiBmaWxlcyAqLwoJCWlmICghKGZpbHRlcl9mbGFncyZURl9ISURERU4pICYmIChlbnRyeS0+ZGF0YS5kd0ZpbGVBdHRyaWJ1dGVzJihGSUxFX0FUVFJJQlVURV9ISURERU58RklMRV9BVFRSSUJVVEVfU1lTVEVNKSkpCgkJCWNvbnRpbnVlOwoKCQkvKiBmaWx0ZXIgbG9va2luZyBhdCB0aGUgZmlsZSB0eXBlICovCgkJaWYgKChmaWx0ZXJfZmxhZ3MmKFRGX1BST0dSQU1TfFRGX0RPQ1VNRU5UU3xURl9PVEhFUlMpKSAhPSAoVEZfUFJPR1JBTVN8VEZfRE9DVU1FTlRTfFRGX09USEVSUykpCgkJCXN3aXRjaChnZXRfZmlsZV90eXBlKGVudHJ5LT5kYXRhLmNGaWxlTmFtZSkpIHsKCQkJICBjYXNlIEZUX0VYRUNVVEFCTEU6CgkJCSAgCWlmICghKGZpbHRlcl9mbGFncyAmIFRGX1BST0dSQU1TKSkKCQkJCQljb250aW51ZTsKCQkJCWJyZWFrOwoKCQkJICBjYXNlIEZUX0RPQ1VNRU5UOgoJCQkJaWYgKCEoZmlsdGVyX2ZsYWdzICYgVEZfRE9DVU1FTlRTKSkKCQkJCQljb250aW51ZTsKCQkJCWJyZWFrOwoKCQkJICBkZWZhdWx0OiAvKiBURl9PVEhFUlMgKi8KCQkJCWlmICghKGZpbHRlcl9mbGFncyAmIFRGX09USEVSUykpCgkJCQkJY29udGludWU7CgkJCX0KCgkJaWYgKGlkeCAhPSAtMSkKCQkJaWR4Kys7CgoJCVNlbmRNZXNzYWdlKHBhbmUtPmh3bmQsIExCX0lOU0VSVFNUUklORywgaWR4LCAoTFBBUkFNKSBlbnRyeSk7CgoJCWlmIChwYW5lLT50cmVlUGFuZSAmJiBlbnRyeS0+ZXhwYW5kZWQpCgkJCWlkeCA9IGluc2VydF9lbnRyaWVzKHBhbmUsIGVudHJ5LT5kb3duLCBwYXR0ZXJuLCBmaWx0ZXJfZmxhZ3MsIGlkeCk7Cgl9CgoJU2hvd1dpbmRvdyhwYW5lLT5od25kLCBTV19TSE9XKTsKCglyZXR1cm4gaWR4Owp9CgoKc3RhdGljIHZvaWQgZm9ybWF0X2J5dGVzKExQVFNUUiBidWZmZXIsIExPTkdMT05HIGJ5dGVzKQp7CglzdGF0aWMgY29uc3QgVENIQVIgc0ZtdEdCW10gPSB7JyUnLCAnLicsICcxJywgJ2YnLCAnICcsICdHJywgJ0InLCAnXDAnfTsKCXN0YXRpYyBjb25zdCBUQ0hBUiBzRm10TUJbXSA9IHsnJScsICcuJywgJzEnLCAnZicsICcgJywgJ00nLCAnQicsICdcMCd9OwoJc3RhdGljIGNvbnN0IFRDSEFSIHNGbXRrQltdID0geyclJywgJy4nLCAnMScsICdmJywgJyAnLCAnaycsICdCJywgJ1wwJ307CgoJZmxvYXQgZkJ5dGVzID0gKGZsb2F0KWJ5dGVzOwoKCWlmIChieXRlcyA+PSAxMDczNzQxODI0KQkvKiAxIEdCICovCgkJX3N0cHJpbnRmKGJ1ZmZlciwgc0ZtdEdCLCBmQnl0ZXMvMTA3Mzc0MTgyNC5mKy41Zik7CgllbHNlIGlmIChieXRlcyA+PSAxMDQ4NTc2KQkvKiAxIE1CICovCgkJX3N0cHJpbnRmKGJ1ZmZlciwgc0ZtdE1CLCBmQnl0ZXMvMTA0ODU3Ni5mKy41Zik7CgllbHNlIGlmIChieXRlcyA+PSAxMDI0KQkJLyogMSBrQiAqLwoJCV9zdHByaW50ZihidWZmZXIsIHNGbXRrQiwgZkJ5dGVzLzEwMjQuZisuNWYpOwoJZWxzZQoJCV9zdHByaW50ZihidWZmZXIsIHNMb25nTnVtRm10LCBieXRlcyk7Cn0KCnN0YXRpYyB2b2lkIHNldF9zcGFjZV9zdGF0dXModm9pZCkKewoJVUxBUkdFX0lOVEVHRVIgdWxGcmVlQnl0ZXNUb0NhbGxlciwgdWxUb3RhbEJ5dGVzLCB1bEZyZWVCeXRlczsKCVRDSEFSIGZtdFs2NF0sIGIxWzY0XSwgYjJbNjRdLCBidWZmZXJbQlVGRkVSX0xFTl07CgoJaWYgKEdldERpc2tGcmVlU3BhY2VFeChOVUxMLCAmdWxGcmVlQnl0ZXNUb0NhbGxlciwgJnVsVG90YWxCeXRlcywgJnVsRnJlZUJ5dGVzKSkgewoJCWZvcm1hdF9ieXRlcyhiMSwgdWxGcmVlQnl0ZXNUb0NhbGxlci5RdWFkUGFydCk7CgkJZm9ybWF0X2J5dGVzKGIyLCB1bFRvdGFsQnl0ZXMuUXVhZFBhcnQpOwoJCXdzcHJpbnRmKGJ1ZmZlciwgUlMoZm10LElEU19GUkVFX1NQQUNFX0ZNVCksIGIxLCBiMik7Cgl9IGVsc2UKCQlsc3RyY3B5KGJ1ZmZlciwgc1FNYXJrcyk7CgoJU2VuZE1lc3NhZ2UoR2xvYmFscy5oc3RhdHVzYmFyLCBTQl9TRVRURVhULCAwLCAoTFBBUkFNKWJ1ZmZlcik7Cn0KCgpzdGF0aWMgV05EUFJPQyBnX29yZ1RyZWVXbmRQcm9jOwoKc3RhdGljIHZvaWQgY3JlYXRlX3RyZWVfd2luZG93KEhXTkQgcGFyZW50LCBQYW5lKiBwYW5lLCBVSU5UIGlkLCBVSU5UIGlkX2hlYWRlciwgTFBDVFNUUiBwYXR0ZXJuLCBpbnQgZmlsdGVyX2ZsYWdzKQp7CglzdGF0aWMgY29uc3QgVENIQVIgc0xpc3RCb3hbXSA9IHsnTCcsJ2knLCdzJywndCcsJ0InLCdvJywneCcsJ1wwJ307CgoJc3RhdGljIGludCBzX2luaXQgPSAwOwoJRW50cnkqIGVudHJ5ID0gcGFuZS0+cm9vdDsKCglwYW5lLT5od25kID0gQ3JlYXRlV2luZG93KHNMaXN0Qm94LCBzRW1wdHksIFdTX0NISUxEfFdTX1ZJU0lCTEV8V1NfSFNDUk9MTHxXU19WU0NST0xMfAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTEJTX0RJU0FCTEVOT1NDUk9MTHxMQlNfTk9JTlRFR1JBTEhFSUdIVHxMQlNfT1dORVJEUkFXRklYRUR8TEJTX05PVElGWSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsIDAsIDAsIDAsIHBhcmVudCwgKEhNRU5VKVVMb25nVG9IYW5kbGUoaWQpLCBHbG9iYWxzLmhJbnN0YW5jZSwgMCk7CgoJU2V0V2luZG93TG9uZ1B0cihwYW5lLT5od25kLCBHV0xQX1VTRVJEQVRBLCAoTFBBUkFNKXBhbmUpOwoJZ19vcmdUcmVlV25kUHJvYyA9IChXTkRQUk9DKSBTZXRXaW5kb3dMb25nUHRyKHBhbmUtPmh3bmQsIEdXTFBfV05EUFJPQywgKExQQVJBTSlUcmVlV25kUHJvYyk7CgoJU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgV01fU0VURk9OVCwgKFdQQVJBTSlHbG9iYWxzLmhmb250LCBGQUxTRSk7CgoJLyogaW5zZXJ0IGVudHJpZXMgaW50byBsaXN0Ym94ICovCglpZiAoZW50cnkpCgkJaW5zZXJ0X2VudHJpZXMocGFuZSwgZW50cnksIHBhdHRlcm4sIGZpbHRlcl9mbGFncywgLTEpOwoKCS8qIGNhbGN1bGF0ZSBjb2x1bW4gd2lkdGhzICovCglpZiAoIXNfaW5pdCkgewoJCXNfaW5pdCA9IDE7CgkJaW5pdF9vdXRwdXQocGFuZS0+aHduZCk7Cgl9CgoJY2FsY193aWR0aHMocGFuZSwgVFJVRSk7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCglwYW5lLT5od25kSGVhZGVyID0gY3JlYXRlX2hlYWRlcihwYXJlbnQsIHBhbmUsIGlkX2hlYWRlcik7CiNlbmRpZgp9CgoKc3RhdGljIHZvaWQgSW5pdENoaWxkV2luZG93KENoaWxkV25kKiBjaGlsZCkKewoJY3JlYXRlX3RyZWVfd2luZG93KGNoaWxkLT5od25kLCAmY2hpbGQtPmxlZnQsIElEV19UUkVFX0xFRlQsIElEV19IRUFERVJfTEVGVCwgTlVMTCwgVEZfQUxMKTsKCWNyZWF0ZV90cmVlX3dpbmRvdyhjaGlsZC0+aHduZCwgJmNoaWxkLT5yaWdodCwgSURXX1RSRUVfUklHSFQsIElEV19IRUFERVJfUklHSFQsIGNoaWxkLT5maWx0ZXJfcGF0dGVybiwgY2hpbGQtPmZpbHRlcl9mbGFncyk7Cn0KCgpzdGF0aWMgdm9pZCBmb3JtYXRfZGF0ZShjb25zdCBGSUxFVElNRSogZnQsIFRDSEFSKiBidWZmZXIsIGludCB2aXNpYmxlX2NvbHMpCnsKCVNZU1RFTVRJTUUgc3lzdGltZTsKCUZJTEVUSU1FIGxmdDsKCWludCBsZW4gPSAwOwoKCSpidWZmZXIgPSAnXDAnOwoKCWlmICghZnQtPmR3TG93RGF0ZVRpbWUgJiYgIWZ0LT5kd0hpZ2hEYXRlVGltZSkKCQlyZXR1cm47CgoJaWYgKCFGaWxlVGltZVRvTG9jYWxGaWxlVGltZShmdCwgJmxmdCkpCgkJe2VycjogbHN0cmNweShidWZmZXIsc1FNYXJrcyk7IHJldHVybjt9CgoJaWYgKCFGaWxlVGltZVRvU3lzdGVtVGltZSgmbGZ0LCAmc3lzdGltZSkpCgkJZ290byBlcnI7CgoJaWYgKHZpc2libGVfY29scyAmIENPTF9EQVRFKSB7CgkJbGVuID0gR2V0RGF0ZUZvcm1hdChMT0NBTEVfVVNFUl9ERUZBVUxULCAwLCAmc3lzdGltZSwgMCwgYnVmZmVyLCBCVUZGRVJfTEVOKTsKCQlpZiAoIWxlbikKCQkJZ290byBlcnI7Cgl9CgoJaWYgKHZpc2libGVfY29scyAmIENPTF9USU1FKSB7CgkJaWYgKGxlbikKCQkJYnVmZmVyW2xlbi0xXSA9ICcgJzsKCgkJYnVmZmVyW2xlbisrXSA9ICcgJzsKCgkJaWYgKCFHZXRUaW1lRm9ybWF0KExPQ0FMRV9VU0VSX0RFRkFVTFQsIDAsICZzeXN0aW1lLCAwLCBidWZmZXIrbGVuLCBCVUZGRVJfTEVOLWxlbikpCgkJCWJ1ZmZlcltsZW5dID0gJ1wwJzsKCX0KfQoKCnN0YXRpYyB2b2lkIGNhbGNfd2lkdGgoUGFuZSogcGFuZSwgTFBEUkFXSVRFTVNUUlVDVCBkaXMsIGludCBjb2wsIExQQ1RTVFIgc3RyKQp7CglSRUNUIHJ0ID0gezAsIDAsIDAsIDB9OwoKCURyYXdUZXh0KGRpcy0+aERDLCBzdHIsIC0xLCAmcnQsIERUX0NBTENSRUNUfERUX1NJTkdMRUxJTkV8RFRfTk9QUkVGSVgpOwoKCWlmIChydC5yaWdodCA+IHBhbmUtPndpZHRoc1tjb2xdKQoJCXBhbmUtPndpZHRoc1tjb2xdID0gcnQucmlnaHQ7Cn0KCnN0YXRpYyB2b2lkIGNhbGNfdGFiYmVkX3dpZHRoKFBhbmUqIHBhbmUsIExQRFJBV0lURU1TVFJVQ1QgZGlzLCBpbnQgY29sLCBMUENUU1RSIHN0cikKewoJUkVDVCBydCA9IHswLCAwLCAwLCAwfTsKCi8qCURSQVdURVhUUEFSQU1TIGR0cCA9IHtzaXplb2YoRFJBV1RFWFRQQVJBTVMpLCAyfTsKCURyYXdUZXh0RXgoZGlzLT5oREMsIChMUFRTVFIpc3RyLCAtMSwgJnJ0LCBEVF9DQUxDUkVDVHxEVF9TSU5HTEVMSU5FfERUX05PUFJFRklYfERUX0VYUEFORFRBQlN8RFRfVEFCU1RPUCwgJmR0cCk7Ki8KCglEcmF3VGV4dChkaXMtPmhEQywgc3RyLCAtMSwgJnJ0LCBEVF9DQUxDUkVDVHxEVF9TSU5HTEVMSU5FfERUX0VYUEFORFRBQlN8RFRfVEFCU1RPUHwoMjw8OCkpOwoJLypGSVhNRSBydCAoMCwwKSA/Pz8gKi8KCglpZiAocnQucmlnaHQgPiBwYW5lLT53aWR0aHNbY29sXSkKCQlwYW5lLT53aWR0aHNbY29sXSA9IHJ0LnJpZ2h0Owp9CgoKc3RhdGljIHZvaWQgb3V0cHV0X3RleHQoUGFuZSogcGFuZSwgTFBEUkFXSVRFTVNUUlVDVCBkaXMsIGludCBjb2wsIExQQ1RTVFIgc3RyLCBEV09SRCBmbGFncykKewoJaW50IHggPSBkaXMtPnJjSXRlbS5sZWZ0OwoJUkVDVCBydDsKCglydC5sZWZ0ICAgPSB4K3BhbmUtPnBvc2l0aW9uc1tjb2xdK0dsb2JhbHMuc3BhY2VTaXplLmN4OwoJcnQudG9wICAgID0gZGlzLT5yY0l0ZW0udG9wOwoJcnQucmlnaHQgID0geCtwYW5lLT5wb3NpdGlvbnNbY29sKzFdLUdsb2JhbHMuc3BhY2VTaXplLmN4OwoJcnQuYm90dG9tID0gZGlzLT5yY0l0ZW0uYm90dG9tOwoKCURyYXdUZXh0KGRpcy0+aERDLCBzdHIsIC0xLCAmcnQsIERUX1NJTkdMRUxJTkV8RFRfTk9QUkVGSVh8ZmxhZ3MpOwp9CgpzdGF0aWMgdm9pZCBvdXRwdXRfdGFiYmVkX3RleHQoUGFuZSogcGFuZSwgTFBEUkFXSVRFTVNUUlVDVCBkaXMsIGludCBjb2wsIExQQ1RTVFIgc3RyKQp7CglpbnQgeCA9IGRpcy0+cmNJdGVtLmxlZnQ7CglSRUNUIHJ0OwoKCXJ0LmxlZnQgICA9IHgrcGFuZS0+cG9zaXRpb25zW2NvbF0rR2xvYmFscy5zcGFjZVNpemUuY3g7CglydC50b3AgICAgPSBkaXMtPnJjSXRlbS50b3A7CglydC5yaWdodCAgPSB4K3BhbmUtPnBvc2l0aW9uc1tjb2wrMV0tR2xvYmFscy5zcGFjZVNpemUuY3g7CglydC5ib3R0b20gPSBkaXMtPnJjSXRlbS5ib3R0b207CgovKglEUkFXVEVYVFBBUkFNUyBkdHAgPSB7c2l6ZW9mKERSQVdURVhUUEFSQU1TKSwgMn07CglEcmF3VGV4dEV4KGRpcy0+aERDLCAoTFBUU1RSKXN0ciwgLTEsICZydCwgRFRfU0lOR0xFTElORXxEVF9OT1BSRUZJWHxEVF9FWFBBTkRUQUJTfERUX1RBQlNUT1AsICZkdHApOyovCgoJRHJhd1RleHQoZGlzLT5oREMsIHN0ciwgLTEsICZydCwgRFRfU0lOR0xFTElORXxEVF9FWFBBTkRUQUJTfERUX1RBQlNUT1B8KDI8PDgpKTsKfQoKc3RhdGljIHZvaWQgb3V0cHV0X251bWJlcihQYW5lKiBwYW5lLCBMUERSQVdJVEVNU1RSVUNUIGRpcywgaW50IGNvbCwgTFBDVFNUUiBzdHIpCnsKCWludCB4ID0gZGlzLT5yY0l0ZW0ubGVmdDsKCVJFQ1QgcnQ7CglMUENUU1RSIHMgPSBzdHI7CglUQ0hBUiBiWzEyOF07CglMUFRTVFIgZCA9IGI7CglpbnQgcG9zOwoKCXJ0LmxlZnQgICA9IHgrcGFuZS0+cG9zaXRpb25zW2NvbF0rR2xvYmFscy5zcGFjZVNpemUuY3g7CglydC50b3AgICAgPSBkaXMtPnJjSXRlbS50b3A7CglydC5yaWdodCAgPSB4K3BhbmUtPnBvc2l0aW9uc1tjb2wrMV0tR2xvYmFscy5zcGFjZVNpemUuY3g7CglydC5ib3R0b20gPSBkaXMtPnJjSXRlbS5ib3R0b207CgoJaWYgKCpzKQoJCSpkKysgPSAqcysrOwoKCS8qIGluc2VydCBudW1iZXIgc2VwYXJhdG9yIGNoYXJhY3RlcnMgKi8KCXBvcyA9IGxzdHJsZW4ocykgJSAzOwoKCXdoaWxlKCpzKQoJCWlmIChwb3MtLSkKCQkJKmQrKyA9ICpzKys7CgkJZWxzZSB7CgkJCSpkKysgPSBHbG9iYWxzLm51bV9zZXA7CgkJCXBvcyA9IDM7CgkJfQoKCURyYXdUZXh0KGRpcy0+aERDLCBiLCBkLWIsICZydCwgRFRfUklHSFR8RFRfU0lOR0xFTElORXxEVF9OT1BSRUZJWHxEVF9FTkRfRUxMSVBTSVMpOwp9CgoKc3RhdGljIEJPT0wgaXNfZXhlX2ZpbGUoTFBDVFNUUiBleHQpCnsKCXN0YXRpYyBjb25zdCBUQ0hBUiBleGVjdXRhYmxlX2V4dGVuc2lvbnNbXVs0XSA9IHsKCQl7J0MnLCdPJywnTScsJ1wwJ30sCgkJeydFJywnWCcsJ0UnLCdcMCd9LAoJCXsnQicsJ0EnLCdUJywnXDAnfSwKCQl7J0MnLCdNJywnRCcsJ1wwJ30sCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQl7J0MnLCdNJywnTScsJ1wwJ30sCgkJeydCJywnVCcsJ00nLCdcMCd9LAoJCXsnQScsJ1cnLCdLJywnXDAnfSwKI2VuZGlmIC8qIF9OT19FWFRFTlNJT05TICovCgkJeydcMCd9Cgl9OwoKCVRDSEFSIGV4dF9idWZmZXJbX01BWF9FWFRdOwoJY29uc3QgVENIQVIgKCpwKVs0XTsKCUxQQ1RTVFIgczsKCUxQVFNUUiBkOwoKCWZvcihzPWV4dCsxLGQ9ZXh0X2J1ZmZlcjsgKCpkPXRvbG93ZXIoKnMpKTsgcysrKQoJCWQrKzsKCglmb3IocD1leGVjdXRhYmxlX2V4dGVuc2lvbnM7ICgqcClbMF07IHArKykKCQlpZiAoIWxzdHJjbXBpKGV4dF9idWZmZXIsICpwKSkKCQkJcmV0dXJuIFRSVUU7CgoJcmV0dXJuIEZBTFNFOwp9CgpzdGF0aWMgQk9PTCBpc19yZWdpc3RlcmVkX3R5cGUoTFBDVFNUUiBleHQpCnsKCS8qIGNoZWNrIGlmIHRoZXJlIGV4aXN0cyBhIGNsYXNzbmFtZSBmb3IgdGhpcyBmaWxlIGV4dGVuc2lvbiBpbiB0aGUgcmVnaXN0cnkgKi8KCWlmICghUmVnUXVlcnlWYWx1ZShIS0VZX0NMQVNTRVNfUk9PVCwgZXh0LCBOVUxMLCBOVUxMKSkKCQlyZXR1cm4gVFJVRTsKCglyZXR1cm4gRkFMU0U7Cn0KCnN0YXRpYyBlbnVtIEZJTEVfVFlQRSBnZXRfZmlsZV90eXBlKExQQ1RTVFIgZmlsZW5hbWUpCnsKCUxQQ1RTVFIgZXh0ID0gX3Rjc3JjaHIoZmlsZW5hbWUsICcuJyk7CglpZiAoIWV4dCkKCQlleHQgPSBzRW1wdHk7CgoJaWYgKGlzX2V4ZV9maWxlKGV4dCkpCgkJcmV0dXJuIEZUX0VYRUNVVEFCTEU7CgllbHNlIGlmIChpc19yZWdpc3RlcmVkX3R5cGUoZXh0KSkKCQlyZXR1cm4gRlRfRE9DVU1FTlQ7CgllbHNlCgkJcmV0dXJuIEZUX09USEVSOwp9CgoKc3RhdGljIHZvaWQgZHJhd19pdGVtKFBhbmUqIHBhbmUsIExQRFJBV0lURU1TVFJVQ1QgZGlzLCBFbnRyeSogZW50cnksIGludCBjYWxjV2lkdGhDb2wpCnsKCVRDSEFSIGJ1ZmZlcltCVUZGRVJfTEVOXTsKCURXT1JEIGF0dHJzOwoJaW50IHZpc2libGVfY29scyA9IHBhbmUtPnZpc2libGVfY29sczsKCUNPTE9SUkVGIGJrY29sb3IsIHRleHRjb2xvcjsKCVJFQ1QgZm9jdXNSZWN0ID0gZGlzLT5yY0l0ZW07CglIQlJVU0ggaGJydXNoOwoJZW51bSBJTUFHRSBpbWc7CglpbnQgaW1nX3BvcywgY3g7CglpbnQgY29sID0gMDsKCglpZiAoZW50cnkpIHsKCQlhdHRycyA9IGVudHJ5LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXM7CgoJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWSkgewoJCQlpZiAoZW50cnktPmRhdGEuY0ZpbGVOYW1lWzBdID09ICcuJyAmJiBlbnRyeS0+ZGF0YS5jRmlsZU5hbWVbMV0gPT0gJy4nCgkJCQkJJiYgZW50cnktPmRhdGEuY0ZpbGVOYW1lWzJdID09ICdcMCcpCgkJCQlpbWcgPSBJTUdfRk9MREVSX1VQOwojaWZuZGVmIF9OT19FWFRFTlNJT05TCgkJCWVsc2UgaWYgKGVudHJ5LT5kYXRhLmNGaWxlTmFtZVswXSA9PSAnLicgJiYgZW50cnktPmRhdGEuY0ZpbGVOYW1lWzFdID09ICdcMCcpCgkJCQlpbWcgPSBJTUdfRk9MREVSX0NVUjsKI2VuZGlmCgkJCWVsc2UgaWYgKAojaWZkZWYgX05PX0VYVEVOU0lPTlMKCQkJCQkgZW50cnktPmV4cGFuZGVkIHx8CiNlbmRpZgoJCQkJCSAocGFuZS0+dHJlZVBhbmUgJiYgKGRpcy0+aXRlbVN0YXRlJk9EU19GT0NVUykpKQoJCQkJaW1nID0gSU1HX09QRU5fRk9MREVSOwoJCQllbHNlCgkJCQlpbWcgPSBJTUdfRk9MREVSOwoJCX0gZWxzZSB7CgkJCXN3aXRjaChnZXRfZmlsZV90eXBlKGVudHJ5LT5kYXRhLmNGaWxlTmFtZSkpIHsKCQkJICBjYXNlIEZUX0VYRUNVVEFCTEU6CWltZyA9IElNR19FWEVDVVRBQkxFOwlicmVhazsKCQkJICBjYXNlIEZUX0RPQ1VNRU5UOgkJaW1nID0gSU1HX0RPQ1VNRU5UOwkJYnJlYWs7CgkJCSAgZGVmYXVsdDoJCQkJaW1nID0gSU1HX0ZJTEU7CgkJCX0KCQl9Cgl9IGVsc2UgewoJCWF0dHJzID0gMDsKCQlpbWcgPSBJTUdfTk9ORTsKCX0KCglpZiAocGFuZS0+dHJlZVBhbmUpIHsKCQlpZiAoZW50cnkpIHsKCQkJaW1nX3BvcyA9IGRpcy0+cmNJdGVtLmxlZnQgKyBlbnRyeS0+bGV2ZWwqKElNQUdFX1dJRFRIK1RSRUVfTElORV9EWCk7CgoJCQlpZiAoY2FsY1dpZHRoQ29sID09IC0xKSB7CgkJCQlpbnQgeDsKCQkJCWludCB5ID0gZGlzLT5yY0l0ZW0udG9wICsgSU1BR0VfSEVJR0hULzI7CgkJCQlFbnRyeSogdXA7CgkJCQlSRUNUIHJ0X2NsaXA7CgkJCQlIUkdOIGhyZ25fb3JnID0gQ3JlYXRlUmVjdFJnbigwLCAwLCAwLCAwKTsKCQkJCUhSR04gaHJnbjsKCgkJCQlydF9jbGlwLmxlZnQgICA9IGRpcy0+cmNJdGVtLmxlZnQ7CgkJCQlydF9jbGlwLnRvcCAgICA9IGRpcy0+cmNJdGVtLnRvcDsKCQkJCXJ0X2NsaXAucmlnaHQgID0gZGlzLT5yY0l0ZW0ubGVmdCtwYW5lLT53aWR0aHNbY29sXTsKCQkJCXJ0X2NsaXAuYm90dG9tID0gZGlzLT5yY0l0ZW0uYm90dG9tOwoKCQkJCWhyZ24gPSBDcmVhdGVSZWN0UmduSW5kaXJlY3QoJnJ0X2NsaXApOwoKCQkJCWlmICghR2V0Q2xpcFJnbihkaXMtPmhEQywgaHJnbl9vcmcpKSB7CgkJCQkJRGVsZXRlT2JqZWN0KGhyZ25fb3JnKTsKCQkJCQlocmduX29yZyA9IDA7CgkJCQl9CgoJCQkJLyogSEdESU9CSiBob2xkUGVuID0gU2VsZWN0T2JqZWN0KGRpcy0+aERDLCBHZXRTdG9ja09iamVjdChCTEFDS19QRU4pKTsgKi8KCQkJCUV4dFNlbGVjdENsaXBSZ24oZGlzLT5oREMsIGhyZ24sIFJHTl9BTkQpOwoJCQkJRGVsZXRlT2JqZWN0KGhyZ24pOwoKCQkJCWlmICgodXA9ZW50cnktPnVwKSAhPSBOVUxMKSB7CgkJCQkJTW92ZVRvRXgoZGlzLT5oREMsIGltZ19wb3MtSU1BR0VfV0lEVEgvMiwgeSwgMCk7CgkJCQkJTGluZVRvKGRpcy0+aERDLCBpbWdfcG9zLTIsIHkpOwoKCQkJCQl4ID0gaW1nX3BvcyAtIElNQUdFX1dJRFRILzI7CgoJCQkJCWRvIHsKCQkJCQkJeCAtPSBJTUFHRV9XSURUSCtUUkVFX0xJTkVfRFg7CgoJCQkJCQlpZiAodXAtPm5leHQKI2lmbmRlZiBfTEVGVF9GSUxFUwoJCQkJCQkJJiYgKHVwLT5uZXh0LT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMgJiBGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpCiNlbmRpZgoJCQkJCQkJKSB7CgkJCQkJCQlNb3ZlVG9FeChkaXMtPmhEQywgeCwgZGlzLT5yY0l0ZW0udG9wLCAwKTsKCQkJCQkJCUxpbmVUbyhkaXMtPmhEQywgeCwgZGlzLT5yY0l0ZW0uYm90dG9tKTsKCQkJCQkJfQoJCQkJCX0gd2hpbGUoKHVwPXVwLT51cCkgIT0gTlVMTCk7CgkJCQl9CgoJCQkJeCA9IGltZ19wb3MgLSBJTUFHRV9XSURUSC8yOwoKCQkJCU1vdmVUb0V4KGRpcy0+aERDLCB4LCBkaXMtPnJjSXRlbS50b3AsIDApOwoJCQkJTGluZVRvKGRpcy0+aERDLCB4LCB5KTsKCgkJCQlpZiAoZW50cnktPm5leHQKI2lmbmRlZiBfTEVGVF9GSUxFUwoJCQkJCSYmIChlbnRyeS0+bmV4dC0+ZGF0YS5kd0ZpbGVBdHRyaWJ1dGVzJkZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWSkKI2VuZGlmCgkJCQkJKQoJCQkJCUxpbmVUbyhkaXMtPmhEQywgeCwgZGlzLT5yY0l0ZW0uYm90dG9tKTsKCgkJCQlTZWxlY3RDbGlwUmduKGRpcy0+aERDLCBocmduX29yZyk7CgkJCQlpZiAoaHJnbl9vcmcpIERlbGV0ZU9iamVjdChocmduX29yZyk7CgkJCQkvKiBTZWxlY3RPYmplY3QoZGlzLT5oREMsIGhvbGRQZW4pOyAqLwoJCQl9IGVsc2UgaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykgewoJCQkJaW50IHJpZ2h0ID0gaW1nX3BvcyArIElNQUdFX1dJRFRIIC0gVFJFRV9MSU5FX0RYOwoKCQkJCWlmIChyaWdodCA+IHBhbmUtPndpZHRoc1tjb2xdKQoJCQkJCXBhbmUtPndpZHRoc1tjb2xdID0gcmlnaHQ7CgkJCX0KCQl9IGVsc2UgIHsKCQkJaW1nX3BvcyA9IGRpcy0+cmNJdGVtLmxlZnQ7CgkJfQoJfSBlbHNlIHsKCQlpbWdfcG9zID0gZGlzLT5yY0l0ZW0ubGVmdDsKCgkJaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykKCQkJcGFuZS0+d2lkdGhzW2NvbF0gPSBJTUFHRV9XSURUSDsKCX0KCglpZiAoY2FsY1dpZHRoQ29sID09IC0xKSB7CgkJZm9jdXNSZWN0LmxlZnQgPSBpbWdfcG9zIC0yOwoKI2lmZGVmIF9OT19FWFRFTlNJT05TCgkJaWYgKHBhbmUtPnRyZWVQYW5lICYmIGVudHJ5KSB7CgkJCVJFQ1QgcnQgPSB7MH07CgoJCQlEcmF3VGV4dChkaXMtPmhEQywgZW50cnktPmRhdGEuY0ZpbGVOYW1lLCAtMSwgJnJ0LCBEVF9DQUxDUkVDVHxEVF9TSU5HTEVMSU5FfERUX05PUFJFRklYKTsKCgkJCWZvY3VzUmVjdC5yaWdodCA9IGRpcy0+cmNJdGVtLmxlZnQrcGFuZS0+cG9zaXRpb25zW2NvbCsxXStUUkVFX0xJTkVfRFggKyBydC5yaWdodCArMjsKCQl9CiNlbHNlCgoJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX0NPTVBSRVNTRUQpCgkJCXRleHRjb2xvciA9IENPTE9SX0NPTVBSRVNTRUQ7CgkJZWxzZQojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KCQkJdGV4dGNvbG9yID0gUkdCKDAsMCwwKTsKCgkJaWYgKGRpcy0+aXRlbVN0YXRlICYgT0RTX0ZPQ1VTKSB7CgkJCXRleHRjb2xvciA9IFJHQigyNTUsMjU1LDI1NSk7CgkJCWJrY29sb3IgPSBDT0xPUl9TRUxFQ1RJT047CgkJfSBlbHNlIHsKCQkJYmtjb2xvciA9IFJHQigyNTUsMjU1LDI1NSk7CgkJfQoKCQloYnJ1c2ggPSBDcmVhdGVTb2xpZEJydXNoKGJrY29sb3IpOwoJCUZpbGxSZWN0KGRpcy0+aERDLCAmZm9jdXNSZWN0LCBoYnJ1c2gpOwoJCURlbGV0ZU9iamVjdChoYnJ1c2gpOwoKCQlTZXRCa01vZGUoZGlzLT5oREMsIFRSQU5TUEFSRU5UKTsKCQlTZXRUZXh0Q29sb3IoZGlzLT5oREMsIHRleHRjb2xvcik7CgoJCWN4ID0gcGFuZS0+d2lkdGhzW2NvbF07CgoJCWlmIChjeCAmJiBpbWchPUlNR19OT05FKSB7CgkJCWlmIChjeCA+IElNQUdFX1dJRFRIKQoJCQkJY3ggPSBJTUFHRV9XSURUSDsKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJCQlpZiAoZW50cnktPmhpY29uICYmIGVudHJ5LT5oaWNvbiE9KEhJQ09OKS0xKQoJCQkJRHJhd0ljb25FeChkaXMtPmhEQywgaW1nX3BvcywgZGlzLT5yY0l0ZW0udG9wLCBlbnRyeS0+aGljb24sIGN4LCBHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZU01JQ09OKSwgMCwgMCwgRElfTk9STUFMKTsKCQkJZWxzZQojZW5kaWYKCQkJCUltYWdlTGlzdF9EcmF3RXgoR2xvYmFscy5oaW1sLCBpbWcsIGRpcy0+aERDLAoJCQkJCQkJCSBpbWdfcG9zLCBkaXMtPnJjSXRlbS50b3AsIGN4LAoJCQkJCQkJCSBJTUFHRV9IRUlHSFQsIGJrY29sb3IsIENMUl9ERUZBVUxULCBJTERfTk9STUFMKTsKCQl9Cgl9CgoJaWYgKCFlbnRyeSkKCQlyZXR1cm47CgojaWZkZWYgX05PX0VYVEVOU0lPTlMKCWlmIChpbWcgPj0gSU1HX0ZPTERFUl9VUCkKCQlyZXR1cm47CiNlbmRpZgoKCWNvbCsrOwoKCS8qIG91cHV0IGZpbGUgbmFtZSAqLwoJaWYgKGNhbGNXaWR0aENvbCA9PSAtMSkKCQlvdXRwdXRfdGV4dChwYW5lLCBkaXMsIGNvbCwgZW50cnktPmRhdGEuY0ZpbGVOYW1lLCAwKTsKCWVsc2UgaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykKCQljYWxjX3dpZHRoKHBhbmUsIGRpcywgY29sLCBlbnRyeS0+ZGF0YS5jRmlsZU5hbWUpOwoKCWNvbCsrOwoKI2lmZGVmIF9OT19FWFRFTlNJT05TCiAgaWYgKCFwYW5lLT50cmVlUGFuZSkgewojZW5kaWYKCiAgICAgICAgLyogZGlzcGxheSBmaWxlIHNpemUgKi8KCWlmICh2aXNpYmxlX2NvbHMgJiBDT0xfU0laRSkgewojaWZkZWYgX05PX0VYVEVOU0lPTlMKCQlpZiAoIShhdHRycyZGSUxFX0FUVFJJQlVURV9ESVJFQ1RPUlkpKQojZW5kaWYKCQl7CgkJCVVMT05HTE9ORyBzaXplOwoKICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZSA9ICgoVUxPTkdMT05HKWVudHJ5LT5kYXRhLm5GaWxlU2l6ZUhpZ2ggPDwgMzIpIHwgZW50cnktPmRhdGEubkZpbGVTaXplTG93OwoKCQkJX3N0cHJpbnRmKGJ1ZmZlciwgc0xvbmdOdW1GbXQsIHNpemUpOwoKCQkJaWYgKGNhbGNXaWR0aENvbCA9PSAtMSkKCQkJCW91dHB1dF9udW1iZXIocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlcik7CgkJCWVsc2UgaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykKCQkJCWNhbGNfd2lkdGgocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlcik7LypUT0RPOiBub3QgZXZlciB0aW1lIGVub3VnaCAqLwoJCX0KCgkJY29sKys7Cgl9CgoJLyogZGlzcGxheSBmaWxlIGRhdGUgKi8KCWlmICh2aXNpYmxlX2NvbHMgJiAoQ09MX0RBVEV8Q09MX1RJTUUpKSB7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQlmb3JtYXRfZGF0ZSgmZW50cnktPmRhdGEuZnRDcmVhdGlvblRpbWUsIGJ1ZmZlciwgdmlzaWJsZV9jb2xzKTsKCQlpZiAoY2FsY1dpZHRoQ29sID09IC0xKQoJCQlvdXRwdXRfdGV4dChwYW5lLCBkaXMsIGNvbCwgYnVmZmVyLCAwKTsKCQllbHNlIGlmIChjYWxjV2lkdGhDb2w9PWNvbCB8fCBjYWxjV2lkdGhDb2w9PUNPTFVNTlMpCgkJCWNhbGNfd2lkdGgocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlcik7CgkJY29sKys7CgoJCWZvcm1hdF9kYXRlKCZlbnRyeS0+ZGF0YS5mdExhc3RBY2Nlc3NUaW1lLCBidWZmZXIsIHZpc2libGVfY29scyk7CgkJaWYgKGNhbGNXaWR0aENvbCA9PSAtMSkKCQkJb3V0cHV0X3RleHQocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlciwgMCk7CgkJZWxzZSBpZiAoY2FsY1dpZHRoQ29sPT1jb2wgfHwgY2FsY1dpZHRoQ29sPT1DT0xVTU5TKQoJCQljYWxjX3dpZHRoKHBhbmUsIGRpcywgY29sLCBidWZmZXIpOwoJCWNvbCsrOwojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KCgkJZm9ybWF0X2RhdGUoJmVudHJ5LT5kYXRhLmZ0TGFzdFdyaXRlVGltZSwgYnVmZmVyLCB2aXNpYmxlX2NvbHMpOwoJCWlmIChjYWxjV2lkdGhDb2wgPT0gLTEpCgkJCW91dHB1dF90ZXh0KHBhbmUsIGRpcywgY29sLCBidWZmZXIsIDApOwoJCWVsc2UgaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykKCQkJY2FsY193aWR0aChwYW5lLCBkaXMsIGNvbCwgYnVmZmVyKTsKCQljb2wrKzsKCX0KCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCWlmIChlbnRyeS0+YmhmaV92YWxpZCkgewogICAgICAgICAgICBVTE9OR0xPTkcgaW5kZXggPSAoKFVMT05HTE9ORyllbnRyeS0+YmhmaS5uRmlsZUluZGV4SGlnaCA8PCAzMikgfCBlbnRyeS0+YmhmaS5uRmlsZUluZGV4TG93OwoKCQlpZiAodmlzaWJsZV9jb2xzICYgQ09MX0lOREVYKSB7CgkJCV9zdHByaW50ZihidWZmZXIsIHNMb25nSGV4Rm10LCBpbmRleCk7CgoJCQlpZiAoY2FsY1dpZHRoQ29sID09IC0xKQoJCQkJb3V0cHV0X3RleHQocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlciwgRFRfUklHSFQpOwoJCQllbHNlIGlmIChjYWxjV2lkdGhDb2w9PWNvbCB8fCBjYWxjV2lkdGhDb2w9PUNPTFVNTlMpCgkJCQljYWxjX3dpZHRoKHBhbmUsIGRpcywgY29sLCBidWZmZXIpOwoKCQkJY29sKys7CgkJfQoKCQlpZiAodmlzaWJsZV9jb2xzICYgQ09MX0xJTktTKSB7CgkJCXdzcHJpbnRmKGJ1ZmZlciwgc051bUZtdCwgZW50cnktPmJoZmkubk51bWJlck9mTGlua3MpOwoKCQkJaWYgKGNhbGNXaWR0aENvbCA9PSAtMSkKCQkJCW91dHB1dF90ZXh0KHBhbmUsIGRpcywgY29sLCBidWZmZXIsIERUX0NFTlRFUik7CgkJCWVsc2UgaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykKCQkJCWNhbGNfd2lkdGgocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlcik7CgoJCQljb2wrKzsKCQl9Cgl9IGVsc2UKCQljb2wgKz0gMjsKI2VuZGlmIC8qIF9OT19FWFRFTlNJT05TICovCgoJLyogc2hvdyBmaWxlIGF0dHJpYnV0ZXMgKi8KCWlmICh2aXNpYmxlX2NvbHMgJiBDT0xfQVRUUklCVVRFUykgewojaWZkZWYgX05PX0VYVEVOU0lPTlMKCQlzdGF0aWMgY29uc3QgVENIQVIgczRUYWJzW10gPSB7JyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcMCd9OwoJCWxzdHJjcHkoYnVmZmVyLCBzNFRhYnMpOwojZWxzZQoJCXN0YXRpYyBjb25zdCBUQ0hBUiBzMTFUYWJzW10gPSB7JyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcMCd9OwoJCWxzdHJjcHkoYnVmZmVyLCBzMTFUYWJzKTsKI2VuZGlmCgoJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX05PUk1BTCkJCQkJCWJ1ZmZlclsgMF0gPSAnTic7CgkJZWxzZSB7CgkJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX1JFQURPTkxZKQkJCWJ1ZmZlclsgMl0gPSAnUic7CgkJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX0hJRERFTikJCQkJYnVmZmVyWyA0XSA9ICdIJzsKCQkJaWYgKGF0dHJzICYgRklMRV9BVFRSSUJVVEVfU1lTVEVNKQkJCQlidWZmZXJbIDZdID0gJ1MnOwoJCQlpZiAoYXR0cnMgJiBGSUxFX0FUVFJJQlVURV9BUkNISVZFKQkJCQlidWZmZXJbIDhdID0gJ0EnOwoJCQlpZiAoYXR0cnMgJiBGSUxFX0FUVFJJQlVURV9DT01QUkVTU0VEKQkJCWJ1ZmZlclsxMF0gPSAnQyc7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJaWYgKGF0dHJzICYgRklMRV9BVFRSSUJVVEVfRElSRUNUT1JZKQkJCWJ1ZmZlclsxMl0gPSAnRCc7CgkJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX0VOQ1JZUFRFRCkJCQlidWZmZXJbMTRdID0gJ0UnOwoJCQlpZiAoYXR0cnMgJiBGSUxFX0FUVFJJQlVURV9URU1QT1JBUlkpCQkJYnVmZmVyWzE2XSA9ICdUJzsKCQkJaWYgKGF0dHJzICYgRklMRV9BVFRSSUJVVEVfU1BBUlNFX0ZJTEUpCQkJYnVmZmVyWzE4XSA9ICdQJzsKCQkJaWYgKGF0dHJzICYgRklMRV9BVFRSSUJVVEVfUkVQQVJTRV9QT0lOVCkJCWJ1ZmZlclsyMF0gPSAnUSc7CgkJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX09GRkxJTkUpCQkJCWJ1ZmZlclsyMl0gPSAnTyc7CgkJCWlmIChhdHRycyAmIEZJTEVfQVRUUklCVVRFX05PVF9DT05URU5UX0lOREVYRUQpCWJ1ZmZlclsyNF0gPSAnWCc7CiNlbmRpZiAvKiBfTk9fRVhURU5TSU9OUyAqLwoJCX0KCgkJaWYgKGNhbGNXaWR0aENvbCA9PSAtMSkKCQkJb3V0cHV0X3RhYmJlZF90ZXh0KHBhbmUsIGRpcywgY29sLCBidWZmZXIpOwoJCWVsc2UgaWYgKGNhbGNXaWR0aENvbD09Y29sIHx8IGNhbGNXaWR0aENvbD09Q09MVU1OUykKCQkJY2FsY190YWJiZWRfd2lkdGgocGFuZSwgZGlzLCBjb2wsIGJ1ZmZlcik7CgoJCWNvbCsrOwoJfQoKLypUT0RPCglpZiAoZmxhZ3Muc2VjdXJpdHkpIHsKCQlzdGF0aWMgY29uc3QgVENIQVIgc1NlY1RhYnNbXSA9IHsKCQkJJyAnLCdcdCcsJyAnLCdcdCcsJyAnLCdcdCcsJyAnLAoJCQknICcsJ1x0JywnICcsCgkJCScgJywnXHQnLCcgJywnXHQnLCcgJywnXHQnLCcgJywKCQkJJyAnLCdcdCcsJyAnLAoJCQknICcsJ1x0JywnICcsJ1x0JywnICcsJ1x0JywnICcsCgkJCSdcMCcKCQl9OwoKCQlEV09SRCByaWdodHMgPSBnZXRfYWNjZXNzX21hc2soKTsKCgkJbHN0cmNweShidWZmZXIsIHNTZWNUYWJzKTsKCgkJaWYgKHJpZ2h0cyAmIEZJTEVfUkVBRF9EQVRBKQkJCWJ1ZmZlclsgMF0gPSAnUic7CgkJaWYgKHJpZ2h0cyAmIEZJTEVfV1JJVEVfREFUQSkJCQlidWZmZXJbIDJdID0gJ1cnOwoJCWlmIChyaWdodHMgJiBGSUxFX0FQUEVORF9EQVRBKQkJCWJ1ZmZlclsgNF0gPSAnQSc7CgkJaWYgKHJpZ2h0cyAmIEZJTEVfUkVBRF9FQSkJCQkJe2J1ZmZlcls2XSA9ICdlbnRyeSc7IGJ1ZmZlclsgN10gPSAnUic7fQoJCWlmIChyaWdodHMgJiBGSUxFX1dSSVRFX0VBKQkJCQl7YnVmZmVyWzldID0gJ2VudHJ5JzsgYnVmZmVyWzEwXSA9ICdXJzt9CgkJaWYgKHJpZ2h0cyAmIEZJTEVfRVhFQ1VURSkJCQkJYnVmZmVyWzEyXSA9ICdYJzsKCQlpZiAocmlnaHRzICYgRklMRV9ERUxFVEVfQ0hJTEQpCQkJYnVmZmVyWzE0XSA9ICdEJzsKCQlpZiAocmlnaHRzICYgRklMRV9SRUFEX0FUVFJJQlVURVMpCQl7YnVmZmVyWzE2XSA9ICdhJzsgYnVmZmVyWzE3XSA9ICdSJzt9CgkJaWYgKHJpZ2h0cyAmIEZJTEVfV1JJVEVfQVRUUklCVVRFUykJCXtidWZmZXJbMTldID0gJ2EnOyBidWZmZXJbMjBdID0gJ1cnO30KCQlpZiAocmlnaHRzICYgV1JJVEVfREFDKQkJCQkJYnVmZmVyWzIyXSA9ICdDJzsKCQlpZiAocmlnaHRzICYgV1JJVEVfT1dORVIpCQkJCWJ1ZmZlclsyNF0gPSAnTyc7CgkJaWYgKHJpZ2h0cyAmIFNZTkNIUk9OSVpFKQkJCQlidWZmZXJbMjZdID0gJ1MnOwoKCQlvdXRwdXRfdGV4dChkaXMsIGNvbCsrLCBidWZmZXIsIERUX0xFRlQsIDMsIHBzaXplKTsKCX0KCglpZiAoZmxhZ3MuZGVzY3JpcHRpb24pIHsKCQlnZXRfZGVzY3JpcHRpb24oYnVmZmVyKTsKCQlvdXRwdXRfdGV4dChkaXMsIGNvbCsrLCBidWZmZXIsIDAsIHBzaXplKTsKCX0KKi8KCiNpZmRlZiBfTk9fRVhURU5TSU9OUwogIH0KCiAgICAgICAgLyogZHJhdyBmb2N1cyBmcmFtZSAqLwoJaWYgKChkaXMtPml0ZW1TdGF0ZSZPRFNfRk9DVVMpICYmIGNhbGNXaWR0aENvbD09LTEpIHsKCSAgICAgICAgLyogQ3VycmVudGx5IFswNC8yMDAwXSBXaW5lIG5laXRoZXIgYmVoYXZlcyBleGFjdGx5IHRoZSBzYW1lICovCgkgICAgICAgIC8qIHdheSBhcyBXSU4gOTUgbm9yIGxpa2UgV2luZG93cyBOVC4uLiAqLwoJCUhHRElPQkogbGFzdEJydXNoOwoJCUhQRU4gbGFzdFBlbjsKCQlIUEVOIGhwZW47CgoJCWlmICghKEdldFZlcnNpb24oKSAmIDB4ODAwMDAwMDApKSB7CS8qIFdpbmRvd3MgTlQ/ICovCgkJCUxPR0JSVVNIIGxiID0ge1BTX1NPTElELCBSR0IoMjU1LDI1NSwyNTUpfTsKCQkJaHBlbiA9IEV4dENyZWF0ZVBlbihQU19DT1NNRVRJQ3xQU19BTFRFUk5BVEUsIDEsICZsYiwgMCwgMCk7CgkJfSBlbHNlCgkJCWhwZW4gPSBDcmVhdGVQZW4oUFNfRE9ULCAwLCBSR0IoMjU1LDI1NSwyNTUpKTsKCgkJbGFzdFBlbiA9IFNlbGVjdFBlbihkaXMtPmhEQywgaHBlbik7CgkJbGFzdEJydXNoID0gU2VsZWN0T2JqZWN0KGRpcy0+aERDLCBHZXRTdG9ja09iamVjdChIT0xMT1dfQlJVU0gpKTsKCQlTZXRST1AyKGRpcy0+aERDLCBSMl9YT1JQRU4pOwoJCVJlY3RhbmdsZShkaXMtPmhEQywgZm9jdXNSZWN0LmxlZnQsIGZvY3VzUmVjdC50b3AsIGZvY3VzUmVjdC5yaWdodCwgZm9jdXNSZWN0LmJvdHRvbSk7CgkJU2VsZWN0T2JqZWN0KGRpcy0+aERDLCBsYXN0QnJ1c2gpOwoJCVNlbGVjdE9iamVjdChkaXMtPmhEQywgbGFzdFBlbik7CgkJRGVsZXRlT2JqZWN0KGhwZW4pOwoJfQojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KfQoKCiNpZmRlZiBfTk9fRVhURU5TSU9OUwoKc3RhdGljIHZvaWQgZHJhd19zcGxpdGJhcihIV05EIGh3bmQsIGludCB4KQp7CglSRUNUIHJ0OwoJSERDIGhkYyA9IEdldERDKGh3bmQpOwoKCUdldENsaWVudFJlY3QoaHduZCwgJnJ0KTsKCglydC5sZWZ0ID0geCAtIFNQTElUX1dJRFRILzI7CglydC5yaWdodCA9IHggKyBTUExJVF9XSURUSC8yKzE7CgoJSW52ZXJ0UmVjdChoZGMsICZydCk7CgoJUmVsZWFzZURDKGh3bmQsIGhkYyk7Cn0KCiNlbmRpZiAvKiBfTk9fRVhURU5TSU9OUyAqLwoKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCnN0YXRpYyB2b2lkIHNldF9oZWFkZXIoUGFuZSogcGFuZSkKewoJSERfSVRFTSBpdGVtOwoJaW50IHNjcm9sbF9wb3MgPSBHZXRTY3JvbGxQb3MocGFuZS0+aHduZCwgU0JfSE9SWik7CglpbnQgaT0wLCB4PTA7CgoJaXRlbS5tYXNrID0gSERJX1dJRFRIOwoJaXRlbS5jeHkgPSAwOwoKCWZvcig7IHgrcGFuZS0+d2lkdGhzW2ldPHNjcm9sbF9wb3MgJiYgaTxDT0xVTU5TOyBpKyspIHsKCQl4ICs9IHBhbmUtPndpZHRoc1tpXTsKCQlTZW5kTWVzc2FnZShwYW5lLT5od25kSGVhZGVyLCBIRE1fU0VUSVRFTSwgaSwgKExQQVJBTSkgJml0ZW0pOwoJfQoKCWlmIChpIDwgQ09MVU1OUykgewoJCXggKz0gcGFuZS0+d2lkdGhzW2ldOwoJCWl0ZW0uY3h5ID0geCAtIHNjcm9sbF9wb3M7CgkJU2VuZE1lc3NhZ2UocGFuZS0+aHduZEhlYWRlciwgSERNX1NFVElURU0sIGkrKywgKExQQVJBTSkgJml0ZW0pOwoKCQlmb3IoOyBpPENPTFVNTlM7IGkrKykgewoJCQlpdGVtLmN4eSA9IHBhbmUtPndpZHRoc1tpXTsKCQkJeCArPSBwYW5lLT53aWR0aHNbaV07CgkJCVNlbmRNZXNzYWdlKHBhbmUtPmh3bmRIZWFkZXIsIEhETV9TRVRJVEVNLCBpLCAoTFBBUkFNKSAmaXRlbSk7CgkJfQoJfQp9CgpzdGF0aWMgTFJFU1VMVCBwYW5lX25vdGlmeShQYW5lKiBwYW5lLCBOTUhEUiogcG5taCkKewoJc3dpdGNoKHBubWgtPmNvZGUpIHsKCQljYXNlIEhETl9JVEVNQ0hBTkdFRDogewoJCQlIRF9OT1RJRlkqIHBoZG4gPSAoSERfTk9USUZZKikgcG5taDsKCQkJaW50IGlkeCA9IHBoZG4tPmlJdGVtOwoJCQlpbnQgZHggPSBwaGRuLT5waXRlbS0+Y3h5IC0gcGFuZS0+d2lkdGhzW2lkeF07CgkJCWludCBpOwoKCQkJUkVDVCBjbG50OwoJCQlHZXRDbGllbnRSZWN0KHBhbmUtPmh3bmQsICZjbG50KTsKCgkJCXBhbmUtPndpZHRoc1tpZHhdICs9IGR4OwoKCQkJZm9yKGk9aWR4OyArK2k8PUNPTFVNTlM7ICkKCQkJCXBhbmUtPnBvc2l0aW9uc1tpXSArPSBkeDsKCgkJCXsKCQkJCWludCBzY3JvbGxfcG9zID0gR2V0U2Nyb2xsUG9zKHBhbmUtPmh3bmQsIFNCX0hPUlopOwoJCQkJUkVDVCBydF9zY3I7CgkJCQlSRUNUIHJ0X2NsaXA7CgoJCQkJcnRfc2NyLmxlZnQgICA9IHBhbmUtPnBvc2l0aW9uc1tpZHgrMV0tc2Nyb2xsX3BvczsKCQkJCXJ0X3Njci50b3AgICAgPSAwOwoJCQkJcnRfc2NyLnJpZ2h0ICA9IGNsbnQucmlnaHQ7CgkJCQlydF9zY3IuYm90dG9tID0gY2xudC5ib3R0b207CgoJCQkJcnRfY2xpcC5sZWZ0ICAgPSBwYW5lLT5wb3NpdGlvbnNbaWR4XS1zY3JvbGxfcG9zOwoJCQkJcnRfY2xpcC50b3AgICAgPSAwOwoJCQkJcnRfY2xpcC5yaWdodCAgPSBjbG50LnJpZ2h0OwoJCQkJcnRfY2xpcC5ib3R0b20gPSBjbG50LmJvdHRvbTsKCgkJCQlpZiAocnRfc2NyLmxlZnQgPCAwKSBydF9zY3IubGVmdCA9IDA7CgkJCQlpZiAocnRfY2xpcC5sZWZ0IDwgMCkgcnRfY2xpcC5sZWZ0ID0gMDsKCgkJCQlTY3JvbGxXaW5kb3dFeChwYW5lLT5od25kLCBkeCwgMCwgJnJ0X3NjciwgJnJ0X2NsaXAsIDAsIDAsIFNXX0lOVkFMSURBVEUpOwoKCQkJCXJ0X2NsaXAucmlnaHQgPSBwYW5lLT5wb3NpdGlvbnNbaWR4KzFdOwoJCQkJUmVkcmF3V2luZG93KHBhbmUtPmh3bmQsICZydF9jbGlwLCAwLCBSRFdfSU5WQUxJREFURXxSRFdfVVBEQVRFTk9XKTsKCgkJCQlpZiAocG5taC0+Y29kZSA9PSBIRE5fRU5EVFJBQ0spIHsKCQkJCQlTZW5kTWVzc2FnZShwYW5lLT5od25kLCBMQl9TRVRIT1JJWk9OVEFMRVhURU5ULCBwYW5lLT5wb3NpdGlvbnNbQ09MVU1OU10sIDApOwoKCQkJCQlpZiAoR2V0U2Nyb2xsUG9zKHBhbmUtPmh3bmQsIFNCX0hPUlopICE9IHNjcm9sbF9wb3MpCgkJCQkJCXNldF9oZWFkZXIocGFuZSk7CgkJCQl9CgkJCX0KCgkJCXJldHVybiBGQUxTRTsKCQl9CgoJCWNhc2UgSEROX0RJVklERVJEQkxDTElDSzogewoJCQlIRF9OT1RJRlkqIHBoZG4gPSAoSERfTk9USUZZKikgcG5taDsKCQkJSERfSVRFTSBpdGVtOwoKCQkJY2FsY19zaW5nbGVfd2lkdGgocGFuZSwgcGhkbi0+aUl0ZW0pOwoJCQlpdGVtLm1hc2sgPSBIRElfV0lEVEg7CgkJCWl0ZW0uY3h5ID0gcGFuZS0+d2lkdGhzW3BoZG4tPmlJdGVtXTsKCgkJCVNlbmRNZXNzYWdlKHBhbmUtPmh3bmRIZWFkZXIsIEhETV9TRVRJVEVNLCBwaGRuLT5pSXRlbSwgKExQQVJBTSkgJml0ZW0pOwoJCQlJbnZhbGlkYXRlUmVjdChwYW5lLT5od25kLCAwLCBUUlVFKTsKCQkJYnJlYWs7fQoJfQoKCXJldHVybiAwOwp9CgojZW5kaWYgLyogX05PX0VYVEVOU0lPTlMgKi8KCgpzdGF0aWMgdm9pZCBzY2FuX2VudHJ5KENoaWxkV25kKiBjaGlsZCwgRW50cnkqIGVudHJ5LCBpbnQgaWR4LCBIV05EIGh3bmQpCnsKCVRDSEFSIHBhdGhbTUFYX1BBVEhdOwoJSENVUlNPUiBvbGRfY3Vyc29yID0gU2V0Q3Vyc29yKExvYWRDdXJzb3IoMCwgSURDX1dBSVQpKTsKCgkvKiBkZWxldGUgc3ViIGVudHJpZXMgaW4gbGVmdCBwYW5lICovCglmb3IoOzspIHsKCQlMUkVTVUxUIHJlcyA9IFNlbmRNZXNzYWdlKGNoaWxkLT5sZWZ0Lmh3bmQsIExCX0dFVElURU1EQVRBLCBpZHgrMSwgMCk7CgkJRW50cnkqIHN1YiA9IChFbnRyeSopIHJlczsKCgkJaWYgKHJlcz09TEJfRVJSIHx8ICFzdWIgfHwgc3ViLT5sZXZlbDw9ZW50cnktPmxldmVsKQoJCQlicmVhazsKCgkJU2VuZE1lc3NhZ2UoY2hpbGQtPmxlZnQuaHduZCwgTEJfREVMRVRFU1RSSU5HLCBpZHgrMSwgMCk7Cgl9CgoJLyogZW1wdHkgcmlnaHQgcGFuZSAqLwoJU2VuZE1lc3NhZ2UoY2hpbGQtPnJpZ2h0Lmh3bmQsIExCX1JFU0VUQ09OVEVOVCwgMCwgMCk7CgoJLyogcmVsZWFzZSBtZW1vcnkgKi8KCWZyZWVfZW50cmllcyhlbnRyeSk7CgoJLyogcmVhZCBjb250ZW50cyBmcm9tIGRpc2sgKi8KI2lmZGVmIF9TSEVMTF9GT0xERVJTCglpZiAoZW50cnktPmV0eXBlID09IEVUX1NIRUxMKQoJewoJCXJlYWRfZGlyZWN0b3J5KGVudHJ5LCBOVUxMLCBjaGlsZC0+c29ydE9yZGVyLCBod25kKTsKCX0KCWVsc2UKI2VuZGlmCgl7CgkJZ2V0X3BhdGgoZW50cnksIHBhdGgpOwoJCXJlYWRfZGlyZWN0b3J5KGVudHJ5LCBwYXRoLCBjaGlsZC0+c29ydE9yZGVyLCBod25kKTsKCX0KCgkvKiBpbnNlcnQgZm91bmQgZW50cmllcyBpbiByaWdodCBwYW5lICovCglpbnNlcnRfZW50cmllcygmY2hpbGQtPnJpZ2h0LCBlbnRyeS0+ZG93biwgY2hpbGQtPmZpbHRlcl9wYXR0ZXJuLCBjaGlsZC0+ZmlsdGVyX2ZsYWdzLCAtMSk7CgljYWxjX3dpZHRocygmY2hpbGQtPnJpZ2h0LCBGQUxTRSk7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCXNldF9oZWFkZXIoJmNoaWxkLT5yaWdodCk7CiNlbmRpZgoKCWNoaWxkLT5oZWFkZXJfd2R0aHNfb2sgPSBGQUxTRTsKCglTZXRDdXJzb3Iob2xkX2N1cnNvcik7Cn0KCgovKiBleHBhbmQgYSBkaXJlY3RvcnkgZW50cnkgKi8KCnN0YXRpYyBCT09MIGV4cGFuZF9lbnRyeShDaGlsZFduZCogY2hpbGQsIEVudHJ5KiBkaXIpCnsKCWludCBpZHg7CglFbnRyeSogcDsKCglpZiAoIWRpciB8fCBkaXItPmV4cGFuZGVkIHx8ICFkaXItPmRvd24pCgkJcmV0dXJuIEZBTFNFOwoKCXAgPSBkaXItPmRvd247CgoJaWYgKHAtPmRhdGEuY0ZpbGVOYW1lWzBdPT0nLicgJiYgcC0+ZGF0YS5jRmlsZU5hbWVbMV09PSdcMCcgJiYgcC0+bmV4dCkgewoJCXAgPSBwLT5uZXh0OwoKCQlpZiAocC0+ZGF0YS5jRmlsZU5hbWVbMF09PScuJyAmJiBwLT5kYXRhLmNGaWxlTmFtZVsxXT09Jy4nICYmCgkJCQlwLT5kYXRhLmNGaWxlTmFtZVsyXT09J1wwJyAmJiBwLT5uZXh0KQoJCQlwID0gcC0+bmV4dDsKCX0KCgkvKiBubyBzdWJkaXJlY3RvcmllcyA/ICovCglpZiAoIShwLT5kYXRhLmR3RmlsZUF0dHJpYnV0ZXMmRklMRV9BVFRSSUJVVEVfRElSRUNUT1JZKSkKCQlyZXR1cm4gRkFMU0U7CgoJaWR4ID0gU2VuZE1lc3NhZ2UoY2hpbGQtPmxlZnQuaHduZCwgTEJfRklORFNUUklORywgMCwgKExQQVJBTSlkaXIpOwoKCWRpci0+ZXhwYW5kZWQgPSBUUlVFOwoKCS8qIGluc2VydCBlbnRyaWVzIGluIGxlZnQgcGFuZSAqLwoJaW5zZXJ0X2VudHJpZXMoJmNoaWxkLT5sZWZ0LCBwLCBOVUxMLCBURl9BTEwsIGlkeCk7CgoJaWYgKCFjaGlsZC0+aGVhZGVyX3dkdGhzX29rKSB7CgkJaWYgKGNhbGNfd2lkdGhzKCZjaGlsZC0+bGVmdCwgRkFMU0UpKSB7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJc2V0X2hlYWRlcigmY2hpbGQtPmxlZnQpOwojZW5kaWYKCgkJCWNoaWxkLT5oZWFkZXJfd2R0aHNfb2sgPSBUUlVFOwoJCX0KCX0KCglyZXR1cm4gVFJVRTsKfQoKCnN0YXRpYyB2b2lkIGNvbGxhcHNlX2VudHJ5KFBhbmUqIHBhbmUsIEVudHJ5KiBkaXIpCnsKCWludCBpZHggPSBTZW5kTWVzc2FnZShwYW5lLT5od25kLCBMQl9GSU5EU1RSSU5HLCAwLCAoTFBBUkFNKWRpcik7CgoJU2hvd1dpbmRvdyhwYW5lLT5od25kLCBTV19ISURFKTsKCgkvKiBoaWRlIHN1YiBlbnRyaWVzICovCglmb3IoOzspIHsKCQlMUkVTVUxUIHJlcyA9IFNlbmRNZXNzYWdlKHBhbmUtPmh3bmQsIExCX0dFVElURU1EQVRBLCBpZHgrMSwgMCk7CgkJRW50cnkqIHN1YiA9IChFbnRyeSopIHJlczsKCgkJaWYgKHJlcz09TEJfRVJSIHx8ICFzdWIgfHwgc3ViLT5sZXZlbDw9ZGlyLT5sZXZlbCkKCQkJYnJlYWs7CgoJCVNlbmRNZXNzYWdlKHBhbmUtPmh3bmQsIExCX0RFTEVURVNUUklORywgaWR4KzEsIDApOwoJfQoKCWRpci0+ZXhwYW5kZWQgPSBGQUxTRTsKCglTaG93V2luZG93KHBhbmUtPmh3bmQsIFNXX1NIT1cpOwp9CgoKc3RhdGljIHZvaWQgcmVmcmVzaF9yaWdodF9wYW5lKENoaWxkV25kKiBjaGlsZCkKewoJU2VuZE1lc3NhZ2UoY2hpbGQtPnJpZ2h0Lmh3bmQsIExCX1JFU0VUQ09OVEVOVCwgMCwgMCk7CglpbnNlcnRfZW50cmllcygmY2hpbGQtPnJpZ2h0LCBjaGlsZC0+cmlnaHQucm9vdCwgY2hpbGQtPmZpbHRlcl9wYXR0ZXJuLCBjaGlsZC0+ZmlsdGVyX2ZsYWdzLCAtMSk7CgljYWxjX3dpZHRocygmY2hpbGQtPnJpZ2h0LCBGQUxTRSk7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCglzZXRfaGVhZGVyKCZjaGlsZC0+cmlnaHQpOwojZW5kaWYKfQoKc3RhdGljIHZvaWQgc2V0X2N1cmRpcihDaGlsZFduZCogY2hpbGQsIEVudHJ5KiBlbnRyeSwgaW50IGlkeCwgSFdORCBod25kKQp7CglUQ0hBUiBwYXRoW01BWF9QQVRIXTsKCglpZiAoIWVudHJ5KQoJCXJldHVybjsKCglwYXRoWzBdID0gJ1wwJzsKCgljaGlsZC0+bGVmdC5jdXIgPSBlbnRyeTsKCgljaGlsZC0+cmlnaHQucm9vdCA9IGVudHJ5LT5kb3duPyBlbnRyeS0+ZG93bjogZW50cnk7CgljaGlsZC0+cmlnaHQuY3VyID0gZW50cnk7CgoJaWYgKCFlbnRyeS0+c2Nhbm5lZCkKCQlzY2FuX2VudHJ5KGNoaWxkLCBlbnRyeSwgaWR4LCBod25kKTsKCWVsc2UKCQlyZWZyZXNoX3JpZ2h0X3BhbmUoY2hpbGQpOwoKCWdldF9wYXRoKGVudHJ5LCBwYXRoKTsKCWxzdHJjcHkoY2hpbGQtPnBhdGgsIHBhdGgpOwoKCWlmIChjaGlsZC0+aHduZCkJLyogb25seSBjaGFuZ2Ugd2luZG93IHRpdGxlLCBpZiB0aGUgd2luZG93IGFscmVhZHkgZXhpc3RzICovCgkJU2V0V2luZG93VGV4dChjaGlsZC0+aHduZCwgcGF0aCk7CgoJaWYgKHBhdGhbMF0pCgkJaWYgKFNldEN1cnJlbnREaXJlY3RvcnkocGF0aCkpCgkJCXNldF9zcGFjZV9zdGF0dXMoKTsKfQoKCnN0YXRpYyB2b2lkIHJlZnJlc2hfY2hpbGQoQ2hpbGRXbmQqIGNoaWxkKQp7CglUQ0hBUiBwYXRoW01BWF9QQVRIXSwgZHJ2W19NQVhfRFJJVkUrMV07CglFbnRyeSogZW50cnk7CglpbnQgaWR4OwoKCWdldF9wYXRoKGNoaWxkLT5sZWZ0LmN1ciwgcGF0aCk7CglfdHNwbGl0cGF0aChwYXRoLCBkcnYsIE5VTEwsIE5VTEwsIE5VTEwpOwoKCWNoaWxkLT5yaWdodC5yb290ID0gTlVMTDsKCglzY2FuX2VudHJ5KGNoaWxkLCAmY2hpbGQtPnJvb3QuZW50cnksIDAsIGNoaWxkLT5od25kKTsKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJaWYgKGNoaWxkLT5yb290LmVudHJ5LmV0eXBlID09IEVUX1NIRUxMKQoJCWVudHJ5ID0gcmVhZF90cmVlKCZjaGlsZC0+cm9vdCwgTlVMTCwgZ2V0X3BhdGhfcGlkbChwYXRoLGNoaWxkLT5od25kKSwgZHJ2LCBjaGlsZC0+c29ydE9yZGVyLCBjaGlsZC0+aHduZCk7CgllbHNlCiNlbmRpZgoJCWVudHJ5ID0gcmVhZF90cmVlKCZjaGlsZC0+cm9vdCwgcGF0aCwgTlVMTCwgZHJ2LCBjaGlsZC0+c29ydE9yZGVyLCBjaGlsZC0+aHduZCk7CgoJaWYgKCFlbnRyeSkKCQllbnRyeSA9ICZjaGlsZC0+cm9vdC5lbnRyeTsKCglpbnNlcnRfZW50cmllcygmY2hpbGQtPmxlZnQsIGNoaWxkLT5yb290LmVudHJ5LmRvd24sIE5VTEwsIFRGX0FMTCwgMCk7CgoJc2V0X2N1cmRpcihjaGlsZCwgZW50cnksIDAsIGNoaWxkLT5od25kKTsKCglpZHggPSBTZW5kTWVzc2FnZShjaGlsZC0+bGVmdC5od25kLCBMQl9GSU5EU1RSSU5HLCAwLCAoTFBBUkFNKWNoaWxkLT5sZWZ0LmN1cik7CglTZW5kTWVzc2FnZShjaGlsZC0+bGVmdC5od25kLCBMQl9TRVRDVVJTRUwsIGlkeCwgMCk7Cn0KCgpzdGF0aWMgdm9pZCBjcmVhdGVfZHJpdmVfYmFyKHZvaWQpCnsKCVRCQlVUVE9OIGRyaXZlYmFyQnRuID0gezAsIDAsIFRCU1RBVEVfRU5BQkxFRCwgQlROU19CVVRUT04sIHswLCAwfSwgMCwgMH07CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCVRDSEFSIGIxW0JVRkZFUl9MRU5dOwojZW5kaWYKCWludCBidG4gPSAxOwoJUFRTVFIgcDsKCglHZXRMb2dpY2FsRHJpdmVTdHJpbmdzKEJVRkZFUl9MRU4sIEdsb2JhbHMuZHJpdmVzKTsKCglHbG9iYWxzLmhkcml2ZWJhciA9IENyZWF0ZVRvb2xiYXJFeChHbG9iYWxzLmhNYWluV25kLCBXU19DSElMRHxXU19WSVNJQkxFfENDU19OT01PVkVZfFRCU1RZTEVfTElTVCwKCQkJCUlEV19EUklWRUJBUiwgMiwgR2xvYmFscy5oSW5zdGFuY2UsIElEQl9EUklWRUJBUiwgJmRyaXZlYmFyQnRuLAoJCQkJMCwgMTYsIDEzLCAxNiwgMTMsIHNpemVvZihUQkJVVFRPTikpOwoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwojaWZkZWYgX19XSU5FX18KCS8qIGluc2VydCB1bml4IGZpbGUgc3lzdGVtIGJ1dHRvbiAqLwoJYjFbMF0gPSAnLyc7CgliMVsxXSA9ICdcMCc7CgliMVsyXSA9ICdcMCc7CglTZW5kTWVzc2FnZShHbG9iYWxzLmhkcml2ZWJhciwgVEJfQUREU1RSSU5HLCAwLCAoTFBBUkFNKWIxKTsKCglkcml2ZWJhckJ0bi5pZENvbW1hbmQgPSBJRF9EUklWRV9VTklYX0ZTOwoJU2VuZE1lc3NhZ2UoR2xvYmFscy5oZHJpdmViYXIsIFRCX0lOU0VSVEJVVFRPTiwgYnRuKyssIChMUEFSQU0pJmRyaXZlYmFyQnRuKTsKCWRyaXZlYmFyQnRuLmlTdHJpbmcrKzsKI2VuZGlmCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJLyogaW5zZXJ0IHNoZWxsIG5hbWVzcGFjZSBidXR0b24gKi8KCWxvYWRfc3RyaW5nKGIxLCBJRFNfU0hFTEwpOwoJYjFbbHN0cmxlbihiMSkrMV0gPSAnXDAnOwoJU2VuZE1lc3NhZ2UoR2xvYmFscy5oZHJpdmViYXIsIFRCX0FERFNUUklORywgMCwgKExQQVJBTSliMSk7CgoJZHJpdmViYXJCdG4uaWRDb21tYW5kID0gSURfRFJJVkVfU0hFTExfTlM7CglTZW5kTWVzc2FnZShHbG9iYWxzLmhkcml2ZWJhciwgVEJfSU5TRVJUQlVUVE9OLCBidG4rKywgKExQQVJBTSkmZHJpdmViYXJCdG4pOwoJZHJpdmViYXJCdG4uaVN0cmluZysrOwojZW5kaWYKCgkvKiByZWdpc3RlciB3aW5kb3dzIGRyaXZlIHJvb3Qgc3RyaW5ncyAqLwoJU2VuZE1lc3NhZ2UoR2xvYmFscy5oZHJpdmViYXIsIFRCX0FERFNUUklORywgMCwgKExQQVJBTSlHbG9iYWxzLmRyaXZlcyk7CiNlbmRpZgoKCWRyaXZlYmFyQnRuLmlkQ29tbWFuZCA9IElEX0RSSVZFX0ZJUlNUOwoKCWZvcihwPUdsb2JhbHMuZHJpdmVzOyAqcDsgKSB7CiNpZmRlZiBfTk9fRVhURU5TSU9OUwoJCS8qIGluc2VydCBkcml2ZSBsZXR0ZXIgKi8KCQlUQ0hBUiBiWzNdID0ge3RvbG93ZXIoKnApfTsKCQlTZW5kTWVzc2FnZShHbG9iYWxzLmhkcml2ZWJhciwgVEJfQUREU1RSSU5HLCAwLCAoTFBBUkFNKWIpOwojZW5kaWYKCQlzd2l0Y2goR2V0RHJpdmVUeXBlKHApKSB7CgkJCWNhc2UgRFJJVkVfUkVNT1ZBQkxFOglkcml2ZWJhckJ0bi5pQml0bWFwID0gMTsJYnJlYWs7CgkJCWNhc2UgRFJJVkVfQ0RST006CQlkcml2ZWJhckJ0bi5pQml0bWFwID0gMzsJYnJlYWs7CgkJCWNhc2UgRFJJVkVfUkVNT1RFOgkJZHJpdmViYXJCdG4uaUJpdG1hcCA9IDQ7CWJyZWFrOwoJCQljYXNlIERSSVZFX1JBTURJU0s6CQlkcml2ZWJhckJ0bi5pQml0bWFwID0gNTsJYnJlYWs7CgkJCWRlZmF1bHQ6LypEUklWRV9GSVhFRCovCWRyaXZlYmFyQnRuLmlCaXRtYXAgPSAyOwoJCX0KCgkJU2VuZE1lc3NhZ2UoR2xvYmFscy5oZHJpdmViYXIsIFRCX0lOU0VSVEJVVFRPTiwgYnRuKyssIChMUEFSQU0pJmRyaXZlYmFyQnRuKTsKCQlkcml2ZWJhckJ0bi5pZENvbW1hbmQrKzsKCQlkcml2ZWJhckJ0bi5pU3RyaW5nKys7CgoJCXdoaWxlKCpwKyspOwoJfQp9CgpzdGF0aWMgdm9pZCByZWZyZXNoX2RyaXZlcyh2b2lkKQp7CglSRUNUIHJlY3Q7CgoJLyogZGVzdHJveSBkcml2ZSBiYXIgKi8KCURlc3Ryb3lXaW5kb3coR2xvYmFscy5oZHJpdmViYXIpOwoJR2xvYmFscy5oZHJpdmViYXIgPSAwOwoKCS8qIHJlLWNyZWF0ZSBkcml2ZSBiYXIgKi8KCWNyZWF0ZV9kcml2ZV9iYXIoKTsKCgkvKiB1cGRhdGUgd2luZG93IGxheW91dCAqLwoJR2V0Q2xpZW50UmVjdChHbG9iYWxzLmhNYWluV25kLCAmcmVjdCk7CglTZW5kTWVzc2FnZShHbG9iYWxzLmhNYWluV25kLCBXTV9TSVpFLCAwLCBNQUtFTE9ORyhyZWN0LnJpZ2h0LCByZWN0LmJvdHRvbSkpOwp9CgoKc3RhdGljIEJPT0wgbGF1bmNoX2ZpbGUoSFdORCBod25kLCBMUENUU1RSIGNtZCwgVUlOVCBuQ21kU2hvdykKewoJSElOU1RBTkNFIGhpbnN0ID0gU2hlbGxFeGVjdXRlKGh3bmQsIE5VTEwvKm9wZXJhdGlvbiovLCBjbWQsIE5VTEwvKnBhcmFtZXRlcnMqLywgTlVMTC8qZGlyKi8sIG5DbWRTaG93KTsKCglpZiAoUHRyVG9VbG9uZyhoaW5zdCkgPD0gMzIpIHsKCQlkaXNwbGF5X2Vycm9yKGh3bmQsIEdldExhc3RFcnJvcigpKTsKCQlyZXR1cm4gRkFMU0U7Cgl9CgoJcmV0dXJuIFRSVUU7Cn0KCgpzdGF0aWMgQk9PTCBsYXVuY2hfZW50cnkoRW50cnkqIGVudHJ5LCBIV05EIGh3bmQsIFVJTlQgbkNtZFNob3cpCnsKCVRDSEFSIGNtZFtNQVhfUEFUSF07CgojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCWlmIChlbnRyeS0+ZXR5cGUgPT0gRVRfU0hFTEwpIHsKCQlCT09MIHJldCA9IFRSVUU7CgoJCVNIRUxMRVhFQ1VURUlORk8gc2hleGluZm87CgoJCXNoZXhpbmZvLmNiU2l6ZSA9IHNpemVvZihTSEVMTEVYRUNVVEVJTkZPKTsKCQlzaGV4aW5mby5mTWFzayA9IFNFRV9NQVNLX0lETElTVDsKCQlzaGV4aW5mby5od25kID0gaHduZDsKCQlzaGV4aW5mby5scFZlcmIgPSBOVUxMOwoJCXNoZXhpbmZvLmxwRmlsZSA9IE5VTEw7CgkJc2hleGluZm8ubHBQYXJhbWV0ZXJzID0gTlVMTDsKCQlzaGV4aW5mby5scERpcmVjdG9yeSA9IE5VTEw7CgkJc2hleGluZm8ublNob3cgPSBuQ21kU2hvdzsKCQlzaGV4aW5mby5scElETGlzdCA9IGdldF90b19hYnNvbHV0ZV9waWRsKGVudHJ5LCBod25kKTsKCgkJaWYgKCFTaGVsbEV4ZWN1dGVFeCgmc2hleGluZm8pKSB7CgkJCWRpc3BsYXlfZXJyb3IoaHduZCwgR2V0TGFzdEVycm9yKCkpOwoJCQlyZXQgPSBGQUxTRTsKCQl9CgoJCWlmIChzaGV4aW5mby5scElETGlzdCAhPSBlbnRyeS0+cGlkbCkKCQkJSU1hbGxvY19GcmVlKEdsb2JhbHMuaU1hbGxvYywgc2hleGluZm8ubHBJRExpc3QpOwoKCQlyZXR1cm4gcmV0OwoJfQojZW5kaWYKCglnZXRfcGF0aChlbnRyeSwgY21kKTsKCgkgLyogc3RhcnQgcHJvZ3JhbSwgb3BlbiBkb2N1bWVudC4uLiAqLwoJcmV0dXJuIGxhdW5jaF9maWxlKGh3bmQsIGNtZCwgbkNtZFNob3cpOwp9CgoKc3RhdGljIHZvaWQgYWN0aXZhdGVfZW50cnkoQ2hpbGRXbmQqIGNoaWxkLCBQYW5lKiBwYW5lLCBIV05EIGh3bmQpCnsKCUVudHJ5KiBlbnRyeSA9IHBhbmUtPmN1cjsKCglpZiAoIWVudHJ5KQoJCXJldHVybjsKCglpZiAoZW50cnktPmRhdGEuZHdGaWxlQXR0cmlidXRlcyAmIEZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWSkgewoJCWludCBzY2FubmVkX29sZCA9IGVudHJ5LT5zY2FubmVkOwoKCQlpZiAoIXNjYW5uZWRfb2xkKQoJCXsKCQkJaW50IGlkeCA9IFNlbmRNZXNzYWdlKGNoaWxkLT5sZWZ0Lmh3bmQsIExCX0dFVENVUlNFTCwgMCwgMCk7CgkJCXNjYW5fZW50cnkoY2hpbGQsIGVudHJ5LCBpZHgsIGh3bmQpOwoJCX0KCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQlpZiAoZW50cnktPmRhdGEuY0ZpbGVOYW1lWzBdPT0nLicgJiYgZW50cnktPmRhdGEuY0ZpbGVOYW1lWzFdPT0nXDAnKQoJCQlyZXR1cm47CiNlbmRpZgoKCQlpZiAoZW50cnktPmRhdGEuY0ZpbGVOYW1lWzBdPT0nLicgJiYgZW50cnktPmRhdGEuY0ZpbGVOYW1lWzFdPT0nLicgJiYgZW50cnktPmRhdGEuY0ZpbGVOYW1lWzJdPT0nXDAnKSB7CgkJCWVudHJ5ID0gY2hpbGQtPmxlZnQuY3VyLT51cDsKCQkJY29sbGFwc2VfZW50cnkoJmNoaWxkLT5sZWZ0LCBlbnRyeSk7CgkJCWdvdG8gZm9jdXNfZW50cnk7CgkJfSBlbHNlIGlmIChlbnRyeS0+ZXhwYW5kZWQpCgkJCWNvbGxhcHNlX2VudHJ5KHBhbmUsIGNoaWxkLT5sZWZ0LmN1cik7CgkJZWxzZSB7CgkJCWV4cGFuZF9lbnRyeShjaGlsZCwgY2hpbGQtPmxlZnQuY3VyKTsKCgkJCWlmICghcGFuZS0+dHJlZVBhbmUpIGZvY3VzX2VudHJ5OiB7CgkJCQlpbnQgaWR4c3RhcnQgPSBTZW5kTWVzc2FnZShjaGlsZC0+bGVmdC5od25kLCBMQl9HRVRDVVJTRUwsIDAsIDApOwoJCQkJaW50IGlkeCA9IFNlbmRNZXNzYWdlKGNoaWxkLT5sZWZ0Lmh3bmQsIExCX0ZJTkRTVFJJTkcsIGlkeHN0YXJ0LCAoTFBBUkFNKWVudHJ5KTsKCQkJCVNlbmRNZXNzYWdlKGNoaWxkLT5sZWZ0Lmh3bmQsIExCX1NFVENVUlNFTCwgaWR4LCAwKTsKCQkJCXNldF9jdXJkaXIoY2hpbGQsIGVudHJ5LCBpZHgsIGh3bmQpOwoJCQl9CgkJfQoKCQlpZiAoIXNjYW5uZWRfb2xkKSB7CgkJCWNhbGNfd2lkdGhzKHBhbmUsIEZBTFNFKTsKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJc2V0X2hlYWRlcihwYW5lKTsKI2VuZGlmCgkJfQoJfSBlbHNlIHsKCQlpZiAoR2V0S2V5U3RhdGUoVktfTUVOVSkgPCAwKQoJCQlzaG93X3Byb3BlcnRpZXNfZGxnKGVudHJ5LCBjaGlsZC0+aHduZCk7CgkJZWxzZQoJCQlsYXVuY2hfZW50cnkoZW50cnksIGNoaWxkLT5od25kLCBTV19TSE9XTk9STUFMKTsKCX0KfQoKCnN0YXRpYyBCT09MIHBhbmVfY29tbWFuZChQYW5lKiBwYW5lLCBVSU5UIGNtZCkKewoJc3dpdGNoKGNtZCkgewoJCWNhc2UgSURfVklFV19OQU1FOgoJCQlpZiAocGFuZS0+dmlzaWJsZV9jb2xzKSB7CgkJCQlwYW5lLT52aXNpYmxlX2NvbHMgPSAwOwoJCQkJY2FsY193aWR0aHMocGFuZSwgVFJVRSk7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJCXNldF9oZWFkZXIocGFuZSk7CiNlbmRpZgoJCQkJSW52YWxpZGF0ZVJlY3QocGFuZS0+aHduZCwgMCwgVFJVRSk7CgkJCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX05BTUUsIE1GX0JZQ09NTUFORHxNRl9DSEVDS0VEKTsKCQkJCUNoZWNrTWVudUl0ZW0oR2xvYmFscy5oTWVudVZpZXcsIElEX1ZJRVdfQUxMX0FUVFJJQlVURVMsIE1GX0JZQ09NTUFORCk7CgkJCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX1NFTEVDVEVEX0FUVFJJQlVURVMsIE1GX0JZQ09NTUFORCk7CgkJCX0KCQkJYnJlYWs7CgoJCWNhc2UgSURfVklFV19BTExfQVRUUklCVVRFUzoKCQkJaWYgKHBhbmUtPnZpc2libGVfY29scyAhPSBDT0xfQUxMKSB7CgkJCQlwYW5lLT52aXNpYmxlX2NvbHMgPSBDT0xfQUxMOwoJCQkJY2FsY193aWR0aHMocGFuZSwgVFJVRSk7CiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQkJCXNldF9oZWFkZXIocGFuZSk7CiNlbmRpZgoJCQkJSW52YWxpZGF0ZVJlY3QocGFuZS0+aHduZCwgMCwgVFJVRSk7CgkJCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX05BTUUsIE1GX0JZQ09NTUFORCk7CgkJCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX0FMTF9BVFRSSUJVVEVTLCBNRl9CWUNPTU1BTkR8TUZfQ0hFQ0tFRCk7CgkJCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX1NFTEVDVEVEX0FUVFJJQlVURVMsIE1GX0JZQ09NTUFORCk7CgkJCX0KCQkJYnJlYWs7CgojaWZuZGVmIF9OT19FWFRFTlNJT05TCgkJY2FzZSBJRF9QUkVGRVJSRURfU0laRVM6IHsKCQkJY2FsY193aWR0aHMocGFuZSwgVFJVRSk7CgkJCXNldF9oZWFkZXIocGFuZSk7CgkJCUludmFsaWRhdGVSZWN0KHBhbmUtPmh3bmQsIDAsIFRSVUUpOwoJCQlicmVhazt9CiNlbmRpZgoKCQkgICAgICAgIC8qIFRPRE86IG1vcmUgY29tbWFuZCBpZHMuLi4gKi8KCgkJZGVmYXVsdDoKCQkJcmV0dXJuIEZBTFNFOwoJfQoKCXJldHVybiBUUlVFOwp9CgoKc3RhdGljIHZvaWQgc2V0X3NvcnRfb3JkZXIoQ2hpbGRXbmQqIGNoaWxkLCBTT1JUX09SREVSIHNvcnRPcmRlcikKewoJaWYgKGNoaWxkLT5zb3J0T3JkZXIgIT0gc29ydE9yZGVyKSB7CgkJY2hpbGQtPnNvcnRPcmRlciA9IHNvcnRPcmRlcjsKCQlyZWZyZXNoX2NoaWxkKGNoaWxkKTsKCX0KfQoKc3RhdGljIHZvaWQgdXBkYXRlX3ZpZXdfbWVudShDaGlsZFduZCogY2hpbGQpCnsKCUNoZWNrTWVudUl0ZW0oR2xvYmFscy5oTWVudVZpZXcsIElEX1ZJRVdfU09SVF9OQU1FLCBjaGlsZC0+c29ydE9yZGVyPT1TT1JUX05BTUU/IE1GX0NIRUNLRUQ6IE1GX1VOQ0hFQ0tFRCk7CglDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX1NPUlRfVFlQRSwgY2hpbGQtPnNvcnRPcmRlcj09U09SVF9FWFQ/IE1GX0NIRUNLRUQ6IE1GX1VOQ0hFQ0tFRCk7CglDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVWaWV3LCBJRF9WSUVXX1NPUlRfU0laRSwgY2hpbGQtPnNvcnRPcmRlcj09U09SVF9TSVpFPyBNRl9DSEVDS0VEOiBNRl9VTkNIRUNLRUQpOwoJQ2hlY2tNZW51SXRlbShHbG9iYWxzLmhNZW51VmlldywgSURfVklFV19TT1JUX0RBVEUsIGNoaWxkLT5zb3J0T3JkZXI9PVNPUlRfREFURT8gTUZfQ0hFQ0tFRDogTUZfVU5DSEVDS0VEKTsKfQoKCnN0YXRpYyBCT09MIGlzX2RpcmVjdG9yeShMUENUU1RSIHRhcmdldCkKewoJLypUT0RPIGNvcnJlY3RseSBoYW5kbGUgVU5JWCBwYXRocyAqLwoJRFdPUkQgdGFyZ2V0X2F0dHIgPSBHZXRGaWxlQXR0cmlidXRlcyh0YXJnZXQpOwoKCWlmICh0YXJnZXRfYXR0ciA9PSBJTlZBTElEX0ZJTEVfQVRUUklCVVRFUykKCQlyZXR1cm4gRkFMU0U7CgoJcmV0dXJuIHRhcmdldF9hdHRyJkZJTEVfQVRUUklCVVRFX0RJUkVDVE9SWT8gVFJVRTogRkFMU0U7Cn0KCQpzdGF0aWMgQk9PTCBwcm9tcHRfdGFyZ2V0KFBhbmUqIHBhbmUsIExQVFNUUiBzb3VyY2UsIExQVFNUUiB0YXJnZXQpCnsKCVRDSEFSIHBhdGhbTUFYX1BBVEhdOwoJaW50IGxlbjsKCglnZXRfcGF0aChwYW5lLT5jdXIsIHBhdGgpOwoKCWlmIChEaWFsb2dCb3hQYXJhbShHbG9iYWxzLmhJbnN0YW5jZSwgTUFLRUlOVFJFU09VUkNFKElERF9TRUxFQ1RfREVTVElOQVRJT04pLCBwYW5lLT5od25kLCBEZXN0aW5hdGlvbkRsZ1Byb2MsIChMUEFSQU0pcGF0aCkgIT0gSURPSykKCQlyZXR1cm4gRkFMU0U7CgoJZ2V0X3BhdGgocGFuZS0+Y3VyLCBzb3VyY2UpOwoKCS8qIGNvbnZlcnQgcmVsYXRpdmUgdGFyZ2V0cyB0byBhYnNvbHV0ZSBwYXRocyAqLwoJaWYgKHBhdGhbMF0hPScvJyAmJiBwYXRoWzFdIT0nOicpIHsKCQlnZXRfcGF0aChwYW5lLT5jdXItPnVwLCB0YXJnZXQpOwoJCWxlbiA9IGxzdHJsZW4odGFyZ2V0KTsKCgkJaWYgKHRhcmdldFtsZW4tMV0hPSdcXCcgJiYgdGFyZ2V0W2xlbi0xXSE9Jy8nKQoJCQl0YXJnZXRbbGVuKytdID0gJy8nOwoKCQlsc3RyY3B5KHRhcmdldCtsZW4sIHBhdGgpOwoJfSBlbHNlCgkJbHN0cmNweSh0YXJnZXQsIHBhdGgpOwoKCS8qIElmIHRoZSB0YXJnZXQgYWxyZWFkeSBleGlzdHMgYXMgZGlyZWN0b3J5LCBjcmVhdGUgYSBuZXcgdGFyZ2V0IGJlbG93IHRoaXMuICovCglpZiAoaXNfZGlyZWN0b3J5KHBhdGgpKSB7CgkJVENIQVIgZm5hbWVbX01BWF9GTkFNRV0sIGV4dFtfTUFYX0VYVF07CgkJc3RhdGljIGNvbnN0IFRDSEFSIHNBcHBlbmRbXSA9IHsnJScsJ3MnLCcvJywnJScsJ3MnLCclJywncycsJ1wwJ307CgoJCV90c3BsaXRwYXRoKHNvdXJjZSwgTlVMTCwgTlVMTCwgZm5hbWUsIGV4dCk7CgoJCXdzcHJpbnRmKHRhcmdldCwgc0FwcGVuZCwgcGF0aCwgZm5hbWUsIGV4dCk7Cgl9CgoJcmV0dXJuIFRSVUU7Cn0KCgpzdGF0aWMgSUNvbnRleHRNZW51Miogc19wY3R4bWVudTIgPSBOVUxMOwpzdGF0aWMgSUNvbnRleHRNZW51Myogc19wY3R4bWVudTMgPSBOVUxMOwoKc3RhdGljIHZvaWQgQ3R4TWVudV9yZXNldCh2b2lkKQp7CglzX3BjdHhtZW51MiA9IE5VTEw7CglzX3BjdHhtZW51MyA9IE5VTEw7Cn0KCnN0YXRpYyBJQ29udGV4dE1lbnUqIEN0eE1lbnVfcXVlcnlfaW50ZXJmYWNlcyhJQ29udGV4dE1lbnUqIHBjbTEpCnsKCUlDb250ZXh0TWVudSogcGNtID0gTlVMTDsKCglDdHhNZW51X3Jlc2V0KCk7CgoJaWYgKElDb250ZXh0TWVudV9RdWVyeUludGVyZmFjZShwY20xLCAmSUlEX0lDb250ZXh0TWVudTMsICh2b2lkKiopJnBjbSkgPT0gTk9FUlJPUikKCQlzX3BjdHhtZW51MyA9IChMUENPTlRFWFRNRU5VMylwY207CgllbHNlIGlmIChJQ29udGV4dE1lbnVfUXVlcnlJbnRlcmZhY2UocGNtMSwgJklJRF9JQ29udGV4dE1lbnUyLCAodm9pZCoqKSZwY20pID09IE5PRVJST1IpCgkJc19wY3R4bWVudTIgPSAoTFBDT05URVhUTUVOVTIpcGNtOwoKCWlmIChwY20pIHsKCQlJQ29udGV4dE1lbnVfUmVsZWFzZShwY20xKTsKCQlyZXR1cm4gcGNtOwoJfSBlbHNlCgkJcmV0dXJuIHBjbTE7Cn0KCnN0YXRpYyBCT09MIEN0eE1lbnVfSGFuZGxlTWVudU1zZyhVSU5UIG5tc2csIFdQQVJBTSB3cGFyYW0sIExQQVJBTSBscGFyYW0pCnsKCWlmIChzX3BjdHhtZW51MykgewoJCWlmIChTVUNDRUVERUQoSUNvbnRleHRNZW51M19IYW5kbGVNZW51TXNnKHNfcGN0eG1lbnUzLCBubXNnLCB3cGFyYW0sIGxwYXJhbSkpKQoJCQlyZXR1cm4gVFJVRTsKCX0KCglpZiAoc19wY3R4bWVudTIpCgkJaWYgKFNVQ0NFRURFRChJQ29udGV4dE1lbnUyX0hhbmRsZU1lbnVNc2coc19wY3R4bWVudTIsIG5tc2csIHdwYXJhbSwgbHBhcmFtKSkpCgkJCXJldHVybiBUUlVFOwoKCXJldHVybiBGQUxTRTsKfQoKCnN0YXRpYyBIUkVTVUxUIFNoZWxsRm9sZGVyQ29udGV4dE1lbnUoSVNoZWxsRm9sZGVyKiBzaGVsbF9mb2xkZXIsIEhXTkQgaHduZFBhcmVudCwgaW50IGNpZGwsIExQQ0lURU1JRExJU1QqIGFwaWRsLCBpbnQgeCwgaW50IHkpCnsKCUlDb250ZXh0TWVudSogcGNtOwoJQk9PTCBleGVjdXRlZCA9IEZBTFNFOwoKCUhSRVNVTFQgaHIgPSBJU2hlbGxGb2xkZXJfR2V0VUlPYmplY3RPZihzaGVsbF9mb2xkZXIsIGh3bmRQYXJlbnQsIGNpZGwsIGFwaWRsLCAmSUlEX0lDb250ZXh0TWVudSwgTlVMTCwgKExQVk9JRCopJnBjbSk7Ci8qCUhSRVNVTFQgaHIgPSBDRGVmRm9sZGVyTWVudV9DcmVhdGUyKGRpcj9kaXItPl9waWRsOkRlc2t0b3BGb2xkZXIoKSwgaHduZFBhcmVudCwgMSwgJnBpZGwsIHNoZWxsX2ZvbGRlciwgTlVMTCwgMCwgTlVMTCwgJnBjbSk7ICovCgoJaWYgKFNVQ0NFRURFRChocikpIHsKCQlITUVOVSBobWVudSA9IENyZWF0ZVBvcHVwTWVudSgpOwoKCQlwY20gPSBDdHhNZW51X3F1ZXJ5X2ludGVyZmFjZXMocGNtKTsKCgkJaWYgKGhtZW51KSB7CgkJCWhyID0gSUNvbnRleHRNZW51X1F1ZXJ5Q29udGV4dE1lbnUocGNtLCBobWVudSwgMCwgRkNJRE1fU0hWSUVXRklSU1QsIEZDSURNX1NIVklFV0xBU1QsIENNRl9OT1JNQUwpOwoKCQkJaWYgKFNVQ0NFRURFRChocikpIHsKCQkJCVVJTlQgaWRDbWQgPSBUcmFja1BvcHVwTWVudShobWVudSwgVFBNX0xFRlRBTElHTnxUUE1fUkVUVVJOQ01EfFRQTV9SSUdIVEJVVFRPTiwgeCwgeSwgMCwgaHduZFBhcmVudCwgTlVMTCk7CgoJCQkJQ3R4TWVudV9yZXNldCgpOwoKCQkJCWlmIChpZENtZCkgewoJCQkJICBDTUlOVk9LRUNPTU1BTkRJTkZPIGNtaTsKCgkJCQkgIGNtaS5jYlNpemUgPSBzaXplb2YoQ01JTlZPS0VDT01NQU5ESU5GTyk7CgkJCQkgIGNtaS5mTWFzayA9IDA7CgkJCQkgIGNtaS5od25kID0gaHduZFBhcmVudDsKCQkJCSAgY21pLmxwVmVyYiA9IChMUENTVFIpKElOVF9QVFIpKGlkQ21kIC0gRkNJRE1fU0hWSUVXRklSU1QpOwoJCQkJICBjbWkubHBQYXJhbWV0ZXJzID0gTlVMTDsKCQkJCSAgY21pLmxwRGlyZWN0b3J5ID0gTlVMTDsKCQkJCSAgY21pLm5TaG93ID0gU1dfU0hPV05PUk1BTDsKCQkJCSAgY21pLmR3SG90S2V5ID0gMDsKCQkJCSAgY21pLmhJY29uID0gMDsKCgkJCQkgIGhyID0gSUNvbnRleHRNZW51X0ludm9rZUNvbW1hbmQocGNtLCAmY21pKTsKCQkJCQlleGVjdXRlZCA9IFRSVUU7CgkJCQl9CgkJCX0gZWxzZQoJCQkJQ3R4TWVudV9yZXNldCgpOwoJCX0KCgkJSUNvbnRleHRNZW51X1JlbGVhc2UocGNtKTsKCX0KCglyZXR1cm4gRkFJTEVEKGhyKT8gaHI6IGV4ZWN1dGVkPyBTX09LOiBTX0ZBTFNFOwp9CgoKc3RhdGljIExSRVNVTFQgQ0FMTEJBQ0sgQ2hpbGRXbmRQcm9jKEhXTkQgaHduZCwgVUlOVCBubXNnLCBXUEFSQU0gd3BhcmFtLCBMUEFSQU0gbHBhcmFtKQp7CglDaGlsZFduZCogY2hpbGQgPSAoQ2hpbGRXbmQqKSBHZXRXaW5kb3dMb25nUHRyKGh3bmQsIEdXTFBfVVNFUkRBVEEpOwoJQVNTRVJUKGNoaWxkKTsKCglzd2l0Y2gobm1zZykgewoJCWNhc2UgV01fRFJBV0lURU06IHsKCQkJTFBEUkFXSVRFTVNUUlVDVCBkaXMgPSAoTFBEUkFXSVRFTVNUUlVDVClscGFyYW07CgkJCUVudHJ5KiBlbnRyeSA9IChFbnRyeSopIGRpcy0+aXRlbURhdGE7CgoJCQlpZiAoZGlzLT5DdGxJRCA9PSBJRFdfVFJFRV9MRUZUKQoJCQkJZHJhd19pdGVtKCZjaGlsZC0+bGVmdCwgZGlzLCBlbnRyeSwgLTEpOwoJCQllbHNlIGlmIChkaXMtPkN0bElEID09IElEV19UUkVFX1JJR0hUKQoJCQkJZHJhd19pdGVtKCZjaGlsZC0+cmlnaHQsIGRpcywgZW50cnksIC0xKTsKCQkJZWxzZQoJCQkJZ290byBkcmF3X21lbnVfaXRlbTsKCgkJCXJldHVybiBUUlVFO30KCgkJY2FzZSBXTV9DUkVBVEU6CgkJCUluaXRDaGlsZFdpbmRvdyhjaGlsZCk7CgkJCWJyZWFrOwoKCQljYXNlIFdNX05DREVTVFJPWToKCQkJZnJlZV9jaGlsZF93aW5kb3coY2hpbGQpOwoJCQlTZXRXaW5kb3dMb25nUHRyKGh3bmQsIEdXTFBfVVNFUkRBVEEsIDApOwoJCQlicmVhazsKCgkJY2FzZSBXTV9QQUlOVDogewoJCQlQQUlOVFNUUlVDVCBwczsKCQkJSEJSVVNIIGxhc3RCcnVzaDsKCQkJUkVDVCBydDsKCQkJR2V0Q2xpZW50UmVjdChod25kLCAmcnQpOwoJCQlCZWdpblBhaW50KGh3bmQsICZwcyk7CgkJCXJ0LmxlZnQgPSBjaGlsZC0+c3BsaXRfcG9zLVNQTElUX1dJRFRILzI7CgkJCXJ0LnJpZ2h0ID0gY2hpbGQtPnNwbGl0X3BvcytTUExJVF9XSURUSC8yKzE7CgkJCWxhc3RCcnVzaCA9IFNlbGVjdE9iamVjdChwcy5oZGMsIEdldFN0b2NrT2JqZWN0KENPTE9SX1NQTElUQkFSKSk7CgkJCVJlY3RhbmdsZShwcy5oZGMsIHJ0LmxlZnQsIHJ0LnRvcC0xLCBydC5yaWdodCwgcnQuYm90dG9tKzEpOwoJCQlTZWxlY3RPYmplY3QocHMuaGRjLCBsYXN0QnJ1c2gpOwojaWZkZWYgX05PX0VYVEVOU0lPTlMKCQkJcnQudG9wID0gcnQuYm90dG9tIC0gR2V0U3lzdGVtTWV0cmljcyhTTV9DWUhTQ1JPTEwpOwoJCQlGaWxsUmVjdChwcy5oZGMsICZydCwgR2V0U3RvY2tPYmplY3QoQkxBQ0tfQlJVU0gpKTsKI2VuZGlmCgkJCUVuZFBhaW50KGh3bmQsICZwcyk7CgkJCWJyZWFrO30KCgkJY2FzZSBXTV9TRVRDVVJTT1I6CgkJCWlmIChMT1dPUkQobHBhcmFtKSA9PSBIVENMSUVOVCkgewoJCQkJUE9JTlQgcHQ7CgkJCQlHZXRDdXJzb3JQb3MoJnB0KTsKCQkJCVNjcmVlblRvQ2xpZW50KGh3bmQsICZwdCk7CgoJCQkJaWYgKHB0Lng+PWNoaWxkLT5zcGxpdF9wb3MtU1BMSVRfV0lEVEgvMiAmJiBwdC54PGNoaWxkLT5zcGxpdF9wb3MrU1BMSVRfV0lEVEgvMisxKSB7CgkJCQkJU2V0Q3Vyc29yKExvYWRDdXJzb3IoMCwgSURDX1NJWkVXRSkpOwoJCQkJCXJldHVybiBUUlVFOwoJCQkJfQoJCQl9CgkJCWdvdG8gZGVmOwoKCQljYXNlIFdNX0xCVVRUT05ET1dOOiB7CgkJCVJFQ1QgcnQ7CiAgICAgICAgICAgICAgICAgICAgICAgIGludCB4ID0gKHNob3J0KUxPV09SRChscGFyYW0pOwoKCQkJR2V0Q2xpZW50UmVjdChod25kLCAmcnQpOwoKCQkJaWYgKHg+PWNoaWxkLT5zcGxpdF9wb3MtU1BMSVRfV0lEVEgvMiAmJiB4PGNoaWxkLT5zcGxpdF9wb3MrU1BMSVRfV0lEVEgvMisxKSB7CgkJCQlsYXN0X3NwbGl0ID0gY2hpbGQtPnNwbGl0X3BvczsKI2lmZGVmIF9OT19FWFRFTlNJT05TCgkJCQlkcmF3X3NwbGl0YmFyKGh3bmQsIGxhc3Rfc3BsaXQpOwojZW5kaWYKCQkJCVNldENhcHR1cmUoaHduZCk7CgkJCX0KCgkJCWJyZWFrO30KCgkJY2FzZSBXTV9MQlVUVE9OVVA6CgkJCWlmIChHZXRDYXB0dXJlKCkgPT0gaHduZCkgewojaWZkZWYgX05PX0VYVEVOU0lPTlMKCQkJCVJFQ1QgcnQ7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50IHggPSAoc2hvcnQpTE9XT1JEKGxwYXJhbSk7CgkJCQlkcmF3X3NwbGl0YmFyKGh3bmQsIGxhc3Rfc3BsaXQpOwoJCQkJbGFzdF9zcGxpdCA9IC0xOwoJCQkJR2V0Q2xpZW50UmVjdChod25kLCAmcnQpOwoJCQkJY2hpbGQtPnNwbGl0X3BvcyA9IHg7CgkJCQlyZXNpemVfdHJlZShjaGlsZCwgcnQucmlnaHQsIHJ0LmJvdHRvbSk7CiNlbmRpZgoJCQkJUmVsZWFzZUNhcHR1cmUoKTsKCQkJfQoJCQlicmVhazsKCiNpZmRlZiBfTk9fRVhURU5TSU9OUwoJCWNhc2UgV01fQ0FQVFVSRUNIQU5HRUQ6CgkJCWlmIChHZXRDYXB0dXJlKCk9PWh3bmQgJiYgbGFzdF9zcGxpdD49MCkKCQkJCWRyYXdfc3BsaXRiYXIoaHduZCwgbGFzdF9zcGxpdCk7CgkJCWJyZWFrOwojZW5kaWYKCgkJY2FzZSBXTV9LRVlET1dOOgoJCQlpZiAod3BhcmFtID09IFZLX0VTQ0FQRSkKCQkJCWlmIChHZXRDYXB0dXJlKCkgPT0gaHduZCkgewoJCQkJCVJFQ1QgcnQ7CiNpZmRlZiBfTk9fRVhURU5TSU9OUwoJCQkJCWRyYXdfc3BsaXRiYXIoaHduZCwgbGFzdF9zcGxpdCk7CiNlbHNlCgkJCQkJY2hpbGQtPnNwbGl0X3BvcyA9IGxhc3Rfc3BsaXQ7CiNlbmRpZgoJCQkJCUdldENsaWVudFJlY3QoaHduZCwgJnJ0KTsKCQkJCQlyZXNpemVfdHJlZShjaGlsZCwgcnQucmlnaHQsIHJ0LmJvdHRvbSk7CgkJCQkJbGFzdF9zcGxpdCA9IC0xOwoJCQkJCVJlbGVhc2VDYXB0dXJlKCk7CgkJCQkJU2V0Q3Vyc29yKExvYWRDdXJzb3IoMCwgSURDX0FSUk9XKSk7CgkJCQl9CgkJCWJyZWFrOwoKCQljYXNlIFdNX01PVVNFTU9WRToKCQkJaWYgKEdldENhcHR1cmUoKSA9PSBod25kKSB7CgkJCQlSRUNUIHJ0OwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCB4ID0gKHNob3J0KUxPV09SRChscGFyYW0pOwoKI2lmZGVmIF9OT19FWFRFTlNJT05TCgkJCQlIREMgaGRjID0gR2V0REMoaHduZCk7CgkJCQlHZXRDbGllbnRSZWN0KGh3bmQsICZydCk7CgoJCQkJcnQubGVmdCA9IGxhc3Rfc3BsaXQtU1BMSVRfV0lEVEgvMjsKCQkJCXJ0LnJpZ2h0ID0gbGFzdF9zcGxpdCtTUExJVF9XSURUSC8yKzE7CgkJCQlJbnZlcnRSZWN0KGhkYywgJnJ0KTsKCgkJCQlsYXN0X3NwbGl0ID0geDsKCQkJCXJ0LmxlZnQgPSB4LVNQTElUX1dJRFRILzI7CgkJCQlydC5yaWdodCA9IHgrU1BMSVRfV0lEVEgvMisxOwoJCQkJSW52ZXJ0UmVjdChoZGMsICZydCk7CgoJCQkJUmVsZWFzZURDKGh3bmQsIGhkYyk7CiNlbHNlCgkJCQlHZXRDbGllbnRSZWN0KGh3bmQsICZydCk7CgoJCQkJaWYgKHg+PTAgJiYgeDxydC5yaWdodCkgewoJCQkJCWNoaWxkLT5zcGxpdF9wb3MgPSB4OwoJCQkJCXJlc2l6ZV90cmVlKGNoaWxkLCBydC5yaWdodCwgcnQuYm90dG9tKTsKCQkJCQlydC5sZWZ0ID0geC1TUExJVF9XSURUSC8yOwoJCQkJCXJ0LnJpZ2h0ID0geCtTUExJVF9XSURUSC8yKzE7CgkJCQkJSW52YWxpZGF0ZVJlY3QoaHduZCwgJnJ0LCBGQUxTRSk7CgkJCQkJVXBkYXRlV2luZG93KGNoaWxkLT5sZWZ0Lmh3bmQpOwoJCQkJCVVwZGF0ZVdpbmRvdyhod25kKTsKCQkJCQlVcGRhdGVXaW5kb3coY2hpbGQtPnJpZ2h0Lmh3bmQpOwoJCQkJfQojZW5kaWYKCQkJfQoJCQlicmVhazsKCiNpZm5kZWYgX05PX0VYVEVOU0lPTlMKCQljYXNlIFdNX0dFVE1JTk1BWElORk86CgkJCURlZk1ESUNoaWxkUHJvYyhod25kLCBubXNnLCB3cGFyYW0sIGxwYXJhbSk7CgoJCQl7TFBNSU5NQVhJTkZPIGxwbW1pID0gKExQTUlOTUFYSU5GTylscGFyYW07CgoJCQlscG1taS0+cHRNYXhUcmFja1NpemUueCA8PD0gMTsvKjIqR2V0U3lzdGVtTWV0cmljcyhTTV9DWFNDUkVFTikgLyBTTV9DWFZJUlRVQUxTQ1JFRU4gKi8KCQkJbHBtbWktPnB0TWF4VHJhY2tTaXplLnkgPDw9IDE7LyoyKkdldFN5c3RlbU1ldHJpY3MoU01fQ1lTQ1JFRU4pIC8gU01fQ1lWSVJUVUFMU0NSRUVOICovCgkJCWJyZWFrO30KI2VuZGlmIC8qIF9OT19FWFRFTlNJT05TICovCgoJCWNhc2UgV01fU0VURk9DVVM6CgkJCWlmIChTZXRDdXJyZW50RGlyZWN0b3J5KGNoaWxkLT5wYXRoKSkKCQkJCXNldF9zcGFjZV9zdGF0dXMoKTsKCQkJU2V0Rm9jdXMoY2hpbGQtPmZvY3VzX3BhbmU/IGNoaWxkLT5yaWdodC5od25kOiBjaGlsZC0+bGVmdC5od25kKTsKCQkJYnJlYWs7CgoJCWNhc2UgV01fRElTUEFUQ0hfQ09NTUFORDogewoJCQlQYW5lKiBwYW5lID0gR2V0Rm9jdXMoKT09Y2hpbGQtPmxlZnQuaHduZD8gJmNoaWxkLT5sZWZ0OiAmY2hpbGQtPnJpZ2h0OwoKCQkJc3dpdGNoKExPV09SRCh3cGFyYW0pKSB7CgkJCQljYXNlIElEX1dJTkRPV19ORVc6IHsKCQkJCQlDaGlsZFduZCogbmV3X2NoaWxkID0gYWxsb2NfY2hpbGRfd2luZG93KGNoaWxkLT5wYXRoLCBOVUxMLCBod25kKTsKCgkJCQkJaWYgKCFjcmVhdGVfY2hpbGRfd2luZG93KG5ld19jaGlsZCkpCgkJCQkJCUhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIG5ld19jaGlsZCk7CgoJCQkJCWJyZWFrO30KCgkJCQljYXNlIElEX1JFRlJFU0g6CgkJCQkJcmVmcmVzaF9kcml2ZXMoKTsKCQkJCQlyZWZyZXNoX2NoaWxkKGNoaWxkKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX0FDVElWQVRFOgoJCQkJCWFjdGl2YXRlX2VudHJ5KGNoaWxkLCBwYW5lLCBod25kKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX0ZJTEVfTU9WRTogewoJCQkJCVRDSEFSIHNvdXJjZVtCVUZGRVJfTEVOXSwgdGFyZ2V0W0JVRkZFUl9MRU5dOwoKCQkJCQlpZiAocHJvbXB0X3RhcmdldChwYW5lLCBzb3VyY2UsIHRhcmdldCkpIHsKCQkJCQkJU0hGSUxFT1BTVFJVQ1Qgc2hmbyA9IHtod25kLCBGT19NT1ZFLCBzb3VyY2UsIHRhcmdldH07CgoJCQkJCQlzb3VyY2VbbHN0cmxlbihzb3VyY2UpKzFdID0gJ1wwJzsKCQkJCQkJdGFyZ2V0W2xzdHJsZW4odGFyZ2V0KSsxXSA9ICdcMCc7CgoJCQkJCQlpZiAoIVNIRmlsZU9wZXJhdGlvbigmc2hmbykpCgkJCQkJCQlyZWZyZXNoX2NoaWxkKGNoaWxkKTsKCQkJCQl9CgkJCQkJYnJlYWs7fQoKCQkJCWNhc2UgSURfRklMRV9DT1BZOiB7CgkJCQkJVENIQVIgc291cmNlW0JVRkZFUl9MRU5dLCB0YXJnZXRbQlVGRkVSX0xFTl07CgoJCQkJCWlmIChwcm9tcHRfdGFyZ2V0KHBhbmUsIHNvdXJjZSwgdGFyZ2V0KSkgewoJCQkJCQlTSEZJTEVPUFNUUlVDVCBzaGZvID0ge2h3bmQsIEZPX0NPUFksIHNvdXJjZSwgdGFyZ2V0fTsKCgkJCQkJCXNvdXJjZVtsc3RybGVuKHNvdXJjZSkrMV0gPSAnXDAnOwoJCQkJCQl0YXJnZXRbbHN0cmxlbih0YXJnZXQpKzFdID0gJ1wwJzsKCgkJCQkJCWlmICghU0hGaWxlT3BlcmF0aW9uKCZzaGZvKSkKCQkJCQkJCXJlZnJlc2hfY2hpbGQoY2hpbGQpOwoJCQkJCX0KCQkJCQlicmVhazt9CgoJCQkJY2FzZSBJRF9GSUxFX0RFTEVURTogewoJCQkJCVRDSEFSIHBhdGhbQlVGRkVSX0xFTl07CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTSEZJTEVPUFNUUlVDVCBzaGZvID0ge2h3bmQsIEZPX0RFTEVURSwgcGF0aCwgTlVMTCwgRk9GX0FMTE9XVU5ET307CgoJCQkJCWdldF9wYXRoKHBhbmUtPmN1ciwgcGF0aCk7CgoJCQkJCXBhdGhbbHN0cmxlbihwYXRoKSsxXSA9ICdcMCc7CgoJCQkJCWlmICghU0hGaWxlT3BlcmF0aW9uKCZzaGZvKSkKCQkJCQkJcmVmcmVzaF9jaGlsZChjaGlsZCk7CgkJCQkJYnJlYWs7fQoKCQkJCWNhc2UgSURfVklFV19TT1JUX05BTUU6CgkJCQkJc2V0X3NvcnRfb3JkZXIoY2hpbGQsIFNPUlRfTkFNRSk7CgkJCQkJYnJlYWs7CgoJCQkJY2FzZSBJRF9WSUVXX1NPUlRfVFlQRToKCQkJCQlzZXRfc29ydF9vcmRlcihjaGlsZCwgU09SVF9FWFQpOwoJCQkJCWJyZWFrOwoKCQkJCWNhc2UgSURfVklFV19TT1JUX1NJWkU6CgkJCQkJc2V0X3NvcnRfb3JkZXIoY2hpbGQsIFNPUlRfU0laRSk7CgkJCQkJYnJlYWs7CgoJCQkJY2FzZSBJRF9WSUVXX1NPUlRfREFURToKCQkJCQlzZXRfc29ydF9vcmRlcihjaGlsZCwgU09SVF9EQVRFKTsKCQkJCQlicmVhazsKCgkJCQljYXNlIElEX1ZJRVdfRklMVEVSOiB7CgkJCQkJc3RydWN0IEZpbHRlckRpYWxvZyBkbGc7CgoJCQkJCW1lbXNldCgmZGxnLCAwLCBzaXplb2Yoc3RydWN0IEZpbHRlckRpYWxvZykpOwoJCQkJCWxzdHJjcHkoZGxnLnBhdHRlcm4sIGNoaWxkLT5maWx0ZXJfcGF0dGVybik7CgkJCQkJZGxnLmZsYWdzID0gY2hpbGQtPmZpbHRlcl9mbGFnczsKCgkJCQkJaWYgKERpYWxvZ0JveFBhcmFtKEdsb2JhbHMuaEluc3RhbmNlLCBNQUtFSU5UUkVTT1VSQ0UoSUREX0RJQUxPR19WSUVXX1RZUEUpLCBod25kLCBGaWx0ZXJEaWFsb2dEbGdQcm9jLCAoTFBBUkFNKSZkbGcpID09IElET0spIHsKCQkJCQkJbHN0cmNweShjaGlsZC0+ZmlsdGVyX3BhdHRlcm4sIGRsZy5wYXR0ZXJuKTsKCQkJCQkJY2hpbGQtPmZpbHRlcl9mbGFncyA9IGRsZy5mbGFnczsKCQkJCQkJcmVmcmVzaF9yaWdodF9wYW5lKGNoaWxkKTsKCQkJCQl9CgkJCQkJYnJlYWs7fQoKCQkJCWNhc2UgSURfVklFV19TUExJVDogewoJCQkJCWxhc3Rfc3BsaXQgPSBjaGlsZC0+c3BsaXRfcG9zOwojaWZkZWYgX05PX0VYVEVOU0lPTlMKCQkJCQlkcmF3X3NwbGl0YmFyKGh3bmQsIGxhc3Rfc3BsaXQpOwojZW5kaWYKCQkJCQlTZXRDYXB0dXJlKGh3bmQpOwoJCQkJCWJyZWFrO30KCgkJCQljYXNlIElEX0VESVRfUFJPUEVSVElFUzoKCQkJCQlzaG93X3Byb3BlcnRpZXNfZGxnKHBhbmUtPmN1ciwgY2hpbGQtPmh3bmQpOwoJCQkJCWJyZWFrOwoKCQkJCWRlZmF1bHQ6CgkJCQkJcmV0dXJuIHBhbmVfY29tbWFuZChwYW5lLCBMT1dPUkQod3BhcmFtKSk7CgkJCX0KCgkJCXJldHVybiBUUlVFO30KCgkJY2FzZSBXTV9DT01NQU5EOiB7CgkJCVBhbmUqIHBhbmUgPSBHZXRGb2N1cygpPT1jaGlsZC0+bGVmdC5od25kPyAmY2hpbGQtPmxlZnQ6ICZjaGlsZC0+cmlnaHQ7CgoJCQlzd2l0Y2goSElXT1JEKHdwYXJhbSkpIHsKCQkJCWNhc2UgTEJOX1NFTENIQU5HRTogewoJCQkJCWludCBpZHggPSBTZW5kTWVzc2FnZShwYW5lLT5od25kLCBMQl9HRVRDVVJTRUwsIDAsIDApOwoJCQkJCUVudHJ5KiBlbnRyeSA9IChFbnRyeSopIFNlbmRNZXNzYWdlKHBhbmUtPmh3bmQsIExCX0dFVElURU1EQVRBLCBpZHgsIDApOwoKCQkJCQlpZiAocGFuZSA9PSAmY2hpbGQtPmxlZnQpCgkJCQkJCXNldF9jdXJkaXIoY2hpbGQsIGVudHJ5LCBpZHgsIGh3bmQpOwoJCQkJCWVsc2UKCQkJCQkJcGFuZS0+Y3VyID0gZW50cnk7CgkJCQkJYnJlYWs7fQoKCQkJCWNhc2UgTEJOX0RCTENMSzoKCQkJCQlhY3RpdmF0ZV9lbnRyeShjaGlsZCwgcGFuZSwgaHduZCk7CgkJCQkJYnJlYWs7CgkJCX0KCQkJYnJlYWs7fQoKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJCWNhc2UgV01fTk9USUZZOiB7CgkJCU5NSERSKiBwbm1oID0gKE5NSERSKikgbHBhcmFtOwoJCQlyZXR1cm4gcGFuZV9ub3RpZnkocG5taC0+aWRGcm9tPT1JRFdfSEVBREVSX0xFRlQ/ICZjaGlsZC0+bGVmdDogJmNoaWxkLT5yaWdodCwgcG5taCk7fQojZW5kaWYKCiNpZmRlZiBfU0hFTExfRk9MREVSUwoJCWNhc2UgV01fQ09OVEVYVE1FTlU6IHsKCQkJUE9JTlQgcHQsIHB0X2NsbnQ7CgkJCVBhbmUqIHBhbmU7CgkJCWludCBpZHg7CgoJCQkgLyogZmlyc3Qgc2VsZWN0IHRoZSBjdXJyZW50IGl0ZW0gaW4gdGhlIGxpc3Rib3ggKi8KCQkJSFdORCBocGFuZWwgPSAoSFdORCkgd3BhcmFtOwoJCQlwdF9jbG50LnggPSBwdC54ID0gKHNob3J0KUxPV09SRChscGFyYW0pOwoJCQlwdF9jbG50LnkgPSBwdC55ID0gKHNob3J0KUhJV09SRChscGFyYW0pOwoJCQlTY3JlZW5Ub0NsaWVudChocGFuZWwsICZwdF9jbG50KTsKCQkJU2VuZE1lc3NhZ2UoaHBhbmVsLCBXTV9MQlVUVE9ORE9XTiwgMCwgTUFLRUxPTkcocHRfY2xudC54LCBwdF9jbG50LnkpKTsKCQkJU2VuZE1lc3NhZ2UoaHBhbmVsLCBXTV9MQlVUVE9OVVAsIDAsIE1BS0VMT05HKHB0X2NsbnQueCwgcHRfY2xudC55KSk7CgoJCQkgLyogbm93IGNyZWF0ZSB0aGUgcG9wdXAgbWVudSB1c2luZyBzaGVsbCBuYW1lc3BhY2UgYW5kIElDb250ZXh0TWVudSAqLwoJCQlwYW5lID0gR2V0Rm9jdXMoKT09Y2hpbGQtPmxlZnQuaHduZD8gJmNoaWxkLT5sZWZ0OiAmY2hpbGQtPnJpZ2h0OwoJCQlpZHggPSBTZW5kTWVzc2FnZShwYW5lLT5od25kLCBMQl9HRVRDVVJTRUwsIDAsIDApOwoKCQkJaWYgKGlkeCAhPSAtMSkgewoJCQkJRW50cnkqIGVudHJ5ID0gKEVudHJ5KikgU2VuZE1lc3NhZ2UocGFuZS0+aHduZCwgTEJfR0VUSVRFTURBVEEsIGlkeCwgMCk7CgoJCQkJTFBJVEVNSURMSVNUIHBpZGxfYWJzID0gZ2V0X3RvX2Fic29sdXRlX3BpZGwoZW50cnksIGh3bmQpOwoKCQkJCWlmIChwaWRsX2FicykgewoJCQkJCUlTaGVsbEZvbGRlciogcGFyZW50Rm9sZGVyOwoJCQkJCUxQQ0lURU1JRExJU1QgcGlkbExhc3Q7CgoJCQkJCSAvKiBnZXQgYW5kIHVzZSB0aGUgcGFyZW50IGZvbGRlciB0byBkaXNwbGF5IGNvcnJlY3QgY29udGV4dCBtZW51IGluIGFsbCBjYXNlcyAqLwoJCQkJCWlmIChTVUNDRUVERUQoU0hCaW5kVG9QYXJlbnQocGlkbF9hYnMsICZJSURfSVNoZWxsRm9sZGVyLCAoTFBWT0lEKikmcGFyZW50Rm9sZGVyLCAmcGlkbExhc3QpKSkgewoJCQkJCQlpZiAoU2hlbGxGb2xkZXJDb250ZXh0TWVudShwYXJlbnRGb2xkZXIsIGh3bmQsIDEsICZwaWRsTGFzdCwgcHQueCwgcHQueSkgPT0gU19PSykKCQkJCQkJCXJlZnJlc2hfY2hpbGQoY2hpbGQpOwoKCQkJCQkJSVNoZWxsRm9sZGVyX1JlbGVhc2UocGFyZW50Rm9sZGVyKTsKCQkJCQl9CgoJCQkJCUlNYWxsb2NfRnJlZShHbG9iYWxzLmlNYWxsb2MsIHBpZGxfYWJzKTsKCQkJCX0KCQkJfQoJCQlicmVhazt9CiNlbmRpZgoKCQkgIGNhc2UgV01fTUVBU1VSRUlURU06CgkJICBkcmF3X21lbnVfaXRlbToKCQkJaWYgKCF3cGFyYW0pCS8qIElzIHRoZSBtZXNzYWdlIG1lbnUtcmVsYXRlZD8gKi8KCQkJCWlmIChDdHhNZW51X0hhbmRsZU1lbnVNc2cobm1zZywgd3BhcmFtLCBscGFyYW0pKQoJCQkJCXJldHVybiBUUlVFOwoKCQkJYnJlYWs7CgoJCSAgY2FzZSBXTV9JTklUTUVOVVBPUFVQOgoJCQlpZiAoQ3R4TWVudV9IYW5kbGVNZW51TXNnKG5tc2csIHdwYXJhbSwgbHBhcmFtKSkKCQkJCXJldHVybiAwOwoKCQkJdXBkYXRlX3ZpZXdfbWVudShjaGlsZCk7CgkJCWJyZWFrOwoKCQkgIGNhc2UgV01fTUVOVUNIQVI6CS8qIG9ubHkgc3VwcG9ydGVkIGJ5IElDb250ZXh0TWVudTMgKi8KCQkgICBpZiAoc19wY3R4bWVudTMpIHsKCQkJICAgTFJFU1VMVCBsUmVzdWx0ID0gMDsKCgkJCSAgIElDb250ZXh0TWVudTNfSGFuZGxlTWVudU1zZzIoc19wY3R4bWVudTMsIG5tc2csIHdwYXJhbSwgbHBhcmFtLCAmbFJlc3VsdCk7CgoJCQkgICByZXR1cm4gbFJlc3VsdDsKCQkgICB9CgoJCSAgIGJyZWFrOwoKCQljYXNlIFdNX1NJWkU6CgkJCWlmICh3cGFyYW0gIT0gU0laRV9NSU5JTUlaRUQpCgkJCQlyZXNpemVfdHJlZShjaGlsZCwgTE9XT1JEKGxwYXJhbSksIEhJV09SRChscGFyYW0pKTsKCQkJLyogZmFsbCB0aHJvdWdoICovCgoJCWRlZmF1bHQ6IGRlZjoKCQkJcmV0dXJuIERlZk1ESUNoaWxkUHJvYyhod25kLCBubXNnLCB3cGFyYW0sIGxwYXJhbSk7Cgl9CgoJcmV0dXJuIDA7Cn0KCgpzdGF0aWMgTFJFU1VMVCBDQUxMQkFDSyBUcmVlV25kUHJvYyhIV05EIGh3bmQsIFVJTlQgbm1zZywgV1BBUkFNIHdwYXJhbSwgTFBBUkFNIGxwYXJhbSkKewoJQ2hpbGRXbmQqIGNoaWxkID0gKENoaWxkV25kKikgR2V0V2luZG93TG9uZ1B0cihHZXRQYXJlbnQoaHduZCksIEdXTFBfVVNFUkRBVEEpOwoJUGFuZSogcGFuZSA9IChQYW5lKikgR2V0V2luZG93TG9uZ1B0cihod25kLCBHV0xQX1VTRVJEQVRBKTsKCUFTU0VSVChjaGlsZCk7CgoJc3dpdGNoKG5tc2cpIHsKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJCWNhc2UgV01fSFNDUk9MTDoKCQkJc2V0X2hlYWRlcihwYW5lKTsKCQkJYnJlYWs7CiNlbmRpZgoKCQljYXNlIFdNX1NFVEZPQ1VTOgoJCQljaGlsZC0+Zm9jdXNfcGFuZSA9IHBhbmU9PSZjaGlsZC0+cmlnaHQ/IDE6IDA7CgkJCVNlbmRNZXNzYWdlKGh3bmQsIExCX1NFVFNFTCwgVFJVRSwgMSk7CgkJCS8qVE9ETzogY2hlY2sgbWVudSBpdGVtcyAqLwoJCQlicmVhazsKCgkJY2FzZSBXTV9LRVlET1dOOgoJCQlpZiAod3BhcmFtID09IFZLX1RBQikgewoJCQkJLypUT0RPOiBTZXRGb2N1cyhHbG9iYWxzLmhkcml2ZWJhcikgKi8KCQkJCVNldEZvY3VzKGNoaWxkLT5mb2N1c19wYW5lPyBjaGlsZC0+bGVmdC5od25kOiBjaGlsZC0+cmlnaHQuaHduZCk7CgkJCX0KCX0KCglyZXR1cm4gQ2FsbFdpbmRvd1Byb2MoZ19vcmdUcmVlV25kUHJvYywgaHduZCwgbm1zZywgd3BhcmFtLCBscGFyYW0pOwp9CgoKc3RhdGljIHZvaWQgSW5pdEluc3RhbmNlKEhJTlNUQU5DRSBoaW5zdGFuY2UpCnsKCXN0YXRpYyBjb25zdCBUQ0hBUiBzRm9udFtdID0geydNJywnaScsJ2MnLCdyJywnbycsJ3MnLCdvJywnZicsJ3QnLCcgJywnUycsJ2EnLCduJywncycsJyAnLCdTJywnZScsJ3InLCdpJywnZicsJ1wwJ307CgoJV05EQ0xBU1NFWCB3Y0ZyYW1lOwoJV05EQ0xBU1Mgd2NDaGlsZDsKCUFUT00gaENoaWxkQ2xhc3M7CglpbnQgY29sOwoKCUlOSVRDT01NT05DT05UUk9MU0VYIGljYyA9IHsKCQlzaXplb2YoSU5JVENPTU1PTkNPTlRST0xTRVgpLAoJCUlDQ19CQVJfQ0xBU1NFUwoJfTsKCglIREMgaGRjID0gR2V0REMoMCk7CgoJc2V0bG9jYWxlKExDX0NPTExBVEUsICIiKTsJLyogc2V0IGNvbGxhdGluZyBydWxlcyB0byBsb2NhbCBzZXR0aW5ncyBmb3IgY29tcGFyZU5hbWUgKi8KCglJbml0Q29tbW9uQ29udHJvbHNFeCgmaWNjKTsKCgoJLyogcmVnaXN0ZXIgZnJhbWUgd2luZG93IGNsYXNzICovCgoJd2NGcmFtZS5jYlNpemUgICAgICAgID0gc2l6ZW9mKFdORENMQVNTRVgpOwoJd2NGcmFtZS5zdHlsZSAgICAgICAgID0gMDsKCXdjRnJhbWUubHBmblduZFByb2MgICA9IEZyYW1lV25kUHJvYzsKCXdjRnJhbWUuY2JDbHNFeHRyYSAgICA9IDA7Cgl3Y0ZyYW1lLmNiV25kRXh0cmEgICAgPSAwOwoJd2NGcmFtZS5oSW5zdGFuY2UgICAgID0gaGluc3RhbmNlOwoJd2NGcmFtZS5oSWNvbiAgICAgICAgID0gTG9hZEljb24oaGluc3RhbmNlLCBNQUtFSU5UUkVTT1VSQ0UoSURJX1dJTkVGSUxFKSk7Cgl3Y0ZyYW1lLmhDdXJzb3IgICAgICAgPSBMb2FkQ3Vyc29yKDAsIElEQ19BUlJPVyk7Cgl3Y0ZyYW1lLmhickJhY2tncm91bmQgPSAwOwoJd2NGcmFtZS5scHN6TWVudU5hbWUgID0gMDsKCXdjRnJhbWUubHBzekNsYXNzTmFtZSA9IHNXSU5FRklMRUZSQU1FOwoJd2NGcmFtZS5oSWNvblNtICAgICAgID0gKEhJQ09OKUxvYWRJbWFnZShoaW5zdGFuY2UsCgkJCQkJCQkJCQkJIE1BS0VJTlRSRVNPVVJDRShJRElfV0lORUZJTEUpLAoJCQkJCQkJCQkJCSBJTUFHRV9JQ09OLAoJCQkJCQkJCQkJCSBHZXRTeXN0ZW1NZXRyaWNzKFNNX0NYU01JQ09OKSwKCQkJCQkJCQkJCQkgR2V0U3lzdGVtTWV0cmljcyhTTV9DWVNNSUNPTiksCgkJCQkJCQkJCQkJIExSX1NIQVJFRCk7CgoJR2xvYmFscy5oZnJhbWVDbGFzcyA9IFJlZ2lzdGVyQ2xhc3NFeCgmd2NGcmFtZSk7CgoKCS8qIHJlZ2lzdGVyIHRyZWUgd2luZG93cyBjbGFzcyAqLwoKCXdjQ2hpbGQuc3R5bGUgICAgICAgICA9IENTX0NMQVNTREN8Q1NfREJMQ0xLU3xDU19WUkVEUkFXOwoJd2NDaGlsZC5scGZuV25kUHJvYyAgID0gQ2hpbGRXbmRQcm9jOwoJd2NDaGlsZC5jYkNsc0V4dHJhICAgID0gMDsKCXdjQ2hpbGQuY2JXbmRFeHRyYSAgICA9IDA7Cgl3Y0NoaWxkLmhJbnN0YW5jZSAgICAgPSBoaW5zdGFuY2U7Cgl3Y0NoaWxkLmhJY29uICAgICAgICAgPSAwOwoJd2NDaGlsZC5oQ3Vyc29yICAgICAgID0gTG9hZEN1cnNvcigwLCBJRENfQVJST1cpOwoJd2NDaGlsZC5oYnJCYWNrZ3JvdW5kID0gMDsKCXdjQ2hpbGQubHBzek1lbnVOYW1lICA9IDA7Cgl3Y0NoaWxkLmxwc3pDbGFzc05hbWUgPSBzV0lORUZJTEVUUkVFOwoKCWhDaGlsZENsYXNzID0gUmVnaXN0ZXJDbGFzcygmd2NDaGlsZCk7CgoKCUdsb2JhbHMuaGFjY2VsID0gTG9hZEFjY2VsZXJhdG9ycyhoaW5zdGFuY2UsIE1BS0VJTlRSRVNPVVJDRShJREFfV0lORUZJTEUpKTsKCglHbG9iYWxzLmhmb250ID0gQ3JlYXRlRm9udCgtTXVsRGl2KDgsR2V0RGV2aWNlQ2FwcyhoZGMsTE9HUElYRUxTWSksNzIpLCAwLCAwLCAwLCAwLCAwLCAwLCAwLCAwLCAwLCAwLCAwLCAwLCBzRm9udCk7CgoJUmVsZWFzZURDKDAsIGhkYyk7CgoJR2xvYmFscy5oSW5zdGFuY2UgPSBoaW5zdGFuY2U7CgojaWZkZWYgX1NIRUxMX0ZPTERFUlMKCUNvSW5pdGlhbGl6ZShOVUxMKTsKCUNvR2V0TWFsbG9jKE1FTUNUWF9UQVNLLCAmR2xvYmFscy5pTWFsbG9jKTsKCVNIR2V0RGVza3RvcEZvbGRlcigmR2xvYmFscy5pRGVza3RvcCk7CglHbG9iYWxzLmNmU3RyRk5hbWUgPSBSZWdpc3RlckNsaXBib2FyZEZvcm1hdChDRlNUUl9GSUxFTkFNRSk7CiNlbmRpZgoKCS8qIGxvYWQgY29sdW1uIHN0cmluZ3MgKi8KCWNvbCA9IDE7CgoJbG9hZF9zdHJpbmcoZ19wb3NfbmFtZXNbY29sKytdLCBJRFNfQ09MX05BTUUpOwoJbG9hZF9zdHJpbmcoZ19wb3NfbmFtZXNbY29sKytdLCBJRFNfQ09MX1NJWkUpOwoJbG9hZF9zdHJpbmcoZ19wb3NfbmFtZXNbY29sKytdLCBJRFNfQ09MX0NEQVRFKTsKI2lmbmRlZiBfTk9fRVhURU5TSU9OUwoJbG9hZF9zdHJpbmcoZ19wb3NfbmFtZXNbY29sKytdLCBJRFNfQ09MX0FEQVRFKTsKCWxvYWRfc3RyaW5nKGdfcG9zX25hbWVzW2NvbCsrXSwgSURTX0NPTF9NREFURSk7Cglsb2FkX3N0cmluZyhnX3Bvc19uYW1lc1tjb2wrK10sIElEU19DT0xfSURYKTsKCWxvYWRfc3RyaW5nKGdfcG9zX25hbWVzW2NvbCsrXSwgSURTX0NPTF9MSU5LUyk7CiNlbmRpZgoJbG9hZF9zdHJpbmcoZ19wb3NfbmFtZXNbY29sKytdLCBJRFNfQ09MX0FUVFIpOwojaWZuZGVmIF9OT19FWFRFTlNJT05TCglsb2FkX3N0cmluZyhnX3Bvc19uYW1lc1tjb2wrK10sIElEU19DT0xfU0VDKTsKI2VuZGlmCn0KCgpzdGF0aWMgdm9pZCBzaG93X2ZyYW1lKEhXTkQgaHduZFBhcmVudCwgaW50IGNtZHNob3csIExQQ1RTVFIgcGF0aCkKewoJc3RhdGljIGNvbnN0IFRDSEFSIHNNRElDTElFTlRbXSA9IHsnTScsJ0QnLCdJJywnQycsJ0wnLCdJJywnRScsJ04nLCdUJywnXDAnfTsKCglUQ0hBUiBidWZmZXJbTUFYX1BBVEhdLCBiMVtCVUZGRVJfTEVOXTsKCUNoaWxkV25kKiBjaGlsZDsKCUhNRU5VIGhNZW51RnJhbWUsIGhNZW51V2luZG93OwoJd2luZG93T3B0aW9ucyBvcHRzOwoKCUNMSUVOVENSRUFURVNUUlVDVCBjY3M7CgoJaWYgKEdsb2JhbHMuaE1haW5XbmQpCgkJcmV0dXJuOwoKCW9wdHMgPSBsb2FkX3JlZ2lzdHJ5X3NldHRpbmdzKCk7CgloTWVudUZyYW1lID0gTG9hZE1lbnUoR2xvYmFscy5oSW5zdGFuY2UsIE1BS0VJTlRSRVNPVVJDRShJRE1fV0lORUZJTEUpKTsKCWhNZW51V2luZG93ID0gR2V0U3ViTWVudShoTWVudUZyYW1lLCBHZXRNZW51SXRlbUNvdW50KGhNZW51RnJhbWUpLTIpOwoKCUdsb2JhbHMuaE1lbnVGcmFtZSA9IGhNZW51RnJhbWU7CglHbG9iYWxzLmhNZW51VmlldyA9IEdldFN1Yk1lbnUoaE1lbnVGcmFtZSwgMyk7CglHbG9iYWxzLmhNZW51T3B0aW9ucyA9IEdldFN1Yk1lbnUoaE1lbnVGcmFtZSwgNCk7CgoJY2NzLmhXaW5kb3dNZW51ICA9IGhNZW51V2luZG93OwoJY2NzLmlkRmlyc3RDaGlsZCA9IElEV19GSVJTVF9DSElMRDsKCgoJLyogY3JlYXRlIG1haW4gd2luZG93ICovCglHbG9iYWxzLmhNYWluV25kID0gQ3JlYXRlV2luZG93RXgoMCwgTUFLRUlOVFJFU09VUkNFKEdsb2JhbHMuaGZyYW1lQ2xhc3MpLCBSUyhiMSxJRFNfV0lORV9GSUxFKSwgV1NfT1ZFUkxBUFBFRFdJTkRPVywKCQkJCQlvcHRzLnN0YXJ0X3gsIG9wdHMuc3RhcnRfeSwgb3B0cy53aWR0aCwgb3B0cy5oZWlnaHQsCgkJCQkJaHduZFBhcmVudCwgR2xvYmFscy5oTWVudUZyYW1lLCBHbG9iYWxzLmhJbnN0YW5jZSwgMC8qbHBQYXJhbSovKTsKCgoJR2xvYmFscy5obWRpY2xpZW50ID0gQ3JlYXRlV2luZG93RXgoMCwgc01ESUNMSUVOVCwgTlVMTCwKCQkJCQlXU19DSElMRHxXU19DTElQQ0hJTERSRU58V1NfVlNDUk9MTHxXU19IU0NST0xMfFdTX1ZJU0lCTEV8V1NfQk9SREVSLAoJCQkJCTAsIDAsIDAsIDAsCgkJCQkJR2xvYmFscy5oTWFpblduZCwgMCwgR2xvYmFscy5oSW5zdGFuY2UsICZjY3MpOwogIAoJQ2hlY2tNZW51SXRlbShHbG9iYWxzLmhNZW51T3B0aW9ucywgSURfVklFV19EUklWRV9CQVIsIE1GX0JZQ09NTUFORHxNRl9DSEVDS0VEKTsKCUNoZWNrTWVudUl0ZW0oR2xvYmFscy5oTWVudU9wdGlvbnMsIElEX1ZJRVdfU0FWRVNFVFRJTkdTLCBNRl9CWUNPTU1BTkQpOwoKCWNyZWF0ZV9kcml2ZV9iYXIoKTsKCgl7CgkJVEJCVVRUT04gdG9vbGJhckJ0bnNbXSA9IHsKCQkJezAsIDAsIDAsIEJUTlNfU0VQLCB7MCwgMH0sIDAsIDB9LAoJCQl7MCwgSURfV0lORE9XX05FVywgVEJTVEFURV9FTkFCTEVELCBCVE5TX0JVVFRPTiwgezAsIDB9LCAwLCAwfSwKCQkJezEsIElEX1dJTkRPV19DQVNDQURFLCBUQlNUQVRFX0VOQUJMRUQsIEJUTlNfQlVUVE9OLCB7MCwgMH0sIDAsIDB9LAoJCQl7MiwgSURfV0lORE9XX1RJTEVfSE9SWiwgVEJTVEFURV9FTkFCTEVELCBCVE5TX0JVVFRPTiwgezAsIDB9LCAwLCAwfSwKCQkJezMsIElEX1dJTkRPV19USUxFX1ZFUlQsIFRCU1RBVEVfRU5BQkxFRCwgQlROU19CVVRUT04sIHswLCAwfSwgMCwgMH0sCi8qVE9ETwoJCQl7NCwgSURfLi4uICwgVEJTVEFURV9FTkFCTEVELCBCVE5TX0JVVFRPTiwgezAsIDB9LCAwLCAwfSwKCQkJezUsIElEXy4uLiAsIFRCU1RBVEVfRU5BQkxFRCwgQlROU19CVVRUT04sIHswLCAwfSwgMCwgMH0sCiovCQl9OwoKCQlHbG9iYWxzLmh0b29sYmFyID0gQ3JlYXRlVG9vbGJhckV4KEdsb2JhbHMuaE1haW5XbmQsIFdTX0NISUxEfFdTX1ZJU0lCTEUsCgkJCUlEV19UT09MQkFSLCAyLCBHbG9iYWxzLmhJbnN0YW5jZSwgSURCX1RPT0xCQVIsIHRvb2xiYXJCdG5zLAoJCQlzaXplb2YodG9vbGJhckJ0bnMpL3NpemVvZihUQkJVVFRPTiksIDE2LCAxNSwgMTYsIDE1LCBzaXplb2YoVEJCVVRUT04pKTsKCQlDaGVja01lbnVJdGVtKEdsb2JhbHMuaE1lbnVPcHRpb25zLCBJRF9WSUVXX1RPT0xfQkFSLCBNRl9CWUNPTU1BTkR8TUZfQ0hFQ0tFRCk7Cgl9CgoJR2xvYmFscy5oc3RhdHVzYmFyID0gQ3JlYXRlU3RhdHVzV2luZG93KFdTX0NISUxEfFdTX1ZJU0lCTEUsIDAsIEdsb2JhbHMuaE1haW5XbmQsIElEV19TVEFUVVNCQVIpOwoJQ2hlY2tNZW51SXRlbShHbG9iYWxzLmhNZW51T3B0aW9ucywgSURfVklFV19TVEFUVVNCQVIsIE1GX0JZQ09NTUFORHxNRl9DSEVDS0VEKTsKCi8qIENyZWF0ZVN0YXR1c1dpbmRvdyBkb2VzIG5vdCBhY2NlcHQgV1NfQk9SREVSCglHbG9iYWxzLmhzdGF0dXNiYXIgPSBDcmVhdGVXaW5kb3dFeChXU19FWF9OT1BBUkVOVE5PVElGWSwgU1RBVFVTQ0xBU1NOQU1FLCAwLAoJCQkJCVdTX0NISUxEfFdTX1ZJU0lCTEV8V1NfQ0xJUFNJQkxJTkdTfFdTX0JPUkRFUnxDQ1NfTk9ESVZJREVSLCAwLDAsMCwwLAoJCQkJCUdsb2JhbHMuaE1haW5XbmQsIChITUVOVSlJRFdfU1RBVFVTQkFSLCBoaW5zdGFuY2UsIDApOyovCgoJLypUT0RPOiByZWFkIHBhdGhzIGZyb20gcmVnaXN0cnkgKi8KCglpZiAoIXBhdGggfHwgISpwYXRoKSB7CgkJR2V0Q3VycmVudERpcmVjdG9yeShNQVhfUEFUSCwgYnVmZmVyKTsKCQlwYXRoID0gYnVmZmVyOwoJfQoKCVNob3dXaW5kb3coR2xvYmFscy5oTWFpblduZCwgY21kc2hvdyk7CgojaWYgZGVmaW5lZChfU0hFTExfRk9MREVSUykgJiYgIWRlZmluZWQoX19XSU5FX18pCgkgLyogU2hlbGwgTmFtZXNwYWNlIGFzIGRlZmF1bHQ6ICovCgljaGlsZCA9IGFsbG9jX2NoaWxkX3dpbmRvdyhwYXRoLCBnZXRfcGF0aF9waWRsKHBhdGgsR2xvYmFscy5oTWFpblduZCksIEdsb2JhbHMuaE1haW5XbmQpOwojZWxzZQoJY2hpbGQgPSBhbGxvY19jaGlsZF93aW5kb3cocGF0aCwgTlVMTCwgR2xvYmFscy5oTWFpblduZCk7CiNlbmRpZgoKCWNoaWxkLT5wb3Muc2hvd0NtZCA9IFNXX1NIT1dNQVhJTUlaRUQ7CgljaGlsZC0+cG9zLnJjTm9ybWFsUG9zaXRpb24ubGVmdCA9IDA7CgljaGlsZC0+cG9zLnJjTm9ybWFsUG9zaXRpb24udG9wID0gMDsKCWNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi5yaWdodCA9IDMyMDsKCWNoaWxkLT5wb3MucmNOb3JtYWxQb3NpdGlvbi5ib3R0b20gPSAyODA7CgoJaWYgKCFjcmVhdGVfY2hpbGRfd2luZG93KGNoaWxkKSkKCQlIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBjaGlsZCk7CgoJU2V0V2luZG93UGxhY2VtZW50KGNoaWxkLT5od25kLCAmY2hpbGQtPnBvcyk7CgoJR2xvYmFscy5oaW1sID0gSW1hZ2VMaXN0X0xvYWRCaXRtYXAoR2xvYmFscy5oSW5zdGFuY2UsIE1BS0VJTlRSRVNPVVJDRShJREJfSU1BR0VTKSwgMTYsIDAsIFJHQigwLDI1NSwwKSk7CgoJR2xvYmFscy5wcmVzY2FuX25vZGUgPSBGQUxTRTsKCglVcGRhdGVXaW5kb3coR2xvYmFscy5oTWFpblduZCk7CgoJaWYgKHBhdGggJiYgcGF0aFswXSkKCXsKCQlpbnQgaW5kZXgsY291bnQ7CgkJVENIQVIgZHJ2W19NQVhfRFJJVkUrMV0sIGRpcltfTUFYX0RJUl0sIG5hbWVbX01BWF9GTkFNRV0sIGV4dFtfTUFYX0VYVF07CgkJVENIQVIgZnVsbG5hbWVbX01BWF9GTkFNRStfTUFYX0VYVCsxXTsKCgkJbWVtc2V0KG5hbWUsMCxzaXplb2YobmFtZSkpOwoJCW1lbXNldChuYW1lLDAsc2l6ZW9mKGV4dCkpOwoJCV90c3BsaXRwYXRoKHBhdGgsIGRydiwgZGlyLCBuYW1lLCBleHQpOwoJCWlmIChuYW1lWzBdKQoJCXsKCQkJY291bnQgPSBTZW5kTWVzc2FnZShjaGlsZC0+cmlnaHQuaHduZCwgTEJfR0VUQ09VTlQsIDAsIDApOwoJCQlsc3RyY3B5KGZ1bGxuYW1lLG5hbWUpOwoJCQlsc3RyY2F0KGZ1bGxuYW1lLGV4dCk7CgoJCQlmb3IgKGluZGV4ID0gMDsgaW5kZXggPCBjb3VudDsgaW5kZXggKyspCgkJCXsKCQkJCUVudHJ5KiBlbnRyeSA9IChFbnRyeSopIFNlbmRNZXNzYWdlKGNoaWxkLT5yaWdodC5od25kLCBMQl9HRVRJVEVNREFUQSwgaW5kZXgsIDApOwoJCQkJaWYgKGxzdHJjbXAoZW50cnktPmRhdGEuY0ZpbGVOYW1lLGZ1bGxuYW1lKT09MCB8fAoJCQkJCQlsc3RyY21wKGVudHJ5LT5kYXRhLmNBbHRlcm5hdGVGaWxlTmFtZSxmdWxsbmFtZSk9PTApCgkJCQl7CgkJCQkJU2VuZE1lc3NhZ2UoY2hpbGQtPnJpZ2h0Lmh3bmQsIExCX1NFVENVUlNFTCwgaW5kZXgsIDApOwoJCQkJCVNldEZvY3VzKGNoaWxkLT5yaWdodC5od25kKTsKCQkJCQlicmVhazsKCQkJCX0KCQkJfQoJCX0KCX0KfQoKc3RhdGljIHZvaWQgRXhpdEluc3RhbmNlKHZvaWQpCnsKI2lmZGVmIF9TSEVMTF9GT0xERVJTCglJU2hlbGxGb2xkZXJfUmVsZWFzZShHbG9iYWxzLmlEZXNrdG9wKTsKCUlNYWxsb2NfUmVsZWFzZShHbG9iYWxzLmlNYWxsb2MpOwoJQ29VbmluaXRpYWxpemUoKTsKI2VuZGlmCgoJRGVsZXRlT2JqZWN0KEdsb2JhbHMuaGZvbnQpOwoJSW1hZ2VMaXN0X0Rlc3Ryb3koR2xvYmFscy5oaW1sKTsKfQoKI2lmZGVmIF9OT19FWFRFTlNJT05TCgovKiBzZWFyY2ggZm9yIGFscmVhZHkgcnVubmluZyB3aW5bZV1maWxlcyAqLwoKc3RhdGljIGludCBnX2ZvdW5kUHJldkluc3RhbmNlID0gMDsKCnN0YXRpYyBCT09MIENBTExCQUNLIEVudW1XbmRQcm9jKEhXTkQgaHduZCwgTFBBUkFNIGxwYXJhbSkKewoJVENIQVIgY2xzWzEyOF07CgoJR2V0Q2xhc3NOYW1lKGh3bmQsIGNscywgMTI4KTsKCglpZiAoIWxzdHJjbXAoY2xzLCAoTFBDVFNUUilscGFyYW0pKSB7CgkJZ19mb3VuZFByZXZJbnN0YW5jZSsrOwoJCXJldHVybiBGQUxTRTsKCX0KCglyZXR1cm4gVFJVRTsKfQoKLyogc2VhcmNoIGZvciB3aW5kb3cgb2YgZ2l2ZW4gY2xhc3MgbmFtZSB0byBhbGxvdyBvbmx5IG9uZSBydW5uaW5nIGluc3RhbmNlICovCnN0YXRpYyBpbnQgZmluZF93aW5kb3dfY2xhc3MoTFBDVFNUUiBjbGFzc25hbWUpCnsKCUVudW1XaW5kb3dzKEVudW1XbmRQcm9jLCAoTFBBUkFNKWNsYXNzbmFtZSk7CgoJaWYgKGdfZm91bmRQcmV2SW5zdGFuY2UpCgkJcmV0dXJuIDE7CgoJcmV0dXJuIDA7Cn0KCiNlbmRpZgoKc3RhdGljIGludCB3aW5lZmlsZV9tYWluKEhJTlNUQU5DRSBoaW5zdGFuY2UsIGludCBjbWRzaG93LCBMUENUU1RSIHBhdGgpCnsKCU1TRyBtc2c7CiAgCglJbml0SW5zdGFuY2UoaGluc3RhbmNlKTsKCglzaG93X2ZyYW1lKDAsIGNtZHNob3csIHBhdGgpOwoKCXdoaWxlKEdldE1lc3NhZ2UoJm1zZywgMCwgMCwgMCkpIHsKCQlpZiAoR2xvYmFscy5obWRpY2xpZW50ICYmIFRyYW5zbGF0ZU1ESVN5c0FjY2VsKEdsb2JhbHMuaG1kaWNsaWVudCwgJm1zZykpCgkJCWNvbnRpbnVlOwoKCQlpZiAoR2xvYmFscy5oTWFpblduZCAmJiBUcmFuc2xhdGVBY2NlbGVyYXRvcihHbG9iYWxzLmhNYWluV25kLCBHbG9iYWxzLmhhY2NlbCwgJm1zZykpCgkJCWNvbnRpbnVlOwoKCQlUcmFuc2xhdGVNZXNzYWdlKCZtc2cpOwoJCURpc3BhdGNoTWVzc2FnZSgmbXNnKTsKCX0KCglFeGl0SW5zdGFuY2UoKTsKCglyZXR1cm4gbXNnLndQYXJhbTsKfQoKCiNpZiBkZWZpbmVkKFVOSUNPREUpICYmIGRlZmluZWQoX01TQ19WRVIpCmludCBBUElFTlRSWSB3V2luTWFpbihISU5TVEFOQ0UgaGluc3RhbmNlLCBISU5TVEFOQ0UgcHJldmluc3RhbmNlLCBMUFdTVFIgY21kbGluZSwgaW50IGNtZHNob3cpCiNlbHNlCmludCBBUElFTlRSWSBXaW5NYWluKEhJTlNUQU5DRSBoaW5zdGFuY2UsIEhJTlNUQU5DRSBwcmV2aW5zdGFuY2UsIExQU1RSIGNtZGxpbmUsIGludCBjbWRzaG93KQojZW5kaWYKewojaWZkZWYgX05PX0VYVEVOU0lPTlMKCWlmIChmaW5kX3dpbmRvd19jbGFzcyhzV0lORUZJTEVGUkFNRSkpCgkJcmV0dXJuIDE7CiNlbmRpZgoKI2lmIGRlZmluZWQoVU5JQ09ERSkgJiYgIWRlZmluZWQoX01TQ19WRVIpCgl7IC8qIGNvbnZlcnQgQU5TSSBjbWRsaW5lIGludG8gV0NTIHBhdGggc3RyaW5nICovCglUQ0hBUiBidWZmZXJbTUFYX1BBVEhdOwoJTXVsdGlCeXRlVG9XaWRlQ2hhcihDUF9BQ1AsIDAsIGNtZGxpbmUsIC0xLCBidWZmZXIsIE1BWF9QQVRIKTsKCXdpbmVmaWxlX21haW4oaGluc3RhbmNlLCBjbWRzaG93LCBidWZmZXIpOwoJfQojZWxzZQoJd2luZWZpbGVfbWFpbihoaW5zdGFuY2UsIGNtZHNob3csIGNtZGxpbmUpOwojZW5kaWYKCglyZXR1cm4gMDsKfQo=