LyogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgogICB1bmljb2RlZGF0YSAtLSBQcm92aWRlcyBhY2Nlc3MgdG8gdGhlIFVuaWNvZGUgNS4xIGRhdGEgYmFzZS4KCiAgIERhdGEgd2FzIGV4dHJhY3RlZCBmcm9tIHRoZSBVbmljb2RlIDUuMSBVbmljb2RlRGF0YS50eHQgZmlsZS4KCiAgIFdyaXR0ZW4gYnkgTWFyYy1BbmRyZSBMZW1idXJnIChtYWxAbGVtYnVyZy5jb20pLgogICBNb2RpZmllZCBmb3IgUHl0aG9uIDIuMCBieSBGcmVkcmlrIEx1bmRoIChmcmVkcmlrQHB5dGhvbndhcmUuY29tKQogICBNb2RpZmllZCBieSBNYXJ0aW4gdi4gTPZ3aXMgKG1hcnRpbkB2LmxvZXdpcy5kZSkKCiAgIENvcHlyaWdodCAoYykgQ29ycG9yYXRpb24gZm9yIE5hdGlvbmFsIFJlc2VhcmNoIEluaXRpYXRpdmVzLgoKICAgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICovCgojaW5jbHVkZSAiUHl0aG9uLmgiCiNpbmNsdWRlICJ1Y25oYXNoLmgiCiNpbmNsdWRlICJzdHJ1Y3RtZW1iZXIuaCIKCi8qIGNoYXJhY3RlciBwcm9wZXJ0aWVzICovCgp0eXBlZGVmIHN0cnVjdCB7CiAgICBjb25zdCB1bnNpZ25lZCBjaGFyIGNhdGVnb3J5OwkvKiBpbmRleCBpbnRvCgkJCQkJICAgX1B5VW5pY29kZV9DYXRlZ29yeU5hbWVzICovCiAgICBjb25zdCB1bnNpZ25lZCBjaGFyCWNvbWJpbmluZzsgCS8qIGNvbWJpbmluZyBjbGFzcyB2YWx1ZSAwIC0gMjU1ICovCiAgICBjb25zdCB1bnNpZ25lZCBjaGFyCWJpZGlyZWN0aW9uYWw7IAkvKiBpbmRleCBpbnRvCgkJCQkJICAgX1B5VW5pY29kZV9CaWRpcmVjdGlvbmFsTmFtZXMgKi8KICAgIGNvbnN0IHVuc2lnbmVkIGNoYXIgbWlycm9yZWQ7CS8qIHRydWUgaWYgbWlycm9yZWQgaW4gYmlkaXIgbW9kZSAqLwogICAgY29uc3QgdW5zaWduZWQgY2hhciBlYXN0X2FzaWFuX3dpZHRoOwkvKiBpbmRleCBpbnRvCgkJCQkJCSAgIF9QeVVuaWNvZGVfRWFzdEFzaWFuV2lkdGggKi8KfSBfUHlVbmljb2RlX0RhdGFiYXNlUmVjb3JkOwoKdHlwZWRlZiBzdHJ1Y3QgY2hhbmdlX3JlY29yZCB7CiAgICAvKiBzZXF1ZW5jZSBvZiBmaWVsZHMgc2hvdWxkIGJlIHRoZSBzYW1lIGFzIGluIG1lcmdlX29sZF92ZXJzaW9uICovCiAgICBjb25zdCB1bnNpZ25lZCBjaGFyIGJpZGlyX2NoYW5nZWQ7CiAgICBjb25zdCB1bnNpZ25lZCBjaGFyIGNhdGVnb3J5X2NoYW5nZWQ7CiAgICBjb25zdCB1bnNpZ25lZCBjaGFyIGRlY2ltYWxfY2hhbmdlZDsKICAgIGNvbnN0IHVuc2lnbmVkIGNoYXIgbWlycm9yZWRfY2hhbmdlZDsKICAgIGNvbnN0IGludCBudW1lcmljX2NoYW5nZWQ7Cn0gY2hhbmdlX3JlY29yZDsKCi8qIGRhdGEgZmlsZSBnZW5lcmF0ZWQgYnkgVG9vbHMvdW5pY29kZS9tYWtldW5pY29kZWRhdGEucHkgKi8KI2luY2x1ZGUgInVuaWNvZGVkYXRhX2RiLmgiCgpzdGF0aWMgY29uc3QgX1B5VW5pY29kZV9EYXRhYmFzZVJlY29yZCoKX2dldHJlY29yZF9leChQeV9VQ1M0IGNvZGUpCnsKICAgIGludCBpbmRleDsKICAgIGlmIChjb2RlID49IDB4MTEwMDAwKQogICAgICAgIGluZGV4ID0gMDsKICAgIGVsc2UgewogICAgICAgIGluZGV4ID0gaW5kZXgxWyhjb2RlPj5TSElGVCldOwogICAgICAgIGluZGV4ID0gaW5kZXgyWyhpbmRleDw8U0hJRlQpKyhjb2RlJigoMTw8U0hJRlQpLTEpKV07CiAgICB9CgogICAgcmV0dXJuICZfUHlVbmljb2RlX0RhdGFiYXNlX1JlY29yZHNbaW5kZXhdOwp9CgovKiAtLS0tLS0tLS0tLS0tIFByZXZpb3VzLXZlcnNpb24gQVBJIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gKi8KdHlwZWRlZiBzdHJ1Y3QgcHJldmlvdXNfdmVyc2lvbiB7CiAgICBQeU9iamVjdF9IRUFECiAgICBjb25zdCBjaGFyICpuYW1lOwogICAgY29uc3QgY2hhbmdlX3JlY29yZCogKCpnZXRyZWNvcmQpKFB5X1VDUzQpOwogICAgUHlfVUNTNCAoKm5vcm1hbGl6YXRpb24pKFB5X1VDUzQpOwp9IFByZXZpb3VzREJWZXJzaW9uOwoKI2RlZmluZSBnZXRfb2xkX3JlY29yZChzZWxmLCB2KSAgICAoKCgoUHJldmlvdXNEQlZlcnNpb24qKXNlbGYpLT5nZXRyZWNvcmQpKHYpKQoKc3RhdGljIFB5TWVtYmVyRGVmIERCX21lbWJlcnNbXSA9IHsKCXsidW5pZGF0YV92ZXJzaW9uIiwgVF9TVFJJTkcsIG9mZnNldG9mKFByZXZpb3VzREJWZXJzaW9uLCBuYW1lKSwgUkVBRE9OTFl9LAogICAgICAgIHtOVUxMfQp9OwoKLyogZm9yd2FyZCBkZWNsYXJhdGlvbiAqLwpzdGF0aWMgUHlUeXBlT2JqZWN0IFVDRF9UeXBlOwojZGVmaW5lIFVDRF9DaGVjayhvKSAoUHlfVFlQRShvKT09JlVDRF9UeXBlKQoKc3RhdGljIFB5T2JqZWN0KgpuZXdfcHJldmlvdXNfdmVyc2lvbihjb25zdCBjaGFyKm5hbWUsIGNvbnN0IGNoYW5nZV9yZWNvcmQqICgqZ2V0cmVjb3JkKShQeV9VQ1M0KSwKICAgICAgICAgICAgICAgICAgICAgUHlfVUNTNCAoKm5vcm1hbGl6YXRpb24pKFB5X1VDUzQpKQp7CglQcmV2aW91c0RCVmVyc2lvbiAqc2VsZjsKCXNlbGYgPSBQeU9iamVjdF9OZXcoUHJldmlvdXNEQlZlcnNpb24sICZVQ0RfVHlwZSk7CglpZiAoc2VsZiA9PSBOVUxMKQoJCXJldHVybiBOVUxMOwoJc2VsZi0+bmFtZSA9IG5hbWU7CglzZWxmLT5nZXRyZWNvcmQgPSBnZXRyZWNvcmQ7CiAgICAgICAgc2VsZi0+bm9ybWFsaXphdGlvbiA9IG5vcm1hbGl6YXRpb247CglyZXR1cm4gKFB5T2JqZWN0KilzZWxmOwp9CgoKc3RhdGljIFB5X1VDUzQgZ2V0dWNoYXIoUHlVbmljb2RlT2JqZWN0ICpvYmopCnsKICAgIFB5X1VOSUNPREUgKnYgPSBQeVVuaWNvZGVfQVNfVU5JQ09ERShvYmopOwoKICAgIGlmIChQeVVuaWNvZGVfR0VUX1NJWkUob2JqKSA9PSAxKQoJcmV0dXJuICp2OwojaWZuZGVmIFB5X1VOSUNPREVfV0lERQogICAgZWxzZSBpZiAoKFB5VW5pY29kZV9HRVRfU0laRShvYmopID09IDIpICYmCiAgICAgICAgICAgICAoMHhEODAwIDw9IHZbMF0gJiYgdlswXSA8PSAweERCRkYpICYmCiAgICAgICAgICAgICAoMHhEQzAwIDw9IHZbMV0gJiYgdlsxXSA8PSAweERGRkYpKQoJcmV0dXJuICgoKHZbMF0gJiAweDNGRik8PDEwKSB8ICh2WzFdICYgMHgzRkYpKSArIDB4MTAwMDA7CiNlbmRpZgogICAgUHlFcnJfU2V0U3RyaW5nKFB5RXhjX1R5cGVFcnJvciwKICAgICAgICAgICAgICAgICAgICAibmVlZCBhIHNpbmdsZSBVbmljb2RlIGNoYXJhY3RlciBhcyBwYXJhbWV0ZXIiKTsKICAgIHJldHVybiAoUHlfVUNTNCktMTsKfQoKLyogLS0tIE1vZHVsZSBBUEkgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICovCgpQeURvY19TVFJWQVIodW5pY29kZWRhdGFfZGVjaW1hbF9fZG9jX18sCiJkZWNpbWFsKHVuaWNoclssIGRlZmF1bHRdKVxuXApcblwKUmV0dXJucyB0aGUgZGVjaW1hbCB2YWx1ZSBhc3NpZ25lZCB0byB0aGUgVW5pY29kZSBjaGFyYWN0ZXIgdW5pY2hyXG5cCmFzIGludGVnZXIuIElmIG5vIHN1Y2ggdmFsdWUgaXMgZGVmaW5lZCwgZGVmYXVsdCBpcyByZXR1cm5lZCwgb3IsIGlmXG5cCm5vdCBnaXZlbiwgVmFsdWVFcnJvciBpcyByYWlzZWQuIik7CgpzdGF0aWMgUHlPYmplY3QgKgp1bmljb2RlZGF0YV9kZWNpbWFsKFB5T2JqZWN0ICpzZWxmLCBQeU9iamVjdCAqYXJncykKewogICAgUHlVbmljb2RlT2JqZWN0ICp2OwogICAgUHlPYmplY3QgKmRlZm9iaiA9IE5VTEw7CiAgICBpbnQgaGF2ZV9vbGQgPSAwOwogICAgbG9uZyByYzsKICAgIFB5X1VDUzQgYzsKCiAgICBpZiAoIVB5QXJnX1BhcnNlVHVwbGUoYXJncywgIk8hfE86ZGVjaW1hbCIsICZQeVVuaWNvZGVfVHlwZSwgJnYsICZkZWZvYmopKQogICAgICAgIHJldHVybiBOVUxMOwogICAgYyA9IGdldHVjaGFyKHYpOwogICAgaWYgKGMgPT0gKFB5X1VDUzQpLTEpCiAgICAgICAgcmV0dXJuIE5VTEw7CgogICAgaWYgKHNlbGYgJiYgVUNEX0NoZWNrKHNlbGYpKSB7CiAgICAgICAgY29uc3QgY2hhbmdlX3JlY29yZCAqb2xkID0gZ2V0X29sZF9yZWNvcmQoc2VsZiwgYyk7CiAgICAgICAgaWYgKG9sZC0+Y2F0ZWdvcnlfY2hhbmdlZCA9PSAwKSB7CiAgICAgICAgICAgIC8qIHVuYXNzaWduZWQgKi8KICAgICAgICAgICAgaGF2ZV9vbGQgPSAxOwogICAgICAgICAgICByYyA9IC0xOwogICAgICAgIH0gCiAgICAgICAgZWxzZSBpZiAob2xkLT5kZWNpbWFsX2NoYW5nZWQgIT0gMHhGRikgewogICAgICAgICAgICBoYXZlX29sZCA9IDE7CiAgICAgICAgICAgIHJjID0gb2xkLT5kZWNpbWFsX2NoYW5nZWQ7CiAgICAgICAgfQogICAgfQoKICAgIGlmICghaGF2ZV9vbGQpCiAgICAgICAgcmMgPSBQeV9VTklDT0RFX1RPREVDSU1BTChjKTsKICAgIGlmIChyYyA8IDApIHsKCWlmIChkZWZvYmogPT0gTlVMTCkgewoJICAgIFB5RXJyX1NldFN0cmluZyhQeUV4Y19WYWx1ZUVycm9yLAoJCQkgICAgIm5vdCBhIGRlY2ltYWwiKTsKICAgICAgICAgICAgcmV0dXJuIE5VTEw7Cgl9CgllbHNlIHsKCSAgICBQeV9JTkNSRUYoZGVmb2JqKTsKCSAgICByZXR1cm4gZGVmb2JqOwoJfQogICAgfQogICAgcmV0dXJuIFB5TG9uZ19Gcm9tTG9uZyhyYyk7Cn0KClB5RG9jX1NUUlZBUih1bmljb2RlZGF0YV9kaWdpdF9fZG9jX18sCiJkaWdpdCh1bmljaHJbLCBkZWZhdWx0XSlcblwKXG5cClJldHVybnMgdGhlIGRpZ2l0IHZhbHVlIGFzc2lnbmVkIHRvIHRoZSBVbmljb2RlIGNoYXJhY3RlciB1bmljaHIgYXNcblwKaW50ZWdlci4gSWYgbm8gc3VjaCB2YWx1ZSBpcyBkZWZpbmVkLCBkZWZhdWx0IGlzIHJldHVybmVkLCBvciwgaWZcblwKbm90IGdpdmVuLCBWYWx1ZUVycm9yIGlzIHJhaXNlZC4iKTsKCnN0YXRpYyBQeU9iamVjdCAqCnVuaWNvZGVkYXRhX2RpZ2l0KFB5T2JqZWN0ICpzZWxmLCBQeU9iamVjdCAqYXJncykKewogICAgUHlVbmljb2RlT2JqZWN0ICp2OwogICAgUHlPYmplY3QgKmRlZm9iaiA9IE5VTEw7CiAgICBsb25nIHJjOwogICAgUHlfVUNTNCBjOwoKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZShhcmdzLCAiTyF8TzpkaWdpdCIsICZQeVVuaWNvZGVfVHlwZSwgJnYsICZkZWZvYmopKQogICAgICAgIHJldHVybiBOVUxMOwogICAgYyA9IGdldHVjaGFyKHYpOwogICAgaWYgKGMgPT0gKFB5X1VDUzQpLTEpCiAgICAgICAgcmV0dXJuIE5VTEw7CiAgICByYyA9IFB5X1VOSUNPREVfVE9ESUdJVChjKTsKICAgIGlmIChyYyA8IDApIHsKCWlmIChkZWZvYmogPT0gTlVMTCkgewoJICAgIFB5RXJyX1NldFN0cmluZyhQeUV4Y19WYWx1ZUVycm9yLCAibm90IGEgZGlnaXQiKTsKICAgICAgICAgICAgcmV0dXJuIE5VTEw7Cgl9CgllbHNlIHsKCSAgICBQeV9JTkNSRUYoZGVmb2JqKTsKCSAgICByZXR1cm4gZGVmb2JqOwoJfQogICAgfQogICAgcmV0dXJuIFB5TG9uZ19Gcm9tTG9uZyhyYyk7Cn0KClB5RG9jX1NUUlZBUih1bmljb2RlZGF0YV9udW1lcmljX19kb2NfXywKIm51bWVyaWModW5pY2hyWywgZGVmYXVsdF0pXG5cClxuXApSZXR1cm5zIHRoZSBudW1lcmljIHZhbHVlIGFzc2lnbmVkIHRvIHRoZSBVbmljb2RlIGNoYXJhY3RlciB1bmljaHJcblwKYXMgZmxvYXQuIElmIG5vIHN1Y2ggdmFsdWUgaXMgZGVmaW5lZCwgZGVmYXVsdCBpcyByZXR1cm5lZCwgb3IsIGlmXG5cCm5vdCBnaXZlbiwgVmFsdWVFcnJvciBpcyByYWlzZWQuIik7CgpzdGF0aWMgUHlPYmplY3QgKgp1bmljb2RlZGF0YV9udW1lcmljKFB5T2JqZWN0ICpzZWxmLCBQeU9iamVjdCAqYXJncykKewogICAgUHlVbmljb2RlT2JqZWN0ICp2OwogICAgUHlPYmplY3QgKmRlZm9iaiA9IE5VTEw7CiAgICBpbnQgaGF2ZV9vbGQgPSAwOwogICAgZG91YmxlIHJjOwogICAgUHlfVUNTNCBjOwoKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZShhcmdzLCAiTyF8TzpudW1lcmljIiwgJlB5VW5pY29kZV9UeXBlLCAmdiwgJmRlZm9iaikpCiAgICAgICAgcmV0dXJuIE5VTEw7CiAgICBjID0gZ2V0dWNoYXIodik7CiAgICBpZiAoYyA9PSAoUHlfVUNTNCktMSkKICAgICAgICByZXR1cm4gTlVMTDsKCiAgICBpZiAoc2VsZiAmJiBVQ0RfQ2hlY2soc2VsZikpIHsKICAgICAgICBjb25zdCBjaGFuZ2VfcmVjb3JkICpvbGQgPSBnZXRfb2xkX3JlY29yZChzZWxmLCBjKTsKICAgICAgICBpZiAob2xkLT5jYXRlZ29yeV9jaGFuZ2VkID09IDApIHsKICAgICAgICAgICAgLyogdW5hc3NpZ25lZCAqLwogICAgICAgICAgICBoYXZlX29sZCA9IDE7CiAgICAgICAgICAgIHJjID0gLTEuMDsKICAgICAgICB9IAogICAgICAgIGVsc2UgaWYgKG9sZC0+ZGVjaW1hbF9jaGFuZ2VkICE9IDB4RkYpIHsKICAgICAgICAgICAgaGF2ZV9vbGQgPSAxOwogICAgICAgICAgICByYyA9IG9sZC0+ZGVjaW1hbF9jaGFuZ2VkOwogICAgICAgIH0KICAgIH0KCiAgICBpZiAoIWhhdmVfb2xkKQogICAgICAgIHJjID0gUHlfVU5JQ09ERV9UT05VTUVSSUMoYyk7CiAgICBpZiAocmMgPT0gLTEuMCkgewoJaWYgKGRlZm9iaiA9PSBOVUxMKSB7CgkgICAgUHlFcnJfU2V0U3RyaW5nKFB5RXhjX1ZhbHVlRXJyb3IsICJub3QgYSBudW1lcmljIGNoYXJhY3RlciIpOwoJICAgIHJldHVybiBOVUxMOwoJfQoJZWxzZSB7CgkgICAgUHlfSU5DUkVGKGRlZm9iaik7CgkgICAgcmV0dXJuIGRlZm9iajsKCX0KICAgIH0KICAgIHJldHVybiBQeUZsb2F0X0Zyb21Eb3VibGUocmMpOwp9CgpQeURvY19TVFJWQVIodW5pY29kZWRhdGFfY2F0ZWdvcnlfX2RvY19fLAoiY2F0ZWdvcnkodW5pY2hyKVxuXApcblwKUmV0dXJucyB0aGUgZ2VuZXJhbCBjYXRlZ29yeSBhc3NpZ25lZCB0byB0aGUgVW5pY29kZSBjaGFyYWN0ZXJcblwKdW5pY2hyIGFzIHN0cmluZy4iKTsKCnN0YXRpYyBQeU9iamVjdCAqCnVuaWNvZGVkYXRhX2NhdGVnb3J5KFB5T2JqZWN0ICpzZWxmLCBQeU9iamVjdCAqYXJncykKewogICAgUHlVbmljb2RlT2JqZWN0ICp2OwogICAgaW50IGluZGV4OwogICAgUHlfVUNTNCBjOwoKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZShhcmdzLCAiTyE6Y2F0ZWdvcnkiLAoJCQkgICZQeVVuaWNvZGVfVHlwZSwgJnYpKQoJcmV0dXJuIE5VTEw7CiAgICBjID0gZ2V0dWNoYXIodik7CiAgICBpZiAoYyA9PSAoUHlfVUNTNCktMSkKICAgICAgICByZXR1cm4gTlVMTDsKICAgIGluZGV4ID0gKGludCkgX2dldHJlY29yZF9leChjKS0+Y2F0ZWdvcnk7CiAgICBpZiAoc2VsZiAmJiBVQ0RfQ2hlY2soc2VsZikpIHsKICAgICAgICBjb25zdCBjaGFuZ2VfcmVjb3JkICpvbGQgPSBnZXRfb2xkX3JlY29yZChzZWxmLCBjKTsKICAgICAgICBpZiAob2xkLT5jYXRlZ29yeV9jaGFuZ2VkICE9IDB4RkYpCiAgICAgICAgICAgIGluZGV4ID0gb2xkLT5jYXRlZ29yeV9jaGFuZ2VkOwogICAgfQogICAgcmV0dXJuIFB5VW5pY29kZV9Gcm9tU3RyaW5nKF9QeVVuaWNvZGVfQ2F0ZWdvcnlOYW1lc1tpbmRleF0pOwp9CgpQeURvY19TVFJWQVIodW5pY29kZWRhdGFfYmlkaXJlY3Rpb25hbF9fZG9jX18sCiJiaWRpcmVjdGlvbmFsKHVuaWNocilcblwKXG5cClJldHVybnMgdGhlIGJpZGlyZWN0aW9uYWwgY2F0ZWdvcnkgYXNzaWduZWQgdG8gdGhlIFVuaWNvZGUgY2hhcmFjdGVyXG5cCnVuaWNociBhcyBzdHJpbmcuIElmIG5vIHN1Y2ggdmFsdWUgaXMgZGVmaW5lZCwgYW4gZW1wdHkgc3RyaW5nIGlzXG5cCnJldHVybmVkLiIpOwoKc3RhdGljIFB5T2JqZWN0ICoKdW5pY29kZWRhdGFfYmlkaXJlY3Rpb25hbChQeU9iamVjdCAqc2VsZiwgUHlPYmplY3QgKmFyZ3MpCnsKICAgIFB5VW5pY29kZU9iamVjdCAqdjsKICAgIGludCBpbmRleDsKICAgIFB5X1VDUzQgYzsKCiAgICBpZiAoIVB5QXJnX1BhcnNlVHVwbGUoYXJncywgIk8hOmJpZGlyZWN0aW9uYWwiLAoJCQkgICZQeVVuaWNvZGVfVHlwZSwgJnYpKQoJcmV0dXJuIE5VTEw7CiAgICBjID0gZ2V0dWNoYXIodik7CiAgICBpZiAoYyA9PSAoUHlfVUNTNCktMSkKICAgICAgICByZXR1cm4gTlVMTDsKICAgIGluZGV4ID0gKGludCkgX2dldHJlY29yZF9leChjKS0+YmlkaXJlY3Rpb25hbDsKICAgIGlmIChzZWxmICYmIFVDRF9DaGVjayhzZWxmKSkgewogICAgICAgIGNvbnN0IGNoYW5nZV9yZWNvcmQgKm9sZCA9IGdldF9vbGRfcmVjb3JkKHNlbGYsIGMpOwogICAgICAgIGlmIChvbGQtPmNhdGVnb3J5X2NoYW5nZWQgPT0gMCkKICAgICAgICAgICAgaW5kZXggPSAwOyAvKiB1bmFzc2lnbmVkICovCiAgICAgICAgZWxzZSBpZiAob2xkLT5iaWRpcl9jaGFuZ2VkICE9IDB4RkYpCiAgICAgICAgICAgIGluZGV4ID0gb2xkLT5iaWRpcl9jaGFuZ2VkOwogICAgfQogICAgcmV0dXJuIFB5VW5pY29kZV9Gcm9tU3RyaW5nKF9QeVVuaWNvZGVfQmlkaXJlY3Rpb25hbE5hbWVzW2luZGV4XSk7Cn0KClB5RG9jX1NUUlZBUih1bmljb2RlZGF0YV9jb21iaW5pbmdfX2RvY19fLAoiY29tYmluaW5nKHVuaWNocilcblwKXG5cClJldHVybnMgdGhlIGNhbm9uaWNhbCBjb21iaW5pbmcgY2xhc3MgYXNzaWduZWQgdG8gdGhlIFVuaWNvZGVcblwKY2hhcmFjdGVyIHVuaWNociBhcyBpbnRlZ2VyLiBSZXR1cm5zIDAgaWYgbm8gY29tYmluaW5nIGNsYXNzIGlzXG5cCmRlZmluZWQuIik7CgpzdGF0aWMgUHlPYmplY3QgKgp1bmljb2RlZGF0YV9jb21iaW5pbmcoUHlPYmplY3QgKnNlbGYsIFB5T2JqZWN0ICphcmdzKQp7CiAgICBQeVVuaWNvZGVPYmplY3QgKnY7CiAgICBpbnQgaW5kZXg7CiAgICBQeV9VQ1M0IGM7CgogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlKGFyZ3MsICJPITpjb21iaW5pbmciLAoJCQkgICZQeVVuaWNvZGVfVHlwZSwgJnYpKQoJcmV0dXJuIE5VTEw7CiAgICBjID0gZ2V0dWNoYXIodik7CiAgICBpZiAoYyA9PSAoUHlfVUNTNCktMSkKICAgICAgICByZXR1cm4gTlVMTDsKICAgIGluZGV4ID0gKGludCkgX2dldHJlY29yZF9leChjKS0+Y29tYmluaW5nOwogICAgaWYgKHNlbGYgJiYgVUNEX0NoZWNrKHNlbGYpKSB7CiAgICAgICAgY29uc3QgY2hhbmdlX3JlY29yZCAqb2xkID0gZ2V0X29sZF9yZWNvcmQoc2VsZiwgYyk7CiAgICAgICAgaWYgKG9sZC0+Y2F0ZWdvcnlfY2hhbmdlZCA9PSAwKQogICAgICAgICAgICBpbmRleCA9IDA7IC8qIHVuYXNzaWduZWQgKi8KICAgIH0KICAgIHJldHVybiBQeUxvbmdfRnJvbUxvbmcoaW5kZXgpOwp9CgpQeURvY19TVFJWQVIodW5pY29kZWRhdGFfbWlycm9yZWRfX2RvY19fLAoibWlycm9yZWQodW5pY2hyKVxuXApcblwKUmV0dXJucyB0aGUgbWlycm9yZWQgcHJvcGVydHkgYXNzaWduZWQgdG8gdGhlIFVuaWNvZGUgY2hhcmFjdGVyXG5cCnVuaWNociBhcyBpbnRlZ2VyLiBSZXR1cm5zIDEgaWYgdGhlIGNoYXJhY3RlciBoYXMgYmVlbiBpZGVudGlmaWVkIGFzXG5cCmEgXCJtaXJyb3JlZFwiIGNoYXJhY3RlciBpbiBiaWRpcmVjdGlvbmFsIHRleHQsIDAgb3RoZXJ3aXNlLiIpOwoKc3RhdGljIFB5T2JqZWN0ICoKdW5pY29kZWRhdGFfbWlycm9yZWQoUHlPYmplY3QgKnNlbGYsIFB5T2JqZWN0ICphcmdzKQp7CiAgICBQeVVuaWNvZGVPYmplY3QgKnY7CiAgICBpbnQgaW5kZXg7CiAgICBQeV9VQ1M0IGM7CgogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlKGFyZ3MsICJPITptaXJyb3JlZCIsCgkJCSAgJlB5VW5pY29kZV9UeXBlLCAmdikpCglyZXR1cm4gTlVMTDsKICAgIGMgPSBnZXR1Y2hhcih2KTsKICAgIGlmIChjID09IChQeV9VQ1M0KS0xKQogICAgICAgIHJldHVybiBOVUxMOwogICAgaW5kZXggPSAoaW50KSBfZ2V0cmVjb3JkX2V4KGMpLT5taXJyb3JlZDsKICAgIGlmIChzZWxmICYmIFVDRF9DaGVjayhzZWxmKSkgewogICAgICAgIGNvbnN0IGNoYW5nZV9yZWNvcmQgKm9sZCA9IGdldF9vbGRfcmVjb3JkKHNlbGYsIGMpOwogICAgICAgIGlmIChvbGQtPmNhdGVnb3J5X2NoYW5nZWQgPT0gMCkKICAgICAgICAgICAgaW5kZXggPSAwOyAvKiB1bmFzc2lnbmVkICovCiAgICAgICAgZWxzZSBpZiAob2xkLT5taXJyb3JlZF9jaGFuZ2VkICE9IDB4RkYpCiAgICAgICAgICAgIGluZGV4ID0gb2xkLT5taXJyb3JlZF9jaGFuZ2VkOwogICAgfQogICAgcmV0dXJuIFB5TG9uZ19Gcm9tTG9uZyhpbmRleCk7Cn0KClB5RG9jX1NUUlZBUih1bmljb2RlZGF0YV9lYXN0X2FzaWFuX3dpZHRoX19kb2NfXywKImVhc3RfYXNpYW5fd2lkdGgodW5pY2hyKVxuXApcblwKUmV0dXJucyB0aGUgZWFzdCBhc2lhbiB3aWR0aCBhc3NpZ25lZCB0byB0aGUgVW5pY29kZSBjaGFyYWN0ZXJcblwKdW5pY2hyIGFzIHN0cmluZy4iKTsKCnN0YXRpYyBQeU9iamVjdCAqCnVuaWNvZGVkYXRhX2Vhc3RfYXNpYW5fd2lkdGgoUHlPYmplY3QgKnNlbGYsIFB5T2JqZWN0ICphcmdzKQp7CiAgICBQeVVuaWNvZGVPYmplY3QgKnY7CiAgICBpbnQgaW5kZXg7CiAgICBQeV9VQ1M0IGM7CgogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlKGFyZ3MsICJPITplYXN0X2FzaWFuX3dpZHRoIiwKCQkJICAmUHlVbmljb2RlX1R5cGUsICZ2KSkKCXJldHVybiBOVUxMOwogICAgYyA9IGdldHVjaGFyKHYpOwogICAgaWYgKGMgPT0gKFB5X1VDUzQpLTEpCiAgICAgICAgcmV0dXJuIE5VTEw7CiAgICBpbmRleCA9IChpbnQpIF9nZXRyZWNvcmRfZXgoYyktPmVhc3RfYXNpYW5fd2lkdGg7CiAgICBpZiAoc2VsZiAmJiBVQ0RfQ2hlY2soc2VsZikpIHsKICAgICAgICBjb25zdCBjaGFuZ2VfcmVjb3JkICpvbGQgPSBnZXRfb2xkX3JlY29yZChzZWxmLCBjKTsKICAgICAgICBpZiAob2xkLT5jYXRlZ29yeV9jaGFuZ2VkID09IDApCiAgICAgICAgICAgIGluZGV4ID0gMDsgLyogdW5hc3NpZ25lZCAqLwogICAgfQogICAgcmV0dXJuIFB5VW5pY29kZV9Gcm9tU3RyaW5nKF9QeVVuaWNvZGVfRWFzdEFzaWFuV2lkdGhOYW1lc1tpbmRleF0pOwp9CgpQeURvY19TVFJWQVIodW5pY29kZWRhdGFfZGVjb21wb3NpdGlvbl9fZG9jX18sCiJkZWNvbXBvc2l0aW9uKHVuaWNocilcblwKXG5cClJldHVybnMgdGhlIGNoYXJhY3RlciBkZWNvbXBvc2l0aW9uIG1hcHBpbmcgYXNzaWduZWQgdG8gdGhlIFVuaWNvZGVcblwKY2hhcmFjdGVyIHVuaWNociBhcyBzdHJpbmcuIEFuIGVtcHR5IHN0cmluZyBpcyByZXR1cm5lZCBpbiBjYXNlIG5vXG5cCnN1Y2ggbWFwcGluZyBpcyBkZWZpbmVkLiIpOwoKc3RhdGljIFB5T2JqZWN0ICoKdW5pY29kZWRhdGFfZGVjb21wb3NpdGlvbihQeU9iamVjdCAqc2VsZiwgUHlPYmplY3QgKmFyZ3MpCnsKICAgIFB5VW5pY29kZU9iamVjdCAqdjsKICAgIGNoYXIgZGVjb21wWzI1Nl07CiAgICBpbnQgY29kZSwgaW5kZXgsIGNvdW50LCBpOwogICAgdW5zaWduZWQgaW50IHByZWZpeF9pbmRleDsKICAgIFB5X1VDUzQgYzsKCiAgICBpZiAoIVB5QXJnX1BhcnNlVHVwbGUoYXJncywgIk8hOmRlY29tcG9zaXRpb24iLAoJCQkgICZQeVVuaWNvZGVfVHlwZSwgJnYpKQoJcmV0dXJuIE5VTEw7CiAgICBjID0gZ2V0dWNoYXIodik7CiAgICBpZiAoYyA9PSAoUHlfVUNTNCktMSkKICAgICAgICByZXR1cm4gTlVMTDsKCiAgICBjb2RlID0gKGludCljOwoKICAgIGlmIChzZWxmICYmIFVDRF9DaGVjayhzZWxmKSkgewogICAgICAgIGNvbnN0IGNoYW5nZV9yZWNvcmQgKm9sZCA9IGdldF9vbGRfcmVjb3JkKHNlbGYsIGMpOwogICAgICAgIGlmIChvbGQtPmNhdGVnb3J5X2NoYW5nZWQgPT0gMCkKICAgICAgICAgICAgcmV0dXJuIFB5VW5pY29kZV9Gcm9tU3RyaW5nKCIiKTsgLyogdW5hc3NpZ25lZCAqLwogICAgfQoKICAgIGlmIChjb2RlIDwgMCB8fCBjb2RlID49IDB4MTEwMDAwKQogICAgICAgIGluZGV4ID0gMDsKICAgIGVsc2UgewogICAgICAgIGluZGV4ID0gZGVjb21wX2luZGV4MVsoY29kZT4+REVDT01QX1NISUZUKV07CiAgICAgICAgaW5kZXggPSBkZWNvbXBfaW5kZXgyWyhpbmRleDw8REVDT01QX1NISUZUKSsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoY29kZSYoKDE8PERFQ09NUF9TSElGVCktMSkpXTsKICAgIH0KCiAgICAvKiBoaWdoIGJ5dGUgaXMgbnVtYmVyIG9mIGhleCBieXRlcyAodXN1YWxseSBvbmUgb3IgdHdvKSwgbG93IGJ5dGUKICAgICAgIGlzIHByZWZpeCBjb2RlIChmcm9tKi8KICAgIGNvdW50ID0gZGVjb21wX2RhdGFbaW5kZXhdID4+IDg7CgogICAgLyogWFhYOiBjb3VsZCBhbGxvY2F0ZSB0aGUgUHlTdHJpbmcgdXAgZnJvbnQgaW5zdGVhZAogICAgICAgKHN0cmxlbihwcmVmaXgpICsgNSAqIGNvdW50ICsgMSBieXRlcykgKi8KCiAgICAvKiBCYXNlZCBvbiBob3cgaW5kZXggaXMgY2FsY3VsYXRlZCBhYm92ZSBhbmQgZGVjb21wX2RhdGEgaXMgZ2VuZXJhdGVkCiAgICAgICBmcm9tIFRvb2xzL3VuaWNvZGUvbWFrZXVuaWNvZGVkYXRhLnB5LCBpdCBzaG91bGQgbm90IGJlIHBvc3NpYmxlCiAgICAgICB0byBvdmVyZmxvdyBkZWNvbXBfcHJlZml4LiAqLwogICAgcHJlZml4X2luZGV4ID0gZGVjb21wX2RhdGFbaW5kZXhdICYgMjU1OwogICAgYXNzZXJ0KHByZWZpeF9pbmRleCA8IChzaXplb2YoZGVjb21wX3ByZWZpeCkvc2l6ZW9mKCpkZWNvbXBfcHJlZml4KSkpOwoKICAgIC8qIGNvcHkgcHJlZml4ICovCiAgICBpID0gc3RybGVuKGRlY29tcF9wcmVmaXhbcHJlZml4X2luZGV4XSk7CiAgICBtZW1jcHkoZGVjb21wLCBkZWNvbXBfcHJlZml4W3ByZWZpeF9pbmRleF0sIGkpOwoKICAgIHdoaWxlIChjb3VudC0tID4gMCkgewogICAgICAgIGlmIChpKQogICAgICAgICAgICBkZWNvbXBbaSsrXSA9ICcgJzsKICAgICAgICBhc3NlcnQoKHNpemVfdClpIDwgc2l6ZW9mKGRlY29tcCkpOwogICAgICAgIFB5T1Nfc25wcmludGYoZGVjb21wICsgaSwgc2l6ZW9mKGRlY29tcCkgLSBpLCAiJTA0WCIsCiAgICAgICAgICAgICAgICAgICAgICBkZWNvbXBfZGF0YVsrK2luZGV4XSk7CiAgICAgICAgaSArPSBzdHJsZW4oZGVjb21wICsgaSk7CiAgICB9CiAgICAKICAgIGRlY29tcFtpXSA9ICdcMCc7CgogICAgcmV0dXJuIFB5VW5pY29kZV9Gcm9tU3RyaW5nKGRlY29tcCk7Cn0KCnN0YXRpYyB2b2lkCmdldF9kZWNvbXBfcmVjb3JkKFB5T2JqZWN0ICpzZWxmLCBQeV9VQ1M0IGNvZGUsIGludCAqaW5kZXgsIGludCAqcHJlZml4LCBpbnQgKmNvdW50KQp7CiAgICBpZiAoY29kZSA+PSAweDExMDAwMCkgewogICAgICAgICppbmRleCA9IDA7CiAgICB9IGVsc2UgaWYgKHNlbGYgJiYgVUNEX0NoZWNrKHNlbGYpICYmIAogICAgICAgICAgICAgICBnZXRfb2xkX3JlY29yZChzZWxmLCBjb2RlKS0+Y2F0ZWdvcnlfY2hhbmdlZD09MCkgewogICAgICAgIC8qIHVuYXNzaWduZWQgaW4gb2xkIHZlcnNpb24gKi8KICAgICAgICAqaW5kZXggPSAwOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgKmluZGV4ID0gZGVjb21wX2luZGV4MVsoY29kZT4+REVDT01QX1NISUZUKV07CiAgICAgICAgKmluZGV4ID0gZGVjb21wX2luZGV4MlsoKmluZGV4PDxERUNPTVBfU0hJRlQpKwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNvZGUmKCgxPDxERUNPTVBfU0hJRlQpLTEpKV07CiAgICB9CgkKICAgIC8qIGhpZ2ggYnl0ZSBpcyBudW1iZXIgb2YgaGV4IGJ5dGVzICh1c3VhbGx5IG9uZSBvciB0d28pLCBsb3cgYnl0ZQogICAgICAgaXMgcHJlZml4IGNvZGUgKGZyb20qLwogICAgKmNvdW50ID0gZGVjb21wX2RhdGFbKmluZGV4XSA+PiA4OwogICAgKnByZWZpeCA9IGRlY29tcF9kYXRhWyppbmRleF0gJiAyNTU7CgogICAgKCppbmRleCkrKzsKfQoKI2RlZmluZSBTQmFzZSAgIDB4QUMwMAojZGVmaW5lIExCYXNlICAgMHgxMTAwCiNkZWZpbmUgVkJhc2UgICAweDExNjEKI2RlZmluZSBUQmFzZSAgIDB4MTFBNwojZGVmaW5lIExDb3VudCAgMTkKI2RlZmluZSBWQ291bnQgIDIxCiNkZWZpbmUgVENvdW50ICAyOAojZGVmaW5lIE5Db3VudCAgKFZDb3VudCpUQ291bnQpCiNkZWZpbmUgU0NvdW50ICAoTENvdW50Kk5Db3VudCkKCnN0YXRpYyBQeU9iamVjdCoKbmZkX25ma2QoUHlPYmplY3QgKnNlbGYsIFB5T2JqZWN0ICppbnB1dCwgaW50IGspCnsKICAgIFB5T2JqZWN0ICpyZXN1bHQ7CiAgICBQeV9VTklDT0RFICppLCAqZW5kLCAqbzsKICAgIC8qIExvbmdlc3QgZGVjb21wb3NpdGlvbiBpbiBVbmljb2RlIDMuMjogVStGREZBICovCiAgICBQeV9VTklDT0RFIHN0YWNrWzIwXTsgCiAgICBQeV9zc2l6ZV90IHNwYWNlLCBpc2l6ZTsKICAgIGludCBpbmRleCwgcHJlZml4LCBjb3VudCwgc3RhY2twdHI7CiAgICB1bnNpZ25lZCBjaGFyIHByZXYsIGN1cjsKCQogICAgc3RhY2twdHIgPSAwOwogICAgaXNpemUgPSBQeVVuaWNvZGVfR0VUX1NJWkUoaW5wdXQpOwogICAgLyogT3ZlcmFsbG9jYXRlIGF0bW9zdCAxMCBjaGFyYWN0ZXJzLiAqLwogICAgc3BhY2UgPSAoaXNpemUgPiAxMCA/IDEwIDogaXNpemUpICsgaXNpemU7CiAgICByZXN1bHQgPSBQeVVuaWNvZGVfRnJvbVVuaWNvZGUoTlVMTCwgc3BhY2UpOwogICAgaWYgKCFyZXN1bHQpCiAgICAgICAgcmV0dXJuIE5VTEw7CiAgICBpID0gUHlVbmljb2RlX0FTX1VOSUNPREUoaW5wdXQpOwogICAgZW5kID0gaSArIGlzaXplOwogICAgbyA9IFB5VW5pY29kZV9BU19VTklDT0RFKHJlc3VsdCk7CgogICAgd2hpbGUgKGkgPCBlbmQpIHsKICAgICAgICBzdGFja1tzdGFja3B0cisrXSA9ICppKys7CiAgICAgICAgd2hpbGUoc3RhY2twdHIpIHsKICAgICAgICAgICAgUHlfVU5JQ09ERSBjb2RlID0gc3RhY2tbLS1zdGFja3B0cl07CiAgICAgICAgICAgIC8qIEhhbmd1bCBEZWNvbXBvc2l0aW9uIGFkZHMgdGhyZWUgY2hhcmFjdGVycyBpbgogICAgICAgICAgICAgICBhIHNpbmdsZSBzdGVwLCBzbyB3ZSBuZWVkIGF0bGVhc3QgdGhhdCBtdWNoIHJvb20uICovCiAgICAgICAgICAgIGlmIChzcGFjZSA8IDMpIHsKICAgICAgICAgICAgICAgIFB5X3NzaXplX3QgbmV3c2l6ZSA9IFB5VW5pY29kZV9HRVRfU0laRShyZXN1bHQpICsgMTA7CiAgICAgICAgICAgICAgICBzcGFjZSArPSAxMDsKICAgICAgICAgICAgICAgIGlmIChQeVVuaWNvZGVfUmVzaXplKCZyZXN1bHQsIG5ld3NpemUpID09IC0xKQogICAgICAgICAgICAgICAgICAgIHJldHVybiBOVUxMOwogICAgICAgICAgICAgICAgbyA9IFB5VW5pY29kZV9BU19VTklDT0RFKHJlc3VsdCkgKyBuZXdzaXplIC0gc3BhY2U7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgLyogSGFuZ3VsIERlY29tcG9zaXRpb24uICovCiAgICAgICAgICAgIGlmIChTQmFzZSA8PSBjb2RlICYmIGNvZGUgPCAoU0Jhc2UrU0NvdW50KSkgewogICAgICAgICAgICAgICAgaW50IFNJbmRleCA9IGNvZGUgLSBTQmFzZTsKICAgICAgICAgICAgICAgIGludCBMID0gTEJhc2UgKyBTSW5kZXggLyBOQ291bnQ7CiAgICAgICAgICAgICAgICBpbnQgViA9IFZCYXNlICsgKFNJbmRleCAlIE5Db3VudCkgLyBUQ291bnQ7CiAgICAgICAgICAgICAgICBpbnQgVCA9IFRCYXNlICsgU0luZGV4ICUgVENvdW50OwogICAgICAgICAgICAgICAgKm8rKyA9IEw7CiAgICAgICAgICAgICAgICAqbysrID0gVjsKICAgICAgICAgICAgICAgIHNwYWNlIC09IDI7CiAgICAgICAgICAgICAgICBpZiAoVCAhPSBUQmFzZSkgewogICAgICAgICAgICAgICAgICAgICpvKysgPSBUOwogICAgICAgICAgICAgICAgICAgIHNwYWNlIC0tOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgLyogbm9ybWFsaXphdGlvbiBjaGFuZ2VzICovCiAgICAgICAgICAgIGlmIChzZWxmICYmIFVDRF9DaGVjayhzZWxmKSkgewogICAgICAgICAgICAgICAgUHlfVUNTNCB2YWx1ZSA9ICgoUHJldmlvdXNEQlZlcnNpb24qKXNlbGYpLT5ub3JtYWxpemF0aW9uKGNvZGUpOwogICAgICAgICAgICAgICAgaWYgKHZhbHVlICE9IDApIHsKICAgICAgICAgICAgICAgICAgICBzdGFja1tzdGFja3B0cisrXSA9IHZhbHVlOwogICAgICAgICAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CgogICAgICAgICAgICAvKiBPdGhlciBkZWNvbXBvc2l0aW9ucy4gKi8KICAgICAgICAgICAgZ2V0X2RlY29tcF9yZWNvcmQoc2VsZiwgY29kZSwgJmluZGV4LCAmcHJlZml4LCAmY291bnQpOwoKICAgICAgICAgICAgLyogQ29weSBjaGFyYWN0ZXIgaWYgaXQgaXMgbm90IGRlY29tcG9zYWJsZSwgb3IgaGFzIGEKICAgICAgICAgICAgICAgY29tcGF0aWJpbGl0eSBkZWNvbXBvc2l0aW9uLCBidXQgd2UgZG8gTkZELiAqLwogICAgICAgICAgICBpZiAoIWNvdW50IHx8IChwcmVmaXggJiYgIWspKSB7CiAgICAgICAgICAgICAgICAqbysrID0gY29kZTsKICAgICAgICAgICAgICAgIHNwYWNlLS07CiAgICAgICAgICAgICAgICBjb250aW51ZTsKICAgICAgICAgICAgfQogICAgICAgICAgICAvKiBDb3B5IGRlY29tcG9zaXRpb24gb250byB0aGUgc3RhY2ssIGluIHJldmVyc2UKICAgICAgICAgICAgICAgb3JkZXIuICAqLwogICAgICAgICAgICB3aGlsZShjb3VudCkgewogICAgICAgICAgICAgICAgY29kZSA9IGRlY29tcF9kYXRhW2luZGV4ICsgKC0tY291bnQpXTsKICAgICAgICAgICAgICAgIHN0YWNrW3N0YWNrcHRyKytdID0gY29kZTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICAvKiBEcm9wIG92ZXJhbGxvY2F0aW9uLiBDYW5ub3QgZmFpbC4gKi8KICAgIFB5VW5pY29kZV9SZXNpemUoJnJlc3VsdCwgUHlVbmljb2RlX0dFVF9TSVpFKHJlc3VsdCkgLSBzcGFjZSk7CgogICAgLyogU29ydCBjYW5vbmljYWxseS4gKi8KICAgIGkgPSBQeVVuaWNvZGVfQVNfVU5JQ09ERShyZXN1bHQpOwogICAgcHJldiA9IF9nZXRyZWNvcmRfZXgoKmkpLT5jb21iaW5pbmc7CiAgICBlbmQgPSBpICsgUHlVbmljb2RlX0dFVF9TSVpFKHJlc3VsdCk7CiAgICBmb3IgKGkrKzsgaSA8IGVuZDsgaSsrKSB7CiAgICAgICAgY3VyID0gX2dldHJlY29yZF9leCgqaSktPmNvbWJpbmluZzsKICAgICAgICBpZiAocHJldiA9PSAwIHx8IGN1ciA9PSAwIHx8IHByZXYgPD0gY3VyKSB7CiAgICAgICAgICAgIHByZXYgPSBjdXI7CiAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgIH0KICAgICAgICAvKiBOb24tY2Fub25pY2FsIG9yZGVyLiBOZWVkIHRvIHN3aXRjaCAqaSB3aXRoIHByZXZpb3VzLiAqLwogICAgICAgIG8gPSBpIC0gMTsKICAgICAgICB3aGlsZSAoMSkgewogICAgICAgICAgICBQeV9VTklDT0RFIHRtcCA9IG9bMV07CiAgICAgICAgICAgIG9bMV0gPSBvWzBdOwogICAgICAgICAgICBvWzBdID0gdG1wOwogICAgICAgICAgICBvLS07CiAgICAgICAgICAgIGlmIChvIDwgUHlVbmljb2RlX0FTX1VOSUNPREUocmVzdWx0KSkKICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICBwcmV2ID0gX2dldHJlY29yZF9leCgqbyktPmNvbWJpbmluZzsKICAgICAgICAgICAgaWYgKHByZXYgPT0gMCB8fCBwcmV2IDw9IGN1cikKICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KICAgICAgICBwcmV2ID0gX2dldHJlY29yZF9leCgqaSktPmNvbWJpbmluZzsKICAgIH0KICAgIHJldHVybiByZXN1bHQ7Cn0KCnN0YXRpYyBpbnQKZmluZF9uZmNfaW5kZXgoUHlPYmplY3QgKnNlbGYsIHN0cnVjdCByZWluZGV4KiBuZmMsIFB5X1VOSUNPREUgY29kZSkKewogICAgaW50IGluZGV4OwogICAgZm9yIChpbmRleCA9IDA7IG5mY1tpbmRleF0uc3RhcnQ7IGluZGV4KyspIHsKICAgICAgICBpbnQgc3RhcnQgPSBuZmNbaW5kZXhdLnN0YXJ0OwogICAgICAgIGlmIChjb2RlIDwgc3RhcnQpCiAgICAgICAgICAgIHJldHVybiAtMTsKICAgICAgICBpZiAoY29kZSA8PSBzdGFydCArIG5mY1tpbmRleF0uY291bnQpIHsKICAgICAgICAgICAgaW50IGRlbHRhID0gY29kZSAtIHN0YXJ0OwogICAgICAgICAgICByZXR1cm4gbmZjW2luZGV4XS5pbmRleCArIGRlbHRhOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiAtMTsKfQoKc3RhdGljIFB5T2JqZWN0KgpuZmNfbmZrYyhQeU9iamVjdCAqc2VsZiwgUHlPYmplY3QgKmlucHV0LCBpbnQgaykKewogICAgUHlPYmplY3QgKnJlc3VsdDsKICAgIFB5X1VOSUNPREUgKmksICppMSwgKm8sICplbmQ7CiAgICBpbnQgZixsLGluZGV4LGluZGV4MSxjb21iOwogICAgUHlfVU5JQ09ERSBjb2RlOwogICAgUHlfVU5JQ09ERSAqc2tpcHBlZFsyMF07CiAgICBpbnQgY3NraXBwZWQgPSAwOwoKICAgIHJlc3VsdCA9IG5mZF9uZmtkKHNlbGYsIGlucHV0LCBrKTsKICAgIGlmICghcmVzdWx0KQogICAgICAgIHJldHVybiBOVUxMOwoKICAgIC8qIFdlIGFyZSBnb2luZyB0byBtb2RpZnkgcmVzdWx0IGluLXBsYWNlLgogICAgICAgSWYgbmZkX25ma2QgaXMgY2hhbmdlZCB0byBzb21ldGltZXMgcmV0dXJuIHRoZSBpbnB1dCwKICAgICAgIHRoaXMgY29kZSBuZWVkcyB0byBiZSByZXZpZXdlZC4gKi8KICAgIGFzc2VydChyZXN1bHQgIT0gaW5wdXQpOwoKICAgIGkgPSBQeVVuaWNvZGVfQVNfVU5JQ09ERShyZXN1bHQpOwogICAgZW5kID0gaSArIFB5VW5pY29kZV9HRVRfU0laRShyZXN1bHQpOwogICAgbyA9IFB5VW5pY29kZV9BU19VTklDT0RFKHJlc3VsdCk7CgkKICBhZ2FpbjoKICAgIHdoaWxlIChpIDwgZW5kKSB7CiAgICAgIGZvciAoaW5kZXggPSAwOyBpbmRleCA8IGNza2lwcGVkOyBpbmRleCsrKSB7CiAgICAgICAgICBpZiAoc2tpcHBlZFtpbmRleF0gPT0gaSkgewogICAgICAgICAgICAgIC8qICppIGNoYXJhY3RlciBpcyBza2lwcGVkLiAKICAgICAgICAgICAgICAgICBSZW1vdmUgZnJvbSBsaXN0LiAqLwogICAgICAgICAgICAgIHNraXBwZWRbaW5kZXhdID0gc2tpcHBlZFtjc2tpcHBlZC0xXTsKICAgICAgICAgICAgICBjc2tpcHBlZC0tOwogICAgICAgICAgICAgIGkrKzsKICAgICAgICAgICAgICBnb3RvIGFnYWluOyAvKiBjb250aW51ZSB3aGlsZSAqLwogICAgICAgICAgfQogICAgICB9CiAgICAgIC8qIEhhbmd1bCBDb21wb3NpdGlvbi4gV2UgZG9uJ3QgbmVlZCB0byBjaGVjayBmb3IgPExWLFQ+CiAgICAgICAgIHBhaXJzLCBzaW5jZSB3ZSBhbHdheXMgaGF2ZSBkZWNvbXBvc2VkIGRhdGEuICovCiAgICAgIGlmIChMQmFzZSA8PSAqaSAmJiAqaSA8IChMQmFzZStMQ291bnQpICYmCiAgICAgICAgICBpICsgMSA8IGVuZCAmJiAKICAgICAgICAgIFZCYXNlIDw9IGlbMV0gJiYgaVsxXSA8PSAoVkJhc2UrVkNvdW50KSkgewogICAgICAgICAgaW50IExJbmRleCwgVkluZGV4OwogICAgICAgICAgTEluZGV4ID0gaVswXSAtIExCYXNlOwogICAgICAgICAgVkluZGV4ID0gaVsxXSAtIFZCYXNlOwogICAgICAgICAgY29kZSA9IFNCYXNlICsgKExJbmRleCpWQ291bnQrVkluZGV4KSpUQ291bnQ7CiAgICAgICAgICBpKz0yOwogICAgICAgICAgaWYgKGkgPCBlbmQgJiYKICAgICAgICAgICAgICBUQmFzZSA8PSAqaSAmJiAqaSA8PSAoVEJhc2UrVENvdW50KSkgewogICAgICAgICAgICAgIGNvZGUgKz0gKmktVEJhc2U7CiAgICAgICAgICAgICAgaSsrOwogICAgICAgICAgfQogICAgICAgICAgKm8rKyA9IGNvZGU7CiAgICAgICAgICBjb250aW51ZTsKICAgICAgfQoKICAgICAgZiA9IGZpbmRfbmZjX2luZGV4KHNlbGYsIG5mY19maXJzdCwgKmkpOwogICAgICBpZiAoZiA9PSAtMSkgewogICAgICAgICAgKm8rKyA9ICppKys7CiAgICAgICAgICBjb250aW51ZTsKICAgICAgfQogICAgICAvKiBGaW5kIG5leHQgdW5ibG9ja2VkIGNoYXJhY3Rlci4gKi8KICAgICAgaTEgPSBpKzE7CiAgICAgIGNvbWIgPSAwOwogICAgICB3aGlsZSAoaTEgPCBlbmQpIHsKICAgICAgICAgIGludCBjb21iMSA9IF9nZXRyZWNvcmRfZXgoKmkxKS0+Y29tYmluaW5nOwogICAgICAgICAgaWYgKGNvbWIxICYmIGNvbWIgPT0gY29tYjEpIHsKICAgICAgICAgICAgICAvKiBDaGFyYWN0ZXIgaXMgYmxvY2tlZC4gKi8KICAgICAgICAgICAgICBpMSsrOwogICAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgICAgfQogICAgICAgICAgbCA9IGZpbmRfbmZjX2luZGV4KHNlbGYsIG5mY19sYXN0LCAqaTEpOwogICAgICAgICAgLyogKmkxIGNhbm5vdCBiZSBjb21iaW5lZCB3aXRoICppLiBJZiAqaTEKICAgICAgICAgICAgIGlzIGEgc3RhcnRlciwgd2UgZG9uJ3QgbmVlZCB0byBsb29rIGZ1cnRoZXIuCiAgICAgICAgICAgICBPdGhlcndpc2UsIHJlY29yZCB0aGUgY29tYmluaW5nIGNsYXNzLiAqLwogICAgICAgICAgaWYgKGwgPT0gLTEpIHsKICAgICAgICAgICAgbm90X2NvbWJpbmFibGU6CiAgICAgICAgICAgICAgaWYgKGNvbWIxID09IDApCiAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgIGNvbWIgPSBjb21iMTsKICAgICAgICAgICAgICBpMSsrOwogICAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgICAgfQogICAgICAgICAgaW5kZXggPSBmKlRPVEFMX0xBU1QgKyBsOwogICAgICAgICAgaW5kZXgxID0gY29tcF9pbmRleFtpbmRleCA+PiBDT01QX1NISUZUXTsKICAgICAgICAgIGNvZGUgPSBjb21wX2RhdGFbKGluZGV4MTw8Q09NUF9TSElGVCkrCiAgICAgICAgICAgICAgICAgICAgICAgICAgIChpbmRleCYoKDE8PENPTVBfU0hJRlQpLTEpKV07CiAgICAgICAgICBpZiAoY29kZSA9PSAwKQogICAgICAgICAgICAgIGdvdG8gbm90X2NvbWJpbmFibGU7CgkJCQogICAgICAgICAgLyogUmVwbGFjZSB0aGUgb3JpZ2luYWwgY2hhcmFjdGVyLiAqLwogICAgICAgICAgKmkgPSBjb2RlOwogICAgICAgICAgLyogTWFyayB0aGUgc2Vjb25kIGNoYXJhY3RlciB1bnVzZWQuICovCiAgICAgICAgICBza2lwcGVkW2Nza2lwcGVkKytdID0gaTE7CiAgICAgICAgICBpMSsrOwogICAgICAgICAgZiA9IGZpbmRfbmZjX2luZGV4KHNlbGYsIG5mY19maXJzdCwgKmkpOwogICAgICAgICAgaWYgKGYgPT0gLTEpCiAgICAgICAgICAgICAgYnJlYWs7CiAgICAgIH0KICAgICAgKm8rKyA9ICppKys7CiAgICB9CiAgICBpZiAobyAhPSBlbmQpCiAgICAgICAgUHlVbmljb2RlX1Jlc2l6ZSgmcmVzdWx0LCBvIC0gUHlVbmljb2RlX0FTX1VOSUNPREUocmVzdWx0KSk7CiAgICByZXR1cm4gcmVzdWx0Owp9CgkJClB5RG9jX1NUUlZBUih1bmljb2RlZGF0YV9ub3JtYWxpemVfX2RvY19fLAoibm9ybWFsaXplKGZvcm0sIHVuaXN0cilcblwKXG5cClJldHVybiB0aGUgbm9ybWFsIGZvcm0gJ2Zvcm0nIGZvciB0aGUgVW5pY29kZSBzdHJpbmcgdW5pc3RyLiAgVmFsaWRcblwKdmFsdWVzIGZvciBmb3JtIGFyZSAnTkZDJywgJ05GS0MnLCAnTkZEJywgYW5kICdORktEJy4iKTsKCnN0YXRpYyBQeU9iamVjdCoKdW5pY29kZWRhdGFfbm9ybWFsaXplKFB5T2JqZWN0ICpzZWxmLCBQeU9iamVjdCAqYXJncykKewogICAgY2hhciAqZm9ybTsKICAgIFB5T2JqZWN0ICppbnB1dDsKCiAgICBpZighUHlBcmdfUGFyc2VUdXBsZShhcmdzLCAic08hOm5vcm1hbGl6ZSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAmZm9ybSwgJlB5VW5pY29kZV9UeXBlLCAmaW5wdXQpKQogICAgICAgIHJldHVybiBOVUxMOwoKICAgIGlmIChQeVVuaWNvZGVfR2V0U2l6ZShpbnB1dCkgPT0gMCkgewogICAgICAgIC8qIFNwZWNpYWwgY2FzZSBlbXB0eSBpbnB1dCBzdHJpbmdzLCBzaW5jZSByZXNpemluZwogICAgICAgICAgIHRoZW0gIGxhdGVyIHdvdWxkIGNhdXNlIGludGVybmFsIGVycm9ycy4gKi8KICAgICAgICBQeV9JTkNSRUYoaW5wdXQpOwogICAgICAgIHJldHVybiBpbnB1dDsKICAgIH0KCiAgICBpZiAoc3RyY21wKGZvcm0sICJORkMiKSA9PSAwKQogICAgICAgIHJldHVybiBuZmNfbmZrYyhzZWxmLCBpbnB1dCwgMCk7CiAgICBpZiAoc3RyY21wKGZvcm0sICJORktDIikgPT0gMCkKICAgICAgICByZXR1cm4gbmZjX25ma2Moc2VsZiwgaW5wdXQsIDEpOwogICAgaWYgKHN0cmNtcChmb3JtLCAiTkZEIikgPT0gMCkKICAgICAgICByZXR1cm4gbmZkX25ma2Qoc2VsZiwgaW5wdXQsIDApOwogICAgaWYgKHN0cmNtcChmb3JtLCAiTkZLRCIpID09IDApCiAgICAgICAgcmV0dXJuIG5mZF9uZmtkKHNlbGYsIGlucHV0LCAxKTsKICAgIFB5RXJyX1NldFN0cmluZyhQeUV4Y19WYWx1ZUVycm9yLCAiaW52YWxpZCBub3JtYWxpemF0aW9uIGZvcm0iKTsKICAgIHJldHVybiBOVUxMOwp9CgovKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAqLwovKiB1bmljb2RlIGNoYXJhY3RlciBuYW1lIHRhYmxlcyAqLwoKLyogZGF0YSBmaWxlIGdlbmVyYXRlZCBieSBUb29scy91bmljb2RlL21ha2V1bmljb2RlZGF0YS5weSAqLwojaW5jbHVkZSAidW5pY29kZW5hbWVfZGIuaCIKCi8qIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICovCi8qIGRhdGFiYXNlIGNvZGUgKGN1dCBhbmQgcGFzdGVkIGZyb20gdGhlIHVuaWRiIHBhY2thZ2UpICovCgpzdGF0aWMgdW5zaWduZWQgbG9uZwpfZ2V0aGFzaChjb25zdCBjaGFyICpzLCBpbnQgbGVuLCBpbnQgc2NhbGUpCnsKICAgIGludCBpOwogICAgdW5zaWduZWQgbG9uZyBoID0gMDsKICAgIHVuc2lnbmVkIGxvbmcgaXg7CiAgICBmb3IgKGkgPSAwOyBpIDwgbGVuOyBpKyspIHsKICAgICAgICBoID0gKGggKiBzY2FsZSkgKyAodW5zaWduZWQgY2hhcikgdG91cHBlcihQeV9DSEFSTUFTSyhzW2ldKSk7CiAgICAgICAgaXggPSBoICYgMHhmZjAwMDAwMDsKICAgICAgICBpZiAoaXgpCiAgICAgICAgICAgIGggPSAoaCBeICgoaXg+PjI0KSAmIDB4ZmYpKSAmIDB4MDBmZmZmZmY7CiAgICB9CiAgICByZXR1cm4gaDsKfQoKc3RhdGljIGNoYXIgKmhhbmd1bF9zeWxsYWJsZXNbXVszXSA9IHsKICAgIHsgIkciLCAgIkEiLCAgICIiICAgfSwKICAgIHsgIkdHIiwgIkFFIiwgICJHIiAgfSwKICAgIHsgIk4iLCAgIllBIiwgICJHRyIgfSwKICAgIHsgIkQiLCAgIllBRSIsICJHUyIgfSwKICAgIHsgIkREIiwgIkVPIiwgICJOIiwgfSwKICAgIHsgIlIiLCAgIkUiLCAgICJOSiIgfSwKICAgIHsgIk0iLCAgIllFTyIsICJOSCIgfSwKICAgIHsgIkIiLCAgIllFIiwgICJEIiAgfSwKICAgIHsgIkJCIiwgIk8iLCAgICJMIiAgfSwKICAgIHsgIlMiLCAgIldBIiwgICJMRyIgfSwKICAgIHsgIlNTIiwgIldBRSIsICJMTSIgfSwKICAgIHsgIiIsICAgIk9FIiwgICJMQiIgfSwKICAgIHsgIkoiLCAgIllPIiwgICJMUyIgfSwKICAgIHsgIkpKIiwgIlUiLCAgICJMVCIgfSwKICAgIHsgIkMiLCAgIldFTyIsICJMUCIgfSwKICAgIHsgIksiLCAgIldFIiwgICJMSCIgfSwKICAgIHsgIlQiLCAgIldJIiwgICJNIiAgfSwKICAgIHsgIlAiLCAgIllVIiwgICJCIiAgfSwKICAgIHsgIkgiLCAgIkVVIiwgICJCUyIgfSwKICAgIHsgMCwgICAgIllJIiwgICJTIiAgfSwKICAgIHsgMCwgICAgIkkiLCAgICJTUyIgfSwKICAgIHsgMCwgICAgMCwgICAgICJORyIgfSwKICAgIHsgMCwgICAgMCwgICAgICJKIiAgfSwKICAgIHsgMCwgICAgMCwgICAgICJDIiAgfSwKICAgIHsgMCwgICAgMCwgICAgICJLIiAgfSwKICAgIHsgMCwgICAgMCwgICAgICJUIiAgfSwKICAgIHsgMCwgICAgMCwgICAgICJQIiAgfSwKICAgIHsgMCwgICAgMCwgICAgICJIIiAgfQp9OwoKc3RhdGljIGludAppc191bmlmaWVkX2lkZW9ncmFwaChQeV9VQ1M0IGNvZGUpCnsKICAgIHJldHVybiAoCiAgICAgICAgKDB4MzQwMCA8PSBjb2RlICYmIGNvZGUgPD0gMHg0REI1KSB8fCAvKiBDSksgSWRlb2dyYXBoIEV4dGVuc2lvbiBBICovCiAgICAgICAgKDB4NEUwMCA8PSBjb2RlICYmIGNvZGUgPD0gMHg5RkJCKSB8fCAvKiBDSksgSWRlb2dyYXBoICovCiAgICAgICAgKDB4MjAwMDAgPD0gY29kZSAmJiBjb2RlIDw9IDB4MkE2RDYpKTsvKiBDSksgSWRlb2dyYXBoIEV4dGVuc2lvbiBCICovCn0KCnN0YXRpYyBpbnQKX2dldHVjbmFtZShQeU9iamVjdCAqc2VsZiwgUHlfVUNTNCBjb2RlLCBjaGFyKiBidWZmZXIsIGludCBidWZsZW4pCnsKICAgIGludCBvZmZzZXQ7CiAgICBpbnQgaTsKICAgIGludCB3b3JkOwogICAgdW5zaWduZWQgY2hhciogdzsKCiAgICBpZiAoY29kZSA+PSAweDExMDAwMCkKICAgICAgICByZXR1cm4gMDsKCiAgICBpZiAoc2VsZiAmJiBVQ0RfQ2hlY2soc2VsZikpIHsKICAgICAgICBjb25zdCBjaGFuZ2VfcmVjb3JkICpvbGQgPSBnZXRfb2xkX3JlY29yZChzZWxmLCBjb2RlKTsKICAgICAgICBpZiAob2xkLT5jYXRlZ29yeV9jaGFuZ2VkID09IDApIHsKICAgICAgICAgICAgLyogdW5hc3NpZ25lZCAqLwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9IAogICAgfQoKICAgIGlmIChTQmFzZSA8PSBjb2RlICYmIGNvZGUgPCBTQmFzZStTQ291bnQpIHsKCS8qIEhhbmd1bCBzeWxsYWJsZS4gKi8KCWludCBTSW5kZXggPSBjb2RlIC0gU0Jhc2U7CglpbnQgTCA9IFNJbmRleCAvIE5Db3VudDsKCWludCBWID0gKFNJbmRleCAlIE5Db3VudCkgLyBUQ291bnQ7CglpbnQgVCA9IFNJbmRleCAlIFRDb3VudDsKCglpZiAoYnVmbGVuIDwgMjcpCgkgICAgLyogV29yc3QgY2FzZTogSEFOR1VMIFNZTExBQkxFIDwxMGNoYXJzPi4gKi8KCSAgICByZXR1cm4gMDsKCXN0cmNweShidWZmZXIsICJIQU5HVUwgU1lMTEFCTEUgIik7CglidWZmZXIgKz0gMTY7CglzdHJjcHkoYnVmZmVyLCBoYW5ndWxfc3lsbGFibGVzW0xdWzBdKTsKCWJ1ZmZlciArPSBzdHJsZW4oaGFuZ3VsX3N5bGxhYmxlc1tMXVswXSk7CglzdHJjcHkoYnVmZmVyLCBoYW5ndWxfc3lsbGFibGVzW1ZdWzFdKTsKCWJ1ZmZlciArPSBzdHJsZW4oaGFuZ3VsX3N5bGxhYmxlc1tWXVsxXSk7CglzdHJjcHkoYnVmZmVyLCBoYW5ndWxfc3lsbGFibGVzW1RdWzJdKTsKCWJ1ZmZlciArPSBzdHJsZW4oaGFuZ3VsX3N5bGxhYmxlc1tUXVsyXSk7CgkqYnVmZmVyID0gJ1wwJzsKCXJldHVybiAxOwogICAgfQoKICAgIGlmIChpc191bmlmaWVkX2lkZW9ncmFwaChjb2RlKSkgewogICAgICAgIGlmIChidWZsZW4gPCAyOCkKICAgICAgICAgICAgLyogV29yc3QgY2FzZTogQ0pLIFVOSUZJRUQgSURFT0dSQVBILTIwMDAwICovCiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIHNwcmludGYoYnVmZmVyLCAiQ0pLIFVOSUZJRUQgSURFT0dSQVBILSVYIiwgY29kZSk7CiAgICAgICAgcmV0dXJuIDE7CiAgICB9CgogICAgLyogZ2V0IG9mZnNldCBpbnRvIHBocmFzZWJvb2sgKi8KICAgIG9mZnNldCA9IHBocmFzZWJvb2tfb2Zmc2V0MVsoY29kZT4+cGhyYXNlYm9va19zaGlmdCldOwogICAgb2Zmc2V0ID0gcGhyYXNlYm9va19vZmZzZXQyWyhvZmZzZXQ8PHBocmFzZWJvb2tfc2hpZnQpICsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjb2RlJigoMTw8cGhyYXNlYm9va19zaGlmdCktMSkpXTsKICAgIGlmICghb2Zmc2V0KQogICAgICAgIHJldHVybiAwOwoKICAgIGkgPSAwOwoKICAgIGZvciAoOzspIHsKICAgICAgICAvKiBnZXQgd29yZCBpbmRleCAqLwogICAgICAgIHdvcmQgPSBwaHJhc2Vib29rW29mZnNldF0gLSBwaHJhc2Vib29rX3Nob3J0OwogICAgICAgIGlmICh3b3JkID49IDApIHsKICAgICAgICAgICAgd29yZCA9ICh3b3JkIDw8IDgpICsgcGhyYXNlYm9va1tvZmZzZXQrMV07CiAgICAgICAgICAgIG9mZnNldCArPSAyOwogICAgICAgIH0gZWxzZQogICAgICAgICAgICB3b3JkID0gcGhyYXNlYm9va1tvZmZzZXQrK107CiAgICAgICAgaWYgKGkpIHsKICAgICAgICAgICAgaWYgKGkgPiBidWZsZW4pCiAgICAgICAgICAgICAgICByZXR1cm4gMDsgLyogYnVmZmVyIG92ZXJmbG93ICovCiAgICAgICAgICAgIGJ1ZmZlcltpKytdID0gJyAnOwogICAgICAgIH0KICAgICAgICAvKiBjb3B5IHdvcmQgc3RyaW5nIGZyb20gbGV4aWNvbi4gIHRoZSBsYXN0IGNoYXJhY3RlciBpbiB0aGUKICAgICAgICAgICB3b3JkIGhhcyBiaXQgNyBzZXQuICB0aGUgbGFzdCB3b3JkIGluIGEgc3RyaW5nIGVuZHMgd2l0aAogICAgICAgICAgIDB4ODAgKi8KICAgICAgICB3ID0gbGV4aWNvbiArIGxleGljb25fb2Zmc2V0W3dvcmRdOwogICAgICAgIHdoaWxlICgqdyA8IDEyOCkgewogICAgICAgICAgICBpZiAoaSA+PSBidWZsZW4pCiAgICAgICAgICAgICAgICByZXR1cm4gMDsgLyogYnVmZmVyIG92ZXJmbG93ICovCiAgICAgICAgICAgIGJ1ZmZlcltpKytdID0gKncrKzsKICAgICAgICB9CiAgICAgICAgaWYgKGkgPj0gYnVmbGVuKQogICAgICAgICAgICByZXR1cm4gMDsgLyogYnVmZmVyIG92ZXJmbG93ICovCiAgICAgICAgYnVmZmVyW2krK10gPSAqdyAmIDEyNzsKICAgICAgICBpZiAoKncgPT0gMTI4KQogICAgICAgICAgICBicmVhazsgLyogZW5kIG9mIHdvcmQgKi8KICAgIH0KCiAgICByZXR1cm4gMTsKfQoKc3RhdGljIGludApfY21wbmFtZShQeU9iamVjdCAqc2VsZiwgaW50IGNvZGUsIGNvbnN0IGNoYXIqIG5hbWUsIGludCBuYW1lbGVuKQp7CiAgICAvKiBjaGVjayBpZiBjb2RlIGNvcnJlc3BvbmRzIHRvIHRoZSBnaXZlbiBuYW1lICovCiAgICBpbnQgaTsKICAgIGNoYXIgYnVmZmVyW05BTUVfTUFYTEVOXTsKICAgIGlmICghX2dldHVjbmFtZShzZWxmLCBjb2RlLCBidWZmZXIsIHNpemVvZihidWZmZXIpKSkKICAgICAgICByZXR1cm4gMDsKICAgIGZvciAoaSA9IDA7IGkgPCBuYW1lbGVuOyBpKyspIHsKICAgICAgICBpZiAodG91cHBlcihQeV9DSEFSTUFTSyhuYW1lW2ldKSkgIT0gYnVmZmVyW2ldKQogICAgICAgICAgICByZXR1cm4gMDsKICAgIH0KICAgIHJldHVybiBidWZmZXJbbmFtZWxlbl0gPT0gJ1wwJzsKfQoKc3RhdGljIHZvaWQgCmZpbmRfc3lsbGFibGUoY29uc3QgY2hhciAqc3RyLCBpbnQgKmxlbiwgaW50ICpwb3MsIGludCBjb3VudCwgaW50IGNvbHVtbikKewogICAgaW50IGksIGxlbjE7CiAgICAqbGVuID0gLTE7CiAgICBmb3IgKGkgPSAwOyBpIDwgY291bnQ7IGkrKykgewoJY2hhciAqcyA9IGhhbmd1bF9zeWxsYWJsZXNbaV1bY29sdW1uXTsKCWxlbjEgPSBzdHJsZW4ocyk7CglpZiAobGVuMSA8PSAqbGVuKQoJICAgIGNvbnRpbnVlOwoJaWYgKHN0cm5jbXAoc3RyLCBzLCBsZW4xKSA9PSAwKSB7CgkgICAgKmxlbiA9IGxlbjE7CgkgICAgKnBvcyA9IGk7Cgl9CiAgICB9CiAgICBpZiAoKmxlbiA9PSAtMSkgewoJKmxlbiA9IDA7CiAgICB9Cn0KCnN0YXRpYyBpbnQKX2dldGNvZGUoUHlPYmplY3QqIHNlbGYsIGNvbnN0IGNoYXIqIG5hbWUsIGludCBuYW1lbGVuLCBQeV9VQ1M0KiBjb2RlKQp7CiAgICB1bnNpZ25lZCBpbnQgaCwgdjsKICAgIHVuc2lnbmVkIGludCBtYXNrID0gY29kZV9zaXplLTE7CiAgICB1bnNpZ25lZCBpbnQgaSwgaW5jcjsKCiAgICAvKiBDaGVjayBmb3IgaGFuZ3VsIHN5bGxhYmxlcy4gKi8KICAgIGlmIChzdHJuY21wKG5hbWUsICJIQU5HVUwgU1lMTEFCTEUgIiwgMTYpID09IDApIHsKCWludCBsZW4sIEwgPSAtMSwgViA9IC0xLCBUID0gLTE7Cgljb25zdCBjaGFyICpwb3MgPSBuYW1lICsgMTY7CglmaW5kX3N5bGxhYmxlKHBvcywgJmxlbiwgJkwsIExDb3VudCwgMCk7Cglwb3MgKz0gbGVuOwoJZmluZF9zeWxsYWJsZShwb3MsICZsZW4sICZWLCBWQ291bnQsIDEpOwoJcG9zICs9IGxlbjsKCWZpbmRfc3lsbGFibGUocG9zLCAmbGVuLCAmVCwgVENvdW50LCAyKTsKCXBvcyArPSBsZW47CglpZiAoTCAhPSAtMSAmJiBWICE9IC0xICYmIFQgIT0gLTEgJiYgcG9zLW5hbWUgPT0gbmFtZWxlbikgewoJICAgICpjb2RlID0gU0Jhc2UgKyAoTCpWQ291bnQrVikqVENvdW50ICsgVDsKCSAgICByZXR1cm4gMTsKCX0KICAgICAgICAvKiBPdGhlcndpc2UsIGl0J3MgYW4gaWxsZWdhbCBzeWxsYWJsZSBuYW1lLiAqLwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIC8qIENoZWNrIGZvciB1bmlmaWVkIGlkZW9ncmFwaHMuICovCiAgICBpZiAoc3RybmNtcChuYW1lLCAiQ0pLIFVOSUZJRUQgSURFT0dSQVBILSIsIDIyKSA9PSAwKSB7CiAgICAgICAgLyogRm91ciBvciBmaXZlIGhleGRpZ2l0cyBtdXN0IGZvbGxvdy4gKi8KICAgICAgICB2ID0gMDsKICAgICAgICBuYW1lICs9IDIyOwogICAgICAgIG5hbWVsZW4gLT0gMjI7CiAgICAgICAgaWYgKG5hbWVsZW4gIT0gNCAmJiBuYW1lbGVuICE9IDUpCiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIHdoaWxlIChuYW1lbGVuLS0pIHsKICAgICAgICAgICAgdiAqPSAxNjsKICAgICAgICAgICAgaWYgKCpuYW1lID49ICcwJyAmJiAqbmFtZSA8PSAnOScpCiAgICAgICAgICAgICAgICB2ICs9ICpuYW1lIC0gJzAnOwogICAgICAgICAgICBlbHNlIGlmICgqbmFtZSA+PSAnQScgJiYgKm5hbWUgPD0gJ0YnKQogICAgICAgICAgICAgICAgdiArPSAqbmFtZSAtICdBJyArIDEwOwogICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICAgICAgbmFtZSsrOwogICAgICAgIH0KICAgICAgICBpZiAoIWlzX3VuaWZpZWRfaWRlb2dyYXBoKHYpKQogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICAqY29kZSA9IHY7CiAgICAgICAgcmV0dXJuIDE7CiAgICB9CgogICAgLyogdGhlIGZvbGxvd2luZyBpcyB0aGUgc2FtZSBhcyBweXRob24ncyBkaWN0aW9uYXJ5IGxvb2t1cCwgd2l0aAogICAgICAgb25seSBtaW5vciBjaGFuZ2VzLiAgc2VlIHRoZSBtYWtldW5pY29kZWRhdGEgc2NyaXB0IGZvciBtb3JlCiAgICAgICBkZXRhaWxzICovCgogICAgaCA9ICh1bnNpZ25lZCBpbnQpIF9nZXRoYXNoKG5hbWUsIG5hbWVsZW4sIGNvZGVfbWFnaWMpOwogICAgaSA9ICh+aCkgJiBtYXNrOwogICAgdiA9IGNvZGVfaGFzaFtpXTsKICAgIGlmICghdikKICAgICAgICByZXR1cm4gMDsKICAgIGlmIChfY21wbmFtZShzZWxmLCB2LCBuYW1lLCBuYW1lbGVuKSkgewogICAgICAgICpjb2RlID0gdjsKICAgICAgICByZXR1cm4gMTsKICAgIH0KICAgIGluY3IgPSAoaCBeIChoID4+IDMpKSAmIG1hc2s7CiAgICBpZiAoIWluY3IpCiAgICAgICAgaW5jciA9IG1hc2s7CiAgICBmb3IgKDs7KSB7CiAgICAgICAgaSA9IChpICsgaW5jcikgJiBtYXNrOwogICAgICAgIHYgPSBjb2RlX2hhc2hbaV07CiAgICAgICAgaWYgKCF2KQogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICBpZiAoX2NtcG5hbWUoc2VsZiwgdiwgbmFtZSwgbmFtZWxlbikpIHsKICAgICAgICAgICAgKmNvZGUgPSB2OwogICAgICAgICAgICByZXR1cm4gMTsKICAgICAgICB9CiAgICAgICAgaW5jciA9IGluY3IgPDwgMTsKICAgICAgICBpZiAoaW5jciA+IG1hc2spCiAgICAgICAgICAgIGluY3IgPSBpbmNyIF4gY29kZV9wb2x5OwogICAgfQp9CgpzdGF0aWMgY29uc3QgX1B5VW5pY29kZV9OYW1lX0NBUEkgaGFzaEFQSSA9IAp7CiAgICBzaXplb2YoX1B5VW5pY29kZV9OYW1lX0NBUEkpLAogICAgX2dldHVjbmFtZSwKICAgIF9nZXRjb2RlCn07CgovKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAqLwovKiBQeXRob24gYmluZGluZ3MgKi8KClB5RG9jX1NUUlZBUih1bmljb2RlZGF0YV9uYW1lX19kb2NfXywKIm5hbWUodW5pY2hyWywgZGVmYXVsdF0pXG5cClJldHVybnMgdGhlIG5hbWUgYXNzaWduZWQgdG8gdGhlIFVuaWNvZGUgY2hhcmFjdGVyIHVuaWNociBhcyBhXG5cCnN0cmluZy4gSWYgbm8gbmFtZSBpcyBkZWZpbmVkLCBkZWZhdWx0IGlzIHJldHVybmVkLCBvciwgaWYgbm90XG5cCmdpdmVuLCBWYWx1ZUVycm9yIGlzIHJhaXNlZC4iKTsKCnN0YXRpYyBQeU9iamVjdCAqCnVuaWNvZGVkYXRhX25hbWUoUHlPYmplY3QqIHNlbGYsIFB5T2JqZWN0KiBhcmdzKQp7CiAgICBjaGFyIG5hbWVbTkFNRV9NQVhMRU5dOwogICAgUHlfVUNTNCBjOwoKICAgIFB5VW5pY29kZU9iamVjdCogdjsKICAgIFB5T2JqZWN0KiBkZWZvYmogPSBOVUxMOwogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlKGFyZ3MsICJPIXxPOm5hbWUiLCAmUHlVbmljb2RlX1R5cGUsICZ2LCAmZGVmb2JqKSkKICAgICAgICByZXR1cm4gTlVMTDsKCiAgICBjID0gZ2V0dWNoYXIodik7CiAgICBpZiAoYyA9PSAoUHlfVUNTNCktMSkKICAgICAgICByZXR1cm4gTlVMTDsKCiAgICBpZiAoIV9nZXR1Y25hbWUoc2VsZiwgYywgbmFtZSwgc2l6ZW9mKG5hbWUpKSkgewoJaWYgKGRlZm9iaiA9PSBOVUxMKSB7CgkgICAgUHlFcnJfU2V0U3RyaW5nKFB5RXhjX1ZhbHVlRXJyb3IsICJubyBzdWNoIG5hbWUiKTsKICAgICAgICAgICAgcmV0dXJuIE5VTEw7Cgl9CgllbHNlIHsKCSAgICBQeV9JTkNSRUYoZGVmb2JqKTsKCSAgICByZXR1cm4gZGVmb2JqOwoJfQogICAgfQoKICAgIHJldHVybiBQeVVuaWNvZGVfRnJvbVN0cmluZyhuYW1lKTsKfQoKUHlEb2NfU1RSVkFSKHVuaWNvZGVkYXRhX2xvb2t1cF9fZG9jX18sCiJsb29rdXAobmFtZSlcblwKXG5cCkxvb2sgdXAgY2hhcmFjdGVyIGJ5IG5hbWUuICBJZiBhIGNoYXJhY3RlciB3aXRoIHRoZVxuXApnaXZlbiBuYW1lIGlzIGZvdW5kLCByZXR1cm4gdGhlIGNvcnJlc3BvbmRpbmcgVW5pY29kZVxuXApjaGFyYWN0ZXIuICBJZiBub3QgZm91bmQsIEtleUVycm9yIGlzIHJhaXNlZC4iKTsKCnN0YXRpYyBQeU9iamVjdCAqCnVuaWNvZGVkYXRhX2xvb2t1cChQeU9iamVjdCogc2VsZiwgUHlPYmplY3QqIGFyZ3MpCnsKICAgIFB5X1VDUzQgY29kZTsKICAgIFB5X1VOSUNPREUgc3RyWzJdOwoKICAgIGNoYXIqIG5hbWU7CiAgICBpbnQgbmFtZWxlbjsKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZShhcmdzLCAicyM6bG9va3VwIiwgJm5hbWUsICZuYW1lbGVuKSkKICAgICAgICByZXR1cm4gTlVMTDsKCiAgICBpZiAoIV9nZXRjb2RlKHNlbGYsIG5hbWUsIG5hbWVsZW4sICZjb2RlKSkgewogICAgICAgIFB5RXJyX0Zvcm1hdChQeUV4Y19LZXlFcnJvciwgInVuZGVmaW5lZCBjaGFyYWN0ZXIgbmFtZSAnJXMnIiwKICAgICAgICAgICAgICAgICAgICAgbmFtZSk7CiAgICAgICAgcmV0dXJuIE5VTEw7CiAgICB9CgojaWZuZGVmIFB5X1VOSUNPREVfV0lERQogICAgaWYgKGNvZGUgPj0gMHgxMDAwMCkgewogICAgICAgIHN0clswXSA9IDB4ZDgwMCArICgoY29kZSAtIDB4MTAwMDApID4+IDEwKTsKICAgICAgICBzdHJbMV0gPSAweGRjMDAgKyAoKGNvZGUgLSAweDEwMDAwKSAmIDB4M2ZmKTsKICAgICAgICByZXR1cm4gUHlVbmljb2RlX0Zyb21Vbmljb2RlKHN0ciwgMik7CiAgICB9CiNlbmRpZgogICAgc3RyWzBdID0gKFB5X1VOSUNPREUpIGNvZGU7CiAgICByZXR1cm4gUHlVbmljb2RlX0Zyb21Vbmljb2RlKHN0ciwgMSk7ICAgIAp9CgovKiBYWFggQWRkIGRvYyBzdHJpbmdzLiAqLwoKc3RhdGljIFB5TWV0aG9kRGVmIHVuaWNvZGVkYXRhX2Z1bmN0aW9uc1tdID0gewogICAgeyJkZWNpbWFsIiwgdW5pY29kZWRhdGFfZGVjaW1hbCwgTUVUSF9WQVJBUkdTLCB1bmljb2RlZGF0YV9kZWNpbWFsX19kb2NfX30sCiAgICB7ImRpZ2l0IiwgdW5pY29kZWRhdGFfZGlnaXQsIE1FVEhfVkFSQVJHUywgdW5pY29kZWRhdGFfZGlnaXRfX2RvY19ffSwKICAgIHsibnVtZXJpYyIsIHVuaWNvZGVkYXRhX251bWVyaWMsIE1FVEhfVkFSQVJHUywgdW5pY29kZWRhdGFfbnVtZXJpY19fZG9jX199LAogICAgeyJjYXRlZ29yeSIsIHVuaWNvZGVkYXRhX2NhdGVnb3J5LCBNRVRIX1ZBUkFSR1MsCiAgICAgICAgICAgICAgICAgdW5pY29kZWRhdGFfY2F0ZWdvcnlfX2RvY19ffSwKICAgIHsiYmlkaXJlY3Rpb25hbCIsIHVuaWNvZGVkYXRhX2JpZGlyZWN0aW9uYWwsIE1FVEhfVkFSQVJHUywKICAgICAgICAgICAgICAgICAgICAgIHVuaWNvZGVkYXRhX2JpZGlyZWN0aW9uYWxfX2RvY19ffSwKICAgIHsiY29tYmluaW5nIiwgdW5pY29kZWRhdGFfY29tYmluaW5nLCBNRVRIX1ZBUkFSR1MsCiAgICAgICAgICAgICAgICAgIHVuaWNvZGVkYXRhX2NvbWJpbmluZ19fZG9jX199LAogICAgeyJtaXJyb3JlZCIsIHVuaWNvZGVkYXRhX21pcnJvcmVkLCBNRVRIX1ZBUkFSR1MsCiAgICAgICAgICAgICAgICAgdW5pY29kZWRhdGFfbWlycm9yZWRfX2RvY19ffSwKICAgIHsiZWFzdF9hc2lhbl93aWR0aCIsIHVuaWNvZGVkYXRhX2Vhc3RfYXNpYW5fd2lkdGgsIE1FVEhfVkFSQVJHUywKICAgICAgICAgICAgICAgICAgICAgICAgIHVuaWNvZGVkYXRhX2Vhc3RfYXNpYW5fd2lkdGhfX2RvY19ffSwKICAgIHsiZGVjb21wb3NpdGlvbiIsIHVuaWNvZGVkYXRhX2RlY29tcG9zaXRpb24sIE1FVEhfVkFSQVJHUywKICAgICAgICAgICAgICAgICAgICAgIHVuaWNvZGVkYXRhX2RlY29tcG9zaXRpb25fX2RvY19ffSwKICAgIHsibmFtZSIsIHVuaWNvZGVkYXRhX25hbWUsIE1FVEhfVkFSQVJHUywgdW5pY29kZWRhdGFfbmFtZV9fZG9jX199LAogICAgeyJsb29rdXAiLCB1bmljb2RlZGF0YV9sb29rdXAsIE1FVEhfVkFSQVJHUywgdW5pY29kZWRhdGFfbG9va3VwX19kb2NfX30sCiAgICB7Im5vcm1hbGl6ZSIsIHVuaWNvZGVkYXRhX25vcm1hbGl6ZSwgTUVUSF9WQVJBUkdTLAogICAgICAgICAgICAgICAgICB1bmljb2RlZGF0YV9ub3JtYWxpemVfX2RvY19ffSwKICAgIHtOVUxMLCBOVUxMfQkJLyogc2VudGluZWwgKi8KfTsKCnN0YXRpYyBQeVR5cGVPYmplY3QgVUNEX1R5cGUgPSB7CgkvKiBUaGUgb2JfdHlwZSBmaWVsZCBtdXN0IGJlIGluaXRpYWxpemVkIGluIHRoZSBtb2R1bGUgaW5pdCBmdW5jdGlvbgoJICogdG8gYmUgcG9ydGFibGUgdG8gV2luZG93cyB3aXRob3V0IHVzaW5nIEMrKy4gKi8KCVB5VmFyT2JqZWN0X0hFQURfSU5JVChOVUxMLCAwKQoJInVuaWNvZGVkYXRhLlVDRCIsCQkvKnRwX25hbWUqLwoJc2l6ZW9mKFByZXZpb3VzREJWZXJzaW9uKSwJLyp0cF9iYXNpY3NpemUqLwoJMCwJCQkvKnRwX2l0ZW1zaXplKi8KCS8qIG1ldGhvZHMgKi8KCShkZXN0cnVjdG9yKVB5T2JqZWN0X0RlbCwgLyp0cF9kZWFsbG9jKi8KCTAsCQkJLyp0cF9wcmludCovCgkwLCAgICAgICAgICAgICAgICAgICAgICAvKnRwX2dldGF0dHIqLwoJMCwJCQkvKnRwX3NldGF0dHIqLwoJMCwJCQkvKnRwX3Jlc2VydmVkKi8KCTAsCQkJLyp0cF9yZXByKi8KCTAsCQkJLyp0cF9hc19udW1iZXIqLwoJMCwJCQkvKnRwX2FzX3NlcXVlbmNlKi8KCTAsCQkJLyp0cF9hc19tYXBwaW5nKi8KCTAsCQkJLyp0cF9oYXNoKi8KICAgICAgICAwLCAgICAgICAgICAgICAgICAgICAgICAvKnRwX2NhbGwqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfc3RyKi8KICAgICAgICBQeU9iamVjdF9HZW5lcmljR2V0QXR0ciwvKnRwX2dldGF0dHJvKi8KICAgICAgICAwLCAgICAgICAgICAgICAgICAgICAgICAvKnRwX3NldGF0dHJvKi8KICAgICAgICAwLCAgICAgICAgICAgICAgICAgICAgICAvKnRwX2FzX2J1ZmZlciovCiAgICAgICAgUHlfVFBGTEFHU19ERUZBVUxULCAgICAgLyp0cF9mbGFncyovCiAgICAgICAgMCwgICAgICAgICAgICAgICAgICAgICAgLyp0cF9kb2MqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfdHJhdmVyc2UqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfY2xlYXIqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfcmljaGNvbXBhcmUqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfd2Vha2xpc3RvZmZzZXQqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfaXRlciovCiAgICAgICAgMCwgICAgICAgICAgICAgICAgICAgICAgLyp0cF9pdGVybmV4dCovCiAgICAgICAgdW5pY29kZWRhdGFfZnVuY3Rpb25zLCAgLyp0cF9tZXRob2RzKi8KICAgICAgICBEQl9tZW1iZXJzLCAgICAgICAgICAgICAvKnRwX21lbWJlcnMqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfZ2V0c2V0Ki8KICAgICAgICAwLCAgICAgICAgICAgICAgICAgICAgICAvKnRwX2Jhc2UqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfZGljdCovCiAgICAgICAgMCwgICAgICAgICAgICAgICAgICAgICAgLyp0cF9kZXNjcl9nZXQqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfZGVzY3Jfc2V0Ki8KICAgICAgICAwLCAgICAgICAgICAgICAgICAgICAgICAvKnRwX2RpY3RvZmZzZXQqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfaW5pdCovCiAgICAgICAgMCwgICAgICAgICAgICAgICAgICAgICAgLyp0cF9hbGxvYyovCiAgICAgICAgMCwgICAgICAgICAgICAgICAgICAgICAgLyp0cF9uZXcqLwogICAgICAgIDAsICAgICAgICAgICAgICAgICAgICAgIC8qdHBfZnJlZSovCiAgICAgICAgMCwgICAgICAgICAgICAgICAgICAgICAgLyp0cF9pc19nYyovCn07CgpQeURvY19TVFJWQVIodW5pY29kZWRhdGFfZG9jc3RyaW5nLAoiVGhpcyBtb2R1bGUgcHJvdmlkZXMgYWNjZXNzIHRvIHRoZSBVbmljb2RlIENoYXJhY3RlciBEYXRhYmFzZSB3aGljaFxuXApkZWZpbmVzIGNoYXJhY3RlciBwcm9wZXJ0aWVzIGZvciBhbGwgVW5pY29kZSBjaGFyYWN0ZXJzLiBUaGUgZGF0YSBpblxuXAp0aGlzIGRhdGFiYXNlIGlzIGJhc2VkIG9uIHRoZSBVbmljb2RlRGF0YS50eHQgZmlsZSB2ZXJzaW9uXG5cCjUuMS4wIHdoaWNoIGlzIHB1YmxpY2FsbHkgYXZhaWxhYmxlIGZyb20gZnRwOi8vZnRwLnVuaWNvZGUub3JnLy5cblwKXG5cClRoZSBtb2R1bGUgdXNlcyB0aGUgc2FtZSBuYW1lcyBhbmQgc3ltYm9scyBhcyBkZWZpbmVkIGJ5IHRoZVxuXApVbmljb2RlRGF0YSBGaWxlIEZvcm1hdCA1LjEuMCAoc2VlXG5cCmh0dHA6Ly93d3cudW5pY29kZS5vcmcvUHVibGljLzUuMS4wL3VjZC9VQ0QuaHRtbCkuIik7CgoKc3RhdGljIHN0cnVjdCBQeU1vZHVsZURlZiB1bmljb2RlZGF0YW1vZHVsZSA9IHsKCVB5TW9kdWxlRGVmX0hFQURfSU5JVCwKCSJ1bmljb2RlZGF0YSIsCgl1bmljb2RlZGF0YV9kb2NzdHJpbmcsCgktMSwKCXVuaWNvZGVkYXRhX2Z1bmN0aW9ucywKCU5VTEwsCglOVUxMLAoJTlVMTCwKCU5VTEwKfTsKClB5TU9ESU5JVF9GVU5DClB5SW5pdF91bmljb2RlZGF0YSh2b2lkKQp7CiAgICBQeU9iamVjdCAqbSwgKnY7CgogICAgUHlfVFlQRSgmVUNEX1R5cGUpID0gJlB5VHlwZV9UeXBlOwoKICAgIG0gPSBQeU1vZHVsZV9DcmVhdGUoJnVuaWNvZGVkYXRhbW9kdWxlKTsKICAgIGlmICghbSkKICAgICAgICByZXR1cm4gTlVMTDsKCiAgICBQeU1vZHVsZV9BZGRTdHJpbmdDb25zdGFudChtLCAidW5pZGF0YV92ZXJzaW9uIiwgVU5JREFUQV9WRVJTSU9OKTsKICAgIFB5X0lOQ1JFRigmVUNEX1R5cGUpOwogICAgUHlNb2R1bGVfQWRkT2JqZWN0KG0sICJVQ0QiLCAoUHlPYmplY3QqKSZVQ0RfVHlwZSk7CgogICAgLyogUHJldmlvdXMgdmVyc2lvbnMgKi8KICAgIHYgPSBuZXdfcHJldmlvdXNfdmVyc2lvbigiMy4yLjAiLCBnZXRfY2hhbmdlXzNfMl8wLCBub3JtYWxpemF0aW9uXzNfMl8wKTsKICAgIGlmICh2ICE9IE5VTEwpCiAgICAgICAgUHlNb2R1bGVfQWRkT2JqZWN0KG0sICJ1Y2RfM18yXzAiLCB2KTsKCiAgICAvKiBFeHBvcnQgQyBBUEkgKi8KICAgIHYgPSBQeUNPYmplY3RfRnJvbVZvaWRQdHIoKHZvaWQgKikgJmhhc2hBUEksIE5VTEwpOwogICAgaWYgKHYgIT0gTlVMTCkKICAgICAgICBQeU1vZHVsZV9BZGRPYmplY3QobSwgInVjbmhhc2hfQ0FQSSIsIHYpOwogICAgcmV0dXJuIG07Cn0KCi8qIApMb2NhbCB2YXJpYWJsZXM6CmMtYmFzaWMtb2Zmc2V0OiA0CmluZGVudC10YWJzLW1vZGU6IG5pbApFbmQ6CiovCg==