I2luY2x1ZGUgImNvc2UuaCIKI2luY2x1ZGUgImNvbmZpZ3VyZS5oIgojaW5jbHVkZSAiY29zZV9pbnQuaCIKI2luY2x1ZGUgImNyeXB0by5oIgoKI2luY2x1ZGUgPGFzc2VydC5oPgojaW5jbHVkZSA8bWVtb3J5Lmg+CgojaWZkZWYgVVNFX01CRURfVExTCgojaW5jbHVkZSAiaW5jbHVkZS9tYmVkdGxzL2NjbS5oIgojaW5jbHVkZSAiaW5jbHVkZS9tYmVkdGxzL21kLmgiCiNpbmNsdWRlICJpbmNsdWRlL21iZWR0bHMvY3RyX2RyYmcuaCIKI2luY2x1ZGUgImluY2x1ZGUvbWJlZHRscy9lbnRyb3B5LmgiCgpib29sIEZVc2VDb21wcmVzc2VkID0gdHJ1ZTsKCiNkZWZpbmUgTUlOKEEsIEIpICgoQSkgPCAoQikgPyAoQSkgOiAoQikpCgpib29sIEFFU19DQ01fRGVjcnlwdChDT1NFX0VudmVsb3BlZCAqIHBjb3NlLCBpbnQgVFNpemUsIGludCBMU2l6ZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkNyeXB0bywgc2l6ZV90IGNiQ3J5cHRvLCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgc2l6ZV90IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCgltYmVkdGxzX2NjbV9jb250ZXh0IGN0eDsKCWludCBjYk91dDsKCWJ5dGUgKiByZ2JPdXQgPSBOVUxMOwoJaW50IE5TaXplID0gMTUgLSAoTFNpemUvOCk7CglieXRlIHJnYklWWzE1XSA9IHsgMCB9OwoJY29uc3QgY25fY2JvciAqIHBJViA9IE5VTEw7CgltYmVkdGxzX2NpcGhlcl9pZF90IGNpcGhlcjsKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgoJbWJlZHRsc19jY21faW5pdCgmY3R4KTsKCQogICAgICAgIC8vICBTZXR1cCB0aGUgSVYvTm9uY2UgYW5kIHB1dCBpdCBpbnRvIHRoZSBtZXNzYWdlCglwSVYgPSBfQ09TRV9tYXBfZ2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgQ09TRV9IZWFkZXJfSVYsIENPU0VfQk9USCwgTlVMTCk7CglpZiAoKHBJViA9PSBOVUxMKSB8fCAocElWLT50eXBlIT0gQ05fQ0JPUl9CWVRFUykpIHsKCQlpZiAocGVyciAhPSBOVUxMKSBwZXJyLT5lcnIgPSBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUjsKCgllcnJvclJldHVybjoKCQlpZiAocmdiT3V0ICE9IE5VTEwpIENPU0VfRlJFRShyZ2JPdXQsIGNvbnRleHQpOwoJCW1iZWR0bHNfY2NtX2ZyZWUoJmN0eCk7CgkJcmV0dXJuIGZhbHNlOwoJfQoJQ0hFQ0tfQ09ORElUSU9OKHBJVi0+bGVuZ3RoID09IE5TaXplLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgltZW1jcHkocmdiSVYsIHBJVi0+di5zdHIsIHBJVi0+bGVuZ3RoKTsKCgkvLyAgU2V0dXAgYW5kIHJ1biB0aGUgbWJlZFRMUyBjb2RlCgljaXBoZXIgPSBNQkVEVExTX0NJUEhFUl9JRF9BRVM7CgkKICAgICAgICBDSEVDS19DT05ESVRJT04oIW1iZWR0bHNfY2NtX3NldGtleSgmY3R4LCBjaXBoZXIsIHBiS2V5LCBjYktleSo4KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJVFNpemUgLz0gODsgLy8gQ29tZXMgaW4gaW4gYml0cyBub3QgYnl0ZXMuCgoJY2JPdXQgPSAoaW50KSAgY2JDcnlwdG8gLSBUU2l6ZTsKCXJnYk91dCA9IChieXRlICopQ09TRV9DQUxMT0MoY2JPdXQsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHJnYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKICAgICAgICAKCQkKCUNIRUNLX0NPTkRJVElPTighbWJlZHRsc19jY21fYXV0aF9kZWNyeXB0KCZjdHgsIGNiT3V0LCByZ2JJViwgTlNpemUsIHBiQXV0aERhdGEsIGNiQXV0aERhdGEsIHBiQ3J5cHRvLCByZ2JPdXQsICZwYkNyeXB0b1tjYk91dF0sIFRTaXplKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwogICAgICAgIAoJbWJlZHRsc19jY21fZnJlZSgmY3R4KTsKCXBjb3NlLT5wYkNvbnRlbnQgPSByZ2JPdXQ7CglwY29zZS0+Y2JDb250ZW50ID0gY2JPdXQ7CgoJcmV0dXJuIHRydWU7Cn0KCgpib29sIEFFU19DQ01fRW5jcnlwdChDT1NFX0VudmVsb3BlZCAqIHBjb3NlLCBpbnQgVFNpemUsIGludCBMU2l6ZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkF1dGhEYXRhLCBzaXplX3QgY2JBdXRoRGF0YSwgY29zZV9lcnJiYWNrICogcGVycikKewoJbWJlZHRsc19jY21fY29udGV4dCBjdHg7CglpbnQgY2JPdXQ7CglieXRlICogcmdiT3V0ID0gTlVMTDsKCWludCBOU2l6ZSA9IDE1IC0gKExTaXplLzgpOwoJY29uc3QgY25fY2JvciAqIGNib3JfaXYgPSBOVUxMOwoJY25fY2JvciAqIGNib3JfaXZfdCA9IE5VTEw7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoJY25fY2JvciAqIGNuVG1wID0gTlVMTDsKCW1iZWR0bHNfY2lwaGVyX2lkX3QgY2lwaGVyOwoJYnl0ZSByZ2JJVlsxNl07CglieXRlICogcGJJViA9IE5VTEw7Cgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCiAgICAgICAgbWJlZHRsc19jY21faW5pdCgmY3R4KTsKCgljaXBoZXIgPSBNQkVEVExTX0NJUEhFUl9JRF9BRVM7CQkJCgkKCS8vICBTZXR1cCB0aGUgSVYvTm9uY2UgYW5kIHB1dCBpdCBpbnRvIHRoZSBtZXNzYWdlCgljYm9yX2l2ID0gX0NPU0VfbWFwX2dldF9pbnQoJnBjb3NlLT5tX21lc3NhZ2UsIENPU0VfSGVhZGVyX0lWLCBDT1NFX0JPVEgsIHBlcnIpOwoJaWYgKGNib3JfaXYgPT0gTlVMTCkgewoJCiAgICAgICAgICAgICAgICBwYklWID0gQ09TRV9DQUxMT0MoTlNpemUsIDEsIGNvbnRleHQpOwoJCUNIRUNLX0NPTkRJVElPTihwYklWICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoJCXJhbmRfYnl0ZXMocGJJViwgTlNpemUpOwoJCW1lbWNweShyZ2JJViwgcGJJViwgTlNpemUpOwoJCWNib3JfaXZfdCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJJViwgTlNpemUsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CgkJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY2Jvcl9pdl90ICE9IE5VTEwsIGNib3JfZXJyb3IpOwoJCXBiSVYgPSBOVUxMOwoKCQlpZiAoIV9DT1NFX21hcF9wdXQoJnBjb3NlLT5tX21lc3NhZ2UsIENPU0VfSGVhZGVyX0lWLCBjYm9yX2l2X3QsIENPU0VfVU5QUk9URUNUX09OTFksIHBlcnIpKSBnb3RvIGVycm9yUmV0dXJuOwoJCWNib3JfaXZfdCA9IE5VTEw7Cgl9CgllbHNlIHsKCQlDSEVDS19DT05ESVRJT04oY2Jvcl9pdi0+dHlwZSA9PSBDTl9DQk9SX0JZVEVTLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgkJQ0hFQ0tfQ09ORElUSU9OKGNib3JfaXYtPmxlbmd0aCA9PSBOU2l6ZSwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJCW1lbWNweShyZ2JJViwgY2Jvcl9pdi0+di5zdHIsIGNib3JfaXYtPmxlbmd0aCk7Cgl9CgoJLy8gIFNldHVwIGFuZCBydW4gdGhlIG1iZWRUTFMgY29kZQoJCgkvL2NiS2V5IGNvbWVzIGluIGJ5dGVzIG5vdCBiaXRzCiAJQ0hFQ0tfQ09ORElUSU9OKCFtYmVkdGxzX2NjbV9zZXRrZXkoJmN0eCwgY2lwaGVyLCBwYktleSwgY2JLZXkqOCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQoJVFNpemUgLz0gODsgLy8gQ29tZXMgaW4gaW4gYml0cyBub3QgYnl0ZXMuCgoJcmdiT3V0ID0gKGJ5dGUgKilDT1NFX0NBTExPQyhjYk91dCtUU2l6ZSwgMSwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocmdiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCUNIRUNLX0NPTkRJVElPTighbWJlZHRsc19jY21fZW5jcnlwdF9hbmRfdGFnKCZjdHgsIHBjb3NlLT5jYkNvbnRlbnQsIHJnYklWLCBOU2l6ZSwgcGJBdXRoRGF0YSwgY2JBdXRoRGF0YSwgcGNvc2UtPnBiQ29udGVudCwgcmdiT3V0LCAmcmdiT3V0W3Bjb3NlLT5jYkNvbnRlbnRdLCBUU2l6ZSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsJCgoJY25UbXAgPSBjbl9jYm9yX2RhdGFfY3JlYXRlKHJnYk91dCwgKGludClwY29zZS0+Y2JDb250ZW50ICsgVFNpemUsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSBOVUxMKTsKCUNIRUNLX0NPTkRJVElPTihjblRtcCAhPSBOVUxMLCBDT1NFX0VSUl9DQk9SKTsKCXJnYk91dCA9IE5VTEw7CgoJQ0hFQ0tfQ09ORElUSU9OKF9DT1NFX2FycmF5X3JlcGxhY2UoJnBjb3NlLT5tX21lc3NhZ2UsIGNuVG1wLCBJTkRFWF9CT0RZLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCksIENPU0VfRVJSX0NCT1IpOwoJY25UbXAgPSBOVUxMOwoKCW1iZWR0bHNfY2NtX2ZyZWUoJmN0eCk7CgkKCXJldHVybiB0cnVlOwoKZXJyb3JSZXR1cm46CglpZiAocGJJViAhPSBOVUxMKSBDT1NFX0ZSRUUocGJJViwgY29udGV4dCk7CglpZiAoY2Jvcl9pdl90ICE9IE5VTEwpIENPU0VfRlJFRShjYm9yX2l2X3QsIGNvbnRleHQpOwoJaWYgKHJnYk91dCAhPSBOVUxMKSBDT1NFX0ZSRUUocmdiT3V0LCBjb250ZXh0KTsKCWlmIChjblRtcCAhPSBOVUxMKSBDT1NFX0ZSRUUoY25UbXAsIGNvbnRleHQpOwoJcHJpbnRmKCJlcnJvclJldHVybiBmcm9tIE9QRU5TU0xcbiIpOwoJbWJlZHRsc19jY21fZnJlZSgmY3R4KTsKCXJldHVybiBmYWxzZTsKfQovKgpib29sIEFFU19HQ01fRGVjcnlwdChDT1NFX0VudmVsb3BlZCAqIHBjb3NlLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgY29uc3QgYnl0ZSAqIHBiQ3J5cHRvLCBzaXplX3QgY2JDcnlwdG8sIGNvbnN0IGJ5dGUgKiBwYkF1dGhEYXRhLCBzaXplX3QgY2JBdXRoRGF0YSwgY29zZV9lcnJiYWNrICogcGVycikKewoJRVZQX0NJUEhFUl9DVFggY3R4OwoJaW50IGNiT3V0OwoJYnl0ZSAqIHJnYk91dCA9IE5VTEw7CglpbnQgb3V0bCA9IDA7CglieXRlIHJnYklWWzE1XSA9IHsgMCB9OwoJY29uc3QgY25fY2JvciAqIHBJViA9IE5VTEw7Cgljb25zdCBFVlBfQ0lQSEVSICogY2lwaGVyOwojaWZkZWYgVVNFX0NCT1JfQ09OVEVYVAoJY25fY2Jvcl9jb250ZXh0ICogY29udGV4dCA9ICZwY29zZS0+bV9tZXNzYWdlLm1fYWxsb2NDb250ZXh0OwojZW5kaWYKCWludCBUU2l6ZSA9IDEyOCAvIDg7CgoJRVZQX0NJUEhFUl9DVFhfaW5pdCgmY3R4KTsKCgkvLyAgU2V0dXAgdGhlIElWL05vbmNlIGFuZCBwdXQgaXQgaW50byB0aGUgbWVzc2FnZQoKCXBJViA9IF9DT1NFX21hcF9nZXRfaW50KCZwY29zZS0+bV9tZXNzYWdlLCBDT1NFX0hlYWRlcl9JViwgQ09TRV9CT1RILCBOVUxMKTsKCWlmICgocElWID09IE5VTEwpIHx8IChwSVYtPnR5cGUgIT0gQ05fQ0JPUl9CWVRFUykpIHsKCQlpZiAocGVyciAhPSBOVUxMKSBwZXJyLT5lcnIgPSBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUjsKCgllcnJvclJldHVybjoKCQlpZiAocmdiT3V0ICE9IE5VTEwpIENPU0VfRlJFRShyZ2JPdXQsIGNvbnRleHQpOwoJCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CgkJcmV0dXJuIGZhbHNlOwoJfQoKCUNIRUNLX0NPTkRJVElPTihwSVYtPmxlbmd0aCA9PSA5Ni84LCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgltZW1jcHkocmdiSVYsIHBJVi0+di5zdHIsIHBJVi0+bGVuZ3RoKTsKCgkvLyAgU2V0dXAgYW5kIHJ1biB0aGUgT3BlblNTTCBjb2RlCgoJc3dpdGNoIChjYktleSkgewoJY2FzZSAxMjggLyA4OgoJCWNpcGhlciA9IEVWUF9hZXNfMTI4X2djbSgpOwoJCWJyZWFrOwoKCWNhc2UgMTkyIC8gODoKCQljaXBoZXIgPSBFVlBfYWVzXzE5Ml9nY20oKTsKCQlicmVhazsKCgljYXNlIDI1NiAvIDg6CgkJY2lwaGVyID0gRVZQX2Flc18yNTZfZ2NtKCk7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgkJYnJlYWs7Cgl9CgoJLy8gIERvIHRoZSBzZXR1cCBmb3IgT3BlblNTTAoKCUNIRUNLX0NPTkRJVElPTihFVlBfRGVjcnlwdEluaXRfZXgoJmN0eCwgY2lwaGVyLCBOVUxMLCBOVUxMLCBOVUxMKSwgQ09TRV9FUlJfREVDUllQVF9GQUlMRUQpOwoKCUNIRUNLX0NPTkRJVElPTihFVlBfQ0lQSEVSX0NUWF9jdHJsKCZjdHgsIEVWUF9DVFJMX0NDTV9TRVRfVEFHLCBUU2l6ZSwgKHZvaWQgKikmcGJDcnlwdG9bY2JDcnlwdG8gLSBUU2l6ZV0pLCBDT1NFX0VSUl9ERUNSWVBUX0ZBSUxFRCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9EZWNyeXB0SW5pdCgmY3R4LCAwLCBwYktleSwgcmdiSVYpLCBDT1NFX0VSUl9ERUNSWVBUX0ZBSUxFRCk7CgkKCS8vICBQdXMgaW4gdGhlIEFBRAoKCUNIRUNLX0NPTkRJVElPTihFVlBfRGVjcnlwdFVwZGF0ZSgmY3R4LCBOVUxMLCAmb3V0bCwgcGJBdXRoRGF0YSwgKGludCkgY2JBdXRoRGF0YSksIENPU0VfRVJSX0RFQ1JZUFRfRkFJTEVEKTsKCgkvLyAgCgoJY2JPdXQgPSAoaW50KWNiQ3J5cHRvIC0gVFNpemU7CglyZ2JPdXQgPSAoYnl0ZSAqKUNPU0VfQ0FMTE9DKGNiT3V0LCAxLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihyZ2JPdXQgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgoJLy8gIFByb2Nlc3MgY29udGVudAoKCUNIRUNLX0NPTkRJVElPTihFVlBfRGVjcnlwdFVwZGF0ZSgmY3R4LCByZ2JPdXQsICZjYk91dCwgcGJDcnlwdG8sIChpbnQpY2JDcnlwdG8gLSBUU2l6ZSksIENPU0VfRVJSX0RFQ1JZUFRfRkFJTEVEKTsKCgkvLyAgUHJvY2VzcyBUYWcKCglDSEVDS19DT05ESVRJT04oRVZQX0NJUEhFUl9DVFhfY3RybCgmY3R4LCBFVlBfQ1RSTF9HQ01fU0VUX1RBRywgVFNpemUsIChieXRlICopcGJDcnlwdG8gKyBjYkNyeXB0byAtIFRTaXplKSwgQ09TRV9FUlJfREVDUllQVF9GQUlMRUQpOwoKCS8vICBDaGVjayB0aGUgcmVzdWx0CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9EZWNyeXB0RmluYWwoJmN0eCwgcmdiT3V0ICsgY2JPdXQsICZjYk91dCksIENPU0VfRVJSX0RFQ1JZUFRfRkFJTEVEKTsKCglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoKCXBjb3NlLT5wYkNvbnRlbnQgPSByZ2JPdXQ7CglwY29zZS0+Y2JDb250ZW50ID0gY2JPdXQ7CgoJcmV0dXJuIHRydWU7Cn0KCmJvb2wgQUVTX0dDTV9FbmNyeXB0KENPU0VfRW52ZWxvcGVkICogcGNvc2UsIGNvbnN0IGJ5dGUgKiBwYktleSwgc2l6ZV90IGNiS2V5LCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgc2l6ZV90IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCUVWUF9DSVBIRVJfQ1RYIGN0eDsKCWludCBjYk91dDsKCWJ5dGUgKiByZ2JPdXQgPSBOVUxMOwoJaW50IG91dGwgPSAwOwoJYnl0ZSByZ2JJVlsxNl0gPSB7IDAgfTsKCWJ5dGUgKiBwYklWID0gTlVMTDsKCWNvbnN0IGNuX2Nib3IgKiBjYm9yX2l2ID0gTlVMTDsKCWNuX2Nib3IgKiBjYm9yX2l2X3QgPSBOVUxMOwoJY29uc3QgRVZQX0NJUEhFUiAqIGNpcGhlcjsKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCgkvLyBNYWtlIGl0IGZpcnN0IHNvIHdlIGNhbiBjbGVhbiBpdCB1cAoJRVZQX0NJUEhFUl9DVFhfaW5pdCgmY3R4KTsKCgkvLyAgU2V0dXAgdGhlIElWL05vbmNlIGFuZCBwdXQgaXQgaW50byB0aGUgbWVzc2FnZQoKCWNib3JfaXYgPSBfQ09TRV9tYXBfZ2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgQ09TRV9IZWFkZXJfSVYsIENPU0VfQk9USCwgcGVycik7CglpZiAoY2Jvcl9pdiA9PSBOVUxMKSB7CgkJcGJJViA9IENPU0VfQ0FMTE9DKDk2LCAxLCBjb250ZXh0KTsKCQlDSEVDS19DT05ESVRJT04ocGJJViAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCQlyYW5kX2J5dGVzKHBiSVYsIDk2IC8gOCk7CgkJbWVtY3B5KHJnYklWLCBwYklWLCA5NiAvIDgpOwoJCWNib3JfaXZfdCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJJViwgOTYgLyA4LCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgJmNib3JfZXJyb3IpOwoJCUNIRUNLX0NPTkRJVElPTl9DQk9SKGNib3JfaXZfdCAhPSBOVUxMLCBjYm9yX2Vycm9yKTsKCQlwYklWID0gTlVMTDsKCgkJaWYgKCFfQ09TRV9tYXBfcHV0KCZwY29zZS0+bV9tZXNzYWdlLCBDT1NFX0hlYWRlcl9JViwgY2Jvcl9pdl90LCBDT1NFX1VOUFJPVEVDVF9PTkxZLCBwZXJyKSkgZ290byBlcnJvclJldHVybjsKCQljYm9yX2l2X3QgPSBOVUxMOwoJfQoJZWxzZSB7CgkJQ0hFQ0tfQ09ORElUSU9OKGNib3JfaXYtPnR5cGUgPT0gQ05fQ0JPUl9CWVRFUywgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJCUNIRUNLX0NPTkRJVElPTihjYm9yX2l2LT5sZW5ndGggPT0gOTYgLyA4LCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgkJbWVtY3B5KHJnYklWLCBjYm9yX2l2LT52LnN0ciwgY2Jvcl9pdi0+bGVuZ3RoKTsKCX0KCgoJc3dpdGNoIChjYktleSo4KSB7CgljYXNlIDEyODoKCQljaXBoZXIgPSBFVlBfYWVzXzEyOF9nY20oKTsKCQlicmVhazsKCgljYXNlIDE5MjoKCQljaXBoZXIgPSBFVlBfYWVzXzE5Ml9nY20oKTsKCQlicmVhazsKCgljYXNlIDI1NjoKCQljaXBoZXIgPSBFVlBfYWVzXzI1Nl9nY20oKTsKCQlicmVhazsKCglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCQlicmVhazsKCX0KCgkvLyAgU2V0dXAgYW5kIHJ1biB0aGUgT3BlblNTTCBjb2RlCgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0SW5pdF9leCgmY3R4LCBjaXBoZXIsIE5VTEwsIE5VTEwsIE5VTEwpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0SW5pdCgmY3R4LCAwLCBwYktleSwgcmdiSVYpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIE5VTEwsICZvdXRsLCBwYkF1dGhEYXRhLCAoaW50KSBjYkF1dGhEYXRhKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCXJnYk91dCA9IChieXRlICopQ09TRV9DQUxMT0MocGNvc2UtPmNiQ29udGVudCArIDEyOC84LCAxLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihyZ2JPdXQgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCBwY29zZS0+cGJDb250ZW50LCAoaW50KXBjb3NlLT5jYkNvbnRlbnQpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0RmluYWxfZXgoJmN0eCwgJnJnYk91dFtjYk91dF0sICZjYk91dCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglDSEVDS19DT05ESVRJT04oRVZQX0NJUEhFUl9DVFhfY3RybCgmY3R4LCBFVlBfQ1RSTF9HQ01fR0VUX1RBRywgMTI4LzgsICZyZ2JPdXRbcGNvc2UtPmNiQ29udGVudF0pLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJY25fY2JvciAqIGNuVG1wID0gY25fY2Jvcl9kYXRhX2NyZWF0ZShyZ2JPdXQsIChpbnQpcGNvc2UtPmNiQ29udGVudCArIDEyOC84LCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCk7CglDSEVDS19DT05ESVRJT04oY25UbXAgIT0gTlVMTCwgQ09TRV9FUlJfQ0JPUik7CglyZ2JPdXQgPSBOVUxMOwoJQ0hFQ0tfQ09ORElUSU9OKF9DT1NFX2FycmF5X3JlcGxhY2UoJnBjb3NlLT5tX21lc3NhZ2UsIGNuVG1wLCBJTkRFWF9CT0RZLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCksIENPU0VfRVJSX0NCT1IpOwoKCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gdHJ1ZTsKCmVycm9yUmV0dXJuOgoJaWYgKHBiSVYgIT0gTlVMTCkgQ09TRV9GUkVFKHBiSVYsIGNvbnRleHQpOwoJaWYgKGNib3JfaXZfdCAhPSBOVUxMKSBDT1NFX0ZSRUUoY2Jvcl9pdl90LCBjb250ZXh0KTsKCWlmIChyZ2JPdXQgIT0gTlVMTCkgQ09TRV9GUkVFKHJnYk91dCwgY29udGV4dCk7CglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJcmV0dXJuIGZhbHNlOwp9CgoKYm9vbCBBRVNfQ0JDX01BQ19DcmVhdGUoQ09TRV9NYWNNZXNzYWdlICogcGNvc2UsIGludCBUU2l6ZSwgY29uc3QgYnl0ZSAqIHBiS2V5LCBzaXplX3QgY2JLZXksIGNvbnN0IGJ5dGUgKiBwYkF1dGhEYXRhLCBzaXplX3QgY2JBdXRoRGF0YSwgY29zZV9lcnJiYWNrICogcGVycikKewoJY29uc3QgRVZQX0NJUEhFUiAqIHBjaXBoZXIgPSBOVUxMOwoJRVZQX0NJUEhFUl9DVFggY3R4OwoJaW50IGNiT3V0OwoJYnl0ZSByZ2JJVlsxNl0gPSB7IDAgfTsKCWJ5dGUgKiByZ2JPdXQgPSBOVUxMOwoJYm9vbCBmID0gZmFsc2U7Cgl1bnNpZ25lZCBpbnQgaTsKCWNuX2Nib3IgKiBjbiA9IE5VTEw7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoKCUVWUF9DSVBIRVJfQ1RYX2luaXQoJmN0eCk7CgoJcmdiT3V0ID0gQ09TRV9DQUxMT0MoMTYsIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHJnYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCglzd2l0Y2ggKGNiS2V5KjgpIHsKCWNhc2UgMTI4OgoJCXBjaXBoZXIgPSBFVlBfYWVzXzEyOF9jYmMoKTsKCQlicmVhazsKCgljYXNlIDI1NjoKCQlwY2lwaGVyID0gRVZQX2Flc18yNTZfY2JjKCk7CgkJYnJlYWs7CgoJZGVmYXVsdDoKCQlGQUlMX0NPTkRJVElPTihDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7Cgl9CgoJLy8gIFNldHVwIGFuZCBydW4gdGhlIE9wZW5TU0wgY29kZQoKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdEluaXRfZXgoJmN0eCwgcGNpcGhlciwgTlVMTCwgcGJLZXksIHJnYklWKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCWZvciAoaSA9IDA7IGkgPCAodW5zaWduZWQgaW50KWNiQXV0aERhdGEgLyAxNjsgaSsrKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCBwYkF1dGhEYXRhICsgKGkgKiAxNiksIDE2KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoJaWYgKGNiQXV0aERhdGEgJSAxNiAhPSAwKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCBwYkF1dGhEYXRhICsgKGkgKiAxNiksIGNiQXV0aERhdGEgJSAxNiksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQlDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRVcGRhdGUoJmN0eCwgcmdiT3V0LCAmY2JPdXQsIHJnYklWLCAxNiAtIChjYkF1dGhEYXRhICUgMTYpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoKCWNuID0gY25fY2Jvcl9kYXRhX2NyZWF0ZShyZ2JPdXQsIFRTaXplIC8gOCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpOwoJQ0hFQ0tfQ09ORElUSU9OKGNuICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoJcmdiT3V0ID0gTlVMTDsKCglDSEVDS19DT05ESVRJT04oX0NPU0VfYXJyYXlfcmVwbGFjZSgmcGNvc2UtPm1fbWVzc2FnZSwgY24sIElOREVYX01BQ19UQUcsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSBOVUxMKSwgQ09TRV9FUlJfQ0JPUik7CgljbiA9IE5VTEw7CgoJRVZQX0NJUEhFUl9DVFhfY2xlYW51cCgmY3R4KTsKCXJldHVybiAhZjsKCmVycm9yUmV0dXJuOgoJaWYgKHJnYk91dCAhPSBOVUxMKSBDT1NFX0ZSRUUocmdiT3V0LCBjb250ZXh0KTsKCWlmIChjbiAhPSBOVUxMKSBDTl9DQk9SX0ZSRUUoY24sIGNvbnRleHQpOwoJRVZQX0NJUEhFUl9DVFhfY2xlYW51cCgmY3R4KTsKCXJldHVybiBmYWxzZTsKfQoKYm9vbCBBRVNfQ0JDX01BQ19WYWxpZGF0ZShDT1NFX01hY01lc3NhZ2UgKiBwY29zZSwgaW50IFRTaXplLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgY29uc3QgYnl0ZSAqIHBiQXV0aERhdGEsIHNpemVfdCBjYkF1dGhEYXRhLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7Cgljb25zdCBFVlBfQ0lQSEVSICogcGNpcGhlciA9IE5VTEw7CglFVlBfQ0lQSEVSX0NUWCBjdHg7CglpbnQgY2JPdXQ7CglieXRlIHJnYklWWzE2XSA9IHsgMCB9OwoJYnl0ZSByZ2JUYWdbMTZdID0geyAwIH07Cglib29sIGYgPSBmYWxzZTsKCXVuc2lnbmVkIGludCBpOwoKCXN3aXRjaCAoY2JLZXkqOCkgewoJY2FzZSAxMjg6CgkJcGNpcGhlciA9IEVWUF9hZXNfMTI4X2NiYygpOwoJCWJyZWFrOwoKCWNhc2UgMjU2OgoJCXBjaXBoZXIgPSBFVlBfYWVzXzI1Nl9jYmMoKTsKCQlicmVhazsKCglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCgkvLyAgU2V0dXAgYW5kIHJ1biB0aGUgT3BlblNTTCBjb2RlCgoJRVZQX0NJUEhFUl9DVFhfaW5pdCgmY3R4KTsKCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdEluaXRfZXgoJmN0eCwgcGNpcGhlciwgTlVMTCwgcGJLZXksIHJnYklWKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCVRTaXplIC89IDg7CgoJZm9yIChpID0gMDsgaSA8ICh1bnNpZ25lZCBpbnQpIGNiQXV0aERhdGEgLyAxNjsgaSsrKSB7CgkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYlRhZywgJmNiT3V0LCBwYkF1dGhEYXRhKyhpKjE2KSwgMTYpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7Cgl9CglpZiAoY2JBdXRoRGF0YSAlIDE2ICE9IDApIHsKCQlDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRVcGRhdGUoJmN0eCwgcmdiVGFnLCAmY2JPdXQsIHBiQXV0aERhdGEgKyAoaSAqIDE2KSwgY2JBdXRoRGF0YSAlIDE2KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdFVwZGF0ZSgmY3R4LCByZ2JUYWcsICZjYk91dCwgcmdiSVYsIDE2IC0gKGNiQXV0aERhdGEgJSAxNikpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7Cgl9CgoJY25fY2JvciAqIGNuID0gX0NPU0VfYXJyYXlnZXRfaW50KCZwY29zZS0+bV9tZXNzYWdlLCBJTkRFWF9NQUNfVEFHKTsKCUNIRUNLX0NPTkRJVElPTihjbiAhPSBOVUxMLCBDT1NFX0VSUl9DQk9SKTsKCglmb3IgKGkgPSAwOyBpIDwgKHVuc2lnbmVkIGludClUU2l6ZTsgaSsrKSBmIHw9IChjbi0+di5ieXRlc1tpXSAhPSByZ2JUYWdbaV0pOwoKCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gIWY7CgplcnJvclJldHVybjoKCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gZmFsc2U7Cn0KCiNpZiAwCi8vICBXZSBhcmUgZG9pbmcgQ0JDLU1BQyBub3QgQ01BQyBhdCB0aGlzIHRpbWUKYm9vbCBBRVNfQ01BQ19WYWxpZGF0ZShDT1NFX01hY01lc3NhZ2UgKiBwY29zZSwgaW50IEtleVNpemUsIGludCBUYWdTaXplLCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgaW50IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCUNNQUNfQ1RYICogcGN0eCA9IE5VTEw7Cgljb25zdCBFVlBfQ0lQSEVSICogcGNpcGhlciA9IE5VTEw7CglieXRlICogcmdiT3V0ID0gTlVMTDsKCXNpemVfdCBjYk91dDsKCWJvb2wgZiA9IGZhbHNlOwoJdW5zaWduZWQgaW50IGk7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX21lc3NhZ2UubV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoKCXBjdHggPSBDTUFDX0NUWF9uZXcoKTsKCgoJc3dpdGNoIChLZXlTaXplKSB7CgljYXNlIDEyODogcGNpcGhlciA9IEVWUF9hZXNfMTI4X2NiYygpOyBicmVhazsKCWNhc2UgMjU2OiBwY2lwaGVyID0gRVZQX2Flc18yNTZfY2JjKCk7IGJyZWFrOwoJZGVmYXVsdDogRkFJTF9DT05ESVRJT04oQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOyBicmVhazsKCX0KCglyZ2JPdXQgPSBDT1NFX0NBTExPQygxMjgvOCwgMSwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocmdiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCUNIRUNLX0NPTkRJVElPTihDTUFDX0luaXQocGN0eCwgcGNvc2UtPnBiS2V5LCBwY29zZS0+Y2JLZXksIHBjaXBoZXIsIE5VTEwgKSA9PSAxLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CglDSEVDS19DT05ESVRJT04oQ01BQ19VcGRhdGUocGN0eCwgcGJBdXRoRGF0YSwgY2JBdXRoRGF0YSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCUNIRUNLX0NPTkRJVElPTihDTUFDX0ZpbmFsKHBjdHgsIHJnYk91dCwgJmNiT3V0KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCWNuX2Nib3IgKiBjbiA9IF9DT1NFX2FycmF5Z2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgSU5ERVhfTUFDX1RBRyk7CglDSEVDS19DT05ESVRJT04oY24gIT0gTlVMTCwgQ09TRV9FUlJfQ0JPUik7CgoJZm9yIChpID0gMDsgaSA8ICh1bnNpZ25lZCBpbnQpVGFnU2l6ZSAvIDg7IGkrKykgZiB8PSAoY24tPnYuYnl0ZXNbaV0gIT0gcmdiT3V0W2ldKTsKCglDT1NFX0ZSRUUocmdiT3V0LCBjb250ZXh0KTsKCUNNQUNfQ1RYX2NsZWFudXAocGN0eCk7CglDTUFDX0NUWF9mcmVlKHBjdHgpOwoJcmV0dXJuICFmOwoKZXJyb3JSZXR1cm46CglDT1NFX0ZSRUUocmdiT3V0LCBjb250ZXh0KTsKCUNNQUNfQ1RYX2NsZWFudXAocGN0eCk7CglDTUFDX0NUWF9mcmVlKHBjdHgpOwoJcmV0dXJuIGZhbHNlOwoKfQojZW5kaWYKCmJvb2wgSEtERl9BRVNfRXhwYW5kKENPU0UgKiBwY29zZSwgc2l6ZV90IGNiaXRLZXksIGNvbnN0IGJ5dGUgKiBwYlBSSywgc2l6ZV90IGNiUFJLLCBjb25zdCBieXRlICogcGJJbmZvLCBzaXplX3QgY2JJbmZvLCBieXRlICogcGJPdXRwdXQsIHNpemVfdCBjYk91dHB1dCwgY29zZV9lcnJiYWNrICogcGVycikKewoJY29uc3QgRVZQX0NJUEhFUiAqIHBjaXBoZXIgPSBOVUxMOwoJRVZQX0NJUEhFUl9DVFggY3R4OwoJaW50IGNiT3V0OwoJYnl0ZSByZ2JJVlsxNl0gPSB7IDAgfTsKCWJ5dGUgYkNvdW50ID0gMTsKCXNpemVfdCBpYjsKCWJ5dGUgcmdiRGlnZXN0WzEyOCAvIDhdOwoJaW50IGNiRGlnZXN0ID0gMDsKCWJ5dGUgcmdiT3V0WzE2XTsKCglFVlBfQ0lQSEVSX0NUWF9pbml0KCZjdHgpOwoKCXN3aXRjaCAoY2JpdEtleSkgewoJY2FzZSAxMjg6CgkJcGNpcGhlciA9IEVWUF9hZXNfMTI4X2NiYygpOwoJCWJyZWFrOwoKCWNhc2UgMjU2OgoJCXBjaXBoZXIgPSBFVlBfYWVzXzI1Nl9jYmMoKTsKCQlicmVhazsKCglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCUNIRUNLX0NPTkRJVElPTihjYlBSSyA9PSBjYml0S2V5IC8gOCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCS8vICBTZXR1cCBhbmQgcnVuIHRoZSBPcGVuU1NMIGNvZGUKCgoJZm9yIChpYiA9IDA7IGliIDwgY2JPdXRwdXQ7IGliICs9IDE2LCBiQ291bnQgKz0gMSkgewoJCXNpemVfdCBpYjI7CgoJCUNIRUNLX0NPTkRJVElPTihFVlBfRW5jcnlwdEluaXRfZXgoJmN0eCwgcGNpcGhlciwgTlVMTCwgcGJQUkssIHJnYklWKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCQlDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRVcGRhdGUoJmN0eCwgcmdiT3V0LCAmY2JPdXQsIHJnYkRpZ2VzdCwgY2JEaWdlc3QpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJZm9yIChpYjIgPSAwOyBpYjIgPCBjYkluZm87IGliMis9MTYpIHsKCQkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCBwYkluZm8raWIyLCAoaW50KSBNSU4oMTYsIGNiSW5mby1pYjIpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCX0KCQlDSEVDS19DT05ESVRJT04oRVZQX0VuY3J5cHRVcGRhdGUoJmN0eCwgcmdiT3V0LCAmY2JPdXQsICZiQ291bnQsIDEpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJaWYgKChjYkluZm8gKyAxKSAlIDE2ICE9IDApIHsKCQkJQ0hFQ0tfQ09ORElUSU9OKEVWUF9FbmNyeXB0VXBkYXRlKCZjdHgsIHJnYk91dCwgJmNiT3V0LCByZ2JJViwgKGludCkgMTYtKGNiSW5mbysxKSUxNiksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCQl9CgkJbWVtY3B5KHJnYkRpZ2VzdCwgcmdiT3V0LCBjYk91dCk7CgkJY2JEaWdlc3QgPSBjYk91dDsKCQltZW1jcHkocGJPdXRwdXQgKyBpYiwgcmdiRGlnZXN0LCBNSU4oMTYsIGNiT3V0cHV0IC0gaWIpKTsKCX0KCglFVlBfQ0lQSEVSX0NUWF9jbGVhbnVwKCZjdHgpOwoJcmV0dXJuIHRydWU7CgplcnJvclJldHVybjoKCUVWUF9DSVBIRVJfQ1RYX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gZmFsc2U7Cn0KCgpib29sIEhLREZfRXh0cmFjdChDT1NFICogcGNvc2UsIGNvbnN0IGJ5dGUgKiBwYktleSwgc2l6ZV90IGNiS2V5LCBzaXplX3QgY2JpdERpZ2VzdCwgYnl0ZSAqIHJnYkRpZ2VzdCwgc2l6ZV90ICogcGNiRGlnZXN0LCBDQk9SX0NPTlRFWFRfQ09NTUEgY29zZV9lcnJiYWNrICogcGVycikKewoJYnl0ZSByZ2JTYWx0W0VWUF9NQVhfTURfU0laRV0gPSB7IDAgfTsKCWludCBjYlNhbHQ7Cgljbl9jYm9yICogY25TYWx0OwoJSE1BQ19DVFggY3R4OwoJY29uc3QgRVZQX01EICogcG1kID0gTlVMTDsKCXVuc2lnbmVkIGludCBjYkRpZ2VzdDsKCglITUFDX0NUWF9pbml0KCZjdHgpOwoKCWlmICgwKSB7CgllcnJvclJldHVybjoKCQlITUFDX2NsZWFudXAoJmN0eCk7CgkJcmV0dXJuIGZhbHNlOwoJfQoKCXN3aXRjaCAoY2JpdERpZ2VzdCkgewoJY2FzZSAyNTY6IHBtZCA9IEVWUF9zaGEyNTYoKTsgY2JTYWx0ID0gMjU2IC8gODsgIGJyZWFrOwoJY2FzZSAzODQ6IHBtZCA9IEVWUF9zaGEzODQoKTsgY2JTYWx0ID0gMzg0IC8gODsgYnJlYWs7CgljYXNlIDUxMjogcG1kID0gRVZQX3NoYTUxMigpOyBjYlNhbHQgPSA1MTIgLyA4OyBicmVhazsKCWRlZmF1bHQ6IEZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsgYnJlYWs7Cgl9CgoJY25TYWx0ID0gX0NPU0VfbWFwX2dldF9pbnQocGNvc2UsIENPU0VfSGVhZGVyX0hLREZfc2FsdCwgQ09TRV9CT1RILCBwZXJyKTsKCglpZiAoY25TYWx0ICE9IE5VTEwpIHsKCQlDSEVDS19DT05ESVRJT04oSE1BQ19Jbml0KCZjdHgsIGNuU2FsdC0+di5ieXRlcywgKGludCkgY25TYWx0LT5sZW5ndGgsIHBtZCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCX0KCWVsc2UgewoJCUNIRUNLX0NPTkRJVElPTihITUFDX0luaXQoJmN0eCwgcmdiU2FsdCwgY2JTYWx0LCBwbWQpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7Cgl9CglDSEVDS19DT05ESVRJT04oSE1BQ19VcGRhdGUoJmN0eCwgcGJLZXksIChpbnQpY2JLZXkpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CglDSEVDS19DT05ESVRJT04oSE1BQ19GaW5hbCgmY3R4LCByZ2JEaWdlc3QsICZjYkRpZ2VzdCksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCSpwY2JEaWdlc3QgPSBjYkRpZ2VzdDsKCUhNQUNfY2xlYW51cCgmY3R4KTsKCXJldHVybiB0cnVlOwp9Cgpib29sIEhLREZfRXhwYW5kKENPU0UgKiBwY29zZSwgc2l6ZV90IGNiaXREaWdlc3QsIGNvbnN0IGJ5dGUgKiBwYlBSSywgc2l6ZV90IGNiUFJLLCBjb25zdCBieXRlICogcGJJbmZvLCBzaXplX3QgY2JJbmZvLCBieXRlICogcGJPdXRwdXQsIHNpemVfdCBjYk91dHB1dCwgY29zZV9lcnJiYWNrICogcGVycikKewoJSE1BQ19DVFggY3R4OwoJY29uc3QgRVZQX01EICogcG1kID0gTlVMTDsKCXNpemVfdCBpYjsKCWludCBjYlNhbHQ7Cgl1bnNpZ25lZCBpbnQgY2JEaWdlc3QgPSAwOwoJYnl0ZSByZ2JEaWdlc3RbRVZQX01BWF9NRF9TSVpFXTsKCWJ5dGUgYkNvdW50ID0gMTsKCglITUFDX0NUWF9pbml0KCZjdHgpOwoKCWlmICgwKSB7CgllcnJvclJldHVybjoKCQlITUFDX2NsZWFudXAoJmN0eCk7CgkJcmV0dXJuIGZhbHNlOwoJfQoKCXN3aXRjaCAoY2JpdERpZ2VzdCkgewoJY2FzZSAyNTY6IHBtZCA9IEVWUF9zaGEyNTYoKTsgY2JTYWx0ID0gMjU2IC8gODsgIGJyZWFrOwoJY2FzZSAzODQ6IHBtZCA9IEVWUF9zaGEzODQoKTsgY2JTYWx0ID0gMzg0IC8gODsgYnJlYWs7CgljYXNlIDUxMjogcG1kID0gRVZQX3NoYTUxMigpOyBjYlNhbHQgPSA1MTIgLyA4OyBicmVhazsKCWRlZmF1bHQ6IEZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsgYnJlYWs7Cgl9CgoKCWZvciAoaWIgPSAwOyBpYiA8IGNiT3V0cHV0OyBpYiArPSBjYkRpZ2VzdCwgYkNvdW50ICs9IDEpIHsKCQlDSEVDS19DT05ESVRJT04oSE1BQ19Jbml0X2V4KCZjdHgsIHBiUFJLLCAoaW50KWNiUFJLLCBwbWQsIE5VTEwpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJQ0hFQ0tfQ09ORElUSU9OKEhNQUNfVXBkYXRlKCZjdHgsIHJnYkRpZ2VzdCwgY2JEaWdlc3QpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJQ0hFQ0tfQ09ORElUSU9OKEhNQUNfVXBkYXRlKCZjdHgsIHBiSW5mbywgY2JJbmZvKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCUNIRUNLX0NPTkRJVElPTihITUFDX1VwZGF0ZSgmY3R4LCAmYkNvdW50LCAxKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCUNIRUNLX0NPTkRJVElPTihITUFDX0ZpbmFsKCZjdHgsIHJnYkRpZ2VzdCwgJmNiRGlnZXN0KSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCQltZW1jcHkocGJPdXRwdXQgKyBpYiwgcmdiRGlnZXN0LCBNSU4oY2JEaWdlc3QsIGNiT3V0cHV0IC0gaWIpKTsKCX0KCglITUFDX2NsZWFudXAoJmN0eCk7CglyZXR1cm4gdHJ1ZTsKCn0KCnZvaWQgZHVtcF9vdXRwdXQoYnl0ZSogYiwgc2l6ZV90IHMpewoJZm9yKGludCBpID0gMDsgaSA8IHM7IGkrKyl7CgkJcHJpbnRmKCIlMDJ4IiwgKmIpOwoJCWIrKzsKCX0KCXByaW50ZigiXG4iKTsKfQoKdm9pZCBkaWZmKHVuc2lnbmVkIGNoYXIqIGEsIHNpemVfdCBhX2wsIHVuc2lnbmVkIGNoYXIqIGIsIHNpemVfdCBiX2wpewoJc2l6ZV90IHM7CglzID0gKGFfbCA8IGJfbCkgPyBhX2wgOiBiX2w7Cgl1bnNpZ25lZCBjaGFyKiB0bXAgPSBhOwoJcHJpbnRmKCJzaXplID0gJWRcbiIscyApOwovLwlwcmludGYoIiUwMnhcbiIsICp0bXApOwoJaW50IGk7Cglmb3IoaSA9IDA7IGkgPCBzOyArK2kpewoJCXByaW50ZigiJTAyeCIsICp0bXApOwoJCXRtcCsrOwoJfQoJcHJpbnRmKCJcbiIpOwoKCXRtcCA9IGI7Cglmb3IoaSA9IDA7IGkgPCBzOyArK2kpewoJCXByaW50ZigiJTAyeCIsICp0bXApOwoJCXRtcCsrOwoJfQoJcHJpbnRmKCJcbiIpOwoKCWZvcihpID0gMDsgaSA8IHM7ICsraSl7CgoJCWlmKCphICE9ICpiKXsKCQkJcHJpbnRmKCJeXiIpOwoKCQl9IGVsc2UgewoJCQlwcmludGYoIl9fIik7CgoJCX0KCgkJYSsrOwoJCWIrKzsKCX0KCXByaW50ZigiXG4iKTsKfQoqLwoKYm9vbCBITUFDX0NyZWF0ZShDT1NFX01hY01lc3NhZ2UgKiBwY29zZSwgaW50IEhTaXplLCBpbnQgVFNpemUsIGNvbnN0IGJ5dGUgKiBwYktleSwgc2l6ZV90IGNiS2V5LCBjb25zdCBieXRlICogcGJBdXRoRGF0YSwgc2l6ZV90IGNiQXV0aERhdGEsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWJ5dGUgKiByZ2JPdXQgPSBOVUxMOwovLwl1bnNpZ25lZCBpbnQgY2JPdXQ7CgltYmVkdGxzX21kX2NvbnRleHRfdCBjb250eDsKCWNvbnN0IGNoYXIqIG1kX25hbWU7Cgljb25zdCBzdHJ1Y3QgbWJlZHRsc19tZF9pbmZvX3QgKiBpbmZvOwoKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgoJc3dpdGNoIChIU2l6ZSkgewoJCWNhc2UgMjU2OiBtZF9uYW1lID0gIlNIQTI1NiI7IGJyZWFrOwoJCWNhc2UgMzg0OiBtZF9uYW1lID0gIlNIQTM4NCI7IGJyZWFrOwoJCWNhc2UgNTEyOiBtZF9uYW1lID0gIlNIQTUxMiI7IGJyZWFrOwoJCWRlZmF1bHQ6IEZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsgYnJlYWs7Cgl9CgoJaWYgKDApIHsKCWVycm9yUmV0dXJuOgoJCUNPU0VfRlJFRShyZ2JPdXQsIGNvbnRleHQpOwoJCW1iZWR0bHNfbWRfZnJlZSgmY29udHgpOyAKCQlyZXR1cm4gZmFsc2U7Cgl9CgoJbWJlZHRsc19tZF9pbml0KCZjb250eCk7CglpbmZvID0gbWJlZHRsc19tZF9pbmZvX2Zyb21fc3RyaW5nIChtZF9uYW1lKTsKCW1iZWR0bHNfbWRfc2V0dXAoICZjb250eCwgaW5mbywgMSApOwoKCXJnYk91dCA9IENPU0VfQ0FMTE9DKG1iZWR0bHNfbWRfZ2V0X3NpemUoaW5mbyksIDEsIGNvbnRleHQpOwoJQ0hFQ0tfQ09ORElUSU9OKHJnYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCglDSEVDS19DT05ESVRJT04oIShtYmVkdGxzX21kX2htYWNfc3RhcnRzICgmY29udHgsIChjaGFyKilwYktleSwgY2JLZXkpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJQ0hFQ0tfQ09ORElUSU9OKCEobWJlZHRsc19tZF9obWFjX3VwZGF0ZSAoJmNvbnR4LCBwYkF1dGhEYXRhLCBjYkF1dGhEYXRhKSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCUNIRUNLX0NPTkRJVElPTighKG1iZWR0bHNfbWRfaG1hY19maW5pc2ggKCZjb250eCwgcmdiT3V0KSksIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglDSEVDS19DT05ESVRJT04oX0NPU0VfYXJyYXlfcmVwbGFjZSgmcGNvc2UtPm1fbWVzc2FnZSwgY25fY2Jvcl9kYXRhX2NyZWF0ZShyZ2JPdXQsIFRTaXplIC8gOCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBJTkRFWF9NQUNfVEFHLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgTlVMTCksIENPU0VfRVJSX0NCT1IpOwoKCW1iZWR0bHNfbWRfZnJlZSgmY29udHgpOyAKCXJldHVybiB0cnVlOwp9Cgpib29sIEhNQUNfVmFsaWRhdGUoQ09TRV9NYWNNZXNzYWdlICogcGNvc2UsIGludCBIU2l6ZSwgaW50IFRTaXplLCBjb25zdCBieXRlICogcGJLZXksIHNpemVfdCBjYktleSwgY29uc3QgYnl0ZSAqIHBiQXV0aERhdGEsIHNpemVfdCBjYkF1dGhEYXRhLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CgltYmVkdGxzX21kX2NvbnRleHRfdCBjb250eDsKCWNvbnN0IGNoYXIqIG1kX25hbWU7Cgljb25zdCBzdHJ1Y3QgbWJlZHRsc19tZF9pbmZvX3QgKiBpbmZvOwoJYnl0ZSAqIHJnYk91dCA9IE5VTEw7Cgl1bnNpZ25lZCBpbnQgY2JPdXQ7Cglib29sIGYgPSBmYWxzZTsKCXVuc2lnbmVkIGludCBpOwoKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcGNvc2UtPm1fbWVzc2FnZS5tX2FsbG9jQ29udGV4dDsKI2VuZGlmCgoJc3dpdGNoIChIU2l6ZSkgewoJCWNhc2UgMjU2OiBtZF9uYW1lID0gIlNIQTI1NiI7IGJyZWFrOwoJCWNhc2UgMzg0OiBtZF9uYW1lID0gIlNIQTM4NCI7IGJyZWFrOwoJCWNhc2UgNTEyOiBtZF9uYW1lID0gIlNIQTUxMiI7IGJyZWFrOwoJCWRlZmF1bHQ6IEZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsgYnJlYWs7Cgl9CgoJbWJlZHRsc19tZF9pbml0KCZjb250eCk7CglpbmZvID0gbWJlZHRsc19tZF9pbmZvX2Zyb21fc3RyaW5nIChtZF9uYW1lKTsKCW1iZWR0bHNfbWRfc2V0dXAoICZjb250eCwgaW5mbywgMSApOwoJCgljYk91dCA9IG1iZWR0bHNfbWRfZ2V0X3NpemUoaW5mbyk7CglyZ2JPdXQgPSBDT1NFX0NBTExPQyhjYk91dCwgMSwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocmdiT3V0ICE9IE5VTEwsIENPU0VfRVJSX09VVF9PRl9NRU1PUlkpOwoKCUNIRUNLX0NPTkRJVElPTighKG1iZWR0bHNfbWRfaG1hY19zdGFydHMgKCZjb250eCwgKGNoYXIqKXBiS2V5LCBjYktleSkpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CglDSEVDS19DT05ESVRJT04oIShtYmVkdGxzX21kX2htYWNfdXBkYXRlICgmY29udHgsIHBiQXV0aERhdGEsIGNiQXV0aERhdGEpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJQ0hFQ0tfQ09ORElUSU9OKCEobWJlZHRsc19tZF9obWFjX2ZpbmlzaCAoJmNvbnR4LCByZ2JPdXQpKSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCWNuX2Nib3IgKiBjbiA9IF9DT1NFX2FycmF5Z2V0X2ludCgmcGNvc2UtPm1fbWVzc2FnZSwgSU5ERVhfTUFDX1RBRyk7CglDSEVDS19DT05ESVRJT04oY24gIT0gTlVMTCwgQ09TRV9FUlJfQ0JPUik7CgoJaWYgKGNuLT5sZW5ndGggPiAoaW50KSBjYk91dCkgcmV0dXJuIGZhbHNlOwoJZm9yIChpID0gMDsgaSA8ICh1bnNpZ25lZCBpbnQpIFRTaXplLzg7IGkrKykgZiB8PSAoY24tPnYuYnl0ZXNbaV0gIT0gcmdiT3V0W2ldKTsKCgltYmVkdGxzX21kX2ZyZWUoJmNvbnR4KTsgCglyZXR1cm4gIWY7CgplcnJvclJldHVybjoKCUNPU0VfRlJFRShyZ2JPdXQsIGNvbnRleHQpOwoJbWJlZHRsc19tZF9mcmVlKCZjb250eCk7IAoJcmV0dXJuIGZhbHNlOwp9Ci8qCgojZGVmaW5lIENPU0VfS2V5X0VDX0N1cnZlIC0xCiNkZWZpbmUgQ09TRV9LZXlfRUNfWCAtMgojZGVmaW5lIENPU0VfS2V5X0VDX1kgLTMKI2RlZmluZSBDT1NFX0tleV9FQ19kIC00CgpFQ19LRVkgKiBFQ0tleV9Gcm9tKGNvbnN0IGNuX2Nib3IgKiBwS2V5LCBpbnQgKiBjYkdyb3VwLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglFQ19LRVkgKiBwTmV3S2V5ID0gRUNfS0VZX25ldygpOwoJYnl0ZSAgcmdiS2V5WzUxMisxXTsKCWludCBjYktleTsKCWNvbnN0IGNuX2Nib3IgKiBwOwoJaW50IG5pZEdyb3VwID0gLTE7CglFQ19QT0lOVCAqIHBQb2ludCA9IE5VTEw7CgoJcCA9IGNuX2Nib3JfbWFwZ2V0X2ludChwS2V5LCBDT1NFX0tleV9FQ19DdXJ2ZSk7CglDSEVDS19DT05ESVRJT04ocCAhPSBOVUxMLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgoJc3dpdGNoIChwLT52LnNpbnQpIHsKCWNhc2UgMTogLy8gUC0yNTYKCQluaWRHcm91cCA9IE5JRF9YOV82Ml9wcmltZTI1NnYxOwoJCSpjYkdyb3VwID0gMjU2IC8gODsKCQlicmVhazsKCgljYXNlIDI6IC8vIFAtMzg0CgkJbmlkR3JvdXAgPSBOSURfc2VjcDM4NHIxOwoJCSpjYkdyb3VwID0gMzg0IC8gODsKCQlicmVhazsKCgljYXNlIDM6IC8vIFAtNTIxCgkJbmlkR3JvdXAgPSBOSURfc2VjcDUyMXIxOwoJCSpjYkdyb3VwID0gKDUyMSArIDcpIC8gODsKCQlicmVhazsKCglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCglFQ19HUk9VUCAqIGVjZ3JvdXAgPSBFQ19HUk9VUF9uZXdfYnlfY3VydmVfbmFtZShuaWRHcm91cCk7CglDSEVDS19DT05ESVRJT04oZWNncm91cCAhPSBOVUxMLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CglDSEVDS19DT05ESVRJT04oRUNfS0VZX3NldF9ncm91cChwTmV3S2V5LCBlY2dyb3VwKSA9PSAxLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJcCA9IGNuX2Nib3JfbWFwZ2V0X2ludChwS2V5LCBDT1NFX0tleV9FQ19YKTsKCUNIRUNLX0NPTkRJVElPTigocCAhPSBOVUxMKSAmJiAocC0+dHlwZSA9PSBDTl9DQk9SX0JZVEVTKSwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJQ0hFQ0tfQ09ORElUSU9OKHAtPmxlbmd0aCA9PSAqY2JHcm91cCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJbWVtY3B5KHJnYktleSsxLCBwLT52LnN0ciwgcC0+bGVuZ3RoKTsKCQoKCXAgPSBjbl9jYm9yX21hcGdldF9pbnQocEtleSwgQ09TRV9LZXlfRUNfWSk7CglDSEVDS19DT05ESVRJT04oKHAgIT0gTlVMTCksIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCWlmIChwLT50eXBlID09IENOX0NCT1JfQllURVMpIHsKCQlyZ2JLZXlbMF0gPSBQT0lOVF9DT05WRVJTSU9OX1VOQ09NUFJFU1NFRDsKCQljYktleSA9ICgqY2JHcm91cCAqIDIpICsgMTsKCQlDSEVDS19DT05ESVRJT04ocC0+bGVuZ3RoID09ICpjYkdyb3VwLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgkJbWVtY3B5KHJnYktleSArIHAtPmxlbmd0aCArIDEsIHAtPnYuc3RyLCBwLT5sZW5ndGgpOwoJfQoJZWxzZSBpZiAocC0+dHlwZSA9PSBDTl9DQk9SX1RSVUUpIHsKCQljYktleSA9ICgqY2JHcm91cCkgKyAxOwoJCXJnYktleVswXSA9IFBPSU5UX0NPTlZFUlNJT05fQ09NUFJFU1NFRCArIDE7Cgl9CgllbHNlIGlmIChwLT50eXBlID09IENOX0NCT1JfRkFMU0UpIHsKCQljYktleSA9ICgqY2JHcm91cCkgKyAxOwoJCXJnYktleVswXSA9IFBPSU5UX0NPTlZFUlNJT05fQ09NUFJFU1NFRDsKCX0KCWVsc2UgRkFJTF9DT05ESVRJT04oQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCXBQb2ludCA9IEVDX1BPSU5UX25ldyhlY2dyb3VwKTsKCUNIRUNLX0NPTkRJVElPTihwUG9pbnQgIT0gTlVMTCwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJQ0hFQ0tfQ09ORElUSU9OKEVDX1BPSU5UX29jdDJwb2ludChlY2dyb3VwLCBwUG9pbnQsIHJnYktleSwgY2JLZXksIE5VTEwpID09IDEsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCUNIRUNLX0NPTkRJVElPTihFQ19LRVlfc2V0X3B1YmxpY19rZXkocE5ld0tleSwgcFBvaW50KSA9PSAxLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJcCA9IGNuX2Nib3JfbWFwZ2V0X2ludChwS2V5LCBDT1NFX0tleV9FQ19kKTsKCWlmIChwICE9IE5VTEwpIHsKCQlCSUdOVU0gKiBwYm47CgoJCXBibiA9IEJOX2JpbjJibihwLT52LmJ5dGVzLCAoaW50KSBwLT5sZW5ndGgsIE5VTEwpOwoJCUNIRUNLX0NPTkRJVElPTihwYm4gIT0gTlVMTCwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCUNIRUNLX0NPTkRJVElPTihFQ19LRVlfc2V0X3ByaXZhdGVfa2V5KHBOZXdLZXksIHBibikgPT0gMSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJfQoJCglyZXR1cm4gcE5ld0tleTsKCmVycm9yUmV0dXJuOgoJaWYgKHBOZXdLZXkgIT0gTlVMTCkgRUNfS0VZX2ZyZWUocE5ld0tleSk7CglyZXR1cm4gTlVMTDsKfQoKY25fY2JvciAqIEVDX0Zyb21LZXkoY29uc3QgRUNfS0VZICogcEtleSwgQ0JPUl9DT05URVhUX0NPTU1BIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWNuX2Nib3IgKiBwa2V5ID0gTlVMTDsKCWNvbnN0IEVDX0dST1VQICogcGdyb3VwOwoJaW50IGNvc2VfZ3JvdXA7Cgljbl9jYm9yICogcCA9IE5VTEw7Cgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCWNvbnN0IEVDX1BPSU5UICogcFBvaW50OwoJc2l6ZV90IGNiU2l6ZTsKCWJ5dGUgKiBwYk91dCA9IE5VTEw7CgoJcGdyb3VwID0gRUNfS0VZX2dldDBfZ3JvdXAocEtleSk7CglDSEVDS19DT05ESVRJT04ocGdyb3VwICE9IE5VTEwsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCglzd2l0Y2ggKEVDX0dST1VQX2dldF9jdXJ2ZV9uYW1lKHBncm91cCkpIHsKCWNhc2UgTklEX1g5XzYyX3ByaW1lMjU2djE6CQljb3NlX2dyb3VwID0gMTsJCWJyZWFrOwoJY2FzZSBOSURfc2VjcDM4NHIxOiBjb3NlX2dyb3VwID0gMjsgYnJlYWs7CgljYXNlIE5JRF9zZWNwNTIxcjE6IGNvc2VfZ3JvdXAgPSAzOyBicmVhazsKCglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCglwa2V5ID0gY25fY2Jvcl9tYXBfY3JlYXRlKENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihwa2V5ICE9IE5VTEwsIGNib3JfZXJyb3IpOwoKCXAgPSBjbl9jYm9yX2ludF9jcmVhdGUoY29zZV9ncm91cCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BICZjYm9yX2Vycm9yKTsKCUNIRUNLX0NPTkRJVElPTl9DQk9SKHAgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihjbl9jYm9yX21hcHB1dF9pbnQocGtleSwgQ09TRV9LZXlfRUNfQ3VydmUsIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJcCA9IE5VTEw7CgoJcFBvaW50ID0gRUNfS0VZX2dldDBfcHVibGljX2tleShwS2V5KTsKCUNIRUNLX0NPTkRJVElPTihwUG9pbnQgIT0gTlVMTCwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoKCWlmIChGVXNlQ29tcHJlc3NlZCkgewoJCWNiU2l6ZSA9IEVDX1BPSU5UX3BvaW50Mm9jdChwZ3JvdXAsIHBQb2ludCwgUE9JTlRfQ09OVkVSU0lPTl9DT01QUkVTU0VELCBOVUxMLCAwLCBOVUxMKTsKCQlDSEVDS19DT05ESVRJT04oY2JTaXplID4gMCwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCXBiT3V0ID0gQ09TRV9DQUxMT0MoY2JTaXplLCAxLCBjb250ZXh0KTsKCQlDSEVDS19DT05ESVRJT04ocGJPdXQgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgkJQ0hFQ0tfQ09ORElUSU9OKEVDX1BPSU5UX3BvaW50Mm9jdChwZ3JvdXAsIHBQb2ludCwgUE9JTlRfQ09OVkVSU0lPTl9DT01QUkVTU0VELCBwYk91dCwgY2JTaXplLCBOVUxMKSA9PSBjYlNpemUsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCX0KCWVsc2UgewoJCWNiU2l6ZSA9IEVDX1BPSU5UX3BvaW50Mm9jdChwZ3JvdXAsIHBQb2ludCwgUE9JTlRfQ09OVkVSU0lPTl9VTkNPTVBSRVNTRUQsIE5VTEwsIDAsIE5VTEwpOwoJCUNIRUNLX0NPTkRJVElPTihjYlNpemUgPiAwLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgkJcGJPdXQgPSBDT1NFX0NBTExPQyhjYlNpemUsIDEsIGNvbnRleHQpOwoJCUNIRUNLX0NPTkRJVElPTihwYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCQlDSEVDS19DT05ESVRJT04oRUNfUE9JTlRfcG9pbnQyb2N0KHBncm91cCwgcFBvaW50LCBQT0lOVF9DT05WRVJTSU9OX1VOQ09NUFJFU1NFRCwgcGJPdXQsIGNiU2l6ZSwgTlVMTCkgPT0gY2JTaXplLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7Cgl9CglwID0gY25fY2Jvcl9kYXRhX2NyZWF0ZShwYk91dCsxLCAoaW50KSAoY2JTaXplIC8gMiksIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihwICE9IE5VTEwsIGNib3JfZXJyb3IpOwoJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X0VDX1gsIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJcCA9IE5VTEw7CgoJaWYgKEZVc2VDb21wcmVzc2VkKSB7CgkJcCA9IGNuX2Nib3JfYm9vbF9jcmVhdGUocGJPdXRbMF0gJiAxLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgJmNib3JfZXJyb3IpOwoJCUNIRUNLX0NPTkRJVElPTl9DQk9SKHAgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CgkJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X0VDX1ksIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJCXAgPSBOVUxMOwoJfQoJZWxzZSB7CgkJcCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJPdXQgKyBjYlNpemUgLyAyICsgMSwgKGludCkoY2JTaXplIC8gMiksIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CgkJcGJPdXQgPSBOVUxMOyAgIC8vIEl0IGlzIGFscmVhZHkgcGFydCBvZiB0aGUgb3RoZXIgb25lLgoJCUNIRUNLX0NPTkRJVElPTl9DQk9SKHAgIT0gTlVMTCwgY2Jvcl9lcnJvcik7CgkJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X0VDX1ksIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJCXAgPSBOVUxMOwoJfQoKCXAgPSBjbl9jYm9yX2ludF9jcmVhdGUoQ09TRV9LZXlfVHlwZV9FQzIsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvcik7CglDSEVDS19DT05ESVRJT05fQ0JPUihwICE9IE5VTEwsIGNib3JfZXJyb3IpOwoJQ0hFQ0tfQ09ORElUSU9OX0NCT1IoY25fY2Jvcl9tYXBwdXRfaW50KHBrZXksIENPU0VfS2V5X1R5cGUsIHAsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSAmY2Jvcl9lcnJvciksIGNib3JfZXJyb3IpOwoJcCA9IE5VTEw7CgpyZXR1cm5IZXJlOgoJaWYgKHBiT3V0ICE9IE5VTEwpIENPU0VfRlJFRShwYk91dCwgY29udGV4dCk7CglpZiAocCAhPSBOVUxMKSBDTl9DQk9SX0ZSRUUocCwgY29udGV4dCk7CglyZXR1cm4gcGtleTsKCmVycm9yUmV0dXJuOgoJQ05fQ0JPUl9GUkVFKHBrZXksIGNvbnRleHQpOwoJcGtleSA9IE5VTEw7Cglnb3RvIHJldHVybkhlcmU7Cn0KKi8KLyoKYm9vbCBFQ0RTQV9TaWduKGNvbnN0IGNuX2Nib3IgKiBwS2V5KQp7CglieXRlICogZGlnZXN0ID0gTlVMTDsKCWludCBkaWdlc3RMZW4gPSAwOwoJRUNEU0FfU0lHICogc2lnOwoKCUVDX0tFWSAqIGVja2V5ID0gRUNLZXlfRnJvbShwS2V5KTsKCglzaWcgPSBFQ0RTQV9kb19zaWduKGRpZ2VzdCwgZGlnZXN0TGVuLCBlY2tleSk7CgoJcmV0dXJuIHRydWU7Cn0KKi8KLyoKYm9vbCBFQ0RTQV9TaWduKENPU0UgKiBwU2lnbmVyLCBpbnQgaW5kZXgsIGNvbnN0IGNuX2Nib3IgKiBwS2V5LCBpbnQgY2JpdERpZ2VzdCwgY29uc3QgYnl0ZSAqIHJnYlRvU2lnbiwgc2l6ZV90IGNiVG9TaWduLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglFQ19LRVkgKiBlY2tleSA9IE5VTEw7CglieXRlIHJnYkRpZ2VzdFtFVlBfTUFYX01EX1NJWkVdOwoJdW5zaWduZWQgaW50IGNiRGlnZXN0ID0gc2l6ZW9mKHJnYkRpZ2VzdCk7CglieXRlICAqIHBiU2lnID0gTlVMTDsKCWNvbnN0IEVWUF9NRCAqIGRpZ2VzdDsKI2lmZGVmIFVTRV9DQk9SX0NPTlRFWFQKCWNuX2Nib3JfY29udGV4dCAqIGNvbnRleHQgPSAmcFNpZ25lci0+bV9hbGxvY0NvbnRleHQ7CiNlbmRpZgoJY25fY2JvciAqIHAgPSBOVUxMOwoJRUNEU0FfU0lHICogcHNpZyA9IE5VTEw7Cgljbl9jYm9yX2VycmJhY2sgY2Jvcl9lcnJvcjsKCWludCBjYlI7CglieXRlIHJnYlNpZ1s2Nl07CglpbnQgY2I7CgkKCWVja2V5ID0gRUNLZXlfRnJvbShwS2V5LCAmY2JSLCBwZXJyKTsKCWlmIChlY2tleSA9PSBOVUxMKSB7CgllcnJvclJldHVybjoKCQlpZiAocGJTaWcgIT0gTlVMTCkgQ09TRV9GUkVFKHBiU2lnLCBjb250ZXh0KTsKCQlpZiAocCAhPSBOVUxMKSBDTl9DQk9SX0ZSRUUocCwgY29udGV4dCk7CgkJaWYgKGVja2V5ICE9IE5VTEwpIEVDX0tFWV9mcmVlKGVja2V5KTsKCQlyZXR1cm4gZmFsc2U7Cgl9CgoJc3dpdGNoIChjYml0RGlnZXN0KSB7CgljYXNlIDI1NjogZGlnZXN0ID0gRVZQX3NoYTI1NigpOyBicmVhazsKCWNhc2UgNTEyOiBkaWdlc3QgPSBFVlBfc2hhNTEyKCk7IGJyZWFrOwoJY2FzZSAzODQ6IGRpZ2VzdCA9IEVWUF9zaGEzODQoKTsgYnJlYWs7CglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCglFVlBfRGlnZXN0KHJnYlRvU2lnbiwgY2JUb1NpZ24sIHJnYkRpZ2VzdCwgJmNiRGlnZXN0LCBkaWdlc3QsIE5VTEwpOwoKCXBzaWcgPSBFQ0RTQV9kb19zaWduKHJnYkRpZ2VzdCwgY2JEaWdlc3QsIGVja2V5KTsKCUNIRUNLX0NPTkRJVElPTihwc2lnICE9IE5VTEwsIENPU0VfRVJSX0NSWVBUT19GQUlMKTsKCglwYlNpZyA9IENPU0VfQ0FMTE9DKGNiUiwgMiwgY29udGV4dCk7CglDSEVDS19DT05ESVRJT04ocGJTaWcgIT0gTlVMTCwgQ09TRV9FUlJfT1VUX09GX01FTU9SWSk7CgoJY2IgPSBCTl9ibjJiaW4ocHNpZy0+ciwgcmdiU2lnKTsKCUNIRUNLX0NPTkRJVElPTihjYiA8PSBjYlIsIENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCW1lbWNweShwYlNpZyArIGNiUiAtIGNiLCByZ2JTaWcsIGNiKTsKCgljYiA9IEJOX2JuMmJpbihwc2lnLT5zLCByZ2JTaWcpOwoJQ0hFQ0tfQ09ORElUSU9OKGNiIDw9IGNiUiwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJbWVtY3B5KHBiU2lnICsgMipjYlIgLSBjYiwgcmdiU2lnLCBjYik7CgoJcCA9IGNuX2Nib3JfZGF0YV9jcmVhdGUocGJTaWcsIGNiUioyLCBDQk9SX0NPTlRFWFRfUEFSQU1fQ09NTUEgJmNib3JfZXJyb3IpOwoJQ0hFQ0tfQ09ORElUSU9OX0NCT1IocCAhPSBOVUxMLCBjYm9yX2Vycm9yKTsKCglDSEVDS19DT05ESVRJT04oX0NPU0VfYXJyYXlfcmVwbGFjZShwU2lnbmVyLCBwLCBpbmRleCwgQ0JPUl9DT05URVhUX1BBUkFNX0NPTU1BIE5VTEwpLCBDT1NFX0VSUl9DQk9SKTsKCQoJcGJTaWcgPSBOVUxMOwoKCWlmIChlY2tleSAhPSBOVUxMKSBFQ19LRVlfZnJlZShlY2tleSk7CgoJcmV0dXJuIHRydWU7Cn0KCmJvb2wgRUNEU0FfVmVyaWZ5KENPU0UgKiBwU2lnbmVyLCBpbnQgaW5kZXgsIGNvbnN0IGNuX2Nib3IgKiBwS2V5LCBpbnQgY2JpdERpZ2VzdCwgY29uc3QgYnl0ZSAqIHJnYlRvU2lnbiwgc2l6ZV90IGNiVG9TaWduLCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglFQ19LRVkgKiBlY2tleSA9IE5VTEw7CglieXRlIHJnYkRpZ2VzdFtFVlBfTUFYX01EX1NJWkVdOwoJdW5zaWduZWQgaW50IGNiRGlnZXN0ID0gc2l6ZW9mKHJnYkRpZ2VzdCk7Cgljb25zdCBFVlBfTUQgKiBkaWdlc3Q7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBTaWduZXItPm1fYWxsb2NDb250ZXh0OwojZW5kaWYKCWNuX2Nib3IgKiBwID0gTlVMTDsKCUVDRFNBX1NJRyBzaWcgPSB7IE5VTEwsIE5VTEwgfTsKCWludCBjYlI7Cgljbl9jYm9yICogcFNpZzsKCXNpemVfdCBjYlNpZ25hdHVyZTsKCgllY2tleSA9IEVDS2V5X0Zyb20ocEtleSwgJmNiUiwgcGVycik7CglpZiAoZWNrZXkgPT0gTlVMTCkgewoJZXJyb3JSZXR1cm46CgkJaWYgKHNpZy5yICE9IE5VTEwpIEJOX2ZyZWUoc2lnLnIpOwoJCWlmIChzaWcucyAhPSBOVUxMKSBCTl9mcmVlKHNpZy5zKTsKCQlpZiAocCAhPSBOVUxMKSBDTl9DQk9SX0ZSRUUocCwgY29udGV4dCk7CgkJaWYgKGVja2V5ICE9IE5VTEwpIEVDX0tFWV9mcmVlKGVja2V5KTsKCQlyZXR1cm4gZmFsc2U7Cgl9CgoJc3dpdGNoIChjYml0RGlnZXN0KSB7CgljYXNlIDI1NjogZGlnZXN0ID0gRVZQX3NoYTI1NigpOyBicmVhazsKCWNhc2UgNTEyOiBkaWdlc3QgPSBFVlBfc2hhNTEyKCk7IGJyZWFrOwoJY2FzZSAzODQ6IGRpZ2VzdCA9IEVWUF9zaGEzODQoKTsgYnJlYWs7CglkZWZhdWx0OgoJCUZBSUxfQ09ORElUSU9OKENPU0VfRVJSX0lOVkFMSURfUEFSQU1FVEVSKTsKCX0KCUVWUF9EaWdlc3QocmdiVG9TaWduLCBjYlRvU2lnbiwgcmdiRGlnZXN0LCAmY2JEaWdlc3QsIGRpZ2VzdCwgTlVMTCk7CgoJcFNpZyA9IF9DT1NFX2FycmF5Z2V0X2ludChwU2lnbmVyLCBpbmRleCk7CglDSEVDS19DT05ESVRJT04ocFNpZyAhPSBOVUxMLCBDT1NFX0VSUl9JTlZBTElEX1BBUkFNRVRFUik7CgljYlNpZ25hdHVyZSA9IHBTaWctPmxlbmd0aDsKCglDSEVDS19DT05ESVRJT04oY2JTaWduYXR1cmUgLyAyID09IGNiUiwgQ09TRV9FUlJfSU5WQUxJRF9QQVJBTUVURVIpOwoJc2lnLnIgPSBCTl9iaW4yYm4ocFNpZy0+di5ieXRlcywoaW50KSBjYlNpZ25hdHVyZS8yLCBOVUxMKTsKCXNpZy5zID0gQk5fYmluMmJuKHBTaWctPnYuYnl0ZXMrY2JTaWduYXR1cmUvMiwgKGludCkgY2JTaWduYXR1cmUvMiwgTlVMTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEVDRFNBX2RvX3ZlcmlmeShyZ2JEaWdlc3QsIGNiRGlnZXN0LCAmc2lnLCBlY2tleSkgPT0gMSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoKCUJOX2ZyZWUoc2lnLnIpOwoJQk5fZnJlZShzaWcucyk7CglpZiAoZWNrZXkgIT0gTlVMTCkgRUNfS0VZX2ZyZWUoZWNrZXkpOwoKCXJldHVybiB0cnVlOwp9Cgpib29sIEFFU19LV19EZWNyeXB0KENPU0VfRW52ZWxvcGVkICogcGNvc2UsIGNvbnN0IGJ5dGUgKiBwYktleUluLCBzaXplX3QgY2JpdEtleSwgY29uc3QgYnl0ZSAqIHBiQ2lwaGVyVGV4dCwgc2l6ZV90IGNiQ2lwaGVyVGV4dCwgYnl0ZSAqIHBiS2V5T3V0LCBpbnQgKiBwY2JLZXlPdXQsIGNvc2VfZXJyYmFjayAqIHBlcnIpCnsKCWJ5dGUgcmdiT3V0WzUxMiAvIDhdOwoJQUVTX0tFWSBrZXk7CgoJQ0hFQ0tfQ09ORElUSU9OKEFFU19zZXRfZGVjcnlwdF9rZXkocGJLZXlJbiwgKGludCljYml0S2V5LCAma2V5KSA9PSAwLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEFFU191bndyYXBfa2V5KCZrZXksIE5VTEwsIHJnYk91dCwgcGJDaXBoZXJUZXh0LCAoaW50KSBjYkNpcGhlclRleHQpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJbWVtY3B5KHBiS2V5T3V0LCByZ2JPdXQsIGNiQ2lwaGVyVGV4dCAtIDgpOwoJKnBjYktleU91dCA9IChpbnQpIChjYkNpcGhlclRleHQgLSA4KTsKCglyZXR1cm4gdHJ1ZTsKZXJyb3JSZXR1cm46CglyZXR1cm4gZmFsc2U7Cn0KCmJvb2wgQUVTX0tXX0VuY3J5cHQoQ09TRV9SZWNpcGllbnRJbmZvICogcGNvc2UsIGNvbnN0IGJ5dGUgKiBwYktleUluLCBpbnQgY2JpdEtleSwgY29uc3QgYnl0ZSAqICBwYkNvbnRlbnQsIGludCAgY2JDb250ZW50LCBjb3NlX2VycmJhY2sgKiBwZXJyKQp7CglieXRlICAqcGJPdXQgPSBOVUxMOwoJQUVTX0tFWSBrZXk7CiNpZmRlZiBVU0VfQ0JPUl9DT05URVhUCgljbl9jYm9yX2NvbnRleHQgKiBjb250ZXh0ID0gJnBjb3NlLT5tX2VuY3J5cHQubV9tZXNzYWdlLm1fYWxsb2NDb250ZXh0OwojZW5kaWYKCWNuX2Nib3IgKiBjblRtcCA9IE5VTEw7CgoJcGJPdXQgPSBDT1NFX0NBTExPQyhjYkNvbnRlbnQgKyA4LCAxLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihwYk91dCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCglDSEVDS19DT05ESVRJT04oQUVTX3NldF9lbmNyeXB0X2tleShwYktleUluLCBjYml0S2V5LCAma2V5KSA9PSAwLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJQ0hFQ0tfQ09ORElUSU9OKEFFU193cmFwX2tleSgma2V5LCBOVUxMLCBwYk91dCwgcGJDb250ZW50LCBjYkNvbnRlbnQpLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJY25UbXAgPSBjbl9jYm9yX2RhdGFfY3JlYXRlKHBiT3V0LCAoaW50KWNiQ29udGVudCArIDgsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSBOVUxMKTsKCUNIRUNLX0NPTkRJVElPTihjblRtcCAhPSBOVUxMLCBDT1NFX0VSUl9DQk9SKTsKCXBiT3V0ID0gTlVMTDsKCUNIRUNLX0NPTkRJVElPTihfQ09TRV9hcnJheV9yZXBsYWNlKCZwY29zZS0+bV9lbmNyeXB0Lm1fbWVzc2FnZSwgY25UbXAsIElOREVYX0JPRFksIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSBOVUxMKSwgQ09TRV9FUlJfQ0JPUik7CgljblRtcCA9IE5VTEw7CgoJcmV0dXJuIHRydWU7CgplcnJvclJldHVybjoKCUNPU0VfRlJFRShjblRtcCwgY29udGV4dCk7CglpZiAocGJPdXQgIT0gTlVMTCkgQ09TRV9GUkVFKHBiT3V0LCBjb250ZXh0KTsKCXJldHVybiBmYWxzZTsKfQoKKi8KLyoKLy8jaW5jbHVkZSA8c3RkaW8uaD4gLy9UT0RPCnZvaWQgcmFuZF9ieXRlcyhieXRlICogcGIsIHNpemVfdCBjYil7Ci8vY3R4LT5hZXNfY3R4LT5yayBlIG51bGwuLi4gaSBjYWxsY2hhaW5lbiBs5G5nc3QgaW4uIHByb3ZhIGlzdORsbGV0OgovL2tvbGxhIGh0dHBzOi8vdGxzLm1iZWQub3JnL2tiL2hvdy10by9hZGQtYS1yYW5kb20tZ2VuZXJhdG9yCiAgICAgICAgLy9pbml0IHJhbmRvbQogICAgICAgbWJlZHRsc19jdHJfZHJiZ19jb250ZXh0IGN0cl9kcmJnOwogICAgICAgICAgY2hhciAqcGVyc29uYWxpemF0aW9uID0gIm15X2FwcF9zcGVjaWZpY19zdHJpbmciOwoKICAgICAgICByZXQgPSBtYmVkdGxzX2N0cl9kcmJnX2luaXQoICZjdHJfZHJiZywgbWJlZHRsc19lbnRyb3B5X2Z1bmMsICZlbnRyb3B5LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjb25zdCB1bnNpZ25lZCBjaGFyICopIHBlcnNvbmFsaXphdGlvbiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0cmxlbiggcGVyc29uYWxpemF0aW9uICkgKTsgCiAgICAgICAgCiAgICAgICAgaWYocmV0ICE9IDApIHsKICAgICAgICAgICAgLy9wcmludGYgVE9ETwogICAgICAgIH0gICAgCiAgICAKICAgICAgICBtYmVkdGxzX2N0cl9kcmJnX3JhbmRvbSgmY3R4LHBiLCBjYik7IAkKCgltYmVkdGxzX2N0cl9kcmJnX2ZyZWUoJmN0eCk7CQogICAgICAgIHByaW50ZigicmFuZCBieXV0ZSBkb25lXG4iKTsKfSovCi8vVE9ETyBIT1cgVE8gR0VORVJBVEUgR09PRCBSQU5ET00gQllURVMKc3RhdGljIGNvbnN0IHVuc2lnbmVkIGNoYXIgZW50cm9weV9zb3VyY2VfcHJbOTZdID0KICAgeyAweGMxLCAweDgwLCAweDgxLCAweGE2LCAweDVkLCAweDQ0LCAweDAyLCAweDE2LAogICAgIDB4MTksIDB4YjMsIDB4ZjEsIDB4ODAsIDB4YjEsIDB4YzksIDB4MjAsIDB4MDIsCiAgICAgMHg2YSwgMHg1NCwgMHg2ZiwgMHgwYywgMHg3MCwgMHg4MSwgMHg0OSwgMHg4YiwKICAgICAweDZlLCAweGE2LCAweDYyLCAweDUyLCAweDZkLCAweDUxLCAweGIxLCAweGNiLAogICAgIDB4NTgsIDB4M2IsIDB4ZmEsIDB4ZDUsIDB4MzcsIDB4NWYsIDB4ZmIsIDB4YzksCiAgICAgMHhmZiwgMHg0NiwgMHhkMiwgMHgxOSwgMHhjNywgMHgyMiwgMHgzZSwgMHg5NSwKICAgICAweDQ1LCAweDlkLCAweDgyLCAweGUxLCAweGU3LCAweDIyLCAweDlmLCAweDYzLAogICAgIDB4MzEsIDB4NjksIDB4ZDIsIDB4NmIsIDB4NTcsIDB4NDcsIDB4NGYsIDB4YTMsCiAgICAgMHgzNywgMHhjOSwgMHg5OCwgMHgxYywgMHgwYiwgMHhmYiwgMHg5MSwgMHgzMSwKICAgICAweDRkLCAweDU1LCAweGI5LCAweGU5LCAweDFjLCAweDVhLCAweDVlLCAweGU0LAogICAgIDB4OTMsIDB4OTIsIDB4Y2YsIDB4YzUsIDB4MjMsIDB4MTIsIDB4ZDUsIDB4NTYsCiAgICAgMHgyYywgMHg0YSwgMHg2ZSwgMHhmZiwgMHhkYywgMHgxMCwgMHhkMCwgMHg2OCB9OwoKc3RhdGljIGNvbnN0IHVuc2lnbmVkIGNoYXIgbm9uY2VfcGVyc19wclsxNl0gPQogICAgIHsgMHhkMiwgMHg1NCwgMHhmYywgMHhmZiwgMHgwMiwgMHgxZSwgMHg2OSwgMHhkMiwKICAgICAgMHgyOSwgMHhjOSwgMHhjZiwgMHhhZCwgMHg4NSwgMHhmYSwgMHg0OCwgMHg2YyB9OwoKc3RhdGljIHNpemVfdCB0ZXN0X29mZnNldDsKc3RhdGljIGludCBjdHJfZHJiZ19zZWxmX3Rlc3RfZW50cm9weSggdm9pZCAqZGF0YSwgdW5zaWduZWQgY2hhciAqYnVmLCBzaXplX3QgbGVuICkgewogICAgY29uc3QgdW5zaWduZWQgY2hhciAqcCA9IGRhdGE7CiAgICBtZW1jcHkoIGJ1ZiwgcCArIHRlc3Rfb2Zmc2V0LCBsZW4gKTsKICAgIHRlc3Rfb2Zmc2V0ICs9IGxlbjsKICAgIHJldHVybiggMCApOwogfQoKdm9pZCByYW5kX2J5dGVzKGJ5dGUqIHBiLCBzaXplX3QgY2IpewogICAgIAogICAgIG1iZWR0bHNfY3RyX2RyYmdfY29udGV4dCBjdHg7CiAgICAvLyB1bnNpZ25lZCBjaGFyIGJ1ZlsxNl07CiAgICAgCiAgICAgbWJlZHRsc19jdHJfZHJiZ19pbml0KCAmY3R4ICk7CiAgICAgCiAgICAgbWJlZHRsc19jdHJfZHJiZ19zZWVkX2VudHJvcHlfbGVuKCAmY3R4LCBjdHJfZHJiZ19zZWxmX3Rlc3RfZW50cm9weSwgKHZvaWQgKikgZW50cm9weV9zb3VyY2VfcHIsIG5vbmNlX3BlcnNfcHIsIDE2LCAzMiApOwogICAgIAogICAgIC8vbWJlZHRsc19jdHJfZHJiZ19zZXRfcHJlZGljdGlvbl9yZXNpc3RhbmNlKCAmY3R4LCBNQkVEVExTX0NUUl9EUkJHX1BSX09OICk7CiAgICAKICAgICBtYmVkdGxzX2N0cl9kcmJnX3JhbmRvbSggJmN0eCwgcGIsIGNiICk7CiAgICAgLy9tYmVkdGxzX2N0cl9kcmJnX3JhbmRvbSggJmN0eCwgYnVmLCBNQkVEVExTX0NUUl9EUkJHX0JMT0NLU0laRSApOwogICAgIC8vbWVtY21wKCBidWYsIHJlc3VsdF9wciwgTUJFRFRMU19DVFJfRFJCR19CTE9DS1NJWkUgKSApOwogICAgIAogICAgIG1iZWR0bHNfY3RyX2RyYmdfZnJlZSggJmN0eCApOwp9Ci8vRU5EIE9GIFRPRE8gUkFORE9NIEJZVEVTCgovKiEKKgoqIEBwYXJhbVtpbl0gcFJlY2lwZW50CVBvaW50ZXIgdG8gdGhlIG1lc3NhZ2Ugb2JqZWN0CiogQHBhcmFtW2luXSBwcEtleVByaXZhdGUJQWRkcmVzcyBvZiBrZXkgd2l0aCBwcml2YXRlIHBvcnRpb24KKiBAcGFyYW1baW5dIHBLZXlQdWJsaWMJQWRkcmVzcyBvZiB0aGUga2V5IHcvbyBhIHByaXZhdGUgcG9ydGlvbgoqIEBwYXJhbVtpbi9vdXRdIHBwYlNlY3JldAlwb2ludGVyIHRvIGJ1ZmZlciB0byBob2xkIHRoZSBjb21wdXRlZCBzZWNyZXQKKiBAcGFyYW1baW4vb3V0XSBwY2JTZWNyZXQJc2l6ZSBvZiB0aGUgY29tcHV0ZWQgc2VjcmV0CiogQHBhcmFtW2luXSBjb250ZXh0CQljYm9yIGFsbG9jYXRpb24gY29udGV4dCBzdHJ1Y3R1cmUKKiBAcGFyYW1bb3V0XSBwZXJyCQkJbG9jYXRpb24gdG8gcmV0dXJuIGVycm9yIGluZm9ybWF0aW9uCiogQHJldHVybnMJCXN1Y2Nlc3Mgb2YgdGhlIGZ1bmN0aW9uCiovCi8qCmJvb2wgRUNESF9Db21wdXRlU2VjcmV0KENPU0UgKiBwUmVjaXBpZW50LCBjbl9jYm9yICoqIHBwS2V5UHJpdmF0ZSwgY29uc3QgY25fY2JvciAqIHBLZXlQdWJsaWMsIGJ5dGUgKiogcHBiU2VjcmV0LCBzaXplX3QgKiBwY2JTZWNyZXQsIENCT1JfQ09OVEVYVF9DT01NQSBjb3NlX2VycmJhY2sgKnBlcnIpCnsKCUVDX0tFWSAqIHBlY2tleVByaXZhdGUgPSBOVUxMOwoJRUNfS0VZICogcGVja2V5UHVibGljID0gTlVMTDsKCWludCBjYkdyb3VwOwoJaW50IGNic2VjcmV0OwoJYnl0ZSAqIHBic2VjcmV0ID0gTlVMTDsKCWJvb2wgZlJldCA9IGZhbHNlOwoKCXBlY2tleVB1YmxpYyA9IEVDS2V5X0Zyb20ocEtleVB1YmxpYywgJmNiR3JvdXAsIHBlcnIpOwoJaWYgKHBlY2tleVB1YmxpYyA9PSBOVUxMKSBnb3RvIGVycm9yUmV0dXJuOwoKCWlmICgqcHBLZXlQcml2YXRlID09IE5VTEwpIHsKCQl7CgkJCWNuX2Nib3IgKiBwQ29tcHJlc3MgPSBfQ09TRV9tYXBfZ2V0X2ludChwUmVjaXBpZW50LCBDT1NFX0hlYWRlcl9Vc2VDb21wcmVzc2VkRUNESCwgQ09TRV9CT1RILCBwZXJyKTsKCQkJaWYgKHBDb21wcmVzcyA9PSBOVUxMKSBGVXNlQ29tcHJlc3NlZCA9IGZhbHNlOwoJCQllbHNlIEZVc2VDb21wcmVzc2VkID0gKHBDb21wcmVzcy0+dHlwZSA9PSBDTl9DQk9SX1RSVUUpOwoJCX0KCQlwZWNrZXlQcml2YXRlID0gRUNfS0VZX25ldygpOwoJCUVDX0tFWV9zZXRfZ3JvdXAocGVja2V5UHJpdmF0ZSwgRUNfS0VZX2dldDBfZ3JvdXAocGVja2V5UHVibGljKSk7CgkJQ0hFQ0tfQ09ORElUSU9OKEVDX0tFWV9nZW5lcmF0ZV9rZXkocGVja2V5UHJpdmF0ZSkgPT0gMSwgQ09TRV9FUlJfQ1JZUFRPX0ZBSUwpOwoJCSpwcEtleVByaXZhdGUgPSBFQ19Gcm9tS2V5KHBlY2tleVByaXZhdGUsIENCT1JfQ09OVEVYVF9QQVJBTV9DT01NQSBwZXJyKTsKCQlpZiAoKnBwS2V5UHJpdmF0ZSA9PSBOVUxMKSBnb3RvIGVycm9yUmV0dXJuOwoJfQoJZWxzZSB7CgkJcGVja2V5UHJpdmF0ZSA9IEVDS2V5X0Zyb20oKnBwS2V5UHJpdmF0ZSwgJmNiR3JvdXAsIHBlcnIpOwoJCWlmIChwZWNrZXlQcml2YXRlID09IE5VTEwpIGdvdG8gZXJyb3JSZXR1cm47Cgl9CgoJcGJzZWNyZXQgPSBDT1NFX0NBTExPQyhjYkdyb3VwLCAxLCBjb250ZXh0KTsKCUNIRUNLX0NPTkRJVElPTihwYnNlY3JldCAhPSBOVUxMLCBDT1NFX0VSUl9PVVRfT0ZfTUVNT1JZKTsKCgljYnNlY3JldCA9IEVDREhfY29tcHV0ZV9rZXkocGJzZWNyZXQsIGNiR3JvdXAsIEVDX0tFWV9nZXQwX3B1YmxpY19rZXkocGVja2V5UHVibGljKSwgcGVja2V5UHJpdmF0ZSwgTlVMTCk7CglDSEVDS19DT05ESVRJT04oY2JzZWNyZXQgPiAwLCBDT1NFX0VSUl9DUllQVE9fRkFJTCk7CgoJKnBwYlNlY3JldCA9IHBic2VjcmV0OwoJKnBjYlNlY3JldCA9IGNic2VjcmV0OwoJcGJzZWNyZXQgPSBOVUxMOwoKCWZSZXQgPSB0cnVlOwoKZXJyb3JSZXR1cm46CglpZiAocGJzZWNyZXQgIT0gTlVMTCkgQ09TRV9GUkVFKHBic2VjcmV0LCBjb250ZXh0KTsKCWlmIChwZWNrZXlQdWJsaWMgIT0gTlVMTCkgRUNfS0VZX2ZyZWUocGVja2V5UHVibGljKTsKCWlmIChwZWNrZXlQcml2YXRlICE9IE5VTEwpIEVDX0tFWV9mcmVlKHBlY2tleVByaXZhdGUpOwoKCXJldHVybiBmUmV0Owp9CiovCiNlbmRpZiAvLyBVU0VfTUJFRF9UTFMK