Lyogdmk6c2V0IHRzPTggc3RzPTQgc3c9NDoKICoKICogVklNIC0gVmkgSU1wcm92ZWQJCWJ5IEJyYW0gTW9vbGVuYWFyCiAqCQkJCUdVSS9Nb3RpZiBzdXBwb3J0IGJ5IFJvYmVydCBXZWJiCiAqCiAqIERvICI6aGVscCB1Z2FuZGEiICBpbiBWaW0gdG8gcmVhZCBjb3B5aW5nIGFuZCB1c2FnZSBjb25kaXRpb25zLgogKiBEbyAiOmhlbHAgY3JlZGl0cyIgaW4gVmltIHRvIHNlZSBhIGxpc3Qgb2YgcGVvcGxlIHdobyBjb250cmlidXRlZC4KICogU2VlIFJFQURNRS50eHQgZm9yIGFuIG92ZXJ2aWV3IG9mIHRoZSBWaW0gc291cmNlIGNvZGUuCiAqLwoKI2luY2x1ZGUgPFhtL0Zvcm0uaD4KI2luY2x1ZGUgPFhtL1Jvd0NvbHVtbi5oPgojaW5jbHVkZSA8WG0vUHVzaEIuaD4KI2luY2x1ZGUgPFhtL1RleHQuaD4KI2luY2x1ZGUgPFhtL1RleHRGLmg+CiNpbmNsdWRlIDxYbS9TZXBhcmF0b3IuaD4KI2luY2x1ZGUgPFhtL0xhYmVsLmg+CiNpbmNsdWRlIDxYbS9DYXNjYWRlQi5oPgojaW5jbHVkZSA8WG0vU2Nyb2xsQmFyLmg+CiNpbmNsdWRlIDxYbS9NZW51U2hlbGwuaD4KI2luY2x1ZGUgPFhtL0RyYXdpbmdBLmg+CiNpZiAoWG1WZXJzaW9uID49IDEwMDIpCiMgaW5jbHVkZSA8WG0vUmVwVHlwZS5oPgojZW5kaWYKI2luY2x1ZGUgPFhtL0ZyYW1lLmg+CiNpbmNsdWRlIDxYbS9MYWJlbEcuaD4KI2luY2x1ZGUgPFhtL1RvZ2dsZUJHLmg+CiNpbmNsdWRlIDxYbS9TZXBhcmF0b0cuaD4KI2luY2x1ZGUgPFhtL1htUC5oPgoKI2luY2x1ZGUgPFgxMS9rZXlzeW0uaD4KI2luY2x1ZGUgPFgxMS9YYXRvbS5oPgojaW5jbHVkZSA8WDExL1N0cmluZ0RlZnMuaD4KI2luY2x1ZGUgPFgxMS9JbnRyaW5zaWMuaD4KCiNpbmNsdWRlICJ2aW0uaCIKCiNpZmRlZiBIQVZFX1gxMV9YUE1fSAojIGluY2x1ZGUgPFgxMS94cG0uaD4KI2Vsc2UKIyBpZmRlZiBIQVZFX1hNX1hQTVBfSAojICBpbmNsdWRlIDxYbS9YcG1QLmg+CiMgZW5kaWYKI2VuZGlmCiNpZmRlZiBIQVZFX1hNX05PVEVCT09LX0gKIyBpbmNsdWRlIDxYbS9Ob3RlYm9vay5oPgojZW5kaWYKCiNpbmNsdWRlICJndWlfeG1lYncuaCIJLyogZm9yIG91ciBFbmhhbmNlZCBCdXR0b24gV2lkZ2V0ICovCgojaWYgZGVmaW5lZChGRUFUX0dVSV9ESUFMT0cpICYmIGRlZmluZWQoSEFWRV9YUE0pCiMgaW5jbHVkZSAiLi4vcGl4bWFwcy9hbGVydC54cG0iCiMgaW5jbHVkZSAiLi4vcGl4bWFwcy9lcnJvci54cG0iCiMgaW5jbHVkZSAiLi4vcGl4bWFwcy9nZW5lcmljLnhwbSIKIyBpbmNsdWRlICIuLi9waXhtYXBzL2luZm8ueHBtIgojIGluY2x1ZGUgIi4uL3BpeG1hcHMvcXVlc3QueHBtIgojZW5kaWYKCiNkZWZpbmUgTU9USUZfUE9QVVAKCmV4dGVybiBXaWRnZXQgdmltU2hlbGw7CgpzdGF0aWMgV2lkZ2V0IHZpbUZvcm07CnN0YXRpYyBXaWRnZXQgdGV4dEFyZWFGb3JtOwpXaWRnZXQgdGV4dEFyZWE7CiNpZmRlZiBGRUFUX1RPT0xCQVIKc3RhdGljIFdpZGdldCB0b29sQmFyRnJhbWU7CnN0YXRpYyBXaWRnZXQgdG9vbEJhcjsKI2VuZGlmCiNpZmRlZiBGRUFUX0dVSV9UQUJMSU5FCnN0YXRpYyBXaWRnZXQJdGFiTGluZTsKc3RhdGljIFdpZGdldAl0YWJMaW5lX21lbnUgPSAwOwpzdGF0aWMgaW50CXNob3dpbmdfdGFibGluZSA9IDA7CiNlbmRpZgojaWZkZWYgRkVBVF9GT09URVIKc3RhdGljIFdpZGdldCBmb290ZXI7CiNlbmRpZgojaWZkZWYgRkVBVF9NRU5VCiMgaWYgKFhtVmVyc2lvbiA+PSAxMDAyKQovKiByZW1lbWJlciB0aGUgbGFzdCBzZXQgdmFsdWUgZm9yIHRoZSB0ZWFyb2ZmIGl0ZW0gKi8Kc3RhdGljIGludCB0ZWFyb2ZmX3ZhbCA9IChpbnQpWG1URUFSX09GRl9FTkFCTEVEOwojIGVuZGlmCnN0YXRpYyBXaWRnZXQgbWVudUJhcjsKI2VuZGlmCgpzdGF0aWMgdm9pZCBzY3JvbGxfY2IgX19BUkdTKChXaWRnZXQgdywgWHRQb2ludGVyIGNsaWVudF9kYXRhLCBYdFBvaW50ZXIgY2FsbF9kYXRhKSk7CiNpZmRlZiBGRUFUX0dVSV9UQUJMSU5FCnN0YXRpYyB2b2lkIHRhYmxpbmVfY2IgX19BUkdTKChXaWRnZXQgdywgWHRQb2ludGVyIGNsaWVudF9kYXRhLCBYdFBvaW50ZXIgY2FsbF9kYXRhKSk7CnN0YXRpYyB2b2lkIHRhYmxpbmVfYnV0dG9uX2NiIF9fQVJHUygoV2lkZ2V0IHcsIFh0UG9pbnRlciBjbGllbnRfZGF0YSwgWHRQb2ludGVyIGNhbGxfZGF0YSkpOwpzdGF0aWMgdm9pZCB0YWJsaW5lX21lbnVfY2IgX19BUkdTKChXaWRnZXQgdywgWHRQb2ludGVyCWNsb3N1cmUsIFhFdmVudAkqZSwgQm9vbGVhbiAqY29udGludWVfZGlzcGF0Y2gpKTsKc3RhdGljIHZvaWQgdGFibGluZV9iYWxsb29uX2NiIF9fQVJHUygoQmFsbG9vbkV2YWwgKmJldmFsLCBpbnQgc3RhdGUpKTsKI2VuZGlmCiNpZmRlZiBGRUFUX1RPT0xCQVIKIyBpZmRlZiBGRUFUX0ZPT1RFUgpzdGF0aWMgdm9pZCB0b29sYmFyYnV0dG9uX2VudGVyX2NiIF9fQVJHUygoV2lkZ2V0LCBYdFBvaW50ZXIsIFhFdmVudCAqLCBCb29sZWFuICopKTsKc3RhdGljIHZvaWQgdG9vbGJhcmJ1dHRvbl9sZWF2ZV9jYiBfX0FSR1MoKFdpZGdldCwgWHRQb2ludGVyLCBYRXZlbnQgKiwgQm9vbGVhbiAqKSk7CiMgZW5kaWYKc3RhdGljIHZvaWQgcmVzZXRfZm9jdXMgX19BUkdTKCh2b2lkKSk7CiNlbmRpZgojaWZkZWYgRkVBVF9GT09URVIKc3RhdGljIGludCBndWlfbWNoX2NvbXB1dGVfZm9vdGVyX2hlaWdodCBfX0FSR1MoKHZvaWQpKTsKI2VuZGlmCiNpZmRlZiBXU0RFQlVHCnN0YXRpYyB2b2lkIGF0dGFjaER1bXAoV2lkZ2V0LCBjaGFyICopOwojZW5kaWYKCnN0YXRpYyB2b2lkIGd1aV9tb3RpZl9tZW51X2NvbG9ycyBfX0FSR1MoKFdpZGdldCBpZCkpOwpzdGF0aWMgdm9pZCBndWlfbW90aWZfc2Nyb2xsX2NvbG9ycyBfX0FSR1MoKFdpZGdldCBpZCkpOwoKI2lmIChYbVZlcnNpb24gPj0gMTAwMikKIyBkZWZpbmUgU1RSSU5HX1RBRyAgWG1GT05UTElTVF9ERUZBVUxUX1RBRwojZWxzZQojIGRlZmluZSBTVFJJTkdfVEFHICBYbVNUUklOR19ERUZBVUxUX0NIQVJTRVQKI2VuZGlmCgovKgogKiBDYWxsLWJhY2sgcm91dGluZXMuCiAqLwoKICAgIHN0YXRpYyB2b2lkCnNjcm9sbF9jYih3LCBjbGllbnRfZGF0YSwgY2FsbF9kYXRhKQogICAgV2lkZ2V0CXcgVU5VU0VEOwogICAgWHRQb2ludGVyCWNsaWVudF9kYXRhLCBjYWxsX2RhdGE7CnsKICAgIHNjcm9sbGJhcl9UICpzYjsKICAgIGxvbmcJdmFsdWU7CiAgICBpbnQJCWRyYWdnaW5nOwoKICAgIHNiID0gZ3VpX2ZpbmRfc2Nyb2xsYmFyKChsb25nKWNsaWVudF9kYXRhKTsKCiAgICB2YWx1ZSA9ICgoWG1TY3JvbGxCYXJDYWxsYmFja1N0cnVjdCAqKWNhbGxfZGF0YSktPnZhbHVlOwogICAgZHJhZ2dpbmcgPSAoKChYbVNjcm9sbEJhckNhbGxiYWNrU3RydWN0ICopY2FsbF9kYXRhKS0+cmVhc29uID09CgkJCQkJCQkgICAgICAoaW50KVhtQ1JfRFJBRyk7CiAgICBndWlfZHJhZ19zY3JvbGxiYXIoc2IsIHZhbHVlLCBkcmFnZ2luZyk7Cn0KCiNpZmRlZiBGRUFUX0dVSV9UQUJMSU5FCiAgICBzdGF0aWMgdm9pZAp0YWJsaW5lX2NiKHcsIGNsaWVudF9kYXRhLCBjYWxsX2RhdGEpCiAgICBXaWRnZXQJdyBVTlVTRUQ7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGEgVU5VU0VEOwogICAgWHRQb2ludGVyCWNhbGxfZGF0YTsKewogICAgWG1Ob3RlYm9va0NhbGxiYWNrU3RydWN0ICpucHRyOwoKICAgIG5wdHIgPSAoWG1Ob3RlYm9va0NhbGxiYWNrU3RydWN0ICopY2FsbF9kYXRhOwogICAgaWYgKG5wdHItPnJlYXNvbiAhPSAoaW50KVhtQ1JfTk9ORSkKCXNlbmRfdGFibGluZV9ldmVudChucHRyLT5wYWdlX251bWJlcik7Cn0KCiAgICBzdGF0aWMgdm9pZAp0YWJsaW5lX2J1dHRvbl9jYih3LCBjbGllbnRfZGF0YSwgY2FsbF9kYXRhKQogICAgV2lkZ2V0CXc7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGEgVU5VU0VEOwogICAgWHRQb2ludGVyCWNhbGxfZGF0YSBVTlVTRUQ7CnsKICAgIGludAkJY21kLCB0YWJfaWR4OwoKICAgIFh0VmFHZXRWYWx1ZXModywgWG1OdXNlckRhdGEsICZjbWQsIE5VTEwpOwogICAgWHRWYUdldFZhbHVlcyh0YWJMaW5lX21lbnUsIFhtTnVzZXJEYXRhLCAmdGFiX2lkeCwgTlVMTCk7CgogICAgc2VuZF90YWJsaW5lX21lbnVfZXZlbnQodGFiX2lkeCwgY21kKTsKfQoKLyoKICogVGFibGluZSBzaW5nbGUgbW91c2UgY2xpY2sgdGltZW91dCBoYW5kbGVyCiAqLwogICAgc3RhdGljIHZvaWQKbW90aWZfdGFibGluZV90aW1lcl9jYiAodGltZWRfb3V0LCBpbnRlcnZhbF9pZCkKICAgIFh0UG9pbnRlcgkJdGltZWRfb3V0OwogICAgWHRJbnRlcnZhbElkCSppbnRlcnZhbF9pZCBVTlVTRUQ7CnsKICAgICooKGludCAqKXRpbWVkX291dCkgPSBUUlVFOwp9CgovKgogKiBjaGVjayBpZiB0aGUgdGFibGluZSB0YWIgc2Nyb2xsZXIgaXMgY2xpY2tlZAogKi8KICAgIHN0YXRpYyBpbnQKdGFibGluZV9zY3JvbGxlcl9jbGlja2VkKHNjcm9sbGVyX25hbWUsIGV2ZW50KQogICAgY2hhcgkJKnNjcm9sbGVyX25hbWU7CiAgICBYQnV0dG9uUHJlc3NlZEV2ZW50ICpldmVudDsKewogICAgV2lkZ2V0CXRhYl9zY3JvbGxfdzsKICAgIFBvc2l0aW9uCXBvc194LCBwb3NfeTsKICAgIERpbWVuc2lvbgl3aWR0aCwgaGVpZ2h0OwoKICAgIHRhYl9zY3JvbGxfdyA9IFh0TmFtZVRvV2lkZ2V0KHRhYkxpbmUsIHNjcm9sbGVyX25hbWUpOwogICAgaWYgKHRhYl9zY3JvbGxfdyAhPSAoV2lkZ2V0KTApIHsKCVh0VmFHZXRWYWx1ZXModGFiX3Njcm9sbF93LCBYbU54LCAmcG9zX3gsIFhtTnksICZwb3NfeSwgWG1Od2lkdGgsCgkJICAgICAgJndpZHRoLCBYbU5oZWlnaHQsICZoZWlnaHQsIE5VTEwpOwoJaWYgKHBvc194ID49IDApIHsKCSAgICAvKiBUYWIgc2Nyb2xsZXIgKG5leHQpIGlzIHZpc2libGUgKi8KCSAgICBpZiAoKGV2ZW50LT54ID49IHBvc194KSAmJiAoZXZlbnQtPnggPD0gcG9zX3ggKyB3aWR0aCkgJiYKCQkoZXZlbnQtPnkgPj0gcG9zX3kpICYmIChldmVudC0+eSA8PSBwb3NfeSArIGhlaWdodCkpIHsKCQkvKiBDbGlja2VkIG9uIHRoZSBzY3JvbGxlciAqLwoJCXJldHVybiBUUlVFOwoJICAgIH0KCX0KICAgIH0KICAgIHJldHVybiBGQUxTRTsKfQoKICAgIHN0YXRpYyB2b2lkCnRhYmxpbmVfbWVudV9jYih3LCBjbG9zdXJlLCBlLCBjb250aW51ZV9kaXNwYXRjaCkKICAgIFdpZGdldAl3OwogICAgWHRQb2ludGVyCWNsb3N1cmUgVU5VU0VEOwogICAgWEV2ZW50CSplOwogICAgQm9vbGVhbgkqY29udGludWVfZGlzcGF0Y2ggVU5VU0VEOwp7CiAgICBXaWRnZXQJCQl0YWJfdzsKICAgIFhCdXR0b25QcmVzc2VkRXZlbnQJCSpldmVudDsKICAgIGludAkJCQl0YWJfaWR4ID0gMDsKICAgIFdpZGdldExpc3QJCQljaGlsZHJlbjsKICAgIENhcmRpbmFsCQkJbnVtQ2hpbGRyZW47CiAgICBzdGF0aWMgWHRJbnRlcnZhbElkCQl0aW1lciA9IChYdEludGVydmFsSWQpMDsKICAgIHN0YXRpYyBpbnQJCQl0aW1lZF9vdXQgPSBUUlVFOwoKICAgIGV2ZW50ID0gKFhCdXR0b25QcmVzc2VkRXZlbnQgKillOwoKICAgIGlmIChldmVudC0+YnV0dG9uID09IEJ1dHRvbjEpCiAgICB7CglpZiAodGFibGluZV9zY3JvbGxlcl9jbGlja2VkKCJNYWpvclRhYlNjcm9sbGVyTmV4dCIsIGV2ZW50KQoJICAgIHx8IHRhYmxpbmVfc2Nyb2xsZXJfY2xpY2tlZCgiTWFqb3JUYWJTY3JvbGxlclByZXZpb3VzIiwgZXZlbnQpKQoJICAgIHJldHVybjsKCglpZiAoIXRpbWVkX291dCkKCXsKCSAgICBYdFJlbW92ZVRpbWVPdXQodGltZXIpOwoJICAgIHRpbWVkX291dCA9IFRSVUU7CgoJICAgIC8qCgkgICAgICogRG91YmxlIGNsaWNrIG9uIHRoZSB0YWJsaW5lIGd1dHRlciwgYWRkIGEgbmV3IHRhYgoJICAgICAqLwoJICAgIHNlbmRfdGFibGluZV9tZW51X2V2ZW50KDAsIFRBQkxJTkVfTUVOVV9ORVcpOwoJfQoJZWxzZQoJewoJICAgIC8qCgkgICAgICogU2luZ2xlIGNsaWNrIG9uIHRoZSB0YWJsaW5lIGd1dHRlciwgc3RhcnQgYSB0aW1lciB0byBjaGVjawoJICAgICAqIGZvciBkb3VibGUgY2xpY2tzCgkgICAgICovCgkgICAgdGltZXIgPSBYdEFwcEFkZFRpbWVPdXQoYXBwX2NvbnRleHQsIChsb25nX3UpcF9tb3VzZXQsCgkJCQkgICAgbW90aWZfdGFibGluZV90aW1lcl9jYiwgJnRpbWVkX291dCk7CgkgICAgdGltZWRfb3V0ID0gRkFMU0U7Cgl9CglyZXR1cm47CiAgICB9CgogICAgaWYgKGV2ZW50LT5idXR0b24gIT0gQnV0dG9uMykKCXJldHVybjsKCiAgICAvKiBXaGVuIGlnbm9yaW5nIGV2ZW50cyBkb24ndCBzaG93IHRoZSBtZW51LiAqLwogICAgaWYgKGhvbGRfZ3VpX2V2ZW50cwojIGlmZGVmIEZFQVRfQ01EV0lOCgkgICAgfHwgY21kd2luX3R5cGUgIT0gMAojIGVuZGlmCiAgICAgICApCglyZXR1cm47CgogICAgaWYgKGV2ZW50LT5zdWJ3aW5kb3cgIT0gTm9uZSkKICAgIHsKCXRhYl93ID0gWHRXaW5kb3dUb1dpZGdldChYdERpc3BsYXkodyksIGV2ZW50LT5zdWJ3aW5kb3cpOwoJLyogTElOVEVEOiBhdm9pZCB3YXJuaW5nOiBkdWJpb3VzIG9wZXJhdGlvbiBvbiBlbnVtICovCglpZiAodGFiX3cgIT0gKFdpZGdldCkwICYmIFhtSXNQdXNoQnV0dG9uKHRhYl93KSkKCSAgICBYdFZhR2V0VmFsdWVzKHRhYl93LCBYbU5wYWdlTnVtYmVyLCAmdGFiX2lkeCwgTlVMTCk7CiAgICB9CgogICAgWHRWYVNldFZhbHVlcyh0YWJMaW5lX21lbnUsIFhtTnVzZXJEYXRhLCB0YWJfaWR4LCBOVUxMKTsKICAgIFh0VmFHZXRWYWx1ZXModGFiTGluZV9tZW51LCBYbU5jaGlsZHJlbiwgJmNoaWxkcmVuLCBYbU5udW1DaGlsZHJlbiwKCQkgICZudW1DaGlsZHJlbiwgTlVMTCk7CiAgICBYdE1hbmFnZUNoaWxkcmVuKGNoaWxkcmVuLCBudW1DaGlsZHJlbik7CiAgICBYbU1lbnVQb3NpdGlvbih0YWJMaW5lX21lbnUsIChYQnV0dG9uUHJlc3NlZEV2ZW50ICopZSkgOwogICAgWHRNYW5hZ2VDaGlsZCh0YWJMaW5lX21lbnUpOwp9CgogICAgc3RhdGljIHZvaWQKdGFibGluZV9iYWxsb29uX2NiKGJldmFsLCBzdGF0ZSkKICAgIEJhbGxvb25FdmFsCSpiZXZhbDsKICAgIGludAkJc3RhdGUgVU5VU0VEOwp7CiAgICBpbnQJCW5yOwogICAgdGFicGFnZV9UCSp0cDsKCiAgICBpZiAoYmV2YWwtPnRhcmdldCA9PSAoV2lkZ2V0KTApCglyZXR1cm47CgogICAgWHRWYUdldFZhbHVlcyhiZXZhbC0+dGFyZ2V0LCBYbU5wYWdlTnVtYmVyLCAmbnIsIE5VTEwpOwogICAgdHAgPSBmaW5kX3RhYnBhZ2UobnIpOwogICAgaWYgKHRwID09IE5VTEwpCglyZXR1cm47CgogICAgZ2V0X3RhYmxpbmVfbGFiZWwodHAsIFRSVUUpOwogICAgZ3VpX21jaF9wb3N0X2JhbGxvb24oYmV2YWwsIE5hbWVCdWZmKTsKfQoKI2VuZGlmCgovKgogKiBFbmQgb2YgY2FsbC1iYWNrIHJvdXRpbmVzCiAqLwoKLyoKICogSW1wbGVtZW50IHRocmVlIGRpbWVuc2lvbmFsIHNoYWRpbmcgb2YgaW5zZW5zaXRpdmUgbGFiZWxzLgogKiBCeSBNYXJjaW4gRGFsZWNraS4KICovCgojaW5jbHVkZSA8WG0vWG1QLmg+CiNpbmNsdWRlIDxYbS9MYWJlbFAuaD4KCnN0YXRpYyBYdEV4cG9zZVByb2Mgb2xkX2xhYmVsX2V4cG9zZSA9IE5VTEw7CgpzdGF0aWMgdm9pZCBsYWJlbF9leHBvc2UgX19BUkdTKChXaWRnZXQgX3csIFhFdmVudCAqX2V2ZW50LCBSZWdpb24gX3JlZ2lvbikpOwoKICAgIHN0YXRpYyB2b2lkCmxhYmVsX2V4cG9zZShfdywgX2V2ZW50LCBfcmVnaW9uKQogICAgV2lkZ2V0CV93OwogICAgWEV2ZW50CSpfZXZlbnQ7CiAgICBSZWdpb24JX3JlZ2lvbjsKewogICAgR0MJCSAgICBpbnNlbnNpdGl2ZUdDOwogICAgWG1MYWJlbFdpZGdldCAgIGx3ID0gKFhtTGFiZWxXaWRnZXQpX3c7CiAgICB1bnNpZ25lZCBjaGFyICAgbGFiZWxfdHlwZSA9IChpbnQpWG1TVFJJTkc7CgogICAgWHRWYUdldFZhbHVlcyhfdywgWG1ObGFiZWxUeXBlLCAmbGFiZWxfdHlwZSwgKFh0UG9pbnRlcikwKTsKCiAgICBpZiAoWHRJc1NlbnNpdGl2ZShfdykgfHwgbGFiZWxfdHlwZSAhPSAoaW50KVhtU1RSSU5HKQoJKCpvbGRfbGFiZWxfZXhwb3NlKShfdywgX2V2ZW50LCBfcmVnaW9uKTsKICAgIGVsc2UKICAgIHsKCVhHQ1ZhbHVlcyAgIHZhbHVlczsKCVh0R0NNYXNrICAgIG1hc2s7CglYdEdDTWFzayAgICBkeW5hbWljOwoJWEZvbnRTdHJ1Y3QgKmZzOwoKCV9YbUZvbnRMaXN0R2V0RGVmYXVsdEZvbnQobHctPmxhYmVsLmZvbnQsICZmcyk7CgoJLyogRklYTUU6IHdlIHNob3VsZCBiZSBkb2luZyB0aGUgd2hvbGUgZHJhd2luZyBvdXJzZWxmIGhlcmUuICovCglpbnNlbnNpdGl2ZUdDID0gbHctPmxhYmVsLmluc2Vuc2l0aXZlX0dDOwoKCW1hc2sgPSBHQ0ZvcmVncm91bmQgfCBHQ0JhY2tncm91bmQgfCBHQ0dyYXBoaWNzRXhwb3N1cmVzOwoJZHluYW1pYyA9IEdDQ2xpcE1hc2sgfCBHQ0NsaXBYT3JpZ2luIHwgR0NDbGlwWU9yaWdpbjsKCXZhbHVlcy5ncmFwaGljc19leHBvc3VyZXMgPSBGYWxzZTsKCglpZiAoZnMgIT0gMCkKCXsKCSAgICBtYXNrIHw9IEdDRm9udDsKCSAgICB2YWx1ZXMuZm9udCA9IGZzLT5maWQ7Cgl9CgoJaWYgKGx3LT5wcmltaXRpdmUudG9wX3NoYWRvd19waXhtYXAgIT0gTm9uZQoJCSYmIGx3LT5wcmltaXRpdmUudG9wX3NoYWRvd19waXhtYXAgIT0gWG1VTlNQRUNJRklFRF9QSVhNQVApCgl7CgkgICAgbWFzayB8PSBHQ0ZpbGxTdHlsZSB8IEdDVGlsZTsKCSAgICB2YWx1ZXMuZmlsbF9zdHlsZSA9IEZpbGxUaWxlZDsKCSAgICB2YWx1ZXMudGlsZSA9IGx3LT5wcmltaXRpdmUudG9wX3NoYWRvd19waXhtYXA7Cgl9CgoJbHctPmxhYmVsLlRleHRSZWN0LnggKz0gMTsKCWx3LT5sYWJlbC5UZXh0UmVjdC55ICs9IDE7CglpZiAobHctPmxhYmVsLl9hY2NfdGV4dCAhPSAwKQoJewoJICAgIGx3LT5sYWJlbC5hY2NfVGV4dFJlY3QueCArPSAxOwoJICAgIGx3LT5sYWJlbC5hY2NfVGV4dFJlY3QueSArPSAxOwoJfQoKCXZhbHVlcy5mb3JlZ3JvdW5kID0gbHctPnByaW1pdGl2ZS50b3Bfc2hhZG93X2NvbG9yOwoJdmFsdWVzLmJhY2tncm91bmQgPSBsdy0+Y29yZS5iYWNrZ3JvdW5kX3BpeGVsOwoKCWx3LT5sYWJlbC5pbnNlbnNpdGl2ZV9HQyA9IFh0QWxsb2NhdGVHQygoV2lkZ2V0KWx3LCAwLCBtYXNrLAoJCQkJCSAgICAgICAmdmFsdWVzLCBkeW5hbWljLCAoWHRHQ01hc2spMCk7CgkoKm9sZF9sYWJlbF9leHBvc2UpKF93LCBfZXZlbnQsIF9yZWdpb24pOwoJWHRSZWxlYXNlR0MoX3csIGx3LT5sYWJlbC5pbnNlbnNpdGl2ZV9HQyk7CgoJbHctPmxhYmVsLlRleHRSZWN0LnggLT0gMTsKCWx3LT5sYWJlbC5UZXh0UmVjdC55IC09IDE7CglpZiAobHctPmxhYmVsLl9hY2NfdGV4dCAhPSAwKQoJewoJICAgIGx3LT5sYWJlbC5hY2NfVGV4dFJlY3QueCAtPSAxOwoJICAgIGx3LT5sYWJlbC5hY2NfVGV4dFJlY3QueSAtPSAxOwoJfQoKCXZhbHVlcy5mb3JlZ3JvdW5kID0gbHctPnByaW1pdGl2ZS5ib3R0b21fc2hhZG93X2NvbG9yOwoJdmFsdWVzLmJhY2tncm91bmQgPSBsdy0+Y29yZS5iYWNrZ3JvdW5kX3BpeGVsOwoKCWx3LT5sYWJlbC5pbnNlbnNpdGl2ZV9HQyA9IFh0QWxsb2NhdGVHQygoV2lkZ2V0KSBsdywgMCwgbWFzaywKCQkJCQkgICAgICAgJnZhbHVlcywgZHluYW1pYywgKFh0R0NNYXNrKTApOwoJKCpvbGRfbGFiZWxfZXhwb3NlKShfdywgX2V2ZW50LCBfcmVnaW9uKTsKCVh0UmVsZWFzZUdDKF93LCBsdy0+bGFiZWwuaW5zZW5zaXRpdmVfR0MpOwoKCWx3LT5sYWJlbC5pbnNlbnNpdGl2ZV9HQyA9IGluc2Vuc2l0aXZlR0M7CiAgICB9Cn0KCi8qCiAqIENyZWF0ZSBhbGwgdGhlIG1vdGlmIHdpZGdldHMgbmVjZXNzYXJ5LgogKi8KICAgIHZvaWQKZ3VpX3gxMV9jcmVhdGVfd2lkZ2V0cygpCnsKI2lmZGVmIEZFQVRfR1VJX1RBQkxJTkUKICAgIFdpZGdldAlidXR0b24sIHNjcm9sbGVyOwogICAgQXJnCQlhcmdzWzEwXTsKICAgIGludAkJbjsKICAgIFhtU3RyaW5nCXhtczsKI2VuZGlmCgogICAgLyoKICAgICAqIEluc3RhbGwgdGhlIDNEIHNoYWRlIGVmZmVjdCBkcmF3aW5nIHJvdXRpbmVzLgogICAgICovCiAgICBpZiAob2xkX2xhYmVsX2V4cG9zZSA9PSBOVUxMKQogICAgewoJb2xkX2xhYmVsX2V4cG9zZSA9IHhtTGFiZWxXaWRnZXRDbGFzcy0+Y29yZV9jbGFzcy5leHBvc2U7Cgl4bUxhYmVsV2lkZ2V0Q2xhc3MtPmNvcmVfY2xhc3MuZXhwb3NlID0gbGFiZWxfZXhwb3NlOwogICAgfQoKICAgIC8qCiAgICAgKiBTdGFydCBvdXQgYnkgYWRkaW5nIHRoZSBjb25maWd1cmVkIGJvcmRlciB3aWR0aCBpbnRvIHRoZSBib3JkZXIgb2Zmc2V0CiAgICAgKi8KICAgIGd1aS5ib3JkZXJfb2Zmc2V0ID0gZ3VpLmJvcmRlcl93aWR0aDsKCiAgICAvKgogICAgICogSW5zdGFsbCB0aGUgdGVhck9mZk1vZGVsIHJlc291cmNlIGNvbnZlcnRlci4KICAgICAqLwojaWYgKFhtVmVyc2lvbiA+PSAxMDAyKQogICAgWG1SZXBUeXBlSW5zdGFsbFRlYXJPZmZNb2RlbENvbnZlcnRlcigpOwojZW5kaWYKCiAgICAvKiBNYWtlIHN1cmUgdGhlICJRdWl0IiBtZW51IGVudHJ5IG9mIHRoZSB3aW5kb3cgbWFuYWdlciBpcyBpZ25vcmVkICovCiAgICBYdFZhU2V0VmFsdWVzKHZpbVNoZWxsLCBYbU5kZWxldGVSZXNwb25zZSwgWG1ET19OT1RISU5HLCBOVUxMKTsKCiAgICB2aW1Gb3JtID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoInZpbUZvcm0iLAoJeG1Gb3JtV2lkZ2V0Q2xhc3MsIHZpbVNoZWxsLAoJWG1OYm9yZGVyV2lkdGgsIDAsCglYbU5oaWdobGlnaHRUaGlja25lc3MsIDAsCglYbU5zaGFkb3dUaGlja25lc3MsIDAsCglYbU5tYXJnaW5XaWR0aCwgMCwKCVhtTm1hcmdpbkhlaWdodCwgMCwKCVhtTnJlc2l6ZVBvbGljeSwgWG1SRVNJWkVfQU5ZLAoJTlVMTCk7CiAgICBndWlfbW90aWZfbWVudV9jb2xvcnModmltRm9ybSk7CgojaWZkZWYgRkVBVF9NRU5VCiAgICB7CglBcmcgYWxbN107IC8qIE1ha2Ugc3VyZSB0aGVyZSBpcyBlbm91Z2ggcm9vbSBmb3IgYXJndW1lbnRzISAqLwoJaW50IGFjID0gMDsKCiMgaWYgKFhtVmVyc2lvbiA+PSAxMDAyKQoJWHRTZXRBcmcoYWxbYWNdLCBYbU50ZWFyT2ZmTW9kZWwsIHRlYXJvZmZfdmFsKTsgYWMrKzsKIyBlbmRpZgoJWHRTZXRBcmcoYWxbYWNdLCBYbU5sZWZ0QXR0YWNobWVudCwgIFhtQVRUQUNIX0ZPUk0pOyBhYysrOwoJWHRTZXRBcmcoYWxbYWNdLCBYbU50b3BBdHRhY2htZW50LCAgIFhtQVRUQUNIX0ZPUk0pOyBhYysrOwoJWHRTZXRBcmcoYWxbYWNdLCBYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0pOyBhYysrOwojIGlmbmRlZiBGRUFUX1RPT0xCQVIKCS8qIEFsd2F5cyBzdGljayB0byByaWdodCBoYW5kIHNpZGUuICovCglYdFNldEFyZyhhbFthY10sIFhtTnJpZ2h0T2Zmc2V0LCAwKTsgYWMrKzsKIyBlbmRpZgoJWHRTZXRBcmcoYWxbYWNdLCBYbU5tYXJnaW5IZWlnaHQsIDApOyBhYysrOwoJbWVudUJhciA9IFhtQ3JlYXRlTWVudUJhcih2aW1Gb3JtLCAibWVudUJhciIsIGFsLCBhYyk7CglYdE1hbmFnZUNoaWxkKG1lbnVCYXIpOwoKCS8qIFJlbWVtYmVyIHRoZSBkZWZhdWx0IGNvbG9ycywgbmVlZGVkIGZvciAiOmhpIGNsZWFyIi4gKi8KCVh0VmFHZXRWYWx1ZXMobWVudUJhciwKCSAgICBYbU5iYWNrZ3JvdW5kLCAmZ3VpLm1lbnVfZGVmX2JnX3BpeGVsLAoJICAgIFhtTmZvcmVncm91bmQsICZndWkubWVudV9kZWZfZmdfcGl4ZWwsCgkgICAgTlVMTCk7CglndWlfbW90aWZfbWVudV9jb2xvcnMobWVudUJhcik7CiAgICB9CiNlbmRpZgoKI2lmZGVmIEZFQVRfVE9PTEJBUgogICAgLyoKICAgICAqIENyZWF0ZSBhbiBlbXB0eSBUb29sQmFyLiBXZSBzaG91bGQgZ2V0IGJ1dHRvbnMgZGVmaW5lZCBmcm9tIG1lbnUudmltLgogICAgICovCiAgICB0b29sQmFyRnJhbWUgPSBYdFZhQ3JlYXRlV2lkZ2V0KCJ0b29sQmFyRnJhbWUiLAoJeG1GcmFtZVdpZGdldENsYXNzLCB2aW1Gb3JtLAoJWG1Oc2hhZG93VGhpY2tuZXNzLCAwLAoJWG1ObWFyZ2luSGVpZ2h0LCAwLAoJWG1ObWFyZ2luV2lkdGgsIDAsCglYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCVhtTnJpZ2h0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCU5VTEwpOwogICAgZ3VpX21vdGlmX21lbnVfY29sb3JzKHRvb2xCYXJGcmFtZSk7CgogICAgdG9vbEJhciA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJ0b29sQmFyIiwKCXhtUm93Q29sdW1uV2lkZ2V0Q2xhc3MsIHRvb2xCYXJGcmFtZSwKCVhtTmNoaWxkVHlwZSwgWG1GUkFNRV9XT1JLQVJFQV9DSElMRCwKCVhtTnJvd0NvbHVtblR5cGUsIFhtV09SS19BUkVBLAoJWG1Ob3JpZW50YXRpb24sIFhtSE9SSVpPTlRBTCwKCVhtTnRyYXZlcnNhbE9uLCBGYWxzZSwKCVhtTmlzSG9tb2dlbmVvdXMsIEZhbHNlLAoJWG1OcGFja2luZywgWG1QQUNLX1RJR0hULAoJWG1Oc3BhY2luZywgMCwKCVhtTnNoYWRvd1RoaWNrbmVzcywgMCwKCVhtTmhpZ2hsaWdodFRoaWNrbmVzcywgMCwKCVhtTm1hcmdpbkhlaWdodCwgMCwKCVhtTm1hcmdpbldpZHRoLCAwLAoJWG1OYWRqdXN0TGFzdCwgVHJ1ZSwKCU5VTEwpOwogICAgZ3VpX21vdGlmX21lbnVfY29sb3JzKHRvb2xCYXIpOwoKI2VuZGlmCgojaWZkZWYgRkVBVF9HVUlfVEFCTElORQogICAgLyogQ3JlYXRlIHRoZSBWaW0gR1VJIHRhYmxpbmUgKi8KICAgIG4gPSAwOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OYmluZGluZ1R5cGUsIFhtTk9ORSk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTm9yaWVudGF0aW9uLCBYbVZFUlRJQ0FMKTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OYmFja1BhZ2VTaXplLCBYbU5PTkUpOyBuKys7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5iYWNrUGFnZU51bWJlciwgMCk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJhY2tQYWdlUGxhY2VtZW50LCBYbVRPUF9SSUdIVCk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTm1ham9yVGFiU3BhY2luZywgMCk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnNoYWRvd1RoaWNrbmVzcywgMCk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwogICAgdGFiTGluZSA9IFhtQ3JlYXRlTm90ZWJvb2sodmltRm9ybSwgIlZpbSB0YWJsaW5lIiwgYXJncywgbik7CgogICAgWHRBZGRDYWxsYmFjayh0YWJMaW5lLCBYbU5wYWdlQ2hhbmdlZENhbGxiYWNrLCAoWHRDYWxsYmFja1Byb2MpdGFibGluZV9jYiwKCQkJTlVMTCk7CiAgICBYdEFkZEV2ZW50SGFuZGxlcih0YWJMaW5lLCBCdXR0b25QcmVzc01hc2ssIEZhbHNlLAoJCQkoWHRFdmVudEhhbmRsZXIpdGFibGluZV9tZW51X2NiLCBOVUxMKTsKCiAgICBndWkudGFibGluZV9oZWlnaHQgPSBUQUJMSU5FX0hFSUdIVDsKCiAgICAvKgogICAgICogU2V0IHRoZSBzaXplIG9mIHRoZSBtaW5vciBuZXh0L3ByZXYgc2Nyb2xsZXJzIHRvIHplcm8sIHNvCiAgICAgKiB0aGF0IHRoZXkgYXJlIG5vdCBkaXNwbGF5ZWQuIER1ZSB0byBhIGJ1ZyBpbiBPcGVuTW90aWYgMi4zLAogICAgICogZXZlbiBpZiB0aGVzZSBjaGlsZHJlbiB3aWRnZXQgYXJlIHVubWFuYWdlZCwgdGhleSBhcmUgYWdhaW4KICAgICAqIG1hbmFnZWQgYnkgdGhlIE5vdGVib29rIHdpZGdldCBhbmQgdGhlIG5vdGVib29rIHdpZGdldCBnZW9tZXRyeQogICAgICogaXMgYWRqdXN0ZWQgdG8gYWNjb3VudCBmb3IgdGhlIG1pbm9yIHNjcm9sbGVyIHdpZGdldHMuCiAgICAgKi8KICAgIHNjcm9sbGVyID0gWHROYW1lVG9XaWRnZXQodGFiTGluZSwgIk1pbm9yVGFiU2Nyb2xsZXJOZXh0Iik7CiAgICBYdFZhU2V0VmFsdWVzKHNjcm9sbGVyLCBYbU53aWR0aCwgMCwgWG1OcmVzaXphYmxlLCBGYWxzZSwKCQkgIFhtTnRyYXZlcnNhbE9uLCBGYWxzZSwgTlVMTCk7CiAgICBzY3JvbGxlciA9IFh0TmFtZVRvV2lkZ2V0KHRhYkxpbmUsICJNaW5vclRhYlNjcm9sbGVyUHJldmlvdXMiKTsKICAgIFh0VmFTZXRWYWx1ZXMoc2Nyb2xsZXIsIFhtTndpZHRoLCAwLCBYbU5yZXNpemFibGUsIEZhbHNlLAoJCSAgWG1OdHJhdmVyc2FsT24sIEZhbHNlLCBOVUxMKTsKCiAgICAvKiBDcmVhdGUgdGhlIHRhYmxpbmUgcG9wdXAgbWVudSAqLwogICAgdGFiTGluZV9tZW51ID0gWG1DcmVhdGVQb3B1cE1lbnUodGFiTGluZSwgInRhYmxpbmUgcG9wdXAiLCBOVUxMLCAwKTsKCiAgICAvKiBBZGQgdGhlIGJ1dHRvbnMgdG8gdGhlIG1lbnUgKi8KICAgIG4gPSAwOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OdXNlckRhdGEsIFRBQkxJTkVfTUVOVV9DTE9TRSk7IG4rKzsKICAgIHhtcyA9IFhtU3RyaW5nQ3JlYXRlKChjaGFyICopIkNsb3NlIHRhYiIsIFNUUklOR19UQUcpOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1ObGFiZWxTdHJpbmcsIHhtcyk7IG4rKzsKICAgIGJ1dHRvbiA9IFhtQ3JlYXRlUHVzaEJ1dHRvbih0YWJMaW5lX21lbnUsICJDbG9zZSIsIGFyZ3MsIG4pOwogICAgWHRBZGRDYWxsYmFjayhidXR0b24sIFhtTmFjdGl2YXRlQ2FsbGJhY2ssCgkJICAoWHRDYWxsYmFja1Byb2MpdGFibGluZV9idXR0b25fY2IsIE5VTEwpOwogICAgWG1TdHJpbmdGcmVlKHhtcyk7CgogICAgbiA9IDA7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU51c2VyRGF0YSwgVEFCTElORV9NRU5VX05FVyk7IG4rKzsKICAgIHhtcyA9IFhtU3RyaW5nQ3JlYXRlKChjaGFyICopIk5ldyBUYWIiLCBTVFJJTkdfVEFHKTsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxhYmVsU3RyaW5nLCB4bXMpOyBuKys7CiAgICBidXR0b24gPSBYbUNyZWF0ZVB1c2hCdXR0b24odGFiTGluZV9tZW51LCAiTmV3IFRhYiIsIGFyZ3MsIG4pOwogICAgWHRBZGRDYWxsYmFjayhidXR0b24sIFhtTmFjdGl2YXRlQ2FsbGJhY2ssCgkJICAoWHRDYWxsYmFja1Byb2MpdGFibGluZV9idXR0b25fY2IsIE5VTEwpOwogICAgWG1TdHJpbmdGcmVlKHhtcyk7CgogICAgbiA9IDA7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU51c2VyRGF0YSwgVEFCTElORV9NRU5VX09QRU4pOyBuKys7CiAgICB4bXMgPSBYbVN0cmluZ0NyZWF0ZSgoY2hhciAqKSJPcGVuIHRhYi4uLiIsIFNUUklOR19UQUcpOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1ObGFiZWxTdHJpbmcsIHhtcyk7IG4rKzsKICAgIGJ1dHRvbiA9IFhtQ3JlYXRlUHVzaEJ1dHRvbih0YWJMaW5lX21lbnUsICJPcGVuIHRhYi4uLiIsIGFyZ3MsIG4pOwogICAgWHRBZGRDYWxsYmFjayhidXR0b24sIFhtTmFjdGl2YXRlQ2FsbGJhY2ssCgkJICAoWHRDYWxsYmFja1Byb2MpdGFibGluZV9idXR0b25fY2IsIE5VTEwpOwogICAgWG1TdHJpbmdGcmVlKHhtcyk7CiNlbmRpZgoKICAgIHRleHRBcmVhRm9ybSA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJ0ZXh0QXJlYUZvcm0iLAoJeG1Gb3JtV2lkZ2V0Q2xhc3MsIHZpbUZvcm0sCglYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCVhtTnJpZ2h0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCVhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCglYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJWG1ObWFyZ2luV2lkdGgsIDAsCglYbU5tYXJnaW5IZWlnaHQsIDAsCglYbU5yZXNpemVQb2xpY3ksIFhtUkVTSVpFX0FOWSwKCU5VTEwpOwogICAgZ3VpX21vdGlmX3Njcm9sbF9jb2xvcnModGV4dEFyZWFGb3JtKTsKCiAgICB0ZXh0QXJlYSA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJ0ZXh0QXJlYSIsCgl4bURyYXdpbmdBcmVhV2lkZ2V0Q2xhc3MsIHRleHRBcmVhRm9ybSwKCVhtTmZvcmVncm91bmQsIGd1aS5ub3JtX3BpeGVsLAoJWG1OYmFja2dyb3VuZCwgZ3VpLmJhY2tfcGl4ZWwsCglYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCVhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCglYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCglYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoKCS8qCgkgKiBUaGVzZSB0YWtlIHNvbWUgY29udHJvbCBhd2F5IGZyb20gdGhlIHVzZXIsIGJ1dCBhdm9pZHMgbWFraW5nIHRoZW0KCSAqIGFkZCByZXNvdXJjZXMgdG8gZ2V0IGEgZGVjZW50IGxvb2tpbmcgc2V0dXAuCgkgKi8KCVhtTmJvcmRlcldpZHRoLCAwLAoJWG1OaGlnaGxpZ2h0VGhpY2tuZXNzLCAwLAoJWG1Oc2hhZG93VGhpY2tuZXNzLCAwLAoJTlVMTCk7CgojaWZkZWYgRkVBVF9GT09URVIKICAgIC8qCiAgICAgKiBDcmVhdGUgdGhlIEZvb3Rlci4KICAgICAqLwogICAgZm9vdGVyID0gWHRWYUNyZWF0ZVdpZGdldCgiZm9vdGVyIiwKCXhtTGFiZWxHYWRnZXRDbGFzcywgdmltRm9ybSwKCVhtTmFsaWdubWVudCwgWG1BTElHTk1FTlRfQkVHSU5OSU5HLAoJWG1ObWFyZ2luSGVpZ2h0LCAwLAoJWG1ObWFyZ2luV2lkdGgsIDAsCglYbU50cmF2ZXJzYWxPbiwgRmFsc2UsCglYbU5yZWNvbXB1dGVTaXplLCBGYWxzZSwKCVhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJWG1ObGVmdE9mZnNldCwgNSwKCVhtTnJpZ2h0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCVhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCglOVUxMKTsKICAgIGd1aV9tY2hfc2V0X2Zvb3RlcigoY2hhcl91ICopICIiKTsKI2VuZGlmCgogICAgLyoKICAgICAqIEluc3RhbGwgdGhlIGNhbGxiYWNrcy4KICAgICAqLwogICAgZ3VpX3gxMV9jYWxsYmFja3ModGV4dEFyZWEsIHZpbUZvcm0pOwoKICAgIC8qIFByZXRlbmQgd2UgZG9uJ3QgaGF2ZSBpbnB1dCBmb2N1cywgd2Ugd2lsbCBnZXQgYW4gZXZlbnQgaWYgd2UgZG8uICovCiAgICBndWkuaW5fZm9jdXMgPSBGQUxTRTsKfQoKLyoKICogQ2FsbGVkIHdoZW4gdGhlIEdVSSBpcyBub3QgZ29pbmcgdG8gc3RhcnQgYWZ0ZXIgYWxsLgogKi8KICAgIHZvaWQKZ3VpX3gxMV9kZXN0cm95X3dpZGdldHMoKQp7CiAgICB0ZXh0QXJlYSA9IE5VTEw7CiNpZmRlZiBGRUFUX01FTlUKICAgIG1lbnVCYXIgPSBOVUxMOwojZW5kaWYKfQoKICAgIHZvaWQKZ3VpX21jaF9zZXRfdGV4dF9hcmVhX3Bvcyh4LCB5LCB3LCBoKQogICAgaW50CSAgICB4IFVOVVNFRDsKICAgIGludAkgICAgeSBVTlVTRUQ7CiAgICBpbnQJICAgIHcgVU5VU0VEOwogICAgaW50CSAgICBoIFVOVVNFRDsKewojaWZkZWYgRkVBVF9UT09MQkFSCiAgICAvKiBHaXZlIGtleWJvYXJkIGZvY3VzIHRvIHRoZSB0ZXh0QXJlYSBpbnN0ZWFkIG9mIHRoZSB0b29sYmFyLiAqLwogICAgcmVzZXRfZm9jdXMoKTsKI2VuZGlmCn0KCiAgICB2b2lkCmd1aV94MTFfc2V0X2JhY2tfY29sb3IoKQp7CiAgICBpZiAodGV4dEFyZWEgIT0gTlVMTCkKI2lmIChYbVZlcnNpb24gPj0gMTAwMikKCVhtQ2hhbmdlQ29sb3IodGV4dEFyZWEsIGd1aS5iYWNrX3BpeGVsKTsKI2Vsc2UKCVh0VmFTZXRWYWx1ZXModGV4dEFyZWEsCgkJICBYbU5iYWNrZ3JvdW5kLCBndWkuYmFja19waXhlbCwKCQkgIE5VTEwpOwojZW5kaWYKfQoKLyoKICogTWFuYWdlIGRpYWxvZyBjZW50ZXJlZCBvbiBwb2ludGVyLiBUaGlzIGNvdWxkIGJlIHVzZWQgYnkgdGhlIEF0aGVuYSBjb2RlIGFzCiAqIHdlbGwuCiAqLwogICAgdm9pZAptYW5hZ2VfY2VudGVyZWQoZGlhbG9nX2NoaWxkKQogICAgV2lkZ2V0IGRpYWxvZ19jaGlsZDsKewogICAgV2lkZ2V0IHNoZWxsID0gWHRQYXJlbnQoZGlhbG9nX2NoaWxkKTsKICAgIFdpbmRvdyByb290LCBjaGlsZDsKICAgIHVuc2lnbmVkIGludCBtYXNrOwogICAgdW5zaWduZWQgaW50IHdpZHRoLCBoZWlnaHQsIGJvcmRlcl93aWR0aCwgZGVwdGg7CiAgICBpbnQgeCwgeSwgd2luX3gsIHdpbl95LCBtYXhYLCBtYXhZOwogICAgQm9vbGVhbiBtYXBwZWRXaGVuTWFuYWdlZDsKCiAgICAvKiBUZW1wb3JhcmlseSBzZXQgdmFsdWUgb2YgWG1ObWFwcGVkV2hlbk1hbmFnZWQKICAgICAgIHRvIHN0b3AgdGhlIGRpYWxvZyBmcm9tIHBvcHBpbmcgdXAgcmlnaHQgYXdheSAqLwogICAgWHRWYUdldFZhbHVlcyhzaGVsbCwgWG1ObWFwcGVkV2hlbk1hbmFnZWQsICZtYXBwZWRXaGVuTWFuYWdlZCwgTlVMTCk7CiAgICBYdFZhU2V0VmFsdWVzKHNoZWxsLCBYbU5tYXBwZWRXaGVuTWFuYWdlZCwgRmFsc2UsIE5VTEwpOwoKICAgIFh0TWFuYWdlQ2hpbGQoZGlhbG9nX2NoaWxkKTsKCiAgICAvKiBHZXQgdGhlIHBvaW50ZXIgcG9zaXRpb24gKHgsIHkpICovCiAgICBYUXVlcnlQb2ludGVyKFh0RGlzcGxheShzaGVsbCksIFh0V2luZG93KHNoZWxsKSwgJnJvb3QsICZjaGlsZCwKCQkgICZ4LCAmeSwgJndpbl94LCAmd2luX3ksICZtYXNrKTsKCiAgICAvKiBUcmFuc2xhdGUgdGhlIHBvaW50ZXIgcG9zaXRpb24gKHgsIHkpIGludG8gYSBwb3NpdGlvbiBmb3IgdGhlIG5ldwogICAgICAgd2luZG93IHRoYXQgd2lsbCBwbGFjZSB0aGUgcG9pbnRlciBhdCBpdHMgY2VudGVyICovCiAgICBYR2V0R2VvbWV0cnkoWHREaXNwbGF5KHNoZWxsKSwgWHRXaW5kb3coc2hlbGwpLCAmcm9vdCwgJndpbl94LCAmd2luX3ksCgkJICZ3aWR0aCwgJmhlaWdodCwgJmJvcmRlcl93aWR0aCwgJmRlcHRoKTsKICAgIHdpZHRoICs9IDIgKiBib3JkZXJfd2lkdGg7CiAgICBoZWlnaHQgKz0gMiAqIGJvcmRlcl93aWR0aDsKICAgIHggLT0gd2lkdGggLyAyOwogICAgeSAtPSBoZWlnaHQgLyAyOwoKICAgIC8qIEVuc3VyZSB0aGF0IHRoZSBkaWFsb2cgcmVtYWlucyBvbiBzY3JlZW4gKi8KICAgIG1heFggPSBYdFNjcmVlbihzaGVsbCktPndpZHRoIC0gd2lkdGg7CiAgICBtYXhZID0gWHRTY3JlZW4oc2hlbGwpLT5oZWlnaHQgLSBoZWlnaHQ7CiAgICBpZiAoeCA8IDApCgl4ID0gMDsKICAgIGlmICh4ID4gbWF4WCkKCXggPSBtYXhYOwogICAgaWYgKHkgPCAwKQoJeSA9IDA7CiAgICBpZiAoeSA+IG1heFkpCgl5ID0gbWF4WTsKCiAgICAvKiBTZXQgZGVzaXJlZCB3aW5kb3cgcG9zaXRpb24gaW4gdGhlIERpYWxvZ1NoZWxsICovCiAgICBYdFZhU2V0VmFsdWVzKHNoZWxsLCBYbU54LCB4LCBYbU55LCB5LCBOVUxMKTsKCiAgICAvKiBNYXAgdGhlIHdpZGdldCAqLwogICAgWHRNYXBXaWRnZXQoc2hlbGwpOwoKICAgIC8qIFJlc3RvcmUgdGhlIHZhbHVlIG9mIFhtTm1hcHBlZFdoZW5NYW5hZ2VkICovCiAgICBYdFZhU2V0VmFsdWVzKHNoZWxsLCBYbU5tYXBwZWRXaGVuTWFuYWdlZCwgbWFwcGVkV2hlbk1hbmFnZWQsIE5VTEwpOwp9CgojaWYgZGVmaW5lZChGRUFUX01FTlUpIHx8IGRlZmluZWQoRkVBVF9TVU5fV09SS1NIT1ApIFwKCXx8IGRlZmluZWQoRkVBVF9HVUlfRElBTE9HKSB8fCBkZWZpbmVkKFBST1RPKQoKLyoKICogRW5jYXBzdWxhdGUgdGhlIHdheSBhbiBYbUZvbnRMaXN0IGlzIGNyZWF0ZWQuCiAqLwogICAgWG1Gb250TGlzdApndWlfbW90aWZfY3JlYXRlX2ZvbnRsaXN0KGZvbnQpCiAgICBYRm9udFN0cnVjdCAgICAqZm9udDsKewogICAgWG1Gb250TGlzdCBmb250X2xpc3Q7CgojIGlmIChYbVZlcnNpb24gPD0gMTAwMSkKICAgIC8qIE1vdGlmIDEuMSBtZXRob2QgKi8KICAgIGZvbnRfbGlzdCA9IFhtRm9udExpc3RDcmVhdGUoZm9udCwgU1RSSU5HX1RBRyk7CiMgZWxzZQogICAgLyogTW90aWYgMS4yIG1ldGhvZCAqLwogICAgWG1Gb250TGlzdEVudHJ5IGZvbnRfbGlzdF9lbnRyeTsKCiAgICBmb250X2xpc3RfZW50cnkgPSBYbUZvbnRMaXN0RW50cnlDcmVhdGUoU1RSSU5HX1RBRywgWG1GT05UX0lTX0ZPTlQsCgkJCQkJICAgIChYdFBvaW50ZXIpZm9udCk7CiAgICBmb250X2xpc3QgPSBYbUZvbnRMaXN0QXBwZW5kRW50cnkoTlVMTCwgZm9udF9saXN0X2VudHJ5KTsKICAgIFhtRm9udExpc3RFbnRyeUZyZWUoJmZvbnRfbGlzdF9lbnRyeSk7CiMgZW5kaWYKICAgIHJldHVybiBmb250X2xpc3Q7Cn0KCiMgaWYgKChYbVZlcnNpb24gPiAxMDAxKSAmJiBkZWZpbmVkKEZFQVRfWEZPTlRTRVQpKSB8fCBkZWZpbmVkKFBST1RPKQogICAgWG1Gb250TGlzdApndWlfbW90aWZfZm9udHNldDJmb250bGlzdChmb250c2V0KQogICAgWEZvbnRTZXQJKmZvbnRzZXQ7CnsKICAgIFhtRm9udExpc3QgZm9udF9saXN0OwoKICAgIC8qIE1vdGlmIDEuMiBtZXRob2QgKi8KICAgIFhtRm9udExpc3RFbnRyeSBmb250X2xpc3RfZW50cnk7CgogICAgZm9udF9saXN0X2VudHJ5ID0gWG1Gb250TGlzdEVudHJ5Q3JlYXRlKFNUUklOR19UQUcsCgkJCQkJICAgIFhtRk9OVF9JU19GT05UU0VULAoJCQkJCSAgICAoWHRQb2ludGVyKSpmb250c2V0KTsKICAgIGZvbnRfbGlzdCA9IFhtRm9udExpc3RBcHBlbmRFbnRyeShOVUxMLCBmb250X2xpc3RfZW50cnkpOwogICAgWG1Gb250TGlzdEVudHJ5RnJlZSgmZm9udF9saXN0X2VudHJ5KTsKICAgIHJldHVybiBmb250X2xpc3Q7Cn0KIyBlbmRpZgoKI2VuZGlmCgojaWYgZGVmaW5lZChGRUFUX01FTlUpIHx8IGRlZmluZWQoUFJPVE8pCi8qCiAqIE1lbnUgc3R1ZmYuCiAqLwoKc3RhdGljIHZvaWQgZ3VpX21vdGlmX2FkZF9hY3RleHQgX19BUkdTKCh2aW1tZW51X1QgKm1lbnUpKTsKI2lmIChYbVZlcnNpb24gPj0gMTAwMikKc3RhdGljIHZvaWQgdG9nZ2xlX3RlYXJvZmYgX19BUkdTKChXaWRnZXQgd2lkKSk7CnN0YXRpYyB2b2lkIGd1aV9tY2hfcmVjdXJzZV90ZWFyb2ZmcyBfX0FSR1MoKHZpbW1lbnVfVCAqbWVudSkpOwojZW5kaWYKc3RhdGljIHZvaWQgc3VibWVudV9jaGFuZ2UgX19BUkdTKCh2aW1tZW51X1QgKm1wLCBpbnQgY29sb3JzKSk7CgpzdGF0aWMgdm9pZCBkb19zZXRfbW5lbW9uaWNzIF9fQVJHUygoaW50IGVuYWJsZSkpOwpzdGF0aWMgaW50IG1lbnVfZW5hYmxlZCA9IFRSVUU7CgogICAgdm9pZApndWlfbWNoX2VuYWJsZV9tZW51KGZsYWcpCiAgICBpbnQJICAgIGZsYWc7CnsKICAgIGlmIChmbGFnKQogICAgewoJWHRNYW5hZ2VDaGlsZChtZW51QmFyKTsKI2lmZGVmIEZFQVRfVE9PTEJBUgoJaWYgKFh0SXNNYW5hZ2VkKFh0UGFyZW50KHRvb2xCYXIpKSkKCXsKCSAgICAvKiB0b29sQmFyIGlzIGF0dGFjaGVkIHRvIHRvcCBmb3JtICovCgkgICAgWHRWYVNldFZhbHVlcyhYdFBhcmVudCh0b29sQmFyKSwKCQlYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJWG1OdG9wV2lkZ2V0LCBtZW51QmFyLAoJCU5VTEwpOwojaWZkZWYgRkVBVF9HVUlfVEFCTElORQoJICAgIGlmIChzaG93aW5nX3RhYmxpbmUpCgkgICAgewoJCVh0VmFTZXRWYWx1ZXModGFiTGluZSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIFh0UGFyZW50KHRvb2xCYXIpLAoJCQkgICAgICBOVUxMKTsKCQlYdFZhU2V0VmFsdWVzKHRleHRBcmVhRm9ybSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIHRhYkxpbmUsCgkJCSAgICAgIE5VTEwpOwoJICAgIH0KCSAgICBlbHNlCiNlbmRpZgoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICAgIFhtTnRvcFdpZGdldCwgWHRQYXJlbnQodG9vbEJhciksCgkJCSAgICAgIE5VTEwpOwoJfQoJZWxzZQojZW5kaWYKCXsKI2lmZGVmIEZFQVRfR1VJX1RBQkxJTkUKCSAgICBpZiAoc2hvd2luZ190YWJsaW5lKQoJICAgIHsKCQlYdFZhU2V0VmFsdWVzKHRhYkxpbmUsCgkJCSAgICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQkJICAgICAgWG1OdG9wV2lkZ2V0LCBtZW51QmFyLAoJCQkgICAgICBOVUxMKTsKCQlYdFZhU2V0VmFsdWVzKHRleHRBcmVhRm9ybSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIHRhYkxpbmUsCgkJCSAgICAgIE5VTEwpOwoJICAgIH0KCSAgICBlbHNlCiNlbmRpZgoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICAgIFhtTnRvcFdpZGdldCwgbWVudUJhciwKCQkJICAgICAgTlVMTCk7Cgl9CiAgICB9CiAgICBlbHNlCiAgICB7CglYdFVubWFuYWdlQ2hpbGQobWVudUJhcik7CiNpZmRlZiBGRUFUX1RPT0xCQVIKCWlmIChYdElzTWFuYWdlZChYdFBhcmVudCh0b29sQmFyKSkpCgl7CgkgICAgWHRWYVNldFZhbHVlcyhYdFBhcmVudCh0b29sQmFyKSwKCQlYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCU5VTEwpOwojaWZkZWYgRkVBVF9HVUlfVEFCTElORQoJICAgIGlmIChzaG93aW5nX3RhYmxpbmUpCgkgICAgewoJCVh0VmFTZXRWYWx1ZXModGFiTGluZSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIFh0UGFyZW50KHRvb2xCYXIpLAoJCQkgICAgICBOVUxMKTsKCQlYdFZhU2V0VmFsdWVzKHRleHRBcmVhRm9ybSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIHRhYkxpbmUsCgkJCSAgICAgIE5VTEwpOwoJICAgIH0KCSAgICBlbHNlCiNlbmRpZgoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICAgIFhtTnRvcFdpZGdldCwgWHRQYXJlbnQodG9vbEJhciksCgkJCSAgICAgIE5VTEwpOwoJfQoJZWxzZQojZW5kaWYKCXsKI2lmZGVmIEZFQVRfR1VJX1RBQkxJTkUKCSAgICBpZiAoc2hvd2luZ190YWJsaW5lKQoJICAgIHsKCQlYdFZhU2V0VmFsdWVzKHRhYkxpbmUsCgkJCSAgICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJCSAgICAgIE5VTEwpOwoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICAgIFhtTnRvcFdpZGdldCwgdGFiTGluZSwKCQkJICAgICAgTlVMTCk7CgkgICAgfQoJICAgIGVsc2UKI2VuZGlmCgkJWHRWYVNldFZhbHVlcyh0ZXh0QXJlYUZvcm0sCgkJCSAgICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJCSAgICAgIE5VTEwpOwoJfQogICAgfQoKfQoKLyoKICogRW5hYmxlIG9yIGRpc2FibGUgbW5lbW9uaWNzIGZvciB0aGUgdG9wbGV2ZWwgbWVudXMuCiAqLwogICAgdm9pZApndWlfbW90aWZfc2V0X21uZW1vbmljcyhlbmFibGUpCiAgICBpbnQJCWVuYWJsZTsKewogICAgLyoKICAgICAqIERvbid0IGVuYWJsZSBtZW51IG1uZW1vbmljcyB3aGVuIHRoZSBtZW51IGJhciBpcyBkaXNhYmxlZCwgTGVzc1RpZgogICAgICogY3Jhc2hlcyB3aGVuIHVzaW5nIGEgbW5lbW9uaWMgdGhlbi4KICAgICAqLwogICAgaWYgKCFtZW51X2VuYWJsZWQpCgllbmFibGUgPSBGQUxTRTsKICAgIGRvX3NldF9tbmVtb25pY3MoZW5hYmxlKTsKfQoKICAgIHN0YXRpYyB2b2lkCmRvX3NldF9tbmVtb25pY3MoZW5hYmxlKQogICAgaW50CQllbmFibGU7CnsKICAgIHZpbW1lbnVfVAkqbWVudTsKCiAgICBmb3IgKG1lbnUgPSByb290X21lbnU7IG1lbnUgIT0gTlVMTDsgbWVudSA9IG1lbnUtPm5leHQpCglpZiAobWVudS0+aWQgIT0gKFdpZGdldCkwKQoJICAgIFh0VmFTZXRWYWx1ZXMobWVudS0+aWQsCgkJICAgIFhtTm1uZW1vbmljLCBlbmFibGUgPyBtZW51LT5tbmVtb25pYyA6IE5VTCwKCQkgICAgTlVMTCk7Cn0KCiAgICB2b2lkCmd1aV9tY2hfYWRkX21lbnUobWVudSwgaWR4KQogICAgdmltbWVudV9UCSptZW51OwogICAgaW50CQlpZHg7CnsKICAgIFhtU3RyaW5nCWxhYmVsOwogICAgV2lkZ2V0CXNoZWxsOwogICAgdmltbWVudV9UCSpwYXJlbnQgPSBtZW51LT5wYXJlbnQ7CgojaWZkZWYgTU9USUZfUE9QVVAKICAgIGlmIChtZW51X2lzX3BvcHVwKG1lbnUtPm5hbWUpKQogICAgewoJQXJnIGFyZ1syXTsKCWludCBuID0gMDsKCgkvKiBPbmx5IGNyZWF0ZSB0aGUgcG9wdXAgbWVudSB3aGVuIGl0J3MgYWN0dWFsbHkgdXNlZCwgb3RoZXJ3aXNlIHRoZXJlCgkgKiBpcyBhIGRlbGF5IHdoZW4gdXNpbmcgdGhlIHJpZ2h0IG1vdXNlIGJ1dHRvbi4gKi8KIyBpZiAoWG1WZXJzaW9uIDw9IDEwMDIpCglpZiAobW91c2VfbW9kZWxfcG9wdXAoKSkKIyBlbmRpZgoJewoJICAgIGlmIChndWkubWVudV9iZ19waXhlbCAhPSBJTlZBTENPTE9SKQoJICAgIHsKCQlYdFNldEFyZyhhcmdbMF0sIFhtTmJhY2tncm91bmQsIGd1aS5tZW51X2JnX3BpeGVsKTsgbisrOwoJICAgIH0KCSAgICBpZiAoZ3VpLm1lbnVfZmdfcGl4ZWwgIT0gSU5WQUxDT0xPUikKCSAgICB7CgkJWHRTZXRBcmcoYXJnWzFdLCBYbU5mb3JlZ3JvdW5kLCBndWkubWVudV9mZ19waXhlbCk7IG4rKzsKCSAgICB9CgkgICAgbWVudS0+c3VibWVudV9pZCA9IFhtQ3JlYXRlUG9wdXBNZW51KHRleHRBcmVhLCAiY29udGV4dE1lbnUiLAoJCQkJCQkJCSAgICAgIGFyZywgbik7CgkgICAgbWVudS0+aWQgPSAoV2lkZ2V0KTA7Cgl9CglyZXR1cm47CiAgICB9CiNlbmRpZgoKICAgIGlmICghbWVudV9pc19tZW51YmFyKG1lbnUtPm5hbWUpCgkgICAgfHwgKHBhcmVudCAhPSBOVUxMICYmIHBhcmVudC0+c3VibWVudV9pZCA9PSAoV2lkZ2V0KTApKQoJcmV0dXJuOwoKICAgIGxhYmVsID0gWG1TdHJpbmdDcmVhdGUoKGNoYXIgKiltZW51LT5kbmFtZSwgU1RSSU5HX1RBRyk7CiAgICBpZiAobGFiZWwgPT0gTlVMTCkKCXJldHVybjsKICAgIG1lbnUtPmlkID0gWHRWYUNyZWF0ZVdpZGdldCgic3ViTWVudSIsCgkgICAgeG1DYXNjYWRlQnV0dG9uV2lkZ2V0Q2xhc3MsCgkgICAgKHBhcmVudCA9PSBOVUxMKSA/IG1lbnVCYXIgOiBwYXJlbnQtPnN1Ym1lbnVfaWQsCgkgICAgWG1ObGFiZWxTdHJpbmcsIGxhYmVsLAoJICAgIFhtTm1uZW1vbmljLCBwX3dha1swXSA9PSAnbicgPyBOVUwgOiBtZW51LT5tbmVtb25pYywKI2lmIChYbVZlcnNpb24gPj0gMTAwMikKCSAgICAvKiBzdWJtZW51OiBjb3VudCB0aGUgdGVhcm9mZiBpdGVtIChuZWVkZWQgZm9yIExlc3NUaWYpICovCgkgICAgWG1OcG9zaXRpb25JbmRleCwgaWR4ICsgKHBhcmVudCAhPSBOVUxMCgkJCSAgICYmIHRlYXJvZmZfdmFsID09IChpbnQpWG1URUFSX09GRl9FTkFCTEVEID8gMSA6IDApLAojZW5kaWYKCSAgICBOVUxMKTsKICAgIGd1aV9tb3RpZl9tZW51X2NvbG9ycyhtZW51LT5pZCk7CiAgICBndWlfbW90aWZfbWVudV9mb250bGlzdChtZW51LT5pZCk7CiAgICBYbVN0cmluZ0ZyZWUobGFiZWwpOwoKICAgIGlmIChtZW51LT5pZCA9PSAoV2lkZ2V0KTApCQkvKiBmYWlsZWQgKi8KCXJldHVybjsKCiAgICAvKiBhZGQgYWNjZWxlcmF0b3IgdGV4dCAqLwogICAgZ3VpX21vdGlmX2FkZF9hY3RleHQobWVudSk7CgogICAgc2hlbGwgPSBYdFZhQ3JlYXRlV2lkZ2V0KCJzdWJNZW51U2hlbGwiLAoJeG1NZW51U2hlbGxXaWRnZXRDbGFzcywgbWVudS0+aWQsCglYbU53aWR0aCwgMSwKCVhtTmhlaWdodCwgMSwKCU5VTEwpOwogICAgZ3VpX21vdGlmX21lbnVfY29sb3JzKHNoZWxsKTsKICAgIG1lbnUtPnN1Ym1lbnVfaWQgPSBYdFZhQ3JlYXRlV2lkZ2V0KCJyb3dDb2x1bW5NZW51IiwKCXhtUm93Q29sdW1uV2lkZ2V0Q2xhc3MsIHNoZWxsLAoJWG1Ocm93Q29sdW1uVHlwZSwgWG1NRU5VX1BVTExET1dOLAoJTlVMTCk7CiAgICBndWlfbW90aWZfbWVudV9jb2xvcnMobWVudS0+c3VibWVudV9pZCk7CgogICAgaWYgKG1lbnUtPnN1Ym1lbnVfaWQgPT0gKFdpZGdldCkwKQkJLyogZmFpbGVkICovCglyZXR1cm47CgojaWYgKFhtVmVyc2lvbiA+PSAxMDAyKQogICAgLyogU2V0IHRoZSBjb2xvcnMgZm9yIHRoZSB0ZWFyIG9mZiB3aWRnZXQgKi8KICAgIHRvZ2dsZV90ZWFyb2ZmKG1lbnUtPnN1Ym1lbnVfaWQpOwojZW5kaWYKCiAgICBYdFZhU2V0VmFsdWVzKG1lbnUtPmlkLAoJWG1Oc3ViTWVudUlkLCBtZW51LT5zdWJtZW51X2lkLAoJTlVMTCk7CgogICAgLyoKICAgICAqIFRoZSAiSGVscCIgbWVudSBpcyBhIHNwZWNpYWwgY2FzZSwgYW5kIHNob3VsZCBiZSBwbGFjZWQgYXQgdGhlIGZhcgogICAgICogcmlnaHQgaGFuZCBzaWRlIG9mIHRoZSBtZW51LWJhci4gIEl0J3MgcmVjb2duaXplZCBieSBpdHMgaGlnaCBwcmlvcml0eS4KICAgICAqLwogICAgaWYgKHBhcmVudCA9PSBOVUxMICYmIG1lbnUtPnByaW9yaXR5ID49IDk5OTkpCglYdFZhU2V0VmFsdWVzKG1lbnVCYXIsCgkJWG1ObWVudUhlbHBXaWRnZXQsIG1lbnUtPmlkLAoJCU5VTEwpOwoKICAgIC8qCiAgICAgKiBXaGVuIHdlIGFkZCBhIHRvcC1sZXZlbCBpdGVtIHRvIHRoZSBtZW51IGJhciwgd2UgY2FuIGZpZ3VyZSBvdXQgaG93CiAgICAgKiBoaWdoIHRoZSBtZW51IGJhciBzaG91bGQgYmUuCiAgICAgKi8KICAgIGlmIChwYXJlbnQgPT0gTlVMTCkKCWd1aV9tY2hfY29tcHV0ZV9tZW51X2hlaWdodChtZW51LT5pZCk7Cn0KCgovKgogKiBBZGQgbW5lbW9uaWMgYW5kIGFjY2VsZXJhdG9yIHRleHQgdG8gYSBtZW51IGJ1dHRvbi4KICovCiAgICBzdGF0aWMgdm9pZApndWlfbW90aWZfYWRkX2FjdGV4dChtZW51KQogICAgdmltbWVudV9UCSptZW51Owp7CiAgICBYbVN0cmluZwlsYWJlbDsKCiAgICAvKiBBZGQgYWNjZWxlcmF0b3IgdGV4dCwgaWYgdGhlcmUgaXMgb25lICovCiAgICBpZiAobWVudS0+YWN0ZXh0ICE9IE5VTEwgJiYgbWVudS0+aWQgIT0gKFdpZGdldCkwKQogICAgewoJbGFiZWwgPSBYbVN0cmluZ0NyZWF0ZSgoY2hhciAqKW1lbnUtPmFjdGV4dCwgU1RSSU5HX1RBRyk7CglpZiAobGFiZWwgPT0gTlVMTCkKCSAgICByZXR1cm47CglYdFZhU2V0VmFsdWVzKG1lbnUtPmlkLCBYbU5hY2NlbGVyYXRvclRleHQsIGxhYmVsLCBOVUxMKTsKCVhtU3RyaW5nRnJlZShsYWJlbCk7CiAgICB9Cn0KCiAgICB2b2lkCmd1aV9tY2hfdG9nZ2xlX3RlYXJvZmZzKGVuYWJsZSkKICAgIGludAkJZW5hYmxlOwp7CiNpZiAoWG1WZXJzaW9uID49IDEwMDIpCiAgICBpZiAoZW5hYmxlKQoJdGVhcm9mZl92YWwgPSAoaW50KVhtVEVBUl9PRkZfRU5BQkxFRDsKICAgIGVsc2UKCXRlYXJvZmZfdmFsID0gKGludClYbVRFQVJfT0ZGX0RJU0FCTEVEOwogICAgdG9nZ2xlX3RlYXJvZmYobWVudUJhcik7CiAgICBndWlfbWNoX3JlY3Vyc2VfdGVhcm9mZnMocm9vdF9tZW51KTsKI2VuZGlmCn0KCiNpZiAoWG1WZXJzaW9uID49IDEwMDIpCi8qCiAqIFNldCB0aGUgdGVhcm9mZiBmb3Igb25lIG1lbnUgd2lkZ2V0IG9uIG9yIG9mZiwgYW5kIHNldCB0aGUgY29sb3Igb2YgdGhlCiAqIHRlYXJvZmYgd2lkZ2V0LgogKi8KICAgIHN0YXRpYyB2b2lkCnRvZ2dsZV90ZWFyb2ZmKHdpZCkKICAgIFdpZGdldAl3aWQ7CnsKICAgIFdpZGdldAl3OwoKICAgIFh0VmFTZXRWYWx1ZXMod2lkLCBYbU50ZWFyT2ZmTW9kZWwsIHRlYXJvZmZfdmFsLCBOVUxMKTsKICAgIGlmICh0ZWFyb2ZmX3ZhbCA9PSAoaW50KVhtVEVBUl9PRkZfRU5BQkxFRAoJICAgICYmICh3ID0gWG1HZXRUZWFyT2ZmQ29udHJvbCh3aWQpKSAhPSAoV2lkZ2V0KTApCglndWlfbW90aWZfbWVudV9jb2xvcnModyk7Cn0KCiAgICBzdGF0aWMgdm9pZApndWlfbWNoX3JlY3Vyc2VfdGVhcm9mZnMobWVudSkKICAgIHZpbW1lbnVfVAkqbWVudTsKewogICAgd2hpbGUgKG1lbnUgIT0gTlVMTCkKICAgIHsKCWlmICghbWVudV9pc19wb3B1cChtZW51LT5uYW1lKSkKCXsKCSAgICBpZiAobWVudS0+c3VibWVudV9pZCAhPSAoV2lkZ2V0KTApCgkJdG9nZ2xlX3RlYXJvZmYobWVudS0+c3VibWVudV9pZCk7CgkgICAgZ3VpX21jaF9yZWN1cnNlX3RlYXJvZmZzKG1lbnUtPmNoaWxkcmVuKTsKCX0KCW1lbnUgPSBtZW51LT5uZXh0OwogICAgfQp9CiNlbmRpZgoKICAgIGludApndWlfbWNoX3RleHRfYXJlYV9leHRyYV9oZWlnaHQoKQp7CiAgICBEaW1lbnNpb24Jc2hhZG93SGVpZ2h0OwoKICAgIFh0VmFHZXRWYWx1ZXModGV4dEFyZWFGb3JtLCBYbU5zaGFkb3dUaGlja25lc3MsICZzaGFkb3dIZWlnaHQsIE5VTEwpOwogICAgcmV0dXJuIHNoYWRvd0hlaWdodDsKfQoKLyoKICogQ29tcHV0ZSB0aGUgaGVpZ2h0IG9mIHRoZSBtZW51IGJhci4KICogV2UgbmVlZCB0byBjaGVjayBhbGwgdGhlIGl0ZW1zIGZvciB0aGVpciBwb3NpdGlvbiBhbmQgaGVpZ2h0LCBmb3IgdGhlIGNhc2UKICogdGhlcmUgYXJlIHNldmVyYWwgcm93cywgYW5kL29yIHNvbWUgY2hhcmFjdGVycyBleHRlbmQgaGlnaGVyIG9yIGxvd2VyLgogKi8KICAgIHZvaWQKZ3VpX21jaF9jb21wdXRlX21lbnVfaGVpZ2h0KGlkKQogICAgV2lkZ2V0CWlkOwkJICAgIC8qIGNhbiBiZSBOVUxMIHdoZW4gZGVsZXRpbmcgbWVudSAqLwp7CiAgICBEaW1lbnNpb24JeSwgbWF4eTsKICAgIERpbWVuc2lvbgltYXJnaW4sIHNoYWRvdzsKICAgIHZpbW1lbnVfVAkqbXA7CiAgICBzdGF0aWMgRGltZW5zaW9uCWhlaWdodCA9IDIxOwkvKiBub3JtYWwgaGVpZ2h0IG9mIGEgbWVudSBpdGVtICovCgogICAgLyoKICAgICAqIEdldCB0aGUgaGVpZ2h0IG9mIHRoZSBuZXcgaXRlbSwgYmVmb3JlIG1hbmFnaW5nIGl0LCBiZWNhdXNlIGl0IHdpbGwKICAgICAqIHN0aWxsIHJlZmxlY3QgdGhlIGZvbnQgc2l6ZS4gIEFmdGVyIG1hbmFnaW5nIGl0IGRlcGVuZHMgb24gdGhlIG1lbnUKICAgICAqIGhlaWdodCwgd2hpY2ggaXMgd2hhdCB3ZSBqdXN0IHdhbnRlZCB0byBnZXQhLgogICAgICovCiAgICBpZiAoaWQgIT0gKFdpZGdldCkwKQoJWHRWYUdldFZhbHVlcyhpZCwgWG1OaGVpZ2h0LCAmaGVpZ2h0LCBOVUxMKTsKCiAgICAvKiBGaW5kIGFueSBtZW51IFdpZGdldCwgdG8gYmUgYWJsZSB0byBjYWxsIFh0TWFuYWdlQ2hpbGQoKSAqLwogICAgZWxzZQoJZm9yIChtcCA9IHJvb3RfbWVudTsgbXAgIT0gTlVMTDsgbXAgPSBtcC0+bmV4dCkKCSAgICBpZiAobXAtPmlkICE9IChXaWRnZXQpMCAmJiBtZW51X2lzX21lbnViYXIobXAtPm5hbWUpKQoJICAgIHsKCQlpZCA9IG1wLT5pZDsKCQlicmVhazsKCSAgICB9CgogICAgLyoKICAgICAqIE5vdyBtYW5hZ2UgdGhlIG1lbnUgaXRlbSwgdG8gbWFrZSB0aGVtIGFsbCBiZSBwb3NpdGlvbmVkIChtYWtlcyBhbgogICAgICogZXh0cmEgcm93IHdoZW4gbmVlZGVkLCByZW1vdmVzIGl0IHdoZW4gbm90IG5lZWRlZCkuCiAgICAgKi8KICAgIGlmIChpZCAhPSAoV2lkZ2V0KTApCglYdE1hbmFnZUNoaWxkKGlkKTsKCiAgICAvKgogICAgICogTm93IGZpbmQgdGhlIG1lbnUgaXRlbSB0aGF0IGlzIHRoZSBmdXJ0aGVzdCBkb3duLCBhbmQgZ2V0IGl0J3MgcG9zaXRpb24uCiAgICAgKi8KICAgIG1heHkgPSAwOwogICAgZm9yIChtcCA9IHJvb3RfbWVudTsgbXAgIT0gTlVMTDsgbXAgPSBtcC0+bmV4dCkKICAgIHsKCWlmIChtcC0+aWQgIT0gKFdpZGdldCkwICYmIG1lbnVfaXNfbWVudWJhcihtcC0+bmFtZSkpCgl7CgkgICAgWHRWYUdldFZhbHVlcyhtcC0+aWQsIFhtTnksICZ5LCBOVUxMKTsKCSAgICBpZiAoeSA+IG1heHkpCgkJbWF4eSA9IHk7Cgl9CiAgICB9CgogICAgWHRWYUdldFZhbHVlcyhtZW51QmFyLAoJWG1ObWFyZ2luSGVpZ2h0LCAmbWFyZ2luLAoJWG1Oc2hhZG93VGhpY2tuZXNzLCAmc2hhZG93LAoJTlVMTCk7CgogICAgLyoKICAgICAqIFRoaXMgY29tcHV0YXRpb24gaXMgdGhlIHJlc3VsdCBvZiB0cmlhbC1hbmQtZXJyb3I6CiAgICAgKiBtYXh5ID0JVGhlIG1heGltdW0gcG9zaXRpb24gb2YgYW4gaXRlbTsgcmVxdWlyZWQgZm9yIHdoZW4gdGhlcmUgYXJlCiAgICAgKgkJdHdvIG9yIG1vcmUgcm93cwogICAgICogaGVpZ2h0ID0gaGVpZ2h0IG9mIGFuIGl0ZW0sIGJlZm9yZSBtYW5hZ2luZyBpdDsJSG9wZWZ1bGx5IHRoaXMgd2lsbAogICAgICoJCWNoYW5nZSB3aXRoIHRoZSBmb250IGhlaWdodC4gIEluY2x1ZGVzIHNoYWRvdy1ib3JkZXIuCiAgICAgKiBzaGFkb3cgPQlzaGFkb3ctYm9yZGVyOyBtdXN0IGJlIHN1YnRyYWN0ZWQgZnJvbSB0aGUgaGVpZ2h0LgogICAgICogbWFyZ2luID0gbWFyZ2luIGFyb3VuZCB0aGUgbWVudSBidXR0b25zOyAgTXVzdCBiZSBhZGRlZC4KICAgICAqIEFkZCA0IGZvciB0aGUgdW5kZXJsaW5pbmcgb2Ygc2hvcnRjdXQga2V5cy4KICAgICAqLwogICAgZ3VpLm1lbnVfaGVpZ2h0ID0gbWF4eSArIGhlaWdodCAtIDIgKiBzaGFkb3cgKyAyICogbWFyZ2luICsgNDsKCiAgICAvKiBTb21laG93IHRoZSBtZW51IGJhciBkb2Vzbid0IHJlc2l6ZSBhdXRvbWF0aWNhbGx5LiAgU2V0IGl0IGhlcmUsCiAgICAgKiBldmVuIHRob3VnaCB0aGlzIGlzIGEgY2F0Y2ggMjIuICBEb24ndCBkbyB0aGlzIHdoZW4gc3RhcnRpbmcgdXAsCiAgICAgKiBzb21laG93IHRoZSBtZW51IGdldHMgdmVyeSBoaWdoIHRoZW4uICovCiAgICBpZiAoZ3VpLnNoZWxsX2NyZWF0ZWQpCglYdFZhU2V0VmFsdWVzKG1lbnVCYXIsIFhtTmhlaWdodCwgZ3VpLm1lbnVfaGVpZ2h0LCBOVUxMKTsKfQoKI2lmZGVmIEZFQVRfVE9PTEJBUgoKLyoKICogSWNvbnMgdXNlZCBieSB0aGUgdG9vbGJhciBjb2RlLgogKi8KI2luY2x1ZGUgImd1aV94MTFfcG0uaCIKCnN0YXRpYyBpbnQgY2hlY2tfeHBtIF9fQVJHUygoY2hhcl91ICpwYXRoKSk7CnN0YXRpYyBjaGFyICoqZ2V0X3Rvb2xiYXJfcGl4bWFwIF9fQVJHUygodmltbWVudV9UICptZW51LCBjaGFyICoqZm5hbWUpKTsKc3RhdGljIGludCBhZGRfcGl4bWFwX2FyZ3MgX19BUkdTKCh2aW1tZW51X1QgKm1lbnUsIEFyZyAqYXJncywgaW50IG4pKTsKCi8qCiAqIFJlYWQgYW4gWHBtIGZpbGUuICBSZXR1cm4gT0sgb3IgRkFJTC4KICovCiAgICBzdGF0aWMgaW50CmNoZWNrX3hwbShwYXRoKQogICAgY2hhcl91CSpwYXRoOwp7CiAgICBYcG1BdHRyaWJ1dGVzIGF0dHJzOwogICAgaW50CQlzdGF0dXM7CiAgICBQaXhtYXAJbWFzazsKICAgIFBpeG1hcAltYXA7CgogICAgYXR0cnMudmFsdWVtYXNrID0gMDsKCiAgICAvKiBDcmVhdGUgdGhlICJzZW5zaXRpdmUiIHBpeG1hcCAqLwogICAgc3RhdHVzID0gWHBtUmVhZEZpbGVUb1BpeG1hcChndWkuZHB5LAoJICAgIFJvb3RXaW5kb3coZ3VpLmRweSwgRGVmYXVsdFNjcmVlbihndWkuZHB5KSksCgkgICAgKGNoYXIgKilwYXRoLCAmbWFwLCAmbWFzaywgJmF0dHJzKTsKICAgIFhwbUZyZWVBdHRyaWJ1dGVzKCZhdHRycyk7CgogICAgaWYgKHN0YXR1cyA9PSBYcG1TdWNjZXNzKQoJcmV0dXJuIE9LOwogICAgcmV0dXJuIEZBSUw7Cn0KCgovKgogKiBBbGxvY2F0ZWQgYSBwaXhtYXAgZm9yIHRvb2xiYXIgbWVudSAibWVudSIuCiAqIFdoZW4gaXQncyB0byBiZSByZWFkIGZyb20gYSBmaWxlLCAiZm5hbWUiIGlzIHNldCB0byB0aGUgZmlsZSBuYW1lCiAqIChpbiBhbGxvY2F0ZWQgbWVtb3J5KS4KICogUmV0dXJuIGEgYmxhbmsgcGl4bWFwIGlmIGl0IGZhaWxzLgogKi8KICAgIHN0YXRpYyBjaGFyICoqCmdldF90b29sYmFyX3BpeG1hcChtZW51LCBmbmFtZSkKICAgIHZpbW1lbnVfVAkqbWVudTsKICAgIGNoYXIJKipmbmFtZTsKewogICAgY2hhcl91CWJ1ZltNQVhQQVRITF07CQkvKiBidWZmZXIgc3RvcmluZyBleHBhbmRlZCBwYXRobmFtZSAqLwogICAgY2hhcgkqKnhwbSA9IE5VTEw7CQkvKiB4cG0gYXJyYXkgKi8KICAgIGludAkJcmVzOwoKICAgICpmbmFtZSA9IE5VTEw7CiAgICBidWZbMF0gPSBOVUw7CQkJLyogc3RhcnQgd2l0aCBOVUxMIHBhdGggKi8KCiAgICBpZiAobWVudS0+aWNvbmZpbGUgIT0gTlVMTCkKICAgIHsKCS8qIFVzZSB0aGUgImljb249IiAgYXJndW1lbnQuICovCglndWlfZmluZF9pY29uZmlsZShtZW51LT5pY29uZmlsZSwgYnVmLCAieHBtIik7CglyZXMgPSBjaGVja194cG0oYnVmKTsKCgkvKiBJZiBpdCBmYWlsZWQsIHRyeSB1c2luZyB0aGUgbWVudSBuYW1lLiAqLwoJaWYgKHJlcyA9PSBGQUlMICYmIGd1aV9maW5kX2JpdG1hcChtZW51LT5uYW1lLCBidWYsICJ4cG0iKSA9PSBPSykKCSAgICByZXMgPSBjaGVja194cG0oYnVmKTsKCWlmIChyZXMgPT0gT0spCgl7CgkgICAgKmZuYW1lID0gKGNoYXIgKil2aW1fc3Ryc2F2ZShidWYpOwoJICAgIHJldHVybiB0Yl9ibGFua194cG07Cgl9CiAgICB9CgogICAgaWYgKG1lbnUtPmljb25fYnVpbHRpbiB8fCBndWlfZmluZF9iaXRtYXAobWVudS0+bmFtZSwgYnVmLCAieHBtIikgPT0gRkFJTCkKICAgIHsKCWlmIChtZW51LT5pY29uaWR4ID49IDAgJiYgbWVudS0+aWNvbmlkeAoJICAgICAgIDwgKGludCkoc2l6ZW9mKGJ1aWx0X2luX3BpeG1hcHMpIC8gc2l6ZW9mKGJ1aWx0X2luX3BpeG1hcHNbMF0pKSkKCSAgICB4cG0gPSBidWlsdF9pbl9waXhtYXBzW21lbnUtPmljb25pZHhdOwoJZWxzZQoJICAgIHhwbSA9IHRiX2JsYW5rX3hwbTsKICAgIH0KCiAgICByZXR1cm4geHBtOwp9CgovKgogKiBBZGQgYXJndW1lbnRzIGZvciB0aGUgdG9vbGJhciBwaXhtYXAgdG8gYSBtZW51IGl0ZW0uCiAqLwogICAgc3RhdGljIGludAphZGRfcGl4bWFwX2FyZ3MobWVudSwgYXJncywgbikKICAgIHZpbW1lbnVfVAkqbWVudTsKICAgIEFyZwkJKmFyZ3M7CiAgICBpbnQJCW47CnsKICAgIHZpbV9mcmVlKG1lbnUtPnhwbV9mbmFtZSk7CiAgICBtZW51LT54cG0gPSBnZXRfdG9vbGJhcl9waXhtYXAobWVudSwgJm1lbnUtPnhwbV9mbmFtZSk7CiAgICBpZiAobWVudS0+eHBtID09IE5VTEwpCiAgICB7CglYdFNldEFyZyhhcmdzW25dLCBYbU5sYWJlbFR5cGUsIFhtU1RSSU5HKTsgbisrOwogICAgfQogICAgZWxzZQogICAgewoJaWYgKG1lbnUtPnhwbV9mbmFtZSAhPSBOVUxMKQoJewoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnBpeG1hcEZpbGUsIG1lbnUtPnhwbV9mbmFtZSk7IG4rKzsKCX0KCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnBpeG1hcERhdGEsIG1lbnUtPnhwbSk7IG4rKzsKCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxhYmVsTG9jYXRpb24sIFhtQk9UVE9NKTsgbisrOwogICAgfQogICAgcmV0dXJuIG47Cn0KI2VuZGlmIC8qIEZFQVRfVE9PTEJBUiAqLwoKICAgIHZvaWQKZ3VpX21jaF9hZGRfbWVudV9pdGVtKG1lbnUsIGlkeCkKICAgIHZpbW1lbnVfVAkqbWVudTsKICAgIGludAkJaWR4Owp7CiAgICBYbVN0cmluZwlsYWJlbDsKICAgIHZpbW1lbnVfVAkqcGFyZW50ID0gbWVudS0+cGFyZW50OwoKIyBpZmRlZiBFQkNESUMKICAgIG1lbnUtPm1uZW1vbmljID0gMDsKIyBlbmRpZgoKIyBpZiAoWG1WZXJzaW9uIDw9IDEwMDIpCiAgICAvKiBEb24ndCBhZGQgUG9wdXAgbWVudSBpdGVtcyB3aGVuIHRoZSBwb3B1cCBtZW51IGlzbid0IHVzZWQuICovCiAgICBpZiAobWVudV9pc19jaGlsZF9vZl9wb3B1cChtZW51KSAmJiAhbW91c2VfbW9kZWxfcG9wdXAoKSkKCXJldHVybjsKIyBlbmRpZgoKIyBpZmRlZiBGRUFUX1RPT0xCQVIKICAgIGlmIChtZW51X2lzX3Rvb2xiYXIocGFyZW50LT5uYW1lKSkKICAgIHsKCVdpZGdldENsYXNzCXR5cGU7CglYbVN0cmluZwl4bXMgPSBOVUxMOyAgICAvKiBmYWxsYmFjayBsYWJlbCBpZiBwaXhtYXAgbm90IGZvdW5kICovCglpbnQJCW47CglBcmcJCWFyZ3NbMThdOwoKCW4gPSAwOwoJaWYgKG1lbnVfaXNfc2VwYXJhdG9yKG1lbnUtPm5hbWUpKQoJewoJICAgIGNoYXIJKmNwOwoJICAgIERpbWVuc2lvbgl3aWQ7CgoJICAgIC8qCgkgICAgICogQSBzZXBhcmF0b3IgaGFzIHRoZSBmb3JtYXQgIi1zZXAlZFs6JWRdLSIuIFRoZSBvcHRpb25hbCA6JWQgaXMKCSAgICAgKiBhIHdpZHRoIHNwZWNpZmllci4gSWYgbm8gd2lkdGggaXMgc3BlY2lmaWVkIHRoZW4gd2UgY2hvb3NlIG9uZS4KCSAgICAgKi8KCSAgICBjcCA9IChjaGFyICopdmltX3N0cmNocihtZW51LT5uYW1lLCAnOicpOwoJICAgIGlmIChjcCAhPSBOVUxMKQoJCXdpZCA9IChEaW1lbnNpb24pYXRvaSgrK2NwKTsKCSAgICBlbHNlCgkJd2lkID0gNDsKCiNpZiAwCgkgICAgLyogV2UgYmV0dGVyIHVzZSBhIEZvcm1XaWRnZXQgaGVyZSwgc2luY2UgaXQncyBmYXIgbW9yZQoJICAgICAqIGZsZXhpYmxlIGluIHRlcm1zIG9mIHNpemUuICAqLwoJICAgIHR5cGUgPSB4bUZvcm1XaWRnZXRDbGFzczsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU53aWR0aCwgd2lkKTsgbisrOwojZWxzZQoJICAgIHR5cGUgPSB4bVNlcGFyYXRvcldpZGdldENsYXNzOwoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTndpZHRoLCB3aWQpOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1ObWluV2lkdGgsIHdpZCk7IG4rKzsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5vcmllbnRhdGlvbiwgWG1WRVJUSUNBTCk7IG4rKzsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5zZXBhcmF0b3JUeXBlLCBYbVNIQURPV19FVENIRURfSU4pOyBuKys7CiNlbmRpZgoJfQoJZWxzZQoJewoJICAgIC8qIFdpdGhvdXQgc2hhZG93cyBvbmUgY2FuJ3Qgc2Vuc2Ugd2hhdGV2ZXIgdGhlIGJ1dHRvbiBoYXMgYmVlbgoJICAgICAqIHByZXNzZWQgb3Igbm90ISBIb3dldmVyIHdlIHdhbid0IHRvIHNhdmUgYSBiaXQgb2Ygc3BhY2UuLi4KCSAgICAgKiBOZWVkIHRoZSBoaWdobGlnaHRUaGlja25lc3MgdG8gc2VlIHRoZSBmb2N1cy4KCSAgICAgKi8KCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5oaWdobGlnaHRUaGlja25lc3MsIDEpOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OaGlnaGxpZ2h0T25FbnRlciwgVHJ1ZSk7IG4rKzsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5tYXJnaW5XaWR0aCwgMCk7IG4rKzsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5tYXJnaW5IZWlnaHQsIDApOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OdHJhdmVyc2FsT24sIEZhbHNlKTsgbisrOwoJICAgIC8qIFNldCB0aGUgbGFiZWwgaGVyZSwgc28gdGhhdCB3ZSBjYW4gc3dpdGNoIGJldHdlZW4gaWNvbnMvdGV4dAoJICAgICAqIGJ5IGNoYW5naW5nIHRoZSBYbU5sYWJlbFR5cGUgcmVzb3VyY2UuICovCgkgICAgeG1zID0gWG1TdHJpbmdDcmVhdGUoKGNoYXIgKiltZW51LT5kbmFtZSwgU1RSSU5HX1RBRyk7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1ObGFiZWxTdHJpbmcsIHhtcyk7IG4rKzsKCgkgICAgbiA9IGFkZF9waXhtYXBfYXJncyhtZW51LCBhcmdzLCBuKTsKCgkgICAgdHlwZSA9IHhtRW5oYW5jZWRCdXR0b25XaWRnZXRDbGFzczsKCX0KCglYdFNldEFyZyhhcmdzW25dLCBYbU5wb3NpdGlvbkluZGV4LCBpZHgpOyBuKys7CglpZiAobWVudS0+aWQgPT0gTlVMTCkKCXsKCSAgICBtZW51LT5pZCA9IFh0Q3JlYXRlTWFuYWdlZFdpZGdldCgoY2hhciAqKW1lbnUtPmRuYW1lLAoJCQl0eXBlLCB0b29sQmFyLCBhcmdzLCBuKTsKCSAgICBpZiAobWVudS0+aWQgIT0gTlVMTCAmJiB0eXBlID09IHhtRW5oYW5jZWRCdXR0b25XaWRnZXRDbGFzcykKCSAgICB7CgkJWHRBZGRDYWxsYmFjayhtZW51LT5pZCwKCQkJWG1OYWN0aXZhdGVDYWxsYmFjaywgZ3VpX3gxMV9tZW51X2NiLCBtZW51KTsKIyBpZmRlZiBGRUFUX0ZPT1RFUgoJCVh0QWRkRXZlbnRIYW5kbGVyKG1lbnUtPmlkLCBFbnRlcldpbmRvd01hc2ssIEZhbHNlLAoJCQl0b29sYmFyYnV0dG9uX2VudGVyX2NiLCBtZW51KTsKCQlYdEFkZEV2ZW50SGFuZGxlcihtZW51LT5pZCwgTGVhdmVXaW5kb3dNYXNrLCBGYWxzZSwKCQkJdG9vbGJhcmJ1dHRvbl9sZWF2ZV9jYiwgbWVudSk7CiMgZW5kaWYKCSAgICB9Cgl9CgllbHNlCgkgICAgWHRTZXRWYWx1ZXMobWVudS0+aWQsIGFyZ3MsIG4pOwoJaWYgKHhtcyAhPSBOVUxMKQoJICAgIFhtU3RyaW5nRnJlZSh4bXMpOwoKIyBpZmRlZiBGRUFUX0JFVkFMCglndWlfbWNoX21lbnVfc2V0X3RpcChtZW51KTsKIyBlbmRpZgoKCW1lbnUtPnBhcmVudCA9IHBhcmVudDsKCW1lbnUtPnN1Ym1lbnVfaWQgPSBOVUxMOwoJLyogV2hlbiBhZGRpbmcgZmlyc3QgaXRlbSB0byB0b29sYmFyIGl0IG1pZ2h0IGhhdmUgdG8gYmUgZW5hYmxlZCAuKi8KCWlmICghWHRJc01hbmFnZWQoWHRQYXJlbnQodG9vbEJhcikpCgkJICAgICYmIHZpbV9zdHJjaHIocF9nbywgR09fVE9PTEJBUikgIT0gTlVMTCkKCSAgICBndWlfbWNoX3Nob3dfdG9vbGJhcihUUlVFKTsKCWd1aS50b29sYmFyX2hlaWdodCA9IGd1aV9tY2hfY29tcHV0ZV90b29sYmFyX2hlaWdodCgpOwoJcmV0dXJuOwogICAgfSAvKiB0b29sYmFyIG1lbnUgaXRlbSAqLwojIGVuZGlmCgogICAgLyogTm8gcGFyZW50LCBtdXN0IGJlIGEgbm9uLW1lbnViYXIgbWVudSAqLwogICAgaWYgKHBhcmVudC0+c3VibWVudV9pZCA9PSAoV2lkZ2V0KTApCglyZXR1cm47CgogICAgbWVudS0+c3VibWVudV9pZCA9IChXaWRnZXQpMDsKCiAgICAvKiBBZGQgbWVudSBzZXBhcmF0b3IgKi8KICAgIGlmIChtZW51X2lzX3NlcGFyYXRvcihtZW51LT5uYW1lKSkKICAgIHsKCW1lbnUtPmlkID0gWHRWYUNyZWF0ZVdpZGdldCgic3ViTWVudSIsCgkJeG1TZXBhcmF0b3JHYWRnZXRDbGFzcywgcGFyZW50LT5zdWJtZW51X2lkLAojaWYgKFhtVmVyc2lvbiA+PSAxMDAyKQoJCS8qIGNvdW50IHRoZSB0ZWFyb2ZmIGl0ZW0gKG5lZWRlZCBmb3IgTGVzc1RpZikgKi8KCQlYbU5wb3NpdGlvbkluZGV4LCBpZHggKyAodGVhcm9mZl92YWwgPT0gKGludClYbVRFQVJfT0ZGX0VOQUJMRUQKCQkJCQkJCQkgICAgID8gMSA6IDApLAojZW5kaWYKCQlOVUxMKTsKCWd1aV9tb3RpZl9tZW51X2NvbG9ycyhtZW51LT5pZCk7CglyZXR1cm47CiAgICB9CgogICAgbGFiZWwgPSBYbVN0cmluZ0NyZWF0ZSgoY2hhciAqKW1lbnUtPmRuYW1lLCBTVFJJTkdfVEFHKTsKICAgIGlmIChsYWJlbCA9PSBOVUxMKQoJcmV0dXJuOwogICAgbWVudS0+aWQgPSBYdFZhQ3JlYXRlV2lkZ2V0KCJzdWJNZW51IiwKCXhtUHVzaEJ1dHRvbldpZGdldENsYXNzLCBwYXJlbnQtPnN1Ym1lbnVfaWQsCglYbU5sYWJlbFN0cmluZywgbGFiZWwsCglYbU5tbmVtb25pYywgbWVudS0+bW5lbW9uaWMsCiNpZiAoWG1WZXJzaW9uID49IDEwMDIpCgkvKiBjb3VudCB0aGUgdGVhcm9mZiBpdGVtIChuZWVkZWQgZm9yIExlc3NUaWYpICovCglYbU5wb3NpdGlvbkluZGV4LCBpZHggKyAodGVhcm9mZl92YWwgPT0gKGludClYbVRFQVJfT0ZGX0VOQUJMRUQKCQkJCQkJCQkgICAgID8gMSA6IDApLAojZW5kaWYKCU5VTEwpOwogICAgZ3VpX21vdGlmX21lbnVfY29sb3JzKG1lbnUtPmlkKTsKICAgIGd1aV9tb3RpZl9tZW51X2ZvbnRsaXN0KG1lbnUtPmlkKTsKICAgIFhtU3RyaW5nRnJlZShsYWJlbCk7CgogICAgaWYgKG1lbnUtPmlkICE9IChXaWRnZXQpMCkKICAgIHsKCVh0QWRkQ2FsbGJhY2sobWVudS0+aWQsIFhtTmFjdGl2YXRlQ2FsbGJhY2ssIGd1aV94MTFfbWVudV9jYiwKCQkoWHRQb2ludGVyKW1lbnUpOwoJLyogYWRkIGFjY2VsZXJhdG9yIHRleHQgKi8KCWd1aV9tb3RpZl9hZGRfYWN0ZXh0KG1lbnUpOwogICAgfQp9CgojaWYgKFhtVmVyc2lvbiA8PSAxMDAyKSB8fCBkZWZpbmVkKFBST1RPKQovKgogKiBUaGlzIGZ1bmN0aW9uIHdpbGwgZGVzdHJveS9jcmVhdGUgdGhlIHBvcHVwIG1lbnVzIGR5bmFtaWNhbGx5LAogKiBhY2NvcmRpbmcgdG8gdGhlIHZhbHVlIG9mICdtb3VzZW1vZGVsJy4KICogVGhpcyB3aWxsIGZpeCB0aGUgInJpZ2h0IG1vdXNlIGJ1dHRvbiBmcmVlemUiIHRoYXQgb2NjdXJzIHdoZW4KICogdGhlcmUgZXhpc3RzIGEgcG9wdXAgbWVudSBidXQgaXQgaXNuJ3QgbWFuYWdlZC4KICovCiAgICB2b2lkCmd1aV9tb3RpZl91cGRhdGVfbW91c2Vtb2RlbChtZW51KQogICAgdmltbWVudV9UCSptZW51Owp7CiAgICBpbnQJCWlkeCA9IDA7CgogICAgLyogV2hlbiBHVUkgaGFzbid0IHN0YXJ0ZWQgdGhlIG1lbnVzIGhhdmUgbm90IGJlZW4gY3JlYXRlZC4gKi8KICAgIGlmICghZ3VpLmluX3VzZSkKICAgICAgcmV0dXJuOwoKICAgIHdoaWxlIChtZW51KQogICAgewogICAgICBpZiAobWVudS0+Y2hpbGRyZW4gIT0gTlVMTCkKICAgICAgewoJICBpZiAobWVudV9pc19wb3B1cChtZW51LT5uYW1lKSkKCSAgewoJICAgICAgaWYgKG1vdXNlX21vZGVsX3BvcHVwKCkpCgkgICAgICB7CgkJICAvKiBQb3B1cCBtZW51IHdpbGwgYmUgdXNlZC4gIENyZWF0ZSB0aGUgcG9wdXAgbWVudXMuICovCgkJICBndWlfbWNoX2FkZF9tZW51KG1lbnUsIGlkeCk7CgkJICBndWlfbW90aWZfdXBkYXRlX21vdXNlbW9kZWwobWVudS0+Y2hpbGRyZW4pOwoJICAgICAgfQoJICAgICAgZWxzZQoJICAgICAgewoJCSAgLyogUG9wdXAgbWVudSB3aWxsIG5vdCBiZSB1c2VkLiAgRGVzdHJveSB0aGUgcG9wdXAgbWVudXMuICovCgkJICBndWlfbW90aWZfdXBkYXRlX21vdXNlbW9kZWwobWVudS0+Y2hpbGRyZW4pOwoJCSAgZ3VpX21jaF9kZXN0cm95X21lbnUobWVudSk7CgkgICAgICB9CgkgIH0KICAgICAgfQogICAgICBlbHNlIGlmIChtZW51X2lzX2NoaWxkX29mX3BvcHVwKG1lbnUpKQogICAgICB7CgkgIGlmIChtb3VzZV9tb2RlbF9wb3B1cCgpKQoJICAgICAgZ3VpX21jaF9hZGRfbWVudV9pdGVtKG1lbnUsIGlkeCk7CgkgIGVsc2UKCSAgICAgIGd1aV9tY2hfZGVzdHJveV9tZW51KG1lbnUpOwogICAgICB9CiAgICAgIG1lbnUgPSBtZW51LT5uZXh0OwogICAgICArK2lkeDsKICAgIH0KfQojZW5kaWYKCiAgICB2b2lkCmd1aV9tY2hfbmV3X21lbnVfY29sb3JzKCkKewogICAgaWYgKG1lbnVCYXIgPT0gKFdpZGdldCkwKQoJcmV0dXJuOwogICAgZ3VpX21vdGlmX21lbnVfY29sb3JzKG1lbnVCYXIpOwojaWZkZWYgRkVBVF9UT09MQkFSCiAgICBndWlfbW90aWZfbWVudV9jb2xvcnModG9vbEJhckZyYW1lKTsKICAgIGd1aV9tb3RpZl9tZW51X2NvbG9ycyh0b29sQmFyKTsKI2VuZGlmCgogICAgc3VibWVudV9jaGFuZ2Uocm9vdF9tZW51LCBUUlVFKTsKfQoKICAgIHZvaWQKZ3VpX21jaF9uZXdfbWVudV9mb250KCkKewogICAgaWYgKG1lbnVCYXIgPT0gKFdpZGdldCkwKQoJcmV0dXJuOwogICAgc3VibWVudV9jaGFuZ2Uocm9vdF9tZW51LCBGQUxTRSk7CiAgICB7CglEaW1lbnNpb24gICBoZWlnaHQ7CglQb3NpdGlvbiB3LCBoOwoKCVh0VmFHZXRWYWx1ZXMobWVudUJhciwgWG1OaGVpZ2h0LCAmaGVpZ2h0LCBOVUxMKTsKCWd1aS5tZW51X2hlaWdodCA9IGhlaWdodDsKCglYdFZhR2V0VmFsdWVzKHZpbVNoZWxsLCBYdE53aWR0aCwgJncsIFh0TmhlaWdodCwgJmgsIE5VTEwpOwoJZ3VpX3Jlc2l6ZV9zaGVsbCh3LCBoCiNpZmRlZiBGRUFUX1hJTQoJCS0geGltX2dldF9zdGF0dXNfYXJlYV9oZWlnaHQoKQojZW5kaWYKCQkgICAgICk7CiAgICB9CiAgICBndWlfc2V0X3NoZWxsc2l6ZShGQUxTRSwgVFJVRSwgUkVTSVpFX1ZFUlQpOwogICAgdWlfbmV3X3NoZWxsc2l6ZSgpOwp9CgojaWYgZGVmaW5lZChGRUFUX0JFVkFMKSB8fCBkZWZpbmVkKFBST1RPKQogICAgdm9pZApndWlfbWNoX25ld190b29sdGlwX2ZvbnQoKQp7CiMgaWZkZWYgRkVBVF9UT09MQkFSCiAgICB2aW1tZW51X1QgICAqbWVudTsKCiAgICBpZiAodG9vbEJhciA9PSAoV2lkZ2V0KTApCglyZXR1cm47CgogICAgbWVudSA9IGd1aV9maW5kX21lbnUoKGNoYXJfdSAqKSJUb29sQmFyIik7CiAgICBpZiAobWVudSAhPSBOVUxMKQoJc3VibWVudV9jaGFuZ2UobWVudSwgRkFMU0UpOwojIGVuZGlmCn0KCiAgICB2b2lkCmd1aV9tY2hfbmV3X3Rvb2x0aXBfY29sb3JzKCkKewojIGlmZGVmIEZFQVRfVE9PTEJBUgogICAgdmltbWVudV9UICAgKnRvb2xiYXI7CgogICAgaWYgKHRvb2xCYXIgPT0gKFdpZGdldCkwKQoJcmV0dXJuOwoKICAgIHRvb2xiYXIgPSBndWlfZmluZF9tZW51KChjaGFyX3UgKikiVG9vbEJhciIpOwogICAgaWYgKHRvb2xiYXIgIT0gTlVMTCkKCXN1Ym1lbnVfY2hhbmdlKHRvb2xiYXIsIFRSVUUpOwojIGVuZGlmCn0KI2VuZGlmCgogICAgc3RhdGljIHZvaWQKc3VibWVudV9jaGFuZ2UobWVudSwgY29sb3JzKQogICAgdmltbWVudV9UCSptZW51OwogICAgaW50CQljb2xvcnM7CQkvKiBUUlVFIGZvciBjb2xvcnMsIEZBTFNFIGZvciBmb250ICovCnsKICAgIHZpbW1lbnVfVAkqbXA7CgogICAgZm9yIChtcCA9IG1lbnU7IG1wICE9IE5VTEw7IG1wID0gbXAtPm5leHQpCiAgICB7CglpZiAobXAtPmlkICE9IChXaWRnZXQpMCkKCXsKCSAgICBpZiAoY29sb3JzKQoJICAgIHsKCQlndWlfbW90aWZfbWVudV9jb2xvcnMobXAtPmlkKTsKI2lmZGVmIEZFQVRfVE9PTEJBUgoJCS8qIEZvciBhIHRvb2xiYXIgaXRlbTogRnJlZSB0aGUgcGl4bWFwIGFuZCBhbGxvY2F0ZSBhIG5ldyBvbmUsCgkJICogc28gdGhhdCB0aGUgYmFja2dyb3VuZCBjb2xvciBpcyByaWdodC4gKi8KCQlpZiAobXAtPnhwbSAhPSBOVUxMKQoJCXsKCQkgICAgaW50CQluID0gMDsKCQkgICAgQXJnCQlhcmdzWzE4XTsKCgkJICAgIG4gPSBhZGRfcGl4bWFwX2FyZ3MobXAsIGFyZ3MsIG4pOwoJCSAgICBYdFNldFZhbHVlcyhtcC0+aWQsIGFyZ3MsIG4pOwoJCX0KIyBpZmRlZiBGRUFUX0JFVkFMCgkJLyogSWYgd2UgaGF2ZSBhIHRvb2x0aXAsIHRoZW4gd2UgbmVlZCB0byBjaGFuZ2UgaXQncyBmb250ICovCgkJaWYgKG1wLT50aXAgIT0gTlVMTCkKCQl7CgkJICAgIEFyZyBhcmdzWzJdOwoKCQkgICAgYXJnc1swXS5uYW1lID0gWG1OYmFja2dyb3VuZDsKCQkgICAgYXJnc1swXS52YWx1ZSA9IGd1aS50b29sdGlwX2JnX3BpeGVsOwoJCSAgICBhcmdzWzFdLm5hbWUgPSBYbU5mb3JlZ3JvdW5kOwoJCSAgICBhcmdzWzFdLnZhbHVlID0gZ3VpLnRvb2x0aXBfZmdfcGl4ZWw7CgkJICAgIFh0U2V0VmFsdWVzKG1wLT50aXAtPmJhbGxvb25MYWJlbCwgJmFyZ3NbMF0sIFh0TnVtYmVyKGFyZ3MpKTsKCQl9CiMgZW5kaWYKI2VuZGlmCgkgICAgfQoJICAgIGVsc2UKCSAgICB7CgkJZ3VpX21vdGlmX21lbnVfZm9udGxpc3QobXAtPmlkKTsKI2lmZGVmIEZFQVRfQkVWQUwKCQkvKiBJZiB3ZSBoYXZlIGEgdG9vbHRpcCwgdGhlbiB3ZSBuZWVkIHRvIGNoYW5nZSBpdCdzIGZvbnQgKi8KCQlpZiAobXAtPnRpcCAhPSBOVUxMKQoJCXsKCQkgICAgQXJnIGFyZ3NbMV07CgoJCSAgICBhcmdzWzBdLm5hbWUgPSBYbU5mb250TGlzdDsKCQkgICAgYXJnc1swXS52YWx1ZSA9IChYdEFyZ1ZhbClndWlfbW90aWZfZm9udHNldDJmb250bGlzdCgKCQkJCQkJICAgICZndWkudG9vbHRpcF9mb250c2V0KTsKCQkgICAgWHRTZXRWYWx1ZXMobXAtPnRpcC0+YmFsbG9vbkxhYmVsLCAmYXJnc1swXSwgWHROdW1iZXIoYXJncykpOwoJCX0KI2VuZGlmCgkgICAgfQoJfQoKCWlmIChtcC0+Y2hpbGRyZW4gIT0gTlVMTCkKCXsKI2lmIChYbVZlcnNpb24gPj0gMTAwMikKCSAgICAvKiBTZXQgdGhlIGNvbG9ycy9mb250IGZvciB0aGUgdGVhciBvZmYgd2lkZ2V0ICovCgkgICAgaWYgKG1wLT5zdWJtZW51X2lkICE9IChXaWRnZXQpMCkKCSAgICB7CgkJaWYgKGNvbG9ycykKCQkgICAgZ3VpX21vdGlmX21lbnVfY29sb3JzKG1wLT5zdWJtZW51X2lkKTsKCQllbHNlCgkJICAgIGd1aV9tb3RpZl9tZW51X2ZvbnRsaXN0KG1wLT5zdWJtZW51X2lkKTsKCQl0b2dnbGVfdGVhcm9mZihtcC0+c3VibWVudV9pZCk7CgkgICAgfQojZW5kaWYKCSAgICAvKiBTZXQgdGhlIGNvbG9ycyBmb3IgdGhlIGNoaWxkcmVuICovCgkgICAgc3VibWVudV9jaGFuZ2UobXAtPmNoaWxkcmVuLCBjb2xvcnMpOwoJfQogICAgfQp9CgovKgogKiBEZXN0cm95IHRoZSBtYWNoaW5lIHNwZWNpZmljIG1lbnUgd2lkZ2V0LgogKi8KICAgIHZvaWQKZ3VpX21jaF9kZXN0cm95X21lbnUobWVudSkKICAgIHZpbW1lbnVfVAkqbWVudTsKewogICAgLyogUGxlYXNlIGJlIHN1cmUgdG8gZGVzdHJveSB0aGUgcGFyZW50IHdpZGdldCBmaXJzdCAoaS5lLiBtZW51LT5pZCkuCiAgICAgKiBPbiB0aGUgb3RoZXIgaGFuZCwgcHJvYmxlbXMgaGF2ZSBiZWVuIHJlcG9ydGVkIHRoYXQgdGhlIHN1Ym1lbnUgbXVzdCBiZQogICAgICogZGVsZXRlZCBmaXJzdC4uLgogICAgICoKICAgICAqIFRoaXMgY29kZSBzaG91bGQgYmUgYmFzaWNhbGx5IGlkZW50aWNhbCB0byB0aGF0IGluIHRoZSBmaWxlIGd1aV9hdGhlbmEuYwogICAgICogYmVjYXVzZSB0aGV5IGFyZSBib3RoIFh0IGJhc2VkLgogICAgICovCiAgICBpZiAobWVudS0+c3VibWVudV9pZCAhPSAoV2lkZ2V0KTApCiAgICB7CglYdERlc3Ryb3lXaWRnZXQobWVudS0+c3VibWVudV9pZCk7CgltZW51LT5zdWJtZW51X2lkID0gKFdpZGdldCkwOwogICAgfQoKICAgIGlmIChtZW51LT5pZCAhPSAoV2lkZ2V0KTApCiAgICB7CglXaWRnZXQJICAgIHBhcmVudDsKCglwYXJlbnQgPSBYdFBhcmVudChtZW51LT5pZCk7CiNpZiBkZWZpbmVkKEZFQVRfVE9PTEJBUikgJiYgZGVmaW5lZChGRUFUX0JFVkFMKQoJaWYgKHBhcmVudCA9PSB0b29sQmFyICYmIG1lbnUtPnRpcCAhPSBOVUxMKQoJewoJICAgIC8qIFdlIHRyeSB0byBkZXN0cm95IHRoaXMgYmVmb3JlIHRoZSBhY3R1YWwgbWVudSwgYmVjYXVzZSB0aGVyZSBhcmUKCSAgICAgKiBjYWxsYmFja3MsIGV0Yy4gdGhhdCB3aWxsIGJlIHVucmVnaXN0ZXJlZCBkdXJpbmcgdGhlIHRvb2x0aXAKCSAgICAgKiBkZXN0cnVjdGlvbi4KCSAgICAgKgoJICAgICAqIElmIHlvdSBjYWxsICJndWlfbWNoX2Rlc3Ryb3lfYmV2YWxfYXJlYSgpIiBhZnRlciBkZXN0cm95aW5nCgkgICAgICogbWVudS0+aWQsIHRoZW4gdGhlIHRvb2x0aXAncyB3aW5kb3cgd2lsbCBoYXZlIGFscmVhZHkgYmVlbgoJICAgICAqIGRlYWxsb2NhdGVkIGJ5IFh0LCBhbmQgdW5rbm93biBiZWhhdmlvdXIgd2lsbCBlbnN1ZSAocHJvYmFibHkKCSAgICAgKiBhIGNvcmUgZHVtcCkuCgkgICAgICovCgkgICAgZ3VpX21jaF9kZXN0cm95X2JldmFsX2FyZWEobWVudS0+dGlwKTsKCSAgICBtZW51LT50aXAgPSBOVUxMOwoJfQojZW5kaWYKCVh0RGVzdHJveVdpZGdldChtZW51LT5pZCk7CgltZW51LT5pZCA9IChXaWRnZXQpMDsKCWlmIChwYXJlbnQgPT0gbWVudUJhcikKCSAgICBndWlfbWNoX2NvbXB1dGVfbWVudV9oZWlnaHQoKFdpZGdldCkwKTsKI2lmZGVmIEZFQVRfVE9PTEJBUgoJZWxzZSBpZiAocGFyZW50ID09IHRvb2xCYXIpCgl7CgkgICAgQ2FyZGluYWwgICAgbnVtX2NoaWxkcmVuOwoKCSAgICAvKiBXaGVuIHJlbW92aW5nIGxhc3QgdG9vbGJhciBpdGVtLCBkb24ndCBkaXNwbGF5IHRoZSB0b29sYmFyLiAqLwoJICAgIFh0VmFHZXRWYWx1ZXModG9vbEJhciwgWG1ObnVtQ2hpbGRyZW4sICZudW1fY2hpbGRyZW4sIE5VTEwpOwoJICAgIGlmIChudW1fY2hpbGRyZW4gPT0gMCkKCQlndWlfbWNoX3Nob3dfdG9vbGJhcihGQUxTRSk7CgkgICAgZWxzZQoJCWd1aS50b29sYmFyX2hlaWdodCA9IGd1aV9tY2hfY29tcHV0ZV90b29sYmFyX2hlaWdodCgpOwoJfQojZW5kaWYKICAgIH0KfQoKICAgIHZvaWQKZ3VpX21jaF9zaG93X3BvcHVwbWVudShtZW51KQogICAgdmltbWVudV9UICptZW51IFVOVVNFRDsKewojaWZkZWYgTU9USUZfUE9QVVAKICAgIFhtTWVudVBvc2l0aW9uKG1lbnUtPnN1Ym1lbnVfaWQsIGd1aV94MTFfZ2V0X2xhc3RfbW91c2VfZXZlbnQoKSk7CiAgICBYdE1hbmFnZUNoaWxkKG1lbnUtPnN1Ym1lbnVfaWQpOwojZW5kaWYKfQoKI2VuZGlmIC8qIEZFQVRfTUVOVSAqLwoKLyoKICogU2V0IHRoZSBtZW51IGFuZCBzY3JvbGxiYXIgY29sb3JzIHRvIHRoZWlyIGRlZmF1bHQgdmFsdWVzLgogKi8KICAgIHZvaWQKZ3VpX21jaF9kZWZfY29sb3JzKCkKewogICAgaWYgKGd1aS5pbl91c2UpCiAgICB7CgkvKiBVc2UgdGhlIHZhbHVlcyBzYXZlZCB3aGVuIHN0YXJ0aW5nIHVwLiAgVGhlc2Ugc2hvdWxkIGNvbWUgZnJvbSB0aGUKCSAqIHdpbmRvdyBtYW5hZ2VyIG9yIGEgcmVzb3VyY2VzIGZpbGUuICovCglndWkubWVudV9mZ19waXhlbCA9IGd1aS5tZW51X2RlZl9mZ19waXhlbDsKCWd1aS5tZW51X2JnX3BpeGVsID0gZ3VpLm1lbnVfZGVmX2JnX3BpeGVsOwoJZ3VpLnNjcm9sbF9mZ19waXhlbCA9IGd1aS5zY3JvbGxfZGVmX2ZnX3BpeGVsOwoJZ3VpLnNjcm9sbF9iZ19waXhlbCA9IGd1aS5zY3JvbGxfZGVmX2JnX3BpeGVsOwojaWZkZWYgRkVBVF9CRVZBTAoJZ3VpLnRvb2x0aXBfZmdfcGl4ZWwgPQoJCQlndWlfZ2V0X2NvbG9yKChjaGFyX3UgKilndWkucnNyY190b29sdGlwX2ZnX25hbWUpOwoJZ3VpLnRvb2x0aXBfYmdfcGl4ZWwgPQoJCQlndWlfZ2V0X2NvbG9yKChjaGFyX3UgKilndWkucnNyY190b29sdGlwX2JnX25hbWUpOwojZW5kaWYKICAgIH0KfQoKCi8qCiAqIFNjcm9sbGJhciBzdHVmZi4KICovCgogICAgdm9pZApndWlfbWNoX3NldF9zY3JvbGxiYXJfdGh1bWIoc2IsIHZhbCwgc2l6ZSwgbWF4KQogICAgc2Nyb2xsYmFyX1QgKnNiOwogICAgbG9uZwl2YWw7CiAgICBsb25nCXNpemU7CiAgICBsb25nCW1heDsKewogICAgaWYgKHNiLT5pZCAhPSAoV2lkZ2V0KTApCglYdFZhU2V0VmFsdWVzKHNiLT5pZCwKCQkgIFhtTnZhbHVlLCB2YWwsCgkJICBYbU5zbGlkZXJTaXplLCBzaXplLAoJCSAgWG1OcGFnZUluY3JlbWVudCwgKHNpemUgPiAyID8gc2l6ZSAtIDIgOiAxKSwKCQkgIFhtTm1heGltdW0sIG1heCArIDEsCSAgICAvKiBNb3RpZiBoYXMgbWF4IG9uZSBwYXN0IHRoZSBlbmQgKi8KCQkgIE5VTEwpOwp9CgogICAgdm9pZApndWlfbWNoX3NldF9zY3JvbGxiYXJfcG9zKHNiLCB4LCB5LCB3LCBoKQogICAgc2Nyb2xsYmFyX1QgKnNiOwogICAgaW50CQl4OwogICAgaW50CQl5OwogICAgaW50CQl3OwogICAgaW50CQloOwp7CiAgICBpZiAoc2ItPmlkICE9IChXaWRnZXQpMCkKICAgIHsKCWlmIChzYi0+dHlwZSA9PSBTQkFSX0xFRlQgfHwgc2ItPnR5cGUgPT0gU0JBUl9SSUdIVCkKCXsKCSAgICBpZiAoeSA9PSAwKQoJCWggLT0gZ3VpLmJvcmRlcl9vZmZzZXQ7CgkgICAgZWxzZQoJCXkgLT0gZ3VpLmJvcmRlcl9vZmZzZXQ7CgkgICAgWHRWYVNldFZhbHVlcyhzYi0+aWQsCgkJCSAgWG1OdG9wT2Zmc2V0LCB5LAoJCQkgIFhtTmJvdHRvbU9mZnNldCwgLXkgLSBoLAoJCQkgIFhtTndpZHRoLCB3LAoJCQkgIE5VTEwpOwoJfQoJZWxzZQoJICAgIFh0VmFTZXRWYWx1ZXMoc2ItPmlkLAoJCQkgIFhtTnRvcE9mZnNldCwgeSwKCQkJICBYbU5sZWZ0T2Zmc2V0LCB4LAoJCQkgIFhtTnJpZ2h0T2Zmc2V0LCBndWkud2hpY2hfc2Nyb2xsYmFyc1tTQkFSX1JJR0hUXQoJCQkJCQkgICAgPyBndWkuc2Nyb2xsYmFyX3dpZHRoIDogMCwKCQkJICBYbU5oZWlnaHQsIGgsCgkJCSAgTlVMTCk7CglYdE1hbmFnZUNoaWxkKHNiLT5pZCk7CiAgICB9Cn0KCiAgICB2b2lkCmd1aV9tY2hfZW5hYmxlX3Njcm9sbGJhcihzYiwgZmxhZykKICAgIHNjcm9sbGJhcl9UICpzYjsKICAgIGludAkJZmxhZzsKewogICAgQXJnCQlhcmdzWzE2XTsKICAgIGludAkJbjsKCiAgICBpZiAoc2ItPmlkICE9IChXaWRnZXQpMCkKICAgIHsKCW4gPSAwOwoJaWYgKGZsYWcpCgl7CgkgICAgc3dpdGNoIChzYi0+dHlwZSkKCSAgICB7CgkJY2FzZSBTQkFSX0xFRlQ6CgkJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxlZnRPZmZzZXQsIGd1aS5zY3JvbGxiYXJfd2lkdGgpOyBuKys7CgkJICAgIGJyZWFrOwoKCQljYXNlIFNCQVJfUklHSFQ6CgkJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnJpZ2h0T2Zmc2V0LCBndWkuc2Nyb2xsYmFyX3dpZHRoKTsgbisrOwoJCSAgICBicmVhazsKCgkJY2FzZSBTQkFSX0JPVFRPTToKCQkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OYm90dG9tT2Zmc2V0LCBndWkuc2Nyb2xsYmFyX2hlaWdodCk7bisrOwoJCSAgICBicmVhazsKCSAgICB9CgkgICAgWHRTZXRWYWx1ZXModGV4dEFyZWEsIGFyZ3MsIG4pOwoJICAgIFh0TWFuYWdlQ2hpbGQoc2ItPmlkKTsKCX0KCWVsc2UKCXsKCSAgICBpZiAoIWd1aS53aGljaF9zY3JvbGxiYXJzW3NiLT50eXBlXSkKCSAgICB7CgkJLyogVGhlIHNjcm9sbGJhcnMgb2YgdGhpcyB0eXBlIGFyZSBhbGwgZGlzYWJsZWQsIGFkanVzdCB0aGUKCQkgKiB0ZXh0QXJlYSBhdHRhY2htZW50IG9mZnNldC4gKi8KCQlzd2l0Y2ggKHNiLT50eXBlKQoJCXsKCQkgICAgY2FzZSBTQkFSX0xFRlQ6CgkJCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxlZnRPZmZzZXQsIDApOyBuKys7CgkJCWJyZWFrOwoKCQkgICAgY2FzZSBTQkFSX1JJR0hUOgoJCQlYdFNldEFyZyhhcmdzW25dLCBYbU5yaWdodE9mZnNldCwgMCk7IG4rKzsKCQkJYnJlYWs7CgoJCSAgICBjYXNlIFNCQVJfQk9UVE9NOgoJCQlYdFNldEFyZyhhcmdzW25dLCBYbU5ib3R0b21PZmZzZXQsIDApO24rKzsKCQkJYnJlYWs7CgkJfQoJCVh0U2V0VmFsdWVzKHRleHRBcmVhLCBhcmdzLCBuKTsKCSAgICB9CgkgICAgWHRVbm1hbmFnZUNoaWxkKHNiLT5pZCk7Cgl9CiAgICB9Cn0KCiAgICB2b2lkCmd1aV9tY2hfY3JlYXRlX3Njcm9sbGJhcihzYiwgb3JpZW50KQogICAgc2Nyb2xsYmFyX1QgKnNiOwogICAgaW50CQlvcmllbnQ7CS8qIFNCQVJfVkVSVCBvciBTQkFSX0hPUklaICovCnsKICAgIEFyZwkJYXJnc1sxNl07CiAgICBpbnQJCW47CgogICAgbiA9IDA7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5taW5pbXVtLCAwKTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1Ob3JpZW50YXRpb24sCgkgICAgKG9yaWVudCA9PSBTQkFSX1ZFUlQpID8gWG1WRVJUSUNBTCA6IFhtSE9SSVpPTlRBTCk7IG4rKzsKCiAgICBzd2l0Y2ggKHNiLT50eXBlKQogICAgewoJY2FzZSBTQkFSX0xFRlQ6CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSk7IG4rKzsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9PUFBPU0lURV9GT1JNKTsgbisrOwoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwoJICAgIGJyZWFrOwoKCWNhc2UgU0JBUl9SSUdIVDoKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX09QUE9TSVRFX0ZPUk0pOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwoJICAgIGJyZWFrOwoKCWNhc2UgU0JBUl9CT1RUT006CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1ObGVmdEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0pOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0pOyBuKys7CgkgICAgYnJlYWs7CiAgICB9CgogICAgc2ItPmlkID0gWHRDcmVhdGVXaWRnZXQoInNjcm9sbEJhciIsCgkgICAgeG1TY3JvbGxCYXJXaWRnZXRDbGFzcywgdGV4dEFyZWFGb3JtLCBhcmdzLCBuKTsKCiAgICAvKiBSZW1lbWJlciB0aGUgZGVmYXVsdCBjb2xvcnMsIG5lZWRlZCBmb3IgIjpoaSBjbGVhciIuICovCiAgICBpZiAoZ3VpLnNjcm9sbF9kZWZfYmdfcGl4ZWwgPT0gKGd1aWNvbG9yX1QpMAoJICAgICYmIGd1aS5zY3JvbGxfZGVmX2ZnX3BpeGVsID09IChndWljb2xvcl9UKTApCglYdFZhR2V0VmFsdWVzKHNiLT5pZCwKCQlYbU5iYWNrZ3JvdW5kLCAmZ3VpLnNjcm9sbF9kZWZfYmdfcGl4ZWwsCgkJWG1OZm9yZWdyb3VuZCwgJmd1aS5zY3JvbGxfZGVmX2ZnX3BpeGVsLAoJCU5VTEwpOwoKICAgIGlmIChzYi0+aWQgIT0gKFdpZGdldCkwKQogICAgewoJZ3VpX21jaF9zZXRfc2Nyb2xsYmFyX2NvbG9ycyhzYik7CglYdEFkZENhbGxiYWNrKHNiLT5pZCwgWG1OdmFsdWVDaGFuZ2VkQ2FsbGJhY2ssCgkJICAgICAgc2Nyb2xsX2NiLCAoWHRQb2ludGVyKXNiLT5pZGVudCk7CglYdEFkZENhbGxiYWNrKHNiLT5pZCwgWG1OZHJhZ0NhbGxiYWNrLAoJCSAgICAgIHNjcm9sbF9jYiwgKFh0UG9pbnRlcilzYi0+aWRlbnQpOwoJWHRBZGRFdmVudEhhbmRsZXIoc2ItPmlkLCBLZXlQcmVzc01hc2ssIEZBTFNFLCBndWlfeDExX2tleV9oaXRfY2IsCgkgICAgKFh0UG9pbnRlcikwKTsKICAgIH0KfQoKI2lmIGRlZmluZWQoRkVBVF9XSU5ET1dTKSB8fCBkZWZpbmVkKFBST1RPKQogICAgdm9pZApndWlfbWNoX2Rlc3Ryb3lfc2Nyb2xsYmFyKHNiKQogICAgc2Nyb2xsYmFyX1QgKnNiOwp7CiAgICBpZiAoc2ItPmlkICE9IChXaWRnZXQpMCkKCVh0RGVzdHJveVdpZGdldChzYi0+aWQpOwp9CiNlbmRpZgoKICAgIHZvaWQKZ3VpX21jaF9zZXRfc2Nyb2xsYmFyX2NvbG9ycyhzYikKICAgIHNjcm9sbGJhcl9UICpzYjsKewogICAgaWYgKHNiLT5pZCAhPSAoV2lkZ2V0KTApCiAgICB7CglpZiAoZ3VpLnNjcm9sbF9iZ19waXhlbCAhPSBJTlZBTENPTE9SKQoJewojaWYgKFhtVmVyc2lvbj49MTAwMikKCSAgICBYbUNoYW5nZUNvbG9yKHNiLT5pZCwgZ3VpLnNjcm9sbF9iZ19waXhlbCk7CiNlbHNlCgkgICAgWHRWYVNldFZhbHVlcyhzYi0+aWQsCgkJICAgIFhtTnRyb3VnaENvbG9yLCBndWkuc2Nyb2xsX2JnX3BpeGVsLAoJCSAgICBOVUxMKTsKI2VuZGlmCgl9CgoJaWYgKGd1aS5zY3JvbGxfZmdfcGl4ZWwgIT0gSU5WQUxDT0xPUikKCSAgICBYdFZhU2V0VmFsdWVzKHNiLT5pZCwKCQkgICAgWG1OZm9yZWdyb3VuZCwgZ3VpLnNjcm9sbF9mZ19waXhlbCwKI2lmIChYbVZlcnNpb248MTAwMikKCQkgICAgWG1OYmFja2dyb3VuZCwgZ3VpLnNjcm9sbF9mZ19waXhlbCwKI2VuZGlmCgkJICAgIE5VTEwpOwogICAgfQoKICAgIC8qIFRoaXMgaXMgbmVlZGVkIGZvciB0aGUgcmVjdGFuZ2xlIGJlbG93IHRoZSB2ZXJ0aWNhbCBzY3JvbGxiYXJzLiAqLwogICAgaWYgKHNiID09ICZndWkuYm90dG9tX3NiYXIgJiYgdGV4dEFyZWFGb3JtICE9IChXaWRnZXQpMCkKCWd1aV9tb3RpZl9zY3JvbGxfY29sb3JzKHRleHRBcmVhRm9ybSk7Cn0KCi8qCiAqIE1pc2NlbGxhbmVvdXMgc3R1ZmY6CiAqLwoKICAgIFdpbmRvdwpndWlfeDExX2dldF93aWQoKQp7CiAgICByZXR1cm4oWHRXaW5kb3codGV4dEFyZWEpKTsKfQoKLyoKICogTG9vayBmb3IgYSB3aWRnZXQgaW4gdGhlIHdpZGdldCB0cmVlIHcsIHdpdGggYSBtbmVtb25pYyBtYXRjaGluZyBrZXljb2RlLgogKiBXaGVuIG9uZSBpcyBmb3VuZCwgc2ltdWxhdGUgYSBidXR0b24gcHJlc3Mgb24gdGhhdCB3aWRnZXQgYW5kIGdpdmUgaXQgdGhlCiAqIGtleWJvYXJkIGZvY3VzLiAgSWYgdGhlIG1uZW1vbmljIGlzIG9uIGEgbGFiZWwsIGxvb2sgaW4gdGhlIHVzZXJEYXRhIGZpZWxkCiAqIG9mIHRoZSBsYWJlbCB0byBzZWUgaWYgaXQgcG9pbnRzIHRvIGFub3RoZXIgd2lkZ2V0LCBhbmQgZ2l2ZSB0aGF0IHRoZSBmb2N1cy4KICovCiAgICBzdGF0aWMgdm9pZApkb19tbmVtb25pYyhXaWRnZXQgdywgdW5zaWduZWQgaW50IGtleWNvZGUpCnsKICAgIFdpZGdldExpc3QJICAgIGNoaWxkcmVuOwogICAgaW50CQkgICAgbnVtQ2hpbGRyZW4sIGk7CiAgICBCb29sZWFuCSAgICBpc01lbnU7CiAgICBLZXlTeW0JICAgIG1uZW1vbmljID0gJ1wwJzsKICAgIGNoYXIJICAgIG1uZVN0cmluZ1syXTsKICAgIFdpZGdldAkgICAgdXNlckRhdGE7CiAgICB1bnNpZ25lZCBjaGFyICAgcm93Q29sVHlwZTsKCiAgICBpZiAoWHRJc0NvbXBvc2l0ZSh3KSkKICAgIHsKCWlmIChYdENsYXNzKHcpID09IHhtUm93Q29sdW1uV2lkZ2V0Q2xhc3MpCgl7CgkgICAgWHRWYUdldFZhbHVlcyh3LCBYbU5yb3dDb2x1bW5UeXBlLCAmcm93Q29sVHlwZSwgTlVMTCk7CgkgICAgaXNNZW51ID0gKHJvd0NvbFR5cGUgIT0gKHVuc2lnbmVkIGNoYXIpWG1XT1JLX0FSRUEpOwoJfQoJZWxzZQoJICAgIGlzTWVudSA9IEZhbHNlOwoJaWYgKCFpc01lbnUpCgl7CgkgICAgWHRWYUdldFZhbHVlcyh3LCBYbU5jaGlsZHJlbiwgJmNoaWxkcmVuLCBYbU5udW1DaGlsZHJlbiwKCQkJICAmbnVtQ2hpbGRyZW4sIE5VTEwpOwoJICAgIGZvciAoaSA9IDA7IGkgPCBudW1DaGlsZHJlbjsgaSsrKQoJCWRvX21uZW1vbmljKGNoaWxkcmVuW2ldLCBrZXljb2RlKTsKCX0KICAgIH0KICAgIGVsc2UKICAgIHsKCVh0VmFHZXRWYWx1ZXModywgWG1ObW5lbW9uaWMsICZtbmVtb25pYywgTlVMTCk7CglpZiAobW5lbW9uaWMgIT0gJ1wwJykKCXsKCSAgICBtbmVTdHJpbmdbMF0gPSBtbmVtb25pYzsKCSAgICBtbmVTdHJpbmdbMV0gPSAnXDAnOwoJICAgIGlmIChYS2V5c3ltVG9LZXljb2RlKFh0RGlzcGxheShYdFBhcmVudCh3KSksCgkJCQkgICAgICAgWFN0cmluZ1RvS2V5c3ltKG1uZVN0cmluZykpID09IGtleWNvZGUpCgkgICAgewoJCWlmIChYdENsYXNzKHcpID09IHhtTGFiZWxXaWRnZXRDbGFzcwoJCQl8fCBYdENsYXNzKHcpID09IHhtTGFiZWxHYWRnZXRDbGFzcykKCQl7CgkJICAgIFh0VmFHZXRWYWx1ZXModywgWG1OdXNlckRhdGEsICZ1c2VyRGF0YSwgTlVMTCk7CgkJICAgIGlmICh1c2VyRGF0YSAhPSBOVUxMICYmIFh0SXNXaWRnZXQodXNlckRhdGEpKQoJCQlYbVByb2Nlc3NUcmF2ZXJzYWwodXNlckRhdGEsIFhtVFJBVkVSU0VfQ1VSUkVOVCk7CgkJfQoJCWVsc2UKCQl7CgkJICAgIFhLZXlQcmVzc2VkRXZlbnQga2V5RXZlbnQ7CgoJCSAgICBYbVByb2Nlc3NUcmF2ZXJzYWwodywgWG1UUkFWRVJTRV9DVVJSRU5UKTsKCgkJICAgIG1lbXNldCgoY2hhciAqKSAma2V5RXZlbnQsIDAsIHNpemVvZihYS2V5UHJlc3NlZEV2ZW50KSk7CgkJICAgIGtleUV2ZW50LnR5cGUgPSBLZXlQcmVzczsKCQkgICAga2V5RXZlbnQuc2VyaWFsID0gMTsKCQkgICAga2V5RXZlbnQuc2VuZF9ldmVudCA9IFRydWU7CgkJICAgIGtleUV2ZW50LmRpc3BsYXkgPSBYdERpc3BsYXkodyk7CgkJICAgIGtleUV2ZW50LndpbmRvdyA9IFh0V2luZG93KHcpOwoJCSAgICBYdENhbGxBY3Rpb25Qcm9jKHcsICJBY3RpdmF0ZSIsIChYRXZlbnQgKikgJiBrZXlFdmVudCwKCQkJCQkJCQkgICAgIE5VTEwsIDApOwoJCX0KCSAgICB9Cgl9CiAgICB9Cn0KCi8qCiAqIENhbGxiYWNrIHJvdXRpbmUgZm9yIGRpYWxvZyBtbmVtb25pYyBwcm9jZXNzaW5nLgogKi8KICAgIHN0YXRpYyB2b2lkCm1uZW1vbmljX2V2ZW50KFdpZGdldCB3LCBYdFBvaW50ZXIgY2FsbF9kYXRhIFVOVVNFRCwgWEtleUV2ZW50ICpldmVudCkKewogICAgZG9fbW5lbW9uaWModywgZXZlbnQtPmtleWNvZGUpOwp9CgoKLyoKICogU2VhcmNoIHRoZSB3aWRnZXQgdHJlZSB1bmRlciB3IGZvciB3aWRnZXRzIHdpdGggbW5lbW9uaWNzLiAgV2hlbiBmb3VuZCwgYWRkCiAqIGEgcGFzc2l2ZSBncmFiIHRvIHRoZSBkaWFsb2cgd2lkZ2V0IGZvciB0aGUgbW5lbW9uaWMgY2hhcmFjdGVyLCB0aHVzCiAqIGRpcmVjdGluZyBtbmVtb25pYyBldmVudHMgdG8gdGhlIGRpYWxvZyB3aWRnZXQuCiAqLwogICAgc3RhdGljIHZvaWQKYWRkX21uZW1vbmljX2dyYWJzKFdpZGdldCBkaWFsb2csIFdpZGdldCB3KQp7CiAgICBjaGFyCSAgICBtbmVTdHJpbmdbMl07CiAgICBXaWRnZXRMaXN0CSAgICBjaGlsZHJlbjsKICAgIGludAkJICAgIG51bUNoaWxkcmVuLCBpOwogICAgQm9vbGVhbgkgICAgaXNNZW51OwogICAgS2V5U3ltCSAgICBtbmVtb25pYyA9ICdcMCc7CiAgICB1bnNpZ25lZCBjaGFyICAgcm93Q29sVHlwZTsKCiAgICBpZiAoWHRJc0NvbXBvc2l0ZSh3KSkKICAgIHsKCWlmIChYdENsYXNzKHcpID09IHhtUm93Q29sdW1uV2lkZ2V0Q2xhc3MpCgl7CgkgICAgWHRWYUdldFZhbHVlcyh3LCBYbU5yb3dDb2x1bW5UeXBlLCAmcm93Q29sVHlwZSwgTlVMTCk7CgkgICAgaXNNZW51ID0gKHJvd0NvbFR5cGUgIT0gKHVuc2lnbmVkIGNoYXIpWG1XT1JLX0FSRUEpOwoJfQoJZWxzZQoJICAgIGlzTWVudSA9IEZhbHNlOwoJaWYgKCFpc01lbnUpCgl7CgkgICAgWHRWYUdldFZhbHVlcyh3LCBYbU5jaGlsZHJlbiwgJmNoaWxkcmVuLCBYbU5udW1DaGlsZHJlbiwKCQkJCQkJCSAgJm51bUNoaWxkcmVuLCBOVUxMKTsKCSAgICBmb3IgKGkgPSAwOyBpIDwgbnVtQ2hpbGRyZW47IGkrKykKCQlhZGRfbW5lbW9uaWNfZ3JhYnMoZGlhbG9nLCBjaGlsZHJlbltpXSk7Cgl9CiAgICB9CiAgICBlbHNlCiAgICB7CglYdFZhR2V0VmFsdWVzKHcsIFhtTm1uZW1vbmljLCAmbW5lbW9uaWMsIE5VTEwpOwoJaWYgKG1uZW1vbmljICE9ICdcMCcpCgl7CgkgICAgbW5lU3RyaW5nWzBdID0gbW5lbW9uaWM7CgkgICAgbW5lU3RyaW5nWzFdID0gJ1wwJzsKCSAgICBYdEdyYWJLZXkoZGlhbG9nLCBYS2V5c3ltVG9LZXljb2RlKFh0RGlzcGxheShkaWFsb2cpLAoJCQkJCQkgIFhTdHJpbmdUb0tleXN5bShtbmVTdHJpbmcpKSwKCQkgICAgTW9kMU1hc2ssIFRydWUsIEdyYWJNb2RlQXN5bmMsIEdyYWJNb2RlQXN5bmMpOwoJfQogICAgfQp9CgovKgogKiBBZGQgYSBoYW5kbGVyIGZvciBtbmVtb25pY3MgaW4gYSBkaWFsb2cuICBNb3RpZiBpdHNlbGYgb25seSBoYW5kbGVzCiAqIG1uZW1vbmljcyBpbiBtZW51cy4gTW5lbW9uaWNzIGFkZGVkIG9yIGNoYW5nZWQgYWZ0ZXIgdGhpcyBjYWxsIHdpbGwgYmUKICogaWdub3JlZC4KICoKICogVG8gYWRkIGEgbW5lbW9uaWMgdG8gYSB0ZXh0IGZpZWxkIG9yIGxpc3QsIHNldCB0aGUgWG1ObW5lbW9uaWMgcmVzb3VyY2Ugb24KICogdGhlIGFwcHJvcHJpYXRlIGxhYmVsIGFuZCBzZXQgdGhlIFhtTnVzZXJEYXRhIHJlc291cmNlIG9mIHRoZSBsYWJlbCB0byB0aGUKICogd2lkZ2V0IHRvIGdldCB0aGUgZm9jdXMgd2hlbiB0aGUgbW5lbW9uaWMgaXMgdHlwZWQuCiAqLwogICAgc3RhdGljIHZvaWQKYWN0aXZhdGVfZGlhbG9nX21uZW1vbmljcyhXaWRnZXQgZGlhbG9nKQp7CiAgICBpZiAoIWRpYWxvZykKCXJldHVybjsKCiAgICBYdEFkZEV2ZW50SGFuZGxlcihkaWFsb2csIEtleVByZXNzTWFzaywgRmFsc2UsCgkJCSAgIChYdEV2ZW50SGFuZGxlcikgbW5lbW9uaWNfZXZlbnQsIChYdFBvaW50ZXIpIE5VTEwpOwogICAgYWRkX21uZW1vbmljX2dyYWJzKGRpYWxvZywgZGlhbG9nKTsKfQoKLyoKICogUmVtb3ZlcyB0aGUgZXZlbnQgaGFuZGxlciBhbmQga2V5LWdyYWJzIGZvciBkaWFsb2cgbW5lbW9uaWMgaGFuZGxpbmcuCiAqLwogICAgc3RhdGljIHZvaWQKc3VwcHJlc3NfZGlhbG9nX21uZW1vbmljcyhXaWRnZXQgZGlhbG9nKQp7CiAgICBpZiAoIWRpYWxvZykKCXJldHVybjsKCiAgICBYdFVuZ3JhYktleShkaWFsb2csIEFueUtleSwgTW9kMU1hc2spOwogICAgWHRSZW1vdmVFdmVudEhhbmRsZXIoZGlhbG9nLCBLZXlQcmVzc01hc2ssIEZhbHNlLAoJCQkgICAoWHRFdmVudEhhbmRsZXIpIG1uZW1vbmljX2V2ZW50LCAoWHRQb2ludGVyKSBOVUxMKTsKfQoKI2lmIGRlZmluZWQoRkVBVF9CUk9XU0UpIHx8IGRlZmluZWQoRkVBVF9HVUlfRElBTE9HKQpzdGF0aWMgdm9pZCBzZXRfZm9udGxpc3QgX19BUkdTKChXaWRnZXQgd2cpKTsKCi8qCiAqIFVzZSB0aGUgJ2d1aWZvbnQnIG9yICdndWlmb250c2V0JyBhcyBhIGZvbnRsaXN0IGZvciBhIGRpYWxvZyB3aWRnZXQuCiAqLwogICAgc3RhdGljIHZvaWQKc2V0X2ZvbnRsaXN0KGlkKQogICAgV2lkZ2V0IGlkOwp7CiAgICBYbUZvbnRMaXN0IGZsOwoKI2lmZGVmIEZPTlRTRVRfQUxXQVlTCiAgICBpZiAoZ3VpLmZvbnRzZXQgIT0gTk9GT05UU0VUKQogICAgewoJZmwgPSBndWlfbW90aWZfZm9udHNldDJmb250bGlzdCgoWEZvbnRTZXQgKikmZ3VpLmZvbnRzZXQpOwoJaWYgKGZsICE9IE5VTEwpCgl7CgkgICAgaWYgKFh0SXNNYW5hZ2VkKGlkKSkKCSAgICB7CgkJWHRVbm1hbmFnZUNoaWxkKGlkKTsKCQlYdFZhU2V0VmFsdWVzKGlkLCBYbU5mb250TGlzdCwgZmwsIE5VTEwpOwoJCS8qIFdlIHNob3VsZCBmb3JjZSB0aGUgd2lkZ2V0IHRvIHJlY2FsY3VsYXRlIGl0J3MKCQkgKiBnZW9tZXRyeSBub3cuICovCgkJWHRNYW5hZ2VDaGlsZChpZCk7CgkgICAgfQoJICAgIGVsc2UKCQlYdFZhU2V0VmFsdWVzKGlkLCBYbU5mb250TGlzdCwgZmwsIE5VTEwpOwoJICAgIFhtRm9udExpc3RGcmVlKGZsKTsKCX0KICAgIH0KI2Vsc2UKICAgIGlmIChndWkubm9ybV9mb250ICE9IE5PRk9OVCkKICAgIHsKCWZsID0gZ3VpX21vdGlmX2NyZWF0ZV9mb250bGlzdCgoWEZvbnRTdHJ1Y3QgKilndWkubm9ybV9mb250KTsKCWlmIChmbCAhPSBOVUxMKQoJewoJICAgIGlmIChYdElzTWFuYWdlZChpZCkpCgkgICAgewoJCVh0VW5tYW5hZ2VDaGlsZChpZCk7CgkJWHRWYVNldFZhbHVlcyhpZCwgWG1OZm9udExpc3QsIGZsLCBOVUxMKTsKCQkvKiBXZSBzaG91bGQgZm9yY2UgdGhlIHdpZGdldCB0byByZWNhbGN1bGF0ZSBpdCdzCgkJICogZ2VvbWV0cnkgbm93LiAqLwoJCVh0TWFuYWdlQ2hpbGQoaWQpOwoJICAgIH0KCSAgICBlbHNlCgkJWHRWYVNldFZhbHVlcyhpZCwgWG1OZm9udExpc3QsIGZsLCBOVUxMKTsKCSAgICBYbUZvbnRMaXN0RnJlZShmbCk7Cgl9CiAgICB9CiNlbmRpZgp9CiNlbmRpZgoKI2lmIGRlZmluZWQoRkVBVF9CUk9XU0UpIHx8IGRlZmluZWQoUFJPVE8pCgovKgogKiBmaWxlIHNlbGVjdG9yIHJlbGF0ZWQgc3R1ZmYKICovCgojaW5jbHVkZSA8WG0vRmlsZVNCLmg+CiNpbmNsdWRlIDxYbS9YbVN0ckRlZnMuaD4KCnR5cGVkZWYgc3RydWN0IGRpYWxvZ19jYWxsYmFja19hcmcKewogICAgY2hhciAqICBhcmdzOyAgIC8qIG5vdCB1c2VkIHJpZ2h0IG5vdyAqLwogICAgaW50CSAgICBpZDsKfSBkY2JhcmdfVDsKCnN0YXRpYyBXaWRnZXQgZGlhbG9nX3dndDsKc3RhdGljIGNoYXIgKmJyb3dzZV9mbmFtZSA9IE5VTEw7CnN0YXRpYyBYbVN0cmluZ0NoYXJTZXQgY2hhcnNldCA9IChYbVN0cmluZ0NoYXJTZXQpIFhtU1RSSU5HX0RFRkFVTFRfQ0hBUlNFVDsKCQkJCS8qIHVzZWQgdG8gc2V0IHVwIFhtU3RyaW5ncyAqLwoKc3RhdGljIHZvaWQgRGlhbG9nQ2FuY2VsQ0IgX19BUkdTKChXaWRnZXQsIFh0UG9pbnRlciwgWHRQb2ludGVyKSk7CnN0YXRpYyB2b2lkIERpYWxvZ0FjY2VwdENCIF9fQVJHUygoV2lkZ2V0LCBYdFBvaW50ZXIsIFh0UG9pbnRlcikpOwoKLyoKICogVGhpcyBmdW5jdGlvbiBpcyB1c2VkIHRvIHRyYW5zbGF0ZSB0aGUgcHJlZGVmaW5lZCBsYWJlbCB0ZXh0IG9mIHRoZQogKiBwcmVjb21wb3NlZCBkaWFsb2dzLiBXZSBkbyB0aGlzIGV4cGxpY2l0bHkgdG8gYWxsb3c6CiAqCiAqIC0gdXNhZ2Ugb2YgZ2V0dGV4dCBmb3IgdHJhbnNsYXRpb24sIGFzIGluIGFsbCB0aGUgb3RoZXIgcGxhY2VzLgogKgogKiAtIGVxdWFsaXplIHRoZSBtZXNzYWdlcyBiZXR3ZWVuIGRpZmZlcmVudCBHVUkgaW1wbGVtZW50YXRpb25zIGFzIGZhciBhcwogKiBwb3NzaWJsZS4KICovCnN0YXRpYyB2b2lkIHNldF9wcmVkZWZpbmVkX2xhYmVsIF9fQVJHUygoV2lkZ2V0IHBhcmVudCwgU3RyaW5nIG5hbWUsIGNoYXIgKm5ld19sYWJlbCkpOwoKc3RhdGljIHZvaWQKc2V0X3ByZWRlZmluZWRfbGFiZWwocGFyZW50LCBuYW1lLCBuZXdfbGFiZWwpCiAgICBXaWRnZXQgIHBhcmVudDsKICAgIFN0cmluZyAgbmFtZTsKICAgIGNoYXIgICAgKm5ld19sYWJlbDsKewogICAgWG1TdHJpbmcJc3RyOwogICAgV2lkZ2V0CXc7CiAgICBjaGFyX3UJKnAsICpuZXh0OwogICAgS2V5U3ltCW1uZW1vbmljID0gTlVMOwoKICAgIHcgPSBYdE5hbWVUb1dpZGdldChwYXJlbnQsIG5hbWUpOwoKICAgIGlmICghdykKCXJldHVybjsKCiAgICBwID0gdmltX3N0cnNhdmUoKGNoYXJfdSAqKW5ld19sYWJlbCk7CiAgICBpZiAocCA9PSBOVUxMKQoJcmV0dXJuOwogICAgZm9yIChuZXh0ID0gcDsgKm5leHQ7ICsrbmV4dCkKICAgIHsKCWlmICgqbmV4dCA9PSBETEdfSE9US0VZX0NIQVIpCgl7CgkgICAgaW50IGxlbiA9IFNUUkxFTihuZXh0KTsKCgkgICAgaWYgKGxlbiA+IDApCgkgICAgewoJCW1jaF9tZW1tb3ZlKG5leHQsIG5leHQgKyAxLCBsZW4pOwoJCW1uZW1vbmljID0gbmV4dFswXTsKCSAgICB9Cgl9CiAgICB9CgogICAgc3RyID0gWG1TdHJpbmdDcmVhdGUoKGNoYXIgKilwLCBTVFJJTkdfVEFHKTsKICAgIHZpbV9mcmVlKHApOwoKICAgIGlmIChzdHIgIT0gTlVMTCkKICAgIHsKCVh0VmFTZXRWYWx1ZXModywKCQlYbU5sYWJlbFN0cmluZywgc3RyLAoJCVhtTm1uZW1vbmljLCBtbmVtb25pYywKCQlOVUxMKTsKCVhtU3RyaW5nRnJlZShzdHIpOwogICAgfQogICAgZ3VpX21vdGlmX21lbnVfZm9udGxpc3Qodyk7Cn0KCnN0YXRpYyB2b2lkCnNldF9wcmVkZWZpbmVkX2ZvbnRsaXN0KHBhcmVudCwgbmFtZSkKICAgIFdpZGdldCBwYXJlbnQ7CiAgICBTdHJpbmcgbmFtZTsKewogICAgV2lkZ2V0IHc7CiAgICB3ID0gWHROYW1lVG9XaWRnZXQocGFyZW50LCBuYW1lKTsKCiAgICBpZiAoIXcpCglyZXR1cm47CgogICAgc2V0X2ZvbnRsaXN0KHcpOwp9CgovKgogKiBQdXQgdXAgYSBmaWxlIHJlcXVlc3Rlci4KICogUmV0dXJucyB0aGUgc2VsZWN0ZWQgbmFtZSBpbiBhbGxvY2F0ZWQgbWVtb3J5LCBvciBOVUxMIGZvciBDYW5jZWwuCiAqLwogICAgY2hhcl91ICoKZ3VpX21jaF9icm93c2Uoc2F2aW5nLCB0aXRsZSwgZGZsdCwgZXh0LCBpbml0ZGlyLCBmaWx0ZXIpCiAgICBpbnQJCXNhdmluZyBVTlVTRUQ7CS8qIHNlbGVjdCBmaWxlIHRvIHdyaXRlICovCiAgICBjaGFyX3UJKnRpdGxlOwkJLyogdGl0bGUgZm9yIHRoZSB3aW5kb3cgKi8KICAgIGNoYXJfdQkqZGZsdDsJCS8qIGRlZmF1bHQgbmFtZSAqLwogICAgY2hhcl91CSpleHQgVU5VU0VEOwkvKiBub3QgdXNlZCAoZXh0ZW5zaW9uIGFkZGVkKSAqLwogICAgY2hhcl91CSppbml0ZGlyOwkvKiBpbml0aWFsIGRpcmVjdG9yeSwgTlVMTCBmb3IgY3VycmVudCBkaXIgKi8KICAgIGNoYXJfdQkqZmlsdGVyOwkvKiBmaWxlIG5hbWUgZmlsdGVyICovCnsKICAgIGNoYXJfdQlkaXJidWZbTUFYUEFUSExdOwogICAgY2hhcl91CWRmbHRidWZbTUFYUEFUSExdOwogICAgY2hhcl91CSpwYXR0ZXJuOwogICAgY2hhcl91CSp0b2ZyZWUgPSBOVUxMOwoKICAgIC8qIFRoZXJlIGEgZGlmZmVyZW5jZSBiZXR3ZWVuIHRoZSByZXNvdXJjZSBuYW1lIGFuZCB2YWx1ZSwgVGhlcmVmb3JlLCB3ZQogICAgICogYXZvaWQgdG8gKGFiLSl1c2UgdGhlIChtYXliZSBpbnRlcm5hdGlvbmFsaXplZCEpIGRpYWxvZyB0aXRsZSBhcyBhCiAgICAgKiBkaWFsb2cgbmFtZS4KICAgICAqLwoKICAgIGRpYWxvZ193Z3QgPSBYbUNyZWF0ZUZpbGVTZWxlY3Rpb25EaWFsb2codmltU2hlbGwsICJicm93c2VEaWFsb2ciLCBOVUxMLCAwKTsKCiAgICBpZiAoaW5pdGRpciA9PSBOVUxMIHx8ICppbml0ZGlyID09IE5VTCkKICAgIHsKCW1jaF9kaXJuYW1lKGRpcmJ1ZiwgTUFYUEFUSEwpOwoJaW5pdGRpciA9IGRpcmJ1ZjsKICAgIH0KCiAgICBpZiAoZGZsdCA9PSBOVUxMKQoJZGZsdCA9IChjaGFyX3UgKikiIjsKICAgIGVsc2UgaWYgKFNUUkxFTihpbml0ZGlyKSArIFNUUkxFTihkZmx0KSArIDIgPCBNQVhQQVRITCkKICAgIHsKCS8qIFRoZSBkZWZhdWx0IHNlbGVjdGlvbiBzaG91bGQgYmUgdGhlIGZ1bGwgcGF0aCwgImRmbHQiIGlzIG9ubHkgdGhlCgkgKiBmaWxlIG5hbWUuICovCglTVFJDUFkoZGZsdGJ1ZiwgaW5pdGRpcik7CglhZGRfcGF0aHNlcChkZmx0YnVmKTsKCVNUUkNBVChkZmx0YnVmLCBkZmx0KTsKCWRmbHQgPSBkZmx0YnVmOwogICAgfQoKICAgIC8qIENhbiBvbmx5IHVzZSBvbmUgcGF0dGVybiBmb3IgYSBmaWxlIG5hbWUuICBHZXQgdGhlIGZpcnN0IHBhdHRlcm4gb3V0IG9mCiAgICAgKiB0aGUgZmlsdGVyLiAgQW4gZW1wdHkgcGF0dGVybiBtZWFucyBldmVyeXRoaW5nIG1hdGNoZXMuICovCiAgICBpZiAoZmlsdGVyID09IE5VTEwpCglwYXR0ZXJuID0gKGNoYXJfdSAqKSIiOwogICAgZWxzZQogICAgewoJY2hhcl91CSpzLCAqcDsKCglzID0gZmlsdGVyOwoJZm9yIChwID0gZmlsdGVyOyAqcCAhPSBOVUw7ICsrcCkKCXsKCSAgICBpZiAoKnAgPT0gJ1x0JykJLyogZW5kIG9mIGRlc2NyaXB0aW9uLCBzdGFydCBvZiBwYXR0ZXJuICovCgkJcyA9IHAgKyAxOwoJICAgIGlmICgqcCA9PSAnOycgfHwgKnAgPT0gJ1xuJykJLyogZW5kIG9mIChmaXJzdCkgcGF0dGVybiAqLwoJCWJyZWFrOwoJfQoJcGF0dGVybiA9IHZpbV9zdHJuc2F2ZShzLCBwIC0gcyk7Cgl0b2ZyZWUgPSBwYXR0ZXJuOwoJaWYgKHBhdHRlcm4gPT0gTlVMTCkKCSAgICBwYXR0ZXJuID0gKGNoYXJfdSAqKSIiOwogICAgfQoKICAgIFh0VmFTZXRWYWx1ZXMoZGlhbG9nX3dndCwKCVh0VmFUeXBlZEFyZywKCSAgICBYbU5kaXJlY3RvcnksIFhtUlN0cmluZywgKGNoYXIgKilpbml0ZGlyLCBTVFJMRU4oaW5pdGRpcikgKyAxLAoJWHRWYVR5cGVkQXJnLAoJICAgIFhtTmRpclNwZWMsCVhtUlN0cmluZywgKGNoYXIgKilkZmx0LCBTVFJMRU4oZGZsdCkgKyAxLAoJWHRWYVR5cGVkQXJnLAoJICAgIFhtTnBhdHRlcm4sCVhtUlN0cmluZywgKGNoYXIgKilwYXR0ZXJuLCBTVFJMRU4ocGF0dGVybikgKyAxLAoJWHRWYVR5cGVkQXJnLAoJICAgIFhtTmRpYWxvZ1RpdGxlLCBYbVJTdHJpbmcsIChjaGFyICopdGl0bGUsIFNUUkxFTih0aXRsZSkgKyAxLAoJTlVMTCk7CgogICAgc2V0X3ByZWRlZmluZWRfbGFiZWwoZGlhbG9nX3dndCwgIkFwcGx5IiwgXygiJkZpbHRlciIpKTsKICAgIHNldF9wcmVkZWZpbmVkX2xhYmVsKGRpYWxvZ193Z3QsICJDYW5jZWwiLCBfKCImQ2FuY2VsIikpOwogICAgc2V0X3ByZWRlZmluZWRfbGFiZWwoZGlhbG9nX3dndCwgIkRpciIsIF8oIkRpcmVjdG9yaWVzIikpOwogICAgc2V0X3ByZWRlZmluZWRfbGFiZWwoZGlhbG9nX3dndCwgIkZpbHRlckxhYmVsIiwgXygiRmlsdGVyIikpOwogICAgc2V0X3ByZWRlZmluZWRfbGFiZWwoZGlhbG9nX3dndCwgIkhlbHAiLCBfKCImSGVscCIpKTsKICAgIHNldF9wcmVkZWZpbmVkX2xhYmVsKGRpYWxvZ193Z3QsICJJdGVtcyIsIF8oIkZpbGVzIikpOwogICAgc2V0X3ByZWRlZmluZWRfbGFiZWwoZGlhbG9nX3dndCwgIk9LIiwgXygiJk9LIikpOwogICAgc2V0X3ByZWRlZmluZWRfbGFiZWwoZGlhbG9nX3dndCwgIlNlbGVjdGlvbiIsIF8oIlNlbGVjdGlvbiIpKTsKCiAgICAvKiBUaGlzIGlzIHRvIHNhdmUgdXMgZnJvbSBzaWxseSBleHRlcm5hbCBzZXR0aW5ncyB1c2luZyBub3QgZml4ZWQgd2l0aAogICAgICogZm9udHMgZm9yIGZpbGUgc2VsZWN0aW9uLgogICAgICovCiAgICBzZXRfcHJlZGVmaW5lZF9mb250bGlzdChkaWFsb2dfd2d0LCAiRGlyTGlzdFNXLkRpckxpc3QiKTsKICAgIHNldF9wcmVkZWZpbmVkX2ZvbnRsaXN0KGRpYWxvZ193Z3QsICJJdGVtc0xpc3RTVy5JdGVtc0xpc3QiKTsKCiAgICBndWlfbW90aWZfbWVudV9jb2xvcnMoZGlhbG9nX3dndCk7CiAgICBpZiAoZ3VpLnNjcm9sbF9iZ19waXhlbCAhPSBJTlZBTENPTE9SKQoJWHRWYVNldFZhbHVlcyhkaWFsb2dfd2d0LCBYbU50cm91Z2hDb2xvciwgZ3VpLnNjcm9sbF9iZ19waXhlbCwgTlVMTCk7CgogICAgWHRBZGRDYWxsYmFjayhkaWFsb2dfd2d0LCBYbU5va0NhbGxiYWNrLCBEaWFsb2dBY2NlcHRDQiwgKFh0UG9pbnRlcikwKTsKICAgIFh0QWRkQ2FsbGJhY2soZGlhbG9nX3dndCwgWG1OY2FuY2VsQ2FsbGJhY2ssIERpYWxvZ0NhbmNlbENCLCAoWHRQb2ludGVyKTApOwogICAgLyogV2UgaGF2ZSBubyBoZWxwIGluIHRoaXMgd2luZG93LCBzbyBoaWRlIGhlbHAgYnV0dG9uICovCiAgICBYdFVubWFuYWdlQ2hpbGQoWG1GaWxlU2VsZWN0aW9uQm94R2V0Q2hpbGQoZGlhbG9nX3dndCwKCQkJCQkodW5zaWduZWQgY2hhcilYbURJQUxPR19IRUxQX0JVVFRPTikpOwoKICAgIG1hbmFnZV9jZW50ZXJlZChkaWFsb2dfd2d0KTsKICAgIGFjdGl2YXRlX2RpYWxvZ19tbmVtb25pY3MoZGlhbG9nX3dndCk7CgogICAgLyogc2l0IGluIGEgbG9vcCB1bnRpbCB0aGUgZGlhbG9nIGJveCBoYXMgZ29uZSBhd2F5ICovCiAgICBkbwogICAgewoJWHRBcHBQcm9jZXNzRXZlbnQoWHRXaWRnZXRUb0FwcGxpY2F0aW9uQ29udGV4dChkaWFsb2dfd2d0KSwKCSAgICAoWHRJbnB1dE1hc2spWHRJTUFsbCk7CiAgICB9IHdoaWxlIChYdElzTWFuYWdlZChkaWFsb2dfd2d0KSk7CgogICAgc3VwcHJlc3NfZGlhbG9nX21uZW1vbmljcyhkaWFsb2dfd2d0KTsKICAgIFh0RGVzdHJveVdpZGdldChkaWFsb2dfd2d0KTsKICAgIHZpbV9mcmVlKHRvZnJlZSk7CgogICAgaWYgKGJyb3dzZV9mbmFtZSA9PSBOVUxMKQoJcmV0dXJuIE5VTEw7CiAgICByZXR1cm4gdmltX3N0cnNhdmUoKGNoYXJfdSAqKWJyb3dzZV9mbmFtZSk7Cn0KCi8qCiAqIFRoZSBjb2RlIGJlbG93IHdhcyBvcmlnaW5hbGx5IHRha2VuIGZyb20KICoJL3Vzci9leGFtcGxlcy9tb3RpZi94bXNhbXBsZXJzL3htZWRpdG9yLmMKICogb24gRGlnaXRhbCBVbml4IDQuMGQsIGJ1dCBoZWF2aWx5IG1vZGlmaWVkLgogKi8KCi8qCiAqIFByb2Nlc3MgY2FsbGJhY2sgZnJvbSBEaWFsb2cgY2FuY2VsIGFjdGlvbnMuCiAqLwogICAgc3RhdGljIHZvaWQKRGlhbG9nQ2FuY2VsQ0IodywgY2xpZW50X2RhdGEsIGNhbGxfZGF0YSkKICAgIFdpZGdldAl3IFVOVVNFRDsJCS8qICB3aWRnZXQgaWQJCSovCiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGEgVU5VU0VEOwkvKiAgZGF0YSBmcm9tIGFwcGxpY2F0aW9uICAgKi8KICAgIFh0UG9pbnRlcgljYWxsX2RhdGEgVU5VU0VEOwkvKiAgZGF0YSBmcm9tIHdpZGdldCBjbGFzcyAgKi8KewogICAgaWYgKGJyb3dzZV9mbmFtZSAhPSBOVUxMKQogICAgewoJWHRGcmVlKGJyb3dzZV9mbmFtZSk7Cglicm93c2VfZm5hbWUgPSBOVUxMOwogICAgfQogICAgWHRVbm1hbmFnZUNoaWxkKGRpYWxvZ193Z3QpOwp9CgovKgogKiBQcm9jZXNzIGNhbGxiYWNrIGZyb20gRGlhbG9nIGFjdGlvbnMuCiAqLwogICAgc3RhdGljIHZvaWQKRGlhbG9nQWNjZXB0Q0IodywgY2xpZW50X2RhdGEsIGNhbGxfZGF0YSkKICAgIFdpZGdldAl3IFVOVVNFRDsJCS8qICB3aWRnZXQgaWQJCSovCiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGEgVU5VU0VEOwkvKiAgZGF0YSBmcm9tIGFwcGxpY2F0aW9uICAgKi8KICAgIFh0UG9pbnRlcgljYWxsX2RhdGE7CQkvKiAgZGF0YSBmcm9tIHdpZGdldCBjbGFzcyAgKi8KewogICAgWG1GaWxlU2VsZWN0aW9uQm94Q2FsbGJhY2tTdHJ1Y3QgKmZjYjsKCiAgICBpZiAoYnJvd3NlX2ZuYW1lICE9IE5VTEwpCiAgICB7CglYdEZyZWUoYnJvd3NlX2ZuYW1lKTsKCWJyb3dzZV9mbmFtZSA9IE5VTEw7CiAgICB9CiAgICBmY2IgPSAoWG1GaWxlU2VsZWN0aW9uQm94Q2FsbGJhY2tTdHJ1Y3QgKiljYWxsX2RhdGE7CgogICAgLyogZ2V0IHRoZSBmaWxlbmFtZSBmcm9tIHRoZSBmaWxlIHNlbGVjdGlvbiBib3ggKi8KICAgIFhtU3RyaW5nR2V0THRvUihmY2ItPnZhbHVlLCBjaGFyc2V0LCAmYnJvd3NlX2ZuYW1lKTsKCiAgICAvKiBwb3Bkb3duIHRoZSBmaWxlIHNlbGVjdGlvbiBib3ggKi8KICAgIFh0VW5tYW5hZ2VDaGlsZChkaWFsb2dfd2d0KTsKfQoKI2VuZGlmIC8qIEZFQVRfQlJPV1NFICovCgojaWYgZGVmaW5lZChGRUFUX0dVSV9ESUFMT0cpIHx8IGRlZmluZWQoUFJPVE8pCgpzdGF0aWMgaW50CWRpYWxvZ1N0YXR1czsKCnN0YXRpYyB2b2lkIGtleWhpdF9jYWxsYmFjayBfX0FSR1MoKFdpZGdldCB3LCBYdFBvaW50ZXIgY2xpZW50X2RhdGEsIFhFdmVudCAqZXZlbnQsIEJvb2xlYW4gKmNvbnQpKTsKc3RhdGljIHZvaWQgYnV0cHJvYyBfX0FSR1MoKFdpZGdldCB3LCBYdFBvaW50ZXIgY2xpZW50X2RhdGEsIFh0UG9pbnRlciBjYWxsX2RhdGEpKTsKCi8qCiAqIENhbGxiYWNrIGZ1bmN0aW9uIGZvciB0aGUgdGV4dGZpZWxkLiAgV2hlbiBDUiBpcyBoaXQgdGhpcyB3b3JrcyBsaWtlCiAqIGhpdHRpbmcgdGhlICJPSyIgYnV0dG9uLCBFU0MgbGlrZSAiQ2FuY2VsIi4KICovCiAgICBzdGF0aWMgdm9pZAprZXloaXRfY2FsbGJhY2sodywgY2xpZW50X2RhdGEsIGV2ZW50LCBjb250KQogICAgV2lkZ2V0CQl3OwogICAgWHRQb2ludGVyCQljbGllbnRfZGF0YSBVTlVTRUQ7CiAgICBYRXZlbnQJCSpldmVudDsKICAgIEJvb2xlYW4JCSpjb250IFVOVVNFRDsKewogICAgY2hhcglidWZbMl07CiAgICBLZXlTeW0Ja2V5X3N5bTsKCiAgICBpZiAoWExvb2t1cFN0cmluZygmKGV2ZW50LT54a2V5KSwgYnVmLCAyLCAma2V5X3N5bSwgTlVMTCkgPT0gMSkKICAgIHsKCWlmICgqYnVmID09IENBUikKCSAgICBkaWFsb2dTdGF0dXMgPSAxOwoJZWxzZSBpZiAoKmJ1ZiA9PSBFU0MpCgkgICAgZGlhbG9nU3RhdHVzID0gMjsKICAgIH0KICAgIGlmICgoa2V5X3N5bSA9PSBYS19MZWZ0IHx8IGtleV9zeW0gPT0gWEtfUmlnaHQpCgkgICAgJiYgIShldmVudC0+eGtleS5zdGF0ZSAmIFNoaWZ0TWFzaykpCglYbVRleHRGaWVsZENsZWFyU2VsZWN0aW9uKHcsIFh0TGFzdFRpbWVzdGFtcFByb2Nlc3NlZChndWkuZHB5KSk7Cn0KCiAgICBzdGF0aWMgdm9pZApidXRwcm9jKHcsIGNsaWVudF9kYXRhLCBjYWxsX2RhdGEpCiAgICBXaWRnZXQJdyBVTlVTRUQ7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGE7CiAgICBYdFBvaW50ZXIJY2FsbF9kYXRhIFVOVVNFRDsKewogICAgZGlhbG9nU3RhdHVzID0gKGludCkobG9uZyljbGllbnRfZGF0YSArIDE7Cn0KCiNpZmRlZiBIQVZFX1hQTQoKc3RhdGljIFdpZGdldCBjcmVhdGVfcGl4bWFwX2xhYmVsKFdpZGdldCBwYXJlbnQsIFN0cmluZyBuYW1lLCBjaGFyICoqZGF0YSwgQXJnTGlzdCBhcmdzLCBDYXJkaW5hbCBhcmcpOwoKICAgIHN0YXRpYyBXaWRnZXQKY3JlYXRlX3BpeG1hcF9sYWJlbChwYXJlbnQsIG5hbWUsIGRhdGEsIGFyZ3MsIGFyZykKICAgIFdpZGdldAlwYXJlbnQ7CiAgICBTdHJpbmcJbmFtZTsKICAgIGNoYXIJKipkYXRhOwogICAgQXJnTGlzdAlhcmdzOwogICAgQ2FyZGluYWwJYXJnOwp7CiAgICBXaWRnZXQJCWxhYmVsOwogICAgRGlzcGxheQkJKmRzcDsKICAgIFNjcmVlbgkJKnNjcjsKICAgIGludAkJCWRlcHRoOwogICAgUGl4bWFwCQlwaXhtYXAgPSAwOwogICAgWHBtQXR0cmlidXRlcwlhdHRyOwogICAgQm9vbGVhbgkJcnM7CiAgICBYcG1Db2xvclN5bWJvbAljb2xvcls1XSA9CiAgICB7Cgl7Im5vbmUiLCBOVUxMLCAwfSwKCXsiaWNvbkNvbG9yMSIsIE5VTEwsIDB9LAoJeyJib3R0b21TaGFkb3dDb2xvciIsIE5VTEwsIDB9LAoJeyJ0b3BTaGFkb3dDb2xvciIsIE5VTEwsIDB9LAoJeyJzZWxlY3RDb2xvciIsIE5VTEwsIDB9CiAgICB9OwoKICAgIGxhYmVsID0gWG1DcmVhdGVMYWJlbEdhZGdldChwYXJlbnQsIG5hbWUsIGFyZ3MsIGFyZyk7CgogICAgLyoKICAgICAqIFdlIG5lZWQgdG8gYmUgY2FyZWZ1bCBoZXJlLCBzaW5jZSBpbiBjYXNlIG9mIGdhZGdldHMsIHRoZXJlIGlzCiAgICAgKiBubyB3YXkgdG8gZ2V0IHRoZSBiYWNrZ3JvdW5kIGNvbG9yIGRpcmVjdGx5IGZyb20gdGhlIHdpZGdldCBpdHNlbGYuCiAgICAgKiBJbiBzdWNoIGNhc2VzIHdlIGdldCBpdCBmcm9tIFRoZSBDb3JlIHBhcnQgb2YgaGlzIHBhcmVudCBpbnN0ZWFkLgogICAgICovCiAgICBkc3AgPSBYdERpc3BsYXlPZk9iamVjdChsYWJlbCk7CiAgICBzY3IgPSBYdFNjcmVlbk9mT2JqZWN0KGxhYmVsKTsKICAgIFh0VmFHZXRWYWx1ZXMoWHRJc1N1YmNsYXNzKGxhYmVsLCBjb3JlV2lkZ2V0Q2xhc3MpCgkgICAgPyAgbGFiZWwgOiBYdFBhcmVudChsYWJlbCksCgkJICBYbU5kZXB0aCwgJmRlcHRoLAoJCSAgWG1OYmFja2dyb3VuZCwgJmNvbG9yWzBdLnBpeGVsLAoJCSAgWG1OZm9yZWdyb3VuZCwgJmNvbG9yWzFdLnBpeGVsLAoJCSAgWG1OYm90dG9tU2hhZG93Q29sb3IsICZjb2xvclsyXS5waXhlbCwKCQkgIFhtTnRvcFNoYWRvd0NvbG9yLCAmY29sb3JbM10ucGl4ZWwsCgkJICBYbU5oaWdobGlnaHQsICZjb2xvcls0XS5waXhlbCwKCQkgIE5VTEwpOwoKICAgIGF0dHIudmFsdWVtYXNrID0gWHBtQ29sb3JTeW1ib2xzIHwgWHBtQ2xvc2VuZXNzIHwgWHBtRGVwdGg7CiAgICBhdHRyLmNvbG9yc3ltYm9scyA9IGNvbG9yOwogICAgYXR0ci5udW1zeW1ib2xzID0gNTsKICAgIGF0dHIuY2xvc2VuZXNzID0gNjU1MzU7CiAgICBhdHRyLmRlcHRoID0gZGVwdGg7CiAgICBYcG1DcmVhdGVQaXhtYXBGcm9tRGF0YShkc3AsIFJvb3RXaW5kb3dPZlNjcmVlbihzY3IpLAoJCSAgICBkYXRhLCAmcGl4bWFwLCBOVUxMLCAmYXR0cik7CgogICAgWHRWYUdldFZhbHVlcyhsYWJlbCwgWG1OcmVjb21wdXRlU2l6ZSwgJnJzLCBOVUxMKTsKICAgIFh0VmFTZXRWYWx1ZXMobGFiZWwsIFhtTnJlY29tcHV0ZVNpemUsIFRydWUsIE5VTEwpOwogICAgWHRWYVNldFZhbHVlcyhsYWJlbCwKCSAgICBYbU5sYWJlbFR5cGUsIFhtUElYTUFQLAoJICAgIFhtTmxhYmVsUGl4bWFwLCBwaXhtYXAsCgkgICAgTlVMTCk7CiAgICBYdFZhU2V0VmFsdWVzKGxhYmVsLCBYbU5yZWNvbXB1dGVTaXplLCBycywgTlVMTCk7CgogICAgcmV0dXJuIGxhYmVsOwp9CiNlbmRpZgoKICAgIGludApndWlfbWNoX2RpYWxvZyh0eXBlLCB0aXRsZSwgbWVzc2FnZSwgYnV0dG9uX25hbWVzLCBkZmx0YnV0dG9uLCB0ZXh0ZmllbGQpCiAgICBpbnQJCXR5cGUgVU5VU0VEOwogICAgY2hhcl91CSp0aXRsZTsKICAgIGNoYXJfdQkqbWVzc2FnZTsKICAgIGNoYXJfdQkqYnV0dG9uX25hbWVzOwogICAgaW50CQlkZmx0YnV0dG9uOwogICAgY2hhcl91CSp0ZXh0ZmllbGQ7CQkvKiBidWZmZXIgb2Ygc2l6ZSBJT1NJWkUgKi8KewogICAgY2hhcl91CQkqYnV0czsKICAgIGNoYXJfdQkJKnAsICpuZXh0OwogICAgWHRBcHBDb250ZXh0CWFwcDsKICAgIFhtU3RyaW5nCQlsYWJlbDsKICAgIGludAkJCWJ1dGNvdW50OwogICAgV2lkZ2V0CQl3OwogICAgV2lkZ2V0CQlkaWFsb2dmb3JtID0gTlVMTDsKICAgIFdpZGdldAkJZm9ybSA9IE5VTEw7CiAgICBXaWRnZXQJCWRpYWxvZ3RleHRmaWVsZCA9IE5VTEw7CiAgICBXaWRnZXQJCSpidXR0b25zOwogICAgV2lkZ2V0CQlzZXBfZm9ybSA9IE5VTEw7CiAgICBCb29sZWFuCQl2ZXJ0aWNhbDsKICAgIFdpZGdldAkJc2VwYXJhdG9yID0gTlVMTDsKICAgIGludAkJCW47CiAgICBBcmcJCQlhcmdzWzZdOwojaWZkZWYgSEFWRV9YUE0KICAgIGNoYXIJCSoqaWNvbl9kYXRhID0gTlVMTDsKICAgIFdpZGdldAkJZGlhbG9ncGl4bWFwID0gTlVMTDsKI2VuZGlmCgogICAgaWYgKHRpdGxlID09IE5VTEwpCgl0aXRsZSA9IChjaGFyX3UgKilfKCJWaW0gZGlhbG9nIik7CgogICAgLyogaWYgb3VyIHBvaW50ZXIgaXMgY3VycmVudGx5IGhpZGRlbiwgdGhlbiB3ZSBzaG91bGQgc2hvdyBpdC4gKi8KICAgIGd1aV9tY2hfbW91c2VoaWRlKEZBTFNFKTsKCiAgICBkaWFsb2dmb3JtID0gWG1DcmVhdGVGb3JtRGlhbG9nKHZpbVNoZWxsLCAoY2hhciAqKSJkaWFsb2ciLCBOVUxMLCAwKTsKCiAgICAvKiBDaGVjayAndicgZmxhZyBpbiAnZ3Vpb3B0aW9ucyc6IHZlcnRpY2FsIGJ1dHRvbiBwbGFjZW1lbnQuICovCiAgICB2ZXJ0aWNhbCA9ICh2aW1fc3RyY2hyKHBfZ28sIEdPX1ZFUlRJQ0FMKSAhPSBOVUxMKTsKCiAgICAvKiBTZXQgdGhlIHRpdGxlIG9mIHRoZSBEaWFsb2cgd2luZG93ICovCiAgICBsYWJlbCA9IFhtU3RyaW5nQ3JlYXRlU2ltcGxlKChjaGFyICopdGl0bGUpOwogICAgaWYgKGxhYmVsID09IE5VTEwpCglyZXR1cm4gLTE7CiAgICBYdFZhU2V0VmFsdWVzKGRpYWxvZ2Zvcm0sCgkgICAgWG1OZGlhbG9nVGl0bGUsIGxhYmVsLAoJICAgIFhtTmhvcml6b250YWxTcGFjaW5nLCA0LAoJICAgIFhtTnZlcnRpY2FsU3BhY2luZywgdmVydGljYWwgPyAwIDogNCwKCSAgICBOVUxMKTsKICAgIFhtU3RyaW5nRnJlZShsYWJlbCk7CgogICAgLyogbWFrZSBhIGNvcHksIHNvIHRoYXQgd2UgY2FuIGluc2VydCBOVUxzICovCiAgICBidXRzID0gdmltX3N0cnNhdmUoYnV0dG9uX25hbWVzKTsKICAgIGlmIChidXRzID09IE5VTEwpCglyZXR1cm4gLTE7CgogICAgLyogQ291bnQgdGhlIG51bWJlciBvZiBidXR0b25zIGFuZCBhbGxvY2F0ZSBidXR0b25zW10uICovCiAgICBidXRjb3VudCA9IDE7CiAgICBmb3IgKHAgPSBidXRzOyAqcDsgKytwKQoJaWYgKCpwID09IERMR19CVVRUT05fU0VQKQoJICAgICsrYnV0Y291bnQ7CiAgICBidXR0b25zID0gKFdpZGdldCAqKWFsbG9jKCh1bnNpZ25lZCkoYnV0Y291bnQgKiBzaXplb2YoV2lkZ2V0KSkpOwogICAgaWYgKGJ1dHRvbnMgPT0gTlVMTCkKICAgIHsKCXZpbV9mcmVlKGJ1dHMpOwoJcmV0dXJuIC0xOwogICAgfQoKICAgIC8qCiAgICAgKiBDcmVhdGUgdGhlIGJ1dHRvbnMuCiAgICAgKi8KICAgIHNlcF9mb3JtID0gKFdpZGdldCkgMDsKICAgIHAgPSBidXRzOwogICAgZm9yIChidXRjb3VudCA9IDA7ICpwOyArK2J1dGNvdW50KQogICAgewoJS2V5U3ltIG1uZW1vbmljID0gTlVMOwoKCWZvciAobmV4dCA9IHA7ICpuZXh0OyArK25leHQpCgl7CgkgICAgaWYgKCpuZXh0ID09IERMR19IT1RLRVlfQ0hBUikKCSAgICB7CgkJaW50IGxlbiA9IFNUUkxFTihuZXh0KTsKCgkJaWYgKGxlbiA+IDApCgkJewoJCSAgICBtY2hfbWVtbW92ZShuZXh0LCBuZXh0ICsgMSwgbGVuKTsKCQkgICAgbW5lbW9uaWMgPSBuZXh0WzBdOwoJCX0KCSAgICB9CgkgICAgaWYgKCpuZXh0ID09IERMR19CVVRUT05fU0VQKQoJICAgIHsKCQkqbmV4dCsrID0gTlVMOwoJCWJyZWFrOwoJICAgIH0KCX0KCWxhYmVsID0gWG1TdHJpbmdDcmVhdGUoXygoY2hhciAqKXApLCBTVFJJTkdfVEFHKTsKCWlmIChsYWJlbCA9PSBOVUxMKQoJICAgIGJyZWFrOwoKCWJ1dHRvbnNbYnV0Y291bnRdID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoImJ1dHRvbiIsCgkJeG1QdXNoQnV0dG9uV2lkZ2V0Q2xhc3MsIGRpYWxvZ2Zvcm0sCgkJWG1ObGFiZWxTdHJpbmcsIGxhYmVsLAoJCVhtTm1uZW1vbmljLCBtbmVtb25pYywKCQlYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCVhtTmJvdHRvbU9mZnNldCwgNCwKCQlYbU5zaG93QXNEZWZhdWx0LCBidXRjb3VudCA9PSBkZmx0YnV0dG9uIC0gMSwKCQlYbU5kZWZhdWx0QnV0dG9uU2hhZG93VGhpY2tuZXNzLCAxLAoJCU5VTEwpOwoJWG1TdHJpbmdGcmVlKGxhYmVsKTsKCWd1aV9tb3RpZl9tZW51X2ZvbnRsaXN0KGJ1dHRvbnNbYnV0Y291bnRdKTsKCgkvKiBMYXlvdXQgcHJvcGVybHkuICovCgoJaWYgKGJ1dGNvdW50ID4gMCkKCXsKCSAgICBpZiAodmVydGljYWwpCgkJWHRWYVNldFZhbHVlcyhidXR0b25zW2J1dGNvdW50XSwKCQkJWG1OdG9wV2lkZ2V0LCBidXR0b25zW2J1dGNvdW50IC0gMV0sCgkJCU5VTEwpOwoJICAgIGVsc2UKCSAgICB7CgkJaWYgKCpuZXh0ID09IE5VTCkKCQl7CgkJICAgIFh0VmFTZXRWYWx1ZXMoYnV0dG9uc1tidXRjb3VudF0sCgkJCSAgICBYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJCSAgICBYbU5yaWdodE9mZnNldCwgNCwKCQkJICAgIE5VTEwpOwoKCQkgICAgLyogZmlsbCBpbiBhIGZvcm0gYXMgaW52aXNpYmxlIHNlcGFyYXRvciAqLwoJCSAgICBzZXBfZm9ybSA9IFh0VmFDcmVhdGVXaWRnZXQoInNlcGFyYXRvckZvcm0iLAoJCQkgICAgeG1Gb3JtV2lkZ2V0Q2xhc3MsCWRpYWxvZ2Zvcm0sCgkJCSAgICBYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgWG1ObGVmdFdpZGdldCwgYnV0dG9uc1tidXRjb3VudCAtIDFdLAoJCQkgICAgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICBYbU5yaWdodFdpZGdldCwgYnV0dG9uc1tidXRjb3VudF0sCgkJCSAgICBYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQkgICAgWG1OYm90dG9tT2Zmc2V0LCA0LAoJCQkgICAgTlVMTCk7CgkJICAgIFh0TWFuYWdlQ2hpbGQoc2VwX2Zvcm0pOwoJCX0KCQllbHNlCgkJewoJCSAgICBYdFZhU2V0VmFsdWVzKGJ1dHRvbnNbYnV0Y291bnRdLAoJCQkgICAgWG1ObGVmdEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQkJICAgIFhtTmxlZnRXaWRnZXQsIGJ1dHRvbnNbYnV0Y291bnQgLSAxXSwKCQkJICAgIE5VTEwpOwoJCX0KCSAgICB9Cgl9CgllbHNlIGlmICghdmVydGljYWwpCgl7CgkgICAgaWYgKCpuZXh0ID09IE5VTCkKCSAgICB7CgkJWHRWYVNldFZhbHVlcyhidXR0b25zWzBdLAoJCQlYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJCVhtTnJpZ2h0T2Zmc2V0LCA0LAoJCQlOVUxMKTsKCgkJLyogZmlsbCBpbiBhIGZvcm0gYXMgaW52aXNpYmxlIHNlcGFyYXRvciAqLwoJCXNlcF9mb3JtID0gWHRWYUNyZWF0ZVdpZGdldCgic2VwYXJhdG9yRm9ybSIsCgkJCXhtRm9ybVdpZGdldENsYXNzLCBkaWFsb2dmb3JtLAoJCQlYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQkJWG1ObGVmdE9mZnNldCwgNCwKCQkJWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCVhtTnJpZ2h0V2lkZ2V0LCBidXR0b25zWzBdLAoJCQlYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQlYbU5ib3R0b21PZmZzZXQsIDQsCgkJCU5VTEwpOwoJCVh0TWFuYWdlQ2hpbGQoc2VwX2Zvcm0pOwoJICAgIH0KCSAgICBlbHNlCgkJWHRWYVNldFZhbHVlcyhidXR0b25zWzBdLAoJCQlYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQkJWG1ObGVmdE9mZnNldCwgNCwKCQkJTlVMTCk7Cgl9CgoJWHRBZGRDYWxsYmFjayhidXR0b25zW2J1dGNvdW50XSwgWG1OYWN0aXZhdGVDYWxsYmFjaywKCQkJICAoWHRDYWxsYmFja1Byb2MpYnV0cHJvYywgKFh0UG9pbnRlcikobG9uZylidXRjb3VudCk7CglwID0gbmV4dDsKICAgIH0KICAgIHZpbV9mcmVlKGJ1dHMpOwoKICAgIHNlcGFyYXRvciA9IChXaWRnZXQpIDA7CiAgICBpZiAoYnV0Y291bnQgPiAwKQogICAgewoJLyogQ3JlYXRlIHRoZSBzZXBhcmF0b3IgZm9yIGJlYXV0eS4gKi8KCW4gPSAwOwoJWHRTZXRBcmcoYXJnc1tuXSwgWG1Ob3JpZW50YXRpb24sIFhtSE9SSVpPTlRBTCk7IG4rKzsKCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCk7IG4rKzsKCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbVdpZGdldCwgYnV0dG9uc1swXSk7IG4rKzsKCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbU9mZnNldCwgNCk7IG4rKzsKCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwoJWHRTZXRBcmcoYXJnc1tuXSwgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwoJc2VwYXJhdG9yID0gWG1DcmVhdGVTZXBhcmF0b3JHYWRnZXQoZGlhbG9nZm9ybSwgInNlcGFyYXRvciIsIGFyZ3MsIG4pOwoJWHRNYW5hZ2VDaGlsZChzZXBhcmF0b3IpOwogICAgfQoKICAgIGlmICh0ZXh0ZmllbGQgIT0gTlVMTCkKICAgIHsKCWRpYWxvZ3RleHRmaWVsZCA9IFh0VmFDcmVhdGVXaWRnZXQoInRleHRGaWVsZCIsCgkJeG1UZXh0RmllbGRXaWRnZXRDbGFzcywgZGlhbG9nZm9ybSwKCQlYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJTlVMTCk7CglpZiAoYnV0Y291bnQgPiAwKQoJICAgIFh0VmFTZXRWYWx1ZXMoZGlhbG9ndGV4dGZpZWxkLAoJCSAgICBYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJICAgIFhtTmJvdHRvbVdpZGdldCwgc2VwYXJhdG9yLAoJCSAgICBOVUxMKTsKCWVsc2UKCSAgICBYdFZhU2V0VmFsdWVzKGRpYWxvZ3RleHRmaWVsZCwKCQkgICAgWG1OYm90dG9tQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQkgICAgTlVMTCk7CgoJc2V0X2ZvbnRsaXN0KGRpYWxvZ3RleHRmaWVsZCk7CglYbVRleHRGaWVsZFNldFN0cmluZyhkaWFsb2d0ZXh0ZmllbGQsIChjaGFyICopdGV4dGZpZWxkKTsKCVh0TWFuYWdlQ2hpbGQoZGlhbG9ndGV4dGZpZWxkKTsKCVh0QWRkRXZlbnRIYW5kbGVyKGRpYWxvZ3RleHRmaWVsZCwgS2V5UHJlc3NNYXNrLCBGYWxzZSwKCQkJICAgIChYdEV2ZW50SGFuZGxlcilrZXloaXRfY2FsbGJhY2ssIChYdFBvaW50ZXIpTlVMTCk7CiAgICB9CgogICAgLyogRm9ybSBob2xkaW5nIGJvdGggbWVzc2FnZSBhbmQgcGl4bWFwIGxhYmVscyAqLwogICAgZm9ybSA9IFh0VmFDcmVhdGVXaWRnZXQoInNlcGFyYXRvckZvcm0iLAoJICAgIHhtRm9ybVdpZGdldENsYXNzLCBkaWFsb2dmb3JtLAoJICAgIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIFhtTnJpZ2h0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIE5VTEwpOwogICAgWHRNYW5hZ2VDaGlsZChmb3JtKTsKCiNpZmRlZiBIQVZFX1hQTQogICAgLyogQWRkIGEgcGl4bWFwLCBsZWZ0IG9mIHRoZSBtZXNzYWdlLiAqLwogICAgc3dpdGNoICh0eXBlKQogICAgewoJY2FzZSBWSU1fR0VORVJJQzoKCSAgICBpY29uX2RhdGEgPSBnZW5lcmljX3hwbTsKCSAgICBicmVhazsKCWNhc2UgVklNX0VSUk9SOgoJICAgIGljb25fZGF0YSA9IGVycm9yX3hwbTsKCSAgICBicmVhazsKCWNhc2UgVklNX1dBUk5JTkc6CgkgICAgaWNvbl9kYXRhID0gYWxlcnRfeHBtOwoJICAgIGJyZWFrOwoJY2FzZSBWSU1fSU5GTzoKCSAgICBpY29uX2RhdGEgPSBpbmZvX3hwbTsKCSAgICBicmVhazsKCWNhc2UgVklNX1FVRVNUSU9OOgoJICAgIGljb25fZGF0YSA9IHF1ZXN0X3hwbTsKCSAgICBicmVhazsKCWRlZmF1bHQ6CgkgICAgaWNvbl9kYXRhID0gZ2VuZXJpY194cG07CiAgICB9CgogICAgbiA9IDA7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OdG9wT2Zmc2V0LCA4KTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OYm90dG9tQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbU9mZnNldCwgOCk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNKTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1ObGVmdE9mZnNldCwgOCk7IG4rKzsKCiAgICBkaWFsb2dwaXhtYXAgPSBjcmVhdGVfcGl4bWFwX2xhYmVsKGZvcm0sICJkaWFsb2dQaXhtYXAiLAoJICAgIGljb25fZGF0YSwgYXJncywgbik7CiAgICBYdE1hbmFnZUNoaWxkKGRpYWxvZ3BpeG1hcCk7CiNlbmRpZgoKICAgIC8qIENyZWF0ZSB0aGUgZGlhbG9nIG1lc3NhZ2UuCiAgICAgKiBTaW5jZSBMZXNzVGlmIGlzIGFwcGFyZW50bHkgaGF2aW5nIHByb2JsZW1zIHdpdGggdGhlIGNyZWF0aW9uIG9mCiAgICAgKiBwcm9wZXJseSBsb2NhbGl6ZWQgc3RyaW5nLCB3ZSB1c2UgTHRvUiBoZXJlLiBUaGUgc3ltcHRvbSBpcyB0aGF0IHRoZQogICAgICogc3RyaW5nIHNpbGwgbm90IHNob3cgcHJvcGVybHkgaW4gbXVsdGlwbGUgbGluZXMgYXMgaXQgZG9lcyBpbiBuYXRpdmUKICAgICAqIE1vdGlmLgogICAgICovCiAgICBsYWJlbCA9IFhtU3RyaW5nQ3JlYXRlTHRvUigoY2hhciAqKW1lc3NhZ2UsIFNUUklOR19UQUcpOwogICAgaWYgKGxhYmVsID09IE5VTEwpCglyZXR1cm4gLTE7CiAgICB3ID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoImRpYWxvZ01lc3NhZ2UiLAoJCQkJeG1MYWJlbEdhZGdldENsYXNzLCBmb3JtLAoJCQkJWG1ObGFiZWxTdHJpbmcsIGxhYmVsLAoJCQkJWG1OYWxpZ25tZW50LCBYbUFMSUdOTUVOVF9CRUdJTk5JTkcsCgkJCQlYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQkJWG1OdG9wT2Zmc2V0LCA4LAojaWZkZWYgSEFWRV9YUE0KCQkJCVhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCQlYbU5sZWZ0V2lkZ2V0LCBkaWFsb2dwaXhtYXAsCiNlbHNlCgkJCQlYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKI2VuZGlmCgkJCQlYbU5sZWZ0T2Zmc2V0LCA4LAoJCQkJWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQkJWG1OcmlnaHRPZmZzZXQsIDgsCgkJCQlYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQkJWG1OYm90dG9tT2Zmc2V0LCA4LAoJCQkJTlVMTCk7CiAgICBYbVN0cmluZ0ZyZWUobGFiZWwpOwogICAgc2V0X2ZvbnRsaXN0KHcpOwoKICAgIGlmICh0ZXh0ZmllbGQgIT0gTlVMTCkKICAgIHsKCVh0VmFTZXRWYWx1ZXMoZm9ybSwKCQlYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJWG1OYm90dG9tV2lkZ2V0LCBkaWFsb2d0ZXh0ZmllbGQsCgkJTlVMTCk7CiAgICB9CiAgICBlbHNlCiAgICB7CglpZiAoYnV0Y291bnQgPiAwKQoJICAgIFh0VmFTZXRWYWx1ZXMoZm9ybSwKCQkgICAgWG1OYm90dG9tQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCSAgICBYbU5ib3R0b21XaWRnZXQsIHNlcGFyYXRvciwKCQkgICAgTlVMTCk7CgllbHNlCgkgICAgWHRWYVNldFZhbHVlcyhmb3JtLAoJCSAgICBYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCSAgICBOVUxMKTsKICAgIH0KCiAgICBpZiAoZGZsdGJ1dHRvbiA8IDEpCglkZmx0YnV0dG9uID0gMTsKICAgIGlmIChkZmx0YnV0dG9uID4gYnV0Y291bnQpCglkZmx0YnV0dG9uID0gYnV0Y291bnQ7CiAgICBYdFZhU2V0VmFsdWVzKGRpYWxvZ2Zvcm0sCgkgICAgWG1OZGVmYXVsdEJ1dHRvbiwgYnV0dG9uc1tkZmx0YnV0dG9uIC0gMV0sIE5VTEwpOwogICAgaWYgKHRleHRmaWVsZCAhPSBOVUxMKQoJWHRWYVNldFZhbHVlcyhkaWFsb2dmb3JtLCBYbU5pbml0aWFsRm9jdXMsIGRpYWxvZ3RleHRmaWVsZCwgTlVMTCk7CiAgICBlbHNlCglYdFZhU2V0VmFsdWVzKGRpYWxvZ2Zvcm0sIFhtTmluaXRpYWxGb2N1cywgYnV0dG9uc1tkZmx0YnV0dG9uIC0gMV0sCgkJCQkJCQkJCU5VTEwpOwoKICAgIG1hbmFnZV9jZW50ZXJlZChkaWFsb2dmb3JtKTsKICAgIGFjdGl2YXRlX2RpYWxvZ19tbmVtb25pY3MoZGlhbG9nZm9ybSk7CgogICAgaWYgKHRleHRmaWVsZCAhPSBOVUxMICYmICp0ZXh0ZmllbGQgIT0gTlVMKQogICAgewoJLyogVGhpcyBvbmx5IHdvcmtzIGFmdGVyIHRoZSB0ZXh0ZmllbGQgaGFzIGJlZW4gcmVhbGlzZWQuICovCglYbVRleHRGaWVsZFNldFNlbGVjdGlvbihkaWFsb2d0ZXh0ZmllbGQsCgkJCSAoWG1UZXh0UG9zaXRpb24pMCwgKFhtVGV4dFBvc2l0aW9uKVNUUkxFTih0ZXh0ZmllbGQpLAoJCQkJCSAgIFh0TGFzdFRpbWVzdGFtcFByb2Nlc3NlZChndWkuZHB5KSk7CglYbVRleHRGaWVsZFNldEN1cnNvclBvc2l0aW9uKGRpYWxvZ3RleHRmaWVsZCwKCQkJCQkgICAoWG1UZXh0UG9zaXRpb24pU1RSTEVOKHRleHRmaWVsZCkpOwogICAgfQoKICAgIGFwcCA9IFh0V2lkZ2V0VG9BcHBsaWNhdGlvbkNvbnRleHQoZGlhbG9nZm9ybSk7CgogICAgLyogTG9vcCB1bnRpbCBhIGJ1dHRvbiBpcyBwcmVzc2VkIG9yIHRoZSBkaWFsb2cgaXMga2lsbGVkIHNvbWVob3cuICovCiAgICBkaWFsb2dTdGF0dXMgPSAtMTsKICAgIGZvciAoOzspCiAgICB7CglYdEFwcFByb2Nlc3NFdmVudChhcHAsIChYdElucHV0TWFzaylYdElNQWxsKTsKCWlmIChkaWFsb2dTdGF0dXMgPj0gMCB8fCAhWHRJc01hbmFnZWQoZGlhbG9nZm9ybSkpCgkgICAgYnJlYWs7CiAgICB9CgogICAgdmltX2ZyZWUoYnV0dG9ucyk7CgogICAgaWYgKHRleHRmaWVsZCAhPSBOVUxMKQogICAgewoJcCA9IChjaGFyX3UgKilYbVRleHRHZXRTdHJpbmcoZGlhbG9ndGV4dGZpZWxkKTsKCWlmIChwID09IE5VTEwgfHwgZGlhbG9nU3RhdHVzIDwgMCkKCSAgICAqdGV4dGZpZWxkID0gTlVMOwoJZWxzZQoJICAgIHZpbV9zdHJuY3B5KHRleHRmaWVsZCwgcCwgSU9TSVpFIC0gMSk7CiAgICB9CgogICAgc3VwcHJlc3NfZGlhbG9nX21uZW1vbmljcyhkaWFsb2dmb3JtKTsKICAgIFh0RGVzdHJveVdpZGdldChkaWFsb2dmb3JtKTsKCiAgICByZXR1cm4gZGlhbG9nU3RhdHVzOwp9CiNlbmRpZiAvKiBGRUFUX0dVSV9ESUFMT0cgKi8KCiNpZiBkZWZpbmVkKEZFQVRfRk9PVEVSKSB8fCBkZWZpbmVkKFBST1RPKQoKICAgIHN0YXRpYyBpbnQKZ3VpX21jaF9jb21wdXRlX2Zvb3Rlcl9oZWlnaHQoKQp7CiAgICBEaW1lbnNpb24JaGVpZ2h0OwkJICAgIC8qIHRvdGFsIFRvb2xiYXIgaGVpZ2h0ICovCiAgICBEaW1lbnNpb24JdG9wOwkJICAgIC8qIFhtTm1hcmdpblRvcCAqLwogICAgRGltZW5zaW9uCWJvdHRvbTsJCSAgICAvKiBYbU5tYXJnaW5Cb3R0b20gKi8KICAgIERpbWVuc2lvbglzaGFkb3c7CQkgICAgLyogWG1Oc2hhZG93VGhpY2tuZXNzICovCgogICAgWHRWYUdldFZhbHVlcyhmb290ZXIsCgkgICAgWG1OaGVpZ2h0LCAmaGVpZ2h0LAoJICAgIFhtTm1hcmdpblRvcCwgJnRvcCwKCSAgICBYbU5tYXJnaW5Cb3R0b20sICZib3R0b20sCgkgICAgWG1Oc2hhZG93VGhpY2tuZXNzLCAmc2hhZG93LAoJICAgIE5VTEwpOwoKICAgIHJldHVybiAoaW50KSBoZWlnaHQgKyB0b3AgKyBib3R0b20gKyAoc2hhZG93IDw8IDEpOwp9CgojaWYgMAkgICAgLyogbm90IHVzZWQgKi8KICAgIHZvaWQKZ3VpX21jaF9zZXRfZm9vdGVyX3BvcyhoKQogICAgaW50CSAgICBoOwkJCSAgICAvKiB0ZXh0QXJlYSBoZWlnaHQgKi8KewogICAgWHRWYVNldFZhbHVlcyhmb290ZXIsCgkJICBYbU50b3BPZmZzZXQsIGggKyA3LAoJCSAgTlVMTCk7Cn0KI2VuZGlmCgogICAgdm9pZApndWlfbWNoX2VuYWJsZV9mb290ZXIoc2hvd2l0KQogICAgaW50CQlzaG93aXQ7CnsKICAgIGlmIChzaG93aXQpCiAgICB7CglndWkuZm9vdGVyX2hlaWdodCA9IGd1aV9tY2hfY29tcHV0ZV9mb290ZXJfaGVpZ2h0KCk7CglYdE1hbmFnZUNoaWxkKGZvb3Rlcik7CiAgICB9CiAgICBlbHNlCiAgICB7CglndWkuZm9vdGVyX2hlaWdodCA9IDA7CglYdFVubWFuYWdlQ2hpbGQoZm9vdGVyKTsKICAgIH0KICAgIFh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLCBYbU5ib3R0b21PZmZzZXQsIGd1aS5mb290ZXJfaGVpZ2h0LCBOVUxMKTsKfQoKICAgIHZvaWQKZ3VpX21jaF9zZXRfZm9vdGVyKHMpCiAgICBjaGFyX3UJKnM7CnsKICAgIFhtU3RyaW5nCXhtczsKCiAgICB4bXMgPSBYbVN0cmluZ0NyZWF0ZSgoY2hhciAqKXMsIFNUUklOR19UQUcpOwogICAgaWYgKHhtcyAhPSBOVUxMKQogICAgewoJWHRWYVNldFZhbHVlcyhmb290ZXIsIFhtTmxhYmVsU3RyaW5nLCB4bXMsIE5VTEwpOwoJWG1TdHJpbmdGcmVlKHhtcyk7CiAgICB9Cn0KCiNlbmRpZgoKCiNpZiBkZWZpbmVkKEZFQVRfVE9PTEJBUikgfHwgZGVmaW5lZChQUk9UTykKICAgIHZvaWQKZ3VpX21jaF9zaG93X3Rvb2xiYXIoaW50IHNob3dpdCkKewogICAgQ2FyZGluYWwJbnVtQ2hpbGRyZW47CSAgICAvKiBob3cgbWFueSBjaGlsZHJlbiB0b29sQmFyIGhhcyAqLwoKICAgIGlmICh0b29sQmFyID09IChXaWRnZXQpMCkKCXJldHVybjsKICAgIFh0VmFHZXRWYWx1ZXModG9vbEJhciwgWG1ObnVtQ2hpbGRyZW4sICZudW1DaGlsZHJlbiwgTlVMTCk7CiAgICBpZiAoc2hvd2l0ICYmIG51bUNoaWxkcmVuID4gMCkKICAgIHsKCS8qIEFzc3VtZSB0aGF0IHdlIHdhbnQgdG8gc2hvdyB0aGUgdG9vbGJhciBpZiBwX3Rvb2xiYXIgY29udGFpbnMKCSAqIHZhbGlkIG9wdGlvbiBzZXR0aW5ncywgdGhlcmVmb3JlIHBfdG9vbGJhciBtdXN0IG5vdCBiZSBOVUxMLgoJICovCglXaWRnZXRMaXN0ICBjaGlsZHJlbjsKCglYdFZhR2V0VmFsdWVzKHRvb2xCYXIsIFhtTmNoaWxkcmVuLCAmY2hpbGRyZW4sIE5VTEwpOwoJewoJICAgIHZvaWQgICAgKCphY3Rpb24pKEJhbGxvb25FdmFsICopOwoJICAgIGludAkgICAgdGV4dCA9IDA7CgoJICAgIGlmIChzdHJzdHIoKGNvbnN0IGNoYXIgKilwX3Rvb2xiYXIsICJ0b29sdGlwcyIpKQoJCWFjdGlvbiA9ICZndWlfbWNoX2VuYWJsZV9iZXZhbF9hcmVhOwoJICAgIGVsc2UKCQlhY3Rpb24gPSAmZ3VpX21jaF9kaXNhYmxlX2JldmFsX2FyZWE7CgkgICAgaWYgKHN0cnN0cigoY29uc3QgY2hhciAqKXBfdG9vbGJhciwgInRleHQiKSkKCQl0ZXh0ID0gMTsKCSAgICBlbHNlIGlmIChzdHJzdHIoKGNvbnN0IGNoYXIgKilwX3Rvb2xiYXIsICJpY29ucyIpKQoJCXRleHQgPSAtMTsKCSAgICBpZiAodGV4dCAhPSAwKQoJICAgIHsKCQl2aW1tZW51X1QgICAqdG9vbGJhcjsKCQl2aW1tZW51X1QgICAqY3VyOwoKCQlmb3IgKHRvb2xiYXIgPSByb290X21lbnU7IHRvb2xiYXI7IHRvb2xiYXIgPSB0b29sYmFyLT5uZXh0KQoJCSAgICBpZiAobWVudV9pc190b29sYmFyKHRvb2xiYXItPmRuYW1lKSkKCQkJYnJlYWs7CgkJLyogQXNzdW1wdGlvbjogdG9vbGJhciBpcyBOVUxMIGlmIHRoZXJlIGlzIG5vIHRvb2xiYXIsCgkJICoJICAgICAgIG90aGVyd2lzZSBpdCBjb250YWlucyB0aGUgdG9vbGJhciBtZW51IHN0cnVjdHVyZS4KCQkgKgoJCSAqIEFzc3VtcHRpb246ICJudW1DaGlsZHJlbiIgPT0gdGhlIG51bWJlciBvZiBpdGVtcyBpbiB0aGUgbGlzdAoJCSAqCSAgICAgICBvZiBpdGVtcyBiZWdpbm5pbmcgd2l0aCB0b29sYmFyLT5jaGlsZHJlbi4KCQkgKi8KCQlpZiAodG9vbGJhcikKCQl7CgkJICAgIGZvciAoY3VyID0gdG9vbGJhci0+Y2hpbGRyZW47IGN1cjsgY3VyID0gY3VyLT5uZXh0KQoJCSAgICB7CgkJCUFyZwkgICAgYXJnc1sxXTsKCQkJaW50CSAgICBuID0gMDsKCgkJCS8qIEVuYWJsZS9EaXNhYmxlIHRvb2x0aXAgKE9LIHRvIGVuYWJsZSB3aGlsZQoJCQkgKiBjdXJyZW50bHkgZW5hYmxlZCkuICovCgkJCWlmIChjdXItPnRpcCAhPSBOVUxMKQoJCQkgICAgKCphY3Rpb24pKGN1ci0+dGlwKTsKCQkJaWYgKCFtZW51X2lzX3NlcGFyYXRvcihjdXItPm5hbWUpKQoJCQl7CgkJCSAgICBpZiAodGV4dCA9PSAxIHx8IGN1ci0+eHBtID09IE5VTEwpCgkJCSAgICB7CgkJCQlYdFNldEFyZyhhcmdzW25dLCBYbU5sYWJlbFR5cGUsIFhtU1RSSU5HKTsKCQkJCSsrbjsKCQkJICAgIH0KCQkJICAgIGlmIChjdXItPmlkICE9IE5VTEwpCgkJCSAgICB7CgkJCQlYdFVubWFuYWdlQ2hpbGQoY3VyLT5pZCk7CgkJCQlYdFNldFZhbHVlcyhjdXItPmlkLCBhcmdzLCBuKTsKCQkJCVh0TWFuYWdlQ2hpbGQoY3VyLT5pZCk7CgkJCSAgICB9CgkJCX0KCQkgICAgfQoJCX0KCSAgICB9Cgl9CglndWkudG9vbGJhcl9oZWlnaHQgPSBndWlfbWNoX2NvbXB1dGVfdG9vbGJhcl9oZWlnaHQoKTsKCVh0TWFuYWdlQ2hpbGQoWHRQYXJlbnQodG9vbEJhcikpOwojaWZkZWYgRkVBVF9HVUlfVEFCTElORQoJaWYgKHNob3dpbmdfdGFibGluZSkKCXsKCSAgICBYdFZhU2V0VmFsdWVzKHRhYkxpbmUsCgkJCSAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgIFhtTnRvcFdpZGdldCwgWHRQYXJlbnQodG9vbEJhciksCgkJCSAgTlVMTCk7CgkgICAgWHRWYVNldFZhbHVlcyh0ZXh0QXJlYUZvcm0sCgkJCSAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgIFhtTnRvcFdpZGdldCwgdGFiTGluZSwKCQkJICBOVUxMKTsKCX0KCWVsc2UKI2VuZGlmCgkgICAgWHRWYVNldFZhbHVlcyh0ZXh0QXJlYUZvcm0sCgkJCSAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgIFhtTnRvcFdpZGdldCwgWHRQYXJlbnQodG9vbEJhciksCgkJCSAgTlVMTCk7CglpZiAoWHRJc01hbmFnZWQobWVudUJhcikpCgkgICAgWHRWYVNldFZhbHVlcyhYdFBhcmVudCh0b29sQmFyKSwKCQkgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCSAgICBYbU50b3BXaWRnZXQsIG1lbnVCYXIsCgkJICAgIE5VTEwpOwoJZWxzZQoJICAgIFh0VmFTZXRWYWx1ZXMoWHRQYXJlbnQodG9vbEJhciksCgkJICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJICAgIE5VTEwpOwogICAgfQogICAgZWxzZQogICAgewoJZ3VpLnRvb2xiYXJfaGVpZ2h0ID0gMDsKCWlmIChYdElzTWFuYWdlZChtZW51QmFyKSkKCXsKI2lmZGVmIEZFQVRfR1VJX1RBQkxJTkUKCSAgICBpZiAoc2hvd2luZ190YWJsaW5lKQoJICAgIHsKCQlYdFZhU2V0VmFsdWVzKHRhYkxpbmUsCgkJCSAgICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQkJICAgICAgWG1OdG9wV2lkZ2V0LCBtZW51QmFyLAoJCQkgICAgICBOVUxMKTsKCQlYdFZhU2V0VmFsdWVzKHRleHRBcmVhRm9ybSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIHRhYkxpbmUsCgkJCSAgICAgIE5VTEwpOwoJICAgIH0KCSAgICBlbHNlCiNlbmRpZgoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICAgIFhtTnRvcFdpZGdldCwgbWVudUJhciwKCQkJICAgICAgTlVMTCk7Cgl9CgllbHNlCgl7CiNpZmRlZiBGRUFUX0dVSV9UQUJMSU5FCgkgICAgaWYgKHNob3dpbmdfdGFibGluZSkKCSAgICB7CgkJWHRWYVNldFZhbHVlcyh0YWJMaW5lLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQkgICAgICBOVUxMKTsKCQlYdFZhU2V0VmFsdWVzKHRleHRBcmVhRm9ybSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIHRhYkxpbmUsCgkJCSAgICAgIE5VTEwpOwoJICAgIH0KCSAgICBlbHNlCiNlbmRpZgoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCQkgICAgICBOVUxMKTsKCX0KCglYdFVubWFuYWdlQ2hpbGQoWHRQYXJlbnQodG9vbEJhcikpOwogICAgfQogICAgZ3VpX3NldF9zaGVsbHNpemUoRkFMU0UsIEZBTFNFLCBSRVNJWkVfVkVSVCk7Cn0KCi8qCiAqIEEgdG9vbGJhciBidXR0b24gaGFzIGJlZW4gcHVzaGVkOyBub3cgcmVzZXQgdGhlIGlucHV0IGZvY3VzCiAqIHN1Y2ggdGhhdCB0aGUgdXNlciBjYW4gdHlwZSBwYWdlIHVwL2Rvd24gZXRjLiBhbmQgaGF2ZSB0aGUKICogaW5wdXQgZ28gdG8gdGhlIGVkaXRvciB3aW5kb3csIG5vdCB0aGUgYnV0dG9uCiAqLwogICAgc3RhdGljIHZvaWQKcmVzZXRfZm9jdXMoKQp7CiAgICBpZiAodGV4dEFyZWEgIT0gTlVMTCkKCVhtUHJvY2Vzc1RyYXZlcnNhbCh0ZXh0QXJlYSwgWG1UUkFWRVJTRV9DVVJSRU5UKTsKfQoKICAgIGludApndWlfbWNoX2NvbXB1dGVfdG9vbGJhcl9oZWlnaHQoKQp7CiAgICBEaW1lbnNpb24JYm9yZGVyczsKICAgIERpbWVuc2lvbgloZWlnaHQ7CQkgICAgLyogdG90YWwgVG9vbGJhciBoZWlnaHQgKi8KICAgIERpbWVuc2lvbgl3aGd0OwkJICAgIC8qIGhlaWdodCBvZiBlYWNoIHdpZGdldCAqLwogICAgV2lkZ2V0TGlzdAljaGlsZHJlbjsJICAgIC8qIGxpc3Qgb2YgdG9vbEJhcidzIGNoaWxkcmVuICovCiAgICBDYXJkaW5hbAludW1DaGlsZHJlbjsJICAgIC8qIGhvdyBtYW55IGNoaWxkcmVuIHRvb2xCYXIgaGFzICovCiAgICBpbnQJCWk7CgogICAgYm9yZGVycyA9IDA7CiAgICBoZWlnaHQgPSAwOwogICAgaWYgKHRvb2xCYXIgIT0gKFdpZGdldCkwICYmIHRvb2xCYXJGcmFtZSAhPSAoV2lkZ2V0KTApCiAgICB7CQkJCSAgICAvKiBnZXQgaGVpZ2h0IG9mIFhtRnJhbWUgcGFyZW50ICovCglEaW1lbnNpb24JZnN0OwoJRGltZW5zaW9uCWZtaDsKCURpbWVuc2lvbgl0c3Q7CglEaW1lbnNpb24JdG1oOwoKCVh0VmFHZXRWYWx1ZXModG9vbEJhckZyYW1lLAoJCVhtTnNoYWRvd1RoaWNrbmVzcywgJmZzdCwKCQlYbU5tYXJnaW5IZWlnaHQsICZmbWgsCgkJTlVMTCk7Cglib3JkZXJzICs9IGZzdCArIGZtaDsKCVh0VmFHZXRWYWx1ZXModG9vbEJhciwKCQlYbU5zaGFkb3dUaGlja25lc3MsICZ0c3QsCgkJWG1ObWFyZ2luSGVpZ2h0LCAmdG1oLAoJCVhtTmNoaWxkcmVuLCAmY2hpbGRyZW4sCgkJWG1ObnVtQ2hpbGRyZW4sICZudW1DaGlsZHJlbiwgTlVMTCk7Cglib3JkZXJzICs9IHRzdCArIHRtaDsKCWZvciAoaSA9IDA7IGkgPCAoaW50KW51bUNoaWxkcmVuOyBpKyspCgl7CgkgICAgd2hndCA9IDA7CgkgICAgWHRWYUdldFZhbHVlcyhjaGlsZHJlbltpXSwgWG1OaGVpZ2h0LCAmd2hndCwgTlVMTCk7CgkgICAgaWYgKGhlaWdodCA8IHdoZ3QpCgkJaGVpZ2h0ID0gd2hndDsKCX0KICAgIH0KI2lmZGVmIExFU1NUSUZfVkVSU0lPTgogICAgLyogSGFjazogV2hlbiBzdGFydGluZyB1cCB3ZSBnZXQgd3JvbmcgZGltZW5zaW9ucy4gKi8KICAgIGlmIChoZWlnaHQgPCAxMCkKCWhlaWdodCA9IDI0OwojZW5kaWYKCiAgICByZXR1cm4gKGludCkoaGVpZ2h0ICsgKGJvcmRlcnMgPDwgMSkpOwp9CgogICAgdm9pZAptb3RpZl9nZXRfdG9vbGJhcl9jb2xvcnMoYmdwLCBmZ3AsIGJzcCwgdHNwLCBoc3ApCiAgICBQaXhlbCAgICAgICAqYmdwOwogICAgUGl4ZWwgICAgICAgKmZncDsKICAgIFBpeGVsICAgICAgICpic3A7CiAgICBQaXhlbCAgICAgICAqdHNwOwogICAgUGl4ZWwgICAgICAgKmhzcDsKewogICAgWHRWYUdldFZhbHVlcyh0b29sQmFyLAoJICAgIFhtTmJhY2tncm91bmQsIGJncCwKCSAgICBYbU5mb3JlZ3JvdW5kLCBmZ3AsCgkgICAgWG1OYm90dG9tU2hhZG93Q29sb3IsIGJzcCwKCSAgICBYbU50b3BTaGFkb3dDb2xvciwgdHNwLAoJICAgIFhtTmhpZ2hsaWdodENvbG9yLCBoc3AsCgkgICAgTlVMTCk7Cn0KCiMgaWZkZWYgRkVBVF9GT09URVIKLyoKICogVGhlIG5leHQgdG9vbGJhciBlbnRlci9sZWF2ZSBjYWxsYmFja3Mgc2hvdWxkIHJlYWxseSBkbyBiYWxsb29uIGhlbHAuICBCdXQKICogSSBoYXZlIHRvIHVzZSBmb290ZXIgaGVscCBmb3IgYmFja3dhcmRzIGNvbXBhdGFiaWxpdHkuICBIb3BlZnVsbHkgYm90aCB3aWxsCiAqIGdldCBpbXBsZW1lbnRlZCBhbmQgdGhlIHVzZXIgd2lsbCBoYXZlIGEgY2hvaWNlLgogKi8KICAgIHN0YXRpYyB2b2lkCnRvb2xiYXJidXR0b25fZW50ZXJfY2IodywgY2xpZW50X2RhdGEsIGV2ZW50LCBjb250KQogICAgV2lkZ2V0CXcgVU5VU0VEOwogICAgWHRQb2ludGVyCWNsaWVudF9kYXRhOwogICAgWEV2ZW50CSpldmVudCBVTlVTRUQ7CiAgICBCb29sZWFuCSpjb250IFVOVVNFRDsKewogICAgdmltbWVudV9UCSptZW51ID0gKHZpbW1lbnVfVCAqKSBjbGllbnRfZGF0YTsKCiAgICBpZiAobWVudS0+c3RyaW5nc1tNRU5VX0lOREVYX1RJUF0gIT0gTlVMTCkKICAgIHsKCWlmICh2aW1fc3RyY2hyKHBfZ28sIEdPX0ZPT1RFUikgIT0gTlVMTCkKCSAgICBndWlfbWNoX3NldF9mb290ZXIobWVudS0+c3RyaW5nc1tNRU5VX0lOREVYX1RJUF0pOwogICAgfQp9CgogICAgc3RhdGljIHZvaWQKdG9vbGJhcmJ1dHRvbl9sZWF2ZV9jYih3LCBjbGllbnRfZGF0YSwgZXZlbnQsIGNvbnQpCiAgICBXaWRnZXQJdyBVTlVTRUQ7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGEgVU5VU0VEOwogICAgWEV2ZW50CSpldmVudCBVTlVTRUQ7CiAgICBCb29sZWFuCSpjb250IFVOVVNFRDsKewogICAgZ3VpX21jaF9zZXRfZm9vdGVyKChjaGFyX3UgKikgIiIpOwp9CiMgZW5kaWYKI2VuZGlmCgojaWYgZGVmaW5lZChGRUFUX0dVSV9UQUJMSU5FKSB8fCBkZWZpbmVkKFBST1RPKQovKgogKiBTaG93IG9yIGhpZGUgdGhlIHRhYmxpbmUuCiAqLwogICAgdm9pZApndWlfbWNoX3Nob3dfdGFibGluZShpbnQgc2hvd2l0KQp7CiAgICBpZiAodGFiTGluZSA9PSAoV2lkZ2V0KTApCglyZXR1cm47CgogICAgaWYgKCFzaG93aXQgIT0gIXNob3dpbmdfdGFibGluZSkKICAgIHsKCWlmIChzaG93aXQpCgl7CgkgICAgWHRNYW5hZ2VDaGlsZCh0YWJMaW5lKTsKCSAgICBYdFVubWFuYWdlQ2hpbGQoWHROYW1lVG9XaWRnZXQodGFiTGluZSwgIlBhZ2VTY3JvbGxlciIpKTsKCSAgICBYdFVubWFuYWdlQ2hpbGQoWHROYW1lVG9XaWRnZXQodGFiTGluZSwgIk1pbm9yVGFiU2Nyb2xsZXJOZXh0IikpOwoJICAgIFh0VW5tYW5hZ2VDaGlsZChYdE5hbWVUb1dpZGdldCh0YWJMaW5lLAoJCQkJCSAgICJNaW5vclRhYlNjcm9sbGVyUHJldmlvdXMiKSk7CiNpZmRlZiBGRUFUX01FTlUKIyBpZmRlZiBGRUFUX1RPT0xCQVIKCSAgICBpZiAoWHRJc01hbmFnZWQoWHRQYXJlbnQodG9vbEJhcikpKQoJCVh0VmFTZXRWYWx1ZXModGFiTGluZSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIFh0UGFyZW50KHRvb2xCYXIpLCBOVUxMKTsKCSAgICBlbHNlCiMgZW5kaWYKCQlpZiAoWHRJc01hbmFnZWQobWVudUJhcikpCgkJWHRWYVNldFZhbHVlcyh0YWJMaW5lLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJCSAgICAgIFhtTnRvcFdpZGdldCwgbWVudUJhciwgTlVMTCk7CgkgICAgZWxzZQojZW5kaWYKCQlYdFZhU2V0VmFsdWVzKHRhYkxpbmUsCgkJCSAgICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sIE5VTEwpOwoJICAgIFh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQkJICBYbU50b3BXaWRnZXQsIHRhYkxpbmUsCgkJCSAgTlVMTCk7Cgl9CgllbHNlCgl7CgkgICAgWHRVbm1hbmFnZUNoaWxkKHRhYkxpbmUpOwojaWZkZWYgRkVBVF9NRU5VCiMgaWZkZWYgRkVBVF9UT09MQkFSCgkgICAgaWYgKFh0SXNNYW5hZ2VkKFh0UGFyZW50KHRvb2xCYXIpKSkKCQlYdFZhU2V0VmFsdWVzKHRleHRBcmVhRm9ybSwKCQkJICAgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCQkgICAgICBYbU50b3BXaWRnZXQsIFh0UGFyZW50KHRvb2xCYXIpLCBOVUxMKTsKCSAgICBlbHNlCiMgZW5kaWYKCQlpZiAoWHRJc01hbmFnZWQobWVudUJhcikpCgkJWHRWYVNldFZhbHVlcyh0ZXh0QXJlYUZvcm0sCgkJCSAgICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQkJICAgICAgWG1OdG9wV2lkZ2V0LCBtZW51QmFyLCBOVUxMKTsKCSAgICBlbHNlCiNlbmRpZgoJCVh0VmFTZXRWYWx1ZXModGV4dEFyZWFGb3JtLAoJCQkgICAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLCBOVUxMKTsKCX0KCXNob3dpbmdfdGFibGluZSA9IHNob3dpdDsKICAgIH0KfQoKLyoKICogUmV0dXJuIFRSVUUgd2hlbiB0YWJsaW5lIGlzIGRpc3BsYXllZC4KICovCiAgICBpbnQKZ3VpX21jaF9zaG93aW5nX3RhYmxpbmUodm9pZCkKewogICAgcmV0dXJuIHRhYkxpbmUgIT0gKFdpZGdldCkwICYmIHNob3dpbmdfdGFibGluZTsKfQoKLyoKICogVXBkYXRlIHRoZSBsYWJlbHMgb2YgdGhlIHRhYmxpbmUuCiAqLwogICAgdm9pZApndWlfbWNoX3VwZGF0ZV90YWJsaW5lKHZvaWQpCnsKICAgIHRhYnBhZ2VfVAkJKnRwOwogICAgaW50CQkJbnIgPSAxLCBuOwogICAgQXJnCQkJYXJnc1sxMF07CiAgICBpbnQJCQljdXJ0YWJpZHggPSAwLCBjdXJyZW50cGFnZTsKICAgIFdpZGdldAkJdGFiOwogICAgWG1Ob3RlYm9va1BhZ2VJbmZvCXBhZ2VfaW5mbzsKICAgIFhtTm90ZWJvb2tQYWdlU3RhdHVzIHBhZ2Vfc3RhdHVzOwogICAgaW50CQkJbGFzdF9wYWdlLCB0YWJfY291bnQ7CiAgICBYbVN0cmluZwkJbGFiZWxfc3RyOwogICAgY2hhcgkJKmxhYmVsX2NzdHI7CiAgICBCYWxsb29uRXZhbAkJKmJldmFsOwoKICAgIGlmICh0YWJMaW5lID09IChXaWRnZXQpMCkKCXJldHVybjsKCiAgICAvKiBBZGQgYSBsYWJlbCBmb3IgZWFjaCB0YWIgcGFnZS4gIFRoZXkgYWxsIGNvbnRhaW4gdGhlIHNhbWUgdGV4dCBhcmVhLiAqLwogICAgZm9yICh0cCA9IGZpcnN0X3RhYnBhZ2U7IHRwICE9IE5VTEw7IHRwID0gdHAtPnRwX25leHQsICsrbnIpCiAgICB7CglpZiAodHAgPT0gY3VydGFiKQoJICAgIGN1cnRhYmlkeCA9IG5yOwoKCXBhZ2Vfc3RhdHVzID0gWG1Ob3RlYm9va0dldFBhZ2VJbmZvKHRhYkxpbmUsIG5yLCAmcGFnZV9pbmZvKTsKCWlmIChwYWdlX3N0YXR1cyA9PSBYbVBBR0VfSU5WQUxJRAoJCXx8IHBhZ2VfaW5mby5tYWpvcl90YWJfd2lkZ2V0ID09IChXaWRnZXQpMCkKCXsKCSAgICAvKiBBZGQgdGhlIHRhYiAqLwoJICAgIG4gPSAwOwoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTm5vdGVib29rQ2hpbGRUeXBlLCBYbU1BSk9SX1RBQik7IG4rKzsKCSAgICBYdFNldEFyZyhhcmdzW25dLCBYbU50cmF2ZXJzYWxPbiwgRmFsc2UpOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OYWxpZ25tZW50LCBYbUFMSUdOTUVOVF9CRUdJTk5JTkcpOyBuKys7CgkgICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OaGlnaGxpZ2h0VGhpY2tuZXNzLCAxKTsgbisrOwoJICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnNoYWRvd1RoaWNrbmVzcyAsIDEpOyBuKys7CgkgICAgdGFiID0gWG1DcmVhdGVQdXNoQnV0dG9uKHRhYkxpbmUsICItRW1wdHktIiwgYXJncywgbik7CgkgICAgWHRNYW5hZ2VDaGlsZCh0YWIpOwoJICAgIGJldmFsID0gZ3VpX21jaF9jcmVhdGVfYmV2YWxfYXJlYSh0YWIsIE5VTEwsIHRhYmxpbmVfYmFsbG9vbl9jYiwKCQkJCQkJCQkJTlVMTCk7CgkgICAgWHRWYVNldFZhbHVlcyh0YWIsIFhtTnVzZXJEYXRhLCBiZXZhbCwgTlVMTCk7Cgl9CgllbHNlCgkgICAgdGFiID0gcGFnZV9pbmZvLm1ham9yX3RhYl93aWRnZXQ7CgoJWHRWYVNldFZhbHVlcyh0YWIsIFhtTnBhZ2VOdW1iZXIsIG5yLCBOVUxMKTsKCgkvKgoJICogQ2hhbmdlIHRoZSBsYWJlbCB0ZXh0IG9ubHkgaWYgaXQgaXMgZGlmZmVyZW50CgkgKi8KCVh0VmFHZXRWYWx1ZXModGFiLCBYbU5sYWJlbFN0cmluZywgJmxhYmVsX3N0ciwgTlVMTCk7CglpZiAoWG1TdHJpbmdHZXRMdG9SKGxhYmVsX3N0ciwgWG1TVFJJTkdfREVGQVVMVF9DSEFSU0VULCAmbGFiZWxfY3N0cikpCgl7CgkgICAgZ2V0X3RhYmxpbmVfbGFiZWwodHAsIEZBTFNFKTsKCSAgICBpZiAoU1RSQ01QKGxhYmVsX2NzdHIsIE5hbWVCdWZmKSAhPSAwKQoJICAgIHsKCQlYdFZhU2V0VmFsdWVzKHRhYiwgWHRWYVR5cGVkQXJnLCBYbU5sYWJlbFN0cmluZywgWG1SU3RyaW5nLAoJCQkgICAgICBOYW1lQnVmZiwgU1RSTEVOKE5hbWVCdWZmKSArIDEsIE5VTEwpOwoJCS8qCgkJICogRm9yY2UgYSByZXNpemUgb2YgdGhlIHRhYiBsYWJlbCBidXR0b24KCQkgKi8KCQlYdFVubWFuYWdlQ2hpbGQodGFiKTsKCQlYdE1hbmFnZUNoaWxkKHRhYik7CgkgICAgfQoJICAgIFh0RnJlZShsYWJlbF9jc3RyKTsKCX0KICAgIH0KCiAgICB0YWJfY291bnQgPSBuciAtIDE7CgogICAgWHRWYUdldFZhbHVlcyh0YWJMaW5lLCBYbU5sYXN0UGFnZU51bWJlciwgJmxhc3RfcGFnZSwgTlVMTCk7CgogICAgLyogUmVtb3ZlIGFueSBvbGQgbGFiZWxzLiAqLwogICAgd2hpbGUgKG5yIDw9IGxhc3RfcGFnZSkKICAgIHsKCWlmIChYbU5vdGVib29rR2V0UGFnZUluZm8odGFiTGluZSwgbnIsICZwYWdlX2luZm8pICE9IFhtUEFHRV9JTlZBTElECgkgICAgJiYgcGFnZV9pbmZvLnBhZ2VfbnVtYmVyID09IG5yCgkgICAgJiYgcGFnZV9pbmZvLm1ham9yX3RhYl93aWRnZXQgIT0gKFdpZGdldCkwKQoJewoJICAgIFh0VmFHZXRWYWx1ZXMocGFnZV9pbmZvLm1ham9yX3RhYl93aWRnZXQsIFhtTnVzZXJEYXRhLCAmYmV2YWwsIE5VTEwpOwoJICAgIGlmIChiZXZhbCAhPSBOVUxMKQoJCWd1aV9tY2hfZGVzdHJveV9iZXZhbF9hcmVhKGJldmFsKTsKCSAgICBYdFVubWFuYWdlQ2hpbGQocGFnZV9pbmZvLm1ham9yX3RhYl93aWRnZXQpOwoJICAgIFh0RGVzdHJveVdpZGdldChwYWdlX2luZm8ubWFqb3JfdGFiX3dpZGdldCk7Cgl9CglucisrOwogICAgfQoKICAgIFh0VmFTZXRWYWx1ZXModGFiTGluZSwgWG1ObGFzdFBhZ2VOdW1iZXIsIHRhYl9jb3VudCwgTlVMTCk7CgogICAgWHRWYUdldFZhbHVlcyh0YWJMaW5lLCBYbU5jdXJyZW50UGFnZU51bWJlciwgJmN1cnJlbnRwYWdlLCBOVUxMKTsKICAgIGlmIChjdXJyZW50cGFnZSAhPSBjdXJ0YWJpZHgpCglYdFZhU2V0VmFsdWVzKHRhYkxpbmUsIFhtTmN1cnJlbnRQYWdlTnVtYmVyLCBjdXJ0YWJpZHgsIE5VTEwpOwp9CgovKgogKiBTZXQgdGhlIGN1cnJlbnQgdGFiIHRvICJuciIuICBGaXJzdCB0YWIgaXMgMS4KICovCiAgICB2b2lkCmd1aV9tY2hfc2V0X2N1cnRhYihucikKICAgIGludAkJbnI7CnsKICAgIGludAkJY3VycmVudHBhZ2U7CgogICAgaWYgKHRhYkxpbmUgPT0gKFdpZGdldCkwKQoJcmV0dXJuOwoKICAgIFh0VmFHZXRWYWx1ZXModGFiTGluZSwgWG1OY3VycmVudFBhZ2VOdW1iZXIsICZjdXJyZW50cGFnZSwgTlVMTCk7CiAgICBpZiAoY3VycmVudHBhZ2UgIT0gbnIpCglYdFZhU2V0VmFsdWVzKHRhYkxpbmUsIFhtTmN1cnJlbnRQYWdlTnVtYmVyLCBuciwgTlVMTCk7Cn0KI2VuZGlmCgovKgogKiBTZXQgdGhlIGNvbG9ycyBvZiBXaWRnZXQgImlkIiB0byB0aGUgbWVudSBjb2xvcnMuCiAqLwogICAgc3RhdGljIHZvaWQKZ3VpX21vdGlmX21lbnVfY29sb3JzKGlkKQogICAgV2lkZ2V0ICBpZDsKewogICAgaWYgKGd1aS5tZW51X2JnX3BpeGVsICE9IElOVkFMQ09MT1IpCiNpZiAoWG1WZXJzaW9uID49IDEwMDIpCglYbUNoYW5nZUNvbG9yKGlkLCBndWkubWVudV9iZ19waXhlbCk7CiNlbHNlCglYdFZhU2V0VmFsdWVzKGlkLCBYbU5iYWNrZ3JvdW5kLCBndWkubWVudV9iZ19waXhlbCwgTlVMTCk7CiNlbmRpZgogICAgaWYgKGd1aS5tZW51X2ZnX3BpeGVsICE9IElOVkFMQ09MT1IpCglYdFZhU2V0VmFsdWVzKGlkLCBYbU5mb3JlZ3JvdW5kLCBndWkubWVudV9mZ19waXhlbCwgTlVMTCk7Cn0KCi8qCiAqIFNldCB0aGUgY29sb3JzIG9mIFdpZGdldCAiaWQiIHRvIHRoZSBzY3JvbGxiYXIgY29sb3JzLgogKi8KICAgIHN0YXRpYyB2b2lkCmd1aV9tb3RpZl9zY3JvbGxfY29sb3JzKGlkKQogICAgV2lkZ2V0ICBpZDsKewogICAgaWYgKGd1aS5zY3JvbGxfYmdfcGl4ZWwgIT0gSU5WQUxDT0xPUikKI2lmIChYbVZlcnNpb24gPj0gMTAwMikKCVhtQ2hhbmdlQ29sb3IoaWQsIGd1aS5zY3JvbGxfYmdfcGl4ZWwpOwojZWxzZQoJWHRWYVNldFZhbHVlcyhpZCwgWG1OYmFja2dyb3VuZCwgZ3VpLnNjcm9sbF9iZ19waXhlbCwgTlVMTCk7CiNlbmRpZgogICAgaWYgKGd1aS5zY3JvbGxfZmdfcGl4ZWwgIT0gSU5WQUxDT0xPUikKCVh0VmFTZXRWYWx1ZXMoaWQsIFhtTmZvcmVncm91bmQsIGd1aS5zY3JvbGxfZmdfcGl4ZWwsIE5VTEwpOwp9CgovKgogKiBTZXQgdGhlIGZvbnRsaXN0IGZvciBXaWRnZXQgImlkIiB0byB1c2UgZ3VpLm1lbnVfZm9udHNldCBvciBndWkubWVudV9mb250LgogKi8KICAgIHZvaWQKZ3VpX21vdGlmX21lbnVfZm9udGxpc3QoaWQpCiAgICBXaWRnZXQgIGlkIFVOVVNFRDsKewojaWZkZWYgRkVBVF9NRU5VCiNpZmRlZiBGT05UU0VUX0FMV0FZUwogICAgaWYgKGd1aS5tZW51X2ZvbnRzZXQgIT0gTk9GT05UU0VUKQogICAgewoJWG1Gb250TGlzdCBmbDsKCglmbCA9IGd1aV9tb3RpZl9mb250c2V0MmZvbnRsaXN0KChYRm9udFNldCAqKSZndWkubWVudV9mb250c2V0KTsKCWlmIChmbCAhPSBOVUxMKQoJewoJICAgIGlmIChYdElzTWFuYWdlZChpZCkpCgkgICAgewoJCVh0VW5tYW5hZ2VDaGlsZChpZCk7CgkJWHRWYVNldFZhbHVlcyhpZCwgWG1OZm9udExpc3QsIGZsLCBOVUxMKTsKCQkvKiBXZSBzaG91bGQgZm9yY2UgdGhlIHdpZGdldCB0byByZWNhbGN1bGF0ZSBpdCdzCgkJICogZ2VvbWV0cnkgbm93LiAqLwoJCVh0TWFuYWdlQ2hpbGQoaWQpOwoJICAgIH0KCSAgICBlbHNlCgkJWHRWYVNldFZhbHVlcyhpZCwgWG1OZm9udExpc3QsIGZsLCBOVUxMKTsKCSAgICBYbUZvbnRMaXN0RnJlZShmbCk7Cgl9CiAgICB9CiNlbHNlCiAgICBpZiAoZ3VpLm1lbnVfZm9udCAhPSBOT0ZPTlQpCiAgICB7CglYbUZvbnRMaXN0IGZsOwoKCWZsID0gZ3VpX21vdGlmX2NyZWF0ZV9mb250bGlzdCgoWEZvbnRTdHJ1Y3QgKilndWkubWVudV9mb250KTsKCWlmIChmbCAhPSBOVUxMKQoJewoJICAgIGlmIChYdElzTWFuYWdlZChpZCkpCgkgICAgewoJCVh0VW5tYW5hZ2VDaGlsZChpZCk7CgkJWHRWYVNldFZhbHVlcyhpZCwgWG1OZm9udExpc3QsIGZsLCBOVUxMKTsKCQkvKiBXZSBzaG91bGQgZm9yY2UgdGhlIHdpZGdldCB0byByZWNhbGN1bGF0ZSBpdCdzCgkJICogZ2VvbWV0cnkgbm93LiAqLwoJCVh0TWFuYWdlQ2hpbGQoaWQpOwoJICAgIH0KCSAgICBlbHNlCgkJWHRWYVNldFZhbHVlcyhpZCwgWG1OZm9udExpc3QsIGZsLCBOVUxMKTsKCSAgICBYbUZvbnRMaXN0RnJlZShmbCk7Cgl9CiAgICB9CiNlbmRpZgojZW5kaWYKfQoKCi8qCiAqIFdlIGRvbid0IGNyZWF0ZSBpdCB0d2ljZSBmb3IgdGhlIHNha2Ugb2Ygc3BlZWQuCiAqLwoKdHlwZWRlZiBzdHJ1Y3QgX1NoYXJlZEZpbmRSZXBsYWNlCnsKICAgIFdpZGdldCBkaWFsb2c7CS8qIHRoZSBtYWluIGRpYWxvZyB3aWRnZXQgKi8KICAgIFdpZGdldCB3d29yZDsJLyogJ0V4YWN0IG1hdGNoJyBjaGVjayBidXR0b24gKi8KICAgIFdpZGdldCBtY2FzZTsJLyogJ21hdGNoIGNhc2UnIGNoZWNrIGJ1dHRvbiAqLwogICAgV2lkZ2V0IHVwOwkJLyogc2VhcmNoIGRpcmVjdGlvbiAnVXAnIHJhZGlvIGJ1dHRvbiAqLwogICAgV2lkZ2V0IGRvd247CS8qIHNlYXJjaCBkaXJlY3Rpb24gJ0Rvd24nIHJhZGlvIGJ1dHRvbiAqLwogICAgV2lkZ2V0IHdoYXQ7CS8qICdGaW5kIHdoYXQnIGVudHJ5IHRleHQgd2lkZ2V0ICovCiAgICBXaWRnZXQgd2l0aDsJLyogJ1JlcGxhY2Ugd2l0aCcgZW50cnkgdGV4dCB3aWRnZXQgKi8KICAgIFdpZGdldCBmaW5kOwkvKiAnRmluZCBOZXh0JyBhY3Rpb24gYnV0dG9uICovCiAgICBXaWRnZXQgcmVwbGFjZTsJLyogJ1JlcGxhY2UgV2l0aCcgYWN0aW9uIGJ1dHRvbiAqLwogICAgV2lkZ2V0IGFsbDsJCS8qICdSZXBsYWNlIEFsbCcgYWN0aW9uIGJ1dHRvbiAqLwogICAgV2lkZ2V0IHVuZG87CS8qICdVbmRvJyBhY3Rpb24gYnV0dG9uICovCgogICAgV2lkZ2V0IGNhbmNlbDsKfSBTaGFyZWRGaW5kUmVwbGFjZTsKCnN0YXRpYyBTaGFyZWRGaW5kUmVwbGFjZSBmaW5kX3dpZGdldHMgPSB7TlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTH07CnN0YXRpYyBTaGFyZWRGaW5kUmVwbGFjZSByZXBsX3dpZGdldHMgPSB7TlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTH07CgpzdGF0aWMgdm9pZCBmaW5kX3JlcGxhY2VfZGVzdHJveV9jYWxsYmFjayBfX0FSR1MoKFdpZGdldCB3LCBYdFBvaW50ZXIgY2xpZW50X2RhdGEsIFh0UG9pbnRlciBjYWxsX2RhdGEpKTsKc3RhdGljIHZvaWQgZmluZF9yZXBsYWNlX2Rpc21pc3NfY2FsbGJhY2sgX19BUkdTKChXaWRnZXQgdywgWHRQb2ludGVyIGNsaWVudF9kYXRhLCBYdFBvaW50ZXIgY2FsbF9kYXRhKSk7CnN0YXRpYyB2b2lkIGVudHJ5X2FjdGl2YXRlX2NhbGxiYWNrIF9fQVJHUygoV2lkZ2V0IHcsIFh0UG9pbnRlciBjbGllbnRfZGF0YSwgWHRQb2ludGVyIGNhbGxfZGF0YSkpOwpzdGF0aWMgdm9pZCBmaW5kX3JlcGxhY2VfY2FsbGJhY2sgX19BUkdTKChXaWRnZXQgdywgWHRQb2ludGVyIGNsaWVudF9kYXRhLCBYdFBvaW50ZXIgY2FsbF9kYXRhKSk7CnN0YXRpYyB2b2lkIGZpbmRfcmVwbGFjZV9rZXlwcmVzcyBfX0FSR1MoKFdpZGdldCB3LCBTaGFyZWRGaW5kUmVwbGFjZSAqIGZyZHAsIFhLZXlFdmVudCAqIGV2ZW50KSk7CnN0YXRpYyB2b2lkIGZpbmRfcmVwbGFjZV9kaWFsb2dfY3JlYXRlIF9fQVJHUygoY2hhcl91ICplbnRyeV90ZXh0LCBpbnQgZG9fcmVwbGFjZSkpOwoKICAgIHN0YXRpYyB2b2lkCmZpbmRfcmVwbGFjZV9kZXN0cm95X2NhbGxiYWNrKHcsIGNsaWVudF9kYXRhLCBjYWxsX2RhdGEpCiAgICBXaWRnZXQJdyBVTlVTRUQ7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGE7CiAgICBYdFBvaW50ZXIJY2FsbF9kYXRhIFVOVVNFRDsKewogICAgU2hhcmVkRmluZFJlcGxhY2UgKmNkID0gKFNoYXJlZEZpbmRSZXBsYWNlICopY2xpZW50X2RhdGE7CgogICAgaWYgKGNkICE9IE5VTEwpCiAgICAgICAvKiBzdXBwcmVzc19kaWFsb2dfbW5lbW9uaWNzKGNkLT5kaWFsb2cpOyAqLwoJY2QtPmRpYWxvZyA9IChXaWRnZXQpMDsKfQoKICAgIHN0YXRpYyB2b2lkCmZpbmRfcmVwbGFjZV9kaXNtaXNzX2NhbGxiYWNrKHcsIGNsaWVudF9kYXRhLCBjYWxsX2RhdGEpCiAgICBXaWRnZXQJdyBVTlVTRUQ7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGE7CiAgICBYdFBvaW50ZXIJY2FsbF9kYXRhIFVOVVNFRDsKewogICAgU2hhcmVkRmluZFJlcGxhY2UgKmNkID0gKFNoYXJlZEZpbmRSZXBsYWNlICopY2xpZW50X2RhdGE7CgogICAgaWYgKGNkICE9IE5VTEwpCglYdFVubWFuYWdlQ2hpbGQoY2QtPmRpYWxvZyk7Cn0KCiAgICBzdGF0aWMgdm9pZAplbnRyeV9hY3RpdmF0ZV9jYWxsYmFjayh3LCBjbGllbnRfZGF0YSwgY2FsbF9kYXRhKQogICAgV2lkZ2V0CXcgVU5VU0VEOwogICAgWHRQb2ludGVyCWNsaWVudF9kYXRhOwogICAgWHRQb2ludGVyCWNhbGxfZGF0YSBVTlVTRUQ7CnsKICAgIFhtUHJvY2Vzc1RyYXZlcnNhbCgoV2lkZ2V0KWNsaWVudF9kYXRhLCBYbVRSQVZFUlNFX0NVUlJFTlQpOwp9CgogICAgc3RhdGljIHZvaWQKZmluZF9yZXBsYWNlX2NhbGxiYWNrKHcsIGNsaWVudF9kYXRhLCBjYWxsX2RhdGEpCiAgICBXaWRnZXQJdyBVTlVTRUQ7CiAgICBYdFBvaW50ZXIJY2xpZW50X2RhdGE7CiAgICBYdFBvaW50ZXIJY2FsbF9kYXRhIFVOVVNFRDsKewogICAgbG9uZ191CWZsYWdzID0gKGxvbmdfdSljbGllbnRfZGF0YTsKICAgIGNoYXIJKmZpbmRfdGV4dCwgKnJlcGxfdGV4dDsKICAgIEJvb2xlYW4JZGlyZWN0aW9uX2Rvd24gPSBUUlVFOwogICAgQm9vbGVhbgl3d29yZDsKICAgIEJvb2xlYW4JbWNhc2U7CiAgICBTaGFyZWRGaW5kUmVwbGFjZSAqc2ZyOwoKICAgIGlmIChmbGFncyA9PSBGUkRfVU5ETykKICAgIHsKCWNoYXJfdQkqc2F2ZV9jcG8gPSBwX2NwbzsKCgkvKiBObyBuZWVkIHRvIGJlIFZpIGNvbXBhdGlibGUgaGVyZS4gKi8KCXBfY3BvID0gKGNoYXJfdSAqKSIiOwoJdV91bmRvKDEpOwoJcF9jcG8gPSBzYXZlX2NwbzsKCWd1aV91cGRhdGVfc2NyZWVuKCk7CglyZXR1cm47CiAgICB9CgogICAgLyogR2V0IHRoZSBzZWFyY2gvcmVwbGFjZSBzdHJpbmdzIGZyb20gdGhlIGRpYWxvZyAqLwogICAgaWYgKGZsYWdzID09IEZSRF9GSU5ETkVYVCkKICAgIHsKCXJlcGxfdGV4dCA9IE5VTEw7CglzZnIgPSAmZmluZF93aWRnZXRzOwogICAgfQogICAgZWxzZQogICAgewoJcmVwbF90ZXh0ID0gWG1UZXh0RmllbGRHZXRTdHJpbmcocmVwbF93aWRnZXRzLndpdGgpOwoJc2ZyID0gJnJlcGxfd2lkZ2V0czsKICAgIH0KICAgIGZpbmRfdGV4dCA9IFhtVGV4dEZpZWxkR2V0U3RyaW5nKHNmci0+d2hhdCk7CiAgICBYdFZhR2V0VmFsdWVzKHNmci0+ZG93biwgWG1Oc2V0LCAmZGlyZWN0aW9uX2Rvd24sIE5VTEwpOwogICAgWHRWYUdldFZhbHVlcyhzZnItPnd3b3JkLCBYbU5zZXQsICZ3d29yZCwgTlVMTCk7CiAgICBYdFZhR2V0VmFsdWVzKHNmci0+bWNhc2UsIFhtTnNldCwgJm1jYXNlLCBOVUxMKTsKICAgIGlmICh3d29yZCkKCWZsYWdzIHw9IEZSRF9XSE9MRV9XT1JEOwogICAgaWYgKG1jYXNlKQoJZmxhZ3MgfD0gRlJEX01BVENIX0NBU0U7CgogICAgKHZvaWQpZ3VpX2RvX2ZpbmRyZXBsKChpbnQpZmxhZ3MsIChjaGFyX3UgKilmaW5kX3RleHQsIChjaGFyX3UgKilyZXBsX3RleHQsCgkJCQkJCQkgICAgICBkaXJlY3Rpb25fZG93bik7CgogICAgaWYgKGZpbmRfdGV4dCAhPSBOVUxMKQoJWHRGcmVlKGZpbmRfdGV4dCk7CiAgICBpZiAocmVwbF90ZXh0ICE9IE5VTEwpCglYdEZyZWUocmVwbF90ZXh0KTsKfQoKICAgIHN0YXRpYyB2b2lkCmZpbmRfcmVwbGFjZV9rZXlwcmVzcyh3LCBmcmRwLCBldmVudCkKICAgIFdpZGdldAkJdyBVTlVTRUQ7CiAgICBTaGFyZWRGaW5kUmVwbGFjZQkqZnJkcDsKICAgIFhLZXlFdmVudAkJKmV2ZW50Owp7CiAgICBLZXlTeW0ga2V5c3ltOwoKICAgIGlmIChmcmRwID09IE5VTEwpCglyZXR1cm47CgogICAga2V5c3ltID0gWExvb2t1cEtleXN5bShldmVudCwgMCk7CgogICAgLyogdGhlIHNjYXBlIGtleSBwb3BzIHRoZSB3aG9sZSBkaWFsb2cgZG93biAqLwogICAgaWYgKGtleXN5bSA9PSBYS19Fc2NhcGUpCglYdFVubWFuYWdlQ2hpbGQoZnJkcC0+ZGlhbG9nKTsKfQoKICAgIHN0YXRpYyB2b2lkCnNldF9sYWJlbCh3LCBsYWJlbCkKICAgIFdpZGdldCB3OwogICAgY2hhcl91ICpsYWJlbDsKewogICAgWG1TdHJpbmcJc3RyOwogICAgY2hhcl91CSpwLCAqbmV4dDsKICAgIEtleVN5bQltbmVtb25pYyA9IE5VTDsKCiAgICBpZiAoIXcpCglyZXR1cm47CgogICAgcCA9IHZpbV9zdHJzYXZlKGxhYmVsKTsKICAgIGlmIChwID09IE5VTEwpCglyZXR1cm47CiAgICBmb3IgKG5leHQgPSBwOyAqbmV4dDsgKytuZXh0KQogICAgewoJaWYgKCpuZXh0ID09IERMR19IT1RLRVlfQ0hBUikKCXsKCSAgICBpbnQgbGVuID0gU1RSTEVOKG5leHQpOwoKCSAgICBpZiAobGVuID4gMCkKCSAgICB7CgkJbWNoX21lbW1vdmUobmV4dCwgbmV4dCArIDEsIGxlbik7CgkJbW5lbW9uaWMgPSBuZXh0WzBdOwoJICAgIH0KCX0KICAgIH0KCiAgICBzdHIgPSBYbVN0cmluZ0NyZWF0ZVNpbXBsZSgoY2hhciAqKXApOwogICAgdmltX2ZyZWUocCk7CiAgICBpZiAoc3RyKQogICAgewoJWHRWYVNldFZhbHVlcyh3LAoJCVhtTmxhYmVsU3RyaW5nLCBzdHIsCgkJWG1ObW5lbW9uaWMsIG1uZW1vbmljLAoJCU5VTEwpOwoJWG1TdHJpbmdGcmVlKHN0cik7CiAgICB9CiAgICBndWlfbW90aWZfbWVudV9mb250bGlzdCh3KTsKfQoKICAgIHN0YXRpYyB2b2lkCmZpbmRfcmVwbGFjZV9kaWFsb2dfY3JlYXRlKGFyZywgZG9fcmVwbGFjZSkKICAgIGNoYXJfdQkqYXJnOwogICAgaW50CQlkb19yZXBsYWNlOwp7CiAgICBTaGFyZWRGaW5kUmVwbGFjZQkqZnJkcDsKICAgIFdpZGdldAkJc2VwYXJhdG9yOwogICAgV2lkZ2V0CQlpbnB1dF9mb3JtOwogICAgV2lkZ2V0CQlidXR0b25fZm9ybTsKICAgIFdpZGdldAkJdG9nZ2xlX2Zvcm07CiAgICBXaWRnZXQJCWZyYW1lOwogICAgWG1TdHJpbmcJCXN0cjsKICAgIGludAkJCW47CiAgICBBcmcJCQlhcmdzWzZdOwogICAgaW50CQkJd3dvcmQgPSBGQUxTRTsKICAgIGludAkJCW1jYXNlID0gIXBfaWM7CiAgICBEaW1lbnNpb24JCXdpZHRoOwogICAgRGltZW5zaW9uCQl3aWRlc3Q7CiAgICBjaGFyX3UJCSplbnRyeV90ZXh0OwoKICAgIGZyZHAgPSBkb19yZXBsYWNlID8gJnJlcGxfd2lkZ2V0cyA6ICZmaW5kX3dpZGdldHM7CgogICAgLyogR2V0IHRoZSBzZWFyY2ggc3RyaW5nIHRvIHVzZS4gKi8KICAgIGVudHJ5X3RleHQgPSBnZXRfZmluZF9kaWFsb2dfdGV4dChhcmcsICZ3d29yZCwgJm1jYXNlKTsKCiAgICAvKiBJZiB0aGUgZGlhbG9nIGFscmVhZHkgZXhpc3RzLCBqdXN0IHJhaXNlIGl0LiAqLwogICAgaWYgKGZyZHAtPmRpYWxvZykKICAgIHsKCWd1aV9tb3RpZl9zeW5jaF9mb250cygpOwoKCS8qIElmIHRoZSB3aW5kb3cgaXMgYWxyZWFkeSB1cCwganVzdCBwb3AgaXQgdG8gdGhlIHRvcCAqLwoJaWYgKFh0SXNNYW5hZ2VkKGZyZHAtPmRpYWxvZykpCgkgICAgWE1hcFJhaXNlZChYdERpc3BsYXkoZnJkcC0+ZGlhbG9nKSwKCQkJCQkgICAgWHRXaW5kb3coWHRQYXJlbnQoZnJkcC0+ZGlhbG9nKSkpOwoJZWxzZQoJICAgIFh0TWFuYWdlQ2hpbGQoZnJkcC0+ZGlhbG9nKTsKCVh0UG9wdXAoWHRQYXJlbnQoZnJkcC0+ZGlhbG9nKSwgWHRHcmFiTm9uZSk7CglYbVByb2Nlc3NUcmF2ZXJzYWwoZnJkcC0+d2hhdCwgWG1UUkFWRVJTRV9DVVJSRU5UKTsKCglpZiAoZW50cnlfdGV4dCAhPSBOVUxMKQoJICAgIFhtVGV4dEZpZWxkU2V0U3RyaW5nKGZyZHAtPndoYXQsIChjaGFyICopZW50cnlfdGV4dCk7Cgl2aW1fZnJlZShlbnRyeV90ZXh0KTsKCglYdFZhU2V0VmFsdWVzKGZyZHAtPnd3b3JkLCBYbU5zZXQsIHd3b3JkLCBOVUxMKTsKCXJldHVybjsKICAgIH0KCiAgICAvKiBDcmVhdGUgYSBmcmVzaCBuZXcgZGlhbG9nIHdpbmRvdyAqLwogICAgaWYgKGRvX3JlcGxhY2UpCgkgc3RyID0gWG1TdHJpbmdDcmVhdGVTaW1wbGUoXygiVklNIC0gU2VhcmNoIGFuZCBSZXBsYWNlLi4uIikpOwogICAgZWxzZQoJIHN0ciA9IFhtU3RyaW5nQ3JlYXRlU2ltcGxlKF8oIlZJTSAtIFNlYXJjaC4uLiIpKTsKCiAgICBuID0gMDsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmF1dG9Vbm1hbmFnZSwgRmFsc2UpOyBuKys7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5ub1Jlc2l6ZSwgVHJ1ZSk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmRpYWxvZ1RpdGxlLCBzdHIpOyBuKys7CgogICAgZnJkcC0+ZGlhbG9nID0gWG1DcmVhdGVGb3JtRGlhbG9nKHZpbVNoZWxsLCAiZmluZFJlcGxhY2VEaWFsb2ciLCBhcmdzLCBuKTsKICAgIFhtU3RyaW5nRnJlZShzdHIpOwogICAgWHRBZGRDYWxsYmFjayhmcmRwLT5kaWFsb2csIFhtTmRlc3Ryb3lDYWxsYmFjaywKCSAgICBmaW5kX3JlcGxhY2VfZGVzdHJveV9jYWxsYmFjaywgZnJkcCk7CgogICAgYnV0dG9uX2Zvcm0gPSBYdFZhQ3JlYXRlV2lkZ2V0KCJidXR0b25Gb3JtIiwKCSAgICB4bUZvcm1XaWRnZXRDbGFzcywJZnJkcC0+ZGlhbG9nLAoJICAgIFhtTnJpZ2h0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBYbU5yaWdodE9mZnNldCwgNCwKCSAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIFhtTnRvcE9mZnNldCwgNCwKCSAgICBYbU5ib3R0b21BdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIFhtTmJvdHRvbU9mZnNldCwgNCwKCSAgICBOVUxMKTsKCiAgICBmcmRwLT5maW5kID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoImZpbmRCdXR0b24iLAoJICAgIHhtUHVzaEJ1dHRvbldpZGdldENsYXNzLCBidXR0b25fZm9ybSwKCSAgICBYbU5zZW5zaXRpdmUsIFRydWUsCgkgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkgICAgTlVMTCk7CiAgICBzZXRfbGFiZWwoZnJkcC0+ZmluZCwgXygiRmluZCAmTmV4dCIpKTsKCiAgICBYdEFkZENhbGxiYWNrKGZyZHAtPmZpbmQsIFhtTmFjdGl2YXRlQ2FsbGJhY2ssCgkgICAgZmluZF9yZXBsYWNlX2NhbGxiYWNrLAoJICAgIChkb19yZXBsYWNlID8gKFh0UG9pbnRlcilGUkRfUl9GSU5ETkVYVCA6IChYdFBvaW50ZXIpRlJEX0ZJTkRORVhUKSk7CgogICAgaWYgKGRvX3JlcGxhY2UpCiAgICB7CglmcmRwLT5yZXBsYWNlID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoInJlcGxhY2VCdXR0b24iLAoJCXhtUHVzaEJ1dHRvbldpZGdldENsYXNzLCBidXR0b25fZm9ybSwKCQlYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkJWG1OdG9wV2lkZ2V0LCBmcmRwLT5maW5kLAoJCVhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCVhtTnJpZ2h0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlOVUxMKTsKCXNldF9sYWJlbChmcmRwLT5yZXBsYWNlLCBfKCImUmVwbGFjZSIpKTsKCVh0QWRkQ2FsbGJhY2soZnJkcC0+cmVwbGFjZSwgWG1OYWN0aXZhdGVDYWxsYmFjaywKCQlmaW5kX3JlcGxhY2VfY2FsbGJhY2ssIChYdFBvaW50ZXIpRlJEX1JFUExBQ0UpOwoKCWZyZHAtPmFsbCA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJyZXBsYWNlQWxsQnV0dG9uIiwKCQl4bVB1c2hCdXR0b25XaWRnZXRDbGFzcywgYnV0dG9uX2Zvcm0sCgkJWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCVhtTnRvcFdpZGdldCwgZnJkcC0+cmVwbGFjZSwKCQlYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJTlVMTCk7CglzZXRfbGFiZWwoZnJkcC0+YWxsLCBfKCJSZXBsYWNlICZBbGwiKSk7CglYdEFkZENhbGxiYWNrKGZyZHAtPmFsbCwgWG1OYWN0aXZhdGVDYWxsYmFjaywKCQlmaW5kX3JlcGxhY2VfY2FsbGJhY2ssIChYdFBvaW50ZXIpRlJEX1JFUExBQ0VBTEwpOwoKCWZyZHAtPnVuZG8gPSBYdFZhQ3JlYXRlTWFuYWdlZFdpZGdldCgidW5kb0J1dHRvbiIsCgkJeG1QdXNoQnV0dG9uV2lkZ2V0Q2xhc3MsIGJ1dHRvbl9mb3JtLAoJCVhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQlYbU50b3BXaWRnZXQsIGZyZHAtPmFsbCwKCQlYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJTlVMTCk7CglzZXRfbGFiZWwoZnJkcC0+dW5kbywgXygiJlVuZG8iKSk7CglYdEFkZENhbGxiYWNrKGZyZHAtPnVuZG8sIFhtTmFjdGl2YXRlQ2FsbGJhY2ssCgkJZmluZF9yZXBsYWNlX2NhbGxiYWNrLCAoWHRQb2ludGVyKUZSRF9VTkRPKTsKICAgIH0KCiAgICBmcmRwLT5jYW5jZWwgPSBYdFZhQ3JlYXRlTWFuYWdlZFdpZGdldCgiY2xvc2VCdXR0b24iLAoJICAgIHhtUHVzaEJ1dHRvbldpZGdldENsYXNzLCBidXR0b25fZm9ybSwKCSAgICBYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkgICAgWG1OYm90dG9tQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBOVUxMKTsKICAgIHNldF9sYWJlbChmcmRwLT5jYW5jZWwsIF8oIiZDYW5jZWwiKSk7CiAgICBYdEFkZENhbGxiYWNrKGZyZHAtPmNhbmNlbCwgWG1OYWN0aXZhdGVDYWxsYmFjaywKCSAgICBmaW5kX3JlcGxhY2VfZGlzbWlzc19jYWxsYmFjaywgZnJkcCk7CiAgICBndWlfbW90aWZfbWVudV9mb250bGlzdChmcmRwLT5jYW5jZWwpOwoKICAgIFh0TWFuYWdlQ2hpbGQoYnV0dG9uX2Zvcm0pOwoKICAgIG4gPSAwOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1Ob3JpZW50YXRpb24sIFhtVkVSVElDQUwpOyBuKys7CiAgICBYdFNldEFyZyhhcmdzW25dLCBYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnJpZ2h0V2lkZ2V0LCBidXR0b25fZm9ybSk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTnJpZ2h0T2Zmc2V0LCA0KTsgbisrOwogICAgWHRTZXRBcmcoYXJnc1tuXSwgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSk7IG4rKzsKICAgIFh0U2V0QXJnKGFyZ3Nbbl0sIFhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0pOyBuKys7CiAgICBzZXBhcmF0b3IgPSBYbUNyZWF0ZVNlcGFyYXRvckdhZGdldChmcmRwLT5kaWFsb2csICJzZXBhcmF0b3IiLCBhcmdzLCBuKTsKICAgIFh0TWFuYWdlQ2hpbGQoc2VwYXJhdG9yKTsKCiAgICBpbnB1dF9mb3JtID0gWHRWYUNyZWF0ZVdpZGdldCgiaW5wdXRGb3JtIiwKCSAgICB4bUZvcm1XaWRnZXRDbGFzcywJZnJkcC0+ZGlhbG9nLAoJICAgIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIFhtTmxlZnRPZmZzZXQsIDQsCgkgICAgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkgICAgWG1OcmlnaHRXaWRnZXQsIHNlcGFyYXRvciwKCSAgICBYbU5yaWdodE9mZnNldCwgNCwKCSAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIFhtTnRvcE9mZnNldCwgNCwKCSAgICBOVUxMKTsKCiAgICB7CglXaWRnZXQgbGFiZWxfd2hhdDsKCVdpZGdldCBsYWJlbF93aXRoID0gKFdpZGdldCkwOwoKCXN0ciA9IFhtU3RyaW5nQ3JlYXRlU2ltcGxlKF8oIkZpbmQgd2hhdDoiKSk7CglsYWJlbF93aGF0ID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoIndoYXRMYWJlbCIsCgkJeG1MYWJlbEdhZGdldENsYXNzLCBpbnB1dF9mb3JtLAoJCVhtTmxhYmVsU3RyaW5nLCBzdHIsCgkJWG1ObGVmdEF0dGFjaG1lbnQsCVhtQVRUQUNIX0ZPUk0sCgkJWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlYbU50b3BPZmZzZXQsIDQsCgkJTlVMTCk7CglYbVN0cmluZ0ZyZWUoc3RyKTsKCWd1aV9tb3RpZl9tZW51X2ZvbnRsaXN0KGxhYmVsX3doYXQpOwoKCWZyZHAtPndoYXQgPSBYdFZhQ3JlYXRlTWFuYWdlZFdpZGdldCgid2hhdFRleHQiLAoJCXhtVGV4dEZpZWxkV2lkZ2V0Q2xhc3MsIGlucHV0X2Zvcm0sCgkJWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJWG1ObGVmdEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJTlVMTCk7CgoJaWYgKGRvX3JlcGxhY2UpCgl7CgkgICAgZnJkcC0+d2l0aCA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJ3aXRoVGV4dCIsCgkJICAgIHhtVGV4dEZpZWxkV2lkZ2V0Q2xhc3MsCWlucHV0X2Zvcm0sCgkJICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCQkgICAgWG1OdG9wV2lkZ2V0LCBmcmRwLT53aGF0LAoJCSAgICBYbU50b3BPZmZzZXQsIDQsCgkJICAgIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJCSAgICBYbU5yaWdodEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJICAgIFhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkJICAgIE5VTEwpOwoKCSAgICBYdEFkZENhbGxiYWNrKGZyZHAtPndpdGgsIFhtTmFjdGl2YXRlQ2FsbGJhY2ssCgkJICAgIGZpbmRfcmVwbGFjZV9jYWxsYmFjaywgKFh0UG9pbnRlcikgRlJEX1JfRklORE5FWFQpOwoKCSAgICBzdHIgPSBYbVN0cmluZ0NyZWF0ZVNpbXBsZShfKCJSZXBsYWNlIHdpdGg6IikpOwoJICAgIGxhYmVsX3dpdGggPSBYdFZhQ3JlYXRlTWFuYWdlZFdpZGdldCgid2l0aExhYmVsIiwKCQkgICAgeG1MYWJlbEdhZGdldENsYXNzLCBpbnB1dF9mb3JtLAoJCSAgICBYbU5sYWJlbFN0cmluZywgc3RyLAoJCSAgICBYbU5sZWZ0QXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQkgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCSAgICBYbU50b3BXaWRnZXQsIGZyZHAtPndoYXQsCgkJICAgIFhtTnRvcE9mZnNldCwgNCwKCQkgICAgWG1OYm90dG9tQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQkgICAgTlVMTCk7CgkgICAgWG1TdHJpbmdGcmVlKHN0cik7CgkgICAgZ3VpX21vdGlmX21lbnVfZm9udGxpc3QobGFiZWxfd2l0aCk7CgoJICAgIC8qCgkgICAgICogTWFrZSB0aGUgZW50cnkgYWN0aXZhdGlvbiBvbmx5IGNoYW5nZSB0aGUgaW5wdXQgZm9jdXMgb250byB0aGUKCSAgICAgKiB3aXRoIGl0ZW0uCgkgICAgICovCgkgICAgWHRBZGRDYWxsYmFjayhmcmRwLT53aGF0LCBYbU5hY3RpdmF0ZUNhbGxiYWNrLAoJCSAgICBlbnRyeV9hY3RpdmF0ZV9jYWxsYmFjaywgZnJkcC0+d2l0aCk7CgkgICAgWHRBZGRFdmVudEhhbmRsZXIoZnJkcC0+d2l0aCwgS2V5UHJlc3NNYXNrLCBGYWxzZSwKCQkJICAgIChYdEV2ZW50SGFuZGxlcilmaW5kX3JlcGxhY2Vfa2V5cHJlc3MsCgkJCSAgICAoWHRQb2ludGVyKSBmcmRwKTsKCgl9CgllbHNlCgl7CgkgICAgLyoKCSAgICAgKiBNYWtlIHRoZSBlbnRyeSBhY3RpdmF0aW9uIGRvIHRoZSBzZWFyY2guCgkgICAgICovCgkgICAgWHRBZGRDYWxsYmFjayhmcmRwLT53aGF0LCBYbU5hY3RpdmF0ZUNhbGxiYWNrLAoJCSAgICBmaW5kX3JlcGxhY2VfY2FsbGJhY2ssIChYdFBvaW50ZXIpRlJEX0ZJTkRORVhUKTsKCX0KCVh0QWRkRXZlbnRIYW5kbGVyKGZyZHAtPndoYXQsIEtleVByZXNzTWFzaywgRmFsc2UsCgkJCSAgICAoWHRFdmVudEhhbmRsZXIpZmluZF9yZXBsYWNlX2tleXByZXNzLAoJCQkgICAgKFh0UG9pbnRlcilmcmRwKTsKCgkvKiBHZXQgdGhlIG1heGltdW0gd2lkdGggYmV0d2VlbiB0aGUgbGFiZWwgd2lkZ2V0cyBhbmQgbGluZSB0aGVtIHVwLgoJICovCgluID0gMDsKCVh0U2V0QXJnKGFyZ3Nbbl0sIFhtTndpZHRoLCAmd2lkdGgpOyBuKys7CglYdEdldFZhbHVlcyhsYWJlbF93aGF0LCBhcmdzLCBuKTsKCXdpZGVzdCA9IHdpZHRoOwoJaWYgKGRvX3JlcGxhY2UpCgl7CgkgICAgWHRHZXRWYWx1ZXMobGFiZWxfd2l0aCwgYXJncywgbik7CgkgICAgaWYgKHdpZHRoID4gd2lkZXN0KQoJCXdpZGVzdCA9IHdpZHRoOwoJfQoKCVh0VmFTZXRWYWx1ZXMoZnJkcC0+d2hhdCwgWG1ObGVmdE9mZnNldCwgd2lkZXN0LCBOVUxMKTsKCWlmIChkb19yZXBsYWNlKQoJICAgIFh0VmFTZXRWYWx1ZXMoZnJkcC0+d2l0aCwgWG1ObGVmdE9mZnNldCwgd2lkZXN0LCBOVUxMKTsKCiAgICB9CgogICAgWHRNYW5hZ2VDaGlsZChpbnB1dF9mb3JtKTsKCiAgICB7CglXaWRnZXQgcmFkaW9fYm94OwoJV2lkZ2V0IHc7CgoJZnJhbWUgPSBYdFZhQ3JlYXRlV2lkZ2V0KCJkaXJlY3Rpb25GcmFtZSIsCgkJeG1GcmFtZVdpZGdldENsYXNzLCBmcmRwLT5kaWFsb2csCgkJWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfV0lER0VULAoJCVhtTnRvcFdpZGdldCwgaW5wdXRfZm9ybSwKCQlYbU50b3BPZmZzZXQsIDQsCgkJWG1OYm90dG9tQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCQlYbU5ib3R0b21PZmZzZXQsIDQsCgkJWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9PUFBPU0lURV9XSURHRVQsCgkJWG1OcmlnaHRXaWRnZXQsIGlucHV0X2Zvcm0sCgkJTlVMTCk7CgoJc3RyID0gWG1TdHJpbmdDcmVhdGVTaW1wbGUoXygiRGlyZWN0aW9uIikpOwoJdyA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJkaXJlY3Rpb25GcmFtZUxhYmVsIiwKCQl4bUxhYmVsR2FkZ2V0Q2xhc3MsIGZyYW1lLAoJCVhtTmxhYmVsU3RyaW5nLCBzdHIsCgkJWG1OY2hpbGRIb3Jpem9udGFsQWxpZ25tZW50LCBYbUFMSUdOTUVOVF9CRUdJTk5JTkcsCgkJWG1OY2hpbGRUeXBlLCBYbUZSQU1FX1RJVExFX0NISUxELAoJCU5VTEwpOwoJWG1TdHJpbmdGcmVlKHN0cik7CglndWlfbW90aWZfbWVudV9mb250bGlzdCh3KTsKCglyYWRpb19ib3ggPSBYbUNyZWF0ZVJhZGlvQm94KGZyYW1lLCAicmFkaW9Cb3giLAoJCShBcmdMaXN0KU5VTEwsIDApOwoKCXN0ciA9IFhtU3RyaW5nQ3JlYXRlU2ltcGxlKCBfKCJVcCIpKTsKCWZyZHAtPnVwID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoInVwUmFkaW9CdXR0b24iLAoJCXhtVG9nZ2xlQnV0dG9uR2FkZ2V0Q2xhc3MsIHJhZGlvX2JveCwKCQlYbU5sYWJlbFN0cmluZywgc3RyLAoJCVhtTnNldCwgRmFsc2UsCgkJTlVMTCk7CglYbVN0cmluZ0ZyZWUoc3RyKTsKCWd1aV9tb3RpZl9tZW51X2ZvbnRsaXN0KGZyZHAtPnVwKTsKCglzdHIgPSBYbVN0cmluZ0NyZWF0ZVNpbXBsZShfKCJEb3duIikpOwoJZnJkcC0+ZG93biA9IFh0VmFDcmVhdGVNYW5hZ2VkV2lkZ2V0KCJkb3duUmFkaW9CdXR0b24iLAoJCXhtVG9nZ2xlQnV0dG9uR2FkZ2V0Q2xhc3MsIHJhZGlvX2JveCwKCQlYbU5sYWJlbFN0cmluZywgc3RyLAoJCVhtTnNldCwgVHJ1ZSwKCQlOVUxMKTsKCVhtU3RyaW5nRnJlZShzdHIpOwoJZ3VpX21vdGlmX21lbnVfZm9udGxpc3QoZnJkcC0+ZG93bik7CgoJWHRNYW5hZ2VDaGlsZChyYWRpb19ib3gpOwoJWHRNYW5hZ2VDaGlsZChmcmFtZSk7CiAgICB9CgogICAgdG9nZ2xlX2Zvcm0gPSBYdFZhQ3JlYXRlV2lkZ2V0KCJ0b2dnbGVGb3JtIiwKCSAgICB4bUZvcm1XaWRnZXRDbGFzcywJZnJkcC0+ZGlhbG9nLAoJICAgIFhtTmxlZnRBdHRhY2htZW50LCBYbUFUVEFDSF9GT1JNLAoJICAgIFhtTmxlZnRPZmZzZXQsIDQsCgkgICAgWG1OcmlnaHRBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkgICAgWG1OcmlnaHRXaWRnZXQsIGZyYW1lLAoJICAgIFhtTnJpZ2h0T2Zmc2V0LCA0LAoJICAgIFhtTnRvcEF0dGFjaG1lbnQsIFhtQVRUQUNIX1dJREdFVCwKCSAgICBYbU50b3BXaWRnZXQsIGlucHV0X2Zvcm0sCgkgICAgWG1OdG9wT2Zmc2V0LCA0LAoJICAgIFhtTmJvdHRvbUF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkgICAgWG1OYm90dG9tT2Zmc2V0LCA0LAoJICAgIE5VTEwpOwoKICAgIHN0ciA9IFhtU3RyaW5nQ3JlYXRlU2ltcGxlKF8oIk1hdGNoIHdob2xlIHdvcmQgb25seSIpKTsKICAgIGZyZHAtPnd3b3JkID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoIndvcmRUb2dnbGUiLAoJICAgIHhtVG9nZ2xlQnV0dG9uR2FkZ2V0Q2xhc3MsIHRvZ2dsZV9mb3JtLAoJICAgIFhtTmxhYmVsU3RyaW5nLCBzdHIsCgkgICAgWG1OdG9wQXR0YWNobWVudCwgWG1BVFRBQ0hfRk9STSwKCSAgICBYbU50b3BPZmZzZXQsIDQsCgkgICAgWG1ObGVmdEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkgICAgWG1ObGVmdE9mZnNldCwgNCwKCSAgICBYbU5zZXQsIHd3b3JkLAoJICAgIE5VTEwpOwogICAgWG1TdHJpbmdGcmVlKHN0cik7CgogICAgc3RyID0gWG1TdHJpbmdDcmVhdGVTaW1wbGUoXygiTWF0Y2ggY2FzZSIpKTsKICAgIGZyZHAtPm1jYXNlID0gWHRWYUNyZWF0ZU1hbmFnZWRXaWRnZXQoImNhc2VUb2dnbGUiLAoJICAgIHhtVG9nZ2xlQnV0dG9uR2FkZ2V0Q2xhc3MsIHRvZ2dsZV9mb3JtLAoJICAgIFhtTmxhYmVsU3RyaW5nLCBzdHIsCgkgICAgWG1ObGVmdEF0dGFjaG1lbnQsIFhtQVRUQUNIX0ZPUk0sCgkgICAgWG1ObGVmdE9mZnNldCwgNCwKCSAgICBYbU50b3BBdHRhY2htZW50LCBYbUFUVEFDSF9XSURHRVQsCgkgICAgWG1OdG9wV2lkZ2V0LCBmcmRwLT53d29yZCwKCSAgICBYbU50b3BPZmZzZXQsIDQsCgkgICAgWG1Oc2V0LCBtY2FzZSwKCSAgICBOVUxMKTsKICAgIFhtU3RyaW5nRnJlZShzdHIpOwogICAgZ3VpX21vdGlmX21lbnVfZm9udGxpc3QoZnJkcC0+d3dvcmQpOwogICAgZ3VpX21vdGlmX21lbnVfZm9udGxpc3QoZnJkcC0+bWNhc2UpOwoKICAgIFh0TWFuYWdlQ2hpbGQodG9nZ2xlX2Zvcm0pOwoKICAgIGlmIChlbnRyeV90ZXh0ICE9IE5VTEwpCglYbVRleHRGaWVsZFNldFN0cmluZyhmcmRwLT53aGF0LCAoY2hhciAqKWVudHJ5X3RleHQpOwogICAgdmltX2ZyZWUoZW50cnlfdGV4dCk7CgogICAgZ3VpX21vdGlmX3N5bmNoX2ZvbnRzKCk7CgogICAgbWFuYWdlX2NlbnRlcmVkKGZyZHAtPmRpYWxvZyk7CiAgICBhY3RpdmF0ZV9kaWFsb2dfbW5lbW9uaWNzKGZyZHAtPmRpYWxvZyk7CiAgICBYbVByb2Nlc3NUcmF2ZXJzYWwoZnJkcC0+d2hhdCwgWG1UUkFWRVJTRV9DVVJSRU5UKTsKfQoKICAgdm9pZApndWlfbWNoX2ZpbmRfZGlhbG9nKGVhcCkKICAgIGV4YXJnX1QJKmVhcDsKewogICAgaWYgKCFndWkuaW5fdXNlKQoJcmV0dXJuOwoKICAgIGZpbmRfcmVwbGFjZV9kaWFsb2dfY3JlYXRlKGVhcC0+YXJnLCBGQUxTRSk7Cn0KCgogICAgdm9pZApndWlfbWNoX3JlcGxhY2VfZGlhbG9nKGVhcCkKICAgIGV4YXJnX1QJKmVhcDsKewogICAgaWYgKCFndWkuaW5fdXNlKQoJcmV0dXJuOwoKICAgIGZpbmRfcmVwbGFjZV9kaWFsb2dfY3JlYXRlKGVhcC0+YXJnLCBUUlVFKTsKfQoKLyoKICogU3luY2hyb25pemUgYWxsIGd1aSBlbGVtZW50cywgd2hpY2ggYXJlIGRlcGVuZGFudCB1cG9uIHRoZQogKiBtYWluIHRleHQgZm9udCB1c2VkLiBUaG9zZSBhcmUgaW4gZXNwLiB0aGUgZmluZC9yZXBsYWNlIGRpYWxvZ3MuCiAqIElmIHlvdSBkb24ndCB1bmRlcnN0YW5kIHdoeSB0aGlzIHNob3VsZCBiZSBuZWVkZWQsIHBsZWFzZSB0cnkgdG8KICogc2VhcmNoIGZvciAicGnqtuYiIGluIGlzbzg4NTktMi4KICovCiAgICB2b2lkCmd1aV9tb3RpZl9zeW5jaF9mb250cyh2b2lkKQp7CiAgICBTaGFyZWRGaW5kUmVwbGFjZSAqZnJkcDsKICAgIGludAkJICAgIGRvX3JlcGxhY2U7CiAgICBYRm9udFN0cnVjdAkgICAgKmZvbnQ7CiAgICBYbUZvbnRMaXN0CSAgICBmb250X2xpc3Q7CgogICAgLyogRklYTUU6IFVubGVzcyB3ZSBmaW5kIG91dCBob3cgdG8gY3JlYXRlIGEgWG1Gb250TGlzdCBmcm9tIGEgWEZvbnRTZXQsCiAgICAgKiB3ZSBqdXN0IGdpdmUgdXAgaGVyZSBvbiBmb250IHN5bmNocm9uaXphdGlvbi4gKi8KICAgIGZvbnQgPSAoWEZvbnRTdHJ1Y3QgKilndWkubm9ybV9mb250OwogICAgaWYgKGZvbnQgPT0gTlVMTCkKCXJldHVybjsKCiAgICBmb250X2xpc3QgPSBndWlfbW90aWZfY3JlYXRlX2ZvbnRsaXN0KGZvbnQpOwoKICAgIC8qIE9LIHRoaXMgbG9vcCBpcyBhIGJpdCB0cmlja3kuLi4gKi8KICAgIGZvciAoZG9fcmVwbGFjZSA9IDA7IGRvX3JlcGxhY2UgPD0gMTsgKytkb19yZXBsYWNlKQogICAgewoJZnJkcCA9IChkb19yZXBsYWNlKSA/ICgmcmVwbF93aWRnZXRzKSA6ICgmZmluZF93aWRnZXRzKTsKCWlmIChmcmRwLT5kaWFsb2cpCgl7CgkgICAgWHRWYVNldFZhbHVlcyhmcmRwLT53aGF0LCBYbU5mb250TGlzdCwgZm9udF9saXN0LCBOVUxMKTsKCSAgICBpZiAoZG9fcmVwbGFjZSkKCQlYdFZhU2V0VmFsdWVzKGZyZHAtPndpdGgsIFhtTmZvbnRMaXN0LCBmb250X2xpc3QsIE5VTEwpOwoJfQogICAgfQoKICAgIFhtRm9udExpc3RGcmVlKGZvbnRfbGlzdCk7Cn0K