Ly8NCi8vIKkgQ29weXJpZ2h0IEhlbnJpayBSYXZuIDIwMDQNCi8vDQovLyBVc2UsIG1vZGlmaWNhdGlvbiBhbmQgZGlzdHJpYnV0aW9uIGFyZSBzdWJqZWN0IHRvIHRoZSBCb29zdCBTb2Z0d2FyZSBMaWNlbnNlLCBWZXJzaW9uIDEuMC4NCi8vIChTZWUgYWNjb21wYW55aW5nIGZpbGUgTElDRU5TRV8xXzAudHh0IG9yIGNvcHkgYXQgaHR0cDovL3d3dy5ib29zdC5vcmcvTElDRU5TRV8xXzAudHh0KQ0KLy8NCg0KdXNpbmcgU3lzdGVtOw0KdXNpbmcgU3lzdGVtLkNvbGxlY3Rpb25zOw0KdXNpbmcgU3lzdGVtLklPOw0KDQovLyB1bmNvbW1lbnQgdGhlIGRlZmluZSBiZWxvdyB0byBpbmNsdWRlIHVuaXQgdGVzdHMNCi8vI2RlZmluZSBudW5pdA0KI2lmIG51bml0DQp1c2luZyBOVW5pdC5GcmFtZXdvcms7DQoNCi8vIFVuaXQgdGVzdHMgZm9yIHRoZSBEb3RaTGliIGNsYXNzIGxpYnJhcnkNCi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCi8vDQovLyBVc2UgdGhpcyB3aXRoIE5Vbml0IDIgZnJvbSBodHRwOi8vd3d3Lm51bml0Lm9yZw0KLy8NCg0KbmFtZXNwYWNlIERvdFpMaWJUZXN0cw0Kew0KICAgIHVzaW5nIERvdFpMaWI7DQoNCiAgICAvLyBoZWxwZXIgbWV0aG9kcw0KICAgIGludGVybmFsIGNsYXNzIFV0aWxzDQogICAgew0KICAgICAgICBwdWJsaWMgc3RhdGljIGJvb2wgYnl0ZUFyckVxdWFsKCBieXRlW10gbGhzLCBieXRlW10gcmhzICkNCiAgICAgICAgew0KICAgICAgICAgICAgaWYgKGxocy5MZW5ndGggIT0gcmhzLkxlbmd0aCkNCiAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7DQogICAgICAgICAgICBmb3IgKGludCBpID0gbGhzLkxlbmd0aC0xOyBpID49IDA7IC0taSkNCiAgICAgICAgICAgICAgICBpZiAobGhzW2ldICE9IHJoc1tpXSkNCiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOw0KICAgICAgICAgICAgcmV0dXJuIHRydWU7DQogICAgICAgIH0NCg0KICAgIH0NCg0KDQogICAgW1Rlc3RGaXh0dXJlXQ0KICAgIHB1YmxpYyBjbGFzcyBDaXJjQnVmZmVyVGVzdHMNCiAgICB7DQogICAgICAgICNyZWdpb24gQ2lyY3VsYXIgYnVmZmVyIHRlc3RzDQogICAgICAgIFtUZXN0XQ0KICAgICAgICBwdWJsaWMgdm9pZCBTaW5nbGVQdXRHZXQoKQ0KICAgICAgICB7DQogICAgICAgICAgICBDaXJjdWxhckJ1ZmZlciBidWYgPSBuZXcgQ2lyY3VsYXJCdWZmZXIoMTApOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCAwLCBidWYuU2l6ZSApOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCAtMSwgYnVmLkdldCgpICk7DQoNCiAgICAgICAgICAgIEFzc2VydC5Jc1RydWUoYnVmLlB1dCggMSApKTsNCiAgICAgICAgICAgIEFzc2VydC5BcmVFcXVhbCggMSwgYnVmLlNpemUgKTsNCiAgICAgICAgICAgIEFzc2VydC5BcmVFcXVhbCggMSwgYnVmLkdldCgpICk7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIDAsIGJ1Zi5TaXplICk7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIC0xLCBidWYuR2V0KCkgKTsNCiAgICAgICAgfQ0KDQogICAgICAgIFtUZXN0XQ0KICAgICAgICBwdWJsaWMgdm9pZCBCbG9ja1B1dEdldCgpDQogICAgICAgIHsNCiAgICAgICAgICAgIENpcmN1bGFyQnVmZmVyIGJ1ZiA9IG5ldyBDaXJjdWxhckJ1ZmZlcigxMCk7DQogICAgICAgICAgICBieXRlW10gYXJyID0gezEsMiwzLDQsNSw2LDcsOCw5LDEwfTsNCiAgICAgICAgICAgIEFzc2VydC5BcmVFcXVhbCggMTAsIGJ1Zi5QdXQoYXJyLDAsMTApICk7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIDEwLCBidWYuU2l6ZSApOw0KICAgICAgICAgICAgQXNzZXJ0LklzRmFsc2UoIGJ1Zi5QdXQoMTEpICk7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIDEsIGJ1Zi5HZXQoKSApOw0KICAgICAgICAgICAgQXNzZXJ0LklzVHJ1ZSggYnVmLlB1dCgxMSkgKTsNCg0KICAgICAgICAgICAgYnl0ZVtdIGFycjIgPSAoYnl0ZVtdKWFyci5DbG9uZSgpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCA5LCBidWYuR2V0KGFycjIsMSw5KSApOw0KICAgICAgICAgICAgQXNzZXJ0LklzVHJ1ZSggVXRpbHMuYnl0ZUFyckVxdWFsKGFycixhcnIyKSApOw0KICAgICAgICB9DQoNCiAgICAgICAgI2VuZHJlZ2lvbg0KICAgIH0NCg0KICAgIFtUZXN0Rml4dHVyZV0NCiAgICBwdWJsaWMgY2xhc3MgQ2hlY2tzdW1UZXN0cw0KICAgIHsNCiAgICAgICAgI3JlZ2lvbiBDUkMzMiBUZXN0cw0KICAgICAgICBbVGVzdF0NCiAgICAgICAgcHVibGljIHZvaWQgQ1JDMzJfTnVsbCgpDQogICAgICAgIHsNCiAgICAgICAgICAgIENSQzMyQ2hlY2tzdW0gY3JjMzIgPSBuZXcgQ1JDMzJDaGVja3N1bSgpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCAwLCBjcmMzMi5WYWx1ZSApOw0KDQogICAgICAgICAgICBjcmMzMiA9IG5ldyBDUkMzMkNoZWNrc3VtKDEpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCAxLCBjcmMzMi5WYWx1ZSApOw0KDQogICAgICAgICAgICBjcmMzMiA9IG5ldyBDUkMzMkNoZWNrc3VtKDU1Nik7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIDU1NiwgY3JjMzIuVmFsdWUgKTsNCiAgICAgICAgfQ0KDQogICAgICAgIFtUZXN0XQ0KICAgICAgICBwdWJsaWMgdm9pZCBDUkMzMl9EYXRhKCkNCiAgICAgICAgew0KICAgICAgICAgICAgQ1JDMzJDaGVja3N1bSBjcmMzMiA9IG5ldyBDUkMzMkNoZWNrc3VtKCk7DQogICAgICAgICAgICBieXRlW10gZGF0YSA9IHsgMSwyLDMsNCw1LDYsNyB9Ow0KICAgICAgICAgICAgY3JjMzIuVXBkYXRlKGRhdGEpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCAweDcwZTQ2ODg4LCBjcmMzMi5WYWx1ZSAgKTsNCg0KICAgICAgICAgICAgY3JjMzIgPSBuZXcgQ1JDMzJDaGVja3N1bSgpOw0KICAgICAgICAgICAgY3JjMzIuVXBkYXRlKCJwZW5ndWluIik7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIDB4MGU1YzFhMTIwLCBjcmMzMi5WYWx1ZSApOw0KDQogICAgICAgICAgICBjcmMzMiA9IG5ldyBDUkMzMkNoZWNrc3VtKDEpOw0KICAgICAgICAgICAgY3JjMzIuVXBkYXRlKCJwZW5ndWluIik7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoMHg0M2I2YWE5NCwgY3JjMzIuVmFsdWUpOw0KDQogICAgICAgIH0NCiAgICAgICAgI2VuZHJlZ2lvbg0KDQogICAgICAgICNyZWdpb24gQWRsZXIgdGVzdHMNCg0KICAgICAgICBbVGVzdF0NCiAgICAgICAgcHVibGljIHZvaWQgQWRsZXJfTnVsbCgpDQogICAgICAgIHsNCiAgICAgICAgICAgIEFkbGVyQ2hlY2tzdW0gYWRsZXIgPSBuZXcgQWRsZXJDaGVja3N1bSgpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKDAsIGFkbGVyLlZhbHVlKTsNCg0KICAgICAgICAgICAgYWRsZXIgPSBuZXcgQWRsZXJDaGVja3N1bSgxKTsNCiAgICAgICAgICAgIEFzc2VydC5BcmVFcXVhbCggMSwgYWRsZXIuVmFsdWUgKTsNCg0KICAgICAgICAgICAgYWRsZXIgPSBuZXcgQWRsZXJDaGVja3N1bSg1NTYpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCA1NTYsIGFkbGVyLlZhbHVlICk7DQogICAgICAgIH0NCg0KICAgICAgICBbVGVzdF0NCiAgICAgICAgcHVibGljIHZvaWQgQWRsZXJfRGF0YSgpDQogICAgICAgIHsNCiAgICAgICAgICAgIEFkbGVyQ2hlY2tzdW0gYWRsZXIgPSBuZXcgQWRsZXJDaGVja3N1bSgxKTsNCiAgICAgICAgICAgIGJ5dGVbXSBkYXRhID0geyAxLDIsMyw0LDUsNiw3IH07DQogICAgICAgICAgICBhZGxlci5VcGRhdGUoZGF0YSk7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIDB4NWIwMDFkLCBhZGxlci5WYWx1ZSAgKTsNCg0KICAgICAgICAgICAgYWRsZXIgPSBuZXcgQWRsZXJDaGVja3N1bSgpOw0KICAgICAgICAgICAgYWRsZXIuVXBkYXRlKCJwZW5ndWluIik7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoMHgwYmNmMDJmNiwgYWRsZXIuVmFsdWUgKTsNCg0KICAgICAgICAgICAgYWRsZXIgPSBuZXcgQWRsZXJDaGVja3N1bSgxKTsNCiAgICAgICAgICAgIGFkbGVyLlVwZGF0ZSgicGVuZ3VpbiIpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKDB4MGJkNjAyZjcsIGFkbGVyLlZhbHVlKTsNCg0KICAgICAgICB9DQogICAgICAgICNlbmRyZWdpb24NCiAgICB9DQoNCiAgICBbVGVzdEZpeHR1cmVdDQogICAgcHVibGljIGNsYXNzIEluZm9UZXN0cw0KICAgIHsNCiAgICAgICAgI3JlZ2lvbiBJbmZvIHRlc3RzDQogICAgICAgIFtUZXN0XQ0KICAgICAgICBwdWJsaWMgdm9pZCBJbmZvX1ZlcnNpb24oKQ0KICAgICAgICB7DQogICAgICAgICAgICBJbmZvIGluZm8gPSBuZXcgSW5mbygpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKCIxLjIuNSIsIEluZm8uVmVyc2lvbik7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoMzIsIGluZm8uU2l6ZU9mVUludCk7DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoMzIsIGluZm8uU2l6ZU9mVUxvbmcpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKDMyLCBpbmZvLlNpemVPZlBvaW50ZXIpOw0KICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKDMyLCBpbmZvLlNpemVPZk9mZnNldCk7DQogICAgICAgIH0NCiAgICAgICAgI2VuZHJlZ2lvbg0KICAgIH0NCg0KICAgIFtUZXN0Rml4dHVyZV0NCiAgICBwdWJsaWMgY2xhc3MgRGVmbGF0ZUluZmxhdGVUZXN0cw0KICAgIHsNCiAgICAgICAgI3JlZ2lvbiBEZWZsYXRlIHRlc3RzDQogICAgICAgIFtUZXN0XQ0KICAgICAgICBwdWJsaWMgdm9pZCBEZWZsYXRlX0luaXQoKQ0KICAgICAgICB7DQogICAgICAgICAgICB1c2luZyAoRGVmbGF0ZXIgZGVmID0gbmV3IERlZmxhdGVyKENvbXByZXNzTGV2ZWwuRGVmYXVsdCkpDQogICAgICAgICAgICB7DQogICAgICAgICAgICB9DQogICAgICAgIH0NCg0KICAgICAgICBwcml2YXRlIEFycmF5TGlzdCBjb21wcmVzc2VkRGF0YSA9IG5ldyBBcnJheUxpc3QoKTsNCiAgICAgICAgcHJpdmF0ZSB1aW50IGFkbGVyMTsNCg0KICAgICAgICBwcml2YXRlIEFycmF5TGlzdCB1bmNvbXByZXNzZWREYXRhID0gbmV3IEFycmF5TGlzdCgpOw0KICAgICAgICBwcml2YXRlIHVpbnQgYWRsZXIyOw0KDQogICAgICAgIHB1YmxpYyB2b2lkIENEYXRhQXZhaWwoYnl0ZVtdIGRhdGEsIGludCBzdGFydEluZGV4LCBpbnQgY291bnQpDQogICAgICAgIHsNCiAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY291bnQ7ICsraSkNCiAgICAgICAgICAgICAgICBjb21wcmVzc2VkRGF0YS5BZGQoZGF0YVtpK3N0YXJ0SW5kZXhdKTsNCiAgICAgICAgfQ0KDQogICAgICAgIFtUZXN0XQ0KICAgICAgICBwdWJsaWMgdm9pZCBEZWZsYXRlX0NvbXByZXNzKCkNCiAgICAgICAgew0KICAgICAgICAgICAgY29tcHJlc3NlZERhdGEuQ2xlYXIoKTsNCg0KICAgICAgICAgICAgYnl0ZVtdIHRlc3REYXRhID0gbmV3IGJ5dGVbMzUwMDBdOw0KICAgICAgICAgICAgZm9yIChpbnQgaSA9IDA7IGkgPCB0ZXN0RGF0YS5MZW5ndGg7ICsraSkNCiAgICAgICAgICAgICAgICB0ZXN0RGF0YVtpXSA9IDU7DQoNCiAgICAgICAgICAgIHVzaW5nIChEZWZsYXRlciBkZWYgPSBuZXcgRGVmbGF0ZXIoKENvbXByZXNzTGV2ZWwpNSkpDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgZGVmLkRhdGFBdmFpbGFibGUgKz0gbmV3IERhdGFBdmFpbGFibGVIYW5kbGVyKENEYXRhQXZhaWwpOw0KICAgICAgICAgICAgICAgIGRlZi5BZGQodGVzdERhdGEpOw0KICAgICAgICAgICAgICAgIGRlZi5GaW5pc2goKTsNCiAgICAgICAgICAgICAgICBhZGxlcjEgPSBkZWYuQ2hlY2tzdW07DQogICAgICAgICAgICB9DQogICAgICAgIH0NCiAgICAgICAgI2VuZHJlZ2lvbg0KDQogICAgICAgICNyZWdpb24gSW5mbGF0ZSB0ZXN0cw0KICAgICAgICBbVGVzdF0NCiAgICAgICAgcHVibGljIHZvaWQgSW5mbGF0ZV9Jbml0KCkNCiAgICAgICAgew0KICAgICAgICAgICAgdXNpbmcgKEluZmxhdGVyIGluZiA9IG5ldyBJbmZsYXRlcigpKQ0KICAgICAgICAgICAgew0KICAgICAgICAgICAgfQ0KICAgICAgICB9DQoNCiAgICAgICAgcHJpdmF0ZSB2b2lkIEREYXRhQXZhaWwoYnl0ZVtdIGRhdGEsIGludCBzdGFydEluZGV4LCBpbnQgY291bnQpDQogICAgICAgIHsNCiAgICAgICAgICAgIGZvciAoaW50IGkgPSAwOyBpIDwgY291bnQ7ICsraSkNCiAgICAgICAgICAgICAgICB1bmNvbXByZXNzZWREYXRhLkFkZChkYXRhW2krc3RhcnRJbmRleF0pOw0KICAgICAgICB9DQoNCiAgICAgICAgW1Rlc3RdDQogICAgICAgIHB1YmxpYyB2b2lkIEluZmxhdGVfRXhwYW5kKCkNCiAgICAgICAgew0KICAgICAgICAgICAgdW5jb21wcmVzc2VkRGF0YS5DbGVhcigpOw0KDQogICAgICAgICAgICB1c2luZyAoSW5mbGF0ZXIgaW5mID0gbmV3IEluZmxhdGVyKCkpDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgaW5mLkRhdGFBdmFpbGFibGUgKz0gbmV3IERhdGFBdmFpbGFibGVIYW5kbGVyKEREYXRhQXZhaWwpOw0KICAgICAgICAgICAgICAgIGluZi5BZGQoKGJ5dGVbXSljb21wcmVzc2VkRGF0YS5Ub0FycmF5KHR5cGVvZihieXRlKSkpOw0KICAgICAgICAgICAgICAgIGluZi5GaW5pc2goKTsNCiAgICAgICAgICAgICAgICBhZGxlcjIgPSBpbmYuQ2hlY2tzdW07DQogICAgICAgICAgICB9DQogICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoIGFkbGVyMSwgYWRsZXIyICk7DQogICAgICAgIH0NCiAgICAgICAgI2VuZHJlZ2lvbg0KICAgIH0NCg0KICAgIFtUZXN0Rml4dHVyZV0NCiAgICBwdWJsaWMgY2xhc3MgR1ppcFN0cmVhbVRlc3RzDQogICAgew0KICAgICAgICAjcmVnaW9uIEdaaXBTdHJlYW0gdGVzdA0KICAgICAgICBbVGVzdF0NCiAgICAgICAgcHVibGljIHZvaWQgR1ppcFN0cmVhbV9Xcml0ZVJlYWQoKQ0KICAgICAgICB7DQogICAgICAgICAgICB1c2luZyAoR1ppcFN0cmVhbSBnek91dCA9IG5ldyBHWmlwU3RyZWFtKCJnenN0cmVhbS5neiIsIENvbXByZXNzTGV2ZWwuQmVzdCkpDQogICAgICAgICAgICB7DQogICAgICAgICAgICAgICAgQmluYXJ5V3JpdGVyIHdyaXRlciA9IG5ldyBCaW5hcnlXcml0ZXIoZ3pPdXQpOw0KICAgICAgICAgICAgICAgIHdyaXRlci5Xcml0ZSgiaGkgdGhlcmUiKTsNCiAgICAgICAgICAgICAgICB3cml0ZXIuV3JpdGUoTWF0aC5QSSk7DQogICAgICAgICAgICAgICAgd3JpdGVyLldyaXRlKDQyKTsNCiAgICAgICAgICAgIH0NCg0KICAgICAgICAgICAgdXNpbmcgKEdaaXBTdHJlYW0gZ3pJbiA9IG5ldyBHWmlwU3RyZWFtKCJnenN0cmVhbS5neiIpKQ0KICAgICAgICAgICAgew0KICAgICAgICAgICAgICAgIEJpbmFyeVJlYWRlciByZWFkZXIgPSBuZXcgQmluYXJ5UmVhZGVyKGd6SW4pOw0KICAgICAgICAgICAgICAgIHN0cmluZyBzID0gcmVhZGVyLlJlYWRTdHJpbmcoKTsNCiAgICAgICAgICAgICAgICBBc3NlcnQuQXJlRXF1YWwoImhpIHRoZXJlIixzKTsNCiAgICAgICAgICAgICAgICBkb3VibGUgZCA9IHJlYWRlci5SZWFkRG91YmxlKCk7DQogICAgICAgICAgICAgICAgQXNzZXJ0LkFyZUVxdWFsKE1hdGguUEksIGQpOw0KICAgICAgICAgICAgICAgIGludCBpID0gcmVhZGVyLlJlYWRJbnQzMigpOw0KICAgICAgICAgICAgICAgIEFzc2VydC5BcmVFcXVhbCg0MixpKTsNCiAgICAgICAgICAgIH0NCg0KICAgICAgICB9DQogICAgICAgICNlbmRyZWdpb24NCgl9DQp9DQoNCiNlbmRpZg0K