LyoKICogYSBHVUkgYXBwbGljYXRpb24gZm9yIGRpc3BsYXlpbmcgYSBjb25zb2xlCiAqCVVTRVIzMiBiYWNrIGVuZAogKiBDb3B5cmlnaHQgMjAwMSBFcmljIFBvdWVjaAogKgogKiBUaGlzIGxpYnJhcnkgaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yCiAqIG1vZGlmeSBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMKICogTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkgdGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbjsgZWl0aGVyCiAqIHZlcnNpb24gMi4xIG9mIHRoZSBMaWNlbnNlLCBvciAoYXQgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLgogKgogKiBUaGlzIGxpYnJhcnkgaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKICogYnV0IFdJVEhPVVQgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YKICogTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiAgU2VlIHRoZSBHTlUKICogTGVzc2VyIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KICoKICogWW91IHNob3VsZCBoYXZlIHJlY2VpdmVkIGEgY29weSBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFsb25nIHdpdGggdGhpcyBsaWJyYXJ5OyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCiAqIEZvdW5kYXRpb24sIEluYy4sIDU5IFRlbXBsZSBQbGFjZSwgU3VpdGUgMzMwLCBCb3N0b24sIE1BICAwMjExMS0xMzA3ICBVU0EKICovCgojaW5jbHVkZSA8c3RkaW8uaD4KI2luY2x1ZGUgPHN0ZGxpYi5oPgojaW5jbHVkZSAid2luZWNvbl91c2VyLmgiCgojaW5jbHVkZSAid2luZS9kZWJ1Zy5oIgoKV0lORV9ERUZBVUxUX0RFQlVHX0NIQU5ORUwod2luZWNvbnNvbGUpOwpXSU5FX0RFQ0xBUkVfREVCVUdfQ0hBTk5FTCh3Y19mb250KTsKCi8qIG1hcHBpbmcgY29uc29sZSBjb2xvcnMgdG8gUkdCIHZhbHVlcyAqLwpDT0xPUlJFRglXQ1VTRVJfQ29sb3JNYXBbMTZdID0gCnsKICAgIFJHQigweDAwLCAweDAwLCAweDAwKSwgUkdCKDB4MDAsIDB4MDAsIDB4ODApLCBSR0IoMHgwMCwgMHg4MCwgMHgwMCksIFJHQigweDAwLCAweDgwLCAweDgwKSwKICAgIFJHQigweDgwLCAweDAwLCAweDAwKSwgUkdCKDB4ODAsIDB4MDAsIDB4ODApLCBSR0IoMHg4MCwgMHg4MCwgMHgwMCksIFJHQigweDgwLCAweDgwLCAweDgwKSwKICAgIFJHQigweEMwLCAweEMwLCAweEMwKSwgUkdCKDB4MDAsIDB4MDAsIDB4RkYpLCBSR0IoMHgwMCwgMHhGRiwgMHgwMCksIFJHQigweDAwLCAweEZGLCAweEZGKSwKICAgIFJHQigweEZGLCAweDAwLCAweDAwKSwgUkdCKDB4RkYsIDB4MDAsIDB4RkYpLCBSR0IoMHhGRiwgMHhGRiwgMHgwMCksIFJHQigweEZGLCAweEZGLCAweEZGKSwKfTsKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9GaWxsTWVtREMKICoKICogRmlsbHMgdGhlIE1lbSBEQyB3aXRoIGN1cnJlbnQgY2VsbHMgdmFsdWVzCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfRmlsbE1lbURDKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBpbnQgdXBkX3RwLCBpbnQgdXBkX2JtKQp7CiAgICB1bnNpZ25lZAkJaSwgaiwgazsKICAgIENIQVJfSU5GTyoJCWNlbGw7CiAgICBIRk9OVAkJaE9sZEZvbnQ7CiAgICBXT1JECQlhdHRyOwogICAgV0NIQVIqCQlsaW5lOwoKICAgIC8qIG5vIGZvbnQgaGFzIGJlZW4gc2V0IHVwIHlldCwgZG9uJ3Qgd29ycnkgYWJvdXQgZmlsbGluZyB0aGUgYml0bWFwLCAKICAgICAqIHdlJ2xsIGRvIGl0IG9uY2UgYSBmb250IGlzIGNob3NlbgogICAgICovCiAgICBpZiAoIVBSSVZBVEUoZGF0YSktPmhGb250KSByZXR1cm47CgogICAgaWYgKCEobGluZSA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggKiBzaXplb2YoV0NIQVIpKSkpCiAgICB7V0lORV9FUlIoIk9PTVxuIik7IHJldHVybjt9CgogICAgaE9sZEZvbnQgPSBTZWxlY3RPYmplY3QoUFJJVkFURShkYXRhKS0+aE1lbURDLCBQUklWQVRFKGRhdGEpLT5oRm9udCk7CiAgICBmb3IgKGogPSB1cGRfdHA7IGogPD0gdXBkX2JtOyBqKyspCiAgICB7CgljZWxsID0gJmRhdGEtPmNlbGxzW2ogKiBkYXRhLT5jdXJjZmcuc2Jfd2lkdGhdOwoJZm9yIChpID0gMDsgaSA8IGRhdGEtPmN1cmNmZy5zYl93aWR0aDsgaSsrKQoJewoJICAgIGF0dHIgPSBjZWxsW2ldLkF0dHJpYnV0ZXM7CgkgICAgU2V0QmtDb2xvcihQUklWQVRFKGRhdGEpLT5oTWVtREMsV0NVU0VSX0NvbG9yTWFwWyhhdHRyPj40KSYweDBGXSk7CgkgICAgU2V0VGV4dENvbG9yKFBSSVZBVEUoZGF0YSktPmhNZW1EQywgV0NVU0VSX0NvbG9yTWFwW2F0dHImMHgwRl0pOwoJICAgIGZvciAoayA9IGk7IGsgPCBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggJiYgY2VsbFtrXS5BdHRyaWJ1dGVzID09IGF0dHI7IGsrKykKCSAgICB7CgkJbGluZVtrIC0gaV0gPSBjZWxsW2tdLkNoYXIuVW5pY29kZUNoYXI7CgkgICAgfQoJICAgIFRleHRPdXQoUFJJVkFURShkYXRhKS0+aE1lbURDLCBpICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsIGogKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQsIAoJCSAgICBsaW5lLCBrIC0gaSk7CgkgICAgaSA9IGsgLSAxOwoJfQogICAgfQogICAgU2VsZWN0T2JqZWN0KFBSSVZBVEUoZGF0YSktPmhNZW1EQywgaE9sZEZvbnQpOwogICAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgbGluZSk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9OZXdCaXRtYXAKICoKICogRWl0aGVyIHRoZSBmb250IGdlb21ldHJ5IG9yIHRoZSBzYiBnZW9tZXRyeSBoYXMgY2hhbmdlZC4gd2UgbmVlZCB0byByZWNyZWF0ZSB0aGUKICogYml0bWFwIGdlb21ldHJ5CiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfTmV3Qml0bWFwKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBCT09MIGZpbGwpCnsKICAgIEhEQyAgICAgICAgIGhEQzsKICAgIEhCSVRNQVAJaG5ldywgaG9sZDsKCiAgICBpZiAoIWRhdGEtPmN1cmNmZy5zYl93aWR0aCB8fCAhZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCB8fCAhKGhEQyA9IEdldERDKFBSSVZBVEUoZGF0YSktPmhXbmQpKSkgCiAgICAgICAgcmV0dXJuOwogICAgaG5ldyA9IENyZWF0ZUNvbXBhdGlibGVCaXRtYXAoaERDLCAKCQkJCSAgZGF0YS0+Y3VyY2ZnLnNiX3dpZHRoICAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLCAKCQkJCSAgZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCk7CiAgICBSZWxlYXNlREMoUFJJVkFURShkYXRhKS0+aFduZCwgaERDKTsKICAgIGhvbGQgPSBTZWxlY3RPYmplY3QoUFJJVkFURShkYXRhKS0+aE1lbURDLCBobmV3KTsKCiAgICBpZiAoUFJJVkFURShkYXRhKS0+aEJpdG1hcCkKICAgIHsKCWlmIChob2xkID09IFBSSVZBVEUoZGF0YSktPmhCaXRtYXApCgkgICAgRGVsZXRlT2JqZWN0KFBSSVZBVEUoZGF0YSktPmhCaXRtYXApOwoJZWxzZQoJICAgIFdJTkVfRklYTUUoImxlYWtcbiIpOwogICAgfQogICAgUFJJVkFURShkYXRhKS0+aEJpdG1hcCA9IGhuZXc7CiAgICBpZiAoZmlsbCkKCVdDVVNFUl9GaWxsTWVtREMoZGF0YSwgMCwgZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAtIDEpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfUmVzaXplU2NyZWVuQnVmZmVyCiAqCiAqCiAqLwpzdGF0aWMgdm9pZCBXQ1VTRVJfUmVzaXplU2NyZWVuQnVmZmVyKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBXQ1VTRVJfTmV3Qml0bWFwKGRhdGEsIEZBTFNFKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1Bvc0N1cnNvcgogKgogKiBTZXQgYSBuZXcgcG9zaXRpb24gZm9yIHRoZSBjdXJzb3IKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9Qb3NDdXJzb3IoY29uc3Qgc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIGlmIChQUklWQVRFKGRhdGEpLT5oV25kICE9IEdldEZvY3VzKCkgfHwgIWRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSkgcmV0dXJuOwoKICAgIFNldENhcmV0UG9zKChkYXRhLT5jdXJzb3IuWCAtIGRhdGEtPmN1cmNmZy53aW5fcG9zLlgpICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsIAoJCShkYXRhLT5jdXJzb3IuWSAtIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0KTsKICAgIFNob3dDYXJldChQUklWQVRFKGRhdGEpLT5oV25kKTsgCn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9TaGFwZUN1cnNvcgogKgogKiBTZXRzIGEgbmV3IHNoYXBlIGZvciB0aGUgY3Vyc29yCiAqLwp2b2lkCVdDVVNFUl9TaGFwZUN1cnNvcihzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgaW50IHNpemUsIGludCB2aXMsIEJPT0wgZm9yY2UpCnsKICAgIGlmIChmb3JjZSB8fCBzaXplICE9IGRhdGEtPmN1cmNmZy5jdXJzb3Jfc2l6ZSkKICAgIHsKCWlmIChkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUgJiYgUFJJVkFURShkYXRhKS0+aFduZCA9PSBHZXRGb2N1cygpKSBEZXN0cm95Q2FyZXQoKTsKCWlmIChQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwKSBEZWxldGVPYmplY3QoUFJJVkFURShkYXRhKS0+Y3Vyc29yX2JpdG1hcCk7CglQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwID0gKEhCSVRNQVApMDsKCWlmIChzaXplICE9IDEwMCkKCXsKCSAgICBpbnQJCXcxNmI7IC8qIG51bWJlciBvZiBieWV0cyBwZXIgcm93LCBhbGlnbmVkIG9uIHdvcmQgc2l6ZSAqLwoJICAgIEJZVEUqCXB0cjsKCSAgICBpbnQJCWksIGosIG5ibDsKCgkgICAgdzE2YiA9ICgoZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGggKyAxNSkgJiB+MTUpIC8gODsKCSAgICBwdHIgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgSEVBUF9aRVJPX01FTU9SWSwgdzE2YiAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCk7CgkgICAgaWYgKCFwdHIpIHtXSU5FX0VSUigiT09NXG4iKTsgcmV0dXJuO30KCSAgICBuYmwgPSBtYXgoKGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCAqIHNpemUpIC8gMTAwLCAxKTsKCSAgICBmb3IgKGogPSBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQgLSBuYmw7IGogPCBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7IGorKykKCSAgICB7CgkJZm9yIChpID0gMDsgaSA8IGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoOyBpKyspCgkJewoJCSAgICBwdHJbdzE2YiAqIGogKyAoaSAvIDgpXSB8PSAweDgwID4+IChpICYgNyk7CgkJfQoJICAgIH0KCSAgICBQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwID0gQ3JlYXRlQml0bWFwKGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQsIDEsIDEsIHB0cik7CgkgICAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgcHRyKTsKCX0KCWRhdGEtPmN1cmNmZy5jdXJzb3Jfc2l6ZSA9IHNpemU7CglkYXRhLT5jdXJjZmcuY3Vyc29yX3Zpc2libGUgPSAtMTsKICAgIH0KCiAgICB2aXMgPSAodmlzKSA/IFRSVUUgOiBGQUxTRTsKICAgIGlmIChmb3JjZSB8fCB2aXMgIT0gZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlKQogICAgewoJZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlID0gdmlzOwoJaWYgKFBSSVZBVEUoZGF0YSktPmhXbmQgPT0gR2V0Rm9jdXMoKSkKCXsKCSAgICBpZiAodmlzKQoJICAgIHsKCQlDcmVhdGVDYXJldChQUklWQVRFKGRhdGEpLT5oV25kLCBQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoLCBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQpOwoJCVdDVVNFUl9Qb3NDdXJzb3IoZGF0YSk7CgkgICAgfQoJICAgIGVsc2UKCSAgICB7CgkJRGVzdHJveUNhcmV0KCk7CgkgICAgfQoJfQogICAgfQp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlJTkVDT05fQ29tcHV0ZVBvc2l0aW9ucwogKgogKiBSZWNvbXB1dGVzIGFsbCB0aGUgY29tcG9uZW50cyAobWFpbmx5IHNjcm9sbCBiYXJzKSBwb3NpdGlvbnMKICovCnZvaWQJV0NVU0VSX0NvbXB1dGVQb3NpdGlvbnMoc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIFJFQ1QJCXI7CiAgICBpbnQJCQlkeCwgZHk7CgogICAgLyogY29tcHV0ZSB3aW5kb3cgc2l6ZSBmcm9tIGRlc2lyZWQgY2xpZW50IHNpemUgKi8KICAgIHIubGVmdCA9IHIudG9wID0gMDsKICAgIHIucmlnaHQgPSBkYXRhLT5jdXJjZmcud2luX3dpZHRoICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGg7CiAgICByLmJvdHRvbSA9IGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0ICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0OwoKICAgIGlmIChJc1JlY3RFbXB0eSgmcikpCiAgICB7CglTaG93V2luZG93KFBSSVZBVEUoZGF0YSktPmhXbmQsIFNXX0hJREUpOwoJcmV0dXJuOwogICAgfQoKICAgIEFkanVzdFdpbmRvd1JlY3QoJnIsIEdldFdpbmRvd0xvbmcoUFJJVkFURShkYXRhKS0+aFduZCwgR1dMX1NUWUxFKSwgRkFMU0UpOwoKICAgIGR4ID0gZHkgPSAwOwogICAgaWYgKGRhdGEtPmN1cmNmZy5zYl93aWR0aCA+IGRhdGEtPmN1cmNmZy53aW5fd2lkdGgpCiAgICB7CglkeSA9IEdldFN5c3RlbU1ldHJpY3MoU01fQ1lIU0NST0xMKTsKCVNldFNjcm9sbFJhbmdlKFBSSVZBVEUoZGF0YSktPmhXbmQsIFNCX0hPUlosIDAsIAogICAgICAgICAgICAgICAgICAgICAgIGRhdGEtPmN1cmNmZy5zYl93aWR0aCAtIGRhdGEtPmN1cmNmZy53aW5fd2lkdGgsIEZBTFNFKTsKCVNldFNjcm9sbFBvcyhQUklWQVRFKGRhdGEpLT5oV25kLCBTQl9IT1JaLCAwLCBGQUxTRSk7IC8qIEZJWE1FICovCglTaG93U2Nyb2xsQmFyKFBSSVZBVEUoZGF0YSktPmhXbmQsIFNCX0hPUlosIFRSVUUpOwogICAgfQogICAgZWxzZQogICAgewoJU2hvd1Njcm9sbEJhcihQUklWQVRFKGRhdGEpLT5oV25kLCBTQl9IT1JaLCBGQUxTRSk7CiAgICB9CgogICAgaWYgKGRhdGEtPmN1cmNmZy5zYl9oZWlnaHQgPiBkYXRhLT5jdXJjZmcud2luX2hlaWdodCkKICAgIHsKCWR4ID0gR2V0U3lzdGVtTWV0cmljcyhTTV9DWFZTQ1JPTEwpOwoJU2V0U2Nyb2xsUmFuZ2UoUFJJVkFURShkYXRhKS0+aFduZCwgU0JfVkVSVCwgMCwgCiAgICAgICAgICAgICAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAtIGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0LCBGQUxTRSk7CglTZXRTY3JvbGxQb3MoUFJJVkFURShkYXRhKS0+aFduZCwgU0JfVkVSVCwgMCwgRkFMU0UpOyAvKiBGSVhNRSAqLwoJU2hvd1Njcm9sbEJhcihQUklWQVRFKGRhdGEpLT5oV25kLCBTQl9WRVJULCBUUlVFKTsKICAgIH0JCiAgICBlbHNlCiAgICB7CglTaG93U2Nyb2xsQmFyKFBSSVZBVEUoZGF0YSktPmhXbmQsIFNCX1ZFUlQsIEZBTFNFKTsKICAgIH0KCiAgICBTZXRXaW5kb3dQb3MoUFJJVkFURShkYXRhKS0+aFduZCwgMCwgMCwgMCwgci5yaWdodCAtIHIubGVmdCArIGR4LCByLmJvdHRvbSAtIHIudG9wICsgZHksCgkJIFNXUF9OT01PVkV8U1dQX05PWk9SREVSfFNXUF9TSE9XV0lORE9XKTsKICAgIFdDVVNFUl9TaGFwZUN1cnNvcihkYXRhLCBkYXRhLT5jdXJjZmcuY3Vyc29yX3NpemUsIGRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSwgVFJVRSk7CiAgICBXQ1VTRVJfUG9zQ3Vyc29yKGRhdGEpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfU2V0VGl0bGUKICoKICogU2V0cyB0aGUgdGl0bGUgdG8gdGhlIHdpbmUgY29uc29sZQogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX1NldFRpdGxlKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBXQ0hBUglidWZmZXJbMjU2XTsJCgogICAgaWYgKFdJTkVDT05fR2V0Q29uc29sZVRpdGxlKGRhdGEtPmhDb25JbiwgYnVmZmVyLCBzaXplb2YoYnVmZmVyKSkpCglTZXRXaW5kb3dUZXh0KFBSSVZBVEUoZGF0YSktPmhXbmQsIGJ1ZmZlcik7Cn0KCnZvaWQgV0NVU0VSX0R1bXBMb2dGb250KGNvbnN0IGNoYXIqIHBmeCwgY29uc3QgTE9HRk9OVCogbGYsIERXT1JEIGZ0KQp7CiAgICBXSU5FX1RSQUNFXyh3Y19mb250KSgiJXMgJXMlcyVzJXNcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdGxmLmxmSGVpZ2h0PSVsZCBsZi5sZldpZHRoPSVsZCBsZi5sZkVzY2FwZW1lbnQ9JWxkIGxmLmxmT3JpZW50YXRpb249JWxkXG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAiXHRsZi5sZldlaWdodD0lbGQgbGYubGZJdGFsaWM9JXUgbGYubGZVbmRlcmxpbmU9JXUgbGYubGZTdHJpa2VPdXQ9JXVcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdGxmLmxmQ2hhclNldD0ldSBsZi5sZk91dFByZWNpc2lvbj0ldSBsZi5sZkNsaXBQcmVjaXNpb249JXUgbGYubGZRdWFsaXR5PSV1XG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAiXHRsZi0+bGZQaXRjaEFuZEZhbWlseT0ldSBsZi5sZkZhY2VOYW1lPSVzXG4iLAogICAgICAgICAgICAgICAgICAgICAgICAgcGZ4LAogICAgICAgICAgICAgICAgICAgICAgICAgKGZ0ICYgUkFTVEVSX0ZPTlRUWVBFKSA/ICJyYXN0ZXIiIDogIiIsIAogICAgICAgICAgICAgICAgICAgICAgICAgKGZ0ICYgVFJVRVRZUEVfRk9OVFRZUEUpID8gInRydWV0eXBlIiA6ICIiLCAKICAgICAgICAgICAgICAgICAgICAgICAgICgoZnQgJiAoUkFTVEVSX0ZPTlRUWVBFfFRSVUVUWVBFX0ZPTlRUWVBFKSkgPT0gMCkgPyAidmVjdG9yIiA6ICIiLCAKICAgICAgICAgICAgICAgICAgICAgICAgIChmdCAmIERFVklDRV9GT05UVFlQRSkgPyAifGRldmljZSIgOiAiIiwgCiAgICAgICAgICAgICAgICAgICAgICAgICBsZi0+bGZIZWlnaHQsIGxmLT5sZldpZHRoLCBsZi0+bGZFc2NhcGVtZW50LCBsZi0+bGZPcmllbnRhdGlvbiwgCiAgICAgICAgICAgICAgICAgICAgICAgICBsZi0+bGZXZWlnaHQsIGxmLT5sZkl0YWxpYywgbGYtPmxmVW5kZXJsaW5lLCBsZi0+bGZTdHJpa2VPdXQsIGxmLT5sZkNoYXJTZXQsCiAgICAgICAgICAgICAgICAgICAgICAgICBsZi0+bGZPdXRQcmVjaXNpb24sIGxmLT5sZkNsaXBQcmVjaXNpb24sIGxmLT5sZlF1YWxpdHksIGxmLT5sZlBpdGNoQW5kRmFtaWx5LCAKICAgICAgICAgICAgICAgICAgICAgICAgIHdpbmVfZGJnc3RyX3cobGYtPmxmRmFjZU5hbWUpKTsKfQoKdm9pZCBXQ1VTRVJfRHVtcFRleHRNZXRyaWMoY29uc3QgVEVYVE1FVFJJQyogdG0sIERXT1JEIGZ0KQp7CiAgICBXSU5FX1RSQUNFXyh3Y19mb250KSgiJXMlcyVzJXNcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdHRtSGVpZ2h0PSVsZCB0bUFzY2VudD0lbGQgdG1EZXNjZW50PSVsZCB0bUludGVybmFsTGVhZGluZz0lbGQgdG1FeHRlcm5hbExlYWRpbmc9JWxkXG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAiXHR0bUF2ZUNoYXJXaWR0aD0lbGQgdG1NYXhDaGFyV2lkdGg9JWxkIHRtV2VpZ2h0PSVsZCB0bU92ZXJoYW5nPSVsZFxuIgogICAgICAgICAgICAgICAgICAgICAgICAgIlx0dG1EaWdpdGl6ZWRBc3BlY3RYPSVsZCB0bURpZ2l0aXplZEFzcGVjdFk9JWxkXG4iCiAgICAgICAgICAgICAgICAgICAgICAgICAiXHR0bUZpcnN0Q2hhcj0lZCB0bUxhc3RDaGFyPSVkIHRtRGVmYXVsdENoYXI9JWQgdG1CcmVha0NoYXI9JWRcbiIKICAgICAgICAgICAgICAgICAgICAgICAgICJcdHRtSXRhbGljPSV1IHRtVW5kZXJsaW5lZD0ldSB0bVN0cnVja091dD0ldSB0bVBpdGNoQW5kRmFtaWx5PSV1IHRtQ2hhclNldD0ldVxuIiwKICAgICAgICAgICAgICAgICAgICAgICAgIChmdCAmIFJBU1RFUl9GT05UVFlQRSkgPyAicmFzdGVyIiA6ICIiLCAKICAgICAgICAgICAgICAgICAgICAgICAgIChmdCAmIFRSVUVUWVBFX0ZPTlRUWVBFKSA/ICJ0cnVldHlwZSIgOiAiIiwgCiAgICAgICAgICAgICAgICAgICAgICAgICAoKGZ0ICYgKFJBU1RFUl9GT05UVFlQRXxUUlVFVFlQRV9GT05UVFlQRSkpID09IDApID8gInZlY3RvciIgOiAiIiwgCiAgICAgICAgICAgICAgICAgICAgICAgICAoZnQgJiBERVZJQ0VfRk9OVFRZUEUpID8gInxkZXZpY2UiIDogIiIsIAogICAgICAgICAgICAgICAgICAgICAgICAgdG0tPnRtSGVpZ2h0LCB0bS0+dG1Bc2NlbnQsIHRtLT50bURlc2NlbnQsIHRtLT50bUludGVybmFsTGVhZGluZywgdG0tPnRtRXh0ZXJuYWxMZWFkaW5nLCB0bS0+dG1BdmVDaGFyV2lkdGgsIAogICAgICAgICAgICAgICAgICAgICAgICAgdG0tPnRtTWF4Q2hhcldpZHRoLCB0bS0+dG1XZWlnaHQsIHRtLT50bU92ZXJoYW5nLCB0bS0+dG1EaWdpdGl6ZWRBc3BlY3RYLCB0bS0+dG1EaWdpdGl6ZWRBc3BlY3RZLCAKICAgICAgICAgICAgICAgICAgICAgICAgIHRtLT50bUZpcnN0Q2hhciwgdG0tPnRtTGFzdENoYXIsIHRtLT50bURlZmF1bHRDaGFyLCB0bS0+dG1CcmVha0NoYXIsIHRtLT50bUl0YWxpYywgdG0tPnRtVW5kZXJsaW5lZCwgdG0tPnRtU3RydWNrT3V0LCAKICAgICAgICAgICAgICAgICAgICAgICAgIHRtLT50bVBpdGNoQW5kRmFtaWx5LCB0bS0+dG1DaGFyU2V0KTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJRm9udEVxdWFsCiAqCiAqCiAqLwpCT09MIFdDVVNFUl9BcmVGb250c0VxdWFsKGNvbnN0IHN0cnVjdCBjb25maWdfZGF0YSogY29uZmlnLCBjb25zdCBMT0dGT05UKiBsZikKewogICAgcmV0dXJuIGxmLT5sZkhlaWdodCA9PSBjb25maWctPmNlbGxfaGVpZ2h0ICYmCiAgICAgICAgbGYtPmxmV2VpZ2h0ID09IGNvbmZpZy0+Zm9udF93ZWlnaHQgJiYKICAgICAgICAhbGYtPmxmSXRhbGljICYmICFsZi0+bGZVbmRlcmxpbmUgJiYgIWxmLT5sZlN0cmlrZU91dCAmJgogICAgICAgICFsc3RyY21wKGxmLT5sZkZhY2VOYW1lLCBjb25maWctPmZhY2VfbmFtZSk7Cn0KCnN0cnVjdCBmb250X2Nob29zZXIgCnsKICAgIHN0cnVjdCBpbm5lcl9kYXRhKglkYXRhOwogICAgaW50CQkJZG9uZTsKfTsKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9WYWxpZGF0ZUZvbnRNZXRyaWMKICoKICogUmV0dXJucyB0cnVlIGlmIHRoZSBmb250IGRlc2NyaWJlZCBpbiB0bSBpcyB1c2FibGUgYXMgYSBmb250IGZvciB0aGUgcmVuZGVyZXIKICovCkJPT0wJV0NVU0VSX1ZhbGlkYXRlRm9udE1ldHJpYyhjb25zdCBzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgY29uc3QgVEVYVE1FVFJJQyogdG0sIERXT1JEIGZvbnRUeXBlKQp7CiAgICBCT09MICAgICAgICByZXQgPSBUUlVFOwoKICAgIGlmIChmb250VHlwZSAmIFJBU1RFUl9GT05UVFlQRSkKICAgICAgICByZXQgPSAodG0tPnRtTWF4Q2hhcldpZHRoICogZGF0YS0+Y3VyY2ZnLndpbl93aWR0aCA8IEdldFN5c3RlbU1ldHJpY3MoU01fQ1hTQ1JFRU4pICYmCiAgICAgICAgICAgICAgIHRtLT50bUhlaWdodCAqIGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0IDwgR2V0U3lzdGVtTWV0cmljcyhTTV9DWVNDUkVFTikpOwogICAgcmV0dXJuIHJldCAmJiAhdG0tPnRtSXRhbGljICYmICF0bS0+dG1VbmRlcmxpbmVkICYmICF0bS0+dG1TdHJ1Y2tPdXQgJiYKICAgICAgICAodG0tPnRtQ2hhclNldCA9PSBERUZBVUxUX0NIQVJTRVQgLyp8fCB0bS0+dG1DaGFyU2V0ID09IEFOU0lfQ0hBUlNFVCovKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1ZhbGlkYXRlRm9udAogKgogKiBSZXR1cm5zIHRydWUgaWYgdGhlIGZvbnQgZmFtaWx5IGRlc2NyaWJlZCBpbiBsZiBpcyB1c2FibGUgYXMgYSBmb250IGZvciB0aGUgcmVuZGVyZXIKICovCkJPT0wJV0NVU0VSX1ZhbGlkYXRlRm9udChjb25zdCBzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgY29uc3QgTE9HRk9OVCogbGYpCnsKICAgIHJldHVybiAobGYtPmxmUGl0Y2hBbmRGYW1pbHkgJiAzKSA9PSBGSVhFRF9QSVRDSCAmJiAKICAgICAgICAvKiAobGYtPmxmUGl0Y2hBbmRGYW1pbHkgJiAweEYwKSA9PSBGRl9NT0RFUk4gJiYgKi8KICAgICAgICAobGYtPmxmQ2hhclNldCA9PSBERUZBVUxUX0NIQVJTRVQgfHwgbGYtPmxmQ2hhclNldCA9PSBBTlNJX0NIQVJTRVQpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfZmlyc3RfZm9udF9lbnVtXzIKICoJCWdldF9maXJzdF9mb250X2VudW0KICoKICogSGVscGVyIGZ1bmN0aW9ucyB0byBnZXQgYSBkZWNlbnQgZm9udCBmb3IgdGhlIHJlbmRlcmVyCiAqLwpzdGF0aWMgaW50IENBTExCQUNLIGdldF9maXJzdF9mb250X2VudW1fMihjb25zdCBMT0dGT05UKiBsZiwgY29uc3QgVEVYVE1FVFJJQyogdG0sIAoJCQkJCSAgRFdPUkQgRm9udFR5cGUsIExQQVJBTSBsUGFyYW0pCnsKICAgIHN0cnVjdCBmb250X2Nob29zZXIqCWZjID0gKHN0cnVjdCBmb250X2Nob29zZXIqKWxQYXJhbTsKCiAgICBXQ1VTRVJfRHVtcFRleHRNZXRyaWModG0sIEZvbnRUeXBlKTsKCiAgICBpZiAoV0NVU0VSX1ZhbGlkYXRlRm9udE1ldHJpYyhmYy0+ZGF0YSwgdG0sIEZvbnRUeXBlKSAmJiAKICAgICAgICBXQ1VTRVJfU2V0Rm9udChmYy0+ZGF0YSwgbGYpKQogICAgewogICAgICAgIGZjLT5kb25lID0gMTsKICAgICAgICAvKiBzaW5jZSB3ZSd2ZSBtb2RpZmllZCB0aGUgY3VycmVudCBjb25maWcgd2l0aCBuZXcgZm9udCBpbmZvcm1hdGlvbiwgCiAgICAgICAgICogc2V0IGl0IGFzIHRoZSBuZXcgZGVmYXVsdC4KICAgICAgICAgKiBGb3JjZSBhbHNvIGl0cyB3cml0aW5nIGJhY2sgdG8gdGhlIHJlZ2lzdHJ5IHNvIHRoYXQgd2UgY2FuIGdldCBpdAogICAgICAgICAqIHRoZSBuZXh0IHRpbWUKICAgICAgICAgKi8KICAgICAgICBmYy0+ZGF0YS0+ZGVmY2ZnID0gZmMtPmRhdGEtPmN1cmNmZzsKICAgICAgICBXSU5FQ09OX1JlZ1NhdmUoJmZjLT5kYXRhLT5kZWZjZmcpOwogICAgICAgIHJldHVybiAwOwogICAgfQogICAgcmV0dXJuIDE7Cn0KCnN0YXRpYyBpbnQgQ0FMTEJBQ0sgZ2V0X2ZpcnN0X2ZvbnRfZW51bShjb25zdCBMT0dGT05UKiBsZiwgY29uc3QgVEVYVE1FVFJJQyogdG0sIAoJCQkJCURXT1JEIEZvbnRUeXBlLCBMUEFSQU0gbFBhcmFtKQp7CiAgICBzdHJ1Y3QgZm9udF9jaG9vc2VyKglmYyA9IChzdHJ1Y3QgZm9udF9jaG9vc2VyKilsUGFyYW07CgogICAgV0NVU0VSX0R1bXBMb2dGb250KCJpbml0IiwgbGYsIEZvbnRUeXBlKTsKCiAgICBpZiAoV0NVU0VSX1ZhbGlkYXRlRm9udChmYy0+ZGF0YSwgbGYpKQogICAgewoJRW51bUZvbnRGYW1pbGllcyhQUklWQVRFKGZjLT5kYXRhKS0+aE1lbURDLCBsZi0+bGZGYWNlTmFtZSwgCiAgICAgICAgICAgICAgICAgICAgICAgICBnZXRfZmlyc3RfZm9udF9lbnVtXzIsIGxQYXJhbSk7CglyZXR1cm4gIWZjLT5kb25lOyAvKiB3ZSBqdXN0IG5lZWQgdGhlIGZpcnN0IG1hdGNoaW5nIG9uZS4uLiAqLwogICAgfQogICAgcmV0dXJuIDE7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUNvcHlGb250CiAqCiAqIGdldCB0aGUgcmVsZXZhbnQgaW5mb3JtYXRpb24gZnJvbSB0aGUgZm9udCBkZXNjcmliZWQgaW4gbGYgYW5kIHN0b3JlIHRoZW0KICogaW4gY29uZmlnCiAqLwpIRk9OVCBXQ1VTRVJfQ29weUZvbnQoc3RydWN0IGNvbmZpZ19kYXRhKiBjb25maWcsIEhXTkQgaFduZCwgY29uc3QgTE9HRk9OVCogbGYpCnsKICAgIFRFWFRNRVRSSUMgIHRtOwogICAgSERDICAgICAgICAgaERDOwogICAgSEZPTlQgICAgICAgaEZvbnQsIGhPbGRGb250OwogICAgaW50ICAgICAgICAgdywgaSwgYnVmWzI1Nl07CgogICAgaWYgKCEoaERDID0gR2V0REMoaFduZCkpKSByZXR1cm4gKEhGT05UKTA7CiAgICBpZiAoIShoRm9udCA9IENyZWF0ZUZvbnRJbmRpcmVjdChsZikpKSBnb3RvIGVycjE7CgogICAgaE9sZEZvbnQgPSBTZWxlY3RPYmplY3QoaERDLCBoRm9udCk7CiAgICBHZXRUZXh0TWV0cmljcyhoREMsICZ0bSk7CgogICAgLyogRklYTUU6CiAgICAgKiB0aGUgY3VycmVudCBmcmVldHlwZSBlbmdpbmUgKGF0IGxlYXN0IDIuMC54IHdpdGggeCA8PSA4KSBhbmQgaXRzIGltcGxlbWVudGF0aW9uCiAgICAgKiBpbiBXaW5lIGRvbid0IHJldHVybiBhZGVxdWF0ZSB2YWx1ZXMgZm9yIGZpeGVkIGZvbnRzCiAgICAgKiBJbiBXaW5kb3dzLCB0aG9zZSBmb250cyBhcmUgZXhwZWN0ZXMgdG8gcmV0dXJuIHRoZSBzYW1lIHZhbHVlIGZvcgogICAgICogIC0gdGhlIGF2ZXJhZ2Ugd2lkdGgKICAgICAqICAtIHRoZSBsYXJnZXN0IHdpZHRoCiAgICAgKiAgLSB0aGUgd2lkdGggb2YgYWxsIGNoYXJhY3RlcnMgaW4gdGhlIGZvbnQKICAgICAqIFRoaXMgaXNuJ3QgdHJ1ZSBpbiBXaW5lLiBBcyBhIHRlbXBvcmFyeSB3b3JrYW91bmQsIHdlIGdldCBhcyB0aGUgd2lkdGggb2YgdGhlIAogICAgICogY2VsbCwgdGhlIHdpZHRoIG9mIHRoZSBmaXJzdCBjaGFyYWN0ZXIgaW4gdGhlIGZvbnQsIGFmdGVyIGNoZWNraW5nIHRoYXQgYWxsCiAgICAgKiBjaGFyYWN0ZXJzIGluIHRoZSBmb250IGhhdmUgdGhlIHNhbWUgd2lkdGggKEkgaGVhciBwYXJhbm/vYSBjb21pbmcpCiAgICAgKiB3aGVuIHRoaXMgZ2V0cyBmaXhlZCwgdGhlIHNob3VsZCBiZSB1c2luZyB0bS50bUF2ZUNoYXJXaWR0aCBvciB0bS50bU1heENoYXJXaWR0aAogICAgICogYXMgdGhlIGNlbGwgd2lkdGguCiAgICAgKi8KICAgIEdldENoYXJXaWR0aDMyKGhEQywgdG0udG1GaXJzdENoYXIsIHRtLnRtRmlyc3RDaGFyLCAmdyk7CiAgICBmb3IgKGkgPSB0bS50bUZpcnN0Q2hhciArIDE7IGkgPD0gdG0udG1MYXN0Q2hhcjsgaSArPSBzaXplb2YoYnVmKSAvIHNpemVvZihidWZbMF0pKQogICAgewogICAgICAgIGludCBqLCBsOwogICAgICAgICAgICAKICAgICAgICBsID0gbWluKHRtLnRtTGFzdENoYXIgLSBpLCBzaXplb2YoYnVmKSAvIHNpemVvZihidWZbMF0pIC0gMSk7CiAgICAgICAgR2V0Q2hhcldpZHRoMzIoaERDLCBpLCBpICsgbCwgYnVmKTsKICAgICAgICBmb3IgKGogPSAwOyBqIDw9IGw7IGorKykgCiAgICAgICAgewogICAgICAgICAgICBpZiAoYnVmW2pdICE9IHcpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFdJTkVfV0FSTigiTm9uIHVuaWZvcm0gY2VsbCB3aWR0aDogWyVkXT0lZCBbJWRdPSVkXG4iLCAKICAgICAgICAgICAgICAgICAgICAgICAgICBpICsgaiwgYnVmW2pdLCB0bS50bUZpcnN0Q2hhciwgdyk7CiAgICAgICAgICAgICAgICBnb3RvIGVycjsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICBTZWxlY3RPYmplY3QoaERDLCBoT2xkRm9udCk7CiAgICBSZWxlYXNlREMoaFduZCwgaERDKTsKCiAgICBjb25maWctPmNlbGxfd2lkdGggID0gdzsKICAgIGNvbmZpZy0+Y2VsbF9oZWlnaHQgPSB0bS50bUhlaWdodDsKICAgIGNvbmZpZy0+Zm9udF93ZWlnaHQgPSB0bS50bVdlaWdodDsKICAgIGxzdHJjcHkoY29uZmlnLT5mYWNlX25hbWUsIGxmLT5sZkZhY2VOYW1lKTsKCiAgICByZXR1cm4gaEZvbnQ7CiBlcnI6CiAgICBpZiAoaERDICYmIGhPbGRGb250KSBTZWxlY3RPYmplY3QoaERDLCBoT2xkRm9udCk7CiAgICBpZiAoaEZvbnQpIERlbGV0ZU9iamVjdChoRm9udCk7CiBlcnIxOgogICAgaWYgKGhEQykgUmVsZWFzZURDKGhXbmQsIGhEQyk7CgogICAgcmV0dXJuIChIRk9OVCkwOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfRmlsbExvZ0ZvbnQKICoKICoKICovCnZvaWQgICAgV0NVU0VSX0ZpbGxMb2dGb250KExPR0ZPTlQqIGxmLCBjb25zdCBXQ0hBUiogbmFtZSwgVUlOVCBoZWlnaHQsIFVJTlQgd2VpZ2h0KQp7CiAgICBsZi0+bGZIZWlnaHQgICAgICAgID0gaGVpZ2h0OwogICAgbGYtPmxmV2lkdGggICAgICAgICA9IDA7CiAgICBsZi0+bGZFc2NhcGVtZW50ICAgID0gMDsKICAgIGxmLT5sZk9yaWVudGF0aW9uICAgPSAwOwogICAgbGYtPmxmV2VpZ2h0ICAgICAgICA9IHdlaWdodDsKICAgIGxmLT5sZkl0YWxpYyAgICAgICAgPSBGQUxTRTsKICAgIGxmLT5sZlVuZGVybGluZSAgICAgPSBGQUxTRTsKICAgIGxmLT5sZlN0cmlrZU91dCAgICAgPSBGQUxTRTsKICAgIGxmLT5sZkNoYXJTZXQgICAgICAgPSBERUZBVUxUX0NIQVJTRVQ7CiAgICBsZi0+bGZPdXRQcmVjaXNpb24gID0gT1VUX0RFRkFVTFRfUFJFQ0lTOwogICAgbGYtPmxmQ2xpcFByZWNpc2lvbiA9IENMSVBfREVGQVVMVF9QUkVDSVM7IAogICAgbGYtPmxmUXVhbGl0eSAgICAgICA9IERFRkFVTFRfUVVBTElUWTsKICAgIGxmLT5sZlBpdGNoQW5kRmFtaWx5ID0gRklYRURfUElUQ0ggfCBGRl9ET05UQ0FSRTsKICAgIGxzdHJjcHkobGYtPmxmRmFjZU5hbWUsIG5hbWUpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfU2V0Rm9udAogKgogKiBzZXRzIGxvZ2ZvbnQgYXMgdGhlIG5ldyBmb250IGZvciB0aGUgY29uc29sZQogKi8KQk9PTAlXQ1VTRVJfU2V0Rm9udChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgY29uc3QgTE9HRk9OVCogbG9nZm9udCkKewogICAgSEZPTlQgICAgICAgaEZvbnQ7CgogICAgaWYgKFBSSVZBVEUoZGF0YSktPmhGb250ICE9IDAgJiYgV0NVU0VSX0FyZUZvbnRzRXF1YWwoJmRhdGEtPmN1cmNmZywgbG9nZm9udCkpIAogICAgICAgIHJldHVybiBUUlVFOwoKICAgIGhGb250ID0gV0NVU0VSX0NvcHlGb250KCZkYXRhLT5jdXJjZmcsIFBSSVZBVEUoZGF0YSktPmhXbmQsIGxvZ2ZvbnQpOwogICAgaWYgKCFoRm9udCkge1dJTkVfRVJSKCJ3cm9uZyBmb250XG4iKTsgcmV0dXJuIEZBTFNFO30KCiAgICBpZiAoUFJJVkFURShkYXRhKS0+aEZvbnQpIERlbGV0ZU9iamVjdChQUklWQVRFKGRhdGEpLT5oRm9udCk7CiAgICBQUklWQVRFKGRhdGEpLT5oRm9udCA9IGhGb250OwoKICAgIFdDVVNFUl9Db21wdXRlUG9zaXRpb25zKGRhdGEpOwogICAgV0NVU0VSX05ld0JpdG1hcChkYXRhLCBUUlVFKTsKICAgIEludmFsaWRhdGVSZWN0KFBSSVZBVEUoZGF0YSktPmhXbmQsIE5VTEwsIEZBTFNFKTsKICAgIFVwZGF0ZVdpbmRvdyhQUklWQVRFKGRhdGEpLT5oV25kKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0luaXRGb250CiAqCiAqIGNyZWF0ZSBhIGhGb250IGZyb20gdGhlIHNldHRpbmdzIHNhdmVkIGluIHJlZ2lzdHJ5Li4uCiAqIChjYWxsZWQgb24gaW5pdCwgYXNzdW1pbmcgbm8gZm9udCBoYXMgYmVlbiBjcmVhdGVkIGJlZm9yZSkKICovCnN0YXRpYyBCT09MCVdDVVNFUl9Jbml0Rm9udChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSkKewogICAgc3RydWN0IGZvbnRfY2hvb3NlciBmYzsKCiAgICBpZiAoZGF0YS0+Y3VyY2ZnLmZhY2VfbmFtZVswXSAhPSAnXDAnICYmCiAgICAgICAgZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0ICE9IDAgJiYKICAgICAgICBkYXRhLT5jdXJjZmcuZm9udF93ZWlnaHQgIT0gMCkKICAgIHsKICAgICAgICBMT0dGT05UICAgICAgICAgICAgIGxmOwoKICAgICAgICBXQ1VTRVJfRmlsbExvZ0ZvbnQoJmxmLCBkYXRhLT5jdXJjZmcuZmFjZV9uYW1lLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0LCBkYXRhLT5jdXJjZmcuZm9udF93ZWlnaHQpOwogICAgICAgIGlmIChQUklWQVRFKGRhdGEpLT5oRm9udCAhPSAwKSBXSU5FX0ZJWE1FKCJPaCBzdHJhbmdlXG4iKTsKCiAgICAgICAgaWYgKFdDVVNFUl9TZXRGb250KGRhdGEsICZsZikpIHJldHVybiBUUlVFOwogICAgfQoKICAgIC8qIHRyeSB0byBmaW5kIGFuIGFjY2VwdGFibGUgZm9udCAqLwogICAgV0lORV9XQVJOKCJDb3VsZG4ndCBtYXRjaCB0aGUgZm9udCBmcm9tIHJlZ2lzdHJ5Li4uIHRyeWluZyB0byBmaW5kIG9uZVxuIik7CiAgICBmYy5kYXRhID0gZGF0YTsKICAgIGZjLmRvbmUgPSAwOwogICAgRW51bUZvbnRGYW1pbGllcyhQUklWQVRFKGRhdGEpLT5oTWVtREMsIE5VTEwsIGdldF9maXJzdF9mb250X2VudW0sIChMUEFSQU0pJmZjKTsKICAgIHJldHVybiBmYy5kb25lOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfR2V0Q2VsbAogKgogKiBHZXQgYSBjZWxsIGZyb20gdGhlIGEgcmVsYXRpdmUgY29vcmRpbmF0ZSBpbiB3aW5kb3cgKHRha2VzIGludG8KICogYWNjb3VudCB0aGUgc2Nyb2xsaW5nKQogKi8Kc3RhdGljIENPT1JECVdDVVNFUl9HZXRDZWxsKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBMUEFSQU0gbFBhcmFtKQp7CiAgICBDT09SRAljOwoKICAgIGMuWCA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlggKyAoc2hvcnQpTE9XT1JEKGxQYXJhbSkgLyBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgIGMuWSA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKyAoc2hvcnQpSElXT1JEKGxQYXJhbSkgLyBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7CgogICAgcmV0dXJuIGM7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9HZXRTZWxlY3Rpb25SZWN0CiAqCiAqIEdldCB0aGUgc2VsZWN0aW9uIHJlY3RhbmdsZQogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX0dldFNlbGVjdGlvblJlY3QoY29uc3Qgc3RydWN0IGlubmVyX2RhdGEqIGRhdGEsIExQUkVDVCByKQp7CiAgICByLT5sZWZ0ICAgPSAobWluKFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5YLCBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIuWCkgICAgKSAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoOwogICAgci0+dG9wICAgID0gKG1pbihQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWSwgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkpICAgICkgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7CiAgICByLT5yaWdodCAgPSAobWF4KFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5YLCBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIuWCkgKyAxKSAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoOwogICAgci0+Ym90dG9tID0gKG1heChQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWSwgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkpICsgMSkgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9TZXRTZWxlY3Rpb24KICoKICoKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9TZXRTZWxlY3Rpb24oY29uc3Qgc3RydWN0IGlubmVyX2RhdGEqIGRhdGEsIEhEQyBoUmVmREMpCnsKICAgIEhEQwkJaERDOwogICAgUkVDVAlyOwoKICAgIFdDVVNFUl9HZXRTZWxlY3Rpb25SZWN0KGRhdGEsICZyKTsKICAgIGhEQyA9IGhSZWZEQyA/IGhSZWZEQyA6IEdldERDKFBSSVZBVEUoZGF0YSktPmhXbmQpOwogICAgaWYgKGhEQykKICAgIHsKCWlmIChQUklWQVRFKGRhdGEpLT5oV25kID09IEdldEZvY3VzKCkgJiYgZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlKQoJICAgIEhpZGVDYXJldChQUklWQVRFKGRhdGEpLT5oV25kKTsKCUludmVydFJlY3QoaERDLCAmcik7CglpZiAoaERDICE9IGhSZWZEQykKCSAgICBSZWxlYXNlREMoUFJJVkFURShkYXRhKS0+aFduZCwgaERDKTsKCWlmIChQUklWQVRFKGRhdGEpLT5oV25kID09IEdldEZvY3VzKCkgJiYgZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlKQoJICAgIFNob3dDYXJldChQUklWQVRFKGRhdGEpLT5oV25kKTsKICAgIH0KfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX01vdmVTZWxlY3Rpb24KICoKICoKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9Nb3ZlU2VsZWN0aW9uKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBDT09SRCBjMSwgQ09PUkQgYzIsIEJPT0wgZmluYWwpCnsKICAgIFJFQ1QJcjsKICAgIEhEQwkJaERDOwoKICAgIFdDVVNFUl9HZXRTZWxlY3Rpb25SZWN0KGRhdGEsICZyKTsKICAgIGhEQyA9IEdldERDKFBSSVZBVEUoZGF0YSktPmhXbmQpOwogICAgaWYgKGhEQykKICAgIHsKCWlmIChQUklWQVRFKGRhdGEpLT5oV25kID09IEdldEZvY3VzKCkgJiYgZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlKQoJICAgIEhpZGVDYXJldChQUklWQVRFKGRhdGEpLT5oV25kKTsKCUludmVydFJlY3QoaERDLCAmcik7CiAgICB9CiAgICBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEgPSBjMTsKICAgIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MiA9IGMyOwogICAgaWYgKGhEQykKICAgIHsKCVdDVVNFUl9HZXRTZWxlY3Rpb25SZWN0KGRhdGEsICZyKTsKCUludmVydFJlY3QoaERDLCAmcik7CglSZWxlYXNlREMoUFJJVkFURShkYXRhKS0+aFduZCwgaERDKTsKCWlmIChQUklWQVRFKGRhdGEpLT5oV25kID09IEdldEZvY3VzKCkgJiYgZGF0YS0+Y3VyY2ZnLmN1cnNvcl92aXNpYmxlKQoJICAgIFNob3dDYXJldChQUklWQVRFKGRhdGEpLT5oV25kKTsKICAgIH0KICAgIGlmIChmaW5hbCkKICAgIHsKCVJlbGVhc2VDYXB0dXJlKCk7CglQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID0gVFJVRTsKICAgIH0KfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0NvcHlTZWxlY3Rpb25Ub0NsaXBib2FyZAogKgogKiBDb3BpZXMgdGhlIGN1cnJlbnQgc2VsZWN0aW9uIGludG8gdGhlIGNsaXBib2FyZAogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX0NvcHlTZWxlY3Rpb25Ub0NsaXBib2FyZChjb25zdCBzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSkKewogICAgSEFORExFCWhNZW07CiAgICBMUFdTVFIJcDsKICAgIHVuc2lnbmVkCXcsIGg7CgogICAgdyA9IGFicyhQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWCAtIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0Mi5YKSArIDI7CiAgICBoID0gYWJzKFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5ZIC0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkpICsgMTsKCiAgICBpZiAoIU9wZW5DbGlwYm9hcmQoUFJJVkFURShkYXRhKS0+aFduZCkpIHJldHVybjsKICAgIEVtcHR5Q2xpcGJvYXJkKCk7CgogICAgaE1lbSA9IEdsb2JhbEFsbG9jKEdNRU1fTU9WRUFCTEUsICh3ICogaCAtIDEpICogc2l6ZW9mKFdDSEFSKSk7CiAgICBpZiAoaE1lbSAmJiAocCA9IEdsb2JhbExvY2soaE1lbSkpKQogICAgewoJQ09PUkQJYzsKCWludAl5OwoKCWMuWCA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlggKyBtaW4oUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLlgsIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0Mi5YKTsKCWMuWSA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKyBtaW4oUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLlksIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0Mi5ZKTsKCQoJZm9yICh5ID0gMDsgeSA8IGg7IHkrKywgYy5ZKyspCgl7CgkgICAgUmVhZENvbnNvbGVPdXRwdXRDaGFyYWN0ZXIoZGF0YS0+aENvbk91dCwgJnBbeSAqIHddLCB3IC0gMSwgYywgTlVMTCk7CgkgICAgaWYgKHkgPCBoIC0gMSkgcFt5ICogdyArIHcgLSAxXSA9ICdcbic7Cgl9CglHbG9iYWxVbmxvY2soaE1lbSk7CglTZXRDbGlwYm9hcmREYXRhKENGX1VOSUNPREVURVhULCBoTWVtKTsKICAgIH0KICAgIENsb3NlQ2xpcGJvYXJkKCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9QYXN0ZUZyb21DbGlwYm9hcmQKICoKICoKICovCnN0YXRpYyB2b2lkCVdDVVNFUl9QYXN0ZUZyb21DbGlwYm9hcmQoc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIEhBTkRMRQloOwogICAgV0NIQVIqCXB0cjsKCiAgICBpZiAoIU9wZW5DbGlwYm9hcmQoUFJJVkFURShkYXRhKS0+aFduZCkpIHJldHVybjsKICAgIGggPSBHZXRDbGlwYm9hcmREYXRhKENGX1VOSUNPREVURVhUKTsKICAgIGlmIChoICYmIChwdHIgPSBHbG9iYWxMb2NrKGgpKSkKICAgIHsKCWludAkJaSwgbGVuID0gR2xvYmFsU2l6ZShoKSAvIHNpemVvZihXQ0hBUik7CglJTlBVVF9SRUNPUkQJaXJbMl07CglEV09SRAkJbjsKCVNIT1JUCQlzaDsKCglpclswXS5FdmVudFR5cGUgPSBLRVlfRVZFTlQ7CglpclswXS5FdmVudC5LZXlFdmVudC53UmVwZWF0Q291bnQgPSAwOwoJaXJbMF0uRXZlbnQuS2V5RXZlbnQuZHdDb250cm9sS2V5U3RhdGUgPSAwOwoJaXJbMF0uRXZlbnQuS2V5RXZlbnQuYktleURvd24gPSBUUlVFOwoKCS8qIGdlbmVyYXRlIHRoZSBjb3JyZXNwb25kaW5nIGlucHV0IHJlY29yZHMgKi8KCWZvciAoaSA9IDA7IGkgPCBsZW47IGkrKykKCXsKCSAgICAvKiBGSVhNRTogdGhlIG1vZGlmeWluZyBrZXlzIGFyZSBub3QgZ2VuZXJhdGVkIChzaGlmdCwgY3RybC4uLikgKi8KCSAgICBzaCA9IFZrS2V5U2NhbihwdHJbaV0pOwoJICAgIGlyWzBdLkV2ZW50LktleUV2ZW50LndWaXJ0dWFsS2V5Q29kZSA9IExPQllURShzaCk7CgkgICAgaXJbMF0uRXZlbnQuS2V5RXZlbnQud1ZpcnR1YWxTY2FuQ29kZSA9IE1hcFZpcnR1YWxLZXkoTE9CWVRFKHNoKSwgMCk7CgkgICAgaXJbMF0uRXZlbnQuS2V5RXZlbnQudUNoYXIuVW5pY29kZUNoYXIgPSBwdHJbaV07CgkgICAgCgkgICAgaXJbMV0gPSBpclswXTsKCSAgICBpclsxXS5FdmVudC5LZXlFdmVudC5iS2V5RG93biA9IEZBTFNFOwoJICAgIAoJICAgIFdyaXRlQ29uc29sZUlucHV0KGRhdGEtPmhDb25JbiwgaXIsIDIsICZuKTsKCX0KCUdsb2JhbFVubG9jayhoKTsKICAgIH0KICAgIENsb3NlQ2xpcGJvYXJkKCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVJlZnJlc2gKICoKICoKICovCnN0YXRpYyB2b2lkIFdDVVNFUl9SZWZyZXNoKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBpbnQgdHAsIGludCBibSkKewogICAgV0NVU0VSX0ZpbGxNZW1EQyhkYXRhLCB0cCwgYm0pOwogICAgaWYgKGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgPD0gYm0gJiYgZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWSArIGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0ID49IHRwKQogICAgewoJUkVDVAlyOwoKCXIubGVmdCAgID0gMDsKCXIucmlnaHQgID0gZGF0YS0+Y3VyY2ZnLndpbl93aWR0aCAqIGRhdGEtPmN1cmNmZy5jZWxsX3dpZHRoOwoJci50b3AgICAgPSAodHAgLSBkYXRhLT5jdXJjZmcud2luX3Bvcy5ZKSAqIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodDsKCXIuYm90dG9tID0gKGJtIC0gZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWSArIDEpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0OwoJSW52YWxpZGF0ZVJlY3QoUFJJVkFURShkYXRhKS0+aFduZCwgJnIsIEZBTFNFKTsKCVVwZGF0ZVdpbmRvdyhQUklWQVRFKGRhdGEpLT5oV25kKTsKICAgIH0KfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1BhaW50CiAqCiAqCiAqLwpzdGF0aWMgdm9pZAlXQ1VTRVJfUGFpbnQoY29uc3Qgc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIFBBSU5UU1RSVUNUCQlwczsKCiAgICBCZWdpblBhaW50KFBSSVZBVEUoZGF0YSktPmhXbmQsICZwcyk7CiAgICBCaXRCbHQocHMuaGRjLCAwLCAwLCAKICAgICAgICAgICBkYXRhLT5jdXJjZmcud2luX3dpZHRoICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsIAogICAgICAgICAgIGRhdGEtPmN1cmNmZy53aW5faGVpZ2h0ICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0LAoJICAgUFJJVkFURShkYXRhKS0+aE1lbURDLCAKICAgICAgICAgICBkYXRhLT5jdXJjZmcud2luX3Bvcy5YICogZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsIAogICAgICAgICAgIGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgKiBkYXRhLT5jdXJjZmcuY2VsbF9oZWlnaHQsCgkgICBTUkNDT1BZKTsKICAgIGlmIChQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uKQoJV0NVU0VSX1NldFNlbGVjdGlvbihkYXRhLCBwcy5oZGMpOwogICAgRW5kUGFpbnQoUFJJVkFURShkYXRhKS0+aFduZCwgJnBzKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX1Njcm9sbAogKgogKgogKi8Kc3RhdGljIHZvaWQgV0NVU0VSX1Njcm9sbChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgaW50IHBvcywgQk9PTCBob3J6KQp7CiAgICBpZiAoaG9yeikKICAgIHsKCVNldFNjcm9sbFBvcyhQUklWQVRFKGRhdGEpLT5oV25kLCBTQl9IT1JaLCBwb3MsIFRSVUUpOwoJZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCA9IHBvczsKICAgIH0KICAgIGVsc2UKICAgIHsKCVNldFNjcm9sbFBvcyhQUklWQVRFKGRhdGEpLT5oV25kLCBTQl9WRVJULCBwb3MsIFRSVUUpOwoJZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWSA9IHBvczsKICAgIH0KICAgIEludmFsaWRhdGVSZWN0KFBSSVZBVEUoZGF0YSktPmhXbmQsIE5VTEwsIEZBTFNFKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0ZpbGxNZW51CiAqCiAqCiAqLwpzdGF0aWMgQk9PTCBXQ1VTRVJfRmlsbE1lbnUoSE1FTlUgaE1lbnUsIEJPT0wgc2VwKQp7CiAgICBITUVOVQkJaFN1Yk1lbnU7CiAgICBISU5TVEFOQ0UJCWhJbnN0YW5jZSA9IEdldE1vZHVsZUhhbmRsZShOVUxMKTsKICAgIFdDSEFSCQlidWZmWzI1Nl07CgogICAgaWYgKCFoTWVudSkgcmV0dXJuIEZBTFNFOwoKICAgIC8qIEZJWE1FOiBlcnJvciBoYW5kbGluZyAmIG1lbW9yeSBjbGVhbnVwICovCiAgICBoU3ViTWVudSA9IENyZWF0ZU1lbnUoKTsKICAgIGlmICghaFN1Yk1lbnUpIHJldHVybiBGQUxTRTsKCiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX01BUkssIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoU3ViTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfTUFSSywgYnVmZik7CiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX0NPUFksIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoU3ViTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfQ09QWSwgYnVmZik7CiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX1BBU1RFLCBidWZmLCBzaXplb2YoYnVmZikgLyBzaXplb2YoV0NIQVIpKTsKICAgIEluc2VydE1lbnUoaFN1Yk1lbnUsIC0xLCBNRl9CWVBPU0lUSU9OfE1GX1NUUklORywgSURTX1BBU1RFLCBidWZmKTsKICAgIExvYWRTdHJpbmcoaEluc3RhbmNlLCBJRFNfU0VMRUNUQUxMLCBidWZmLCBzaXplb2YoYnVmZikgLyBzaXplb2YoV0NIQVIpKTsKICAgIEluc2VydE1lbnUoaFN1Yk1lbnUsIC0xLCBNRl9CWVBPU0lUSU9OfE1GX1NUUklORywgSURTX1NFTEVDVEFMTCwgYnVmZik7CiAgICBMb2FkU3RyaW5nKGhJbnN0YW5jZSwgSURTX1NDUk9MTCwgYnVmZiwgc2l6ZW9mKGJ1ZmYpIC8gc2l6ZW9mKFdDSEFSKSk7CiAgICBJbnNlcnRNZW51KGhTdWJNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TVFJJTkcsIElEU19TQ1JPTEwsIGJ1ZmYpOwogICAgTG9hZFN0cmluZyhoSW5zdGFuY2UsIElEU19TRUFSQ0gsIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoU3ViTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfU0VBUkNILCBidWZmKTsKCiAgICBpZiAoc2VwKSBJbnNlcnRNZW51KGhNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TRVBBUkFUT1IsIDAsIE5VTEwpOwogICAgTG9hZFN0cmluZyhoSW5zdGFuY2UsIElEU19FRElULCBidWZmLCBzaXplb2YoYnVmZikgLyBzaXplb2YoV0NIQVIpKTsKICAgIEluc2VydE1lbnUoaE1lbnUsIC0xLCBNRl9CWVBPU0lUSU9OfE1GX1NUUklOR3xNRl9QT1BVUCwgKFVJTlRfUFRSKWhTdWJNZW51LCBidWZmKTsKICAgIExvYWRTdHJpbmcoaEluc3RhbmNlLCBJRFNfREVGQVVMVCwgYnVmZiwgc2l6ZW9mKGJ1ZmYpIC8gc2l6ZW9mKFdDSEFSKSk7CiAgICBJbnNlcnRNZW51KGhNZW51LCAtMSwgTUZfQllQT1NJVElPTnxNRl9TVFJJTkcsIElEU19ERUZBVUxULCBidWZmKTsKICAgIExvYWRTdHJpbmcoaEluc3RhbmNlLCBJRFNfUFJPUEVSVFksIGJ1ZmYsIHNpemVvZihidWZmKSAvIHNpemVvZihXQ0hBUikpOwogICAgSW5zZXJ0TWVudShoTWVudSwgLTEsIE1GX0JZUE9TSVRJT058TUZfU1RSSU5HLCBJRFNfUFJPUEVSVFksIGJ1ZmYpOwoKICAgIHJldHVybiBUUlVFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfQ3JlYXRlCiAqCiAqIENyZWF0ZXMgdGhlIHdpbmRvdyBmb3IgdGhlIHJlbmRlcmluZwogKi8Kc3RhdGljIExSRVNVTFQgV0NVU0VSX0NyZWF0ZShIV05EIGhXbmQsIExQQ1JFQVRFU1RSVUNUIGxwY3MpCnsKICAgIHN0cnVjdCBpbm5lcl9kYXRhKglkYXRhOwogICAgSE1FTlUJCWhTeXNNZW51OwoKICAgIGRhdGEgPSBscGNzLT5scENyZWF0ZVBhcmFtczsKICAgIFNldFdpbmRvd0xvbmcoaFduZCwgMEwsIChEV09SRClkYXRhKTsKICAgIFBSSVZBVEUoZGF0YSktPmhXbmQgPSBoV25kOwoKICAgIGRhdGEtPmN1cmNmZy5jdXJzb3Jfc2l6ZSA9IDEwMTsgLyogaW52YWxpZCB2YWx1ZSwgd2lsbCB0cmlnZ2VyIGEgY29tcGxldGUgY2xlYW51cCAqLwoKICAgIGhTeXNNZW51ID0gR2V0U3lzdGVtTWVudShoV25kLCBGQUxTRSk7CiAgICBpZiAoIWhTeXNNZW51KSByZXR1cm4gMDsKICAgIFBSSVZBVEUoZGF0YSktPmhQb3BNZW51ID0gQ3JlYXRlUG9wdXBNZW51KCk7CiAgICBpZiAoIVBSSVZBVEUoZGF0YSktPmhQb3BNZW51KSByZXR1cm4gMDsKCiAgICBXQ1VTRVJfRmlsbE1lbnUoaFN5c01lbnUsIFRSVUUpOwogICAgV0NVU0VSX0ZpbGxNZW51KFBSSVZBVEUoZGF0YSktPmhQb3BNZW51LCBGQUxTRSk7CgogICAgUFJJVkFURShkYXRhKS0+aE1lbURDID0gQ3JlYXRlQ29tcGF0aWJsZURDKDApOwogICAgaWYgKCFQUklWQVRFKGRhdGEpLT5oTWVtREMpIHtXSU5FX0VSUigibm8gbWVtIGRjXG4iKTtyZXR1cm4gMDt9CgogICAgZGF0YS0+Y3VyY2ZnLnF1aWNrX2VkaXQgPSBGQUxTRTsKICAgIHJldHVybiAwOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfU2V0TWVudURldGFpbHMKICoKICogR3JheXMgLyB1bmdyYXlzIHRoZSBtZW51IGl0ZW1zIGFjY29yZGluZyB0byB0aGVpciBzdGF0ZQogKi8Kc3RhdGljIHZvaWQJV0NVU0VSX1NldE1lbnVEZXRhaWxzKGNvbnN0IHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhLCBITUVOVSBoTWVudSkKewogICAgaWYgKCFoTWVudSkge1dJTkVfRVJSKCJJc3N1ZSBpbiBnZXR0aW5nIG1lbnUgYml0c1xuIik7cmV0dXJuO30KCiAgICBFbmFibGVNZW51SXRlbShoTWVudSwgSURTX0NPUFksIAogICAgICAgICAgICAgICAgICAgTUZfQllDT01NQU5EfChQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID8gTUZfRU5BQkxFRCA6IE1GX0dSQVlFRCkpOwogICAgRW5hYmxlTWVudUl0ZW0oaE1lbnUsIElEU19QQVNURSwgCgkJICAgTUZfQllDT01NQU5EfChJc0NsaXBib2FyZEZvcm1hdEF2YWlsYWJsZShDRl9VTklDT0RFVEVYVCkgCgkJCQkgPyBNRl9FTkFCTEVEIDogTUZfR1JBWUVEKSk7CiAgICBFbmFibGVNZW51SXRlbShoTWVudSwgSURTX1NDUk9MTCwgTUZfQllDT01NQU5EfE1GX0dSQVlFRCk7CiAgICBFbmFibGVNZW51SXRlbShoTWVudSwgSURTX1NFQVJDSCwgTUZfQllDT01NQU5EfE1GX0dSQVlFRCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUNVU0VSX0dldEN0cmxLZXlTdGF0ZQogKgogKiBHZXQgdGhlIGNvbnNvbGUgYml0IG1hc2sgZXF1aXZhbGVudCB0byB0aGUgVktfIHN0YXR1cyBpbiBrZXlTdGF0ZQogKi8Kc3RhdGljIERXT1JEICAgIFdDVVNFUl9HZXRDdHJsS2V5U3RhdGUoQllURSoga2V5U3RhdGUpCnsKICAgIERXT1JEICAgICAgICAgICAgICAgcmV0ID0gMDsKCiAgICBHZXRLZXlib2FyZFN0YXRlKGtleVN0YXRlKTsKICAgIGlmIChrZXlTdGF0ZVtWS19TSElGVF0gICAgJiAweDgwKQlyZXQgfD0gU0hJRlRfUFJFU1NFRDsKICAgIGlmIChrZXlTdGF0ZVtWS19DT05UUk9MXSAgJiAweDgwKQlyZXQgfD0gTEVGVF9DVFJMX1BSRVNTRUQ7IC8qIEZJWE1FOiBnb3R0YSBjaG9vc2Ugb25lICovCiAgICBpZiAoa2V5U3RhdGVbVktfTENPTlRST0xdICYgMHg4MCkJcmV0IHw9IExFRlRfQ1RSTF9QUkVTU0VEOwogICAgaWYgKGtleVN0YXRlW1ZLX1JDT05UUk9MXSAmIDB4ODApCXJldCB8PSBSSUdIVF9DVFJMX1BSRVNTRUQ7CiAgICBpZiAoa2V5U3RhdGVbVktfTE1FTlVdICAgICYgMHg4MCkJcmV0IHw9IExFRlRfQUxUX1BSRVNTRUQ7CiAgICBpZiAoa2V5U3RhdGVbVktfUk1FTlVdICAgICYgMHg4MCkJcmV0IHw9IFJJR0hUX0FMVF9QUkVTU0VEOwogICAgaWYgKGtleVN0YXRlW1ZLX0NBUElUQUxdICAmIDB4MDEpCXJldCB8PSBDQVBTTE9DS19PTjsJICAgIAogICAgaWYgKGtleVN0YXRlW1ZLX05VTUxPQ0tdICAmIDB4MDEpCXJldCB8PSBOVU1MT0NLX09OOwogICAgaWYgKGtleVN0YXRlW1ZLX1NDUk9MTF0gICAmIDB4MDEpCXJldCB8PSBTQ1JPTExMT0NLX09OOwogICAgCiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfSGFuZGxlU2VsZWN0aW9uS2V5CiAqCiAqIEhhbmRsZXMga2V5cyB3aGlsZSBzZWxlY3RpbmcgYW4gYXJlYQogKi8Kc3RhdGljIHZvaWQgV0NVU0VSX0hhbmRsZVNlbGVjdGlvbktleShzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgQk9PTCBkb3duLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBXUEFSQU0gd1BhcmFtLCBMUEFSQU0gbFBhcmFtKQp7CiAgICBCWVRFCWtleVN0YXRlWzI1Nl07CiAgICBEV09SRCAgICAgICBzdGF0ZSA9IFdDVVNFUl9HZXRDdHJsS2V5U3RhdGUoa2V5U3RhdGUpICYgfihDQVBTTE9DS19PTnxOVU1MT0NLX09OfFNDUk9MTExPQ0tfT04pOwogICAgQ09PUkQgICAgICAgYzEsIGMyOwoKICAgIGlmIChkb3duKSByZXR1cm47CgogICAgc3dpdGNoIChzdGF0ZSkKICAgIHsKICAgIGNhc2UgMDoKICAgICAgICBzd2l0Y2ggKHdQYXJhbSkKICAgICAgICB7CiAgICAgICAgY2FzZSBWS19SRVRVUk46CiAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBGQUxTRTsKICAgICAgICAgICAgV0NVU0VSX1NldFNlbGVjdGlvbihkYXRhLCAwKTsKICAgICAgICAgICAgV0NVU0VSX0NvcHlTZWxlY3Rpb25Ub0NsaXBib2FyZChkYXRhKTsKICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgY2FzZSBWS19SSUdIVDoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMS5YKys7IGMyLlgrKzsKICAgICAgICAgICAgaWYgKGMxLlggPCBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggJiYgYzIuWCA8IGRhdGEtPmN1cmNmZy5zYl93aWR0aCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyLCBGQUxTRSk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgY2FzZSBWS19MRUZUOgogICAgICAgICAgICBjMSA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MTsKICAgICAgICAgICAgYzIgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDI7CiAgICAgICAgICAgIGMxLlgtLTsgYzIuWC0tOwogICAgICAgICAgICBpZiAoYzEuWCA+PSAwICYmIGMyLlggPj0gMCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyLCBGQUxTRSk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgY2FzZSBWS19VUDoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMS5ZLS07IGMyLlktLTsKICAgICAgICAgICAgaWYgKGMxLlkgPj0gMCAmJiBjMi5ZID49IDApCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFdDVVNFUl9Nb3ZlU2VsZWN0aW9uKGRhdGEsIGMxLCBjMiwgRkFMU0UpOwogICAgICAgICAgICB9CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIGNhc2UgVktfRE9XTjoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMS5ZKys7IGMyLlkrKzsKICAgICAgICAgICAgaWYgKGMxLlggPCBkYXRhLT5jdXJjZmcuc2JfaGVpZ2h0ICYmIGMyLlggPCBkYXRhLT5jdXJjZmcuc2JfaGVpZ2h0KQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBXQ1VTRVJfTW92ZVNlbGVjdGlvbihkYXRhLCBjMSwgYzIsIEZBTFNFKTsKICAgICAgICAgICAgfQogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIFNISUZUX1BSRVNTRUQ6CiAgICAgICAgc3dpdGNoICh3UGFyYW0pCiAgICAgICAgewogICAgICAgIGNhc2UgVktfUklHSFQ6CiAgICAgICAgICAgIGMxID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxOwogICAgICAgICAgICBjMiA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MjsKICAgICAgICAgICAgYzIuWCsrOwogICAgICAgICAgICBpZiAoYzIuWCA8IGRhdGEtPmN1cmNmZy5zYl93aWR0aCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyLCBGQUxTRSk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgY2FzZSBWS19MRUZUOgogICAgICAgICAgICBjMSA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MTsKICAgICAgICAgICAgYzIgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDI7CiAgICAgICAgICAgIGMyLlgtLTsKICAgICAgICAgICAgaWYgKGMyLlggPj0gYzEuWCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyLCBGQUxTRSk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgY2FzZSBWS19VUDoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMi5ZLS07CiAgICAgICAgICAgIGlmIChjMi5ZID49IGMxLlkpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFdDVVNFUl9Nb3ZlU2VsZWN0aW9uKGRhdGEsIGMxLCBjMiwgRkFMU0UpOwogICAgICAgICAgICB9CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIGNhc2UgVktfRE9XTjoKICAgICAgICAgICAgYzEgPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDE7CiAgICAgICAgICAgIGMyID0gUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyOwogICAgICAgICAgICBjMi5ZKys7CiAgICAgICAgICAgIGlmIChjMi5YIDwgZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgYzEsIGMyLCBGQUxTRSk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgfQogICAgICAgIGJyZWFrOwogICAgfQp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfR2VuZXJhdGVLZXlJbnB1dFJlY29yZAogKgogKiBnZW5lcmF0ZXMgaW5wdXRfcmVjb3JkIGZyb20gd2luZG93cyBXTV9LRVlVUC9XTV9LRVlET1dOIG1lc3NhZ2VzCiAqLwpzdGF0aWMgdm9pZCAgICBXQ1VTRVJfR2VuZXJhdGVLZXlJbnB1dFJlY29yZChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgQk9PTCBkb3duLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSwgQk9PTCBzeXMpCnsKICAgIElOUFVUX1JFQ09SRAlpcjsKICAgIERXT1JECQluOwogICAgV0NIQVIJCWJ1ZlsyXTsKICAgIHN0YXRpYwlXQ0hBUglsYXN0OyAvKiBrZWVwIGxhc3QgY2hhciBzZWVuIGFzIGZlZWQgZm9yIGtleSB1cCBtZXNzYWdlICovCiAgICBCWVRFCQlrZXlTdGF0ZVsyNTZdOwoKICAgIGlyLkV2ZW50VHlwZSA9IEtFWV9FVkVOVDsKICAgIGlyLkV2ZW50LktleUV2ZW50LmJLZXlEb3duID0gZG93bjsKICAgIGlyLkV2ZW50LktleUV2ZW50LndSZXBlYXRDb3VudCA9IExPV09SRChsUGFyYW0pOwogICAgaXIuRXZlbnQuS2V5RXZlbnQud1ZpcnR1YWxLZXlDb2RlID0gd1BhcmFtOwogICAgCiAgICBpci5FdmVudC5LZXlFdmVudC53VmlydHVhbFNjYW5Db2RlID0gSElXT1JEKGxQYXJhbSkgJiAweEZGOwogICAgCiAgICBpci5FdmVudC5LZXlFdmVudC51Q2hhci5Vbmljb2RlQ2hhciA9IDA7CiAgICBpci5FdmVudC5LZXlFdmVudC5kd0NvbnRyb2xLZXlTdGF0ZSA9IFdDVVNFUl9HZXRDdHJsS2V5U3RhdGUoa2V5U3RhdGUpOwogICAgaWYgKGxQYXJhbSAmICgxTCA8PCAyNCkpCQlpci5FdmVudC5LZXlFdmVudC5kd0NvbnRyb2xLZXlTdGF0ZSB8PSBFTkhBTkNFRF9LRVk7CiAgICBpZiAoc3lzKQkJCQlpci5FdmVudC5LZXlFdmVudC5kd0NvbnRyb2xLZXlTdGF0ZSB8PSBMRUZUX0FMVF9QUkVTU0VEOyAvKiBGSVhNRTogZ290dGEgY2hvb3NlIG9uZSAqLwoKICAgIGlmICghKGlyLkV2ZW50LktleUV2ZW50LmR3Q29udHJvbEtleVN0YXRlICYgRU5IQU5DRURfS0VZKSkKICAgIHsKCWlmIChkb3duKQoJewoJICAgIHN3aXRjaCAoVG9Vbmljb2RlKHdQYXJhbSwgSElXT1JEKGxQYXJhbSksIGtleVN0YXRlLCBidWYsIDIsIDApKQoJICAgIHsKCSAgICBjYXNlIDI6CgkJLyogRklYTUUuLi4gc2hvdWxkIGdlbmVyYXRlIHR3byBldmVudHMuLi4gKi8KCQkvKiBmYWxsIHRocnUgKi8KCSAgICBjYXNlIDE6CQoJCWxhc3QgPSBidWZbMF07CgkJYnJlYWs7CgkgICAgZGVmYXVsdDoKCQlsYXN0ID0gMDsKCQlicmVhazsKCSAgICB9Cgl9Cglpci5FdmVudC5LZXlFdmVudC51Q2hhci5Vbmljb2RlQ2hhciA9IGxhc3Q7IC8qIEZJWE1FIEhBQ0tZLi4uIGFuZCBidWdneSAnY296IGl0IHNob3VsZCBiZSBhIHN0YWNrLCBub3QgYSBzaW5nbGUgdmFsdWUgKi8KCWlmICghZG93bikgbGFzdCA9IDA7CiAgICB9CgogICAgV3JpdGVDb25zb2xlSW5wdXQoZGF0YS0+aENvbkluLCAmaXIsIDEsICZuKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJV0NVU0VSX0dlbmVyYXRlTW91c2VJbnB1dFJlY29yZAogKgogKgogKi8Kc3RhdGljIHZvaWQgICAgV0NVU0VSX0dlbmVyYXRlTW91c2VJbnB1dFJlY29yZChzdHJ1Y3QgaW5uZXJfZGF0YSogZGF0YSwgQ09PUkQgYywgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV1BBUkFNIHdQYXJhbSwgRFdPUkQgZXZlbnQpCnsKICAgIElOUFVUX1JFQ09SRAlpcjsKICAgIEJZVEUJCWtleVN0YXRlWzI1Nl07CiAgICBEV09SRCAgICAgICAgICAgICAgIG1vZGUsIG47CgogICAgLyogTU9VU0VfRVZFTlRzIHNob3VsZG4ndCBiZSBzZW50IHVubGVzcyBFTkFCTEVfTU9VU0VfSU5QVVQgaXMgYWN0aXZlICovCiAgICBpZiAoIUdldENvbnNvbGVNb2RlKGRhdGEtPmhDb25JbiwgJm1vZGUpIHx8ICEobW9kZSAmIEVOQUJMRV9NT1VTRV9JTlBVVCkpCiAgICAgICAgcmV0dXJuOwoKICAgIGlyLkV2ZW50VHlwZSA9IE1PVVNFX0VWRU5UOwogICAgaXIuRXZlbnQuTW91c2VFdmVudC5kd01vdXNlUG9zaXRpb24gPSBjOwogICAgaXIuRXZlbnQuTW91c2VFdmVudC5kd0J1dHRvblN0YXRlID0gMDsKICAgIGlmICh3UGFyYW0gJiBNS19MQlVUVE9OKSBpci5FdmVudC5Nb3VzZUV2ZW50LmR3QnV0dG9uU3RhdGUgfD0gRlJPTV9MRUZUXzFTVF9CVVRUT05fUFJFU1NFRDsKICAgIGlmICh3UGFyYW0gJiBNS19NQlVUVE9OKSBpci5FdmVudC5Nb3VzZUV2ZW50LmR3QnV0dG9uU3RhdGUgfD0gRlJPTV9MRUZUXzJORF9CVVRUT05fUFJFU1NFRDsKICAgIGlmICh3UGFyYW0gJiBNS19SQlVUVE9OKSBpci5FdmVudC5Nb3VzZUV2ZW50LmR3QnV0dG9uU3RhdGUgfD0gUklHSFRNT1NUX0JVVFRPTl9QUkVTU0VEOwogICAgaXIuRXZlbnQuTW91c2VFdmVudC5kd0NvbnRyb2xLZXlTdGF0ZSA9IFdDVVNFUl9HZXRDdHJsS2V5U3RhdGUoa2V5U3RhdGUpOwogICAgaXIuRXZlbnQuTW91c2VFdmVudC5kd0V2ZW50RmxhZ3MgPSBldmVudDsKCiAgICBXcml0ZUNvbnNvbGVJbnB1dChkYXRhLT5oQ29uSW4sICZpciwgMSwgJm4pOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfUHJvYwogKgogKgogKi8Kc3RhdGljIExSRVNVTFQgQ0FMTEJBQ0sgV0NVU0VSX1Byb2MoSFdORCBoV25kLCBVSU5UIHVNc2csIFdQQVJBTSB3UGFyYW0sIExQQVJBTSBsUGFyYW0pCnsKICAgIHN0cnVjdCBpbm5lcl9kYXRhKglkYXRhID0gKHN0cnVjdCBpbm5lcl9kYXRhKilHZXRXaW5kb3dMb25nKGhXbmQsIDApOwoKICAgIHN3aXRjaCAodU1zZykKICAgIHsKICAgIGNhc2UgV01fQ1JFQVRFOgogICAgICAgIHJldHVybiBXQ1VTRVJfQ3JlYXRlKGhXbmQsIChMUENSRUFURVNUUlVDVClsUGFyYW0pOwogICAgY2FzZSBXTV9ERVNUUk9ZOgoJUFJJVkFURShkYXRhKS0+aFduZCA9IDA7CglQb3N0UXVpdE1lc3NhZ2UoMCk7CglicmVhazsKICAgIGNhc2UgV01fUEFJTlQ6CglXQ1VTRVJfUGFpbnQoZGF0YSk7CglicmVhazsKICAgIGNhc2UgV01fS0VZRE9XTjoKICAgIGNhc2UgV01fS0VZVVA6CiAgICAgICAgaWYgKFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24pCiAgICAgICAgICAgIFdDVVNFUl9IYW5kbGVTZWxlY3Rpb25LZXkoZGF0YSwgdU1zZyA9PSBXTV9LRVlET1dOLCB3UGFyYW0sIGxQYXJhbSk7CiAgICAgICAgZWxzZQogICAgICAgICAgICBXQ1VTRVJfR2VuZXJhdGVLZXlJbnB1dFJlY29yZChkYXRhLCB1TXNnID09IFdNX0tFWURPV04sIHdQYXJhbSwgbFBhcmFtLCBGQUxTRSk7CglicmVhazsKICAgIGNhc2UgV01fU1lTS0VZRE9XTjoKICAgIGNhc2UgV01fU1lTS0VZVVA6CglXQ1VTRVJfR2VuZXJhdGVLZXlJbnB1dFJlY29yZChkYXRhLCB1TXNnID09IFdNX1NZU0tFWURPV04sIHdQYXJhbSwgbFBhcmFtLCBUUlVFKTsKCWJyZWFrOwogICAgY2FzZSBXTV9MQlVUVE9ORE9XTjoKICAgICAgICBpZiAoZGF0YS0+Y3VyY2ZnLnF1aWNrX2VkaXQpCiAgICAgICAgewogICAgICAgICAgICBpZiAoUFJJVkFURShkYXRhKS0+aGFzX3NlbGVjdGlvbikKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+aGFzX3NlbGVjdGlvbiA9IEZBTFNFOwogICAgICAgICAgICAgICAgV0NVU0VSX1NldFNlbGVjdGlvbihkYXRhLCAwKTsKICAgICAgICAgICAgfQogICAgICAgICAgICBlbHNlCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MSA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MiA9IFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSk7CiAgICAgICAgICAgICAgICBTZXRDYXB0dXJlKFBSSVZBVEUoZGF0YSktPmhXbmQpOwogICAgICAgICAgICAgICAgV0NVU0VSX1NldFNlbGVjdGlvbihkYXRhLCAwKTsKICAgICAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBUUlVFOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGVsc2UgCiAgICAgICAgewogICAgICAgICAgICBXQ1VTRVJfR2VuZXJhdGVNb3VzZUlucHV0UmVjb3JkKGRhdGEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgMCk7CiAgICAgICAgfQoJYnJlYWs7CiAgICBjYXNlIFdNX01PVVNFTU9WRToKICAgICAgICBpZiAoZGF0YS0+Y3VyY2ZnLnF1aWNrX2VkaXQpCiAgICAgICAgewogICAgICAgICAgICBpZiAoR2V0Q2FwdHVyZSgpID09IFBSSVZBVEUoZGF0YSktPmhXbmQgJiYgUFJJVkFURShkYXRhKS0+aGFzX3NlbGVjdGlvbiAmJgogICAgICAgICAgICAgICAgKHdQYXJhbSAmIE1LX0xCVVRUT04pKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBXQ1VTRVJfTW92ZVNlbGVjdGlvbihkYXRhLCBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIEZBTFNFKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICBXQ1VTRVJfR2VuZXJhdGVNb3VzZUlucHV0UmVjb3JkKGRhdGEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgTU9VU0VfTU9WRUQpOwogICAgICAgIH0KCWJyZWFrOwogICAgY2FzZSBXTV9MQlVUVE9OVVA6CiAgICAgICAgaWYgKGRhdGEtPmN1cmNmZy5xdWlja19lZGl0KQogICAgICAgIHsKICAgICAgICAgICAgaWYgKEdldENhcHR1cmUoKSA9PSBQUklWQVRFKGRhdGEpLT5oV25kICYmIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gJiYgCiAgICAgICAgICAgICAgICAod1BhcmFtJiBNS19MQlVUVE9OKSkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NVU0VSX01vdmVTZWxlY3Rpb24oZGF0YSwgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQxLCBXQ1VTRVJfR2V0Q2VsbChkYXRhLCBsUGFyYW0pLCBUUlVFKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICBXQ1VTRVJfR2VuZXJhdGVNb3VzZUlucHV0UmVjb3JkKGRhdGEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgMCk7CiAgICAgICAgfQoJYnJlYWs7CiAgICBjYXNlIFdNX1JCVVRUT05ET1dOOgogICAgICAgIGlmICgod1BhcmFtICYgKE1LX0NPTlRST0x8TUtfU0hJRlQpKSA9PSBkYXRhLT5jdXJjZmcubWVudV9tYXNrKQogICAgICAgIHsKICAgICAgICAgICAgUkVDVCAgICAgICAgcjsKCiAgICAgICAgICAgIEdldFdpbmRvd1JlY3QoaFduZCwgJnIpOwogICAgICAgICAgICBXQ1VTRVJfU2V0TWVudURldGFpbHMoZGF0YSwgUFJJVkFURShkYXRhKS0+aFBvcE1lbnUpOwogICAgICAgICAgICBUcmFja1BvcHVwTWVudShQUklWQVRFKGRhdGEpLT5oUG9wTWVudSwgVFBNX0xFRlRBTElHTnxUUE1fVE9QQUxJR04sIAogICAgICAgICAgICAgICAgICAgICAgICAgICByLmxlZnQgKyBMT1dPUkQobFBhcmFtKSwgci50b3AgKyBISVdPUkQobFBhcmFtKSwgMCwgaFduZCwgTlVMTCk7CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIFdDVVNFUl9HZW5lcmF0ZU1vdXNlSW5wdXRSZWNvcmQoZGF0YSwgV0NVU0VSX0dldENlbGwoZGF0YSwgbFBhcmFtKSwgd1BhcmFtLCAwKTsKICAgICAgICB9CglicmVhazsgICAgCiAgICBjYXNlIFdNX1JCVVRUT05VUDoKICAgICAgICAvKiBubyBuZWVkIHRvIHRyYWNrIGZvciByYnV0dG9uIHVwIHdoZW4gb3BlbmluZyB0aGUgcG9wdXAuLi4gdGhlIGV2ZW50IHdpbGwgYmUKICAgICAgICAgKiBzd2FsbG93ZWQgYnkgVHJhY2tQb3B1cE1lbnUgKi8KICAgICAgICBXQ1VTRVJfR2VuZXJhdGVNb3VzZUlucHV0UmVjb3JkKGRhdGEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgMCk7CglicmVhazsgICAgCiAgICBjYXNlIFdNX01PVVNFV0hFRUw6CiAgICAgICAgLyogRklYTUU6IHNob3VsZCB3ZSBzY3JvbGwgdG9vID8gKi8KICAgICAgICBXQ1VTRVJfR2VuZXJhdGVNb3VzZUlucHV0UmVjb3JkKGRhdGEsIFdDVVNFUl9HZXRDZWxsKGRhdGEsIGxQYXJhbSksIHdQYXJhbSwgTU9VU0VfV0hFRUxFRCk7CglicmVhazsgICAgCiAgICBjYXNlIFdNX1NFVEZPQ1VTOgoJaWYgKGRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSkKCXsKCSAgICBDcmVhdGVDYXJldChQUklWQVRFKGRhdGEpLT5oV25kLCBQUklWQVRFKGRhdGEpLT5jdXJzb3JfYml0bWFwLCAKICAgICAgICAgICAgICAgICAgICAgICAgZGF0YS0+Y3VyY2ZnLmNlbGxfd2lkdGgsIGRhdGEtPmN1cmNmZy5jZWxsX2hlaWdodCk7IAoJICAgIFdDVVNFUl9Qb3NDdXJzb3IoZGF0YSk7Cgl9CiAgICAgICAgYnJlYWs7IAogICAgY2FzZSBXTV9LSUxMRk9DVVM6IAoJaWYgKGRhdGEtPmN1cmNmZy5jdXJzb3JfdmlzaWJsZSkKCSAgICBEZXN0cm95Q2FyZXQoKTsgCglicmVhazsKICAgIGNhc2UgV01fSFNDUk9MTDogCiAgICAgICAgewogICAgICAgICAgICBpbnQJcG9zID0gZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWDsKCiAgICAgICAgICAgIHN3aXRjaCAoTE9XT1JEKHdQYXJhbSkpIAogICAgICAgICAgICB7IAogICAgICAgICAgICBjYXNlIFNCX1BBR0VVUDogCXBvcyAtPSA4OyAJCWJyZWFrOyAKICAgICAgICAgICAgY2FzZSBTQl9QQUdFRE9XTjogCXBvcyArPSA4OyAJCWJyZWFrOyAKICAgICAgICAgICAgY2FzZSBTQl9MSU5FVVA6IAlwb3MtLTsJCQlicmVhazsKICAgICAgICAgICAgY2FzZSBTQl9MSU5FRE9XTjogCXBvcysrOwkgCQlicmVhazsKICAgICAgICAgICAgY2FzZSBTQl9USFVNQlRSQUNLOiBwb3MgPSBISVdPUkQod1BhcmFtKTsJYnJlYWs7CiAgICAgICAgICAgIGRlZmF1bHQ6IAkJCQkJYnJlYWs7CiAgICAgICAgICAgIH0gCiAgICAgICAgICAgIGlmIChwb3MgPCAwKSBwb3MgPSAwOwogICAgICAgICAgICBpZiAocG9zID4gZGF0YS0+Y3VyY2ZnLnNiX3dpZHRoIC0gZGF0YS0+Y3VyY2ZnLndpbl93aWR0aCkgCiAgICAgICAgICAgICAgICBwb3MgPSBkYXRhLT5jdXJjZmcuc2Jfd2lkdGggLSBkYXRhLT5jdXJjZmcud2luX3dpZHRoOwogICAgICAgICAgICBpZiAocG9zICE9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlgpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFNjcm9sbFdpbmRvdyhoV25kLCAoZGF0YS0+Y3VyY2ZnLndpbl9wb3MuWCAtIHBvcykgKiBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aCwgMCwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTlVMTCwgTlVMTCk7CiAgICAgICAgICAgICAgICBkYXRhLT5jdXJjZmcud2luX3Bvcy5YID0gcG9zOwogICAgICAgICAgICAgICAgU2V0U2Nyb2xsUG9zKGhXbmQsIFNCX0hPUlosIHBvcywgVFJVRSk7IAogICAgICAgICAgICAgICAgVXBkYXRlV2luZG93KGhXbmQpOyAKICAgICAgICAgICAgICAgIFdDVVNFUl9Qb3NDdXJzb3IoZGF0YSk7CiAgICAgICAgICAgICAgICBXSU5FQ09OX05vdGlmeVdpbmRvd0NoYW5nZShkYXRhKTsKICAgICAgICAgICAgfQogICAgICAgIH0KCWJyZWFrOwogICAgY2FzZSBXTV9WU0NST0xMOiAKICAgICAgICB7CgkgICAgaW50CXBvcyA9IGRhdGEtPmN1cmNmZy53aW5fcG9zLlk7CgoJICAgIHN3aXRjaCAoTE9XT1JEKHdQYXJhbSkpIAoJICAgIHsgCiAgICAgICAgICAgIGNhc2UgU0JfUEFHRVVQOiAJcG9zIC09IDg7IAkJYnJlYWs7IAogICAgICAgICAgICBjYXNlIFNCX1BBR0VET1dOOiAJcG9zICs9IDg7IAkJYnJlYWs7IAogICAgICAgICAgICBjYXNlIFNCX0xJTkVVUDogCXBvcy0tOwkJCWJyZWFrOwoJICAgIGNhc2UgU0JfTElORURPV046IAlwb3MrKzsJIAkJYnJlYWs7CiAgICAgICAgICAgIGNhc2UgU0JfVEhVTUJUUkFDSzogcG9zID0gSElXT1JEKHdQYXJhbSk7CWJyZWFrOwogICAgICAgICAgICBkZWZhdWx0OiAJCQkJCWJyZWFrOwoJICAgIH0gCgkgICAgaWYgKHBvcyA8IDApIHBvcyA9IDA7CgkgICAgaWYgKHBvcyA+IGRhdGEtPmN1cmNmZy5zYl9oZWlnaHQgLSBkYXRhLT5jdXJjZmcud2luX2hlaWdodCkgCiAgICAgICAgICAgICAgICBwb3MgPSBkYXRhLT5jdXJjZmcuc2JfaGVpZ2h0IC0gZGF0YS0+Y3VyY2ZnLndpbl9oZWlnaHQ7CgkgICAgaWYgKHBvcyAhPSBkYXRhLT5jdXJjZmcud2luX3Bvcy5ZKQoJICAgIHsKCQlTY3JvbGxXaW5kb3coaFduZCwgMCwgKGRhdGEtPmN1cmNmZy53aW5fcG9zLlkgLSBwb3MpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0LCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOVUxMLCBOVUxMKTsKCQlkYXRhLT5jdXJjZmcud2luX3Bvcy5ZID0gcG9zOwoJCVNldFNjcm9sbFBvcyhoV25kLCBTQl9WRVJULCBwb3MsIFRSVUUpOyAKCQlVcGRhdGVXaW5kb3coaFduZCk7IAoJCVdDVVNFUl9Qb3NDdXJzb3IoZGF0YSk7CgkJV0lORUNPTl9Ob3RpZnlXaW5kb3dDaGFuZ2UoZGF0YSk7CgkgICAgfQoKICAgICAgICB9IAogICAgY2FzZSBXTV9TWVNDT01NQU5EOgoJc3dpdGNoICh3UGFyYW0pCgl7CgljYXNlIElEU19ERUZBVUxUOgoJICAgIFdDVVNFUl9HZXRQcm9wZXJ0aWVzKGRhdGEsIEZBTFNFKTsKCSAgICBicmVhazsKCWNhc2UgSURTX1BST1BFUlRZOgoJICAgIFdDVVNFUl9HZXRQcm9wZXJ0aWVzKGRhdGEsIFRSVUUpOwoJICAgIGJyZWFrOwoJZGVmYXVsdDogCgkgICAgcmV0dXJuIERlZldpbmRvd1Byb2MoaFduZCwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwoJfQoJYnJlYWs7CiAgICBjYXNlIFdNX0NPTU1BTkQ6Cglzd2l0Y2ggKHdQYXJhbSkKCXsKCWNhc2UgSURTX0RFRkFVTFQ6CgkgICAgV0NVU0VSX0dldFByb3BlcnRpZXMoZGF0YSwgRkFMU0UpOwoJICAgIGJyZWFrOwoJY2FzZSBJRFNfUFJPUEVSVFk6CgkgICAgV0NVU0VSX0dldFByb3BlcnRpZXMoZGF0YSwgVFJVRSk7CgkgICAgYnJlYWs7CgljYXNlIElEU19NQVJLOgogICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWCA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5ZID0gMDsKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlggPSBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDIuWSA9IDA7CiAgICAgICAgICAgIFdDVVNFUl9TZXRTZWxlY3Rpb24oZGF0YSwgMCk7CiAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBUUlVFOwoJICAgIGJyZWFrOwoJY2FzZSBJRFNfQ09QWToKICAgICAgICAgICAgaWYgKFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24pCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFBSSVZBVEUoZGF0YSktPmhhc19zZWxlY3Rpb24gPSBGQUxTRTsKICAgICAgICAgICAgICAgIFdDVVNFUl9TZXRTZWxlY3Rpb24oZGF0YSwgMCk7CiAgICAgICAgICAgICAgICBXQ1VTRVJfQ29weVNlbGVjdGlvblRvQ2xpcGJvYXJkKGRhdGEpOwogICAgICAgICAgICB9CgkgICAgYnJlYWs7CgljYXNlIElEU19QQVNURToKCSAgICBXQ1VTRVJfUGFzdGVGcm9tQ2xpcGJvYXJkKGRhdGEpOwoJICAgIGJyZWFrOwoJY2FzZSBJRFNfU0VMRUNUQUxMOgogICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5zZWxlY3RQdDEuWCA9IFBSSVZBVEUoZGF0YSktPnNlbGVjdFB0MS5ZID0gMDsKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlggPSAoZGF0YS0+Y3VyY2ZnLnNiX3dpZHRoIC0gMSkgKiBkYXRhLT5jdXJjZmcuY2VsbF93aWR0aDsKICAgICAgICAgICAgUFJJVkFURShkYXRhKS0+c2VsZWN0UHQyLlkgPSAoZGF0YS0+Y3VyY2ZnLnNiX2hlaWdodCAtIDEpICogZGF0YS0+Y3VyY2ZnLmNlbGxfaGVpZ2h0OwogICAgICAgICAgICBXQ1VTRVJfU2V0U2VsZWN0aW9uKGRhdGEsIDApOwogICAgICAgICAgICBQUklWQVRFKGRhdGEpLT5oYXNfc2VsZWN0aW9uID0gVFJVRTsKCSAgICBicmVhazsKCWNhc2UgSURTX1NDUk9MTDoKCWNhc2UgSURTX1NFQVJDSDoKCSAgICBXSU5FX0ZJWE1FKCJVbmhhbmRsZWQgeWV0IGNvbW1hbmQ6ICV4XG4iLCB3UGFyYW0pOwoJICAgIGJyZWFrOwoJZGVmYXVsdDogCgkgICAgcmV0dXJuIERlZldpbmRvd1Byb2MoaFduZCwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwoJfQoJYnJlYWs7CiAgICBjYXNlIFdNX0lOSVRNRU5VUE9QVVA6CglpZiAoIUhJV09SRChsUGFyYW0pKQlyZXR1cm4gRGVmV2luZG93UHJvYyhoV25kLCB1TXNnLCB3UGFyYW0sIGxQYXJhbSk7CglXQ1VTRVJfU2V0TWVudURldGFpbHMoZGF0YSwgR2V0U3lzdGVtTWVudShQUklWQVRFKGRhdGEpLT5oV25kLCBGQUxTRSkpOwoJYnJlYWs7CiAgICBkZWZhdWx0OgoJcmV0dXJuIERlZldpbmRvd1Byb2MoaFduZCwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwogICAgfQogICAgcmV0dXJuIDA7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVdDVVNFUl9EZWxldGVCYWNrZW5kCiAqCiAqCiAqLwp2b2lkIFdDVVNFUl9EZWxldGVCYWNrZW5kKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBpZiAoIVBSSVZBVEUoZGF0YSkpIHJldHVybjsKICAgIGlmIChQUklWQVRFKGRhdGEpLT5oV25kKQkJRGVzdHJveVdpbmRvdyhQUklWQVRFKGRhdGEpLT5oV25kKTsKICAgIGlmIChQUklWQVRFKGRhdGEpLT5oRm9udCkJCURlbGV0ZU9iamVjdChQUklWQVRFKGRhdGEpLT5oRm9udCk7CiAgICBpZiAoUFJJVkFURShkYXRhKS0+Y3Vyc29yX2JpdG1hcCkJRGVsZXRlT2JqZWN0KFBSSVZBVEUoZGF0YSktPmN1cnNvcl9iaXRtYXApOwogICAgaWYgKFBSSVZBVEUoZGF0YSktPmhNZW1EQykJCURlbGV0ZURDKFBSSVZBVEUoZGF0YSktPmhNZW1EQyk7CiAgICBpZiAoUFJJVkFURShkYXRhKS0+aEJpdG1hcCkJCURlbGV0ZU9iamVjdChQUklWQVRFKGRhdGEpLT5oQml0bWFwKTsKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIFBSSVZBVEUoZGF0YSkpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfTWFpbkxvb3AKICoKICoKICovCnN0YXRpYyBpbnQgV0NVU0VSX01haW5Mb29wKHN0cnVjdCBpbm5lcl9kYXRhKiBkYXRhKQp7CiAgICBNU0cJCW1zZzsKCiAgICBmb3IgKDs7KSAKICAgIHsKCXN3aXRjaCAoTXNnV2FpdEZvck11bHRpcGxlT2JqZWN0cygxLCAmZGF0YS0+aFN5bmNocm8sIEZBTFNFLCBJTkZJTklURSwgUVNfQUxMSU5QVVQpKQoJewoJY2FzZSBXQUlUX09CSkVDVF8wOgoJICAgIGlmICghV0lORUNPTl9HcmFiQ2hhbmdlcyhkYXRhKSkKCQlQb3N0UXVpdE1lc3NhZ2UoMCk7CgkgICAgYnJlYWs7CgljYXNlIFdBSVRfT0JKRUNUXzArMToKCSAgICBzd2l0Y2ggKEdldE1lc3NhZ2UoJm1zZywgMCwgMCwgMCkpCgkgICAgewoJICAgIGNhc2UgLTE6IC8qIHRoZSBldmVudCBoYW5kbGUgYmVjYW1lIGludmFsaWQsIHNvIGV4aXQgKi8KCQlyZXR1cm4gLTE7CgkgICAgY2FzZSAwOiAvKiBXTV9RVUlUIGhhcyBiZWVuIHBvc3RlZCAqLwoJCXJldHVybiAwOwoJICAgIGRlZmF1bHQ6CgkJRGlzcGF0Y2hNZXNzYWdlKCZtc2cpOwoJCWJyZWFrOwoJICAgIH0KCSAgICBicmVhazsKCWRlZmF1bHQ6CgkgICAgV0lORV9FUlIoImdvdCBwYlxuIik7CgkgICAgLyogZXJyICovCgkgICAgYnJlYWs7Cgl9CiAgICB9CQp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlXQ1VTRVJfSW5pdEJhY2tlbmQKICoKICogSW5pdGlhbGlzYXRpb24gcGFydCBJSTogY3JlYXRpb24gb2Ygd2luZG93LgogKgogKi8KQk9PTCBXQ1VTRVJfSW5pdEJhY2tlbmQoc3RydWN0IGlubmVyX2RhdGEqIGRhdGEpCnsKICAgIHN0YXRpYyBXQ0hBUiB3Q2xhc3NOYW1lW10gPSB7J1cnLCdpJywnbicsJ2UnLCdDJywnbycsJ24nLCdzJywnbycsJ2wnLCdlJywnQycsJ2wnLCdhJywncycsJ3MnLDB9OwoKICAgIFdORENMQVNTCQl3bmRjbGFzczsKCiAgICBkYXRhLT5wcml2YXRlID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIEhFQVBfWkVST19NRU1PUlksIHNpemVvZihzdHJ1Y3QgaW5uZXJfZGF0YV91c2VyKSk7CiAgICBpZiAoIWRhdGEtPnByaXZhdGUpIHJldHVybiBGQUxTRTsKCiAgICBkYXRhLT5mbk1haW5Mb29wID0gV0NVU0VSX01haW5Mb29wOwogICAgZGF0YS0+Zm5Qb3NDdXJzb3IgPSBXQ1VTRVJfUG9zQ3Vyc29yOwogICAgZGF0YS0+Zm5TaGFwZUN1cnNvciA9IFdDVVNFUl9TaGFwZUN1cnNvcjsKICAgIGRhdGEtPmZuQ29tcHV0ZVBvc2l0aW9ucyA9IFdDVVNFUl9Db21wdXRlUG9zaXRpb25zOwogICAgZGF0YS0+Zm5SZWZyZXNoID0gV0NVU0VSX1JlZnJlc2g7CiAgICBkYXRhLT5mblJlc2l6ZVNjcmVlbkJ1ZmZlciA9IFdDVVNFUl9SZXNpemVTY3JlZW5CdWZmZXI7CiAgICBkYXRhLT5mblNldFRpdGxlID0gV0NVU0VSX1NldFRpdGxlOwogICAgZGF0YS0+Zm5TY3JvbGwgPSBXQ1VTRVJfU2Nyb2xsOwogICAgZGF0YS0+Zm5EZWxldGVCYWNrZW5kID0gV0NVU0VSX0RlbGV0ZUJhY2tlbmQ7CgogICAgd25kY2xhc3Muc3R5bGUgICAgICAgICA9IDA7CiAgICB3bmRjbGFzcy5scGZuV25kUHJvYyAgID0gV0NVU0VSX1Byb2M7CiAgICB3bmRjbGFzcy5jYkNsc0V4dHJhICAgID0gMDsKICAgIHduZGNsYXNzLmNiV25kRXh0cmEgICAgPSBzaXplb2YoRFdPUkQpOwogICAgd25kY2xhc3MuaEluc3RhbmNlICAgICA9IEdldE1vZHVsZUhhbmRsZShOVUxMKTsKICAgIHduZGNsYXNzLmhJY29uICAgICAgICAgPSBMb2FkSWNvbigwLCBJRElfV0lOTE9HTyk7CiAgICB3bmRjbGFzcy5oQ3Vyc29yICAgICAgID0gTG9hZEN1cnNvcigwLCBJRENfQVJST1cpOwogICAgd25kY2xhc3MuaGJyQmFja2dyb3VuZCA9IEdldFN0b2NrT2JqZWN0KEJMQUNLX0JSVVNIKTsKICAgIHduZGNsYXNzLmxwc3pNZW51TmFtZSAgPSBOVUxMOwogICAgd25kY2xhc3MubHBzekNsYXNzTmFtZSA9IHdDbGFzc05hbWU7CiAgCiAgICBSZWdpc3RlckNsYXNzKCZ3bmRjbGFzcyk7CgogICAgQ3JlYXRlV2luZG93KHduZGNsYXNzLmxwc3pDbGFzc05hbWUsIE5VTEwsCgkJIFdTX09WRVJMQVBQRUR8V1NfQ0FQVElPTnxXU19TWVNNRU5VfFdTX1RISUNLRlJBTUV8V1NfTUlOSU1JWkVCT1h8V1NfSFNDUk9MTHxXU19WU0NST0xMLCAKCQkgQ1dfVVNFREVGQVVMVCwgQ1dfVVNFREVGQVVMVCwgMCwgMCwgMCwgMCwgd25kY2xhc3MuaEluc3RhbmNlLCBkYXRhKTsgICAKICAgIGlmICghUFJJVkFURShkYXRhKS0+aFduZCkgcmV0dXJuIEZBTFNFOwoKICAgIC8qIGZvcmNlIHVwZGF0ZSBvZiBjdXJyZW50IGRhdGEgKi8KICAgIGlmICghV0lORUNPTl9HcmFiQ2hhbmdlcyhkYXRhKSkgcmV0dXJuIEZBTFNFOwoKICAgIGlmICghV0NVU0VSX0luaXRGb250KGRhdGEpKSByZXR1cm4gRkFMU0U7CgogICAgcmV0dXJuIFRSVUU7Cn0KCg==