I2luY2x1ZGUgImNvc2UuaCIKI2luY2x1ZGUgImNvbmZpZ3VyZS5oIgojaW5jbHVkZSAiY29zZV9pbnQuaCIKI2luY2x1ZGUgImNyeXB0by5oIgoKI2luY2x1ZGUgPGFzc2VydC5oPgojaWZuZGVmIF9fTUJFRF9fCiNpbmNsdWRlIDxtZW1vcnkuaD4KI2VuZGlmCiNpbmNsdWRlIDxzdGRsaWIuaD4KCiNpZmRlZiBVU0VfTUJFRF9UTFMKCiNpbmNsdWRlICJtYmVkdGxzL2NjbS5oIgojaW5jbHVkZSAibWJlZHRscy9tZC5oIgojaW5jbHVkZSAibWJlZHRscy9jdHJfZHJiZy5oIgojaW5jbHVkZSAibWJlZHRscy9lbnRyb3B5LmgiCiNpbmNsdWRlICJtYmVkdGxzL2VjZHNhLmgiCgpib29sIEZVc2VDb21wcmVzc2VkID0gdHJ1ZTsKCiNkZWZpbmUgTUlOKEEsIEIpICgoQSkgPCAoQikgPyAoQSkgOiAoQikpCgpib29sIEFFU19DQ01fRGVjcnlwdChDT1NFX0VudmVsb3BlZCAqIHBjb3NlLCBpbnQgVFNpemUsIGludCBMU2l6ZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkNyeXB0bywgc2l6ZV90IGNiQ3J5cHRvLCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgc2l6ZV90IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCgltYmVkdGxzX2NjbV9jb250ZXh0IGN0eDsKCWludCBjYk91dDsKCWJ5dGUgKiByZ2JPdXQgPSBOVUxMOwoJaW50IE5TaXplID0gMTUgLSAoTFNpemUvOCk7CglieXRlIHJnYklWWzE1XSA9IHsgMCB9OwoJY29uc3QgY25fY2JvciAqIHBJViA9IE5VTEw7CgltYmVkdGxzX2NpcGhlcl9pZF90IGNpcGhlcjsKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgoJbWJlZHRsc19jY21faW5pdCgmY3R4KTsKCQogICAgICAgIC8vICBTZXR1cCB0aGUgSVYvTm9uY2UgYW5kIHB1dCBpdCBpbnRvIHRoZSBtZXNzYWdlCglwSVYgPSBfQ09TRV9tYXBfZ2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgQ09TRV9IZWFkZXJfSVYsIENPU0VfQk9USCwgTlVMTCk7CglpZiAoKHBJViA9PSBOVUxMKSB8fCAocElWLT50eXBlIT0gQ05fQ0JPUl9CWVRFUykpIHsKCQlpZiAocGVyciAhPSBOVUxMKSBwZXJyLT5lcnIgPSBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUjsKCgllcnJvclJldHVybjoKCQlpZiAocmdiT3V0ICE9IE5VTEwpIENPU0VfRlJFRShyZ2JPdXQsIGNvbnRleHQpOwoJCW1iZWR0bHNfY2NtX2ZyZWUoJmN0eCk7CgkJcmV0dXJuIGZhbHNlOwoJfQoJQ0hFQ0tfQ09ORElUSU9OKHBJVi0+bGVuZ3RoID09IE5TaXplLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgltZW1jcHkocmdiSVYsIHBJVi0+di5zdHIsIHBJVi0+bGVuZ3RoKTsKCgkvLyAgU2V0dXAgYW5kIHJ1biB0aGUgbWJlZFRMUyBjb2RlCgljaXBoZXIgPSBNQkVEVExTX0NJUEhFUl9JRF9BRVM7CgkKICAgICAgICBDSEVDS19DT05ESVRJT04oIW1iZWR0bHNfY2NtX3NldGtleSgmY3R4LCBjaXBoZXIsIHBiS2V5LCBjYktleSo4KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJVFNpemUgLz0gODsgLy8gQ29tZXMgaW4gaW4gYml0cyBub3QgYnl0ZXMuCgoJY2JPdXQgPSAoaW50KSAgY2JDcnlwdG8gLSBUU2l6ZTsKCXJnYk91dCA9IChieXRlICopQ09TRV9DQUxMT0MoY2JPdXQsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHJnYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKICAgICAgICAKCQkKCUNIRUNLX0NPTkRJVElPTighbWJlZHRsc19jY21fYXV0aF9kZWNyeXB0KCZjdHgsIGNiT3V0LCByZ2JJViwgTlNpemUsIHBiQXV0aERhdGEsIGNiQXV0aERhdGEsIHBiQ3J5cHRvLCByZ2JPdXQsICZwYkNyeXB0b1tjYk91dF0sIFRTaXplKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwogICAgICAgIAoJbWJlZHRsc19jY21fZnJlZSgmY3R4KTsKCXBjb3NlLT5wYkNvbnRlbnQgPSByZ2JPdXQ7CglwY29zZS0+Y2JDb250ZW50ID0gY2JPdXQ7CgoJcmV0dXJuIHRydWU7Cn0KCgpib29sIEFFU19DQ01fRW5jcnlwdChDT1NFX0VudmVsb3BlZCAqIHBjb3NlLCBpbnQgVFNpemUsIGludCBMU2l6ZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkF1dGhEYXRhLCBzaXplX3QgY2JBdXRoRGF0YSwgY29zZV9lcnJiYWNrICogcGVycikKewoJbWJlZHRsc19jY21fY29udGV4dCBjdHg7CglpbnQgY2JPdXQ7CglieXRlICogcmdiT3V0ID0gTlVMTDsKCWludCBOU2l6ZSA9IDE1IC0gKExTaXplLzgpOwoJY29uc3QgY25fY2JvciAqIGNib3JfaXYgPSBOVUxMOwoJY25fY2JvciAqIGNib3JfaXZfdCA9IE5VTEw7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoJY25fY2JvciAqIGNuVG1wID0gTlVMTDsKCW1iZWR0bHNfY2lwaGVyX2lkX3QgY2lwaGVyOwoJYnl0ZSByZ2JJVlsxNl07CglieXRlICogcGJJViA9IE5VTEw7Cgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCiAgICAgICAgbWJlZHRsc19jY21faW5pdCgmY3R4KTsKCgljaXBoZXIgPSBNQkVEVExTX0NJUEhFUl9JRF9BRVM7CQkJCgkKCS8vICBTZXR1cCB0aGUgSVYvTm9uY2UgYW5kIHB1dCBpdCBpbnRvIHRoZSBtZXNzYWdlCgljYm9yX2l2ID0gX0NPU0VfbWFwX2dldF9pbnQoJnBjb3NlLT5tX21lc3NhZ2UsIENPU0VfSGVhZGVyX0lWLCBDT1NFX0JPVEgsIHBlcnIpOwoJaWYgKGNib3JfaXYgPT0gTlVMTCkgewoJCiAgICAgICAgICAgICAgICBwYklWID0gQ09TRV9DQUxMT0MoTlNpemUsIDEsIGNvbnRleHQpOwoJCUNIRUNLX0NPTkRJVElPTihwYklWICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoJCXJhbmRfYnl0ZXMocGJJViwgTlNpemUpOwoJCW1lbWNweShyZ2JJViwgcGJJViwgTlNpemUpOwoJCWNib3JfaXZfdCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJJViwgTlNpemUsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CgkJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY2Jvcl9pdl90ICE9IE5VTEwsIGNib3JfZXJyb3IpOwoJCXBiSVYgPSBOVUxMOwoKCQlpZiAoIV9DT1NFX21hcF9wdXQoJnBjb3NlLT5tX21lc3NhZ2UsIENPU0VfSGVhZGVyX0lWLCBjYm9yX2l2X3QsIENPU0VfVU5QUk9URUNUX09OTFksIHBlcnIpKSBnb3RvIGVycm9yUmV0dXJuOwoJCWNib3JfaXZfdCA9IE5VTEw7Cgl9CgllbHNlIHsKCQlDSEVDS19DT05ESVRJT04oY2Jvcl9pdi0+dHlwZSA9PSBDTl9DQk9SX0JZVEVTLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgkJQ0hFQ0tfQ09ORElUSU9OKGNib3JfaXYtPmxlbmd0aCA9PSBOU2l6ZSwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJCW1lbWNweShyZ2JJViwgY2Jvcl9pdi0+di5zdHIsIGNib3JfaXYtPmxlbmd0aCk7Cgl9CgoJLy8gIFNldHVwIGFuZCBydW4gdGhlIG1iZWRUTFMgY29kZQoJCgkvL2NiS2V5IGNvbWVzIGluIGJ5dGVzIG5vdCBiaXRzCiAJQ0hFQ0tfQ09ORElUSU9OKCFtYmVkdGxzX2NjbV9zZXRrZXkoJmN0eCwgY2lwaGVyLCBwYktleSwgY2JLZXkqOCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQoJVFNpemUgLz0gODsgLy8gQ29tZXMgaW4gaW4gYml0cyBub3QgYnl0ZXMuCgogICAgICAgIGNiT3V0ID0gcGNvc2UtPmNiQ29udGVudDsgLy8gTTAwQlVHIC0gVGhpcyBpcyBhIG1pc3NpbmcgY2FsbD8KCXJnYk91dCA9IChieXRlICopQ09TRV9DQUxMT0MoY2JPdXQrVFNpemUsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHJnYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCglDSEVDS19DT05ESVRJT04oIW1iZWR0bHNfY2NtX2VuY3J5cHRfYW5kX3RhZygmY3R4LCBwY29zZS0+Y2JDb250ZW50LCByZ2JJViwgTlNpemUsIHBiQXV0aERhdGEsIGNiQXV0aERhdGEsIHBjb3NlLT5wYkNvbnRlbnQsIHJnYk91dCwgJnJnYk91dFtwY29zZS0+Y2JDb250ZW50XSwgVFNpemUpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CQoKCWNuVG1wID0gY25fY2Jvcl9kYXRhX2NyZWF0ZShyZ2JPdXQsIChpbnQpcGNvc2UtPmNiQ29udGVudCArIFRTaXplLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCk7CglDSEVDS19DT05ESVRJT04oY25UbXAgIT0gTlVMTCwgQ09TRV9FUlJfQ0JPUik7CglyZ2JPdXQgPSBOVUxMOwoKCUNIRUNLX0NPTkRJVElPTihfQ09TRV9hcnJheV9yZXBsYWNlKCZwY29zZS0+bV9tZXNzYWdlLCBjblRtcCwgSU5ERVhfQk9EWSwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBDT1NFX0VSUl9DQk9SKTsKCWNuVG1wID0gTlVMTDsKCgltYmVkdGxzX2NjbV9mcmVlKCZjdHgpOwoJCglyZXR1cm4gdHJ1ZTsKCmVycm9yUmV0dXJuOgoJaWYgKHBiSVYgIT0gTlVMTCkgQ09TRV9GUkVFKHBiSVYsIGNvbnRleHQpOwoJaWYgKGNib3JfaXZfdCAhPSBOVUxMKSBDT1NFX0ZSRUUoY2Jvcl9pdl90LCBjb250ZXh0KTsKCWlmIChyZ2JPdXQgIT0gTlVMTCkgQ09TRV9GUkVFKHJnYk91dCwgY29udGV4dCk7CglpZiAoY25UbXAgIT0gTlVMTCkgQ09TRV9GUkVFKGNuVG1wLCBjb250ZXh0KTsKCW1iZWR0bHNfY2NtX2ZyZWUoJmN0eCk7CglyZXR1cm4gZmFsc2U7Cn0KLyoKYm9vbCBBRVNfR0NNX0RlY3J5cHQoQ09TRV9FbnZlbG9wZWQgKiBwY29zZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkNyeXB0bywgc2l6ZV90IGNiQ3J5cHRvLCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgc2l6ZV90IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCUVWUF9DSVBIRVJfQ1RYIGN0eDsKCWludCBjYk91dDsKCWJ5dGUgKiByZ2JPdXQgPSBOVUxMOwoJaW50IG91dGwgPSAwOwoJYnl0ZSByZ2JJVlsxNV0gPSB7IDAgfTsKCWNvbnN0IGNuX2Nib3IgKiBwSVYgPSBOVUxMOwoJY29uc3QgRVZQX0NJUEhFUiAqIGNpcGhlcjsKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCglpbnQgVFNpemUgPSAxMjggLyA4OwoKCUVWUF9DSVBIRVJfQ1RYX2luaXQoJmN0eCk7CgoJLy8gIFNldHVwIHRoZSBJVi9Ob25jZSBhbmQgcHV0IGl0IGludG8gdGhlIG1lc3NhZ2UKCglwSVYgPSBfQ09TRV9tYXBfZ2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgQ09TRV9IZWFkZXJfSVYsIENPU0VfQk9USCwgTlVMTCk7CglpZiAoKHBJViA9PSBOVUxMKSB8fCAocElWLT50eXBlICE9IENOX0NCT1JfQllURVMpKSB7CgkJaWYgKHBlcnIgIT0gTlVMTCkgcGVyci0+ZXJyID0gQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVI7CgoJZXJyb3JSZXR1cm46CgkJaWYgKHJnYk91dCAhPSBOVUxMKSBDT1NFX0ZSRUUocmdiT3V0LCBjb250ZXh0KTsKCQlFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJCXJldHVybiBmYWxzZTsKCX0KCglDSEVDS19DT05ESVRJT04ocElWLT5sZW5ndGggPT0gOTYvOCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJbWVtY3B5KHJnYklWLCBwSVYtPnYuc3RyLCBwSVYtPmxlbmd0aCk7CgoJLy8gIFNldHVwIGFuZCBydW4gdGhlIE9wZW5TU0wgY29kZQoKCXN3aXRjaCAoY2JLZXkpIHsKCWNhc2UgMTI4IC8gODoKCQljaXBoZXIgPSBFVlBfYWVzXzEyOF9nY20oKTsKCQlicmVhazsKCgljYXNlIDE5MiAvIDg6CgkJY2lwaGVyID0gRVZQX2Flc18xOTJfZ2NtKCk7CgkJYnJlYWs7CgoJY2FzZSAyNTYgLyA4OgoJCWNpcGhlciA9IEVWUF9hZXNfMjU2X2djbSgpOwoJCWJyZWFrOwoKCWRlZmF1bHQ6CgkJRkFJTF9DT05ESVRJT04oQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJCWJyZWFrOwoJfQoKCS8vICBEbyB0aGUgc2V0dXAgZm9yIE9wZW5TU0wKCglDSEVDS19DT05ESVRJT04oRVZQX0RlY3J5cHRJbml0X2V4KCZjdHgsIGNpcGhlciwgTlVMTCwgTlVMTCwgTlVMTCksIENPU0VfRVJSX0RFQ1JZUFRfRkFJTEVEKTsKCglDSEVDS19DT05ESVRJT04oRVZQX0NJUEhFUl9DVFhfY3RybCgmY3R4LCBFVlBfQ1RSTF9DQ01fU0VUX1RBRywgVFNpemUsICh2b2lkICopJnBiQ3J5cHRvW2NiQ3J5cHRvIC0gVFNpemVdKSwgQ09TRV9FUlJfREVDUllQVF9GQUlMRUQpOwoKCUNIRUNLX0NPTkRJVElPTihFVlBfRGVjcnlwdEluaXQoJmN0eCwgMCwgcGJLZXksIHJnYklWKSwgQ09TRV9FUlJfREVDUllQVF9GQUlMRUQpOwoJCgkvLyAgUHVzIGluIHRoZSBBQUQKCglDSEVDS19DT05ESVRJT04oRVZQX0RlY3J5cHRVcGRhdGUoJmN0eCwgTlVMTCwgJm91dGwsIHBiQXV0aERhdGEsIChpbnQpIGNiQXV0aERhdGEpLCBDT1NFX0VSUl9ERUNSWVBUX0ZBSUxFRCk7CgoJLy8gIAoKCWNiT3V0ID0gKGludCljYkNyeXB0byAtIFRTaXplOwoJcmdiT3V0ID0gKGJ5dGUgKilDT1NFX0NBTExPQyhjYk91dCwgMSwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocmdiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCS8vICBQcm9jZXNzIGNvbnRlbnQKCglDSEVDS19DT05ESVRJT04oRVZQX0RlY3J5cHRVcGRhdGUoJmN0eCwgcmdiT3V0LCAmY2JPdXQsIHBiQ3J5cHRvLCAoaW50KWNiQ3J5cHRvIC0gVFNpemUpLCBDT1NFX0VSUl9ERUNSWVBUX0ZBSUxFRCk7CgoJLy8gIFByb2Nlc3MgVGFnCgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9DSVBIRVJfQ1RYX2N0cmwoJmN0eCwgRVZQX0NUUkxfR0NNX1NFVF9UQUcsIFRTaXplLCAoYnl0ZSAqKXBiQ3J5cHRvICsgY2JDcnlwdG8gLSBUU2l6ZSksIENPU0VfRVJSX0RFQ1JZUFRfRkFJTEVEKTsKCgkvLyAgQ2hlY2sgdGhlIHJlc3VsdAoKCUNIRUNLX0NPTkRJVElPTihFVlBfRGVjcnlwdEZpbmFsKCZjdHgsIHJnYk91dCArIGNiT3V0LCAmY2JPdXQpLCBDT1NFX0VSUl9ERUNSWVBUX0ZBSUxFRCk7CgoJRVZQX0NJUEhFUl9DVFhfY2xlYW51cCgmY3R4KTsKCglwY29zZS0+cGJDb250ZW50ID0gcmdiT3V0OwoJcGNvc2UtPmNiQ29udGVudCA9IGNiT3V0OwoKCXJldHVybiB0cnVlOwp9Cgpib29sIEFFU19HQ01fRW5jcnlwdChDT1NFX0VudmVsb3BlZCAqIHBjb3NlLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgY29uc3QgYnl0ZSAqIHBiQXV0aERhdGEsIHNpemVfdCBjYkF1dGhEYXRhLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglFVlBfQ0lQSEVSX0NUWCBjdHg7CglpbnQgY2JPdXQ7CglieXRlICogcmdiT3V0ID0gTlVMTDsKCWludCBvdXRsID0gMDsKCWJ5dGUgcmdiSVZbMTZdID0geyAwIH07CglieXRlICogcGJJViA9IE5VTEw7Cgljb25zdCBjbl9jYm9yICogY2Jvcl9pdiA9IE5VTEw7Cgljbl9jYm9yICogY2Jvcl9pdl90ID0gTlVMTDsKCWNvbnN0IEVWUF9DSVBIRVIgKiBjaXBoZXI7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoJY25fY2Jvcl9lcnJiYWNrIGNib3JfZXJyb3I7CgoJLy8gTWFrZSBpdCBmaXJzdCBzbyB3ZSBjYW4gY2xlYW4gaXQgdXAKCUVWUF9DSVBIRVJfQ1RYX2luaXQoJmN0eCk7CgoJLy8gIFNldHVwIHRoZSBJVi9Ob25jZSBhbmQgcHV0IGl0IGludG8gdGhlIG1lc3NhZ2UKCgljYm9yX2l2ID0gX0NPU0VfbWFwX2dldF9pbnQoJnBjb3NlLT5tX21lc3NhZ2UsIENPU0VfSGVhZGVyX0lWLCBDT1NFX0JPVEgsIHBlcnIpOwoJaWYgKGNib3JfaXYgPT0gTlVMTCkgewoJCXBiSVYgPSBDT1NFX0NBTExPQyg5NiwgMSwgY29udGV4dCk7CgkJQ0hFQ0tfQ09ORElUSU9OKHBiSVYgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgkJcmFuZF9ieXRlcyhwYklWLCA5NiAvIDgpOwoJCW1lbWNweShyZ2JJViwgcGJJViwgOTYgLyA4KTsKCQljYm9yX2l2X3QgPSBjbl9jYm9yX2RhdGFfY3JlYXRlKHBiSVYsIDk2IC8gOCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BICZjYm9yX2Vycm9yKTsKCQlDSEVDS19DT05ESVRJT05fQ0JPUihjYm9yX2l2X3QgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CgkJcGJJViA9IE5VTEw7CgoJCWlmICghX0NPU0VfbWFwX3B1dCgmcGNvc2UtPm1fbWVzc2FnZSwgQ09TRV9IZWFkZXJfSVYsIGNib3JfaXZfdCwgQ09TRV9VTlBST1RFQ1RfT05MWSwgcGVycikpIGdvdG8gZXJyb3JSZXR1cm47CgkJY2Jvcl9pdl90ID0gTlVMTDsKCX0KCWVsc2UgewoJCUNIRUNLX0NPTkRJVElPTihjYm9yX2l2LT50eXBlID09IENOX0NCT1JfQllURVMsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCQlDSEVDS19DT05ESVRJT04oY2Jvcl9pdi0+bGVuZ3RoID09IDk2IC8gOCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJCW1lbWNweShyZ2JJViwgY2Jvcl9pdi0+di5zdHIsIGNib3JfaXYtPmxlbmd0aCk7Cgl9CgoKCXN3aXRjaCAoY2JLZXkqOCkgewoJY2FzZSAxMjg6CgkJY2lwaGVyID0gRVZQX2Flc18xMjhfZ2NtKCk7CgkJYnJlYWs7CgoJY2FzZSAxOTI6CgkJY2lwaGVyID0gRVZQX2Flc18xOTJfZ2NtKCk7CgkJYnJlYWs7CgoJY2FzZSAyNTY6CgkJY2lwaGVyID0gRVZQX2Flc18yNTZfZ2NtKCk7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgkJYnJlYWs7Cgl9CgoJLy8gIFNldHVwIGFuZCBydW4gdGhlIE9wZW5TU0wgY29kZQoKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdEluaXRfZXgoJmN0eCwgY2lwaGVyLCBOVUxMLCBOVUxMLCBOVUxMKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdEluaXQoJmN0eCwgMCwgcGJLZXksIHJnYklWKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCBOVUxMLCAmb3V0bCwgcGJBdXRoRGF0YSwgKGludCkgY2JBdXRoRGF0YSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglyZ2JPdXQgPSAoYnl0ZSAqKUNPU0VfQ0FMTE9DKHBjb3NlLT5jYkNvbnRlbnQgKyAxMjgvOCwgMSwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocmdiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JPdXQsICZjYk91dCwgcGNvc2UtPnBiQ29udGVudCwgKGludClwY29zZS0+Y2JDb250ZW50KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdEZpbmFsX2V4KCZjdHgsICZyZ2JPdXRbY2JPdXRdLCAmY2JPdXQpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9DSVBIRVJfQ1RYX2N0cmwoJmN0eCwgRVZQX0NUUkxfR0NNX0dFVF9UQUcsIDEyOC84LCAmcmdiT3V0W3Bjb3NlLT5jYkNvbnRlbnRdKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCWNuX2Nib3IgKiBjblRtcCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocmdiT3V0LCAoaW50KXBjb3NlLT5jYkNvbnRlbnQgKyAxMjgvOCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpOwoJQ0hFQ0tfQ09ORElUSU9OKGNuVG1wICE9IE5VTEwsIENPU0VfRVJSX0NCT1IpOwoJcmdiT3V0ID0gTlVMTDsKCUNIRUNLX0NPTkRJVElPTihfQ09TRV9hcnJheV9yZXBsYWNlKCZwY29zZS0+bV9tZXNzYWdlLCBjblRtcCwgSU5ERVhfQk9EWSwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBDT1NFX0VSUl9DQk9SKTsKCglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJcmV0dXJuIHRydWU7CgplcnJvclJldHVybjoKCWlmIChwYklWICE9IE5VTEwpIENPU0VfRlJFRShwYklWLCBjb250ZXh0KTsKCWlmIChjYm9yX2l2X3QgIT0gTlVMTCkgQ09TRV9GUkVFKGNib3JfaXZfdCwgY29udGV4dCk7CglpZiAocmdiT3V0ICE9IE5VTEwpIENPU0VfRlJFRShyZ2JPdXQsIGNvbnRleHQpOwoJRVZQX0NJUEhFUl9DVFhfY2xlYW51cCgmY3R4KTsKCXJldHVybiBmYWxzZTsKfQoKCmJvb2wgQUVTX0NCQ19NQUNfQ3JlYXRlKENPU0VfTWFjTWVzc2FnZSAqIHBjb3NlLCBpbnQgVFNpemUsIGNvbnN0IGJ5dGUgKiBwYktleSwgc2l6ZV90IGNiS2V5LCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgc2l6ZV90IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWNvbnN0IEVWUF9DSVBIRVIgKiBwY2lwaGVyID0gTlVMTDsKCUVWUF9DSVBIRVJfQ1RYIGN0eDsKCWludCBjYk91dDsKCWJ5dGUgcmdiSVZbMTZdID0geyAwIH07CglieXRlICogcmdiT3V0ID0gTlVMTDsKCWJvb2wgZiA9IGZhbHNlOwoJdW5zaWduZWQgaW50IGk7Cgljbl9jYm9yICogY24gPSBOVUxMOwojaWZkZWYgVVNFX0NCT1JfQ09OVEVYVAoJY25fY2Jvcl9jb250ZXh0ICogY29udGV4dCA9ICZwY29zZS0+bV9tZXNzYWdlLm1fYWxsb2NDb250ZXh0OwojZW5kaWYKCglFVlBfQ0lQSEVSX0NUWF9pbml0KCZjdHgpOwoKCXJnYk91dCA9IENPU0VfQ0FMTE9DKDE2LCAxLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihyZ2JPdXQgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgoJc3dpdGNoIChjYktleSo4KSB7CgljYXNlIDEyODoKCQlwY2lwaGVyID0gRVZQX2Flc18xMjhfY2JjKCk7CgkJYnJlYWs7CgoJY2FzZSAyNTY6CgkJcGNpcGhlciA9IEVWUF9hZXNfMjU2X2NiYygpOwoJCWJyZWFrOwoKCWRlZmF1bHQ6CgkJRkFJTF9DT05ESVRJT04oQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJfQoKCS8vICBTZXR1cCBhbmQgcnVuIHRoZSBPcGVuU1NMIGNvZGUKCglDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRJbml0X2V4KCZjdHgsIHBjaXBoZXIsIE5VTEwsIHBiS2V5LCByZ2JJViksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglmb3IgKGkgPSAwOyBpIDwgKHVuc2lnbmVkIGludCljYkF1dGhEYXRhIC8gMTY7IGkrKykgewoJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JPdXQsICZjYk91dCwgcGJBdXRoRGF0YSArIChpICogMTYpLCAxNiksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCX0KCWlmIChjYkF1dGhEYXRhICUgMTYgIT0gMCkgewoJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JPdXQsICZjYk91dCwgcGJBdXRoRGF0YSArIChpICogMTYpLCBjYkF1dGhEYXRhICUgMTYpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCByZ2JJViwgMTYgLSAoY2JBdXRoRGF0YSAlIDE2KSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCX0KCgljbiA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocmdiT3V0LCBUU2l6ZSAvIDgsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSBOVUxMKTsKCUNIRUNLX0NPTkRJVElPTihjbiAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCXJnYk91dCA9IE5VTEw7CgoJQ0hFQ0tfQ09ORElUSU9OKF9DT1NFX2FycmF5X3JlcGxhY2UoJnBjb3NlLT5tX21lc3NhZ2UsIGNuLCBJTkRFWF9NQUNfVEFHLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCksIENPU0VfRVJSX0NCT1IpOwoJY24gPSBOVUxMOwoKCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gIWY7CgplcnJvclJldHVybjoKCWlmIChyZ2JPdXQgIT0gTlVMTCkgQ09TRV9GUkVFKHJnYk91dCwgY29udGV4dCk7CglpZiAoY24gIT0gTlVMTCkgQ05fQ0JPUl9GUkVFKGNuLCBjb250ZXh0KTsKCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gZmFsc2U7Cn0KCmJvb2wgQUVTX0NCQ19NQUNfVmFsaWRhdGUoQ09TRV9NYWNNZXNzYWdlICogcGNvc2UsIGludCBUU2l6ZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkF1dGhEYXRhLCBzaXplX3QgY2JBdXRoRGF0YSwgY29zZV9lcnJiYWNrICogcGVycikKewoJY29uc3QgRVZQX0NJUEhFUiAqIHBjaXBoZXIgPSBOVUxMOwoJRVZQX0NJUEhFUl9DVFggY3R4OwoJaW50IGNiT3V0OwoJYnl0ZSByZ2JJVlsxNl0gPSB7IDAgfTsKCWJ5dGUgcmdiVGFnWzE2XSA9IHsgMCB9OwoJYm9vbCBmID0gZmFsc2U7Cgl1bnNpZ25lZCBpbnQgaTsKCglzd2l0Y2ggKGNiS2V5KjgpIHsKCWNhc2UgMTI4OgoJCXBjaXBoZXIgPSBFVlBfYWVzXzEyOF9jYmMoKTsKCQlicmVhazsKCgljYXNlIDI1NjoKCQlwY2lwaGVyID0gRVZQX2Flc18yNTZfY2JjKCk7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7Cgl9CgoJLy8gIFNldHVwIGFuZCBydW4gdGhlIE9wZW5TU0wgY29kZQoKCUVWUF9DSVBIRVJfQ1RYX2luaXQoJmN0eCk7CglDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRJbml0X2V4KCZjdHgsIHBjaXBoZXIsIE5VTEwsIHBiS2V5LCByZ2JJViksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglUU2l6ZSAvPSA4OwoKCWZvciAoaSA9IDA7IGkgPCAodW5zaWduZWQgaW50KSBjYkF1dGhEYXRhIC8gMTY7IGkrKykgewoJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JUYWcsICZjYk91dCwgcGJBdXRoRGF0YSsoaSoxNiksIDE2KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoJaWYgKGNiQXV0aERhdGEgJSAxNiAhPSAwKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYlRhZywgJmNiT3V0LCBwYkF1dGhEYXRhICsgKGkgKiAxNiksIGNiQXV0aERhdGEgJSAxNiksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQlDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRVcGRhdGUoJmN0eCwgcmdiVGFnLCAmY2JPdXQsIHJnYklWLCAxNiAtIChjYkF1dGhEYXRhICUgMTYpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoKCWNuX2Nib3IgKiBjbiA9IF9DT1NFX2FycmF5Z2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgSU5ERVhfTUFDX1RBRyk7CglDSEVDS19DT05ESVRJT04oY24gIT0gTlVMTCwgQ09TRV9FUlJfQ0JPUik7CgoJZm9yIChpID0gMDsgaSA8ICh1bnNpZ25lZCBpbnQpVFNpemU7IGkrKykgZiB8PSAoY24tPnYuYnl0ZXNbaV0gIT0gcmdiVGFnW2ldKTsKCglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJcmV0dXJuICFmOwoKZXJyb3JSZXR1cm46CglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJcmV0dXJuIGZhbHNlOwp9CgojaWYgMAovLyAgV2UgYXJlIGRvaW5nIENCQy1NQUMgbm90IENNQUMgYXQgdGhpcyB0aW1lCmJvb2wgQUVTX0NNQUNfVmFsaWRhdGUoQ09TRV9NYWNNZXNzYWdlICogcGNvc2UsIGludCBLZXlTaXplLCBpbnQgVGFnU2l6ZSwgY29uc3QgYnl0ZSAqIHBiQXV0aERhdGEsIGludCBjYkF1dGhEYXRhLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglDTUFDX0NUWCAqIHBjdHggPSBOVUxMOwoJY29uc3QgRVZQX0NJUEhFUiAqIHBjaXBoZXIgPSBOVUxMOwoJYnl0ZSAqIHJnYk91dCA9IE5VTEw7CglzaXplX3QgY2JPdXQ7Cglib29sIGYgPSBmYWxzZTsKCXVuc2lnbmVkIGludCBpOwojaWZkZWYgVVNFX0NCT1JfQ09OVEVYVAoJY25fY2Jvcl9jb250ZXh0ICogY29udGV4dCA9ICZwY29zZS0+bV9tZXNzYWdlLm1fYWxsb2NDb250ZXh0OwojZW5kaWYKCglwY3R4ID0gQ01BQ19DVFhfbmV3KCk7CgoKCXN3aXRjaCAoS2V5U2l6ZSkgewoJY2FzZSAxMjg6IHBjaXBoZXIgPSBFVlBfYWVzXzEyOF9jYmMoKTsgYnJlYWs7CgljYXNlIDI1NjogcGNpcGhlciA9IEVWUF9hZXNfMjU2X2NiYygpOyBicmVhazsKCWRlZmF1bHQ6IEZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsgYnJlYWs7Cgl9CgoJcmdiT3V0ID0gQ09TRV9DQUxMT0MoMTI4LzgsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHJnYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCglDSEVDS19DT05ESVRJT04oQ01BQ19Jbml0KHBjdHgsIHBjb3NlLT5wYktleSwgcGNvc2UtPmNiS2V5LCBwY2lwaGVyLCBOVUxMICkgPT0gMSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJQ0hFQ0tfQ09ORElUSU9OKENNQUNfVXBkYXRlKHBjdHgsIHBiQXV0aERhdGEsIGNiQXV0aERhdGEpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CglDSEVDS19DT05ESVRJT04oQ01BQ19GaW5hbChwY3R4LCByZ2JPdXQsICZjYk91dCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCgljbl9jYm9yICogY24gPSBfQ09TRV9hcnJheWdldF9pbnQoJnBjb3NlLT5tX21lc3NhZ2UsIElOREVYX01BQ19UQUcpOwoJQ0hFQ0tfQ09ORElUSU9OKGNuICE9IE5VTEwsIENPU0VfRVJSX0NCT1IpOwoKCWZvciAoaSA9IDA7IGkgPCAodW5zaWduZWQgaW50KVRhZ1NpemUgLyA4OyBpKyspIGYgfD0gKGNuLT52LmJ5dGVzW2ldICE9IHJnYk91dFtpXSk7CgoJQ09TRV9GUkVFKHJnYk91dCwgY29udGV4dCk7CglDTUFDX0NUWF9jbGVhbnVwKHBjdHgpOwoJQ01BQ19DVFhfZnJlZShwY3R4KTsKCXJldHVybiAhZjsKCmVycm9yUmV0dXJuOgoJQ09TRV9GUkVFKHJnYk91dCwgY29udGV4dCk7CglDTUFDX0NUWF9jbGVhbnVwKHBjdHgpOwoJQ01BQ19DVFhfZnJlZShwY3R4KTsKCXJldHVybiBmYWxzZTsKCn0KI2VuZGlmCgpib29sIEhLREZfQUVTX0V4cGFuZChDT1NFICogcGNvc2UsIHNpemVfdCBjYml0S2V5LCBjb25zdCBieXRlICogcGJQUkssIHNpemVfdCBjYlBSSywgY29uc3QgYnl0ZSAqIHBiSW5mbywgc2l6ZV90IGNiSW5mbywgYnl0ZSAqIHBiT3V0cHV0LCBzaXplX3QgY2JPdXRwdXQsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWNvbnN0IEVWUF9DSVBIRVIgKiBwY2lwaGVyID0gTlVMTDsKCUVWUF9DSVBIRVJfQ1RYIGN0eDsKCWludCBjYk91dDsKCWJ5dGUgcmdiSVZbMTZdID0geyAwIH07CglieXRlIGJDb3VudCA9IDE7CglzaXplX3QgaWI7CglieXRlIHJnYkRpZ2VzdFsxMjggLyA4XTsKCWludCBjYkRpZ2VzdCA9IDA7CglieXRlIHJnYk91dFsxNl07CgoJRVZQX0NJUEhFUl9DVFhfaW5pdCgmY3R4KTsKCglzd2l0Y2ggKGNiaXRLZXkpIHsKCWNhc2UgMTI4OgoJCXBjaXBoZXIgPSBFVlBfYWVzXzEyOF9jYmMoKTsKCQlicmVhazsKCgljYXNlIDI1NjoKCQlwY2lwaGVyID0gRVZQX2Flc18yNTZfY2JjKCk7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7Cgl9CglDSEVDS19DT05ESVRJT04oY2JQUksgPT0gY2JpdEtleSAvIDgsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCgkvLyAgU2V0dXAgYW5kIHJ1biB0aGUgT3BlblNTTCBjb2RlCgoKCWZvciAoaWIgPSAwOyBpYiA8IGNiT3V0cHV0OyBpYiArPSAxNiwgYkNvdW50ICs9IDEpIHsKCQlzaXplX3QgaWIyOwoKCQlDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRJbml0X2V4KCZjdHgsIHBjaXBoZXIsIE5VTEwsIHBiUFJLLCByZ2JJViksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCByZ2JEaWdlc3QsIGNiRGlnZXN0KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCWZvciAoaWIyID0gMDsgaWIyIDwgY2JJbmZvOyBpYjIrPTE2KSB7CgkJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JPdXQsICZjYk91dCwgcGJJbmZvK2liMiwgKGludCkgTUlOKDE2LCBjYkluZm8taWIyKSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQl9CgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCAmYkNvdW50LCAxKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCWlmICgoY2JJbmZvICsgMSkgJSAxNiAhPSAwKSB7CgkJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JPdXQsICZjYk91dCwgcmdiSVYsIChpbnQpIDE2LShjYkluZm8rMSklMTYpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJfQoJCW1lbWNweShyZ2JEaWdlc3QsIHJnYk91dCwgY2JPdXQpOwoJCWNiRGlnZXN0ID0gY2JPdXQ7CgkJbWVtY3B5KHBiT3V0cHV0ICsgaWIsIHJnYkRpZ2VzdCwgTUlOKDE2LCBjYk91dHB1dCAtIGliKSk7Cgl9CgoJRVZQX0NJUEhFUl9DVFhfY2xlYW51cCgmY3R4KTsKCXJldHVybiB0cnVlOwoKZXJyb3JSZXR1cm46CglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJcmV0dXJuIGZhbHNlOwp9CgoKYm9vbCBIS0RGX0V4dHJhY3QoQ09TRSAqIHBjb3NlLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgc2l6ZV90IGNiaXREaWdlc3QsIGJ5dGUgKiByZ2JEaWdlc3QsIHNpemVfdCAqIHBjYkRpZ2VzdCwgQ0JPUl9DT05URVhUX0NPTU1BIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWJ5dGUgcmdiU2FsdFtFVlBfTUFYX01EX1NJWkVdID0geyAwIH07CglpbnQgY2JTYWx0OwoJY25fY2JvciAqIGNuU2FsdDsKCUhNQUNfQ1RYIGN0eDsKCWNvbnN0IEVWUF9NRCAqIHBtZCA9IE5VTEw7Cgl1bnNpZ25lZCBpbnQgY2JEaWdlc3Q7CgoJSE1BQ19DVFhfaW5pdCgmY3R4KTsKCglpZiAoMCkgewoJZXJyb3JSZXR1cm46CgkJSE1BQ19jbGVhbnVwKCZjdHgpOwoJCXJldHVybiBmYWxzZTsKCX0KCglzd2l0Y2ggKGNiaXREaWdlc3QpIHsKCWNhc2UgMjU2OiBwbWQgPSBFVlBfc2hhMjU2KCk7IGNiU2FsdCA9IDI1NiAvIDg7ICBicmVhazsKCWNhc2UgMzg0OiBwbWQgPSBFVlBfc2hhMzg0KCk7IGNiU2FsdCA9IDM4NCAvIDg7IGJyZWFrOwoJY2FzZSA1MTI6IHBtZCA9IEVWUF9zaGE1MTIoKTsgY2JTYWx0ID0gNTEyIC8gODsgYnJlYWs7CglkZWZhdWx0OiBGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7IGJyZWFrOwoJfQoKCWNuU2FsdCA9IF9DT1NFX21hcF9nZXRfaW50KHBjb3NlLCBDT1NFX0hlYWRlcl9IS0RGX3NhbHQsIENPU0VfQk9USCwgcGVycik7CgoJaWYgKGNuU2FsdCAhPSBOVUxMKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKEhNQUNfSW5pdCgmY3R4LCBjblNhbHQtPnYuYnl0ZXMsIChpbnQpIGNuU2FsdC0+bGVuZ3RoLCBwbWQpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7Cgl9CgllbHNlIHsKCQlDSEVDS19DT05ESVRJT04oSE1BQ19Jbml0KCZjdHgsIHJnYlNhbHQsIGNiU2FsdCwgcG1kKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoJQ0hFQ0tfQ09ORElUSU9OKEhNQUNfVXBkYXRlKCZjdHgsIHBiS2V5LCAoaW50KWNiS2V5KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJQ0hFQ0tfQ09ORElUSU9OKEhNQUNfRmluYWwoJmN0eCwgcmdiRGlnZXN0LCAmY2JEaWdlc3QpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkqcGNiRGlnZXN0ID0gY2JEaWdlc3Q7CglITUFDX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gdHJ1ZTsKfQoKYm9vbCBIS0RGX0V4cGFuZChDT1NFICogcGNvc2UsIHNpemVfdCBjYml0RGlnZXN0LCBjb25zdCBieXRlICogcGJQUkssIHNpemVfdCBjYlBSSywgY29uc3QgYnl0ZSAqIHBiSW5mbywgc2l6ZV90IGNiSW5mbywgYnl0ZSAqIHBiT3V0cHV0LCBzaXplX3QgY2JPdXRwdXQsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCUhNQUNfQ1RYIGN0eDsKCWNvbnN0IEVWUF9NRCAqIHBtZCA9IE5VTEw7CglzaXplX3QgaWI7CglpbnQgY2JTYWx0OwoJdW5zaWduZWQgaW50IGNiRGlnZXN0ID0gMDsKCWJ5dGUgcmdiRGlnZXN0W0VWUF9NQVhfTURfU0laRV07CglieXRlIGJDb3VudCA9IDE7CgoJSE1BQ19DVFhfaW5pdCgmY3R4KTsKCglpZiAoMCkgewoJZXJyb3JSZXR1cm46CgkJSE1BQ19jbGVhbnVwKCZjdHgpOwoJCXJldHVybiBmYWxzZTsKCX0KCglzd2l0Y2ggKGNiaXREaWdlc3QpIHsKCWNhc2UgMjU2OiBwbWQgPSBFVlBfc2hhMjU2KCk7IGNiU2FsdCA9IDI1NiAvIDg7ICBicmVhazsKCWNhc2UgMzg0OiBwbWQgPSBFVlBfc2hhMzg0KCk7IGNiU2FsdCA9IDM4NCAvIDg7IGJyZWFrOwoJY2FzZSA1MTI6IHBtZCA9IEVWUF9zaGE1MTIoKTsgY2JTYWx0ID0gNTEyIC8gODsgYnJlYWs7CglkZWZhdWx0OiBGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7IGJyZWFrOwoJfQoKCglmb3IgKGliID0gMDsgaWIgPCBjYk91dHB1dDsgaWIgKz0gY2JEaWdlc3QsIGJDb3VudCArPSAxKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKEhNQUNfSW5pdF9leCgmY3R4LCBwYlBSSywgKGludCljYlBSSywgcG1kLCBOVUxMKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCUNIRUNLX0NPTkRJVElPTihITUFDX1VwZGF0ZSgmY3R4LCByZ2JEaWdlc3QsIGNiRGlnZXN0KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCUNIRUNLX0NPTkRJVElPTihITUFDX1VwZGF0ZSgmY3R4LCBwYkluZm8sIGNiSW5mbyksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQlDSEVDS19DT05ESVRJT04oSE1BQ19VcGRhdGUoJmN0eCwgJmJDb3VudCwgMSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQlDSEVDS19DT05ESVRJT04oSE1BQ19GaW5hbCgmY3R4LCByZ2JEaWdlc3QsICZjYkRpZ2VzdCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCgkJbWVtY3B5KHBiT3V0cHV0ICsgaWIsIHJnYkRpZ2VzdCwgTUlOKGNiRGlnZXN0LCBjYk91dHB1dCAtIGliKSk7Cgl9CgoJSE1BQ19jbGVhbnVwKCZjdHgpOwoJcmV0dXJuIHRydWU7Cgp9Cgp2b2lkIGR1bXBfb3V0cHV0KGJ5dGUqIGIsIHNpemVfdCBzKXsKCWZvcihpbnQgaSA9IDA7IGkgPCBzOyBpKyspewoJCXByaW50ZigiJTAyeCIsICpiKTsKCQliKys7Cgl9CglwcmludGYoIlxuIik7Cn0KCnZvaWQgZGlmZih1bnNpZ25lZCBjaGFyKiBhLCBzaXplX3QgYV9sLCB1bnNpZ25lZCBjaGFyKiBiLCBzaXplX3QgYl9sKXsKCXNpemVfdCBzOwoJcyA9IChhX2wgPCBiX2wpID8gYV9sIDogYl9sOwoJdW5zaWduZWQgY2hhciogdG1wID0gYTsKCXByaW50Zigic2l6ZSA9ICVkXG4iLHMgKTsKLy8JcHJpbnRmKCIlMDJ4XG4iLCAqdG1wKTsKCWludCBpOwoJZm9yKGkgPSAwOyBpIDwgczsgKytpKXsKCQlwcmludGYoIiUwMngiLCAqdG1wKTsKCQl0bXArKzsKCX0KCXByaW50ZigiXG4iKTsKCgl0bXAgPSBiOwoJZm9yKGkgPSAwOyBpIDwgczsgKytpKXsKCQlwcmludGYoIiUwMngiLCAqdG1wKTsKCQl0bXArKzsKCX0KCXByaW50ZigiXG4iKTsKCglmb3IoaSA9IDA7IGkgPCBzOyArK2kpewoKCQlpZigqYSAhPSAqYil7CgkJCXByaW50ZigiXl4iKTsKCgkJfSBlbHNlIHsKCQkJcHJpbnRmKCJfXyIpOwoKCQl9CgoJCWErKzsKCQliKys7Cgl9CglwcmludGYoIlxuIik7Cn0KKi8KCmJvb2wgSE1BQ19DcmVhdGUoQ09TRV9NYWNNZXNzYWdlICogcGNvc2UsIGludCBIU2l6ZSwgaW50IFRTaXplLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgY29uc3QgYnl0ZSAqIHBiQXV0aERhdGEsIHNpemVfdCBjYkF1dGhEYXRhLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglieXRlICogcmdiT3V0ID0gTlVMTDsKLy8JdW5zaWduZWQgaW50IGNiT3V0OwoJbWJlZHRsc19tZF9jb250ZXh0X3QgY29udHg7Cgljb25zdCBjaGFyKiBtZF9uYW1lOwoJY29uc3Qgc3RydWN0IG1iZWR0bHNfbWRfaW5mb190ICogaW5mbzsKCiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoKCXN3aXRjaCAoSFNpemUpIHsKCQljYXNlIDI1NjogbWRfbmFtZSA9ICJTSEEyNTYiOyBicmVhazsKCQljYXNlIDM4NDogbWRfbmFtZSA9ICJTSEEzODQiOyBicmVhazsKCQljYXNlIDUxMjogbWRfbmFtZSA9ICJTSEE1MTIiOyBicmVhazsKCQlkZWZhdWx0OiBGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7IGJyZWFrOwoJfQoKCWlmICgwKSB7CgllcnJvclJldHVybjoKCQlDT1NFX0ZSRUUocmdiT3V0LCBjb250ZXh0KTsKCQltYmVkdGxzX21kX2ZyZWUoJmNvbnR4KTsgCgkJcmV0dXJuIGZhbHNlOwoJfQoKCW1iZWR0bHNfbWRfaW5pdCgmY29udHgpOwoJaW5mbyA9IG1iZWR0bHNfbWRfaW5mb19mcm9tX3N0cmluZyAobWRfbmFtZSk7CgltYmVkdGxzX21kX3NldHVwKCAmY29udHgsIGluZm8sIDEgKTsKCglyZ2JPdXQgPSBDT1NFX0NBTExPQyhtYmVkdGxzX21kX2dldF9zaXplKGluZm8pLCAxLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihyZ2JPdXQgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgoJQ0hFQ0tfQ09ORElUSU9OKCEobWJlZHRsc19tZF9obWFjX3N0YXJ0cyAoJmNvbnR4LCBwYktleSwgY2JLZXkpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJQ0hFQ0tfQ09ORElUSU9OKCEobWJlZHRsc19tZF9obWFjX3VwZGF0ZSAoJmNvbnR4LCBwYkF1dGhEYXRhLCBjYkF1dGhEYXRhKSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCUNIRUNLX0NPTkRJVElPTighKG1iZWR0bHNfbWRfaG1hY19maW5pc2ggKCZjb250eCwgcmdiT3V0KSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglDSEVDS19DT05ESVRJT04oX0NPU0VfYXJyYXlfcmVwbGFjZSgmcGNvc2UtPm1fbWVzc2FnZSwgY25fY2Jvcl9kYXRhX2NyZWF0ZShyZ2JPdXQsIFRTaXplIC8gOCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBJTkRFWF9NQUNfVEFHLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCksIENPU0VfRVJSX0NCT1IpOwoKCW1iZWR0bHNfbWRfZnJlZSgmY29udHgpOyAKCXJldHVybiB0cnVlOwp9Cgpib29sIEhNQUNfVmFsaWRhdGUoQ09TRV9NYWNNZXNzYWdlICogcGNvc2UsIGludCBIU2l6ZSwgaW50IFRTaXplLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgY29uc3QgYnl0ZSAqIHBiQXV0aERhdGEsIHNpemVfdCBjYkF1dGhEYXRhLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CgltYmVkdGxzX21kX2NvbnRleHRfdCBjb250eDsKCWNvbnN0IGNoYXIqIG1kX25hbWU7Cgljb25zdCBzdHJ1Y3QgbWJlZHRsc19tZF9pbmZvX3QgKiBpbmZvOwoJYnl0ZSAqIHJnYk91dCA9IE5VTEw7Cgl1bnNpZ25lZCBpbnQgY2JPdXQ7Cglib29sIGYgPSBmYWxzZTsKCXVuc2lnbmVkIGludCBpOwoKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgoJc3dpdGNoIChIU2l6ZSkgewoJCWNhc2UgMjU2OiBtZF9uYW1lID0gIlNIQTI1NiI7IGJyZWFrOwoJCWNhc2UgMzg0OiBtZF9uYW1lID0gIlNIQTM4NCI7IGJyZWFrOwoJCWNhc2UgNTEyOiBtZF9uYW1lID0gIlNIQTUxMiI7IGJyZWFrOwoJCWRlZmF1bHQ6IEZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsgYnJlYWs7Cgl9CgoJbWJlZHRsc19tZF9pbml0KCZjb250eCk7CglpbmZvID0gbWJlZHRsc19tZF9pbmZvX2Zyb21fc3RyaW5nIChtZF9uYW1lKTsKCW1iZWR0bHNfbWRfc2V0dXAoICZjb250eCwgaW5mbywgMSApOwoJCgljYk91dCA9IG1iZWR0bHNfbWRfZ2V0X3NpemUoaW5mbyk7CglyZ2JPdXQgPSBDT1NFX0NBTExPQyhjYk91dCwgMSwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocmdiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCUNIRUNLX0NPTkRJVElPTighKG1iZWR0bHNfbWRfaG1hY19zdGFydHMgKCZjb250eCwgcGJLZXksIGNiS2V5KSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCUNIRUNLX0NPTkRJVElPTighKG1iZWR0bHNfbWRfaG1hY191cGRhdGUgKCZjb250eCwgcGJBdXRoRGF0YSwgY2JBdXRoRGF0YSkpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CglDSEVDS19DT05ESVRJT04oIShtYmVkdGxzX21kX2htYWNfZmluaXNoICgmY29udHgsIHJnYk91dCkpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJY25fY2JvciAqIGNuID0gX0NPU0VfYXJyYXlnZXRfaW50KCZwY29zZS0+bV9tZXNzYWdlLCBJTkRFWF9NQUNfVEFHKTsKCUNIRUNLX0NPTkRJVElPTihjbiAhPSBOVUxMLCBDT1NFX0VSUl9DQk9SKTsKCglpZiAoY24tPmxlbmd0aCA+IChpbnQpIGNiT3V0KSByZXR1cm4gZmFsc2U7Cglmb3IgKGkgPSAwOyBpIDwgKHVuc2lnbmVkIGludCkgVFNpemUvODsgaSsrKSBmIHw9IChjbi0+di5ieXRlc1tpXSAhPSByZ2JPdXRbaV0pOwoKCW1iZWR0bHNfbWRfZnJlZSgmY29udHgpOyAKCXJldHVybiAhZjsKCmVycm9yUmV0dXJuOgoJQ09TRV9GUkVFKHJnYk91dCwgY29udGV4dCk7CgltYmVkdGxzX21kX2ZyZWUoJmNvbnR4KTsgCglyZXR1cm4gZmFsc2U7Cn0KCiNkZWZpbmUgQ09TRV9LZXlfRUNfQ3VydmUgLTEKI2RlZmluZSBDT1NFX0tleV9FQ19YIC0yCiNkZWZpbmUgQ09TRV9LZXlfRUNfWSAtMwojZGVmaW5lIENPU0VfS2V5X0VDX2QgLTQKCmJvb2wgRUNLZXlfRnJvbShjb25zdCBjbl9jYm9yICogcEtleSwgbWJlZHRsc19lY3Bfa2V5cGFpciAqa2V5cGFpciwgY29zZV9lcnJiYWNrICogcGVycikKewoJYnl0ZSAgcmdiS2V5W01CRURUTFNfRUNQX01BWF9QVF9MRU5dOwoJaW50IGNiS2V5OwoJaW50IGNiR3JvdXA7Cgljb25zdCBjbl9jYm9yICogcDsKCW1iZWR0bHNfZWNwX2dyb3VwX2lkIGdyb3VwSWQ7CgoJcCA9IGNuX2Nib3JfbWFwZ2V0X2ludChwS2V5LCBDT1NFX0tleV9UeXBlKTsKCUNIRUNLX0NPTkRJVElPTihwICE9IE5VTEwsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCWlmKHAtPnR5cGUgPT0gQ05fQ0JPUl9VSU5UKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKHAtPnYudWludCA9PSBDT1NFX0tleV9UeXBlX0VDMiwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJfQoJZWxzZSB7CgkJRkFJTF9DT05ESVRJT04oQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJfQoKCXAgPSBjbl9jYm9yX21hcGdldF9pbnQocEtleSwgQ09TRV9LZXlfRUNfQ3VydmUpOwoJQ0hFQ0tfQ09ORElUSU9OKChwICE9IE5VTEwpICYmIChwLT50eXBlID09IENOX0NCT1JfVUlOVCksIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCglzd2l0Y2ggKHAtPnYudWludCkgewoJY2FzZSAxOiAvLyBQLTI1NgoJCWdyb3VwSWQgPSBNQkVEVExTX0VDUF9EUF9TRUNQMjU2UjE7CgkJYnJlYWs7CgoJY2FzZSAyOiAvLyBQLTM4NAoJCWdyb3VwSWQgPSBNQkVEVExTX0VDUF9EUF9TRUNQMzg0UjE7CgkJYnJlYWs7CgoJY2FzZSAzOiAvLyBQLTUyMQoJCWdyb3VwSWQgPSBNQkVEVExTX0VDUF9EUF9TRUNQNTIxUjE7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7Cgl9CglDSEVDS19DT05ESVRJT04obWJlZHRsc19lY3BfZ3JvdXBfbG9hZCgma2V5cGFpci0+Z3JwLCBncm91cElkKSA9PSAwLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgljYkdyb3VwID0gKGtleXBhaXItPmdycC5uYml0cyArIDcpIC8gODsKCglwID0gY25fY2Jvcl9tYXBnZXRfaW50KHBLZXksIENPU0VfS2V5X0VDX1gpOwoJQ0hFQ0tfQ09ORElUSU9OKChwICE9IE5VTEwpICYmIChwLT50eXBlID09IENOX0NCT1JfQllURVMpLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CglDSEVDS19DT05ESVRJT04ocC0+bGVuZ3RoID09IGNiR3JvdXAsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCW1lbWNweShyZ2JLZXkrMSwgcC0+di5zdHIsIHAtPmxlbmd0aCk7CgkKCglwID0gY25fY2Jvcl9tYXBnZXRfaW50KHBLZXksIENPU0VfS2V5X0VDX1kpOwoJQ0hFQ0tfQ09ORElUSU9OKChwICE9IE5VTEwpLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CglpZiAocC0+dHlwZSA9PSBDTl9DQk9SX0JZVEVTKSB7CgkJcmdiS2V5WzBdID0gMHgwNDsKCQljYktleSA9IGNiR3JvdXAgKiAyICsgMTsKCQlDSEVDS19DT05ESVRJT04ocC0+bGVuZ3RoID09IGNiR3JvdXAsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCQltZW1jcHkocmdiS2V5ICsgcC0+bGVuZ3RoICsgMSwgcC0+di5zdHIsIHAtPmxlbmd0aCk7Cgl9CgllbHNlIGlmIChwLT50eXBlID09IENOX0NCT1JfVFJVRSkgewoJCWNiS2V5ID0gY2JHcm91cCArIDE7CgkJcmdiS2V5WzBdID0gMHgwMzsKCX0KCWVsc2UgaWYgKHAtPnR5cGUgPT0gQ05fQ0JPUl9GQUxTRSkgewoJCWNiS2V5ID0gY2JHcm91cCArIDE7CgkJcmdiS2V5WzBdID0gMHgwMjsKCX0KCWVsc2UgRkFJTF9DT05ESVRJT04oQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCUNIRUNLX0NPTkRJVElPTihtYmVkdGxzX2VjcF9wb2ludF9yZWFkX2JpbmFyeSgma2V5cGFpci0+Z3JwLCAma2V5cGFpci0+USwgcmdiS2V5LCBjYktleSkgPT0gMCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCXAgPSBjbl9jYm9yX21hcGdldF9pbnQocEtleSwgQ09TRV9LZXlfRUNfZCk7CglpZiAocCAhPSBOVUxMKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKHAtPnR5cGUgPT0gQ05fQ0JPUl9CWVRFUywgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJCUNIRUNLX0NPTkRJVElPTihtYmVkdGxzX21waV9yZWFkX2JpbmFyeSggJmtleXBhaXItPmQsIHAtPnYuYnl0ZXMsIHAtPmxlbmd0aCkgPT0gMCwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoJcmV0dXJuIHRydWU7CgplcnJvclJldHVybjoKCXJldHVybiBmYWxzZTsKfQoKLyoKY25fY2JvciAqIEVDX0Zyb21LZXkoY29uc3QgRUNfS0VZICogcEtleSwgQ0JPUl9DT05URVhUX0NPTU1BIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWNuX2Nib3IgKiBwa2V5ID0gTlVMTDsKCWNvbnN0IEVDX0dST1VQICogcGdyb3VwOwoJaW50IGNvc2VfZ3JvdXA7Cgljbl9jYm9yICogcCA9IE5VTEw7Cgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCWNvbnN0IEVDX1BPSU5UICogcFBvaW50OwoJc2l6ZV90IGNiU2l6ZTsKCWJ5dGUgKiBwYk91dCA9IE5VTEw7CgoJcGdyb3VwID0gRUNfS0VZX2dldDBfZ3JvdXAocEtleSk7CglDSEVDS19DT05ESVRJT04ocGdyb3VwICE9IE5VTEwsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCglzd2l0Y2ggKEVDX0dST1VQX2dldF9jdXJ2ZV9uYW1lKHBncm91cCkpIHsKCWNhc2UgTklEX1g5XzYyX3ByaW1lMjU2djE6CQljb3NlX2dyb3VwID0gMTsJCWJyZWFrOwoJY2FzZSBOSURfc2VjcDM4NHIxOiBjb3NlX2dyb3VwID0gMjsgYnJlYWs7CgljYXNlIE5JRF9zZWNwNTIxcjE6IGNvc2VfZ3JvdXAgPSAzOyBicmVhazsKCglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCglwa2V5ID0gY25fY2Jvcl9tYXBfY3JlYXRlKENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihwa2V5ICE9IE5VTEwsIGNib3JfZXJyb3IpOwoKCXAgPSBjbl9jYm9yX2ludF9jcmVhdGUoY29zZV9ncm91cCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BICZjYm9yX2Vycm9yKTsKCUNIRUNLX0NPTkRJVElPTl9DQk9SKHAgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihjbl9jYm9yX21hcHB1dF9pbnQocGtleSwgQ09TRV9LZXlfRUNfQ3VydmUsIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJcCA9IE5VTEw7CgoJcFBvaW50ID0gRUNfS0VZX2dldDBfcHVibGljX2tleShwS2V5KTsKCUNIRUNLX0NPTkRJVElPTihwUG9pbnQgIT0gTlVMTCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCWlmIChGVXNlQ29tcHJlc3NlZCkgewoJCWNiU2l6ZSA9IEVDX1BPSU5UX3BvaW50Mm9jdChwZ3JvdXAsIHBQb2ludCwgUE9JTlRfQ09OVkVSU0lPTl9DT01QUkVTU0VELCBOVUxMLCAwLCBOVUxMKTsKCQlDSEVDS19DT05ESVRJT04oY2JTaXplID4gMCwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCXBiT3V0ID0gQ09TRV9DQUxMT0MoY2JTaXplLCAxLCBjb250ZXh0KTsKCQlDSEVDS19DT05ESVRJT04ocGJPdXQgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgkJQ0hFQ0tfQ09ORElUSU9OKEVDX1BPSU5UX3BvaW50Mm9jdChwZ3JvdXAsIHBQb2ludCwgUE9JTlRfQ09OVkVSU0lPTl9DT01QUkVTU0VELCBwYk91dCwgY2JTaXplLCBOVUxMKSA9PSBjYlNpemUsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCX0KCWVsc2UgewoJCWNiU2l6ZSA9IEVDX1BPSU5UX3BvaW50Mm9jdChwZ3JvdXAsIHBQb2ludCwgUE9JTlRfQ09OVkVSU0lPTl9VTkNPTVBSRVNTRUQsIE5VTEwsIDAsIE5VTEwpOwoJCUNIRUNLX0NPTkRJVElPTihjYlNpemUgPiAwLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJcGJPdXQgPSBDT1NFX0NBTExPQyhjYlNpemUsIDEsIGNvbnRleHQpOwoJCUNIRUNLX0NPTkRJVElPTihwYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCQlDSEVDS19DT05ESVRJT04oRUNfUE9JTlRfcG9pbnQyb2N0KHBncm91cCwgcFBvaW50LCBQT0lOVF9DT05WRVJTSU9OX1VOQ09NUFJFU1NFRCwgcGJPdXQsIGNiU2l6ZSwgTlVMTCkgPT0gY2JTaXplLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7Cgl9CglwID0gY25fY2Jvcl9kYXRhX2NyZWF0ZShwYk91dCsxLCAoaW50KSAoY2JTaXplIC8gMiksIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihwICE9IE5VTEwsIGNib3JfZXJyb3IpOwoJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X0VDX1gsIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJcCA9IE5VTEw7CgoJaWYgKEZVc2VDb21wcmVzc2VkKSB7CgkJcCA9IGNuX2Nib3JfYm9vbF9jcmVhdGUocGJPdXRbMF0gJiAxLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgJmNib3JfZXJyb3IpOwoJCUNIRUNLX0NPTkRJVElPTl9DQk9SKHAgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CgkJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X0VDX1ksIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJCXAgPSBOVUxMOwoJfQoJZWxzZSB7CgkJcCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJPdXQgKyBjYlNpemUgLyAyICsgMSwgKGludCkoY2JTaXplIC8gMiksIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CgkJcGJPdXQgPSBOVUxMOyAgIC8vIEl0IGlzIGFscmVhZHkgcGFydCBvZiB0aGUgb3RoZXIgb25lLgoJCUNIRUNLX0NPTkRJVElPTl9DQk9SKHAgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CgkJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X0VDX1ksIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJCXAgPSBOVUxMOwoJfQoKCXAgPSBjbl9jYm9yX2ludF9jcmVhdGUoQ09TRV9LZXlfVHlwZV9FQzIsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihwICE9IE5VTEwsIGNib3JfZXJyb3IpOwoJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X1R5cGUsIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJcCA9IE5VTEw7CgpyZXR1cm5IZXJlOgoJaWYgKHBiT3V0ICE9IE5VTEwpIENPU0VfRlJFRShwYk91dCwgY29udGV4dCk7CglpZiAocCAhPSBOVUxMKSBDTl9DQk9SX0ZSRUUocCwgY29udGV4dCk7CglyZXR1cm4gcGtleTsKCmVycm9yUmV0dXJuOgoJQ05fQ0JPUl9GUkVFKHBrZXksIGNvbnRleHQpOwoJcGtleSA9IE5VTEw7Cglnb3RvIHJldHVybkhlcmU7Cn0KKi8KCmJvb2wgRUNEU0FfU2lnbihDT1NFICogcFNpZ25lciwgaW50IGluZGV4LCBjb25zdCBjbl9jYm9yICogcEtleSwgaW50IGNiaXREaWdlc3QsIGNvbnN0IGJ5dGUgKiByZ2JUb1NpZ24sIHNpemVfdCBjYlRvU2lnbiwgY29zZV9lcnJiYWNrICogcGVycikKewojaWYgZGVmaW5lZChNQkVEVExTX0VDRFNBX0RFVEVSTUlOSVNUSUMpCglieXRlIHJnYkRpZ2VzdFtNQkVEVExTX01EX01BWF9TSVpFXTsKCXVpbnQ4X3QgKiBwYlNpZyA9IE5VTEw7Cgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCWludCBjYlI7CgltYmVkdGxzX21kX3R5cGVfdCBtZFR5cGU7Cgljb25zdCBtYmVkdGxzX21kX2luZm9fdCAqcG1kSW5mbzsKCW1iZWR0bHNfZWNwX2tleXBhaXIga2V5cGFpcjsKCW1iZWR0bHNfbXBpIHI7CgltYmVkdGxzX21waSBzOwojaWZkZWYgVVNFX0NCT1JfQ09OVEVYVAoJY25fY2Jvcl9jb250ZXh0ICogY29udGV4dCA9ICZwU2lnbmVyLT5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgljbl9jYm9yICogcCA9IE5VTEw7Cglib29sIHJlc3VsdCA9IGZhbHNlOwoKCW1iZWR0bHNfZWNwX2tleXBhaXJfaW5pdCgma2V5cGFpcik7CgltYmVkdGxzX21waV9pbml0KCZyKTsKCW1iZWR0bHNfbXBpX2luaXQoJnMpOwoKCWlmKCFFQ0tleV9Gcm9tKHBLZXksICZrZXlwYWlyLCBwZXJyKSkgZ290byBlcnJvclJldHVybjsKCglDSEVDS19DT05ESVRJT04oa2V5cGFpci5kLm4gIT0gMCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCXN3aXRjaChjYml0RGlnZXN0KQoJewoJY2FzZSAyNTY6CgkJbWRUeXBlID0gTUJFRFRMU19NRF9TSEEyNTY7CgkJYnJlYWs7CgoJY2FzZSAzODQ6CgkJbWRUeXBlID0gTUJFRFRMU19NRF9TSEEzODQ7CgkJYnJlYWs7CgoJY2FzZSA1MTI6CgkJbWRUeXBlID0gTUJFRFRMU19NRF9TSEE1MTI7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7Cgl9CglwbWRJbmZvID0gbWJlZHRsc19tZF9pbmZvX2Zyb21fdHlwZShtZFR5cGUpOwoJQ0hFQ0tfQ09ORElUSU9OKHBtZEluZm8gIT0gTlVMTCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJQ0hFQ0tfQ09ORElUSU9OKG1iZWR0bHNfbWQocG1kSW5mbywgcmdiVG9TaWduLCBjYlRvU2lnbiwgcmdiRGlnZXN0KSA9PSAwLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgoJQ0hFQ0tfQ09ORElUSU9OKG1iZWR0bHNfZWNkc2Ffc2lnbl9kZXQoJmtleXBhaXIuZ3JwLCAmciwgJnMsICZrZXlwYWlyLmQsIHJnYkRpZ2VzdCwgbWJlZHRsc19tZF9nZXRfc2l6ZShwbWRJbmZvKSwgbWRUeXBlKSA9PSAwLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJY2JSID0gKGtleXBhaXIuZ3JwLm5iaXRzICsgNykgLyA4OwoKCXBiU2lnID0gQ09TRV9DQUxMT0MoY2JSLCAyLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihwYlNpZyAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCglDSEVDS19DT05ESVRJT04obWJlZHRsc19tcGlfd3JpdGVfYmluYXJ5KCZyLCBwYlNpZywgY2JSKSA9PSAwLCBDT1NFX0VSUl9JTlRFUk5BTCk7CglDSEVDS19DT05ESVRJT04obWJlZHRsc19tcGlfd3JpdGVfYmluYXJ5KCZzLCBwYlNpZyArIGNiUiwgY2JSKSA9PSAwLCBDT1NFX0VSUl9JTlRFUk5BTCk7CgoJcCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJTaWcsIGNiUioyLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgJmNib3JfZXJyb3IpOwoJQ0hFQ0tfQ09ORElUSU9OX0NCT1IocCAhPSBOVUxMLCBjYm9yX2Vycm9yKTsKCglDSEVDS19DT05ESVRJT04oX0NPU0VfYXJyYXlfcmVwbGFjZShwU2lnbmVyLCBwLCBpbmRleCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBDT1NFX0VSUl9DQk9SKTsKCglwID0gTlVMTDsKCXBiU2lnID0gTlVMTDsKCXJlc3VsdCA9IHRydWU7CgplcnJvclJldHVybjoKCWNuX2Nib3JfZnJlZShwIENCT1JfQ09OVEVYVF9QQVJBTSk7CglDT1NFX0ZSRUUocGJTaWcsIGNvbnRleHQpOwoJbWJlZHRsc19tcGlfZnJlZSgmcik7CgltYmVkdGxzX21waV9mcmVlKCZzKTsKCW1iZWR0bHNfZWNwX2tleXBhaXJfZnJlZSgma2V5cGFpcik7CglyZXR1cm4gcmVzdWx0OwojZWxzZQoJcmV0dXJuIGZhbHNlOwojZW5kaWYKfQoKCmJvb2wgRUNEU0FfVmVyaWZ5KENPU0UgKiBwU2lnbmVyLCBpbnQgaW5kZXgsIGNvbnN0IGNuX2Nib3IgKiBwS2V5LCBpbnQgY2JpdERpZ2VzdCwgY29uc3QgYnl0ZSAqIHJnYlRvU2lnbiwgc2l6ZV90IGNiVG9TaWduLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CgltYmVkdGxzX2VjcF9rZXlwYWlyIGtleXBhaXI7CgltYmVkdGxzX21waSByOwoJbWJlZHRsc19tcGkgczsKCW1iZWR0bHNfbWRfdHlwZV90IG1kVHlwZTsKCWNvbnN0IG1iZWR0bHNfbWRfaW5mb190ICpwbWRJbmZvOwoJYnl0ZSByZ2JEaWdlc3RbTUJFRFRMU19NRF9NQVhfU0laRV07Cgljbl9jYm9yICogcFNpZzsKCWJvb2wgcmVzdWx0ID0gZmFsc2U7CgoJbWJlZHRsc19lY3Bfa2V5cGFpcl9pbml0KCZrZXlwYWlyKTsKCW1iZWR0bHNfbXBpX2luaXQoJnIpOwoJbWJlZHRsc19tcGlfaW5pdCgmcyk7CgoJaWYoIUVDS2V5X0Zyb20ocEtleSwgJmtleXBhaXIsIHBlcnIpKSBnb3RvIGVycm9yUmV0dXJuOwoKCXN3aXRjaChjYml0RGlnZXN0KQoJewoJY2FzZSAyNTY6CgkJbWRUeXBlID0gTUJFRFRMU19NRF9TSEEyNTY7CgkJYnJlYWs7CgoJY2FzZSAzODQ6CgkJbWRUeXBlID0gTUJFRFRMU19NRF9TSEEzODQ7CgkJYnJlYWs7CgoJY2FzZSA1MTI6CgkJbWRUeXBlID0gTUJFRFRMU19NRF9TSEE1MTI7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7Cgl9CglwbWRJbmZvID0gbWJlZHRsc19tZF9pbmZvX2Zyb21fdHlwZShtZFR5cGUpOwoJQ0hFQ0tfQ09ORElUSU9OKHBtZEluZm8gIT0gTlVMTCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJQ0hFQ0tfQ09ORElUSU9OKG1iZWR0bHNfbWQocG1kSW5mbywgcmdiVG9TaWduLCBjYlRvU2lnbiwgcmdiRGlnZXN0KSA9PSAwLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgoJcFNpZyA9IF9DT1NFX2FycmF5Z2V0X2ludChwU2lnbmVyLCBpbmRleCk7CglDSEVDS19DT05ESVRJT04oKHBTaWcgIT0gTlVMTCkgJiYgKHBTaWctPnR5cGUgPT0gQ05fQ0JPUl9CWVRFUyksIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCglDSEVDS19DT05ESVRJT04obWJlZHRsc19tcGlfcmVhZF9iaW5hcnkoICZyLCBwU2lnLT52LmJ5dGVzLCBwU2lnLT5sZW5ndGggLyAyICkgPT0gMCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CglDSEVDS19DT05ESVRJT04obWJlZHRsc19tcGlfcmVhZF9iaW5hcnkoICZzLCBwU2lnLT52LmJ5dGVzICsgcFNpZy0+bGVuZ3RoIC8gMiwgcFNpZy0+bGVuZ3RoIC8gMiApID09IDAsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoJQ0hFQ0tfQ09ORElUSU9OKG1iZWR0bHNfZWNkc2FfdmVyaWZ5KCZrZXlwYWlyLmdycCwgcmdiRGlnZXN0LCBtYmVkdGxzX21kX2dldF9zaXplKHBtZEluZm8pLCAma2V5cGFpci5RLCAmciwgJnMpID09IDAsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglyZXN1bHQgPSB0cnVlOwoKZXJyb3JSZXR1cm46CgltYmVkdGxzX21waV9mcmVlKCZyKTsKCW1iZWR0bHNfbXBpX2ZyZWUoJnMpOwoJbWJlZHRsc19lY3Bfa2V5cGFpcl9mcmVlKCZrZXlwYWlyKTsKCXJldHVybiByZXN1bHQ7Cn0KCi8qCmJvb2wgQUVTX0tXX0RlY3J5cHQoQ09TRV9FbnZlbG9wZWQgKiBwY29zZSwgY29uc3QgYnl0ZSAqIHBiS2V5SW4sIHNpemVfdCBjYml0S2V5LCBjb25zdCBieXRlICogcGJDaXBoZXJUZXh0LCBzaXplX3QgY2JDaXBoZXJUZXh0LCBieXRlICogcGJLZXlPdXQsIGludCAqIHBjYktleU91dCwgY29zZV9lcnJiYWNrICogcGVycikKewoJYnl0ZSByZ2JPdXRbNTEyIC8gOF07CglBRVNfS0VZIGtleTsKCglDSEVDS19DT05ESVRJT04oQUVTX3NldF9kZWNyeXB0X2tleShwYktleUluLCAoaW50KWNiaXRLZXksICZrZXkpID09IDAsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglDSEVDS19DT05ESVRJT04oQUVTX3Vud3JhcF9rZXkoJmtleSwgTlVMTCwgcmdiT3V0LCBwYkNpcGhlclRleHQsIChpbnQpIGNiQ2lwaGVyVGV4dCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCgltZW1jcHkocGJLZXlPdXQsIHJnYk91dCwgY2JDaXBoZXJUZXh0IC0gOCk7CgkqcGNiS2V5T3V0ID0gKGludCkgKGNiQ2lwaGVyVGV4dCAtIDgpOwoKCXJldHVybiB0cnVlOwplcnJvclJldHVybjoKCXJldHVybiBmYWxzZTsKfQoKYm9vbCBBRVNfS1dfRW5jcnlwdChDT1NFX1JlY2lwaWVudEluZm8gKiBwY29zZSwgY29uc3QgYnl0ZSAqIHBiS2V5SW4sIGludCBjYml0S2V5LCBjb25zdCBieXRlICogIHBiQ29udGVudCwgaW50ICBjYkNvbnRlbnQsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWJ5dGUgICpwYk91dCA9IE5VTEw7CglBRVNfS0VZIGtleTsKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fZW5jcnlwdC5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoJY25fY2JvciAqIGNuVG1wID0gTlVMTDsKCglwYk91dCA9IENPU0VfQ0FMTE9DKGNiQ29udGVudCArIDgsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHBiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCUNIRUNLX0NPTkRJVElPTihBRVNfc2V0X2VuY3J5cHRfa2V5KHBiS2V5SW4sIGNiaXRLZXksICZrZXkpID09IDAsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglDSEVDS19DT05ESVRJT04oQUVTX3dyYXBfa2V5KCZrZXksIE5VTEwsIHBiT3V0LCBwYkNvbnRlbnQsIGNiQ29udGVudCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCgljblRtcCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJPdXQsIChpbnQpY2JDb250ZW50ICsgOCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpOwoJQ0hFQ0tfQ09ORElUSU9OKGNuVG1wICE9IE5VTEwsIENPU0VfRVJSX0NCT1IpOwoJcGJPdXQgPSBOVUxMOwoJQ0hFQ0tfQ09ORElUSU9OKF9DT1NFX2FycmF5X3JlcGxhY2UoJnBjb3NlLT5tX2VuY3J5cHQubV9tZXNzYWdlLCBjblRtcCwgSU5ERVhfQk9EWSwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBDT1NFX0VSUl9DQk9SKTsKCWNuVG1wID0gTlVMTDsKCglyZXR1cm4gdHJ1ZTsKCmVycm9yUmV0dXJuOgoJQ09TRV9GUkVFKGNuVG1wLCBjb250ZXh0KTsKCWlmIChwYk91dCAhPSBOVUxMKSBDT1NFX0ZSRUUocGJPdXQsIGNvbnRleHQpOwoJcmV0dXJuIGZhbHNlOwp9CgoqLwovKgovLyNpbmNsdWRlIDxzdGRpby5oPiAvL1RPRE8Kdm9pZCByYW5kX2J5dGVzKGJ5dGUgKiBwYiwgc2l6ZV90IGNiKXsKLy9jdHgtPmFlc19jdHgtPnJrIGUgbnVsbC4uLiBpIGNhbGxjaGFpbmVuIGzkbmdzdCBpbi4gcHJvdmEgaXN05GxsZXQ6Ci8va29sbGEgaHR0cHM6Ly90bHMubWJlZC5vcmcva2IvaG93LXRvL2FkZC1hLXJhbmRvbS1nZW5lcmF0b3IKICAgICAgICAvL2luaXQgcmFuZG9tCiAgICAgICBtYmVkdGxzX2N0cl9kcmJnX2NvbnRleHQgY3RyX2RyYmc7CiAgICAgICAgICBjaGFyICpwZXJzb25hbGl6YXRpb24gPSAibXlfYXBwX3NwZWNpZmljX3N0cmluZyI7CgogICAgICAgIHJldCA9IG1iZWR0bHNfY3RyX2RyYmdfaW5pdCggJmN0cl9kcmJnLCBtYmVkdGxzX2VudHJvcHlfZnVuYywgJmVudHJvcHksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNvbnN0IHVuc2lnbmVkIGNoYXIgKikgcGVyc29uYWxpemF0aW9uLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RybGVuKCBwZXJzb25hbGl6YXRpb24gKSApOyAKICAgICAgICAKICAgICAgICBpZihyZXQgIT0gMCkgewogICAgICAgICAgICAvL3ByaW50ZiBUT0RPCiAgICAgICAgfSAgICAKICAgIAogICAgICAgIG1iZWR0bHNfY3RyX2RyYmdfcmFuZG9tKCZjdHgscGIsIGNiKTsgCQoKCW1iZWR0bHNfY3RyX2RyYmdfZnJlZSgmY3R4KTsJCiAgICAgICAgcHJpbnRmKCJyYW5kIGJ5dXRlIGRvbmVcbiIpOwp9Ki8KLy9UT0RPIEhPVyBUTyBHRU5FUkFURSBHT09EIFJBTkRPTSBCWVRFUwpzdGF0aWMgY29uc3QgdW5zaWduZWQgY2hhciBlbnRyb3B5X3NvdXJjZV9wcls5Nl0gPQogICB7IDB4YzEsIDB4ODAsIDB4ODEsIDB4YTYsIDB4NWQsIDB4NDQsIDB4MDIsIDB4MTYsCiAgICAgMHgxOSwgMHhiMywgMHhmMSwgMHg4MCwgMHhiMSwgMHhjOSwgMHgyMCwgMHgwMiwKICAgICAweDZhLCAweDU0LCAweDZmLCAweDBjLCAweDcwLCAweDgxLCAweDQ5LCAweDhiLAogICAgIDB4NmUsIDB4YTYsIDB4NjIsIDB4NTIsIDB4NmQsIDB4NTEsIDB4YjEsIDB4Y2IsCiAgICAgMHg1OCwgMHgzYiwgMHhmYSwgMHhkNSwgMHgzNywgMHg1ZiwgMHhmYiwgMHhjOSwKICAgICAweGZmLCAweDQ2LCAweGQyLCAweDE5LCAweGM3LCAweDIyLCAweDNlLCAweDk1LAogICAgIDB4NDUsIDB4OWQsIDB4ODIsIDB4ZTEsIDB4ZTcsIDB4MjIsIDB4OWYsIDB4NjMsCiAgICAgMHgzMSwgMHg2OSwgMHhkMiwgMHg2YiwgMHg1NywgMHg0NywgMHg0ZiwgMHhhMywKICAgICAweDM3LCAweGM5LCAweDk4LCAweDFjLCAweDBiLCAweGZiLCAweDkxLCAweDMxLAogICAgIDB4NGQsIDB4NTUsIDB4YjksIDB4ZTksIDB4MWMsIDB4NWEsIDB4NWUsIDB4ZTQsCiAgICAgMHg5MywgMHg5MiwgMHhjZiwgMHhjNSwgMHgyMywgMHgxMiwgMHhkNSwgMHg1NiwKICAgICAweDJjLCAweDRhLCAweDZlLCAweGZmLCAweGRjLCAweDEwLCAweGQwLCAweDY4IH07CgpzdGF0aWMgY29uc3QgdW5zaWduZWQgY2hhciBub25jZV9wZXJzX3ByWzE2XSA9CiAgICAgeyAweGQyLCAweDU0LCAweGZjLCAweGZmLCAweDAyLCAweDFlLCAweDY5LCAweGQyLAogICAgICAweDI5LCAweGM5LCAweGNmLCAweGFkLCAweDg1LCAweGZhLCAweDQ4LCAweDZjIH07CgpzdGF0aWMgc2l6ZV90IHRlc3Rfb2Zmc2V0OwpzdGF0aWMgaW50IGN0cl9kcmJnX3NlbGZfdGVzdF9lbnRyb3B5KCB2b2lkICpkYXRhLCB1bnNpZ25lZCBjaGFyICpidWYsIHNpemVfdCBsZW4gKSB7CiAgICBjb25zdCB1bnNpZ25lZCBjaGFyICpwID0gZGF0YTsKICAgIG1lbWNweSggYnVmLCBwICsgdGVzdF9vZmZzZXQsIGxlbiApOwogICAgdGVzdF9vZmZzZXQgKz0gbGVuOwogICAgcmV0dXJuKCAwICk7CiB9Cgp2b2lkIHJhbmRfYnl0ZXMoYnl0ZSogcGIsIHNpemVfdCBjYil7CiAgICAgCiAgICAgbWJlZHRsc19jdHJfZHJiZ19jb250ZXh0IGN0eDsKICAgIC8vIHVuc2lnbmVkIGNoYXIgYnVmWzE2XTsKICAgICAKICAgICBtYmVkdGxzX2N0cl9kcmJnX2luaXQoICZjdHggKTsKICAgICAKICAgICBtYmVkdGxzX2N0cl9kcmJnX3NlZWRfZW50cm9weV9sZW4oICZjdHgsIGN0cl9kcmJnX3NlbGZfdGVzdF9lbnRyb3B5LCAodm9pZCAqKSBlbnRyb3B5X3NvdXJjZV9wciwgbm9uY2VfcGVyc19wciwgMTYsIDMyICk7CiAgICAgCiAgICAgLy9tYmVkdGxzX2N0cl9kcmJnX3NldF9wcmVkaWN0aW9uX3Jlc2lzdGFuY2UoICZjdHgsIE1CRURUTFNfQ1RSX0RSQkdfUFJfT04gKTsKICAgIAogICAgIG1iZWR0bHNfY3RyX2RyYmdfcmFuZG9tKCAmY3R4LCBwYiwgY2IgKTsKICAgICAvL21iZWR0bHNfY3RyX2RyYmdfcmFuZG9tKCAmY3R4LCBidWYsIE1CRURUTFNfQ1RSX0RSQkdfQkxPQ0tTSVpFICk7CiAgICAgLy9tZW1jbXAoIGJ1ZiwgcmVzdWx0X3ByLCBNQkVEVExTX0NUUl9EUkJHX0JMT0NLU0laRSApICk7CiAgICAgCiAgICAgbWJlZHRsc19jdHJfZHJiZ19mcmVlKCAmY3R4ICk7Cn0KLy9FTkQgT0YgVE9ETyBSQU5ET00gQllURVMKCi8qIQoqCiogQHBhcmFtW2luXSBwUmVjaXBlbnQJUG9pbnRlciB0byB0aGUgbWVzc2FnZSBvYmplY3QKKiBAcGFyYW1baW5dIHBwS2V5UHJpdmF0ZQlBZGRyZXNzIG9mIGtleSB3aXRoIHByaXZhdGUgcG9ydGlvbgoqIEBwYXJhbVtpbl0gcEtleVB1YmxpYwlBZGRyZXNzIG9mIHRoZSBrZXkgdy9vIGEgcHJpdmF0ZSBwb3J0aW9uCiogQHBhcmFtW2luL291dF0gcHBiU2VjcmV0CXBvaW50ZXIgdG8gYnVmZmVyIHRvIGhvbGQgdGhlIGNvbXB1dGVkIHNlY3JldAoqIEBwYXJhbVtpbi9vdXRdIHBjYlNlY3JldAlzaXplIG9mIHRoZSBjb21wdXRlZCBzZWNyZXQKKiBAcGFyYW1baW5dIGNvbnRleHQJCWNib3IgYWxsb2NhdGlvbiBjb250ZXh0IHN0cnVjdHVyZQoqIEBwYXJhbVtvdXRdIHBlcnIJCQlsb2NhdGlvbiB0byByZXR1cm4gZXJyb3IgaW5mb3JtYXRpb24KKiBAcmV0dXJucwkJc3VjY2VzcyBvZiB0aGUgZnVuY3Rpb24KKi8KLyoKYm9vbCBFQ0RIX0NvbXB1dGVTZWNyZXQoQ09TRSAqIHBSZWNpcGllbnQsIGNuX2Nib3IgKiogcHBLZXlQcml2YXRlLCBjb25zdCBjbl9jYm9yICogcEtleVB1YmxpYywgYnl0ZSAqKiBwcGJTZWNyZXQsIHNpemVfdCAqIHBjYlNlY3JldCwgQ0JPUl9DT05URVhUX0NPTU1BIGNvc2VfZXJyYmFjayAqcGVycikKewoJRUNfS0VZICogcGVja2V5UHJpdmF0ZSA9IE5VTEw7CglFQ19LRVkgKiBwZWNrZXlQdWJsaWMgPSBOVUxMOwoJaW50IGNiR3JvdXA7CglpbnQgY2JzZWNyZXQ7CglieXRlICogcGJzZWNyZXQgPSBOVUxMOwoJYm9vbCBmUmV0ID0gZmFsc2U7CgoJcGVja2V5UHVibGljID0gRUNLZXlfRnJvbShwS2V5UHVibGljLCAmY2JHcm91cCwgcGVycik7CglpZiAocGVja2V5UHVibGljID09IE5VTEwpIGdvdG8gZXJyb3JSZXR1cm47CgoJaWYgKCpwcEtleVByaXZhdGUgPT0gTlVMTCkgewoJCXsKCQkJY25fY2JvciAqIHBDb21wcmVzcyA9IF9DT1NFX21hcF9nZXRfaW50KHBSZWNpcGllbnQsIENPU0VfSGVhZGVyX1VzZUNvbXByZXNzZWRFQ0RILCBDT1NFX0JPVEgsIHBlcnIpOwoJCQlpZiAocENvbXByZXNzID09IE5VTEwpIEZVc2VDb21wcmVzc2VkID0gZmFsc2U7CgkJCWVsc2UgRlVzZUNvbXByZXNzZWQgPSAocENvbXByZXNzLT50eXBlID09IENOX0NCT1JfVFJVRSk7CgkJfQoJCXBlY2tleVByaXZhdGUgPSBFQ19LRVlfbmV3KCk7CgkJRUNfS0VZX3NldF9ncm91cChwZWNrZXlQcml2YXRlLCBFQ19LRVlfZ2V0MF9ncm91cChwZWNrZXlQdWJsaWMpKTsKCQlDSEVDS19DT05ESVRJT04oRUNfS0VZX2dlbmVyYXRlX2tleShwZWNrZXlQcml2YXRlKSA9PSAxLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJKnBwS2V5UHJpdmF0ZSA9IEVDX0Zyb21LZXkocGVja2V5UHJpdmF0ZSwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIHBlcnIpOwoJCWlmICgqcHBLZXlQcml2YXRlID09IE5VTEwpIGdvdG8gZXJyb3JSZXR1cm47Cgl9CgllbHNlIHsKCQlwZWNrZXlQcml2YXRlID0gRUNLZXlfRnJvbSgqcHBLZXlQcml2YXRlLCAmY2JHcm91cCwgcGVycik7CgkJaWYgKHBlY2tleVByaXZhdGUgPT0gTlVMTCkgZ290byBlcnJvclJldHVybjsKCX0KCglwYnNlY3JldCA9IENPU0VfQ0FMTE9DKGNiR3JvdXAsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHBic2VjcmV0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCWNic2VjcmV0ID0gRUNESF9jb21wdXRlX2tleShwYnNlY3JldCwgY2JHcm91cCwgRUNfS0VZX2dldDBfcHVibGljX2tleShwZWNrZXlQdWJsaWMpLCBwZWNrZXlQcml2YXRlLCBOVUxMKTsKCUNIRUNLX0NPTkRJVElPTihjYnNlY3JldCA+IDAsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCgkqcHBiU2VjcmV0ID0gcGJzZWNyZXQ7CgkqcGNiU2VjcmV0ID0gY2JzZWNyZXQ7CglwYnNlY3JldCA9IE5VTEw7CgoJZlJldCA9IHRydWU7CgplcnJvclJldHVybjoKCWlmIChwYnNlY3JldCAhPSBOVUxMKSBDT1NFX0ZSRUUocGJzZWNyZXQsIGNvbnRleHQpOwoJaWYgKHBlY2tleVB1YmxpYyAhPSBOVUxMKSBFQ19LRVlfZnJlZShwZWNrZXlQdWJsaWMpOwoJaWYgKHBlY2tleVByaXZhdGUgIT0gTlVMTCkgRUNfS0VZX2ZyZWUocGVja2V5UHJpdmF0ZSk7CgoJcmV0dXJuIGZSZXQ7Cn0KKi8KI2VuZGlmIC8vIFVTRV9NQkVEX1RMUwo=