LyoKICogZW50aXRpZXMuYyA6IGltcGxlbWVudGF0aW9uIGZvciB0aGUgWE1MIGVudGl0aWVzIGhhbmRsaW5nCiAqCiAqIFNlZSBDb3B5cmlnaHQgZm9yIHRoZSBzdGF0dXMgb2YgdGhpcyBzb2Z0d2FyZS4KICoKICogZGFuaWVsQHZlaWxsYXJkLmNvbQogKi8KCiNkZWZpbmUgSU5fTElCWE1MCiNpbmNsdWRlICJsaWJ4bWwuaCIKCiNpbmNsdWRlIDxzdHJpbmcuaD4KI2lmZGVmIEhBVkVfU1RETElCX0gKI2luY2x1ZGUgPHN0ZGxpYi5oPgojZW5kaWYKI2luY2x1ZGUgPGxpYnhtbC94bWxtZW1vcnkuaD4KI2luY2x1ZGUgPGxpYnhtbC9oYXNoLmg+CiNpbmNsdWRlIDxsaWJ4bWwvZW50aXRpZXMuaD4KI2luY2x1ZGUgPGxpYnhtbC9wYXJzZXIuaD4KI2luY2x1ZGUgPGxpYnhtbC9wYXJzZXJJbnRlcm5hbHMuaD4KI2luY2x1ZGUgPGxpYnhtbC94bWxlcnJvci5oPgojaW5jbHVkZSA8bGlieG1sL2dsb2JhbHMuaD4KCi8qCiAqIFRoZSBYTUwgcHJlZGVmaW5lZCBlbnRpdGllcy4KICovCgpzdGF0aWMgeG1sRW50aXR5IHhtbEVudGl0eUx0ID0gewogICAgTlVMTCwgWE1MX0VOVElUWV9ERUNMLCBCQURfQ0FTVCAibHQiLAogICAgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgCiAgICBCQURfQ0FTVCAiPCIsIEJBRF9DQVNUICI8IiwgMSwKICAgIFhNTF9JTlRFUk5BTF9QUkVERUZJTkVEX0VOVElUWSwKICAgIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIDAKfTsKc3RhdGljIHhtbEVudGl0eSB4bWxFbnRpdHlHdCA9IHsKICAgIE5VTEwsIFhNTF9FTlRJVFlfREVDTCwgQkFEX0NBU1QgImd0IiwKICAgIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIAogICAgQkFEX0NBU1QgIj4iLCBCQURfQ0FTVCAiPiIsIDEsCiAgICBYTUxfSU5URVJOQUxfUFJFREVGSU5FRF9FTlRJVFksCiAgICBOVUxMLCBOVUxMLCBOVUxMLCBOVUxMLCAwCn07CnN0YXRpYyB4bWxFbnRpdHkgeG1sRW50aXR5QW1wID0gewogICAgTlVMTCwgWE1MX0VOVElUWV9ERUNMLCBCQURfQ0FTVCAiYW1wIiwKICAgIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIAogICAgQkFEX0NBU1QgIiYiLCBCQURfQ0FTVCAiJiIsIDEsCiAgICBYTUxfSU5URVJOQUxfUFJFREVGSU5FRF9FTlRJVFksCiAgICBOVUxMLCBOVUxMLCBOVUxMLCBOVUxMLCAwCn07CnN0YXRpYyB4bWxFbnRpdHkgeG1sRW50aXR5UXVvdCA9IHsKICAgIE5VTEwsIFhNTF9FTlRJVFlfREVDTCwgQkFEX0NBU1QgInF1b3QiLAogICAgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgCiAgICBCQURfQ0FTVCAiXCIiLCBCQURfQ0FTVCAiXCIiLCAxLAogICAgWE1MX0lOVEVSTkFMX1BSRURFRklORURfRU5USVRZLAogICAgTlVMTCwgTlVMTCwgTlVMTCwgTlVMTCwgMAp9OwpzdGF0aWMgeG1sRW50aXR5IHhtbEVudGl0eUFwb3MgPSB7CiAgICBOVUxMLCBYTUxfRU5USVRZX0RFQ0wsIEJBRF9DQVNUICJhcG9zIiwKICAgIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIE5VTEwsIAogICAgQkFEX0NBU1QgIiciLCBCQURfQ0FTVCAiJyIsIDEsCiAgICBYTUxfSU5URVJOQUxfUFJFREVGSU5FRF9FTlRJVFksCiAgICBOVUxMLCBOVUxMLCBOVUxMLCBOVUxMLCAwCn07CgovKioKICogeG1sRW50aXRpZXNFcnJNZW1vcnk6CiAqIEBleHRyYTogIGV4dHJhIGluZm9ybWF0aW9ucwogKgogKiBIYW5kbGUgYW4gb3V0IG9mIG1lbW9yeSBjb25kaXRpb24KICovCnN0YXRpYyB2b2lkCnhtbEVudGl0aWVzRXJyTWVtb3J5KGNvbnN0IGNoYXIgKmV4dHJhKQp7CiAgICBfX3htbFNpbXBsZUVycm9yKFhNTF9GUk9NX1RSRUUsIFhNTF9FUlJfTk9fTUVNT1JZLCBOVUxMLCBOVUxMLCBleHRyYSk7Cn0KCi8qKgogKiB4bWxFbnRpdGllc0VycjoKICogQGNvZGU6ICB0aGUgZXJyb3IgY29kZQogKiBAbXNnOiAgdGhlIG1lc3NhZ2UKICoKICogSGFuZGxlIGFuIG91dCBvZiBtZW1vcnkgY29uZGl0aW9uCiAqLwpzdGF0aWMgdm9pZAp4bWxFbnRpdGllc0Vycih4bWxQYXJzZXJFcnJvcnMgY29kZSwgY29uc3QgY2hhciAqbXNnKQp7CiAgICBfX3htbFNpbXBsZUVycm9yKFhNTF9GUk9NX1RSRUUsIGNvZGUsIE5VTEwsIG1zZywgTlVMTCk7Cn0KCi8qCiAqIHhtbEZyZWVFbnRpdHkgOiBjbGVhbi11cCBhbiBlbnRpdHkgcmVjb3JkLgogKi8Kc3RhdGljIHZvaWQgeG1sRnJlZUVudGl0eSh4bWxFbnRpdHlQdHIgZW50aXR5KSB7CiAgICBpZiAoZW50aXR5ID09IE5VTEwpIHJldHVybjsKCiAgICBpZiAoKGVudGl0eS0+Y2hpbGRyZW4pICYmIChlbnRpdHktPm93bmVyID09IDEpICYmCgkoZW50aXR5ID09ICh4bWxFbnRpdHlQdHIpIGVudGl0eS0+Y2hpbGRyZW4tPnBhcmVudCkpCgl4bWxGcmVlTm9kZUxpc3QoZW50aXR5LT5jaGlsZHJlbik7CiAgICBpZiAoZW50aXR5LT5uYW1lICE9IE5VTEwpCgl4bWxGcmVlKChjaGFyICopIGVudGl0eS0+bmFtZSk7CiAgICBpZiAoZW50aXR5LT5FeHRlcm5hbElEICE9IE5VTEwpCiAgICAgICAgeG1sRnJlZSgoY2hhciAqKSBlbnRpdHktPkV4dGVybmFsSUQpOwogICAgaWYgKGVudGl0eS0+U3lzdGVtSUQgIT0gTlVMTCkKICAgICAgICB4bWxGcmVlKChjaGFyICopIGVudGl0eS0+U3lzdGVtSUQpOwogICAgaWYgKGVudGl0eS0+VVJJICE9IE5VTEwpCiAgICAgICAgeG1sRnJlZSgoY2hhciAqKSBlbnRpdHktPlVSSSk7CiAgICBpZiAoZW50aXR5LT5jb250ZW50ICE9IE5VTEwpCiAgICAgICAgeG1sRnJlZSgoY2hhciAqKSBlbnRpdHktPmNvbnRlbnQpOwogICAgaWYgKGVudGl0eS0+b3JpZyAhPSBOVUxMKQogICAgICAgIHhtbEZyZWUoKGNoYXIgKikgZW50aXR5LT5vcmlnKTsKICAgIHhtbEZyZWUoZW50aXR5KTsKfQoKLyoKICogeG1sQWRkRW50aXR5IDogcmVnaXN0ZXIgYSBuZXcgZW50aXR5IGZvciBhbiBlbnRpdGllcyB0YWJsZS4KICovCnN0YXRpYyB4bWxFbnRpdHlQdHIKeG1sQWRkRW50aXR5KHhtbER0ZFB0ciBkdGQsIGNvbnN0IHhtbENoYXIgKm5hbWUsIGludCB0eXBlLAoJICBjb25zdCB4bWxDaGFyICpFeHRlcm5hbElELCBjb25zdCB4bWxDaGFyICpTeXN0ZW1JRCwKCSAgY29uc3QgeG1sQ2hhciAqY29udGVudCkgewogICAgeG1sRW50aXRpZXNUYWJsZVB0ciB0YWJsZSA9IE5VTEw7CiAgICB4bWxFbnRpdHlQdHIgcmV0OwoKICAgIGlmIChuYW1lID09IE5VTEwpCglyZXR1cm4oTlVMTCk7CiAgICBzd2l0Y2ggKHR5cGUpIHsKICAgICAgICBjYXNlIFhNTF9JTlRFUk5BTF9HRU5FUkFMX0VOVElUWToKICAgICAgICBjYXNlIFhNTF9FWFRFUk5BTF9HRU5FUkFMX1BBUlNFRF9FTlRJVFk6CiAgICAgICAgY2FzZSBYTUxfRVhURVJOQUxfR0VORVJBTF9VTlBBUlNFRF9FTlRJVFk6CgkgICAgaWYgKGR0ZC0+ZW50aXRpZXMgPT0gTlVMTCkKCQlkdGQtPmVudGl0aWVzID0geG1sSGFzaENyZWF0ZSgwKTsKCSAgICB0YWJsZSA9IGR0ZC0+ZW50aXRpZXM7CgkgICAgYnJlYWs7CiAgICAgICAgY2FzZSBYTUxfSU5URVJOQUxfUEFSQU1FVEVSX0VOVElUWToKICAgICAgICBjYXNlIFhNTF9FWFRFUk5BTF9QQVJBTUVURVJfRU5USVRZOgoJICAgIGlmIChkdGQtPnBlbnRpdGllcyA9PSBOVUxMKQoJCWR0ZC0+cGVudGl0aWVzID0geG1sSGFzaENyZWF0ZSgwKTsKCSAgICB0YWJsZSA9IGR0ZC0+cGVudGl0aWVzOwoJICAgIGJyZWFrOwogICAgICAgIGNhc2UgWE1MX0lOVEVSTkFMX1BSRURFRklORURfRU5USVRZOgoJICAgIHJldHVybihOVUxMKTsKICAgIH0KICAgIGlmICh0YWJsZSA9PSBOVUxMKQoJcmV0dXJuKE5VTEwpOwogICAgcmV0ID0gKHhtbEVudGl0eVB0cikgeG1sTWFsbG9jKHNpemVvZih4bWxFbnRpdHkpKTsKICAgIGlmIChyZXQgPT0gTlVMTCkgewogICAgICAgIHhtbEVudGl0aWVzRXJyTWVtb3J5KCJ4bWxBZGRFbnRpdHk6OiBtYWxsb2MgZmFpbGVkIik7CglyZXR1cm4oTlVMTCk7CiAgICB9CiAgICBtZW1zZXQocmV0LCAwLCBzaXplb2YoeG1sRW50aXR5KSk7CiAgICByZXQtPnR5cGUgPSBYTUxfRU5USVRZX0RFQ0w7CgogICAgLyoKICAgICAqIGZpbGwgdGhlIHN0cnVjdHVyZS4KICAgICAqLwogICAgcmV0LT5uYW1lID0geG1sU3RyZHVwKG5hbWUpOwogICAgcmV0LT5ldHlwZSA9ICh4bWxFbnRpdHlUeXBlKSB0eXBlOwogICAgaWYgKEV4dGVybmFsSUQgIT0gTlVMTCkKCXJldC0+RXh0ZXJuYWxJRCA9IHhtbFN0cmR1cChFeHRlcm5hbElEKTsKICAgIGlmIChTeXN0ZW1JRCAhPSBOVUxMKQoJcmV0LT5TeXN0ZW1JRCA9IHhtbFN0cmR1cChTeXN0ZW1JRCk7CiAgICBpZiAoY29udGVudCAhPSBOVUxMKSB7CiAgICAgICAgcmV0LT5sZW5ndGggPSB4bWxTdHJsZW4oY29udGVudCk7CglyZXQtPmNvbnRlbnQgPSB4bWxTdHJuZHVwKGNvbnRlbnQsIHJldC0+bGVuZ3RoKTsKICAgICB9IGVsc2UgewogICAgICAgIHJldC0+bGVuZ3RoID0gMDsKICAgICAgICByZXQtPmNvbnRlbnQgPSBOVUxMOwogICAgfQogICAgcmV0LT5VUkkgPSBOVUxMOyAvKiB0byBiZSBjb21wdXRlZCBieSB0aGUgbGF5ZXIga25vd2luZwoJCQl0aGUgZGVmaW5pbmcgZW50aXR5ICovCiAgICByZXQtPm9yaWcgPSBOVUxMOwogICAgcmV0LT5vd25lciA9IDA7CgogICAgaWYgKHhtbEhhc2hBZGRFbnRyeSh0YWJsZSwgbmFtZSwgcmV0KSkgewoJLyoKCSAqIGVudGl0eSB3YXMgYWxyZWFkeSBkZWZpbmVkIGF0IGFub3RoZXIgbGV2ZWwuCgkgKi8KICAgICAgICB4bWxGcmVlRW50aXR5KHJldCk7CglyZXR1cm4oTlVMTCk7CiAgICB9CiAgICByZXR1cm4ocmV0KTsKfQoKLyoqCiAqIHhtbEdldFByZWRlZmluZWRFbnRpdHk6CiAqIEBuYW1lOiAgdGhlIGVudGl0eSBuYW1lCiAqCiAqIENoZWNrIHdoZXRoZXIgdGhpcyBuYW1lIGlzIGFuIHByZWRlZmluZWQgZW50aXR5LgogKgogKiBSZXR1cm5zIE5VTEwgaWYgbm90LCBvdGhlcndpc2UgdGhlIGVudGl0eQogKi8KeG1sRW50aXR5UHRyCnhtbEdldFByZWRlZmluZWRFbnRpdHkoY29uc3QgeG1sQ2hhciAqbmFtZSkgewogICAgaWYgKG5hbWUgPT0gTlVMTCkgcmV0dXJuKE5VTEwpOwogICAgc3dpdGNoIChuYW1lWzBdKSB7CiAgICAgICAgY2FzZSAnbCc6CgkgICAgaWYgKHhtbFN0ckVxdWFsKG5hbWUsIEJBRF9DQVNUICJsdCIpKQoJICAgICAgICByZXR1cm4oJnhtbEVudGl0eUx0KTsKCSAgICBicmVhazsKICAgICAgICBjYXNlICdnJzoKCSAgICBpZiAoeG1sU3RyRXF1YWwobmFtZSwgQkFEX0NBU1QgImd0IikpCgkgICAgICAgIHJldHVybigmeG1sRW50aXR5R3QpOwoJICAgIGJyZWFrOwogICAgICAgIGNhc2UgJ2EnOgoJICAgIGlmICh4bWxTdHJFcXVhbChuYW1lLCBCQURfQ0FTVCAiYW1wIikpCgkgICAgICAgIHJldHVybigmeG1sRW50aXR5QW1wKTsKCSAgICBpZiAoeG1sU3RyRXF1YWwobmFtZSwgQkFEX0NBU1QgImFwb3MiKSkKCSAgICAgICAgcmV0dXJuKCZ4bWxFbnRpdHlBcG9zKTsKCSAgICBicmVhazsKICAgICAgICBjYXNlICdxJzoKCSAgICBpZiAoeG1sU3RyRXF1YWwobmFtZSwgQkFEX0NBU1QgInF1b3QiKSkKCSAgICAgICAgcmV0dXJuKCZ4bWxFbnRpdHlRdW90KTsKCSAgICBicmVhazsKCWRlZmF1bHQ6CgkgICAgYnJlYWs7CiAgICB9CiAgICByZXR1cm4oTlVMTCk7Cn0KCi8qKgogKiB4bWxBZGREdGRFbnRpdHk6CiAqIEBkb2M6ICB0aGUgZG9jdW1lbnQKICogQG5hbWU6ICB0aGUgZW50aXR5IG5hbWUKICogQHR5cGU6ICB0aGUgZW50aXR5IHR5cGUgWE1MX3h4eF95eXlfRU5USVRZCiAqIEBFeHRlcm5hbElEOiAgdGhlIGVudGl0eSBleHRlcm5hbCBJRCBpZiBhdmFpbGFibGUKICogQFN5c3RlbUlEOiAgdGhlIGVudGl0eSBzeXN0ZW0gSUQgaWYgYXZhaWxhYmxlCiAqIEBjb250ZW50OiAgdGhlIGVudGl0eSBjb250ZW50CiAqCiAqIFJlZ2lzdGVyIGEgbmV3IGVudGl0eSBmb3IgdGhpcyBkb2N1bWVudCBEVEQgZXh0ZXJuYWwgc3Vic2V0LgogKgogKiBSZXR1cm5zIGEgcG9pbnRlciB0byB0aGUgZW50aXR5IG9yIE5VTEwgaW4gY2FzZSBvZiBlcnJvcgogKi8KeG1sRW50aXR5UHRyCnhtbEFkZER0ZEVudGl0eSh4bWxEb2NQdHIgZG9jLCBjb25zdCB4bWxDaGFyICpuYW1lLCBpbnQgdHlwZSwKCSAgICAgICAgY29uc3QgeG1sQ2hhciAqRXh0ZXJuYWxJRCwgY29uc3QgeG1sQ2hhciAqU3lzdGVtSUQsCgkJY29uc3QgeG1sQ2hhciAqY29udGVudCkgewogICAgeG1sRW50aXR5UHRyIHJldDsKICAgIHhtbER0ZFB0ciBkdGQ7CgogICAgaWYgKGRvYyA9PSBOVUxMKSB7Cgl4bWxFbnRpdGllc0VycihYTUxfRFREX05PX0RPQywKCSAgICAgICAgInhtbEFkZER0ZEVudGl0eTogZG9jdW1lbnQgaXMgTlVMTCIpOwoJcmV0dXJuKE5VTEwpOwogICAgfQogICAgaWYgKGRvYy0+ZXh0U3Vic2V0ID09IE5VTEwpIHsKCXhtbEVudGl0aWVzRXJyKFhNTF9EVERfTk9fRFRELAoJICAgICAgICAieG1sQWRkRHRkRW50aXR5OiBkb2N1bWVudCB3aXRob3V0IGV4dGVybmFsIHN1YnNldCIpOwoJcmV0dXJuKE5VTEwpOwogICAgfQogICAgZHRkID0gZG9jLT5leHRTdWJzZXQ7CiAgICByZXQgPSB4bWxBZGRFbnRpdHkoZHRkLCBuYW1lLCB0eXBlLCBFeHRlcm5hbElELCBTeXN0ZW1JRCwgY29udGVudCk7CiAgICBpZiAocmV0ID09IE5VTEwpIHJldHVybihOVUxMKTsKCiAgICAvKgogICAgICogTGluayBpdCB0byB0aGUgRFRECiAgICAgKi8KICAgIHJldC0+cGFyZW50ID0gZHRkOwogICAgcmV0LT5kb2MgPSBkdGQtPmRvYzsKICAgIGlmIChkdGQtPmxhc3QgPT0gTlVMTCkgewoJZHRkLT5jaGlsZHJlbiA9IGR0ZC0+bGFzdCA9ICh4bWxOb2RlUHRyKSByZXQ7CiAgICB9IGVsc2UgewogICAgICAgIGR0ZC0+bGFzdC0+bmV4dCA9ICh4bWxOb2RlUHRyKSByZXQ7CglyZXQtPnByZXYgPSBkdGQtPmxhc3Q7CglkdGQtPmxhc3QgPSAoeG1sTm9kZVB0cikgcmV0OwogICAgfQogICAgcmV0dXJuKHJldCk7Cn0KCi8qKgogKiB4bWxBZGREb2NFbnRpdHk6CiAqIEBkb2M6ICB0aGUgZG9jdW1lbnQKICogQG5hbWU6ICB0aGUgZW50aXR5IG5hbWUKICogQHR5cGU6ICB0aGUgZW50aXR5IHR5cGUgWE1MX3h4eF95eXlfRU5USVRZCiAqIEBFeHRlcm5hbElEOiAgdGhlIGVudGl0eSBleHRlcm5hbCBJRCBpZiBhdmFpbGFibGUKICogQFN5c3RlbUlEOiAgdGhlIGVudGl0eSBzeXN0ZW0gSUQgaWYgYXZhaWxhYmxlCiAqIEBjb250ZW50OiAgdGhlIGVudGl0eSBjb250ZW50CiAqCiAqIFJlZ2lzdGVyIGEgbmV3IGVudGl0eSBmb3IgdGhpcyBkb2N1bWVudC4KICoKICogUmV0dXJucyBhIHBvaW50ZXIgdG8gdGhlIGVudGl0eSBvciBOVUxMIGluIGNhc2Ugb2YgZXJyb3IKICovCnhtbEVudGl0eVB0cgp4bWxBZGREb2NFbnRpdHkoeG1sRG9jUHRyIGRvYywgY29uc3QgeG1sQ2hhciAqbmFtZSwgaW50IHR5cGUsCgkgICAgICAgIGNvbnN0IHhtbENoYXIgKkV4dGVybmFsSUQsIGNvbnN0IHhtbENoYXIgKlN5c3RlbUlELAoJICAgICAgICBjb25zdCB4bWxDaGFyICpjb250ZW50KSB7CiAgICB4bWxFbnRpdHlQdHIgcmV0OwogICAgeG1sRHRkUHRyIGR0ZDsKCiAgICBpZiAoZG9jID09IE5VTEwpIHsKCXhtbEVudGl0aWVzRXJyKFhNTF9EVERfTk9fRE9DLAoJICAgICAgICAieG1sQWRkRG9jRW50aXR5OiBkb2N1bWVudCBpcyBOVUxMIik7CglyZXR1cm4oTlVMTCk7CiAgICB9CiAgICBpZiAoZG9jLT5pbnRTdWJzZXQgPT0gTlVMTCkgewoJeG1sRW50aXRpZXNFcnIoWE1MX0RURF9OT19EVEQsCgkgICAgICAgICJ4bWxBZGREb2NFbnRpdHk6IGRvY3VtZW50IHdpdGhvdXQgaW50ZXJuYWwgc3Vic2V0Iik7CglyZXR1cm4oTlVMTCk7CiAgICB9CiAgICBkdGQgPSBkb2MtPmludFN1YnNldDsKICAgIHJldCA9IHhtbEFkZEVudGl0eShkdGQsIG5hbWUsIHR5cGUsIEV4dGVybmFsSUQsIFN5c3RlbUlELCBjb250ZW50KTsKICAgIGlmIChyZXQgPT0gTlVMTCkgcmV0dXJuKE5VTEwpOwoKICAgIC8qCiAgICAgKiBMaW5rIGl0IHRvIHRoZSBEVEQKICAgICAqLwogICAgcmV0LT5wYXJlbnQgPSBkdGQ7CiAgICByZXQtPmRvYyA9IGR0ZC0+ZG9jOwogICAgaWYgKGR0ZC0+bGFzdCA9PSBOVUxMKSB7CglkdGQtPmNoaWxkcmVuID0gZHRkLT5sYXN0ID0gKHhtbE5vZGVQdHIpIHJldDsKICAgIH0gZWxzZSB7CglkdGQtPmxhc3QtPm5leHQgPSAoeG1sTm9kZVB0cikgcmV0OwoJcmV0LT5wcmV2ID0gZHRkLT5sYXN0OwoJZHRkLT5sYXN0ID0gKHhtbE5vZGVQdHIpIHJldDsKICAgIH0KICAgIHJldHVybihyZXQpOwp9CgovKioKICogeG1sR2V0RW50aXR5RnJvbVRhYmxlOgogKiBAdGFibGU6ICBhbiBlbnRpdHkgdGFibGUKICogQG5hbWU6ICB0aGUgZW50aXR5IG5hbWUKICogQHBhcmFtZXRlcjogIGxvb2sgZm9yIHBhcmFtZXRlciBlbnRpdGllcwogKgogKiBEbyBhbiBlbnRpdHkgbG9va3VwIGluIHRoZSB0YWJsZS4KICogcmV0dXJucyB0aGUgY29ycmVzcG9uZGluZyBwYXJhbWV0ZXIgZW50aXR5LCBpZiBmb3VuZC4KICogCiAqIFJldHVybnMgQSBwb2ludGVyIHRvIHRoZSBlbnRpdHkgc3RydWN0dXJlIG9yIE5VTEwgaWYgbm90IGZvdW5kLgogKi8Kc3RhdGljIHhtbEVudGl0eVB0cgp4bWxHZXRFbnRpdHlGcm9tVGFibGUoeG1sRW50aXRpZXNUYWJsZVB0ciB0YWJsZSwgY29uc3QgeG1sQ2hhciAqbmFtZSkgewogICAgcmV0dXJuKCh4bWxFbnRpdHlQdHIpIHhtbEhhc2hMb29rdXAodGFibGUsIG5hbWUpKTsKfQoKLyoqCiAqIHhtbEdldFBhcmFtZXRlckVudGl0eToKICogQGRvYzogIHRoZSBkb2N1bWVudCByZWZlcmVuY2luZyB0aGUgZW50aXR5CiAqIEBuYW1lOiAgdGhlIGVudGl0eSBuYW1lCiAqCiAqIERvIGFuIGVudGl0eSBsb29rdXAgaW4gdGhlIGludGVybmFsIGFuZCBleHRlcm5hbCBzdWJzZXRzIGFuZAogKiByZXR1cm5zIHRoZSBjb3JyZXNwb25kaW5nIHBhcmFtZXRlciBlbnRpdHksIGlmIGZvdW5kLgogKiAKICogUmV0dXJucyBBIHBvaW50ZXIgdG8gdGhlIGVudGl0eSBzdHJ1Y3R1cmUgb3IgTlVMTCBpZiBub3QgZm91bmQuCiAqLwp4bWxFbnRpdHlQdHIKeG1sR2V0UGFyYW1ldGVyRW50aXR5KHhtbERvY1B0ciBkb2MsIGNvbnN0IHhtbENoYXIgKm5hbWUpIHsKICAgIHhtbEVudGl0aWVzVGFibGVQdHIgdGFibGU7CiAgICB4bWxFbnRpdHlQdHIgcmV0OwoKICAgIGlmIChkb2MgPT0gTlVMTCkKCXJldHVybihOVUxMKTsKICAgIGlmICgoZG9jLT5pbnRTdWJzZXQgIT0gTlVMTCkgJiYgKGRvYy0+aW50U3Vic2V0LT5wZW50aXRpZXMgIT0gTlVMTCkpIHsKCXRhYmxlID0gKHhtbEVudGl0aWVzVGFibGVQdHIpIGRvYy0+aW50U3Vic2V0LT5wZW50aXRpZXM7CglyZXQgPSB4bWxHZXRFbnRpdHlGcm9tVGFibGUodGFibGUsIG5hbWUpOwoJaWYgKHJldCAhPSBOVUxMKQoJICAgIHJldHVybihyZXQpOwogICAgfQogICAgaWYgKChkb2MtPmV4dFN1YnNldCAhPSBOVUxMKSAmJiAoZG9jLT5leHRTdWJzZXQtPnBlbnRpdGllcyAhPSBOVUxMKSkgewoJdGFibGUgPSAoeG1sRW50aXRpZXNUYWJsZVB0cikgZG9jLT5leHRTdWJzZXQtPnBlbnRpdGllczsKCXJldHVybih4bWxHZXRFbnRpdHlGcm9tVGFibGUodGFibGUsIG5hbWUpKTsKICAgIH0KICAgIHJldHVybihOVUxMKTsKfQoKLyoqCiAqIHhtbEdldER0ZEVudGl0eToKICogQGRvYzogIHRoZSBkb2N1bWVudCByZWZlcmVuY2luZyB0aGUgZW50aXR5CiAqIEBuYW1lOiAgdGhlIGVudGl0eSBuYW1lCiAqCiAqIERvIGFuIGVudGl0eSBsb29rdXAgaW4gdGhlIERURCBlbnRpdHkgaGFzaCB0YWJsZSBhbmQKICogcmV0dXJucyB0aGUgY29ycmVzcG9uZGluZyBlbnRpdHksIGlmIGZvdW5kLgogKiBOb3RlOiB0aGUgZmlyc3QgYXJndW1lbnQgaXMgdGhlIGRvY3VtZW50IG5vZGUsIG5vdCB0aGUgRFREIG5vZGUuCiAqIAogKiBSZXR1cm5zIEEgcG9pbnRlciB0byB0aGUgZW50aXR5IHN0cnVjdHVyZSBvciBOVUxMIGlmIG5vdCBmb3VuZC4KICovCnhtbEVudGl0eVB0cgp4bWxHZXREdGRFbnRpdHkoeG1sRG9jUHRyIGRvYywgY29uc3QgeG1sQ2hhciAqbmFtZSkgewogICAgeG1sRW50aXRpZXNUYWJsZVB0ciB0YWJsZTsKCiAgICBpZiAoZG9jID09IE5VTEwpCglyZXR1cm4oTlVMTCk7CiAgICBpZiAoKGRvYy0+ZXh0U3Vic2V0ICE9IE5VTEwpICYmIChkb2MtPmV4dFN1YnNldC0+ZW50aXRpZXMgIT0gTlVMTCkpIHsKCXRhYmxlID0gKHhtbEVudGl0aWVzVGFibGVQdHIpIGRvYy0+ZXh0U3Vic2V0LT5lbnRpdGllczsKCXJldHVybih4bWxHZXRFbnRpdHlGcm9tVGFibGUodGFibGUsIG5hbWUpKTsKICAgIH0KICAgIHJldHVybihOVUxMKTsKfQoKLyoqCiAqIHhtbEdldERvY0VudGl0eToKICogQGRvYzogIHRoZSBkb2N1bWVudCByZWZlcmVuY2luZyB0aGUgZW50aXR5CiAqIEBuYW1lOiAgdGhlIGVudGl0eSBuYW1lCiAqCiAqIERvIGFuIGVudGl0eSBsb29rdXAgaW4gdGhlIGRvY3VtZW50IGVudGl0eSBoYXNoIHRhYmxlIGFuZAogKiByZXR1cm5zIHRoZSBjb3JyZXNwb25kaW5nIGVudGl0eSwgb3RoZXJ3aXNlIGEgbG9va3VwIGlzIGRvbmUKICogaW4gdGhlIHByZWRlZmluZWQgZW50aXRpZXMgdG9vLgogKiAKICogUmV0dXJucyBBIHBvaW50ZXIgdG8gdGhlIGVudGl0eSBzdHJ1Y3R1cmUgb3IgTlVMTCBpZiBub3QgZm91bmQuCiAqLwp4bWxFbnRpdHlQdHIKeG1sR2V0RG9jRW50aXR5KHhtbERvY1B0ciBkb2MsIGNvbnN0IHhtbENoYXIgKm5hbWUpIHsKICAgIHhtbEVudGl0eVB0ciBjdXI7CiAgICB4bWxFbnRpdGllc1RhYmxlUHRyIHRhYmxlOwoKICAgIGlmIChkb2MgIT0gTlVMTCkgewoJaWYgKChkb2MtPmludFN1YnNldCAhPSBOVUxMKSAmJiAoZG9jLT5pbnRTdWJzZXQtPmVudGl0aWVzICE9IE5VTEwpKSB7CgkgICAgdGFibGUgPSAoeG1sRW50aXRpZXNUYWJsZVB0cikgZG9jLT5pbnRTdWJzZXQtPmVudGl0aWVzOwoJICAgIGN1ciA9IHhtbEdldEVudGl0eUZyb21UYWJsZSh0YWJsZSwgbmFtZSk7CgkgICAgaWYgKGN1ciAhPSBOVUxMKQoJCXJldHVybihjdXIpOwoJfQoJaWYgKGRvYy0+c3RhbmRhbG9uZSAhPSAxKSB7CgkgICAgaWYgKChkb2MtPmV4dFN1YnNldCAhPSBOVUxMKSAmJgoJCShkb2MtPmV4dFN1YnNldC0+ZW50aXRpZXMgIT0gTlVMTCkpIHsKCQl0YWJsZSA9ICh4bWxFbnRpdGllc1RhYmxlUHRyKSBkb2MtPmV4dFN1YnNldC0+ZW50aXRpZXM7CgkJY3VyID0geG1sR2V0RW50aXR5RnJvbVRhYmxlKHRhYmxlLCBuYW1lKTsKCQlpZiAoY3VyICE9IE5VTEwpCgkJICAgIHJldHVybihjdXIpOwoJICAgIH0KCX0KICAgIH0KICAgIHJldHVybih4bWxHZXRQcmVkZWZpbmVkRW50aXR5KG5hbWUpKTsKfQoKLyoKICogTWFjcm8gdXNlZCB0byBncm93IHRoZSBjdXJyZW50IGJ1ZmZlci4KICovCiNkZWZpbmUgZ3Jvd0J1ZmZlclJlZW50cmFudCgpIHsJCQkJCQlcCiAgICBidWZmZXJfc2l6ZSAqPSAyOwkJCQkJCQlcCiAgICBidWZmZXIgPSAoeG1sQ2hhciAqKQkJCQkJCVwKICAgIAkJeG1sUmVhbGxvYyhidWZmZXIsIGJ1ZmZlcl9zaXplICogc2l6ZW9mKHhtbENoYXIpKTsJXAogICAgaWYgKGJ1ZmZlciA9PSBOVUxMKSB7CQkJCQkJXAogICAgICAgIHhtbEVudGl0aWVzRXJyTWVtb3J5KCJ4bWxFbmNvZGVFbnRpdGllc1JlZW50cmFudDogcmVhbGxvYyBmYWlsZWQiKTtcCglyZXR1cm4oTlVMTCk7CQkJCQkJCVwKICAgIH0JCQkJCQkJCQlcCn0KCgovKioKICogeG1sRW5jb2RlRW50aXRpZXNSZWVudHJhbnQ6CiAqIEBkb2M6ICB0aGUgZG9jdW1lbnQgY29udGFpbmluZyB0aGUgc3RyaW5nCiAqIEBpbnB1dDogIEEgc3RyaW5nIHRvIGNvbnZlcnQgdG8gWE1MLgogKgogKiBEbyBhIGdsb2JhbCBlbmNvZGluZyBvZiBhIHN0cmluZywgcmVwbGFjaW5nIHRoZSBwcmVkZWZpbmVkIGVudGl0aWVzCiAqIGFuZCBub24gQVNDSUkgdmFsdWVzIHdpdGggdGhlaXIgZW50aXRpZXMgYW5kIENoYXJSZWYgY291bnRlcnBhcnRzLgogKiBDb250cmFyeSB0byB4bWxFbmNvZGVFbnRpdGllcywgdGhpcyByb3V0aW5lIGlzIHJlZW50cmFudCwgYW5kIHJlc3VsdAogKiBtdXN0IGJlIGRlYWxsb2NhdGVkLgogKgogKiBSZXR1cm5zIEEgbmV3bHkgYWxsb2NhdGVkIHN0cmluZyB3aXRoIHRoZSBzdWJzdGl0dXRpb24gZG9uZS4KICovCnhtbENoYXIgKgp4bWxFbmNvZGVFbnRpdGllc1JlZW50cmFudCh4bWxEb2NQdHIgZG9jLCBjb25zdCB4bWxDaGFyICppbnB1dCkgewogICAgY29uc3QgeG1sQ2hhciAqY3VyID0gaW5wdXQ7CiAgICB4bWxDaGFyICpidWZmZXIgPSBOVUxMOwogICAgeG1sQ2hhciAqb3V0ID0gTlVMTDsKICAgIGludCBidWZmZXJfc2l6ZSA9IDA7CiAgICBpbnQgaHRtbCA9IDA7CgogICAgaWYgKGlucHV0ID09IE5VTEwpIHJldHVybihOVUxMKTsKICAgIGlmIChkb2MgIT0gTlVMTCkKICAgICAgICBodG1sID0gKGRvYy0+dHlwZSA9PSBYTUxfSFRNTF9ET0NVTUVOVF9OT0RFKTsKCiAgICAvKgogICAgICogYWxsb2NhdGUgYW4gdHJhbnNsYXRpb24gYnVmZmVyLgogICAgICovCiAgICBidWZmZXJfc2l6ZSA9IDEwMDA7CiAgICBidWZmZXIgPSAoeG1sQ2hhciAqKSB4bWxNYWxsb2MoYnVmZmVyX3NpemUgKiBzaXplb2YoeG1sQ2hhcikpOwogICAgaWYgKGJ1ZmZlciA9PSBOVUxMKSB7CiAgICAgICAgeG1sRW50aXRpZXNFcnJNZW1vcnkoInhtbEVuY29kZUVudGl0aWVzUmVlbnRyYW50OiBtYWxsb2MgZmFpbGVkIik7CglyZXR1cm4oTlVMTCk7CiAgICB9CiAgICBvdXQgPSBidWZmZXI7CgogICAgd2hpbGUgKCpjdXIgIT0gJ1wwJykgewogICAgICAgIGlmIChvdXQgLSBidWZmZXIgPiBidWZmZXJfc2l6ZSAtIDEwMCkgewoJICAgIGludCBpbmR4ID0gb3V0IC0gYnVmZmVyOwoKCSAgICBncm93QnVmZmVyUmVlbnRyYW50KCk7CgkgICAgb3V0ID0gJmJ1ZmZlcltpbmR4XTsKCX0KCgkvKgoJICogQnkgZGVmYXVsdCBvbmUgaGF2ZSB0byBlbmNvZGUgYXQgbGVhc3QgJzwnLCAnPicsICciJyBhbmQgJyYnICEKCSAqLwoJaWYgKCpjdXIgPT0gJzwnKSB7CgkgICAgKm91dCsrID0gJyYnOwoJICAgICpvdXQrKyA9ICdsJzsKCSAgICAqb3V0KysgPSAndCc7CgkgICAgKm91dCsrID0gJzsnOwoJfSBlbHNlIGlmICgqY3VyID09ICc+JykgewoJICAgICpvdXQrKyA9ICcmJzsKCSAgICAqb3V0KysgPSAnZyc7CgkgICAgKm91dCsrID0gJ3QnOwoJICAgICpvdXQrKyA9ICc7JzsKCX0gZWxzZSBpZiAoKmN1ciA9PSAnJicpIHsKCSAgICAqb3V0KysgPSAnJic7CgkgICAgKm91dCsrID0gJ2EnOwoJICAgICpvdXQrKyA9ICdtJzsKCSAgICAqb3V0KysgPSAncCc7CgkgICAgKm91dCsrID0gJzsnOwoJfSBlbHNlIGlmICgoKCpjdXIgPj0gMHgyMCkgJiYgKCpjdXIgPCAweDgwKSkgfHwKCSAgICAoKmN1ciA9PSAnXG4nKSB8fCAoKmN1ciA9PSAnXHQnKSB8fCAoKGh0bWwpICYmICgqY3VyID09ICdccicpKSkgewoJICAgIC8qCgkgICAgICogZGVmYXVsdCBjYXNlLCBqdXN0IGNvcHkgIQoJICAgICAqLwoJICAgICpvdXQrKyA9ICpjdXI7Cgl9IGVsc2UgaWYgKCpjdXIgPj0gMHg4MCkgewoJICAgIGlmICgoKGRvYyAhPSBOVUxMKSAmJiAoZG9jLT5lbmNvZGluZyAhPSBOVUxMKSkgfHwgKGh0bWwpKSB7CgkJLyoKCQkgKiBCavhybiBSZWVzZSA8YnJAc3NldXNhLmNvbT4gcHJvdmlkZWQgdGhlIHBhdGNoCgkgICAgICAgIHhtbENoYXIgeGM7CgkgICAgICAgIHhjID0gKCpjdXIgJiAweDNGKSA8PCA2OwoJICAgICAgICBpZiAoY3VyWzFdICE9IDApIHsKCQkgICAgeGMgKz0gKigrK2N1cikgJiAweDNGOwoJCSAgICAqb3V0KysgPSB4YzsKCSAgICAgICAgfSBlbHNlCgkJICovCgkJICAgICpvdXQrKyA9ICpjdXI7CgkgICAgfSBlbHNlIHsKCQkvKgoJCSAqIFdlIGFzc3VtZSB3ZSBoYXZlIFVURi04IGlucHV0LgoJCSAqLwoJCWNoYXIgYnVmWzExXSwgKnB0cjsKCQlpbnQgdmFsID0gMCwgbCA9IDE7CgoJCWlmICgqY3VyIDwgMHhDMCkgewoJCSAgICB4bWxFbnRpdGllc0VycihYTUxfQ0hFQ0tfTk9UX1VURjgsCgkJCSAgICAieG1sRW5jb2RlRW50aXRpZXNSZWVudHJhbnQgOiBpbnB1dCBub3QgVVRGLTgiKTsKCQkgICAgaWYgKGRvYyAhPSBOVUxMKQoJCQlkb2MtPmVuY29kaW5nID0geG1sU3RyZHVwKEJBRF9DQVNUICJJU08tODg1OS0xIik7CgkJICAgIHNucHJpbnRmKGJ1Ziwgc2l6ZW9mKGJ1ZiksICImIyVkOyIsICpjdXIpOwoJCSAgICBidWZbc2l6ZW9mKGJ1ZikgLSAxXSA9IDA7CgkJICAgIHB0ciA9IGJ1ZjsKCQkgICAgd2hpbGUgKCpwdHIgIT0gMCkgKm91dCsrID0gKnB0cisrOwoJCSAgICBjdXIrKzsKCQkgICAgY29udGludWU7CgkJfSBlbHNlIGlmICgqY3VyIDwgMHhFMCkgewogICAgICAgICAgICAgICAgICAgIHZhbCA9IChjdXJbMF0pICYgMHgxRjsKCQkgICAgdmFsIDw8PSA2OwoJCSAgICB2YWwgfD0gKGN1clsxXSkgJiAweDNGOwoJCSAgICBsID0gMjsKCQl9IGVsc2UgaWYgKCpjdXIgPCAweEYwKSB7CiAgICAgICAgICAgICAgICAgICAgdmFsID0gKGN1clswXSkgJiAweDBGOwoJCSAgICB2YWwgPDw9IDY7CgkJICAgIHZhbCB8PSAoY3VyWzFdKSAmIDB4M0Y7CgkJICAgIHZhbCA8PD0gNjsKCQkgICAgdmFsIHw9IChjdXJbMl0pICYgMHgzRjsKCQkgICAgbCA9IDM7CgkJfSBlbHNlIGlmICgqY3VyIDwgMHhGOCkgewogICAgICAgICAgICAgICAgICAgIHZhbCA9IChjdXJbMF0pICYgMHgwNzsKCQkgICAgdmFsIDw8PSA2OwoJCSAgICB2YWwgfD0gKGN1clsxXSkgJiAweDNGOwoJCSAgICB2YWwgPDw9IDY7CgkJICAgIHZhbCB8PSAoY3VyWzJdKSAmIDB4M0Y7CgkJICAgIHZhbCA8PD0gNjsKCQkgICAgdmFsIHw9IChjdXJbM10pICYgMHgzRjsKCQkgICAgbCA9IDQ7CgkJfQoJCWlmICgobCA9PSAxKSB8fCAoIUlTX0NIQVIodmFsKSkpIHsKCQkgICAgeG1sRW50aXRpZXNFcnIoWE1MX0VSUl9JTlZBTElEX0NIQVIsCgkJCSJ4bWxFbmNvZGVFbnRpdGllc1JlZW50cmFudCA6IGNoYXIgb3V0IG9mIHJhbmdlXG4iKTsKCQkgICAgaWYgKGRvYyAhPSBOVUxMKQoJCQlkb2MtPmVuY29kaW5nID0geG1sU3RyZHVwKEJBRF9DQVNUICJJU08tODg1OS0xIik7CgkJICAgIHNucHJpbnRmKGJ1Ziwgc2l6ZW9mKGJ1ZiksICImIyVkOyIsICpjdXIpOwoJCSAgICBidWZbc2l6ZW9mKGJ1ZikgLSAxXSA9IDA7CgkJICAgIHB0ciA9IGJ1ZjsKCQkgICAgd2hpbGUgKCpwdHIgIT0gMCkgKm91dCsrID0gKnB0cisrOwoJCSAgICBjdXIrKzsKCQkgICAgY29udGludWU7CgkJfQoJCS8qCgkJICogV2UgY291bGQgZG8gbXVsdGlwbGUgdGhpbmdzIGhlcmUuIEp1c3Qgc2F2ZSBhcyBhIGNoYXIgcmVmCgkJICovCgkJaWYgKGh0bWwpCgkJICAgIHNucHJpbnRmKGJ1Ziwgc2l6ZW9mKGJ1ZiksICImIyVkOyIsIHZhbCk7CgkJZWxzZQoJCSAgICBzbnByaW50ZihidWYsIHNpemVvZihidWYpLCAiJiN4JVg7IiwgdmFsKTsKCQlidWZbc2l6ZW9mKGJ1ZikgLSAxXSA9IDA7CgkJcHRyID0gYnVmOwoJCXdoaWxlICgqcHRyICE9IDApICpvdXQrKyA9ICpwdHIrKzsKCQljdXIgKz0gbDsKCQljb250aW51ZTsKCSAgICB9Cgl9IGVsc2UgaWYgKElTX0JZVEVfQ0hBUigqY3VyKSkgewoJICAgIGNoYXIgYnVmWzExXSwgKnB0cjsKCgkgICAgc25wcmludGYoYnVmLCBzaXplb2YoYnVmKSwgIiYjJWQ7IiwgKmN1cik7CgkgICAgYnVmW3NpemVvZihidWYpIC0gMV0gPSAwOwogICAgICAgICAgICBwdHIgPSBidWY7CgkgICAgd2hpbGUgKCpwdHIgIT0gMCkgKm91dCsrID0gKnB0cisrOwoJfQoJY3VyKys7CiAgICB9CiAgICAqb3V0KysgPSAwOwogICAgcmV0dXJuKGJ1ZmZlcik7Cn0KCi8qKgogKiB4bWxFbmNvZGVTcGVjaWFsQ2hhcnM6CiAqIEBkb2M6ICB0aGUgZG9jdW1lbnQgY29udGFpbmluZyB0aGUgc3RyaW5nCiAqIEBpbnB1dDogIEEgc3RyaW5nIHRvIGNvbnZlcnQgdG8gWE1MLgogKgogKiBEbyBhIGdsb2JhbCBlbmNvZGluZyBvZiBhIHN0cmluZywgcmVwbGFjaW5nIHRoZSBwcmVkZWZpbmVkIGVudGl0aWVzCiAqIHRoaXMgcm91dGluZSBpcyByZWVudHJhbnQsIGFuZCByZXN1bHQgbXVzdCBiZSBkZWFsbG9jYXRlZC4KICoKICogUmV0dXJucyBBIG5ld2x5IGFsbG9jYXRlZCBzdHJpbmcgd2l0aCB0aGUgc3Vic3RpdHV0aW9uIGRvbmUuCiAqLwp4bWxDaGFyICoKeG1sRW5jb2RlU3BlY2lhbENoYXJzKHhtbERvY1B0ciBkb2MgQVRUUklCVVRFX1VOVVNFRCwgY29uc3QgeG1sQ2hhciAqaW5wdXQpIHsKICAgIGNvbnN0IHhtbENoYXIgKmN1ciA9IGlucHV0OwogICAgeG1sQ2hhciAqYnVmZmVyID0gTlVMTDsKICAgIHhtbENoYXIgKm91dCA9IE5VTEw7CiAgICBpbnQgYnVmZmVyX3NpemUgPSAwOwogICAgaWYgKGlucHV0ID09IE5VTEwpIHJldHVybihOVUxMKTsKCiAgICAvKgogICAgICogYWxsb2NhdGUgYW4gdHJhbnNsYXRpb24gYnVmZmVyLgogICAgICovCiAgICBidWZmZXJfc2l6ZSA9IDEwMDA7CiAgICBidWZmZXIgPSAoeG1sQ2hhciAqKSB4bWxNYWxsb2MoYnVmZmVyX3NpemUgKiBzaXplb2YoeG1sQ2hhcikpOwogICAgaWYgKGJ1ZmZlciA9PSBOVUxMKSB7CiAgICAgICAgeG1sRW50aXRpZXNFcnJNZW1vcnkoInhtbEVuY29kZVNwZWNpYWxDaGFyczogbWFsbG9jIGZhaWxlZCIpOwoJcmV0dXJuKE5VTEwpOwogICAgfQogICAgb3V0ID0gYnVmZmVyOwoKICAgIHdoaWxlICgqY3VyICE9ICdcMCcpIHsKICAgICAgICBpZiAob3V0IC0gYnVmZmVyID4gYnVmZmVyX3NpemUgLSAxMCkgewoJICAgIGludCBpbmR4ID0gb3V0IC0gYnVmZmVyOwoKCSAgICBncm93QnVmZmVyUmVlbnRyYW50KCk7CgkgICAgb3V0ID0gJmJ1ZmZlcltpbmR4XTsKCX0KCgkvKgoJICogQnkgZGVmYXVsdCBvbmUgaGF2ZSB0byBlbmNvZGUgYXQgbGVhc3QgJzwnLCAnPicsICciJyBhbmQgJyYnICEKCSAqLwoJaWYgKCpjdXIgPT0gJzwnKSB7CgkgICAgKm91dCsrID0gJyYnOwoJICAgICpvdXQrKyA9ICdsJzsKCSAgICAqb3V0KysgPSAndCc7CgkgICAgKm91dCsrID0gJzsnOwoJfSBlbHNlIGlmICgqY3VyID09ICc+JykgewoJICAgICpvdXQrKyA9ICcmJzsKCSAgICAqb3V0KysgPSAnZyc7CgkgICAgKm91dCsrID0gJ3QnOwoJICAgICpvdXQrKyA9ICc7JzsKCX0gZWxzZSBpZiAoKmN1ciA9PSAnJicpIHsKCSAgICAqb3V0KysgPSAnJic7CgkgICAgKm91dCsrID0gJ2EnOwoJICAgICpvdXQrKyA9ICdtJzsKCSAgICAqb3V0KysgPSAncCc7CgkgICAgKm91dCsrID0gJzsnOwoJfSBlbHNlIGlmICgqY3VyID09ICciJykgewoJICAgICpvdXQrKyA9ICcmJzsKCSAgICAqb3V0KysgPSAncSc7CgkgICAgKm91dCsrID0gJ3UnOwoJICAgICpvdXQrKyA9ICdvJzsKCSAgICAqb3V0KysgPSAndCc7CgkgICAgKm91dCsrID0gJzsnOwoJfSBlbHNlIGlmICgqY3VyID09ICdccicpIHsKCSAgICAqb3V0KysgPSAnJic7CgkgICAgKm91dCsrID0gJyMnOwoJICAgICpvdXQrKyA9ICcxJzsKCSAgICAqb3V0KysgPSAnMyc7CgkgICAgKm91dCsrID0gJzsnOwoJfSBlbHNlIHsKCSAgICAvKgoJICAgICAqIFdvcmtzIGJlY2F1c2Ugb24gVVRGLTgsIGFsbCBleHRlbmRlZCBzZXF1ZW5jZXMgY2Fubm90CgkgICAgICogcmVzdWx0IGluIGJ5dGVzIGluIHRoZSBBU0NJSSByYW5nZS4KCSAgICAgKi8KCSAgICAqb3V0KysgPSAqY3VyOwoJfQoJY3VyKys7CiAgICB9CiAgICAqb3V0KysgPSAwOwogICAgcmV0dXJuKGJ1ZmZlcik7Cn0KCi8qKgogKiB4bWxDcmVhdGVFbnRpdGllc1RhYmxlOgogKgogKiBjcmVhdGUgYW5kIGluaXRpYWxpemUgYW4gZW1wdHkgZW50aXRpZXMgaGFzaCB0YWJsZS4KICoKICogUmV0dXJucyB0aGUgeG1sRW50aXRpZXNUYWJsZVB0ciBqdXN0IGNyZWF0ZWQgb3IgTlVMTCBpbiBjYXNlIG9mIGVycm9yLgogKi8KeG1sRW50aXRpZXNUYWJsZVB0cgp4bWxDcmVhdGVFbnRpdGllc1RhYmxlKHZvaWQpIHsKICAgIHJldHVybigoeG1sRW50aXRpZXNUYWJsZVB0cikgeG1sSGFzaENyZWF0ZSgwKSk7Cn0KCi8qKgogKiB4bWxGcmVlRW50aXR5V3JhcHBlcjoKICogQGVudGl0eTogIEFuIGVudGl0eQogKiBAbmFtZTogIGl0cyBuYW1lCiAqCiAqIERlYWxsb2NhdGUgdGhlIG1lbW9yeSB1c2VkIGJ5IGFuIGVudGl0aWVzIGluIHRoZSBoYXNoIHRhYmxlLgogKi8Kc3RhdGljIHZvaWQKeG1sRnJlZUVudGl0eVdyYXBwZXIoeG1sRW50aXR5UHRyIGVudGl0eSwKCSAgICAgICAgICAgICAgIGNvbnN0IHhtbENoYXIgKm5hbWUgQVRUUklCVVRFX1VOVVNFRCkgewogICAgaWYgKGVudGl0eSAhPSBOVUxMKQoJeG1sRnJlZUVudGl0eShlbnRpdHkpOwp9CgovKioKICogeG1sRnJlZUVudGl0aWVzVGFibGU6CiAqIEB0YWJsZTogIEFuIGVudGl0eSB0YWJsZQogKgogKiBEZWFsbG9jYXRlIHRoZSBtZW1vcnkgdXNlZCBieSBhbiBlbnRpdGllcyBoYXNoIHRhYmxlLgogKi8Kdm9pZAp4bWxGcmVlRW50aXRpZXNUYWJsZSh4bWxFbnRpdGllc1RhYmxlUHRyIHRhYmxlKSB7CiAgICB4bWxIYXNoRnJlZSh0YWJsZSwgKHhtbEhhc2hEZWFsbG9jYXRvcikgeG1sRnJlZUVudGl0eVdyYXBwZXIpOwp9CgojaWZkZWYgTElCWE1MX1RSRUVfRU5BQkxFRAovKioKICogeG1sQ29weUVudGl0eToKICogQGVudDogIEFuIGVudGl0eQogKgogKiBCdWlsZCBhIGNvcHkgb2YgYW4gZW50aXR5CiAqIAogKiBSZXR1cm5zIHRoZSBuZXcgeG1sRW50aXRpZXNQdHIgb3IgTlVMTCBpbiBjYXNlIG9mIGVycm9yLgogKi8Kc3RhdGljIHhtbEVudGl0eVB0cgp4bWxDb3B5RW50aXR5KHhtbEVudGl0eVB0ciBlbnQpIHsKICAgIHhtbEVudGl0eVB0ciBjdXI7CgogICAgY3VyID0gKHhtbEVudGl0eVB0cikgeG1sTWFsbG9jKHNpemVvZih4bWxFbnRpdHkpKTsKICAgIGlmIChjdXIgPT0gTlVMTCkgewogICAgICAgIHhtbEVudGl0aWVzRXJyTWVtb3J5KCJ4bWxDb3B5RW50aXR5OjogbWFsbG9jIGZhaWxlZCIpOwoJcmV0dXJuKE5VTEwpOwogICAgfQogICAgbWVtc2V0KGN1ciwgMCwgc2l6ZW9mKHhtbEVudGl0eSkpOwogICAgY3VyLT50eXBlID0gWE1MX0VOVElUWV9ERUNMOwoKICAgIGN1ci0+ZXR5cGUgPSBlbnQtPmV0eXBlOwogICAgaWYgKGVudC0+bmFtZSAhPSBOVUxMKQoJY3VyLT5uYW1lID0geG1sU3RyZHVwKGVudC0+bmFtZSk7CiAgICBpZiAoZW50LT5FeHRlcm5hbElEICE9IE5VTEwpCgljdXItPkV4dGVybmFsSUQgPSB4bWxTdHJkdXAoZW50LT5FeHRlcm5hbElEKTsKICAgIGlmIChlbnQtPlN5c3RlbUlEICE9IE5VTEwpCgljdXItPlN5c3RlbUlEID0geG1sU3RyZHVwKGVudC0+U3lzdGVtSUQpOwogICAgaWYgKGVudC0+Y29udGVudCAhPSBOVUxMKQoJY3VyLT5jb250ZW50ID0geG1sU3RyZHVwKGVudC0+Y29udGVudCk7CiAgICBpZiAoZW50LT5vcmlnICE9IE5VTEwpCgljdXItPm9yaWcgPSB4bWxTdHJkdXAoZW50LT5vcmlnKTsKICAgIGlmIChlbnQtPlVSSSAhPSBOVUxMKQoJY3VyLT5VUkkgPSB4bWxTdHJkdXAoZW50LT5VUkkpOwogICAgcmV0dXJuKGN1cik7Cn0KCi8qKgogKiB4bWxDb3B5RW50aXRpZXNUYWJsZToKICogQHRhYmxlOiAgQW4gZW50aXR5IHRhYmxlCiAqCiAqIEJ1aWxkIGEgY29weSBvZiBhbiBlbnRpdHkgdGFibGUuCiAqIAogKiBSZXR1cm5zIHRoZSBuZXcgeG1sRW50aXRpZXNUYWJsZVB0ciBvciBOVUxMIGluIGNhc2Ugb2YgZXJyb3IuCiAqLwp4bWxFbnRpdGllc1RhYmxlUHRyCnhtbENvcHlFbnRpdGllc1RhYmxlKHhtbEVudGl0aWVzVGFibGVQdHIgdGFibGUpIHsKICAgIHJldHVybih4bWxIYXNoQ29weSh0YWJsZSwgKHhtbEhhc2hDb3BpZXIpIHhtbENvcHlFbnRpdHkpKTsKfQojZW5kaWYgLyogTElCWE1MX1RSRUVfRU5BQkxFRCAqLwoKI2lmZGVmIExJQlhNTF9PVVRQVVRfRU5BQkxFRAoKLyoqCiAqIHhtbER1bXBFbnRpdHlDb250ZW50OgogKiBAYnVmOiAgQW4gWE1MIGJ1ZmZlci4KICogQGNvbnRlbnQ6ICBUaGUgZW50aXR5IGNvbnRlbnQuCiAqCiAqIFRoaXMgd2lsbCBkdW1wIHRoZSBxdW90ZWQgc3RyaW5nIHZhbHVlLCB0YWtpbmcgY2FyZSBvZiB0aGUgc3BlY2lhbAogKiB0cmVhdG1lbnQgcmVxdWlyZWQgYnkgJQogKi8Kc3RhdGljIHZvaWQKeG1sRHVtcEVudGl0eUNvbnRlbnQoeG1sQnVmZmVyUHRyIGJ1ZiwgY29uc3QgeG1sQ2hhciAqY29udGVudCkgewogICAgaWYgKGJ1Zi0+YWxsb2MgPT0gWE1MX0JVRkZFUl9BTExPQ19JTU1VVEFCTEUpIHJldHVybjsKICAgIGlmICh4bWxTdHJjaHIoY29udGVudCwgJyUnKSkgewogICAgICAgIGNvbnN0IHhtbENoYXIgKiBiYXNlLCAqY3VyOwoKCXhtbEJ1ZmZlckNDYXQoYnVmLCAiXCIiKTsKCWJhc2UgPSBjdXIgPSBjb250ZW50OwoJd2hpbGUgKCpjdXIgIT0gMCkgewoJICAgIGlmICgqY3VyID09ICciJykgewoJCWlmIChiYXNlICE9IGN1cikKCQkgICAgeG1sQnVmZmVyQWRkKGJ1ZiwgYmFzZSwgY3VyIC0gYmFzZSk7CgkJeG1sQnVmZmVyQWRkKGJ1ZiwgQkFEX0NBU1QgIiZxdW90OyIsIDYpOwoJCWN1cisrOwoJCWJhc2UgPSBjdXI7CgkgICAgfSBlbHNlIGlmICgqY3VyID09ICclJykgewoJCWlmIChiYXNlICE9IGN1cikKCQkgICAgeG1sQnVmZmVyQWRkKGJ1ZiwgYmFzZSwgY3VyIC0gYmFzZSk7CgkJeG1sQnVmZmVyQWRkKGJ1ZiwgQkFEX0NBU1QgIiYjeDI1OyIsIDYpOwoJCWN1cisrOwoJCWJhc2UgPSBjdXI7CgkgICAgfSBlbHNlIHsKCQljdXIrKzsKCSAgICB9Cgl9CglpZiAoYmFzZSAhPSBjdXIpCgkgICAgeG1sQnVmZmVyQWRkKGJ1ZiwgYmFzZSwgY3VyIC0gYmFzZSk7Cgl4bWxCdWZmZXJDQ2F0KGJ1ZiwgIlwiIik7CiAgICB9IGVsc2UgewogICAgICAgIHhtbEJ1ZmZlcldyaXRlUXVvdGVkU3RyaW5nKGJ1ZiwgY29udGVudCk7CiAgICB9Cn0KCi8qKgogKiB4bWxEdW1wRW50aXR5RGVjbDoKICogQGJ1ZjogIEFuIFhNTCBidWZmZXIuCiAqIEBlbnQ6ICBBbiBlbnRpdHkgdGFibGUKICoKICogVGhpcyB3aWxsIGR1bXAgdGhlIGNvbnRlbnQgb2YgdGhlIGVudGl0eSB0YWJsZSBhcyBhbiBYTUwgRFREIGRlZmluaXRpb24KICovCnZvaWQKeG1sRHVtcEVudGl0eURlY2woeG1sQnVmZmVyUHRyIGJ1ZiwgeG1sRW50aXR5UHRyIGVudCkgewogICAgaWYgKChidWYgPT0gTlVMTCkgfHwgKGVudCA9PSBOVUxMKSkgcmV0dXJuOwogICAgc3dpdGNoIChlbnQtPmV0eXBlKSB7CgljYXNlIFhNTF9JTlRFUk5BTF9HRU5FUkFMX0VOVElUWToKCSAgICB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiPCFFTlRJVFkgIik7CgkgICAgeG1sQnVmZmVyV3JpdGVDSEFSKGJ1ZiwgZW50LT5uYW1lKTsKCSAgICB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiICIpOwoJICAgIGlmIChlbnQtPm9yaWcgIT0gTlVMTCkKCQl4bWxCdWZmZXJXcml0ZVF1b3RlZFN0cmluZyhidWYsIGVudC0+b3JpZyk7CgkgICAgZWxzZQoJCXhtbER1bXBFbnRpdHlDb250ZW50KGJ1ZiwgZW50LT5jb250ZW50KTsKCSAgICB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiPlxuIik7CgkgICAgYnJlYWs7CgljYXNlIFhNTF9FWFRFUk5BTF9HRU5FUkFMX1BBUlNFRF9FTlRJVFk6CgkgICAgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIjwhRU5USVRZICIpOwoJICAgIHhtbEJ1ZmZlcldyaXRlQ0hBUihidWYsIGVudC0+bmFtZSk7CgkgICAgaWYgKGVudC0+RXh0ZXJuYWxJRCAhPSBOVUxMKSB7CgkJIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICIgUFVCTElDICIpOwoJCSB4bWxCdWZmZXJXcml0ZVF1b3RlZFN0cmluZyhidWYsIGVudC0+RXh0ZXJuYWxJRCk7CgkJIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICIgIik7CgkJIHhtbEJ1ZmZlcldyaXRlUXVvdGVkU3RyaW5nKGJ1ZiwgZW50LT5TeXN0ZW1JRCk7CgkgICAgfSBlbHNlIHsKCQkgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIiBTWVNURU0gIik7CgkJIHhtbEJ1ZmZlcldyaXRlUXVvdGVkU3RyaW5nKGJ1ZiwgZW50LT5TeXN0ZW1JRCk7CgkgICAgfQoJICAgIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICI+XG4iKTsKCSAgICBicmVhazsKCWNhc2UgWE1MX0VYVEVSTkFMX0dFTkVSQUxfVU5QQVJTRURfRU5USVRZOgoJICAgIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICI8IUVOVElUWSAiKTsKCSAgICB4bWxCdWZmZXJXcml0ZUNIQVIoYnVmLCBlbnQtPm5hbWUpOwoJICAgIGlmIChlbnQtPkV4dGVybmFsSUQgIT0gTlVMTCkgewoJCSB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiIFBVQkxJQyAiKTsKCQkgeG1sQnVmZmVyV3JpdGVRdW90ZWRTdHJpbmcoYnVmLCBlbnQtPkV4dGVybmFsSUQpOwoJCSB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiICIpOwoJCSB4bWxCdWZmZXJXcml0ZVF1b3RlZFN0cmluZyhidWYsIGVudC0+U3lzdGVtSUQpOwoJICAgIH0gZWxzZSB7CgkJIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICIgU1lTVEVNICIpOwoJCSB4bWxCdWZmZXJXcml0ZVF1b3RlZFN0cmluZyhidWYsIGVudC0+U3lzdGVtSUQpOwoJICAgIH0KCSAgICBpZiAoZW50LT5jb250ZW50ICE9IE5VTEwpIHsgLyogU2hvdWxkIGJlIHRydWUgISAqLwoJCXhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICIgTkRBVEEgIik7CgkJaWYgKGVudC0+b3JpZyAhPSBOVUxMKQoJCSAgICB4bWxCdWZmZXJXcml0ZUNIQVIoYnVmLCBlbnQtPm9yaWcpOwoJCWVsc2UKCQkgICAgeG1sQnVmZmVyV3JpdGVDSEFSKGJ1ZiwgZW50LT5jb250ZW50KTsKCSAgICB9CgkgICAgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIj5cbiIpOwoJICAgIGJyZWFrOwoJY2FzZSBYTUxfSU5URVJOQUxfUEFSQU1FVEVSX0VOVElUWToKCSAgICB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiPCFFTlRJVFkgJSAiKTsKCSAgICB4bWxCdWZmZXJXcml0ZUNIQVIoYnVmLCBlbnQtPm5hbWUpOwoJICAgIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICIgIik7CgkgICAgaWYgKGVudC0+b3JpZyA9PSBOVUxMKQoJCXhtbER1bXBFbnRpdHlDb250ZW50KGJ1ZiwgZW50LT5jb250ZW50KTsKCSAgICBlbHNlCgkJeG1sQnVmZmVyV3JpdGVRdW90ZWRTdHJpbmcoYnVmLCBlbnQtPm9yaWcpOwoJICAgIHhtbEJ1ZmZlcldyaXRlQ2hhcihidWYsICI+XG4iKTsKCSAgICBicmVhazsKCWNhc2UgWE1MX0VYVEVSTkFMX1BBUkFNRVRFUl9FTlRJVFk6CgkgICAgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIjwhRU5USVRZICUgIik7CgkgICAgeG1sQnVmZmVyV3JpdGVDSEFSKGJ1ZiwgZW50LT5uYW1lKTsKCSAgICBpZiAoZW50LT5FeHRlcm5hbElEICE9IE5VTEwpIHsKCQkgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIiBQVUJMSUMgIik7CgkJIHhtbEJ1ZmZlcldyaXRlUXVvdGVkU3RyaW5nKGJ1ZiwgZW50LT5FeHRlcm5hbElEKTsKCQkgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIiAiKTsKCQkgeG1sQnVmZmVyV3JpdGVRdW90ZWRTdHJpbmcoYnVmLCBlbnQtPlN5c3RlbUlEKTsKCSAgICB9IGVsc2UgewoJCSB4bWxCdWZmZXJXcml0ZUNoYXIoYnVmLCAiIFNZU1RFTSAiKTsKCQkgeG1sQnVmZmVyV3JpdGVRdW90ZWRTdHJpbmcoYnVmLCBlbnQtPlN5c3RlbUlEKTsKCSAgICB9CgkgICAgeG1sQnVmZmVyV3JpdGVDaGFyKGJ1ZiwgIj5cbiIpOwoJICAgIGJyZWFrOwoJZGVmYXVsdDoKCSAgICB4bWxFbnRpdGllc0VycihYTUxfRFREX1VOS05PV05fRU5USVRZLAoJCSJ4bWxEdW1wRW50aXRpZXNEZWNsOiBpbnRlcm5hbDogdW5rbm93biB0eXBlIGVudGl0eSB0eXBlIik7CiAgICB9Cn0KCi8qKgogKiB4bWxEdW1wRW50aXR5RGVjbFNjYW46CiAqIEBlbnQ6ICBBbiBlbnRpdHkgdGFibGUKICogQGJ1ZjogIEFuIFhNTCBidWZmZXIuCiAqCiAqIFdoZW4gdXNpbmcgdGhlIGhhc2ggdGFibGUgc2NhbiBmdW5jdGlvbiwgYXJndW1lbnRzIG5lZWQgdG8gYmUgcmV2ZXJzZWQKICovCnN0YXRpYyB2b2lkCnhtbER1bXBFbnRpdHlEZWNsU2Nhbih4bWxFbnRpdHlQdHIgZW50LCB4bWxCdWZmZXJQdHIgYnVmKSB7CiAgICB4bWxEdW1wRW50aXR5RGVjbChidWYsIGVudCk7Cn0KICAgICAgCi8qKgogKiB4bWxEdW1wRW50aXRpZXNUYWJsZToKICogQGJ1ZjogIEFuIFhNTCBidWZmZXIuCiAqIEB0YWJsZTogIEFuIGVudGl0eSB0YWJsZQogKgogKiBUaGlzIHdpbGwgZHVtcCB0aGUgY29udGVudCBvZiB0aGUgZW50aXR5IHRhYmxlIGFzIGFuIFhNTCBEVEQgZGVmaW5pdGlvbgogKi8Kdm9pZAp4bWxEdW1wRW50aXRpZXNUYWJsZSh4bWxCdWZmZXJQdHIgYnVmLCB4bWxFbnRpdGllc1RhYmxlUHRyIHRhYmxlKSB7CiAgICB4bWxIYXNoU2Nhbih0YWJsZSwgKHhtbEhhc2hTY2FubmVyKXhtbER1bXBFbnRpdHlEZWNsU2NhbiwgYnVmKTsKfQojZW5kaWYgLyogTElCWE1MX09VVFBVVF9FTkFCTEVEICovCg==