LyoKICogYSBHVUkgYXBwbGljYXRpb24gZm9yIGRpc3BsYXlpbmcgYSBjb25zb2xlCiAqCVVTRVIzMiBiYWNrIGVuZAogKiBDb3B5cmlnaHQgMjAwMSBFcmljIFBvdWVjaAogKgogKiBUaGlzIGxpYnJhcnkgaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yCiAqIG1vZGlmeSBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMKICogTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkgdGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbjsgZWl0aGVyCiAqIHZlcnNpb24gMi4xIG9mIHRoZSBMaWNlbnNlLCBvciAoYXQgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLgogKgogKiBUaGlzIGxpYnJhcnkgaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKICogYnV0IFdJVEhPVVQgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YKICogTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiAgU2VlIHRoZSBHTlUKICogTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KICoKICogWW91IHNob3VsZCBoYXZlIHJlY2VpdmVkIGEgY29weSBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFsb25nIHdpdGggdGhpcyBsaWJyYXJ5OyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCiAqIEZvdW5kYXRpb24sIEluYy4sIDUxIEZyYW5rbGluIFN0LCBGaWZ0aCBGbG9vciwgQm9zdG9uLCBNQSAwMjExMC0xMzAxLCBVU0EKICovCgojaW5jbHVkZSA8c3RkaW8uaD4KI2luY2x1ZGUgPHN0ZGxpYi5oPgojaW5jbHVkZSAid2luZWNvbl91c2VyLmgiCiNpbmNsdWRlICJ3aW5ubHMuaCIKCiNpbmNsdWRlICJ3aW5lL2RlYnVnLmgiCgpXSU5FX0RFRkFVTFRfREVCVUdfQ0hBTk5FTCh3aW5lY29uc29sZSk7CldJTkVfREVDTEFSRV9ERUJVR19DSEFOTkVMKHdjX2ZvbnQpOwoKVUlOVCBnX3VpRGVmYXVsdENoYXJzZXQ7CgovKiBtYXBwaW5nIGNvbnNvbGUgY29sb3JzIHRvIFJHQiB2YWx1ZXMgKi8KY29uc3QgQ09MT1JSRUYgV0NVU0VSX0NvbG9yTWFwWzE2XSA9CnsKICAgIFJHQigweDAwLCAweDAwLCAweDAwKSwgUkdCKDB4MDAsIDB4MDAsIDB4ODApLCBSR0IoMHgwMCwgMHg4MCwgMHgwMCksIFJHQigweDAwLCAweDgwLCAweDgwKSwKICAgIFJHQigweDgwLCAweDAwLCAweDAwKSwgUkdCKDB4ODAsIDB4MDAsIDB4ODApLCBSR0IoMHg4MCwgMHg4MCwgMHgwMCksIFJHQigweDgwLCAweDgwLCAweDgwKSwKICAgIFJHQigweEMwLCAweEMwLCAweEMwKSwgUkdCKDB4MDAsIDB4MDAsIDB4RkYpLCBSR0IoMHgwMCwgMHhGRiwgMHgwMCksIFJHQigweDAwLCAweEZGLCAweEZGKSwKICAgIFJHQigweEZGLCAweDAwLCAweDAwKSwgUkdCKDB4RkYsIDB4MDAsIDB4RkYpLCBSR0IoMHhGRiwgMHhGRiwgMHgwMCksIFJHQigweEZGLCAweEZGLCAweEZGKSwKfTsKCnN0YXRpYyBCT09MIFdDVVNFUl9TZXRGb250KHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBjb25zdCBMT0dGT05UKiBmb250KTsKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9GaWxsTWVtREMKICoKICogRmlsbHMgdGhlIE1lbSBEQyB3aXRoIGN1cnJlbnQgY2VsbHMgdmFsdWVzCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfRmlsbE1lbURDKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBpbnQgdXBkX3RwLCBpbnQgdXBkX2JtKQp7CiAgICB1bnNpZ25lZAkJaSwgaiwgazsKICAgIENIQVJfSU5GTyoJCWNlbGw7CiAgICBIRk9OVAkJaE9sZEZvbnQ7CiAgICBXT1JECQlhdHRyOwogICAgV0NIQVIqCQlsaW5lOwogICAgUkVDVCAgICAgICAgICAgICAgICByOwogICAgSEJSVVNIICAgICAgICAgICAgICBoYnI7CgogICAgLyogbm8gZm9udCBoYXMgYmVlbiBzZXQgdXAgeWV0LCBkb24ndCB3b3JyeSBhYm91dCBmaWxsaW5nIHRoZSBiaXRtYXAsCiAgICAgKiB3ZSdsbCBkbyBpdCBvbmNlIGEgZm9udCBpcyBjaG9zZW4KICAgICAqLwogICAgaWYgKCFQUklWQVRFKGRhdGEpLT5oRm9udCkgcmV0dXJuOwoKICAgIC8qIEZJWE1FOiBjb3VsZCBzZXQgdXAgYSBtZWNoYW5pc20gdG8gcmV1c2UgdGhlIGxpbmUgYmV0d2VlbiBkaWZmZXJlbnQKICAgICAqIGNhbGxzIHRvIHRoaXMgZnVuY3Rpb24KICAgICAqLwogICAgaWYgKCEobGluZSA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggKiBzaXplb2YoV0NIQVIpKSkpCiAgICAgICAgV0lORUNPTl9GYXRhbCgiT09NXG4iKTsKCiAgICBoT2xkRm9udCA9IFNlbGVjdE9iamVjdChQUklWQVRFKGRhdGEpLT5oTWVtREMsIFBSSVZBVEUoZGF0YSktPmhGb250KTsKICAgIGZvciAoaiA9IHVwZF90cDsgaiA8PSB1cGRfYm07IGorKykKICAgIHsKCWNlbGwgPSAmZGF0YS0+Y2VsbHNbaiAqIGRhdGEtPmN1cmNmZy5zYl93aWR0aF07Cglmb3IgKGkgPSAwOyBpIDwgZGF0YS0+Y3VyY2ZnLnNiX3dpZHRoOyBpKyspCgl7CgkgICAgYXR0ciA9IGNlbGxbaV0uQXR0cmlidXRlczsKCSAgICBTZXRCa0NvbG9yKFBSSVZBVEUoZGF0YSktPmhNZW1EQywgV0NVU0VSX0NvbG9yTWFwWyhhdHRyPj40KSYweDBGXSk7CgkgICAgU2V0VGV4dENvbG9yKFBSSVZBVEUoZGF0YSktPmhNZW1EQywgV0NVU0VSX0NvbG9yTWFwW2F0dHImMHgwRl0pOwoJICAgIGZvciAoayA9IGk7IGsgPCBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggJiYgY2VsbFtrXS5BdHRyaWJ1dGVzID09IGF0dHI7IGsrKykKCSAgICB7CgkJbGluZVtrIC0gaV0gPSBjZWxsW2tdLkNoYXIuVW5pY29kZUNoYXI7CgkgICAgfQoJICAgIFRleHRPdXQoUFJJVkFURShkYXRhKS0+aE1lbURDLCBpICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsIGogKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQsCgkJICAgIGxpbmUsIGsgLSBpKTsKICAgICAgICAgICAgaWYgKFBSSVZBVEUoZGF0YSktPmV4dF9sZWFkaW5nICYmIAogICAgICAgICAgICAgICAgKGhiciA9IENyZWF0ZVNvbGlkQnJ1c2goV0NVU0VSX0NvbG9yTWFwWyhhdHRyPj40KSYweDBGXSkpKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICByLmxlZnQgICA9IGkgKiBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgICAgICAgICAgICAgIHIudG9wICAgID0gKGogKyAxKSAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCAtIFBSSVZBVEUoZGF0YSktPmV4dF9sZWFkaW5nOwogICAgICAgICAgICAgICAgci5yaWdodCAgPSBrICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGg7CiAgICAgICAgICAgICAgICByLmJvdHRvbSA9IChqICsgMSkgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7CiAgICAgICAgICAgICAgICBGaWxsUmVjdChQUklWQVRFKGRhdGEpLT5oTWVtREMsICZyLCBoYnIpOwogICAgICAgICAgICAgICAgRGVsZXRlT2JqZWN0KGhicik7CiAgICAgICAgICAgIH0KCSAgICBpID0gayAtIDE7Cgl9CiAgICB9CiAgICBTZWxlY3RPYmplY3QoUFJJVkFURShkYXRhKS0+aE1lbURDLCBoT2xkRm9udCk7CiAgICBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBsaW5lKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX05ld0JpdG1hcAogKgogKiBFaXRoZXIgdGhlIGZvbnQgZ2VvbWV0cnkgb3IgdGhlIHNiIGdlb21ldHJ5IGhhcyBjaGFuZ2VkLiB3ZSBuZWVkCiAqIHRvIHJlY3JlYXRlIHRoZSBiaXRtYXAgZ2VvbWV0cnkuCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfTmV3Qml0bWFwKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBIREMgICAgICAgICBoREM7CiAgICBIQklUTUFQCWhuZXcsIGhvbGQ7CgogICAgaWYgKCFkYXRhLT5jdXJjZmcuc2Jfd2lkdGggfHwgIWRhdGEtPmN1cmNmZy5zYl9oZWlnaHQgfHwKICAgICAgICAhUFJJVkFURShkYXRhKS0+aEZvbnQgfHwgIShoREMgPSBHZXREQyhkYXRhLT5oV25kKSkpCiAgICAgICAgcmV0dXJuOwogICAgaG5ldyA9IENyZWF0ZUNvbXBhdGlibGVCaXRtYXAoaERDLAoJCQkJICBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsCgkJCQkgIGRhdGEtPmN1cmNmZy5zYl9oZWlnaHQgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQpOwogICAgUmVsZWFzZURDKGRhdGEtPmhXbmQsIGhEQyk7CiAgICBob2xkID0gU2VsZWN0T2JqZWN0KFBSSVZBVEUoZGF0YSktPmhNZW1EQywgaG5ldyk7CgogICAgaWYgKFBSSVZBVEUoZGF0YSktPmhCaXRtYXApCiAgICB7CglpZiAoaG9sZCA9PSBQUklWQVRFKGRhdGEpLT5oQml0bWFwKQoJICAgIERlbGV0ZU9iamVjdChQUklWQVRFKGRhdGEpLT5oQml0bWFwKTsKCWVsc2UKCSAgICBXSU5FX0ZJWE1FKCJsZWFrXG4iKTsKICAgIH0KICAgIFBSSVZBVEUoZGF0YSktPmhCaXRtYXAgPSBobmV3OwogICAgV0NVU0VSX0ZpbGxNZW1EQyhkYXRhLCAwLCBkYXRhLT5jdXJjZmcuc2JfaGVpZ2h0IC0gMSk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9SZXNpemVTY3JlZW5CdWZmZXIKICoKICoKICovCnN0YXRpYyB2b2lkIFdDVVNFUl9SZXNpemVTY3JlZW5CdWZmZXIoc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIFdDVVNFUl9OZXdCaXRtYXAoZGF0YSk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9Qb3NDdXJzb3IKICoKICogU2V0IGEgbmV3IHBvc2l0aW9uIGZvciB0aGUgY3Vyc29yCiAqLwpzdGF0aWMgdm9pZAlXQ1VTRVJfUG9zQ3Vyc29yKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBpZiAoZGF0YS0+aFduZCAhPSBHZXRGb2N1cygpIHx8ICFkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUpIHJldHVybjsKCiAgICBTZXRDYXJldFBvcygoZGF0YS0+Y3Vyc29yLlggLSBkYXRhLT5jdXJjZmcud2luX3Bvcy5YKSAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLAoJCShkYXRhLT5jdXJzb3IuWSAtIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0KTsKICAgIFNob3dDYXJldChkYXRhLT5oV25kKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1NoYXBlQ3Vyc29yCiAqCiAqIFNldHMgYSBuZXcgc2hhcGUgZm9yIHRoZSBjdXJzb3IKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9TaGFwZUN1cnNvcihzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgaW50IHNpemUsIGludCB2aXMsIEJPT0wgZm9yY2UpCnsKICAgIGlmIChmb3JjZSB8fCBzaXplICE9IGRhdGEtPmN1cmNmZy5jdXJzb3Jfc2l6ZSkKICAgIHsKCWlmIChkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUgJiYgZGF0YS0+aFduZCA9PSBHZXRGb2N1cygpKSBEZXN0cm95Q2FyZXQoKTsKCWlmIChQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwKSBEZWxldGVPYmplY3QoUFJJVkFURShkYXRhKS0+Y3Vyc29yX2JpdG1hcCk7CglQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwID0gTlVMTDsKCWlmIChzaXplICE9IDEwMCkKCXsKCSAgICBpbnQJCXcxNmI7IC8qIG51bWJlciBvZiBieXRlcyBwZXIgcm93LCBhbGlnbmVkIG9uIHdvcmQgc2l6ZSAqLwoJICAgIEJZVEUqCXB0cjsKCSAgICBpbnQJCWksIGosIG5ibDsKCgkgICAgdzE2YiA9ICgoZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGggKyAxNSkgJiB+MTUpIC8gODsKCSAgICBwdHIgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgSEVBUF9aRVJPX01FTU9SWSwgdzE2YiAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCk7CgkgICAgaWYgKCFwdHIpIFdJTkVDT05fRmF0YWwoIk9PTSIpOwoJICAgIG5ibCA9IG1heCgoZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0ICogc2l6ZSkgLyAxMDAsIDEpOwoJICAgIGZvciAoaiA9IGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCAtIG5ibDsgaiA8IGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodDsgaisrKQoJICAgIHsKCQlmb3IgKGkgPSAwOyBpIDwgZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGg7IGkrKykKCQl7CgkJICAgIHB0clt3MTZiICogaiArIChpIC8gOCldIHw9IDB4ODAgPj4gKGkgJiA3KTsKCQl9CgkgICAgfQoJICAgIFBSSVZBVEUoZGF0YSktPmN1cnNvcl9iaXRtYXAgPSBDcmVhdGVCaXRtYXAoZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0LCAxLCAxLCBwdHIpOwoJICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHB0cik7Cgl9CglkYXRhLT5jdXJjZmcuY3Vyc29yX3NpemUgPSBzaXplOwoJZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlID0gLTE7CiAgICB9CgogICAgdmlzID0gKHZpcykgPyBUUlVFIDogRkFMU0U7CiAgICBpZiAoZm9yY2UgfHwgdmlzICE9IGRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSkKICAgIHsKCWRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSA9IHZpczsKCWlmIChkYXRhLT5oV25kID09IEdldEZvY3VzKCkpCgl7CgkgICAgaWYgKHZpcykKCSAgICB7CgkJQ3JlYXRlQ2FyZXQoZGF0YS0+aFduZCwgUFJJVkFURShkYXRhKS0+Y3Vyc29yX2JpdG1hcCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLCBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQpOwoJCVdDVVNFUl9Qb3NDdXJzb3IoZGF0YSk7CgkgICAgfQoJICAgIGVsc2UKCSAgICB7CgkJRGVzdHJveUNhcmV0KCk7CgkgICAgfQoJfQogICAgfQogICAgV0lORUNPTl9EdW1wQ29uZmlnKCJjcnNyIiwgJmRhdGEtPmN1cmNmZyk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9Db21wdXRlUG9zaXRpb25zCiAqCiAqIFJlY29tcHV0ZXMgYWxsIHRoZSBjb21wb25lbnRzIChtYWlubHkgc2Nyb2xsIGJhcnMpIHBvc2l0aW9ucwogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX0NvbXB1dGVQb3NpdGlvbnMoc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIFJFQ1QJCXI7CiAgICBpbnQJCQlkeCwgZHk7CgogICAgLyogY29tcHV0ZSB3aW5kb3cgc2l6ZSBmcm9tIGRlc2lyZWQgY2xpZW50IHNpemUgKi8KICAgIHIubGVmdCA9IHIudG9wID0gMDsKICAgIHIucmlnaHQgPSBkYXRhLT5jdXJjZmcud2luX3dpZHRoICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGg7CiAgICByLmJvdHRvbSA9IGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0ICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0OwoKICAgIGlmIChJc1JlY3RFbXB0eSgmcikpIHJldHVybjsKCiAgICBBZGp1c3RXaW5kb3dSZWN0KCZyLCBHZXRXaW5kb3dMb25nKGRhdGEtPmhXbmQsIEdXTF9TVFlMRSksIEZBTFNFKTsKCiAgICBkeCA9IGR5ID0gMDsKICAgIGlmIChkYXRhLT5jdXJjZmcuc2Jfd2lkdGggPiBkYXRhLT5jdXJjZmcud2luX3dpZHRoKQogICAgewoJZHkgPSBHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZSFNDUk9MTCk7CglTZXRTY3JvbGxSYW5nZShkYXRhLT5oV25kLCBTQl9IT1JaLCAwLAogICAgICAgICAgICAgICAgICAgICAgIGRhdGEtPmN1cmNmZy5zYl93aWR0aCAtIGRhdGEtPmN1cmNmZy53aW5fd2lkdGgsIEZBTFNFKTsKCVNldFNjcm9sbFBvcyhkYXRhLT5oV25kLCBTQl9IT1JaLCAwLCBGQUxTRSk7IC8qIEZJWE1FICovCglTaG93U2Nyb2xsQmFyKGRhdGEtPmhXbmQsIFNCX0hPUlosIFRSVUUpOwogICAgfQogICAgZWxzZQogICAgewoJU2hvd1Njcm9sbEJhcihkYXRhLT5oV25kLCBTQl9IT1JaLCBGQUxTRSk7CiAgICB9CgogICAgaWYgKGRhdGEtPmN1cmNmZy5zYl9oZWlnaHQgPiBkYXRhLT5jdXJjZmcud2luX2hlaWdodCkKICAgIHsKCWR4ID0gR2V0U3lzdGVtTWV0cmljcyhTTV9DWFZTQ1JPTEwpOwoJU2V0U2Nyb2xsUmFuZ2UoZGF0YS0+aFduZCwgU0JfVkVSVCwgMCwKICAgICAgICAgICAgICAgICAgICAgICBkYXRhLT5jdXJjZmcuc2JfaGVpZ2h0IC0gZGF0YS0+Y3VyY2ZnLndpbl9oZWlnaHQsIEZBTFNFKTsKCVNldFNjcm9sbFBvcyhkYXRhLT5oV25kLCBTQl9WRVJULCAwLCBGQUxTRSk7IC8qIEZJWE1FICovCglTaG93U2Nyb2xsQmFyKGRhdGEtPmhXbmQsIFNCX1ZFUlQsIFRSVUUpOwogICAgfQogICAgZWxzZQogICAgewoJU2hvd1Njcm9sbEJhcihkYXRhLT5oV25kLCBTQl9WRVJULCBGQUxTRSk7CiAgICB9CgogICAgU2V0V2luZG93UG9zKGRhdGEtPmhXbmQsIDAsIDAsIDAsIHIucmlnaHQgLSByLmxlZnQgKyBkeCwgci5ib3R0b20gLSByLnRvcCArIGR5LAoJCSBTV1BfTk9NT1ZFfFNXUF9OT1pPUkRFUik7CiAgICBXQ1VTRVJfU2hhcGVDdXJzb3IoZGF0YSwgZGF0YS0+Y3VyY2ZnLmN1cnNvcl9zaXplLCBkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUsIFRSVUUpOwogICAgV0NVU0VSX1Bvc0N1cnNvcihkYXRhKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1NldFRpdGxlCiAqCiAqIFNldHMgdGhlIHRpdGxlIHRvIHRoZSB3aW5lIGNvbnNvbGUKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9TZXRUaXRsZShjb25zdCBzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSkKewogICAgV0NIQVIJYnVmZmVyWzI1Nl07CgogICAgaWYgKFdJTkVDT05fR2V0Q29uc29sZVRpdGxlKGRhdGEtPmhDb25JbiwgYnVmZmVyLCBzaXplb2YoYnVmZmVyKSkpCglTZXRXaW5kb3dUZXh0KGRhdGEtPmhXbmQsIGJ1ZmZlcik7Cn0KCnZvaWQgV0NVU0VSX0R1bXBMb2dGb250KGNvbnN0IGNoYXIqIHBmeCwgY29uc3QgTE9HRk9OVCogbGYsIERXT1JEIGZ0KQp7CiAgICBXSU5FX1RSQUNFXyh3Y19mb250KSgiJXMgJXMlcyVzJXNcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdGxmLmxmSGVpZ2h0PSVkIGxmLmxmV2lkdGg9JWQgbGYubGZFc2NhcGVtZW50PSVkIGxmLmxmT3JpZW50YXRpb249JWRcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdGxmLmxmV2VpZ2h0PSVkIGxmLmxmSXRhbGljPSV1IGxmLmxmVW5kZXJsaW5lPSV1IGxmLmxmU3RyaWtlT3V0PSV1XG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAiXHRsZi5sZkNoYXJTZXQ9JXUgbGYubGZPdXRQcmVjaXNpb249JXUgbGYubGZDbGlwUHJlY2lzaW9uPSV1IGxmLmxmUXVhbGl0eT0ldVxuIgogICAgICAgICAgICAgICAgICAgICAgICAgIlx0bGYtPmxmUGl0Y2hBbmRGYW1pbHk9JXUgbGYubGZGYWNlTmFtZT0lc1xuIiwKICAgICAgICAgICAgICAgICAgICAgICAgIHBmeCwKICAgICAgICAgICAgICAgICAgICAgICAgIChmdCAmIFJBU1RFUl9GT05UVFlQRSkgPyAicmFzdGVyIiA6ICIiLAogICAgICAgICAgICAgICAgICAgICAgICAgKGZ0ICYgVFJVRVRZUEVfRk9OVFRZUEUpID8gInRydWV0eXBlIiA6ICIiLAogICAgICAgICAgICAgICAgICAgICAgICAgKChmdCAmIChSQVNURVJfRk9OVFRZUEV8VFJVRVRZUEVfRk9OVFRZUEUpKSA9PSAwKSA/ICJ2ZWN0b3IiIDogIiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAoZnQgJiBERVZJQ0VfRk9OVFRZUEUpID8gInxkZXZpY2UiIDogIiIsCiAgICAgICAgICAgICAgICAgICAgICAgICBsZi0+bGZIZWlnaHQsIGxmLT5sZldpZHRoLCBsZi0+bGZFc2NhcGVtZW50LCBsZi0+bGZPcmllbnRhdGlvbiwKICAgICAgICAgICAgICAgICAgICAgICAgIGxmLT5sZldlaWdodCwgbGYtPmxmSXRhbGljLCBsZi0+bGZVbmRlcmxpbmUsIGxmLT5sZlN0cmlrZU91dCwgbGYtPmxmQ2hhclNldCwKICAgICAgICAgICAgICAgICAgICAgICAgIGxmLT5sZk91dFByZWNpc2lvbiwgbGYtPmxmQ2xpcFByZWNpc2lvbiwgbGYtPmxmUXVhbGl0eSwgbGYtPmxmUGl0Y2hBbmRGYW1pbHksCiAgICAgICAgICAgICAgICAgICAgICAgICB3aW5lX2RiZ3N0cl93KGxmLT5sZkZhY2VOYW1lKSk7Cn0KCnZvaWQgV0NVU0VSX0R1bXBUZXh0TWV0cmljKGNvbnN0IFRFWFRNRVRSSUMqIHRtLCBEV09SRCBmdCkKewogICAgV0lORV9UUkFDRV8od2NfZm9udCkoIiVzJXMlcyVzXG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAiXHR0bUhlaWdodD0lZCB0bUFzY2VudD0lZCB0bURlc2NlbnQ9JWQgdG1JbnRlcm5hbExlYWRpbmc9JWQgdG1FeHRlcm5hbExlYWRpbmc9JWRcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdHRtQXZlQ2hhcldpZHRoPSVkIHRtTWF4Q2hhcldpZHRoPSVkIHRtV2VpZ2h0PSVkIHRtT3Zlcmhhbmc9JWRcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdHRtRGlnaXRpemVkQXNwZWN0WD0lZCB0bURpZ2l0aXplZEFzcGVjdFk9JWRcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdHRtRmlyc3RDaGFyPSVkIHRtTGFzdENoYXI9JWQgdG1EZWZhdWx0Q2hhcj0lZCB0bUJyZWFrQ2hhcj0lZFxuIgogICAgICAgICAgICAgICAgICAgICAgICAgIlx0dG1JdGFsaWM9JXUgdG1VbmRlcmxpbmVkPSV1IHRtU3RydWNrT3V0PSV1IHRtUGl0Y2hBbmRGYW1pbHk9JXUgdG1DaGFyU2V0PSV1XG4iLAogICAgICAgICAgICAgICAgICAgICAgICAgKGZ0ICYgUkFTVEVSX0ZPTlRUWVBFKSA/ICJyYXN0ZXIiIDogIiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAoZnQgJiBUUlVFVFlQRV9GT05UVFlQRSkgPyAidHJ1ZXR5cGUiIDogIiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAoKGZ0ICYgKFJBU1RFUl9GT05UVFlQRXxUUlVFVFlQRV9GT05UVFlQRSkpID09IDApID8gInZlY3RvciIgOiAiIiwKICAgICAgICAgICAgICAgICAgICAgICAgIChmdCAmIERFVklDRV9GT05UVFlQRSkgPyAifGRldmljZSIgOiAiIiwKICAgICAgICAgICAgICAgICAgICAgICAgIHRtLT50bUhlaWdodCwgdG0tPnRtQXNjZW50LCB0bS0+dG1EZXNjZW50LCB0bS0+dG1JbnRlcm5hbExlYWRpbmcsIHRtLT50bUV4dGVybmFsTGVhZGluZywgdG0tPnRtQXZlQ2hhcldpZHRoLAogICAgICAgICAgICAgICAgICAgICAgICAgdG0tPnRtTWF4Q2hhcldpZHRoLCB0bS0+dG1XZWlnaHQsIHRtLT50bU92ZXJoYW5nLCB0bS0+dG1EaWdpdGl6ZWRBc3BlY3RYLCB0bS0+dG1EaWdpdGl6ZWRBc3BlY3RZLAogICAgICAgICAgICAgICAgICAgICAgICAgdG0tPnRtRmlyc3RDaGFyLCB0bS0+dG1MYXN0Q2hhciwgdG0tPnRtRGVmYXVsdENoYXIsIHRtLT50bUJyZWFrQ2hhciwgdG0tPnRtSXRhbGljLCB0bS0+dG1VbmRlcmxpbmVkLCB0bS0+dG1TdHJ1Y2tPdXQsCiAgICAgICAgICAgICAgICAgICAgICAgICB0bS0+dG1QaXRjaEFuZEZhbWlseSwgdG0tPnRtQ2hhclNldCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9BcmVGb250c0VxdWFsCiAqCiAqCiAqLwpCT09MIFdDVVNFUl9BcmVGb250c0VxdWFsKGNvbnN0IHN0cnVjdCBjb25maWdfZGF0YSogY29uZmlnLCBjb25zdCBMT0dGT05UKiBsZikKewogICAgcmV0dXJuIGxmLT5sZkhlaWdodCA9PSBjb25maWctPmNlbGxfaGVpZ2h0ICYmCiAgICAgICAgbGYtPmxmV2VpZ2h0ID09IGNvbmZpZy0+Zm9udF93ZWlnaHQgJiYKICAgICAgICAhbGYtPmxmSXRhbGljICYmICFsZi0+bGZVbmRlcmxpbmUgJiYgIWxmLT5sZlN0cmlrZU91dCAmJgogICAgICAgICFsc3RyY21wKGxmLT5sZkZhY2VOYW1lLCBjb25maWctPmZhY2VfbmFtZSk7Cn0KCnN0cnVjdCBmb250X2Nob29zZXIKewogICAgc3RydWN0IGlubmVyX2RhdGEqCWRhdGE7CiAgICBpbnQJCQlkb25lOwp9OwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1ZhbGlkYXRlRm9udE1ldHJpYwogKgogKiBSZXR1cm5zIHRydWUgaWYgdGhlIGZvbnQgZGVzY3JpYmVkIGluIHRtIGlzIHVzYWJsZSBhcyBhIGZvbnQgZm9yIHRoZSByZW5kZXJlcgogKi8KQk9PTAlXQ1VTRVJfVmFsaWRhdGVGb250TWV0cmljKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBjb25zdCBURVhUTUVUUklDKiB0bSwgRFdPUkQgZm9udFR5cGUpCnsKICAgIEJPT0wgICAgICAgIHJldCA9IFRSVUU7CgogICAgaWYgKGZvbnRUeXBlICYgUkFTVEVSX0ZPTlRUWVBFKQogICAgICAgIHJldCA9ICh0bS0+dG1NYXhDaGFyV2lkdGggKiBkYXRhLT5jdXJjZmcud2luX3dpZHRoIDwgR2V0U3lzdGVtTWV0cmljcyhTTV9DWFNDUkVFTikgJiYKICAgICAgICAgICAgICAgdG0tPnRtSGVpZ2h0ICogZGF0YS0+Y3VyY2ZnLndpbl9oZWlnaHQgPCBHZXRTeXN0ZW1NZXRyaWNzKFNNX0NZU0NSRUVOKSk7CiAgICByZXR1cm4gcmV0ICYmICF0bS0+dG1JdGFsaWMgJiYgIXRtLT50bVVuZGVybGluZWQgJiYgIXRtLT50bVN0cnVja091dCAmJgogICAgICAgICh0bS0+dG1DaGFyU2V0ID09IERFRkFVTFRfQ0hBUlNFVCB8fCB0bS0+dG1DaGFyU2V0ID09IGdfdWlEZWZhdWx0Q2hhcnNldCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9WYWxpZGF0ZUZvbnQKICoKICogUmV0dXJucyB0cnVlIGlmIHRoZSBmb250IGZhbWlseSBkZXNjcmliZWQgaW4gbGYgaXMgdXNhYmxlIGFzIGEgZm9udCBmb3IgdGhlIHJlbmRlcmVyCiAqLwpCT09MCVdDVVNFUl9WYWxpZGF0ZUZvbnQoY29uc3Qgc3RydWN0IGlubmVyX2RhdGEqIGRhdGEsIGNvbnN0IExPR0ZPTlQqIGxmKQp7CiAgICByZXR1cm4gKGxmLT5sZlBpdGNoQW5kRmFtaWx5ICYgMykgPT0gRklYRURfUElUQ0ggJiYKICAgICAgICAvKiAobGYtPmxmUGl0Y2hBbmRGYW1pbHkgJiAweEYwKSA9PSBGRl9NT0RFUk4gJiYgKi8KICAgICAgICAobGYtPmxmQ2hhclNldCA9PSBERUZBVUxUX0NIQVJTRVQgfHwgbGYtPmxmQ2hhclNldCA9PSBnX3VpRGVmYXVsdENoYXJzZXQpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfZmlyc3RfZm9udF9lbnVtXzIKICoJCWdldF9maXJzdF9mb250X2VudW0KICoKICogSGVscGVyIGZ1bmN0aW9ucyB0byBnZXQgYSBkZWNlbnQgZm9udCBmb3IgdGhlIHJlbmRlcmVyCiAqLwpzdGF0aWMgaW50IENBTExCQUNLIGdldF9maXJzdF9mb250X2VudW1fMihjb25zdCBMT0dGT05UKiBsZiwgY29uc3QgVEVYVE1FVFJJQyogdG0sCgkJCQkJICBEV09SRCBGb250VHlwZSwgTFBBUkFNIGxQYXJhbSkKewogICAgc3RydWN0IGZvbnRfY2hvb3NlcioJZmMgPSAoc3RydWN0IGZvbnRfY2hvb3NlciopbFBhcmFtOwoKICAgIFdDVVNFUl9EdW1wVGV4dE1ldHJpYyh0bSwgRm9udFR5cGUpOwogICAgaWYgKFdDVVNFUl9WYWxpZGF0ZUZvbnRNZXRyaWMoZmMtPmRhdGEsIHRtLCBGb250VHlwZSkpCiAgICB7CiAgICAgICAgTE9HRk9OVCBtbGYgPSAqbGY7CgogICAgICAgIC8qIFVzZSB0aGUgZGVmYXVsdCBzaXplcyBmb3IgdGhlIGZvbnQgKHRoaXMgaXMgbmVlZGVkLCBlc3BlY2lhbGx5IGZvcgogICAgICAgICAqIFRydWVUeXBlIGZvbnRzLCBzbyB0aGF0IHdlIGdldCBhIGRlY2VudCBzaXplLCBub3QgdGhlIG1heCBzaXplKQogICAgICAgICAqLwogICAgICAgIG1sZi5sZldpZHRoICA9IGZjLT5kYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgICAgICBtbGYubGZIZWlnaHQgPSBmYy0+ZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0OwogICAgICAgIGlmIChXQ1VTRVJfU2V0Rm9udChmYy0+ZGF0YSwgJm1sZikpCiAgICAgICAgewogICAgICAgICAgICBzdHJ1Y3QgICAgICBjb25maWdfZGF0YSAgICAgZGVmY2ZnOwoKICAgICAgICAgICAgV0NVU0VSX0R1bXBMb2dGb250KCJJbml0Q2hvb3Npbmc6ICIsICZtbGYsIEZvbnRUeXBlKTsKICAgICAgICAgICAgZmMtPmRvbmUgPSAxOwogICAgICAgICAgICAvKiBzaW5jZSB3ZSd2ZSBtb2RpZmllZCB0aGUgY3VycmVudCBjb25maWcgd2l0aCBuZXcgZm9udCBpbmZvcm1hdGlvbiwKICAgICAgICAgICAgICogc2V0IHRoaXMgaW5mb3JtYXRpb24gYXMgdGhlIG5ldyBkZWZhdWx0LgogICAgICAgICAgICAgKi8KICAgICAgICAgICAgV0lORUNPTl9SZWdMb2FkKE5VTEwsICZkZWZjZmcpOwogICAgICAgICAgICBkZWZjZmcuY2VsbF93aWR0aCA9IGZjLT5kYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgICAgICAgICAgZGVmY2ZnLmNlbGxfaGVpZ2h0ID0gZmMtPmRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodDsKICAgICAgICAgICAgbHN0cmNweVcoZGVmY2ZnLmZhY2VfbmFtZSwgZmMtPmRhdGEtPmN1cmNmZy5mYWNlX25hbWUpOwogICAgICAgICAgICAvKiBGb3JjZSBhbHNvIGl0cyB3cml0aW5nIGJhY2sgdG8gdGhlIHJlZ2lzdHJ5IHNvIHRoYXQgd2UgY2FuIGdldCBpdAogICAgICAgICAgICAgKiB0aGUgbmV4dCB0aW1lLgogICAgICAgICAgICAgKi8KICAgICAgICAgICAgV0lORUNPTl9SZWdTYXZlKCZkZWZjZmcpOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICB9CiAgICByZXR1cm4gMTsKfQoKc3RhdGljIGludCBDQUxMQkFDSyBnZXRfZmlyc3RfZm9udF9lbnVtKGNvbnN0IExPR0ZPTlQqIGxmLCBjb25zdCBURVhUTUVUUklDKiB0bSwKCQkJCQlEV09SRCBGb250VHlwZSwgTFBBUkFNIGxQYXJhbSkKewogICAgc3RydWN0IGZvbnRfY2hvb3NlcioJZmMgPSAoc3RydWN0IGZvbnRfY2hvb3NlciopbFBhcmFtOwoKICAgIFdDVVNFUl9EdW1wTG9nRm9udCgiSW5pdEZhbWlseTogIiwgbGYsIEZvbnRUeXBlKTsKICAgIGlmIChXQ1VTRVJfVmFsaWRhdGVGb250KGZjLT5kYXRhLCBsZikpCiAgICB7CglFbnVtRm9udEZhbWlsaWVzKFBSSVZBVEUoZmMtPmRhdGEpLT5oTWVtREMsIGxmLT5sZkZhY2VOYW1lLAogICAgICAgICAgICAgICAgICAgICAgICAgZ2V0X2ZpcnN0X2ZvbnRfZW51bV8yLCBsUGFyYW0pOwoJcmV0dXJuICFmYy0+ZG9uZTsgLyogd2UganVzdCBuZWVkIHRoZSBmaXJzdCBtYXRjaGluZyBvbmUuLi4gKi8KICAgIH0KICAgIHJldHVybiAxOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfQ29weUZvbnQKICoKICogZ2V0IHRoZSByZWxldmFudCBpbmZvcm1hdGlvbiBmcm9tIHRoZSBmb250IGRlc2NyaWJlZCBpbiBsZiBhbmQgc3RvcmUgdGhlbQogKiBpbiBjb25maWcKICovCkhGT05UIFdDVVNFUl9Db3B5Rm9udChzdHJ1Y3QgY29uZmlnX2RhdGEqIGNvbmZpZywgSFdORCBoV25kLCBjb25zdCBMT0dGT05UKiBsZiwgTE9ORyogZWwpCnsKICAgIFRFWFRNRVRSSUMgIHRtOwogICAgSERDICAgICAgICAgaERDOwogICAgSEZPTlQgICAgICAgaEZvbnQsIGhPbGRGb250OwogICAgaW50ICAgICAgICAgdywgaSwgYnVmWzI1Nl07CgogICAgaWYgKCEoaERDID0gR2V0REMoaFduZCkpKSByZXR1cm4gTlVMTDsKICAgIGlmICghKGhGb250ID0gQ3JlYXRlRm9udEluZGlyZWN0KGxmKSkpIGdvdG8gZXJyMTsKCiAgICBoT2xkRm9udCA9IFNlbGVjdE9iamVjdChoREMsIGhGb250KTsKICAgIEdldFRleHRNZXRyaWNzKGhEQywgJnRtKTsKCiAgICAvKiBGSVhNRToKICAgICAqIHRoZSBjdXJyZW50IGZyZWV0eXBlIGVuZ2luZSAoYXQgbGVhc3QgMi4wLnggd2l0aCB4IDw9IDgpIGFuZCBpdHMgaW1wbGVtZW50YXRpb24KICAgICAqIGluIFdpbmUgZG9uJ3QgcmV0dXJuIGFkZXF1YXRlIHZhbHVlcyBmb3IgZml4ZWQgZm9udHMKICAgICAqIEluIFdpbmRvd3MsIHRob3NlIGZvbnRzIGFyZSBleHBlY3RlZCB0byByZXR1cm4gdGhlIHNhbWUgdmFsdWUgZm9yCiAgICAgKiAgLSB0aGUgYXZlcmFnZSB3aWR0aAogICAgICogIC0gdGhlIGxhcmdlc3Qgd2lkdGgKICAgICAqICAtIHRoZSB3aWR0aCBvZiBhbGwgY2hhcmFjdGVycyBpbiB0aGUgZm9udAogICAgICogVGhpcyBpc24ndCB0cnVlIGluIFdpbmUuIEFzIGEgdGVtcG9yYXJ5IHdvcmthcm91bmQsIHdlIGdldCBhcyB0aGUgd2lkdGggb2YgdGhlCiAgICAgKiBjZWxsLCB0aGUgd2lkdGggb2YgdGhlIGZpcnN0IGNoYXJhY3RlciBpbiB0aGUgZm9udCwgYWZ0ZXIgY2hlY2tpbmcgdGhhdCBhbGwKICAgICAqIGNoYXJhY3RlcnMgaW4gdGhlIGZvbnQgaGF2ZSB0aGUgc2FtZSB3aWR0aCAoSSBoZWFyIHBhcmFub+9hIGNvbWluZykKICAgICAqIHdoZW4gdGhpcyBnZXRzIGZpeGVkLCB0aGUgY29kZSBzaG91bGQgYmUgdXNpbmcgdG0udG1BdmVDaGFyV2lkdGgKICAgICAqIG9yIHRtLnRtTWF4Q2hhcldpZHRoIGFzIHRoZSBjZWxsIHdpZHRoLgogICAgICovCiAgICBHZXRDaGFyV2lkdGgzMihoREMsIHRtLnRtRmlyc3RDaGFyLCB0bS50bUZpcnN0Q2hhciwgJncpOwogICAgZm9yIChpID0gdG0udG1GaXJzdENoYXIgKyAxOyBpIDw9IHRtLnRtTGFzdENoYXI7IGkgKz0gc2l6ZW9mKGJ1ZikgLyBzaXplb2YoYnVmWzBdKSkKICAgIHsKICAgICAgICBpbnQgaiwgazsKCiAgICAgICAgayA9IG1pbih0bS50bUxhc3RDaGFyIC0gaSwgc2l6ZW9mKGJ1ZikgLyBzaXplb2YoYnVmWzBdKSAtIDEpOwogICAgICAgIEdldENoYXJXaWR0aDMyKGhEQywgaSwgaSArIGssIGJ1Zik7CiAgICAgICAgZm9yIChqID0gMDsgaiA8PSBrOyBqKyspCiAgICAgICAgewogICAgICAgICAgICBpZiAoYnVmW2pdICE9IHcpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFdJTkVfV0FSTigiTm9uIHVuaWZvcm0gY2VsbCB3aWR0aDogWyVkXT0lZCBbJWRdPSVkXG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAgIlRoaXMgbWF5IGJlIGNhdXNlZCBieSBvbGQgZnJlZXR5cGUgbGlicmFyaWVzLCA+PSAyLjAuOCBpcyByZWNvbW1lbmRlZFxuIiwKICAgICAgICAgICAgICAgICAgICAgICAgICBpICsgaiwgYnVmW2pdLCB0bS50bUZpcnN0Q2hhciwgdyk7CiAgICAgICAgICAgICAgICBnb3RvIGVycjsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KICAgIFNlbGVjdE9iamVjdChoREMsIGhPbGRGb250KTsKICAgIFJlbGVhc2VEQyhoV25kLCBoREMpOwoKICAgIGNvbmZpZy0+Y2VsbF93aWR0aCAgPSB3OwogICAgY29uZmlnLT5jZWxsX2hlaWdodCA9IHRtLnRtSGVpZ2h0ICsgdG0udG1FeHRlcm5hbExlYWRpbmc7CiAgICBjb25maWctPmZvbnRfd2VpZ2h0ID0gdG0udG1XZWlnaHQ7CiAgICBsc3RyY3B5KGNvbmZpZy0+ZmFjZV9uYW1lLCBsZi0+bGZGYWNlTmFtZSk7CiAgICBpZiAoZWwpICplbCA9IHRtLnRtRXh0ZXJuYWxMZWFkaW5nOwoKICAgIHJldHVybiBoRm9udDsKIGVycjoKICAgIGlmIChoREMgJiYgaE9sZEZvbnQpIFNlbGVjdE9iamVjdChoREMsIGhPbGRGb250KTsKICAgIGlmIChoRm9udCkgRGVsZXRlT2JqZWN0KGhGb250KTsKIGVycjE6CiAgICBpZiAoaERDKSBSZWxlYXNlREMoaFduZCwgaERDKTsKCiAgICByZXR1cm4gTlVMTDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0ZpbGxMb2dGb250CiAqCiAqCiAqLwp2b2lkICAgIFdDVVNFUl9GaWxsTG9nRm9udChMT0dGT05UKiBsZiwgY29uc3QgV0NIQVIqIG5hbWUsIFVJTlQgaGVpZ2h0LCBVSU5UIHdlaWdodCkKewogICAgbGYtPmxmSGVpZ2h0ICAgICAgICA9IGhlaWdodDsKICAgIGxmLT5sZldpZHRoICAgICAgICAgPSAwOwogICAgbGYtPmxmRXNjYXBlbWVudCAgICA9IDA7CiAgICBsZi0+bGZPcmllbnRhdGlvbiAgID0gMDsKICAgIGxmLT5sZldlaWdodCAgICAgICAgPSB3ZWlnaHQ7CiAgICBsZi0+bGZJdGFsaWMgICAgICAgID0gRkFMU0U7CiAgICBsZi0+bGZVbmRlcmxpbmUgICAgID0gRkFMU0U7CiAgICBsZi0+bGZTdHJpa2VPdXQgICAgID0gRkFMU0U7CiAgICBsZi0+bGZDaGFyU2V0ICAgICAgID0gREVGQVVMVF9DSEFSU0VUOwogICAgbGYtPmxmT3V0UHJlY2lzaW9uICA9IE9VVF9ERUZBVUxUX1BSRUNJUzsKICAgIGxmLT5sZkNsaXBQcmVjaXNpb24gPSBDTElQX0RFRkFVTFRfUFJFQ0lTOwogICAgbGYtPmxmUXVhbGl0eSAgICAgICA9IERFRkFVTFRfUVVBTElUWTsKICAgIGxmLT5sZlBpdGNoQW5kRmFtaWx5ID0gRklYRURfUElUQ0ggfCBGRl9ET05UQ0FSRTsKICAgIGxzdHJjcHkobGYtPmxmRmFjZU5hbWUsIG5hbWUpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfU2V0Rm9udAogKgogKiBzZXRzIGxvZ2ZvbnQgYXMgdGhlIG5ldyBmb250IGZvciB0aGUgY29uc29sZQogKi8KQk9PTAlXQ1VTRVJfU2V0Rm9udChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgY29uc3QgTE9HRk9OVCogbG9nZm9udCkKewogICAgSEZPTlQgICAgICAgaEZvbnQ7CiAgICBMT05HICAgICAgICBlbDsKCiAgICBpZiAoUFJJVkFURShkYXRhKS0+aEZvbnQgIT0gMCAmJiBXQ1VTRVJfQXJlRm9udHNFcXVhbCgmZGF0YS0+Y3VyY2ZnLCBsb2dmb250KSkKICAgICAgICByZXR1cm4gVFJVRTsKCiAgICBoRm9udCA9IFdDVVNFUl9Db3B5Rm9udCgmZGF0YS0+Y3VyY2ZnLCBkYXRhLT5oV25kLCBsb2dmb250LCAmZWwpOwogICAgaWYgKCFoRm9udCkge1dJTkVfRVJSKCJ3cm9uZyBmb250XG4iKTsgcmV0dXJuIEZBTFNFO30KCiAgICBpZiAoUFJJVkFURShkYXRhKS0+aEZvbnQpIERlbGV0ZU9iamVjdChQUklWQVRFKGRhdGEpLT5oRm9udCk7CiAgICBQUklWQVRFKGRhdGEpLT5oRm9udCA9IGhGb250OwogICAgUFJJVkFURShkYXRhKS0+ZXh0X2xlYWRpbmcgPSBlbDsKCiAgICBXQ1VTRVJfQ29tcHV0ZVBvc2l0aW9ucyhkYXRhKTsKICAgIFdDVVNFUl9OZXdCaXRtYXAoZGF0YSk7CiAgICBJbnZhbGlkYXRlUmVjdChkYXRhLT5oV25kLCBOVUxMLCBGQUxTRSk7CiAgICBVcGRhdGVXaW5kb3coZGF0YS0+aFduZCk7CgogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9TZXRGb250UG10CiAqCiAqIFNldHMgYSBuZXcgZm9udCBmb3IgdGhlIGNvbnNvbGUuCiAqIEluIGZhY3QgYSB3cmFwcGVyIGZvciBXQ1VTRVJfU2V0Rm9udAogKi8Kc3RhdGljIHZvaWQgICAgIFdDVVNFUl9TZXRGb250UG10KHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBjb25zdCBXQ0hBUiogZm9udCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVuc2lnbmVkIGhlaWdodCwgdW5zaWduZWQgd2VpZ2h0KQp7CiAgICBMT0dGT05UICAgICAgICAgICAgIGxmOwogICAgc3RydWN0IGZvbnRfY2hvb3NlciBmYzsKCiAgICBXSU5FX1RSQUNFXyh3Y19mb250KSgiPT4gJXMgaD0ldSB3PSV1XG4iLAogICAgICAgICAgICAgICAgICAgICAgICAgd2luZV9kYmdzdHJfd24oZm9udCwgLTEpLCBoZWlnaHQsIHdlaWdodCk7CgogICAgaWYgKGZvbnRbMF0gIT0gJ1wwJyAmJiBoZWlnaHQgIT0gMCAmJiB3ZWlnaHQgIT0gMCkKICAgIHsKICAgICAgICBXQ1VTRVJfRmlsbExvZ0ZvbnQoJmxmLCBmb250LCBoZWlnaHQsIHdlaWdodCk7CiAgICAgICAgaWYgKFdDVVNFUl9TZXRGb250KGRhdGEsICZsZikpCiAgICAgICAgewogICAgICAgICAgICBXQ1VTRVJfRHVtcExvZ0ZvbnQoIkluaXRSZXVzZXM6ICIsICZsZiwgMCk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgLyogdHJ5IHRvIGZpbmQgYW4gYWNjZXB0YWJsZSBmb250ICovCiAgICBXSU5FX1dBUk4oIkNvdWxkbid0IG1hdGNoIHRoZSBmb250IGZyb20gcmVnaXN0cnkuLi4gdHJ5aW5nIHRvIGZpbmQgb25lXG4iKTsKICAgIGZjLmRhdGEgPSBkYXRhOwogICAgZmMuZG9uZSA9IDA7CiAgICBFbnVtRm9udEZhbWlsaWVzKFBSSVZBVEUoZGF0YSktPmhNZW1EQywgTlVMTCwgZ2V0X2ZpcnN0X2ZvbnRfZW51bSwgKExQQVJBTSkmZmMpOwogICAgaWYgKCFmYy5kb25lKSBXSU5FQ09OX0ZhdGFsKCJDb3VsZG4ndCBmaW5kIGEgZGVjZW50IGZvbnQsIGFib3J0aW5nXG4iKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0dldENlbGwKICoKICogR2V0IGEgY2VsbCBmcm9tIGEgcmVsYXRpdmUgY29vcmRpbmF0ZSBpbiB3aW5kb3cgKHRha2VzIGludG8KICogYWNjb3VudCB0aGUgc2Nyb2xsaW5nKQogKi8Kc3RhdGljIENPT1JECVdDVVNFUl9HZXRDZWxsKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBMUEFSQU0gbFBhcmFtKQp7CiAgICBDT09SRAljOwoKICAgIGMuWCA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlggKyAoc2hvcnQpTE9XT1JEKGxQYXJhbSkgLyBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgIGMuWSA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKyAoc2hvcnQpSElXT1JEKGxQYXJhbSkgLyBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7CgogICAgcmV0dXJuIGM7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9HZXRTZWxlY3Rpb25SZWN0CiAqCiAqIEdldCB0aGUgc2VsZWN0aW9uIHJlY3RhbmdsZQogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX0dldFNlbGVjdGlvblJlY3QoY29uc3Qgc3RydWN0IGlubmVyX2RhdGEqIGRhdGEsIExQUkVDVCByKQp7CiAgICByLT5sZWZ0ICAgPSAobWluKFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5YLCBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIuWCkgICAgIC0gZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCkgKiBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgIHItPnRvcCAgICA9IChtaW4oUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLlksIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0Mi5ZKSAgICAgLSBkYXRhLT5jdXJjZmcud2luX3Bvcy5ZKSAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodDsKICAgIHItPnJpZ2h0ICA9IChtYXgoUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLlgsIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0Mi5YKSArIDEgLSBkYXRhLT5jdXJjZmcud2luX3Bvcy5YKSAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoOwogICAgci0+Ym90dG9tID0gKG1heChQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWSwgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkpICsgMSAtIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfU2V0U2VsZWN0aW9uCiAqCiAqCiAqLwpzdGF0aWMgdm9pZAlXQ1VTRVJfU2V0U2VsZWN0aW9uKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBIREMgaFJlZkRDKQp7CiAgICBIREMJCWhEQzsKICAgIFJFQ1QJcjsKCiAgICBXQ1VTRVJfR2V0U2VsZWN0aW9uUmVjdChkYXRhLCAmcik7CiAgICBoREMgPSBoUmVmREMgPyBoUmVmREMgOiBHZXREQyhkYXRhLT5oV25kKTsKICAgIGlmIChoREMpCiAgICB7CglpZiAoZGF0YS0+aFduZCA9PSBHZXRGb2N1cygpICYmIGRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSkKCSAgICBIaWRlQ2FyZXQoZGF0YS0+aFduZCk7CglJbnZlcnRSZWN0KGhEQywgJnIpOwoJaWYgKGhEQyAhPSBoUmVmREMpCgkgICAgUmVsZWFzZURDKGRhdGEtPmhXbmQsIGhEQyk7CglpZiAoZGF0YS0+aFduZCA9PSBHZXRGb2N1cygpICYmIGRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSkKCSAgICBTaG93Q2FyZXQoZGF0YS0+aFduZCk7CiAgICB9Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9Nb3ZlU2VsZWN0aW9uCiAqCiAqCiAqLwpzdGF0aWMgdm9pZAlXQ1VTRVJfTW92ZVNlbGVjdGlvbihzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgQ09PUkQgYzEsIENPT1JEIGMyKQp7CiAgICBSRUNUCXI7CiAgICBIREMJCWhEQzsKCiAgICBpZiAoYzEuWCA8IDAgfHwgYzEuWCA+PSBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggfHwKICAgICAgICBjMi5YIDwgMCB8fCBjMi5YID49IGRhdGEtPmN1cmNmZy5zYl93aWR0aCB8fAogICAgICAgIGMxLlkgPCAwIHx8IGMxLlkgPj0gZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCB8fAogICAgICAgIGMyLlkgPCAwIHx8IGMyLlkgPj0gZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCkKICAgICAgICByZXR1cm47CgogICAgV0NVU0VSX0dldFNlbGVjdGlvblJlY3QoZGF0YSwgJnIpOwogICAgaERDID0gR2V0REMoZGF0YS0+aFduZCk7CiAgICBpZiAoaERDKQogICAgewoJaWYgKGRhdGEtPmhXbmQgPT0gR2V0Rm9jdXMoKSAmJiBkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUpCgkgICAgSGlkZUNhcmV0KGRhdGEtPmhXbmQpOwoJSW52ZXJ0UmVjdChoREMsICZyKTsKICAgIH0KICAgIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MSA9IGMxOwogICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyID0gYzI7CiAgICBpZiAoaERDKQogICAgewoJV0NVU0VSX0dldFNlbGVjdGlvblJlY3QoZGF0YSwgJnIpOwoJSW52ZXJ0UmVjdChoREMsICZyKTsKCVJlbGVhc2VEQyhkYXRhLT5oV25kLCBoREMpOwoJaWYgKGRhdGEtPmhXbmQgPT0gR2V0Rm9jdXMoKSAmJiBkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUpCgkgICAgU2hvd0NhcmV0KGRhdGEtPmhXbmQpOwogICAgfQp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfQ29weVNlbGVjdGlvblRvQ2xpcGJvYXJkCiAqCiAqIENvcGllcyB0aGUgY3VycmVudCBzZWxlY3Rpb24gaW50byB0aGUgY2xpcGJvYXJkCiAqLwpzdGF0aWMgdm9pZAlXQ1VTRVJfQ29weVNlbGVjdGlvblRvQ2xpcGJvYXJkKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBIQU5ETEUJaE1lbTsKICAgIExQV1NUUglwOwogICAgdW5zaWduZWQJdywgaDsKCiAgICB3ID0gYWJzKFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5YIC0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlgpICsgMjsKICAgIGggPSBhYnMoUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLlkgLSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIuWSkgKyAxOwoKICAgIGlmICghT3BlbkNsaXBib2FyZChkYXRhLT5oV25kKSkgcmV0dXJuOwogICAgRW1wdHlDbGlwYm9hcmQoKTsKCiAgICBoTWVtID0gR2xvYmFsQWxsb2MoR01FTV9NT1ZFQUJMRSwgKHcgKiBoKSAqIHNpemVvZihXQ0hBUikpOwogICAgaWYgKGhNZW0gJiYgKHAgPSBHbG9iYWxMb2NrKGhNZW0pKSkKICAgIHsKCUNPT1JECWM7CglpbnQJeTsKCgljLlggPSBtaW4oUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLlgsIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0Mi5YKTsKCWMuWSA9IG1pbihQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWSwgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkpOwoKCWZvciAoeSA9IDA7IHkgPCBoOyB5KyssIGMuWSsrKQoJewoJICAgIExQV1NUUiBlbmQ7CgkgICAgUmVhZENvbnNvbGVPdXRwdXRDaGFyYWN0ZXIoZGF0YS0+aENvbk91dCwgcCwgdyAtIDEsIGMsIE5VTEwpOwoJICAgIAoJICAgIC8qIHN0cmlwIHNwYWNlcyBmcm9tIHRoZSBlbmQgb2YgdGhlIGxpbmUgKi8KCSAgICBlbmQgPSBwICsgdyAtIDE7CgkgICAgd2hpbGUgKGVuZCA+IHAgJiYgKihlbmQgLSAxKSA9PSAnICcpCgkgICAgICAgIGVuZC0tOwoJICAgICplbmQgPSAoeSA8IGggLSAxKSA/ICdcbicgOiAnXDAnOwoJICAgIHAgPSBlbmQgKyAxOwoJfQoJR2xvYmFsVW5sb2NrKGhNZW0pOwoJU2V0Q2xpcGJvYXJkRGF0YShDRl9VTklDT0RFVEVYVCwgaE1lbSk7CiAgICB9CiAgICBDbG9zZUNsaXBib2FyZCgpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfUGFzdGVGcm9tQ2xpcGJvYXJkCiAqCiAqCiAqLwpzdGF0aWMgdm9pZAlXQ1VTRVJfUGFzdGVGcm9tQ2xpcGJvYXJkKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBIQU5ETEUJaDsKICAgIFdDSEFSKglwdHI7CgogICAgaWYgKCFPcGVuQ2xpcGJvYXJkKGRhdGEtPmhXbmQpKSByZXR1cm47CiAgICBoID0gR2V0Q2xpcGJvYXJkRGF0YShDRl9VTklDT0RFVEVYVCk7CiAgICBpZiAoaCAmJiAocHRyID0gR2xvYmFsTG9jayhoKSkpCiAgICB7CglpbnQJCWksIGxlbiA9IEdsb2JhbFNpemUoaCkgLyBzaXplb2YoV0NIQVIpOwoJSU5QVVRfUkVDT1JECWlyWzJdOwoJRFdPUkQJCW47CglTSE9SVAkJc2g7CgoJaXJbMF0uRXZlbnRUeXBlID0gS0VZX0VWRU5UOwoJaXJbMF0uRXZlbnQuS2V5RXZlbnQud1JlcGVhdENvdW50ID0gMDsKCWlyWzBdLkV2ZW50LktleUV2ZW50LmR3Q29udHJvbEtleVN0YXRlID0gMDsKCWlyWzBdLkV2ZW50LktleUV2ZW50LmJLZXlEb3duID0gVFJVRTsKCgkvKiBnZW5lcmF0ZSB0aGUgY29ycmVzcG9uZGluZyBpbnB1dCByZWNvcmRzICovCglmb3IgKGkgPSAwOyBpIDwgbGVuOyBpKyspCgl7CgkgICAgLyogRklYTUU6IHRoZSBtb2RpZnlpbmcga2V5cyBhcmUgbm90IGdlbmVyYXRlZCAoc2hpZnQsIGN0cmwuLi4pICovCgkgICAgc2ggPSBWa0tleVNjYW4ocHRyW2ldKTsKCSAgICBpclswXS5FdmVudC5LZXlFdmVudC53VmlydHVhbEtleUNvZGUgPSBMT0JZVEUoc2gpOwoJICAgIGlyWzBdLkV2ZW50LktleUV2ZW50LndWaXJ0dWFsU2NhbkNvZGUgPSBNYXBWaXJ0dWFsS2V5KExPQllURShzaCksIDApOwoJICAgIGlyWzBdLkV2ZW50LktleUV2ZW50LnVDaGFyLlVuaWNvZGVDaGFyID0gcHRyW2ldOwoKCSAgICBpclsxXSA9IGlyWzBdOwoJICAgIGlyWzFdLkV2ZW50LktleUV2ZW50LmJLZXlEb3duID0gRkFMU0U7CgoJICAgIFdyaXRlQ29uc29sZUlucHV0KGRhdGEtPmhDb25JbiwgaXIsIDIsICZuKTsKCX0KCUdsb2JhbFVubG9jayhoKTsKICAgIH0KICAgIENsb3NlQ2xpcGJvYXJkKCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9SZWZyZXNoCiAqCiAqCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfUmVmcmVzaChjb25zdCBzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgaW50IHRwLCBpbnQgYm0pCnsKICAgIFdDVVNFUl9GaWxsTWVtREMoZGF0YSwgdHAsIGJtKTsKICAgIGlmIChkYXRhLT5jdXJjZmcud2luX3Bvcy5ZIDw9IGJtICYmIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKyBkYXRhLT5jdXJjZmcud2luX2hlaWdodCA+PSB0cCkKICAgIHsKCVJFQ1QJcjsKCglyLmxlZnQgICA9IDA7CglyLnJpZ2h0ICA9IGRhdGEtPmN1cmNmZy53aW5fd2lkdGggKiBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKCXIudG9wICAgID0gKHRwIC0gZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWSkgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7CglyLmJvdHRvbSA9IChibSAtIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKyAxKSAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodDsKCUludmFsaWRhdGVSZWN0KGRhdGEtPmhXbmQsICZyLCBGQUxTRSk7CglVcGRhdGVXaW5kb3coZGF0YS0+aFduZCk7CiAgICB9Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9QYWludAogKgogKgogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX1BhaW50KGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBQQUlOVFNUUlVDVAkJcHM7CgogICAgQmVnaW5QYWludChkYXRhLT5oV25kLCAmcHMpOwogICAgQml0Qmx0KHBzLmhkYywgMCwgMCwKICAgICAgICAgICBkYXRhLT5jdXJjZmcud2luX3dpZHRoICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsCiAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLndpbl9oZWlnaHQgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQsCgkgICBQUklWQVRFKGRhdGEpLT5oTWVtREMsCiAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLAogICAgICAgICAgIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQsCgkgICBTUkNDT1BZKTsKICAgIGlmIChQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uKQoJV0NVU0VSX1NldFNlbGVjdGlvbihkYXRhLCBwcy5oZGMpOwogICAgRW5kUGFpbnQoZGF0YS0+aFduZCwgJnBzKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1Njcm9sbAogKgogKgogKi8Kc3RhdGljIHZvaWQgV0NVU0VSX1Njcm9sbChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgaW50IHBvcywgQk9PTCBob3J6KQp7CiAgICBpZiAoaG9yeikKICAgIHsKCVNldFNjcm9sbFBvcyhkYXRhLT5oV25kLCBTQl9IT1JaLCBwb3MsIFRSVUUpOwoJZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCA9IHBvczsKICAgIH0KICAgIGVsc2UKICAgIHsKCVNldFNjcm9sbFBvcyhkYXRhLT5oV25kLCBTQl9WRVJULCBwb3MsIFRSVUUpOwoJZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWSA9IHBvczsKICAgIH0KICAgIEludmFsaWRhdGVSZWN0KGRhdGEtPmhXbmQsIE5VTEwsIEZBTFNFKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0ZpbGxNZW51CiAqCiAqCiAqLwpzdGF0aWMgQk9PTCBXQ1VTRVJfRmlsbE1lbnUoSE1FTlUgaE1lbnUsIEJPT0wgc2VwKQp7CiAgICBITUVOVQkJaFN1Yk1lbnU7CiAgICBISU5TVEFOQ0UJCWhJbnN0YW5jZSA9IEdldE1vZHVsZUhhbmRsZShOVUxMKTsKICAgIFdDSEFSCQlidWZmWzI1Nl07CgogICAgaWYgKCFoTWVudSkgcmV0dXJuIEZBTFNFOwoKICAgIC8qIEZJWE1FOiBlcnJvciBoYW5kbGluZyAmIG1lbW9yeSBjbGVhbnVwICovCiAgICBoU3ViTWVudSA9IENyZWF0ZU1lbnUoKTsKICAgIGlmICghaFN1Yk1lbnUpIHJldHVybiBGQUxTRTsKCiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX01BUkssIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoU3ViTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfTUFSSywgYnVmZik7CiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX0NPUFksIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoU3ViTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfQ09QWSwgYnVmZik7CiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX1BBU1RFLCBidWZmLCBzaXplb2YoYnVmZikgLyBzaXplb2YoV0NIQVIpKTsKICAgIEluc2VydE1lbnUoaFN1Yk1lbnUsIC0xLCBNRl9CWVBPU0lUSU9OfE1GX1NUUklORywgSURTX1BBU1RFLCBidWZmKTsKICAgIExvYWRTdHJpbmcoaEluc3RhbmNlLCBJRFNfU0VMRUNUQUxMLCBidWZmLCBzaXplb2YoYnVmZikgLyBzaXplb2YoV0NIQVIpKTsKICAgIEluc2VydE1lbnUoaFN1Yk1lbnUsIC0xLCBNRl9CWVBPU0lUSU9OfE1GX1NUUklORywgSURTX1NFTEVDVEFMTCwgYnVmZik7CiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX1NDUk9MTCwgYnVmZiwgc2l6ZW9mKGJ1ZmYpIC8gc2l6ZW9mKFdDSEFSKSk7CiAgICBJbnNlcnRNZW51KGhTdWJNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TVFJJTkcsIElEU19TQ1JPTEwsIGJ1ZmYpOwogICAgTG9hZFN0cmluZyhoSW5zdGFuY2UsIElEU19TRUFSQ0gsIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoU3ViTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfU0VBUkNILCBidWZmKTsKCiAgICBpZiAoc2VwKSBJbnNlcnRNZW51KGhNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TRVBBUkFUT1IsIDAsIE5VTEwpOwogICAgTG9hZFN0cmluZyhoSW5zdGFuY2UsIElEU19FRElULCBidWZmLCBzaXplb2YoYnVmZikgLyBzaXplb2YoV0NIQVIpKTsKICAgIEluc2VydE1lbnUoaE1lbnUsIC0xLCBNRl9CWVBPU0lUSU9OfE1GX1NUUklOR3xNRl9QT1BVUCwgKFVJTlRfUFRSKWhTdWJNZW51LCBidWZmKTsKICAgIExvYWRTdHJpbmcoaEluc3RhbmNlLCBJRFNfREVGQVVMVCwgYnVmZiwgc2l6ZW9mKGJ1ZmYpIC8gc2l6ZW9mKFdDSEFSKSk7CiAgICBJbnNlcnRNZW51KGhNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TVFJJTkcsIElEU19ERUZBVUxULCBidWZmKTsKICAgIExvYWRTdHJpbmcoaEluc3RhbmNlLCBJRFNfUFJPUEVSVElFUywgYnVmZiwgc2l6ZW9mKGJ1ZmYpIC8gc2l6ZW9mKFdDSEFSKSk7CiAgICBJbnNlcnRNZW51KGhNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TVFJJTkcsIElEU19QUk9QRVJUSUVTLCBidWZmKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1NldE1lbnVEZXRhaWxzCiAqCiAqIEdyYXlzIC8gdW5ncmF5cyB0aGUgbWVudSBpdGVtcyBhY2NvcmRpbmcgdG8gdGhlaXIgc3RhdGUKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9TZXRNZW51RGV0YWlscyhjb25zdCBzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgSE1FTlUgaE1lbnUpCnsKICAgIGlmICghaE1lbnUpIHtXSU5FX0VSUigiSXNzdWUgaW4gZ2V0dGluZyBtZW51IGJpdHNcbiIpO3JldHVybjt9CgogICAgRW5hYmxlTWVudUl0ZW0oaE1lbnUsIElEU19DT1BZLAogICAgICAgICAgICAgICAgICAgTUZfQllDT01NQU5EfChQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID8gTUZfRU5BQkxFRCA6IE1GX0dSQVlFRCkpOwogICAgRW5hYmxlTWVudUl0ZW0oaE1lbnUsIElEU19QQVNURSwKCQkgICBNRl9CWUNPTU1BTkR8KElzQ2xpcGJvYXJkRm9ybWF0QXZhaWxhYmxlKENGX1VOSUNPREVURVhUKQoJCQkJID8gTUZfRU5BQkxFRCA6IE1GX0dSQVlFRCkpOwogICAgRW5hYmxlTWVudUl0ZW0oaE1lbnUsIElEU19TQ1JPTEwsIE1GX0JZQ09NTUFORHxNRl9HUkFZRUQpOwogICAgRW5hYmxlTWVudUl0ZW0oaE1lbnUsIElEU19TRUFSQ0gsIE1GX0JZQ09NTUFORHxNRl9HUkFZRUQpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfQ3JlYXRlCiAqCiAqIENyZWF0ZXMgdGhlIHdpbmRvdyBmb3IgdGhlIHJlbmRlcmluZwogKi8Kc3RhdGljIExSRVNVTFQgV0NVU0VSX0NyZWF0ZShIV05EIGhXbmQsIExQQ1JFQVRFU1RSVUNUIGxwY3MpCnsKICAgIHN0cnVjdCBpbm5lcl9kYXRhKglkYXRhOwogICAgSE1FTlUJCWhTeXNNZW51OwoKICAgIGRhdGEgPSBscGNzLT5scENyZWF0ZVBhcmFtczsKICAgIFNldFdpbmRvd0xvbmdQdHIoaFduZCwgMEwsIChEV09SRF9QVFIpZGF0YSk7CiAgICBkYXRhLT5oV25kID0gaFduZDsKCiAgICBoU3lzTWVudSA9IEdldFN5c3RlbU1lbnUoaFduZCwgRkFMU0UpOwogICAgaWYgKCFoU3lzTWVudSkgcmV0dXJuIDA7CiAgICBQUklWQVRFKGRhdGEpLT5oUG9wTWVudSA9IENyZWF0ZVBvcHVwTWVudSgpOwogICAgaWYgKCFQUklWQVRFKGRhdGEpLT5oUG9wTWVudSkgcmV0dXJuIDA7CgogICAgV0NVU0VSX0ZpbGxNZW51KGhTeXNNZW51LCBUUlVFKTsKICAgIFdDVVNFUl9GaWxsTWVudShQUklWQVRFKGRhdGEpLT5oUG9wTWVudSwgRkFMU0UpOwoKICAgIFBSSVZBVEUoZGF0YSktPmhNZW1EQyA9IENyZWF0ZUNvbXBhdGlibGVEQygwKTsKICAgIGlmICghUFJJVkFURShkYXRhKS0+aE1lbURDKSB7V0lORV9FUlIoIm5vIG1lbSBkY1xuIik7cmV0dXJuIDA7fQoKICAgIGRhdGEtPmN1cmNmZy5xdWlja19lZGl0ID0gRkFMU0U7CiAgICByZXR1cm4gMDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0dldEN0cmxLZXlTdGF0ZQogKgogKiBHZXQgdGhlIGNvbnNvbGUgYml0IG1hc2sgZXF1aXZhbGVudCB0byB0aGUgVktfIHN0YXR1cyBpbiBrZXlTdGF0ZQogKi8Kc3RhdGljIERXT1JEICAgIFdDVVNFUl9HZXRDdHJsS2V5U3RhdGUoQllURSoga2V5U3RhdGUpCnsKICAgIERXT1JEICAgICAgICAgICAgICAgcmV0ID0gMDsKCiAgICBHZXRLZXlib2FyZFN0YXRlKGtleVN0YXRlKTsKICAgIGlmIChrZXlTdGF0ZVtWS19TSElGVF0gICAgJiAweDgwKQlyZXQgfD0gU0hJRlRfUFJFU1NFRDsKICAgIGlmIChrZXlTdGF0ZVtWS19MQ09OVFJPTF0gJiAweDgwKQlyZXQgfD0gTEVGVF9DVFJMX1BSRVNTRUQ7CiAgICBpZiAoa2V5U3RhdGVbVktfUkNPTlRST0xdICYgMHg4MCkJcmV0IHw9IFJJR0hUX0NUUkxfUFJFU1NFRDsKICAgIGlmIChrZXlTdGF0ZVtWS19MTUVOVV0gICAgJiAweDgwKQlyZXQgfD0gTEVGVF9BTFRfUFJFU1NFRDsKICAgIGlmIChrZXlTdGF0ZVtWS19STUVOVV0gICAgJiAweDgwKQlyZXQgfD0gUklHSFRfQUxUX1BSRVNTRUQ7CiAgICBpZiAoa2V5U3RhdGVbVktfQ0FQSVRBTF0gICYgMHgwMSkJcmV0IHw9IENBUFNMT0NLX09OOwogICAgaWYgKGtleVN0YXRlW1ZLX05VTUxPQ0tdICAmIDB4MDEpCXJldCB8PSBOVU1MT0NLX09OOwogICAgaWYgKGtleVN0YXRlW1ZLX1NDUk9MTF0gICAmIDB4MDEpCXJldCB8PSBTQ1JPTExMT0NLX09OOwoKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9IYW5kbGVTZWxlY3Rpb25LZXkKICoKICogSGFuZGxlcyBrZXlzIHdoaWxlIHNlbGVjdGluZyBhbiBhcmVhCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfSGFuZGxlU2VsZWN0aW9uS2V5KHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBCT09MIGRvd24sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSkKewogICAgQllURQlrZXlTdGF0ZVsyNTZdOwogICAgRFdPUkQgICAgICAgc3RhdGUgPSBXQ1VTRVJfR2V0Q3RybEtleVN0YXRlKGtleVN0YXRlKSAmIH4oQ0FQU0xPQ0tfT058TlVNTE9DS19PTnxTQ1JPTExMT0NLX09OKTsKICAgIENPT1JEICAgICAgIGMxLCBjMjsKCiAgICBpZiAoIWRvd24pIHJldHVybjsKCiAgICBzd2l0Y2ggKHN0YXRlKQogICAgewogICAgY2FzZSAwOgogICAgICAgIHN3aXRjaCAod1BhcmFtKQogICAgICAgIHsKICAgICAgICBjYXNlIFZLX1JFVFVSTjoKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+aGFzX3NlbGVjdGlvbiA9IEZBTFNFOwogICAgICAgICAgICBXQ1VTRVJfU2V0U2VsZWN0aW9uKGRhdGEsIDApOwogICAgICAgICAgICBXQ1VTRVJfQ29weVNlbGVjdGlvblRvQ2xpcGJvYXJkKGRhdGEpOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgY2FzZSBWS19SSUdIVDoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMS5YKys7IGMyLlgrKzsKICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIGNhc2UgVktfTEVGVDoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMS5YLS07IGMyLlgtLTsKICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIGNhc2UgVktfVVA6CiAgICAgICAgICAgIGMxID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxOwogICAgICAgICAgICBjMiA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MjsKICAgICAgICAgICAgYzEuWS0tOyBjMi5ZLS07CiAgICAgICAgICAgIFdDVVNFUl9Nb3ZlU2VsZWN0aW9uKGRhdGEsIGMxLCBjMik7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICBjYXNlIFZLX0RPV046CiAgICAgICAgICAgIGMxID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxOwogICAgICAgICAgICBjMiA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MjsKICAgICAgICAgICAgYzEuWSsrOyBjMi5ZKys7CiAgICAgICAgICAgIFdDVVNFUl9Nb3ZlU2VsZWN0aW9uKGRhdGEsIGMxLCBjMik7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIFNISUZUX1BSRVNTRUQ6CiAgICAgICAgc3dpdGNoICh3UGFyYW0pCiAgICAgICAgewogICAgICAgIGNhc2UgVktfUklHSFQ6CiAgICAgICAgICAgIGMxID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxOwogICAgICAgICAgICBjMiA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MjsKICAgICAgICAgICAgYzIuWCsrOwogICAgICAgICAgICBXQ1VTRVJfTW92ZVNlbGVjdGlvbihkYXRhLCBjMSwgYzIpOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgY2FzZSBWS19MRUZUOgogICAgICAgICAgICBjMSA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MTsKICAgICAgICAgICAgYzIgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDI7CiAgICAgICAgICAgIGMyLlgtLTsKICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIGNhc2UgVktfVVA6CiAgICAgICAgICAgIGMxID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxOwogICAgICAgICAgICBjMiA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MjsKICAgICAgICAgICAgYzIuWS0tOwogICAgICAgICAgICBXQ1VTRVJfTW92ZVNlbGVjdGlvbihkYXRhLCBjMSwgYzIpOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgY2FzZSBWS19ET1dOOgogICAgICAgICAgICBjMSA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MTsKICAgICAgICAgICAgYzIgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDI7CiAgICAgICAgICAgIGMyLlkrKzsKICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAod1BhcmFtIDwgVktfU1BBQ0UpICAvKiBTaGlmdCwgQWx0LCBDdHJsLCBOdW0gTG9jayBldGMuICovCiAgICAgICAgcmV0dXJuOwogICAgCiAgICBXQ1VTRVJfU2V0U2VsZWN0aW9uKGRhdGEsIDApOyAgICAKICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBGQUxTRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0dlbmVyYXRlS2V5SW5wdXRSZWNvcmQKICoKICogZ2VuZXJhdGVzIGlucHV0X3JlY29yZCBmcm9tIHdpbmRvd3MgV01fS0VZVVAvV01fS0VZRE9XTiBtZXNzYWdlcwogKi8Kc3RhdGljIHZvaWQgICAgV0NVU0VSX0dlbmVyYXRlS2V5SW5wdXRSZWNvcmQoc3RydWN0IGlubmVyX2RhdGEqIGRhdGEsIEJPT0wgZG93biwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSwgQk9PTCBzeXMpCnsKICAgIElOUFVUX1JFQ09SRAlpcjsKICAgIERXT1JECQluOwogICAgV0NIQVIJCWJ1ZlsyXTsKICAgIHN0YXRpYwlXQ0hBUglsYXN0OyAvKiBrZWVwIGxhc3QgY2hhciBzZWVuIGFzIGZlZWQgZm9yIGtleSB1cCBtZXNzYWdlICovCiAgICBCWVRFCQlrZXlTdGF0ZVsyNTZdOwoKICAgIGlyLkV2ZW50VHlwZSA9IEtFWV9FVkVOVDsKICAgIGlyLkV2ZW50LktleUV2ZW50LmJLZXlEb3duID0gZG93bjsKICAgIGlyLkV2ZW50LktleUV2ZW50LndSZXBlYXRDb3VudCA9IExPV09SRChsUGFyYW0pOwogICAgaXIuRXZlbnQuS2V5RXZlbnQud1ZpcnR1YWxLZXlDb2RlID0gd1BhcmFtOwoKICAgIGlyLkV2ZW50LktleUV2ZW50LndWaXJ0dWFsU2NhbkNvZGUgPSBISVdPUkQobFBhcmFtKSAmIDB4RkY7CgogICAgaXIuRXZlbnQuS2V5RXZlbnQudUNoYXIuVW5pY29kZUNoYXIgPSAwOwogICAgaXIuRXZlbnQuS2V5RXZlbnQuZHdDb250cm9sS2V5U3RhdGUgPSBXQ1VTRVJfR2V0Q3RybEtleVN0YXRlKGtleVN0YXRlKTsKICAgIGlmIChsUGFyYW0gJiAoMUwgPDwgMjQpKQkJaXIuRXZlbnQuS2V5RXZlbnQuZHdDb250cm9sS2V5U3RhdGUgfD0gRU5IQU5DRURfS0VZOwogICAgaWYgKHN5cykJCQkJaXIuRXZlbnQuS2V5RXZlbnQuZHdDb250cm9sS2V5U3RhdGUgfD0gTEVGVF9BTFRfUFJFU1NFRDsgLyogRklYTUU6IGdvdHRhIGNob29zZSBvbmUgKi8KCiAgICBpZiAoIShpci5FdmVudC5LZXlFdmVudC5kd0NvbnRyb2xLZXlTdGF0ZSAmIEVOSEFOQ0VEX0tFWSkpCiAgICB7CglpZiAoZG93bikKCXsKCSAgICBzd2l0Y2ggKFRvVW5pY29kZSh3UGFyYW0sIEhJV09SRChsUGFyYW0pLCBrZXlTdGF0ZSwgYnVmLCAyLCAwKSkKCSAgICB7CgkgICAgY2FzZSAyOgoJCS8qIEZJWE1FLi4uIHNob3VsZCBnZW5lcmF0ZSB0d28gZXZlbnRzLi4uICovCgkJLyogZmFsbCB0aHJ1ICovCgkgICAgY2FzZSAxOgoJCWxhc3QgPSBidWZbMF07CgkJYnJlYWs7CgkgICAgZGVmYXVsdDoKCQlsYXN0ID0gMDsKCQlicmVhazsKCSAgICB9Cgl9Cglpci5FdmVudC5LZXlFdmVudC51Q2hhci5Vbmljb2RlQ2hhciA9IGxhc3Q7IC8qIEZJWE1FIEhBQ0tZLi4uIGFuZCBidWdneSAnY296IGl0IHNob3VsZCBiZSBhIHN0YWNrLCBub3QgYSBzaW5nbGUgdmFsdWUgKi8KCWlmICghZG93bikgbGFzdCA9IDA7CiAgICB9CgogICAgV3JpdGVDb25zb2xlSW5wdXQoZGF0YS0+aENvbkluLCAmaXIsIDEsICZuKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0dlbmVyYXRlTW91c2VJbnB1dFJlY29yZAogKgogKgogKi8Kc3RhdGljIHZvaWQgICAgV0NVU0VSX0dlbmVyYXRlTW91c2VJbnB1dFJlY29yZChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgQ09PUkQgYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBXUEFSQU0gd1BhcmFtLCBEV09SRCBldmVudCkKewogICAgSU5QVVRfUkVDT1JECWlyOwogICAgQllURQkJa2V5U3RhdGVbMjU2XTsKICAgIERXT1JEICAgICAgICAgICAgICAgbW9kZSwgbjsKCiAgICAvKiBNT1VTRV9FVkVOVHMgc2hvdWxkbid0IGJlIHNlbnQgdW5sZXNzIEVOQUJMRV9NT1VTRV9JTlBVVCBpcyBhY3RpdmUgKi8KICAgIGlmICghR2V0Q29uc29sZU1vZGUoZGF0YS0+aENvbkluLCAmbW9kZSkgfHwgIShtb2RlICYgRU5BQkxFX01PVVNFX0lOUFVUKSkKICAgICAgICByZXR1cm47CgogICAgaXIuRXZlbnRUeXBlID0gTU9VU0VfRVZFTlQ7CiAgICBpci5FdmVudC5Nb3VzZUV2ZW50LmR3TW91c2VQb3NpdGlvbiA9IGM7CiAgICBpci5FdmVudC5Nb3VzZUV2ZW50LmR3QnV0dG9uU3RhdGUgPSAwOwogICAgaWYgKHdQYXJhbSAmIE1LX0xCVVRUT04pIGlyLkV2ZW50Lk1vdXNlRXZlbnQuZHdCdXR0b25TdGF0ZSB8PSBGUk9NX0xFRlRfMVNUX0JVVFRPTl9QUkVTU0VEOwogICAgaWYgKHdQYXJhbSAmIE1LX01CVVRUT04pIGlyLkV2ZW50Lk1vdXNlRXZlbnQuZHdCdXR0b25TdGF0ZSB8PSBGUk9NX0xFRlRfMk5EX0JVVFRPTl9QUkVTU0VEOwogICAgaWYgKHdQYXJhbSAmIE1LX1JCVVRUT04pIGlyLkV2ZW50Lk1vdXNlRXZlbnQuZHdCdXR0b25TdGF0ZSB8PSBSSUdIVE1PU1RfQlVUVE9OX1BSRVNTRUQ7CiAgICBpZiAod1BhcmFtICYgTUtfQ09OVFJPTCkgaXIuRXZlbnQuTW91c2VFdmVudC5kd0J1dHRvblN0YXRlIHw9IExFRlRfQ1RSTF9QUkVTU0VEOwogICAgaWYgKHdQYXJhbSAmIE1LX1NISUZUKSAgIGlyLkV2ZW50Lk1vdXNlRXZlbnQuZHdCdXR0b25TdGF0ZSB8PSBTSElGVF9QUkVTU0VEOwogICAgaWYgKGV2ZW50ID09IE1PVVNFX1dIRUVMRUQpIGlyLkV2ZW50Lk1vdXNlRXZlbnQuZHdCdXR0b25TdGF0ZSB8PSB3UGFyYW0gJiAweEZGRkYwMDAwOwogICAgaXIuRXZlbnQuTW91c2VFdmVudC5kd0NvbnRyb2xLZXlTdGF0ZSA9IFdDVVNFUl9HZXRDdHJsS2V5U3RhdGUoa2V5U3RhdGUpOwogICAgaXIuRXZlbnQuTW91c2VFdmVudC5kd0V2ZW50RmxhZ3MgPSBldmVudDsKCiAgICBXcml0ZUNvbnNvbGVJbnB1dChkYXRhLT5oQ29uSW4sICZpciwgMSwgJm4pOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfUHJvYwogKgogKgogKi8Kc3RhdGljIExSRVNVTFQgQ0FMTEJBQ0sgV0NVU0VSX1Byb2MoSFdORCBoV25kLCBVSU5UIHVNc2csIFdQQVJBTSB3UGFyYW0sIExQQVJBTSBsUGFyYW0pCnsKICAgIHN0cnVjdCBpbm5lcl9kYXRhKglkYXRhID0gKHN0cnVjdCBpbm5lcl9kYXRhKilHZXRXaW5kb3dMb25nUHRyKGhXbmQsIDApOwoKICAgIHN3aXRjaCAodU1zZykKICAgIHsKICAgIGNhc2UgV01fQ1JFQVRFOgogICAgICAgIHJldHVybiBXQ1VTRVJfQ3JlYXRlKGhXbmQsIChMUENSRUFURVNUUlVDVClsUGFyYW0pOwogICAgY2FzZSBXTV9ERVNUUk9ZOgoJZGF0YS0+aFduZCA9IDA7CglQb3N0UXVpdE1lc3NhZ2UoMCk7CglicmVhazsKICAgIGNhc2UgV01fUEFJTlQ6CglXQ1VTRVJfUGFpbnQoZGF0YSk7CglicmVhazsKICAgIGNhc2UgV01fS0VZRE9XTjoKICAgIGNhc2UgV01fS0VZVVA6CiAgICAgICAgaWYgKFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24pCiAgICAgICAgICAgIFdDVVNFUl9IYW5kbGVTZWxlY3Rpb25LZXkoZGF0YSwgdU1zZyA9PSBXTV9LRVlET1dOLCB3UGFyYW0sIGxQYXJhbSk7CiAgICAgICAgZWxzZQogICAgICAgICAgICBXQ1VTRVJfR2VuZXJhdGVLZXlJbnB1dFJlY29yZChkYXRhLCB1TXNnID09IFdNX0tFWURPV04sIHdQYXJhbSwgbFBhcmFtLCBGQUxTRSk7CglicmVhazsKICAgIGNhc2UgV01fU1lTS0VZRE9XTjoKICAgIGNhc2UgV01fU1lTS0VZVVA6CglXQ1VTRVJfR2VuZXJhdGVLZXlJbnB1dFJlY29yZChkYXRhLCB1TXNnID09IFdNX1NZU0tFWURPV04sIHdQYXJhbSwgbFBhcmFtLCBUUlVFKTsKCWJyZWFrOwogICAgY2FzZSBXTV9MQlVUVE9ORE9XTjoKICAgICAgICBpZiAoZGF0YS0+Y3VyY2ZnLnF1aWNrX2VkaXQgfHwgUFJJVkFURShkYXRhKS0+aGFzX3NlbGVjdGlvbikKICAgICAgICB7CiAgICAgICAgICAgIGlmIChQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uKQogICAgICAgICAgICAgICAgV0NVU0VSX1NldFNlbGVjdGlvbihkYXRhLCAwKTsKICAgICAgICAgICAgCiAgICAgICAgICAgIGlmIChkYXRhLT5jdXJjZmcucXVpY2tfZWRpdCAmJiBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID0gRkFMU0U7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgZWxzZQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIgPSBXQ1VTRVJfR2V0Q2VsbChkYXRhLCBsUGFyYW0pOwogICAgICAgICAgICAgICAgU2V0Q2FwdHVyZShkYXRhLT5oV25kKTsKICAgICAgICAgICAgICAgIFdDVVNFUl9TZXRTZWxlY3Rpb24oZGF0YSwgMCk7CiAgICAgICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID0gVFJVRTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICBXQ1VTRVJfR2VuZXJhdGVNb3VzZUlucHV0UmVjb3JkKGRhdGEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgMCk7CiAgICAgICAgfQoJYnJlYWs7CiAgICBjYXNlIFdNX01PVVNFTU9WRToKICAgICAgICBpZiAoZGF0YS0+Y3VyY2ZnLnF1aWNrX2VkaXQgfHwgUFJJVkFURShkYXRhKS0+aGFzX3NlbGVjdGlvbikKICAgICAgICB7CiAgICAgICAgICAgIGlmIChHZXRDYXB0dXJlKCkgPT0gZGF0YS0+aFduZCAmJiBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uICYmCiAgICAgICAgICAgICAgICAod1BhcmFtICYgTUtfTEJVVFRPTikpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFdDVVNFUl9Nb3ZlU2VsZWN0aW9uKGRhdGEsIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSk7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgZWxzZQogICAgICAgIHsKICAgICAgICAgICAgV0NVU0VSX0dlbmVyYXRlTW91c2VJbnB1dFJlY29yZChkYXRhLCBXQ1VTRVJfR2V0Q2VsbChkYXRhLCBsUGFyYW0pLCB3UGFyYW0sIE1PVVNFX01PVkVEKTsKICAgICAgICB9CglicmVhazsKICAgIGNhc2UgV01fTEJVVFRPTlVQOgogICAgICAgIGlmIChkYXRhLT5jdXJjZmcucXVpY2tfZWRpdCB8fCBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uKQogICAgICAgIHsKICAgICAgICAgICAgaWYgKEdldENhcHR1cmUoKSA9PSBkYXRhLT5oV25kICYmIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24pCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFdDVVNFUl9Nb3ZlU2VsZWN0aW9uKGRhdGEsIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSk7CiAgICAgICAgICAgICAgICBSZWxlYXNlQ2FwdHVyZSgpOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIFdDVVNFUl9HZW5lcmF0ZU1vdXNlSW5wdXRSZWNvcmQoZGF0YSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSwgd1BhcmFtLCAwKTsKICAgICAgICB9CglicmVhazsKICAgIGNhc2UgV01fUkJVVFRPTkRPV046CiAgICAgICAgaWYgKCh3UGFyYW0gJiAoTUtfQ09OVFJPTHxNS19TSElGVCkpID09IGRhdGEtPmN1cmNmZy5tZW51X21hc2spCiAgICAgICAgewogICAgICAgICAgICBQT0lOVCAgICAgICBwdDsKCiAgICAgICAgICAgIHB0LnggPSAoc2hvcnQpTE9XT1JEKGxQYXJhbSk7CiAgICAgICAgICAgIHB0LnkgPSAoc2hvcnQpSElXT1JEKGxQYXJhbSk7CiAgICAgICAgICAgIENsaWVudFRvU2NyZWVuKGhXbmQsICZwdCk7CiAgICAgICAgICAgIFdDVVNFUl9TZXRNZW51RGV0YWlscyhkYXRhLCBQUklWQVRFKGRhdGEpLT5oUG9wTWVudSk7CiAgICAgICAgICAgIFRyYWNrUG9wdXBNZW51KFBSSVZBVEUoZGF0YSktPmhQb3BNZW51LCBUUE1fTEVGVEFMSUdOfFRQTV9UT1BBTElHTiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgcHQueCwgcHQueSwgMCwgaFduZCwgTlVMTCk7CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIFdDVVNFUl9HZW5lcmF0ZU1vdXNlSW5wdXRSZWNvcmQoZGF0YSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSwgd1BhcmFtLCAwKTsKICAgICAgICB9CglicmVhazsKICAgIGNhc2UgV01fUkJVVFRPTlVQOgogICAgICAgIC8qIG5vIG5lZWQgdG8gdHJhY2sgZm9yIHJidXR0b24gdXAgd2hlbiBvcGVuaW5nIHRoZSBwb3B1cC4uLiB0aGUgZXZlbnQgd2lsbCBiZQogICAgICAgICAqIHN3YWxsb3dlZCBieSBUcmFja1BvcHVwTWVudSAqLwogICAgY2FzZSBXTV9NQlVUVE9ORE9XTjoKICAgIGNhc2UgV01fTUJVVFRPTlVQOgogICAgICAgIFdDVVNFUl9HZW5lcmF0ZU1vdXNlSW5wdXRSZWNvcmQoZGF0YSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSwgd1BhcmFtLCAwKTsKCWJyZWFrOwogICAgY2FzZSBXTV9MQlVUVE9OREJMQ0xLOgogICAgY2FzZSBXTV9NQlVUVE9OREJMQ0xLOgogICAgY2FzZSBXTV9SQlVUVE9OREJMQ0xLOgogICAgICAgIFdDVVNFUl9HZW5lcmF0ZU1vdXNlSW5wdXRSZWNvcmQoZGF0YSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSwgd1BhcmFtLCBET1VCTEVfQ0xJQ0spOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBXTV9NT1VTRVdIRUVMOgogICAgICAgIFdDVVNFUl9HZW5lcmF0ZU1vdXNlSW5wdXRSZWNvcmQoZGF0YSwgIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgTU9VU0VfV0hFRUxFRCk7CglicmVhazsKICAgIGNhc2UgV01fU0VURk9DVVM6CglpZiAoZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlKQoJewoJICAgIENyZWF0ZUNhcmV0KGRhdGEtPmhXbmQsIFBSSVZBVEUoZGF0YSktPmN1cnNvcl9iaXRtYXAsCiAgICAgICAgICAgICAgICAgICAgICAgIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLCBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQpOwoJICAgIFdDVVNFUl9Qb3NDdXJzb3IoZGF0YSk7Cgl9CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIFdNX0tJTExGT0NVUzoKCWlmIChkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUpCgkgICAgRGVzdHJveUNhcmV0KCk7CglicmVhazsKICAgIGNhc2UgV01fSFNDUk9MTDoKICAgICAgICB7CiAgICAgICAgICAgIGludAlwb3MgPSBkYXRhLT5jdXJjZmcud2luX3Bvcy5YOwoKICAgICAgICAgICAgc3dpdGNoIChMT1dPUkQod1BhcmFtKSkKICAgICAgICAgICAgewogICAgICAgICAgICBjYXNlIFNCX1BBR0VVUDogCXBvcyAtPSA4OyAJCWJyZWFrOwogICAgICAgICAgICBjYXNlIFNCX1BBR0VET1dOOiAJcG9zICs9IDg7IAkJYnJlYWs7CiAgICAgICAgICAgIGNhc2UgU0JfTElORVVQOiAJcG9zLS07CQkJYnJlYWs7CiAgICAgICAgICAgIGNhc2UgU0JfTElORURPV046IAlwb3MrKzsJIAkJYnJlYWs7CiAgICAgICAgICAgIGNhc2UgU0JfVEhVTUJUUkFDSzogcG9zID0gSElXT1JEKHdQYXJhbSk7CWJyZWFrOwogICAgICAgICAgICBkZWZhdWx0OiAJCQkJCWJyZWFrOwogICAgICAgICAgICB9CiAgICAgICAgICAgIGlmIChwb3MgPCAwKSBwb3MgPSAwOwogICAgICAgICAgICBpZiAocG9zID4gZGF0YS0+Y3VyY2ZnLnNiX3dpZHRoIC0gZGF0YS0+Y3VyY2ZnLndpbl93aWR0aCkKICAgICAgICAgICAgICAgIHBvcyA9IGRhdGEtPmN1cmNmZy5zYl93aWR0aCAtIGRhdGEtPmN1cmNmZy53aW5fd2lkdGg7CiAgICAgICAgICAgIGlmIChwb3MgIT0gZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgU2Nyb2xsV2luZG93KGhXbmQsIChkYXRhLT5jdXJjZmcud2luX3Bvcy5YIC0gcG9zKSAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLCAwLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIE5VTEwsIE5VTEwpOwogICAgICAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCA9IHBvczsKICAgICAgICAgICAgICAgIFNldFNjcm9sbFBvcyhoV25kLCBTQl9IT1JaLCBwb3MsIFRSVUUpOwogICAgICAgICAgICAgICAgVXBkYXRlV2luZG93KGhXbmQpOwogICAgICAgICAgICAgICAgV0NVU0VSX1Bvc0N1cnNvcihkYXRhKTsKICAgICAgICAgICAgICAgIFdJTkVDT05fTm90aWZ5V2luZG93Q2hhbmdlKGRhdGEpOwogICAgICAgICAgICB9CiAgICAgICAgfQoJYnJlYWs7CiAgICBjYXNlIFdNX1ZTQ1JPTEw6CiAgICAgICAgewoJICAgIGludAlwb3MgPSBkYXRhLT5jdXJjZmcud2luX3Bvcy5ZOwoKCSAgICBzd2l0Y2ggKExPV09SRCh3UGFyYW0pKQoJICAgIHsKICAgICAgICAgICAgY2FzZSBTQl9QQUdFVVA6IAlwb3MgLT0gODsgCQlicmVhazsKICAgICAgICAgICAgY2FzZSBTQl9QQUdFRE9XTjogCXBvcyArPSA4OyAJCWJyZWFrOwogICAgICAgICAgICBjYXNlIFNCX0xJTkVVUDogCXBvcy0tOwkJCWJyZWFrOwoJICAgIGNhc2UgU0JfTElORURPV046IAlwb3MrKzsJIAkJYnJlYWs7CiAgICAgICAgICAgIGNhc2UgU0JfVEhVTUJUUkFDSzogcG9zID0gSElXT1JEKHdQYXJhbSk7CWJyZWFrOwogICAgICAgICAgICBkZWZhdWx0OiAJCQkJCWJyZWFrOwoJICAgIH0KCSAgICBpZiAocG9zIDwgMCkgcG9zID0gMDsKCSAgICBpZiAocG9zID4gZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAtIGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0KQogICAgICAgICAgICAgICAgcG9zID0gZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAtIGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0OwoJICAgIGlmIChwb3MgIT0gZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWSkKCSAgICB7CgkJU2Nyb2xsV2luZG93KGhXbmQsIDAsIChkYXRhLT5jdXJjZmcud2luX3Bvcy5ZIC0gcG9zKSAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOVUxMLCBOVUxMKTsKCQlkYXRhLT5jdXJjZmcud2luX3Bvcy5ZID0gcG9zOwoJCVNldFNjcm9sbFBvcyhoV25kLCBTQl9WRVJULCBwb3MsIFRSVUUpOwoJCVVwZGF0ZVdpbmRvdyhoV25kKTsKCQlXQ1VTRVJfUG9zQ3Vyc29yKGRhdGEpOwoJCVdJTkVDT05fTm90aWZ5V2luZG93Q2hhbmdlKGRhdGEpOwoJICAgIH0KCiAgICAgICAgfQogICAgY2FzZSBXTV9TWVNDT01NQU5EOgoJc3dpdGNoICh3UGFyYW0pCgl7CgljYXNlIElEU19ERUZBVUxUOgoJICAgIFdDVVNFUl9HZXRQcm9wZXJ0aWVzKGRhdGEsIEZBTFNFKTsKCSAgICBicmVhazsKCWNhc2UgSURTX1BST1BFUlRJRVM6CgkgICAgV0NVU0VSX0dldFByb3BlcnRpZXMoZGF0YSwgVFJVRSk7CgkgICAgYnJlYWs7CglkZWZhdWx0OgoJICAgIHJldHVybiBEZWZXaW5kb3dQcm9jKGhXbmQsIHVNc2csIHdQYXJhbSwgbFBhcmFtKTsKCX0KCWJyZWFrOwogICAgY2FzZSBXTV9DT01NQU5EOgoJc3dpdGNoICh3UGFyYW0pCgl7CgljYXNlIElEU19ERUZBVUxUOgoJICAgIFdDVVNFUl9HZXRQcm9wZXJ0aWVzKGRhdGEsIEZBTFNFKTsKCSAgICBicmVhazsKCWNhc2UgSURTX1BST1BFUlRJRVM6CgkgICAgV0NVU0VSX0dldFByb3BlcnRpZXMoZGF0YSwgVFJVRSk7CgkgICAgYnJlYWs7CgljYXNlIElEU19NQVJLOgogICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWCA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5ZID0gMDsKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlggPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIuWSA9IDA7CiAgICAgICAgICAgIFdDVVNFUl9TZXRTZWxlY3Rpb24oZGF0YSwgMCk7CiAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBUUlVFOwoJICAgIGJyZWFrOwoJY2FzZSBJRFNfQ09QWToKICAgICAgICAgICAgaWYgKFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24pCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBGQUxTRTsKICAgICAgICAgICAgICAgIFdDVVNFUl9TZXRTZWxlY3Rpb24oZGF0YSwgMCk7CiAgICAgICAgICAgICAgICBXQ1VTRVJfQ29weVNlbGVjdGlvblRvQ2xpcGJvYXJkKGRhdGEpOwogICAgICAgICAgICB9CgkgICAgYnJlYWs7CgljYXNlIElEU19QQVNURToKCSAgICBXQ1VTRVJfUGFzdGVGcm9tQ2xpcGJvYXJkKGRhdGEpOwoJICAgIGJyZWFrOwoJY2FzZSBJRFNfU0VMRUNUQUxMOgogICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWCA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5ZID0gMDsKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlggPSAoZGF0YS0+Y3VyY2ZnLnNiX3dpZHRoIC0gMSkgKiBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkgPSAoZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAtIDEpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0OwogICAgICAgICAgICBXQ1VTRVJfU2V0U2VsZWN0aW9uKGRhdGEsIDApOwogICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID0gVFJVRTsKCSAgICBicmVhazsKCWNhc2UgSURTX1NDUk9MTDoKCWNhc2UgSURTX1NFQVJDSDoKCSAgICBXSU5FX0ZJWE1FKCJVbmhhbmRsZWQgeWV0IGNvbW1hbmQ6ICVseFxuIiwgd1BhcmFtKTsKCSAgICBicmVhazsKCWRlZmF1bHQ6CgkgICAgcmV0dXJuIERlZldpbmRvd1Byb2MoaFduZCwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwoJfQoJYnJlYWs7CiAgICBjYXNlIFdNX0lOSVRNRU5VUE9QVVA6CglpZiAoIUhJV09SRChsUGFyYW0pKQlyZXR1cm4gRGVmV2luZG93UHJvYyhoV25kLCB1TXNnLCB3UGFyYW0sIGxQYXJhbSk7CglXQ1VTRVJfU2V0TWVudURldGFpbHMoZGF0YSwgR2V0U3lzdGVtTWVudShkYXRhLT5oV25kLCBGQUxTRSkpOwoJYnJlYWs7CiAgICBkZWZhdWx0OgoJcmV0dXJuIERlZldpbmRvd1Byb2MoaFduZCwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwogICAgfQogICAgcmV0dXJuIDA7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9EZWxldGVCYWNrZW5kCiAqCiAqCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfRGVsZXRlQmFja2VuZChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSkKewogICAgaWYgKCFQUklWQVRFKGRhdGEpKSByZXR1cm47CiAgICBpZiAoUFJJVkFURShkYXRhKS0+aE1lbURDKQkJRGVsZXRlREMoUFJJVkFURShkYXRhKS0+aE1lbURDKTsKICAgIGlmIChkYXRhLT5oV25kKQkJICAgICAgICBEZXN0cm95V2luZG93KGRhdGEtPmhXbmQpOwogICAgaWYgKFBSSVZBVEUoZGF0YSktPmhGb250KQkJRGVsZXRlT2JqZWN0KFBSSVZBVEUoZGF0YSktPmhGb250KTsKICAgIGlmIChQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwKQlEZWxldGVPYmplY3QoUFJJVkFURShkYXRhKS0+Y3Vyc29yX2JpdG1hcCk7CiAgICBpZiAoUFJJVkFURShkYXRhKS0+aEJpdG1hcCkJCURlbGV0ZU9iamVjdChQUklWQVRFKGRhdGEpLT5oQml0bWFwKTsKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIFBSSVZBVEUoZGF0YSkpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfTWFpbkxvb3AKICoKICoKICovCnN0YXRpYyBpbnQgV0NVU0VSX01haW5Mb29wKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBNU0cJCW1zZzsKCiAgICBTaG93V2luZG93KGRhdGEtPmhXbmQsIGRhdGEtPm5DbWRTaG93KTsKICAgIGZvciAoOzspCiAgICB7Cglzd2l0Y2ggKE1zZ1dhaXRGb3JNdWx0aXBsZU9iamVjdHMoMSwgJmRhdGEtPmhTeW5jaHJvLCBGQUxTRSwgSU5GSU5JVEUsIFFTX0FMTElOUFVUKSkKCXsKCWNhc2UgV0FJVF9PQkpFQ1RfMDoKCSAgICBpZiAoIVdJTkVDT05fR3JhYkNoYW5nZXMoZGF0YSkgJiYgZGF0YS0+Y3VyY2ZnLmV4aXRfb25fZGllKQogICAgICAgICAgICAgICAgUG9zdFF1aXRNZXNzYWdlKDApOwoJICAgIGJyZWFrOwoJY2FzZSBXQUlUX09CSkVDVF8wKzE6CgkgICAgLyogbmVlZCB0byB1c2UgUGVla01lc3NhZ2UgbG9vcCBpbnN0ZWFkIG9mIHNpbXBsZSBHZXRNZXNzYWdlOgoJICAgICAqIG11bHRpcGxlIG1lc3NhZ2VzIG1pZ2h0IGhhdmUgYXJyaXZlZCBpbiBiZXR3ZWVuLAoJICAgICAqIHNvIEdldE1lc3NhZ2Ugd291bGQgbGVhZCB0byBkZWxheWVkIHByb2Nlc3NpbmcgKi8KCSAgICB3aGlsZSAoUGVla01lc3NhZ2UoJm1zZywgMCwgMCwgMCwgUE1fUkVNT1ZFKSkKCSAgICB7CiAgICAgICAgICAgICAgICBpZiAobXNnLm1lc3NhZ2UgPT0gV01fUVVJVCkgcmV0dXJuIDA7CiAgICAgICAgICAgICAgICBXSU5FX1RSQUNFKCJkaXNwYXRjaGluZyBtc2cgJTA0eFxuIiwgbXNnLm1lc3NhZ2UpOwogICAgICAgICAgICAgICAgRGlzcGF0Y2hNZXNzYWdlKCZtc2cpOwoJICAgIH0KCSAgICBicmVhazsKCWRlZmF1bHQ6CgkgICAgV0lORV9FUlIoImdvdCBwYlxuIik7CgkgICAgLyogZXJyICovCgkgICAgYnJlYWs7Cgl9CiAgICB9Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9Jbml0QmFja2VuZAogKgogKiBJbml0aWFsaXNhdGlvbiBwYXJ0IElJOiBjcmVhdGlvbiBvZiB3aW5kb3cuCiAqCiAqLwplbnVtIGluaXRfcmV0dXJuIFdDVVNFUl9Jbml0QmFja2VuZChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIHdDbGFzc05hbWVbXSA9IHsnVycsJ2knLCduJywnZScsJ0MnLCdvJywnbicsJ3MnLCdvJywnbCcsJ2UnLCdDJywnbCcsJ2EnLCdzJywncycsMH07CgogICAgV05EQ0xBU1MJCXduZGNsYXNzOwogICAgQ0hBUlNFVElORk8gICAgICAgICBjaTsKICAgIAogICAgaWYgKCFUcmFuc2xhdGVDaGFyc2V0SW5mbygoRFdPUkQgKikoSU5UX1BUUilHZXRBQ1AoKSwgJmNpLCBUQ0lfU1JDQ09ERVBBR0UpKQogICAgICAgIHJldHVybiBpbml0X2ZhaWxlZDsKICAgIGdfdWlEZWZhdWx0Q2hhcnNldCA9IGNpLmNpQ2hhcnNldDsKICAgIFdJTkVfVFJBQ0VfKHdjX2ZvbnQpKCJDb2RlIHBhZ2UgJWQgPT4gRGVmYXVsdCBjaGFyc2V0OiAlZFxuIiwgR2V0QUNQKCksIGdfdWlEZWZhdWx0Q2hhcnNldCk7CgogICAgZGF0YS0+cHJpdmF0ZSA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCBIRUFQX1pFUk9fTUVNT1JZLCBzaXplb2Yoc3RydWN0IGlubmVyX2RhdGFfdXNlcikpOwogICAgaWYgKCFkYXRhLT5wcml2YXRlKSByZXR1cm4gaW5pdF9mYWlsZWQ7CgogICAgZGF0YS0+Zm5NYWluTG9vcCA9IFdDVVNFUl9NYWluTG9vcDsKICAgIGRhdGEtPmZuUG9zQ3Vyc29yID0gV0NVU0VSX1Bvc0N1cnNvcjsKICAgIGRhdGEtPmZuU2hhcGVDdXJzb3IgPSBXQ1VTRVJfU2hhcGVDdXJzb3I7CiAgICBkYXRhLT5mbkNvbXB1dGVQb3NpdGlvbnMgPSBXQ1VTRVJfQ29tcHV0ZVBvc2l0aW9uczsKICAgIGRhdGEtPmZuUmVmcmVzaCA9IFdDVVNFUl9SZWZyZXNoOwogICAgZGF0YS0+Zm5SZXNpemVTY3JlZW5CdWZmZXIgPSBXQ1VTRVJfUmVzaXplU2NyZWVuQnVmZmVyOwogICAgZGF0YS0+Zm5TZXRUaXRsZSA9IFdDVVNFUl9TZXRUaXRsZTsKICAgIGRhdGEtPmZuU2V0Rm9udCA9IFdDVVNFUl9TZXRGb250UG10OwogICAgZGF0YS0+Zm5TY3JvbGwgPSBXQ1VTRVJfU2Nyb2xsOwogICAgZGF0YS0+Zm5EZWxldGVCYWNrZW5kID0gV0NVU0VSX0RlbGV0ZUJhY2tlbmQ7CgogICAgd25kY2xhc3Muc3R5bGUgICAgICAgICA9IENTX0RCTENMS1M7CiAgICB3bmRjbGFzcy5scGZuV25kUHJvYyAgID0gV0NVU0VSX1Byb2M7CiAgICB3bmRjbGFzcy5jYkNsc0V4dHJhICAgID0gMDsKICAgIHduZGNsYXNzLmNiV25kRXh0cmEgICAgPSBzaXplb2YoRFdPUkRfUFRSKTsKICAgIHduZGNsYXNzLmhJbnN0YW5jZSAgICAgPSBHZXRNb2R1bGVIYW5kbGUoTlVMTCk7CiAgICB3bmRjbGFzcy5oSWNvbiAgICAgICAgID0gTG9hZEljb24oMCwgSURJX1dJTkxPR08pOwogICAgd25kY2xhc3MuaEN1cnNvciAgICAgICA9IExvYWRDdXJzb3IoMCwgSURDX0FSUk9XKTsKICAgIHduZGNsYXNzLmhickJhY2tncm91bmQgPSBHZXRTdG9ja09iamVjdChCTEFDS19CUlVTSCk7CiAgICB3bmRjbGFzcy5scHN6TWVudU5hbWUgID0gTlVMTDsKICAgIHduZGNsYXNzLmxwc3pDbGFzc05hbWUgPSB3Q2xhc3NOYW1lOwoKICAgIFJlZ2lzdGVyQ2xhc3MoJnduZGNsYXNzKTsKCiAgICBDcmVhdGVXaW5kb3cod25kY2xhc3MubHBzekNsYXNzTmFtZSwgTlVMTCwKCQkgV1NfT1ZFUkxBUFBFRHxXU19DQVBUSU9OfFdTX1NZU01FTlV8V1NfVEhJQ0tGUkFNRXxXU19NSU5JTUlaRUJPWHxXU19IU0NST0xMfFdTX1ZTQ1JPTEwsCgkJIENXX1VTRURFRkFVTFQsIENXX1VTRURFRkFVTFQsIDAsIDAsIDAsIDAsIHduZGNsYXNzLmhJbnN0YW5jZSwgZGF0YSk7CiAgICBpZiAoIWRhdGEtPmhXbmQpIHJldHVybiBpbml0X25vdF9zdXBwb3J0ZWQ7CgogICAgcmV0dXJuIGluaXRfc3VjY2VzczsKfQo=