LyoKICogQ29weXJpZ2h0IDE5OTctMjAwMCBNYXJjdXMgTWVpc3NuZXIKICogQ29weXJpZ2h0IDE5OTgtMjAwMCBMaW9uZWwgVWxtZXIKICogQ29weXJpZ2h0IDIwMDAtMjAwMSBUcmFuc0dhbWluZyBUZWNobm9sb2dpZXMgSW5jLgogKiBDb3B5cmlnaHQgMjAwNiBTdGVmYW4gRPZzaW5nZXIKICoKICogVGhpcyBsaWJyYXJ5IGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vcgogKiBtb2RpZnkgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCiAqIExpY2Vuc2UgYXMgcHVibGlzaGVkIGJ5IHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb247IGVpdGhlcgogKiB2ZXJzaW9uIDIuMSBvZiB0aGUgTGljZW5zZSwgb3IgKGF0IHlvdXIgb3B0aW9uKSBhbnkgbGF0ZXIgdmVyc2lvbi4KICoKICogVGhpcyBsaWJyYXJ5IGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCiAqIGJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mCiAqIE1FUkNIQU5UQUJJTElUWSBvciBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRS4gIFNlZSB0aGUgR05VCiAqIExlc3NlciBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGZvciBtb3JlIGRldGFpbHMuCiAqCiAqIFlvdSBzaG91bGQgaGF2ZSByZWNlaXZlZCBhIGNvcHkgb2YgdGhlIEdOVSBMZXNzZXIgR2VuZXJhbCBQdWJsaWMKICogTGljZW5zZSBhbG9uZyB3aXRoIHRoaXMgbGlicmFyeTsgaWYgbm90LCB3cml0ZSB0byB0aGUgRnJlZSBTb2Z0d2FyZQogKiBGb3VuZGF0aW9uLCBJbmMuLCA1MSBGcmFua2xpbiBTdCwgRmlmdGggRmxvb3IsIEJvc3RvbiwgTUEgMDIxMTAtMTMwMSwgVVNBCiAqLwoKI2luY2x1ZGUgImNvbmZpZy5oIgojaW5jbHVkZSAid2luZS9wb3J0LmgiCgojaW5jbHVkZSA8YXNzZXJ0Lmg+CiNpbmNsdWRlIDxzdGRhcmcuaD4KI2luY2x1ZGUgPHN0cmluZy5oPgojaW5jbHVkZSA8c3RkbGliLmg+CgojZGVmaW5lIENPQkpNQUNST1MKI2RlZmluZSBOT05BTUVMRVNTVU5JT04KCiNpbmNsdWRlICJ3aW5kZWYuaCIKI2luY2x1ZGUgIndpbmJhc2UuaCIKI2luY2x1ZGUgIndpbm5scy5oIgojaW5jbHVkZSAid2luZXJyb3IuaCIKI2luY2x1ZGUgIndpbmdkaS5oIgojaW5jbHVkZSAid2luZS9leGNlcHRpb24uaCIKI2luY2x1ZGUgImV4Y3B0LmgiCgojaW5jbHVkZSAiZGRyYXcuaCIKI2luY2x1ZGUgImQzZC5oIgoKI2luY2x1ZGUgImRkcmF3X3ByaXZhdGUuaCIKI2luY2x1ZGUgIndpbmUvZGVidWcuaCIKCldJTkVfREVGQVVMVF9ERUJVR19DSEFOTkVMKGRkcmF3KTsKCnN0YXRpYyBCT09MIElEaXJlY3REcmF3SW1wbF9ERFNEX01hdGNoKGNvbnN0IEREU1VSRkFDRURFU0MyKiByZXF1ZXN0ZWQsIGNvbnN0IEREU1VSRkFDRURFU0MyKiBwcm92aWRlZCk7CnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJRGlyZWN0RHJhd0ltcGxfQXR0YWNoRDNERGV2aWNlKElEaXJlY3REcmF3SW1wbCAqVGhpcywgSURpcmVjdERyYXdTdXJmYWNlSW1wbCAqcHJpbWFyeSk7CnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJRGlyZWN0RHJhd0ltcGxfQ3JlYXRlTmV3U3VyZmFjZShJRGlyZWN0RHJhd0ltcGwgKlRoaXMsIEREU1VSRkFDRURFU0MyICpwRERTRCwgSURpcmVjdERyYXdTdXJmYWNlSW1wbCAqKnBwU3VyZiwgVUlOVCBsZXZlbCk7CgovKiBEZXZpY2UgaWRlbnRpZmllci4gRG9uJ3QgcmVsYXkgaXQgdG8gV2luZUQzRCAqLwpzdGF0aWMgY29uc3QgRERERVZJQ0VJREVOVElGSUVSMiBkZXZpY2VpZGVudGlmaWVyID0KewogICAgImRpc3BsYXkiLAogICAgIkRpcmVjdERyYXcgSEFMIiwKICAgIHsgeyAweDAwMDEwMDAxLCAweDAwMDEwMDAxIH0gfSwKICAgIDAsIDAsIDAsIDAsCiAgICAvKiBhODM3M2MxMC03YWM0LTRkZWItODQ5YS0wMDk4NDRkMDhiMmQgKi8KICAgIHsweGE4MzczYzEwLDB4N2FjNCwweDRkZWIsIHsweDg0LDB4OWEsMHgwMCwweDk4LDB4NDQsMHhkMCwweDhiLDB4MmR9fSwKICAgIDAKfTsKCi8qIFRoaXMgaXMgZm9yIGNsZWFudXAgaWYgYSBicm9rZW4gYXBwIGRvZXNuJ3QgUmVsZWFzZSBpdHMgb2JqZWN0cyAqLwpJRGlyZWN0RHJhd0ltcGwgKmRkcmF3X2xpc3Q7CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSVVua25vd24gTWV0aG9kcwogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpRdWVyeUludGVyZmFjZQogKgogKiBRdWVyaWVzIGRpZmZlcmVudCBpbnRlcmZhY2VzIG9mIHRoZSBEaXJlY3REcmF3IG9iamVjdC4gSXQgY2FuIHJldHVybgogKiBJRGlyZWN0RHJhdyBpbnRlcmZhY2VzIGluIHZlcnNpb24gMSwgMiwgNCBhbmQgNywgYW5kIElEaXJlY3QzRCBpbnRlcmZhY2VzCiAqIGluIHZlcnNpb24gMSwgMiwgMyBhbmQgNy4gQW4gSURpcmVjdDNERGV2aWNlIGNhbiBiZSBjcmVhdGVkIHdpdGggdGhpcwogKiBtZXRob2QuCiAqIFRoZSByZXR1cm5lZCBpbnRlcmZhY2UgaXMgQWRkUmVmKCktZWQgYmVmb3JlIGl0J3MgcmV0dXJuZWQKICoKICogUnVsZXMgZm9yIFF1ZXJ5SW50ZXJmYWNlOgogKiAgaHR0cDovL21zZG4ubWljcm9zb2Z0LmNvbS9saWJyYXJ5L2RlZmF1bHQuYXNwPyBcCiAqICAgIHVybD0vbGlicmFyeS9lbi11cy9jb20vaHRtbC82ZGIxN2VkOC0wNmU0LTRiYWUtYmMyNi0xMTMxNzZjYzdlMGUuYXNwCiAqCiAqIFVzZWQgZm9yIHZlcnNpb24gMSwgMiwgNCBhbmQgNwogKgogKiBQYXJhbXM6CiAqICByZWZpaWQ6IEludGVyZmFjZSBJRCBhc2tlZCBmb3IKICogIG9iajogVXNlZCB0byByZXR1cm4gdGhlIGludGVyZmFjZSBwb2ludGVyCiAqCiAqIFJldHVybnM6CiAqICBTX09LIGlmIGFuIGludGVyZmFjZSB3YXMgZm91bmQKICogIEVfTk9JTlRFUkZBQ0UgaWYgdGhlIHJlcXVlc3RlZCBpbnRlcmZhY2Ugd2Fzbid0IGZvdW5kCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9RdWVyeUludGVyZmFjZShJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUkVGSUlEIHJlZmlpZCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKipvYmopCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CgogICAgVFJBQ0UoIiglcCktPiglcywlcClcbiIsIFRoaXMsIGRlYnVnc3RyX2d1aWQocmVmaWlkKSwgb2JqKTsKCiAgICAvKiBBY2NvcmRpbmcgdG8gQ09NIGRvY3MsIGlmIHRoZSBRdWVyeUludGVyZmFjZSBmYWlscywgb2JqIHNob3VsZCBiZSBzZXQgdG8gTlVMTCAqLwogICAgKm9iaiA9IE5VTEw7CgogICAgaWYoIXJlZmlpZCkKICAgICAgICByZXR1cm4gRERFUlJfSU5WQUxJRFBBUkFNUzsKCiAgICAvKiBDaGVjayBEaXJlY3REcmF3IEludGVyZmFjZXMgKi8KICAgIGlmICggSXNFcXVhbEdVSUQoICZJSURfSVVua25vd24sIHJlZmlpZCApIHx8CiAgICAgICAgIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3REcmF3NywgcmVmaWlkICkgKQogICAgewogICAgICAgICpvYmogPSBJQ09NX0lOVEVSRkFDRShUaGlzLCBJRGlyZWN0RHJhdzcpOwogICAgICAgIFRSQUNFKCIoJXApIFJldHVybmluZyBJRGlyZWN0RHJhdzcgaW50ZXJmYWNlIGF0ICVwXG4iLCBUaGlzLCAqb2JqKTsKICAgIH0KICAgIGVsc2UgaWYgKCBJc0VxdWFsR1VJRCggJklJRF9JRGlyZWN0RHJhdzQsIHJlZmlpZCApICkKICAgIHsKICAgICAgICAqb2JqID0gSUNPTV9JTlRFUkZBQ0UoVGhpcywgSURpcmVjdERyYXc0KTsKICAgICAgICBUUkFDRSgiKCVwKSBSZXR1cm5pbmcgSURpcmVjdERyYXc0IGludGVyZmFjZSBhdCAlcFxuIiwgVGhpcywgKm9iaik7CiAgICB9CiAgICBlbHNlIGlmICggSXNFcXVhbEdVSUQoICZJSURfSURpcmVjdERyYXcyLCByZWZpaWQgKSApCiAgICB7CiAgICAgICAgKm9iaiA9IElDT01fSU5URVJGQUNFKFRoaXMsIElEaXJlY3REcmF3Mik7CiAgICAgICAgVFJBQ0UoIiglcCkgUmV0dXJuaW5nIElEaXJlY3REcmF3MiBpbnRlcmZhY2UgYXQgJXBcbiIsIFRoaXMsICpvYmopOwogICAgfQogICAgZWxzZSBpZiAoIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3REcmF3LCByZWZpaWQgKSApCiAgICB7CiAgICAgICAgKm9iaiA9IElDT01fSU5URVJGQUNFKFRoaXMsIElEaXJlY3REcmF3KTsKICAgICAgICBUUkFDRSgiKCVwKSBSZXR1cm5pbmcgSURpcmVjdERyYXcgaW50ZXJmYWNlIGF0ICVwXG4iLCBUaGlzLCAqb2JqKTsKICAgIH0KCiAgICAvKiBEaXJlY3QzRAogICAgICogVGhlIHJlZmNvdW50IHVuaXQgdGVzdCByZXZlYWxlZCB0aGF0IGFuIElEaXJlY3QzRDcgaW50ZXJmYWNlIGNhbiBvbmx5IGJlIHF1ZXJpZWQKICAgICAqIGZyb20gYSBEaXJlY3REcmF3IG9iamVjdCB0aGF0IHdhcyBjcmVhdGVkIGFzIGFuIElEaXJlY3REcmF3NyBpbnRlcmZhY2UuIE5vIGlkZWEKICAgICAqIHdobyBoYWQgdGhpcyBpZGVhIGFuZCB3aHkuIFRoZSBvbGRlciBpbnRlcmZhY2VzIGNhbiBxdWVyeSBhbmQgSURpcmVjdDNEIHZlcnNpb24KICAgICAqIGJlY2F1c2UgdGhleSBhcmUgYWxsIGNyZWF0ZWQgYXMgSURpcmVjdERyYXcoMSkuIFRoaXMgaXNuJ3QgcmVhbGx5IGNydWNpYWwgYmVoYXZpb3IsCiAgICAgKiBhbmQgbWVzc3kgdG8gaW1wbGVtZW50IHdpdGggdGhlIGNvbW1vbiBjcmVhdGlvbiBmdW5jdGlvbiwgc28gaXQgaGFzIGJlZW4gbGVmdCBvdXQgaGVyZS4KICAgICAqLwogICAgZWxzZSBpZiAoIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3QzRCAgLCByZWZpaWQgKSB8fAogICAgICAgICAgICAgIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3QzRDIgLCByZWZpaWQgKSB8fAogICAgICAgICAgICAgIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3QzRDMgLCByZWZpaWQgKSB8fAogICAgICAgICAgICAgIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3QzRDcgLCByZWZpaWQgKSApCiAgICB7CiAgICAgICAgLyogQ2hlY2sgdGhlIHN1cmZhY2UgaW1wbGVtZW50YXRpb24gKi8KICAgICAgICBpZihUaGlzLT5JbXBsVHlwZSA9PSBTVVJGQUNFX1VOS05PV04pCiAgICAgICAgewogICAgICAgICAgICAvKiBBcHBzIG1heSBjcmVhdGUgdGhlIElEaXJlY3QzRCBJbnRlcmZhY2UgYmVmb3JlIHRoZSBwcmltYXJ5IHN1cmZhY2UuCiAgICAgICAgICAgICAqIHNldCB0aGUgc3VyZmFjZSBpbXBsZW1lbnRhdGlvbiAqLwogICAgICAgICAgICBUaGlzLT5JbXBsVHlwZSA9IFNVUkZBQ0VfT1BFTkdMOwogICAgICAgICAgICBUUkFDRSgiKCVwKSBDaG9vc2luZyBPcGVuR0wgc3VyZmFjZXMgYmVjYXVzZSBhIERpcmVjdDNEIGludGVyZmFjZSB3YXMgcmVxdWVzdGVkXG4iLCBUaGlzKTsKICAgICAgICB9CiAgICAgICAgZWxzZSBpZihUaGlzLT5JbXBsVHlwZSAhPSBTVVJGQUNFX09QRU5HTCAmJiBEZWZhdWx0U3VyZmFjZVR5cGUgPT0gU1VSRkFDRV9VTktOT1dOKQogICAgICAgIHsKICAgICAgICAgICAgRVJSKCIoJXApIFRoZSBBcHAgaXMgcmVxdWVzdGluZyBhIEQzRCBkZXZpY2UsIGJ1dCBhIG5vbi1PcGVuR0wgc3VyZmFjZSB0eXBlIHdhcyBjaG9vc2VuLiBQcmVwYXJlIGZvciB0cm91YmxlIVxuIiwgVGhpcyk7CiAgICAgICAgICAgIEVSUigiICglcCkgWW91IG1heSB3YW50IHRvIGNvbnRhY3Qgd2luZS1kZXZlbCBmb3IgaGVscFxuIiwgVGhpcyk7CiAgICAgICAgICAgIC8qIFNob3VsZCBJIGFzc2VydCgwKSBoZXJlPz8/ICovCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoVGhpcy0+SW1wbFR5cGUgIT0gU1VSRkFDRV9PUEVOR0wpCiAgICAgICAgewogICAgICAgICAgICBXQVJOKCJUaGUgYXBwIHJlcXVlc3RzIGEgRGlyZWN0M0QgaW50ZXJmYWNlLCBidXQgbm9uLW9wZW5nbCBzdXJmYWNlcyB3aGVyZSBzZXQgaW4gd2luZWNmZ1xuIik7CiAgICAgICAgICAgIC8qIERvIG5vdCBhYm9ydCBoZXJlLCBvbmx5IHJlamVjdCAzRCBEZXZpY2UgY3JlYXRpb24gKi8KICAgICAgICB9CgogICAgICAgIGlmICggSXNFcXVhbEdVSUQoICZJSURfSURpcmVjdDNEICAsIHJlZmlpZCApICkKICAgICAgICB7CiAgICAgICAgICAgIFRoaXMtPmQzZHZlcnNpb24gPSAxOwogICAgICAgICAgICAqb2JqID0gSUNPTV9JTlRFUkZBQ0UoVGhpcywgSURpcmVjdDNEKTsKICAgICAgICAgICAgVFJBQ0UoIiByZXR1cm5pbmcgRGlyZWN0M0QgaW50ZXJmYWNlIGF0ICVwLlxuIiwgKm9iaik7CiAgICAgICAgfQogICAgICAgIGVsc2UgaWYgKCBJc0VxdWFsR1VJRCggJklJRF9JRGlyZWN0M0QyICAsIHJlZmlpZCApICkKICAgICAgICB7CiAgICAgICAgICAgIFRoaXMtPmQzZHZlcnNpb24gPSAyOwogICAgICAgICAgICAqb2JqID0gSUNPTV9JTlRFUkZBQ0UoVGhpcywgSURpcmVjdDNEMik7CiAgICAgICAgICAgIFRSQUNFKCIgcmV0dXJuaW5nIERpcmVjdDNEMiBpbnRlcmZhY2UgYXQgJXAuXG4iLCAqb2JqKTsKICAgICAgICB9CiAgICAgICAgZWxzZSBpZiAoIElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3QzRDMgICwgcmVmaWlkICkgKQogICAgICAgIHsKICAgICAgICAgICAgVGhpcy0+ZDNkdmVyc2lvbiA9IDM7CiAgICAgICAgICAgICpvYmogPSBJQ09NX0lOVEVSRkFDRShUaGlzLCBJRGlyZWN0M0QzKTsKICAgICAgICAgICAgVFJBQ0UoIiByZXR1cm5pbmcgRGlyZWN0M0QzIGludGVyZmFjZSBhdCAlcC5cbiIsICpvYmopOwogICAgICAgIH0KICAgICAgICBlbHNlIGlmKElzRXF1YWxHVUlEKCAmSUlEX0lEaXJlY3QzRDcgICwgcmVmaWlkICkpCiAgICAgICAgewogICAgICAgICAgICBUaGlzLT5kM2R2ZXJzaW9uID0gNzsKICAgICAgICAgICAgKm9iaiA9IElDT01fSU5URVJGQUNFKFRoaXMsIElEaXJlY3QzRDcpOwogICAgICAgICAgICBUUkFDRSgiIHJldHVybmluZyBEaXJlY3QzRDcgaW50ZXJmYWNlIGF0ICVwLlxuIiwgKm9iaik7CiAgICAgICAgfQogICAgfQoKICAgIC8qIFVua25vd24gaW50ZXJmYWNlICovCiAgICBlbHNlCiAgICB7CiAgICAgICAgRVJSKCIoJXApLT4oJXMsICVwKTogTm8gaW50ZXJmYWNlIGZvdW5kXG4iLCBUaGlzLCBkZWJ1Z3N0cl9ndWlkKHJlZmlpZCksIG9iaik7CiAgICAgICAgcmV0dXJuIEVfTk9JTlRFUkZBQ0U7CiAgICB9CgogICAgSVVua25vd25fQWRkUmVmKCAoSVVua25vd24gKikgKm9iaiApOwogICAgcmV0dXJuIFNfT0s7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OkFkZFJlZgogKgogKiBJbmNyZWFzZXMgdGhlIGludGVyZmFjZXMgcmVmY291bnQsIGJhc2ljYWxseQogKgogKiBERHJhdyByZWZjb3VudGluZyBpcyBhIGJpdCB0cmlja3kuIFRoZSBkaWZmZXJlbnQgRGlyZWN0RHJhdyBpbnRlcmZhY2UKICogdmVyc2lvbnMgaGF2ZSBpbmRpdmlkdWFsIHJlZmNvdW50cywgYnV0IHRoZSBJRGlyZWN0M0QgaW50ZXJmYWNlcyBkbyBub3QuCiAqIEFsbCBpbnRlcmZhY2VzIGFyZSBmcm9tIG9uZSBvYmplY3QsIHRoYXQgbWVhbnMgY2FsbGluZyBRdWVyeUludGVyZmFjZSBvbiBhbgogKiBJRGlyZWN0RHJhdzcgaW50ZXJmYWNlIGZvciBhbiBJRGlyZWN0RHJhdzQgaW50ZXJmYWNlIGRvZXMgbm90IGNyZWF0ZSBhIG5ldwogKiBJRGlyZWN0RHJhd0ltcGwgb2JqZWN0LgogKgogKiBUaGF0IG1lYW5zIGFsbCBBZGRSZWYgYW5kIFJlbGVhc2UgaW1wbGVtZW50YXRpb25zIG9mIElEaXJlY3REcmF3WCB3b3JrCiAqIHdpdGggdGhlaXIgb3duIGNvdW50ZXIsIGFuZCBJRGlyZWN0M0RYOjpBZGRSZWYgdGh1bmsgdG8gSURpcmVjdERyYXcgKDEpLAogKiBleGNlcHQgb2YgSURpcmVjdDNENyB3aGljaCB0aHVua3MgdG8gSURpcmVjdERyYXc3CiAqCiAqIFJldHVybnM6IFRoZSBuZXcgcmVmY291bnQKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgVUxPTkcgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9BZGRSZWYoSURpcmVjdERyYXc3ICppZmFjZSkKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIFVMT05HIHJlZiA9IEludGVybG9ja2VkSW5jcmVtZW50KCZUaGlzLT5yZWY3KTsKCiAgICBUUkFDRSgiKCVwKSA6IGluY3JlbWVudGluZyBJRGlyZWN0RHJhdzcgcmVmY291bnQgZnJvbSAlbHUuXG4iLCBUaGlzLCByZWYgLTEpOwoKICAgIGlmKHJlZiA9PSAxKSBJbnRlcmxvY2tlZEluY3JlbWVudCgmVGhpcy0+bnVtSWZhY2VzKTsKCiAgICByZXR1cm4gcmVmOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXdJbXBsX0Rlc3Ryb3kKICoKICogRGVzdHJveXMgYSBkZHJhdyBvYmplY3QgaWYgYWxsIHJlZmNvdW50cyBhcmUgMC4gVGhpcyBpcyB0byBzaGFyZSBjb2RlCiAqIGJldHdlZW4gdGhlIElEaXJlY3REcmF3WDo6UmVsZWFzZSBmdW5jdGlvbnMKICoKICogUGFyYW1zOgogKiAgVGhpczogRGlyZWN0RHJhdyBvYmplY3QgdG8gZGVzdHJveQogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnZvaWQKSURpcmVjdERyYXdJbXBsX0Rlc3Ryb3koSURpcmVjdERyYXdJbXBsICpUaGlzKQp7CiAgICBJRGlyZWN0RHJhd0ltcGwgKnByZXY7CgogICAgLyogQ2xlYXIgdGhlIGNvb3BsZXZlbCB0byByZXN0b3JlIHdpbmRvdyBhbmQgZGlzcGxheSBtb2RlICovCiAgICBJRGlyZWN0RHJhdzdfU2V0Q29vcGVyYXRpdmVMZXZlbChJQ09NX0lOVEVSRkFDRShUaGlzLCBJRGlyZWN0RHJhdzcpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTlVMTCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEREU0NMX05PUk1BTCk7CgogICAgLyogRGVzdHJveSB0aGUgZGV2aWNlIHdpbmRvdyBpZiB3ZSBjcmVhdGVkIG9uZSAqLwogICAgaWYoVGhpcy0+ZGV2aWNld2luZG93ICE9IDApCiAgICB7CiAgICAgICAgVFJBQ0UoIiAoJXApIERlc3Ryb3lpbmcgdGhlIGRldmljZSB3aW5kb3cgJXBcbiIsIFRoaXMsIFRoaXMtPmRldmljZXdpbmRvdyk7CiAgICAgICAgRGVzdHJveVdpbmRvdyhUaGlzLT5kZXZpY2V3aW5kb3cpOwogICAgICAgIFRoaXMtPmRldmljZXdpbmRvdyA9IDA7CiAgICB9CgogICAgLyogVW5yZWdpc3RlciB0aGUgd2luZG93IGNsYXNzICovCiAgICBVbnJlZ2lzdGVyQ2xhc3NBKFRoaXMtPmNsYXNzbmFtZSwgMCk7CgogICAgLyogVW5jaGFpbiBpdCBmcm9tIHRoZSBkZHJhdyBsaXN0ICovCiAgICBpZihkZHJhd19saXN0ID09IFRoaXMpCiAgICB7CiAgICAgICAgZGRyYXdfbGlzdCA9IFRoaXMtPm5leHQ7CiAgICAgICAgLyogTm8gbmVlZCB0byBzZWFyY2ggZm9yIGEgcHJlZGVjZXNzb3IgaGVyZSAqLwogICAgfQogICAgZWxzZQogICAgewogICAgICAgIGZvcihwcmV2ID0gZGRyYXdfbGlzdDsgcHJldjsgcHJldiA9IHByZXYtPm5leHQpCiAgICAgICAgICAgIGlmKHByZXYtPm5leHQgPT0gVGhpcykgYnJlYWs7CgogICAgICAgIGlmKHByZXYpCiAgICAgICAgICAgIHByZXYtPm5leHQgPSBUaGlzLT5uZXh0OwogICAgICAgIGVsc2UKICAgICAgICAgICAgRVJSKCJEaWRuJ3QgZmluZCB0aGUgcHJldmlvdXMgZGRyYXcgZWxlbWVudCBpbiB0aGUgbGlzdFxuIik7CiAgICB9CgogICAgLyogUmVsZWFzZSB0aGUgYXR0YWNoZWQgV2luZUQzRCBzdHVmZiAqLwogICAgSVdpbmVEM0REZXZpY2VfUmVsZWFzZShUaGlzLT53aW5lRDNERGV2aWNlKTsKICAgIElXaW5lRDNEX1JlbGVhc2UoVGhpcy0+d2luZUQzRCk7CgogICAgLyogTm93IGZyZWUgdGhlIG9iamVjdCAqLwogICAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgVGhpcyk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OlJlbGVhc2UKICoKICogRGVjcmVhc2VzIHRoZSByZWZjb3VudC4gSWYgdGhlIHJlZmNvdW50IGZhbGxzIHRvIDAsIHRoZSBvYmplY3QgaXMgZGVzdHJveWVkCiAqCiAqIFJldHVybnM6IFRoZSBuZXcgcmVmY291bnQKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgVUxPTkcgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9SZWxlYXNlKElEaXJlY3REcmF3NyAqaWZhY2UpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBVTE9ORyByZWYgPSBJbnRlcmxvY2tlZERlY3JlbWVudCgmVGhpcy0+cmVmNyk7CgogICAgVFJBQ0UoIiglcCktPigpIGRlY3JlbWVudGluZyBJRGlyZWN0RHJhdzcgcmVmY291bnQgZnJvbSAlbHUuXG4iLCBUaGlzLCByZWYgKzEpOwoKICAgIGlmKHJlZiA9PSAwKQogICAgewogICAgICAgIFVMT05HIGlmYWNlY291bnQgPSBJbnRlcmxvY2tlZERlY3JlbWVudCgmVGhpcy0+bnVtSWZhY2VzKTsKICAgICAgICBpZihpZmFjZWNvdW50ID09IDApIElEaXJlY3REcmF3SW1wbF9EZXN0cm95KFRoaXMpOwogICAgfQoKICAgIHJldHVybiByZWY7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdyBtZXRob2RzCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhd0ltcGxfU2V0dXBFeGNsdXNpdmVXaW5kb3cKICoKICogSGVscGVyIGZ1bmN0aW9uIHRoYXQgbW9kaWZpZXMgYSBIV05EJ3MgU3R5bGUgYW5kIEV4U3R5bGUgZm9yIHByb3BlcgogKiBmdWxsc2NyZWVuIHVzZS4KICoKICogUGFyYW1zOgogKiAgVGhpczogUG9pbnRlciB0byB0aGUgRGlyZWN0RHJhdyBpbXBsZW1lbnRhdGlvbgogKiAgSFdORDogV2luZG93IHRvIHNldHVwCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIHZvaWQKSURpcmVjdERyYXdJbXBsX1NldHVwRnVsbHNjcmVlbldpbmRvdyhJRGlyZWN0RHJhd0ltcGwgKlRoaXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSFdORCB3aW5kb3cpCnsKICAgIExPTkcgc3R5bGUsIGV4U3R5bGU7CiAgICAvKiBEb24ndCBkbyBhbnl0aGluZyBpZiBhbiBvcmlnaW5hbCBzdHlsZSBpcyBzdG9yZWQuCiAgICAgKiBUaGF0IHNob3VsZG4ndCBoYXBwZW4KICAgICAqLwogICAgVFJBQ0UoIiglcCk6IFNldHRpbmcgdXAgd2luZG93ICVwIGZvciBleGNsdXNpdmUgbW9kZVxuIiwgVGhpcywgd2luZG93KTsKICAgIGlmKCAoVGhpcy0+c3R5bGUgIT0gMCkgJiYgKFRoaXMtPmV4U3R5bGUgIT0gMCkgKQogICAgewogICAgICAgIEVSUigiKCVwKSBXYW50IHRvIGNoYW5nZSB0aGUgd2luZG93IHBhcmFtZXRlcnMgb2YgSFdORCAlcCwgYnV0IFwKICAgICAgICAgICAgIGFub3RoZXIgc3R5bGUgaXMgc3RvcmVkIGZvciByZXN0YXVyYXRpb24gYWZ0ZXJ3YXJkc1xuIiwgVGhpcywgd2luZG93KTsKICAgIH0KCiAgICAvKiBHZXQgdGhlIHBhcmFtZXRlcnMgYW5kIHNhdmUgdGhlbSAqLwogICAgc3R5bGUgPSBHZXRXaW5kb3dMb25nVyh3aW5kb3csIEdXTF9TVFlMRSk7CiAgICBleFN0eWxlID0gR2V0V2luZG93TG9uZ1cod2luZG93LCBHV0xfRVhTVFlMRSk7CiAgICBUaGlzLT5zdHlsZSA9IHN0eWxlOwogICAgVGhpcy0+ZXhTdHlsZSA9IGV4U3R5bGU7CgogICAgLyogRmlsdGVyIG91dCB3aW5kb3cgZGVjb3JhdGlvbnMgKi8KICAgIHN0eWxlICY9IH5XU19DQVBUSU9OOwogICAgc3R5bGUgJj0gfldTX1RISUNLRlJBTUU7CiAgICBleFN0eWxlICY9IH5XU19FWF9XSU5ET1dFREdFOwogICAgZXhTdHlsZSAmPSB+V1NfRVhfQ0xJRU5URURHRTsKCiAgICAvKiBNYWtlIHN1cmUgdGhlIHdpbmRvdyBpcyBtYW5hZ2VkLCBvdGhlcndpc2Ugd2Ugd29uJ3QgZ2V0IGtleWJvYXJkIGlucHV0ICovCiAgICBzdHlsZSB8PSBXU19QT1BVUCB8IFdTX1NZU01FTlU7CgogICAgVFJBQ0UoIk9sZCBzdHlsZSB3YXMgJTA4bHgsJTA4bHgsIHNldHRpbmcgdG8gJTA4bHgsJTA4bHhcbiIsCiAgICAgICAgICBUaGlzLT5zdHlsZSwgVGhpcy0+ZXhTdHlsZSwgc3R5bGUsIGV4U3R5bGUpOwoKICAgIFNldFdpbmRvd0xvbmdXKHdpbmRvdywgR1dMX1NUWUxFLCBzdHlsZSk7CiAgICBTZXRXaW5kb3dMb25nVyh3aW5kb3csIEdXTF9FWFNUWUxFLCBleFN0eWxlKTsKCiAgICAvKiBJbmZvcm0gdGhlIHdpbmRvdyBhYm91dCB0aGUgdXBkYXRlLgogICAgICogVE9ETzogU2hvdWxkIEkgbW92ZSBpdCB0byAwLzAgdG9vPwogICAgICovCiAgICBTZXRXaW5kb3dQb3Mod2luZG93LCAwIC8qIEluc2VydEFmdGVyLCBpZ25vcmVkICovLAogICAgICAgICAgICAgICAgIDAsIDAsIDAsIDAsIC8qIFBvcywgU2l6ZSwgaWdub3JlZCAqLwogICAgICAgICAgICAgICAgIFNXUF9GUkFNRUNIQU5HRUQgfCBTV1BfTk9TSVpFIHwgU1dQX05PTU9WRSB8IFNXUF9OT1pPUkRFUik7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhd0ltcGxfUmVzdG9yZVdpbmRvdwogKgogKiBIZWxwZXIgZnVuY3Rpb24gdGhhdCByZXN0b3JlcyBhIHdpbmRvd3MnIHByb3BlcnRpZXMgd2hlbiB0YWtpbmcgaXQgb3V0CiAqIG9mIGZ1bGxzY3JlZW4gbW9kZQogKgogKiBQYXJhbXM6CiAqICBUaGlzOiBQb2ludGVyIHRvIHRoZSBEaXJlY3REcmF3IGltcGxlbWVudGF0aW9uCiAqICBIV05EOiBXaW5kb3cgdG8gc2V0dXAKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgdm9pZApJRGlyZWN0RHJhd0ltcGxfUmVzdG9yZVdpbmRvdyhJRGlyZWN0RHJhd0ltcGwgKlRoaXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEhXTkQgd2luZG93KQp7CiAgICBpZiggKFRoaXMtPnN0eWxlID09IDApICYmIChUaGlzLT5leFN0eWxlID09IDApICkKICAgIHsKICAgICAgICAvKiBUaGlzIGNvdWxkIGJlIGEgRERTQ0xfTk9STUFMIC0+IEREU0NMX05PUk1BTAogICAgICAgICAqIHN3aXRjaCwgZG8gbm90aGluZwogICAgICAgICAqLwogICAgICAgIHJldHVybjsKICAgIH0KICAgIFRSQUNFKCIoJXApOiBSZXN0b3Jpbmcgd2luZG93IHNldHRpbmdzIG9mIHdpbmRvdyAlcCB0byAlMDhseCwgJTA4bHhcbiIsCiAgICAgICAgICBUaGlzLCB3aW5kb3csIFRoaXMtPnN0eWxlLCBUaGlzLT5leFN0eWxlKTsKCiAgICBTZXRXaW5kb3dMb25nVyh3aW5kb3csIEdXTF9TVFlMRSwgVGhpcy0+c3R5bGUpOwogICAgU2V0V2luZG93TG9uZ1cod2luZG93LCBHV0xfRVhTVFlMRSwgVGhpcy0+ZXhTdHlsZSk7CgogICAgLyogRGVsZXRlIHRoZSBvbGQgdmFsdWVzICovCiAgICBUaGlzLT5zdHlsZSA9IDA7CiAgICBUaGlzLT5leFN0eWxlID0gMDsKCiAgICAvKiBJbmZvcm0gdGhlIHdpbmRvdyBhYm91dCB0aGUgdXBkYXRlICovCiAgICBTZXRXaW5kb3dQb3Mod2luZG93LCAwIC8qIEluc2VydEFmdGVyLCBpZ25vcmVkICovLAogICAgICAgICAgICAgICAgIDAsIDAsIDAsIDAsIC8qIFBvcywgU2l6ZSwgaWdub3JlZCAqLwogICAgICAgICAgICAgICAgIFNXUF9GUkFNRUNIQU5HRUQgfCBTV1BfTk9NT1ZFIHwgU1dQX05PU0laRSB8IFNXUF9OT1pPUkRFUik7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OlNldENvb3BlcmF0aXZlTGV2ZWwKICoKICogU2V0cyB0aGUgY29vcGVyYXRpdmUgbGV2ZWwgZm9yIHRoZSBEaXJlY3REcmF3IG9iamVjdCwgYW5kIHRoZSB3aW5kb3cKICogYXNzaWduZWQgdG8gaXQuIFRoZSBjb29wZXJhdGl2ZSBsZXZlbCBkZXRlcm1pbmVzIHRoZSBnZW5lcmFsIGJlaGF2aW9yCiAqIG9mIHRoZSBEaXJlY3REcmF3IGFwcGxpY2F0aW9uCiAqCiAqIFdhcm5pbmc6IFRoaXMgaXMgcXVpdGUgdHJpY2t5LCBhcyBpdCdzIG5vdCByZWFsbHkgZG9jdW1lbnRlZCB3aGljaAogKiBjb29wZXJhdGl2ZSBsZXZlbHMgY2FuIGJlIGNvbWJpbmVkIHdpdGggZWFjaCBvdGhlci4gSWYgYSBnYW1lIGZhaWxzCiAqIGFmdGVyIHRoaXMgZnVuY3Rpb24sIHRyeSB0byBjaGVjayB0aGUgY29vcGVyYXRpdmUgbGV2ZWxzIHBhc3NlZCBvbgogKiBXaW5kb3dzLCBhbmQgaWYgaXQgcmV0dXJucyBzb21ldGhpbmcgZGlmZmVyZW50LgogKgogKiBJZiB5b3UgdGhpbmsgdGhhdCB0aGlzIGZ1bmN0aW9uIGNhdXNlZCB0aGUgZmFpbHVyZSBiZWNhdXNlIGl0IHdyaXRlcyBhCiAqIGZpeG1lLCBiZSBzdXJlIHRvIHJ1biBhZ2FpbiB3aXRoIGEgK2RkcmF3IHRyYWNlLgogKgogKiBXaGF0IGlzIGtub3duIGFib3V0IGNvb3BlcmF0aXZlIGxldmVscyAoU2VlIHRoZSBkZHJhdyBtb2RlcyB0ZXN0KToKICogRERTQ0xfRVhDTFVTSVZFIGFuZCBERFNDTF9GVUxMU0NSRUVOIG11c3QgYmUgdXNlZCB3aXRoIGVhY2ggb3RoZXIKICogRERTQ0xfTk9STUFMIGlzIG5vdCBjb21wYXRpYmxlIHdpdGggRERTQ0xfRVhDTFVTSVZFIG9yIEREU0NMX0ZVTExTQ1JFRU4KICogRERTQ0xfU0VURk9DVVNXSU5ET1cgY2FuIGJlIHBhc3NlZCBvbmx5IGluIEREU0NMX05PUk1BTCBtb2RlLCBidXQgYWZ0ZXIgdGhhdAogKiBERFNDTF9GVUxMU0NSRUVOIGNhbiBiZSBhY3RpdmF0ZWQKICogRERTQ0xfU0VURk9DVVNXSU5ET1cgbWF5IG9ubHkgYmUgdXNlZCB3aXRoIEREU0NMX05PV0lORE9XQ0hBTkdFUwogKgogKiBIYW5kbGVkIGZsYWdzOiBERFNDTF9OT1JNQUwsIEREU0NMX0ZVTExTQ1JFRU4sIEREU0NMX0VYQ0xVU0lWRSwKICogICAgICAgICAgICAgICAgRERTQ0xfU0VURk9DVVNXSU5ET1cgKHBhcnRpYWxseSkKICoKICogVW5oYW5kbGVkIGZsYWdzLCB3aGljaCBzaG91bGQgYmUgaW1wbGVtZW50ZWQKICogIEREU0NMX1NFVERFVklDRVdJTkRPVzogU2V0cyBhIHdpbmRvdyBzcGVjaWFsbHkgdXNlZCBmb3IgcmVuZGVyaW5nIChJIGRvbid0CiAqICBleHBlY3QgYW55IGRpZmZlcmVuY2UgdG8gYSBub3JtYWwgd2luZG93IGZvciB3aW5lKQogKiAgRERTQ0xfQ1JFQVRFREVWSUNFV0lORE9XOiBUZWxscyBkZHJhdyB0byBjcmVhdGUgaXRzIG93biB3aW5kb3cgZm9yCiAqICByZW5kZXJpbmcgKFBvc3NpYmxlIHRlc3QgY2FzZTogSGFsZi1saWZlKQogKgogKiBVbnN1cmUgYWJvdXQgdGhlc2U6IEREU0NMX0ZQVVNFVFVQIEREU0NMX0ZQVVJFU0VSVkUKICoKICogVGhlc2Ugc2VlbSBub3QgcmVhbGx5IGltcG9yYW50IGZvciB3aW5lCiAqICBERFNDTF9BTExPV1JFQk9PVCwgRERTQ0xfTk9XSU5ET1dDSEFOR0VTLCBERFNDTF9BTExPV01PREVYLAogKiAgRERTQ0xfTVVMVElUSFJFREVECiAqCiAqIFJldHVybnM6CiAqICBERF9PSyBpZiB0aGUgY29vcGVyYXRpdmUgbGV2ZWwgd2FzIHNldCBzdWNjZXNzZnVsbHkKICogIERERVJSX0lOVkFMSURQQVJBTVMgaWYgdGhlIHBhc3NlZCBjb29wZXJhdGl2ZSBsZXZlbCBjb21iaW5hdGlvbiBpcyBpbnZhbGlkCiAqICBEREVSUl9IV05EQUxSRUFEWVNFVCBpZiBERFNDTF9TRVRGT0NVU1dJTkRPVyBpcyBwYXNzZWQgaW4gZXhjbHVzaXZlIG1vZGUKICogICAoUHJvYmFibHkgb3RoZXJzIHRvbywgaGF2ZSB0byBpbnZlc3RpZ2F0ZSkKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX1NldENvb3BlcmF0aXZlTGV2ZWwoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSFdORCBod25kLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBjb29wbGV2ZWwpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBIV05EIHdpbmRvdzsKICAgIEhSRVNVTFQgaHI7CgogICAgRklYTUUoIiglcCktPiglcCwlMDhseClcbiIsVGhpcyxod25kLGNvb3BsZXZlbCk7CiAgICBERFJBV19kdW1wX2Nvb3BlcmF0aXZlbGV2ZWwoY29vcGxldmVsKTsKCiAgICAvKiBHZXQgdGhlIG9sZCB3aW5kb3cgKi8KICAgIGhyID0gSVdpbmVEM0REZXZpY2VfR2V0SFdORChUaGlzLT53aW5lRDNERGV2aWNlLCAmd2luZG93KTsKICAgIGlmKGhyICE9IEQzRF9PSykKICAgIHsKICAgICAgICBFUlIoIklXaW5lRDNERGV2aWNlOjpHZXRIV05EIGZhaWxlZCwgaHIgPSAlMDhseFxuIiwgaHIpOwogICAgICAgIHJldHVybiBocjsKICAgIH0KCiAgICAvKiBUZXN0cyBzdWdnZXN0IHRoYXQgd2UgbmVlZCBvbmUgb2YgdGhlbTogKi8KICAgIGlmKCEoY29vcGxldmVsICYgKEREU0NMX1NFVEZPQ1VTV0lORE9XIHwKICAgICAgICAgICAgICAgICAgICAgIEREU0NMX05PUk1BTCAgICAgICAgIHwKICAgICAgICAgICAgICAgICAgICAgIEREU0NMX0VYQ0xVU0lWRSAgICAgICkpKQogICAgewogICAgICAgIFRSQUNFKCJJbmNvcnJlY3QgY29vcGxldmVsIGZsYWdzLCByZXR1cm5pbmcgRERFUlJfSU5WQUxJRFBBUkFNU1xuIik7CiAgICAgICAgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CiAgICB9CgogICAgLyogSGFuZGxlIHRob3NlIGxldmVscyBmaXJzdCB3aGljaCBzZXQgdmFyaW91cyBod25kcyAqLwogICAgaWYoY29vcGxldmVsICYgRERTQ0xfU0VURk9DVVNXSU5ET1cpCiAgICB7CiAgICAgICAgLyogVGhpcyBpc24ndCBjb21wYXRpYmxlIHdpdGggYSBsb3Qgb2YgZmxhZ3MgKi8KICAgICAgICBpZihjb29wbGV2ZWwgJiAoIEREU0NMX01VTFRJVEhSRUFERUQgICB8CiAgICAgICAgICAgICAgICAgICAgICAgICBERFNDTF9GUFVTRVRVUCAgICAgICAgfAogICAgICAgICAgICAgICAgICAgICAgICAgRERTQ0xfRlBVUFJFU0VSVkUgICAgIHwKICAgICAgICAgICAgICAgICAgICAgICAgIEREU0NMX0FMTE9XUkVCT09UICAgICB8CiAgICAgICAgICAgICAgICAgICAgICAgICBERFNDTF9BTExPV01PREVYICAgICAgfAogICAgICAgICAgICAgICAgICAgICAgICAgRERTQ0xfU0VUREVWSUNFV0lORE9XIHwKICAgICAgICAgICAgICAgICAgICAgICAgIEREU0NMX05PUk1BTCAgICAgICAgICB8CiAgICAgICAgICAgICAgICAgICAgICAgICBERFNDTF9FWENMVVNJVkUgICAgICAgfAogICAgICAgICAgICAgICAgICAgICAgICAgRERTQ0xfRlVMTFNDUkVFTiAgICAgICkgKQogICAgICAgIHsKICAgICAgICAgICAgVFJBQ0UoIkNhbGxlZCB3aXRoIGluY29tcGF0aWJsZSBmbGFncywgcmV0dXJuaW5nIERERVJSX0lOVkFMSURQQVJBTVNcbiIpOwogICAgICAgICAgICByZXR1cm4gRERFUlJfSU5WQUxJRFBBUkFNUzsKICAgICAgICB9CiAgICAgICAgZWxzZSBpZiggKFRoaXMtPmNvb3BlcmF0aXZlX2xldmVsICYgRERTQ0xfRlVMTFNDUkVFTikgJiYgd2luZG93KQogICAgICAgIHsKICAgICAgICAgICAgVFJBQ0UoIlNldHRpbmcgRERTQ0xfU0VURk9DVVNXSU5ET1cgd2l0aCBhbiBhbHJlYWR5IHNldCB3aW5kb3csIHJldHVybmluZyBEREVSUl9IV05EQUxSRUFEWVNFVFxuIik7CiAgICAgICAgICAgIHJldHVybiBEREVSUl9IV05EQUxSRUFEWVNFVDsKICAgICAgICB9CgogICAgICAgIFRoaXMtPmZvY3Vzd2luZG93ID0gaHduZDsKICAgICAgICAvKiBXb24ndCB1c2UgdGhlIGh3bmQgcGFyYW0gZm9yIGFueXRoaW5nIGVsc2UgKi8KICAgICAgICBod25kID0gTlVMTDsKCiAgICAgICAgLyogVXNlIHRoZSBmb2N1cyB3aW5kb3cgZm9yIGRyYXdpbmcgdG9vICovCiAgICAgICAgSVdpbmVEM0REZXZpY2VfU2V0SFdORChUaGlzLT53aW5lRDNERGV2aWNlLCBUaGlzLT5mb2N1c3dpbmRvdyk7CgogICAgICAgIC8qIERlc3Ryb3kgdGhlIGRldmljZSB3aW5kb3csIGlmIHdlIGhhdmUgb25lICovCiAgICAgICAgaWYoVGhpcy0+ZGV2aWNld2luZG93KQogICAgICAgIHsKICAgICAgICAgICAgRGVzdHJveVdpbmRvdyhUaGlzLT5kZXZpY2V3aW5kb3cpOwogICAgICAgICAgICBUaGlzLT5kZXZpY2V3aW5kb3cgPSBOVUxMOwogICAgICAgIH0KICAgIH0KICAgIC8qIEREU0NMX05PUk1BTCBvciBERFNDTF9GVUxMU0NSRUVOIHwgRERTQ0xfRVhDTFVTSVZFICovCiAgICBpZihjb29wbGV2ZWwgJiBERFNDTF9OT1JNQUwpCiAgICB7CiAgICAgICAgLyogQ2FuJ3QgY29leGlzdCB3aXRoIGZ1bGxzY3JlZW4gb3IgZXhjbHVzaXZlICovCiAgICAgICAgaWYoY29vcGxldmVsICYgKEREU0NMX0ZVTExTQ1JFRU4gfCBERFNDTF9FWENMVVNJVkUpICkKICAgICAgICB7CiAgICAgICAgICAgIFRSQUNFKCIoJXApIEREU0NMX05PUk1BTCBpcyBub3QgY29tcGF0aXZlIHdpdGggRERTQ0xfRlVMTFNDUkVFTiBvciBERFNDTF9FWENMVVNJVkVcbiIsIFRoaXMpOwogICAgICAgICAgICByZXR1cm4gRERFUlJfSU5WQUxJRFBBUkFNUzsKICAgICAgICB9CgogICAgICAgIC8qIFN3aXRjaGluZyBmcm9tIGZ1bGxzY3JlZW4/ICovCiAgICAgICAgaWYoVGhpcy0+Y29vcGVyYXRpdmVfbGV2ZWwgJiBERFNDTF9GVUxMU0NSRUVOKQogICAgICAgIHsKICAgICAgICAgICAgLyogUmVzdG9yZSB0aGUgZGlzcGxheSBtb2RlICovCiAgICAgICAgICAgIElEaXJlY3REcmF3N19SZXN0b3JlRGlzcGxheU1vZGUoaWZhY2UpOwoKICAgICAgICAgICAgaWYod2luZG93KQogICAgICAgICAgICAgICAgSURpcmVjdERyYXdJbXBsX1Jlc3RvcmVXaW5kb3coVGhpcywgd2luZG93KTsKCiAgICAgICAgICAgIFRoaXMtPmNvb3BlcmF0aXZlX2xldmVsICY9IH5ERFNDTF9GVUxMU0NSRUVOOwogICAgICAgICAgICBUaGlzLT5jb29wZXJhdGl2ZV9sZXZlbCAmPSB+RERTQ0xfRVhDTFVTSVZFOwogICAgICAgICAgICBUaGlzLT5jb29wZXJhdGl2ZV9sZXZlbCAmPSB+RERTQ0xfQUxMT1dNT0RFWDsKICAgICAgICB9CgogICAgICAgIC8qIERvbid0IG92ZXJyaWRlIGZvY3VzIHdpbmRvd3Mgb3IgcHJpdmF0ZSBkZXZpY2Ugd2luZG93cyAqLwogICAgICAgIGlmKCBod25kICYmCiAgICAgICAgICAgICEoVGhpcy0+Zm9jdXN3aW5kb3cpICYmCiAgICAgICAgICAgICEoVGhpcy0+ZGV2aWNld2luZG93KSAmJgogICAgICAgICAgICAoaHduZCAhPSB3aW5kb3cpICkKICAgICAgICB7CiAgICAgICAgICAgIElXaW5lRDNERGV2aWNlX1NldEhXTkQoVGhpcy0+d2luZUQzRERldmljZSwgaHduZCk7CiAgICAgICAgfQogICAgfQogICAgZWxzZSBpZihjb29wbGV2ZWwgJiBERFNDTF9GVUxMU0NSRUVOKQogICAgewogICAgICAgIC8qIE5lZWRzIEREU0NMX0VYQ0xVU0lWRSAqLwogICAgICAgIGlmKCEoY29vcGxldmVsICYgRERTQ0xfRVhDTFVTSVZFKSApCiAgICAgICAgewogICAgICAgICAgICBUUkFDRSgiKCVwKSBERFNDTF9GVUxMU0NSRUVOIG5lZWRzIEREU0NMX0VYQ0xVU0lWRVxuIiwgVGhpcyk7CiAgICAgICAgICAgIHJldHVybiBEREVSUl9JTlZBTElEUEFSQU1TOwogICAgICAgIH0KICAgICAgICAvKiBOZWVkIGEgSFdORAogICAgICAgIGlmKGh3bmQgPT0gMCkKICAgICAgICB7CiAgICAgICAgICAgIFRSQUNFKCIoJXApIEREU0NMX0ZVTExTQ1JFRU4gbmVlZHMgYSBIV05EXG4iLCBUaGlzKTsKICAgICAgICAgICAgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CiAgICAgICAgfQogICAgICAgICovCgogICAgICAgIC8qIFN3aXRjaCBmcm9tIG5vcm1hbCB0byBmdWxsIHNjcmVlbiBtb2RlPyAqLwogICAgICAgIGlmKFRoaXMtPmNvb3BlcmF0aXZlX2xldmVsICYgRERTQ0xfTk9STUFMKQogICAgICAgIHsKICAgICAgICAgICAgVGhpcy0+Y29vcGVyYXRpdmVfbGV2ZWwgJj0gfkREU0NMX05PUk1BTDsKICAgICAgICB9CgogICAgICAgIC8qIERvbid0IG92ZXJyaWRlIGZvY3VzIHdpbmRvd3Mgb3IgcHJpdmF0ZSBkZXZpY2Ugd2luZG93cyAqLwogICAgICAgIGlmKCBod25kICYmCiAgICAgICAgICAgICEoVGhpcy0+Zm9jdXN3aW5kb3cpICYmCiAgICAgICAgICAgICEoVGhpcy0+ZGV2aWNld2luZG93KSAmJgogICAgICAgICAgICAoaHduZCAhPSB3aW5kb3cpICkKICAgICAgICB7CiAgICAgICAgICAgIC8qIE9uIGEgd2luZG93IGNoYW5nZSwgcmVzdG9yZSB0aGUgb2xkIHdpbmRvdyBhbmQgc2V0IHRoZSBuZXcgb25lICovCiAgICAgICAgICAgIGlmKHdpbmRvdyAhPSBod25kKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBpZih3aW5kb3cpCiAgICAgICAgICAgICAgICAgICAgSURpcmVjdERyYXdJbXBsX1Jlc3RvcmVXaW5kb3coVGhpcywgd2luZG93KTsKICAgICAgICAgICAgICAgIElEaXJlY3REcmF3SW1wbF9TZXR1cEZ1bGxzY3JlZW5XaW5kb3coVGhpcywgaHduZCk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgSVdpbmVEM0REZXZpY2VfU2V0SFdORChUaGlzLT53aW5lRDNERGV2aWNlLCBod25kKTsKICAgICAgICB9CiAgICB9CiAgICBlbHNlIGlmKGNvb3BsZXZlbCAmIEREU0NMX0VYQ0xVU0lWRSkKICAgIHsKICAgICAgICBUUkFDRSgiKCVwKSBERFNDTF9FWENMVVNJVkUgbmVlZHMgRERTQ0xfRlVMTFNDUkVFTlxuIiwgVGhpcyk7CiAgICAgICAgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CiAgICB9CgogICAgaWYoY29vcGxldmVsICYgRERTQ0xfQ1JFQVRFREVWSUNFV0lORE9XKQogICAgewogICAgICAgIC8qIERvbid0IGNyZWF0ZSBhIGRldmljZSB3aW5kb3cgaWYgYSBmb2N1cyB3aW5kb3cgaXMgc2V0ICovCiAgICAgICAgaWYoICEoVGhpcy0+Zm9jdXN3aW5kb3cpICkKICAgICAgICB7CiAgICAgICAgICAgIEhXTkQgZGV2aWNld2luZG93ID0gQ3JlYXRlV2luZG93RXhBKDAsIFRoaXMtPmNsYXNzbmFtZSwgIkREcmF3IGRldmljZSB3aW5kb3ciLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBXU19QT1BVUCwgMCwgMCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR2V0U3lzdGVtTWV0cmljcyhTTV9DWFNDUkVFTiksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdldFN5c3RlbU1ldHJpY3MoU01fQ1lTQ1JFRU4pLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOVUxMLCBOVUxMLCBHZXRNb2R1bGVIYW5kbGVBKDApLCBOVUxMKTsKCiAgICAgICAgICAgIFNob3dXaW5kb3coZGV2aWNld2luZG93LCBTV19TSE9XKTsgICAvKiBKdXN0IHRvIGJlIHN1cmUgKi8KICAgICAgICAgICAgVFJBQ0UoIiglcCkgQ3JlYXRlZCBhIEREcmF3IGRldmljZSB3aW5kb3cuIEhXTkQ9JXBcbiIsIFRoaXMsIGRldmljZXdpbmRvdyk7CgogICAgICAgICAgICBJV2luZUQzRERldmljZV9TZXRIV05EKFRoaXMtPndpbmVEM0REZXZpY2UsIGRldmljZXdpbmRvdyk7CiAgICAgICAgICAgIFRoaXMtPmRldmljZXdpbmRvdyA9IGRldmljZXdpbmRvdzsKICAgICAgICB9CiAgICB9CgogICAgLyogVW5oYW5kbGVkIGZsYWdzICovCiAgICBpZihjb29wbGV2ZWwgJiBERFNDTF9BTExPV1JFQk9PVCkKICAgICAgICBXQVJOKCIoJXApIFVuaGFuZGxlZCBmbGFnIEREU0NMX0FMTE9XUkVCT09ULCBoYXJtbGVzc1xuIiwgVGhpcyk7CiAgICBpZihjb29wbGV2ZWwgJiBERFNDTF9BTExPV01PREVYKQogICAgICAgIFdBUk4oIiglcCkgVW5oYW5kbGVkIGZsYWcgRERTQ0xfQUxMT1dNT0RFWCwgaGFybWxlc3NcbiIsIFRoaXMpOwogICAgaWYoY29vcGxldmVsICYgRERTQ0xfTVVMVElUSFJFQURFRCkKICAgICAgICBGSVhNRSgiKCVwKSBVbmhhbmRsZWQgZmxhZyBERFNDTF9NVUxUSVRIUkVBREVELCBVaCBPaC4uLlxuIiwgVGhpcyk7CiAgICBpZihjb29wbGV2ZWwgJiBERFNDTF9GUFVTRVRVUCkKICAgICAgICBXQVJOKCIoJXApIFVuaGFuZGxlZCBmbGFnIEREU0NMX0ZQVVNFVFVQLCBoYXJtbGVzc1xuIiwgVGhpcyk7CiAgICBpZihjb29wbGV2ZWwgJiBERFNDTF9GUFVQUkVTRVJWRSkKICAgICAgICBXQVJOKCIoJXApIFVuaGFuZGxlZCBmbGFnIEREU0NMX0ZQVVBSRVNFUlZFLCBoYXJtbGVzc1xuIiwgVGhpcyk7CgogICAgLyogU3RvcmUgdGhlIGNvb3BlcmF0aXZlX2xldmVsICovCiAgICBUaGlzLT5jb29wZXJhdGl2ZV9sZXZlbCB8PSBjb29wbGV2ZWw7CiAgICBUUkFDRSgiU2V0Q29vcGVyYXRpdmVMZXZlbCByZXR1bmluZyBERF9PS1xuIik7CiAgICByZXR1cm4gRERfT0s7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OlNldERpc3BsYXlNb2RlCiAqCiAqIFNldHMgdGhlIGRpc3BsYXkgc2NyZWVuIHJlc29sdXRpb24sIGNvbG9yIGRlcHRoIGFuZCByZWZyZXNoIGZyZXF1ZW5jeQogKiB3aGVuIGluIGZ1bGxzY3JlZW4gbW9kZSAoaW4gdGhlb3J5KS4KICogUG9zc2libGUgcmV0dXJuIHZhbHVlcyBsaXN0ZWQgaW4gdGhlIFNESyBzdWdnZXN0IHRoYXQgdGhpcyBtZXRob2QgZmFpbHMKICogd2hlbiBub3QgaW4gZnVsbHNjcmVlbiBtb2RlLCBidXQgdGhpcyBpcyB3cm9uZy4gV2luZG93cyAyMDAwIGhhcHBpbHkgc2V0cwogKiB0aGUgZGlzcGxheSBtb2RlIGluIEREU0NMX05PUk1BTCBtb2RlIHdpdGhvdXQgYW4gaHduZCBzcGVjaWZpZWQuCiAqIEl0IHNlZW1zIHRvIGJlIHZhbGlkIHRvIHBhc3MgMCBmb3IgV2l0aCBhbmQgSGVpZ2h0LCB0aGlzIGhhcyB0byBiZSB0ZXN0ZWQKICogSXQgY291bGQgbWVhbiB0aGF0IHRoZSBjdXJyZW50IHZpZGVvIG1vZGUgc2hvdWxkIGJlIGxlZnQgYXMtaXMuIChCdXQgd2h5CiAqIGNhbGwgaXQgdGhlbj8pCiAqCiAqIFBhcmFtczoKICogIEhlaWdodCwgV2lkdGg6IFNjcmVlbiBkaW1lbnNpb24KICogIEJQUDogQ29sb3IgZGVwdGggaW4gQml0cyBwZXIgcGl4ZWwKICogIFJlZnJlc2hyYXRlOiBTY3JlZW4gcmVmcmVzaCByYXRlCiAqICBGbGFnczogT3RoZXIgc3R1ZmYKICoKICogUmV0dXJucwogKiAgRERfT0sgb24gc3VjY2VzcwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfU2V0RGlzcGxheU1vZGUoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIFdpZHRoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgSGVpZ2h0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgQlBQLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgUmVmcmVzaFJhdGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBGbGFncykKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIFdJTkVEM0RESVNQTEFZTU9ERSBNb2RlOwogICAgVFJBQ0UoIiglcCktPiglbGQsJWxkLCVsZCwlbGQsJWx4OiBSZWxheSFcbiIsIFRoaXMsIFdpZHRoLCBIZWlnaHQsIEJQUCwgUmVmcmVzaFJhdGUsIEZsYWdzKTsKCiAgICBpZiggIVdpZHRoIHx8ICFIZWlnaHQgKQogICAgewogICAgICAgIEVSUigiV2lkdGg9JWxkLCBIZWlnaHQ9JWxkLCB3aGF0IHRvIGRvP1xuIiwgV2lkdGgsIEhlaWdodCk7CiAgICAgICAgLyogSXQgbG9va3MgbGlrZSBOZWVkIGZvciBTcGVlZCBQb3JzY2hlIFVubGVhc2hlZCBleHBlY3RzIEREX09LIGhlcmUgKi8KICAgICAgICByZXR1cm4gRERfT0s7CiAgICB9CgogICAgLyogQ2hlY2sgdGhlIGV4Y2x1c2l2ZSBtb2RlCiAgICBpZighKFRoaXMtPmNvb3BlcmF0aXZlX2xldmVsICYgRERTQ0xfRVhDTFVTSVZFKSkKICAgICAgICByZXR1cm4gRERFUlJfTk9FWENMVVNJVkVNT0RFOwogICAgICogVGhpcyBpcyBXUk9ORy4gRG9uJ3Qga25vdyBpZiB0aGUgU0RLIGlzIGNvbXBsZXRlbHkKICAgICAqIHdyb25nIGFuZCBpZiB0aGVyZSBhcmUgYW55IGNvbmRpdGlvbnMgd2hlbiBEREVSUl9OT0VYQ0xVU0lWRQogICAgICogaXMgcmV0dXJuZWQsIGJ1dCBIYWxmLUxpZmUgMS4xLjEuMSAoU3RlYW0gdmVyc2lvbikKICAgICAqIGRlcGVuZHMgb24gdGhpcwogICAgICovCgogICAgTW9kZS5XaWR0aCA9IFdpZHRoOwogICAgTW9kZS5IZWlnaHQgPSBIZWlnaHQ7CiAgICBNb2RlLlJlZnJlc2hSYXRlID0gUmVmcmVzaFJhdGU7CiAgICBzd2l0Y2goQlBQKQogICAgewogICAgICAgIGNhc2UgODogIE1vZGUuRm9ybWF0ID0gV0lORUQzREZNVF9QODsgICAgICAgYnJlYWs7CiAgICAgICAgY2FzZSAxNTogTW9kZS5Gb3JtYXQgPSBXSU5FRDNERk1UX1gxUjVHNUI1OyBicmVhazsKICAgICAgICBjYXNlIDE2OiBNb2RlLkZvcm1hdCA9IFdJTkVEM0RGTVRfUjVHNkI1OyAgIGJyZWFrOwogICAgICAgIGNhc2UgMjQ6IE1vZGUuRm9ybWF0ID0gV0lORUQzREZNVF9SOEc4Qjg7ICAgYnJlYWs7CiAgICAgICAgY2FzZSAzMjogTW9kZS5Gb3JtYXQgPSBXSU5FRDNERk1UX0E4UjhHOEI4OyBicmVhazsKICAgIH0KCiAgICAvKiBUT0RPOiBUaGUgcG9zc2libGUgcmV0dXJuIHZhbHVlcyBmcm9tIG1zZG4gc3VnZ2VzdCB0aGF0CiAgICAgKiB0aGUgc2NyZWVuIG1vZGUgY2FuJ3QgYmUgY2hhbmdlZCBpZiBhIHN1cmZhY2UgaXMgbG9ja2VkCiAgICAgKiBvciBzb21lIGRyYXdpbmcgaXMgaW4gcHJvZ3Jlc3MKICAgICAqLwoKICAgIC8qIFRPRE86IExvc2UgdGhlIHByaW1hcnkgc3VyZmFjZSAqLwogICAgcmV0dXJuIElXaW5lRDNERGV2aWNlX1NldERpc3BsYXlNb2RlKFRoaXMtPndpbmVEM0REZXZpY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwgLyogRmlyc3Qgc3dhcGNoYWluICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJk1vZGUpOwoKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6UmVzdG9yZURpc3BsYXlNb2RlCiAqCiAqIFJlc3RvcmVzIHRoZSBkaXNwbGF5IG1vZGUgdG8gd2hhdCBpdCB3YXMgYXQgY3JlYXRpb24gdGltZS4gQmFzaWNhbGx5LgogKgogKiBBIHByb2JsZW0gYXJpc2VzIHdoZW4gdGhlcmUgYXJlIDIgRGlyZWN0RHJhdyBvYmplY3RzIHVzaW5nIHRoZSBzYW1lIGh3bmQ6CiAqICAtPiBERF8xIGZpbmRzIHRoZSBzY3JlZW4gYXQgMTQwMHgxMDUweDMyIHdoZW4gY3JlYXRlZCwgc2V0cyBpdCB0byA2NDB4NDgweDE2CiAqICAtPiBERF8yIGlzIGNyZWF0ZWQsIGZpbmRzIHRoZSBzY3JlZW4gYXQgNjQweDQ4MHgxNiwgc2V0cyBpdCB0byAxMDI0eDc2OHgzMgogKiAgLT4gRERfMSBpcyByZWxlYXNlZC4gVGhlIHNjcmVlbiBzaG91bGQgYmUgbGVmdCBhdCAxMDI0eDc2OHgzMi4KICogIC0+IEREXzIgaXMgcmVsZWFzZWQuIFRoZSBzY3JlZW4gc2hvdWxkIGJlIHNldCB0byAxNDAweDEwNTB4MzIKICogVGhpcyBjYXNlIGlzIHVuaGFuZGxlZCByaWdodCBub3csIGJ1dCBFbXBpcmUgRWFydGggZG9lcyBpdCB0aGlzIHdheS4KICogKEJ1dCBwZXJoYXBzIHRoZXJlIGlzIHNvbWV0aGluZyBpbiBTZXRDb29wZXJhdGl2ZUxldmVsIHRvIHByZXZlbnQgdGhpcykKICoKICogVGhlIG1zZG4gc2F5cyB0aGF0IHRoaXMgbWV0aG9kIHJlc2V0cyB0aGUgZGlzcGxheSBtb2RlIHRvIHdoYXQgaXQgd2FzIGJlZm9yZQogKiBTZXREaXNwbGF5TW9kZSB3YXMgY2FsbGVkLiBXaGF0IGlmIFNldERpc3BsYXlNb2RlcyBpcyBjYWxsZWQgMiB0aW1lcz8/CiAqCiAqIFJldHVybnMKICogIEREX09LIG9uIHN1Y2Nlc3MKICogIERERVJSX05PRVhDTFVTSVZFIG1vZGUgaWYgdGhlIGRldmljZSBpc24ndCBpbiBmdWxsc2NyZWVuIG1vZGUKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX1Jlc3RvcmVEaXNwbGF5TW9kZShJRGlyZWN0RHJhdzcgKmlmYWNlKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgVFJBQ0UoIiglcClcbiIsIFRoaXMpOwoKICAgIHJldHVybiBJRGlyZWN0RHJhdzdfU2V0RGlzcGxheU1vZGUoSUNPTV9JTlRFUkZBQ0UoVGhpcywgSURpcmVjdERyYXc3KSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVGhpcy0+b3JpZ193aWR0aCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVGhpcy0+b3JpZ19oZWlnaHQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRoaXMtPm9yaWdfYnBwLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAwLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAwKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6R2V0Q2FwcwogKgogKiBSZXR1cm5zIHRoZSBkcml2ZXMgY2FwYWJpbGl0aWVzCiAqCiAqIFVzZWQgZm9yIHZlcnNpb24gMSwgMiwgNCBhbmQgNwogKgogKiBQYXJhbXM6CiAqICBEcml2ZXJDYXBzOiBTdHJ1Y3R1cmUgdG8gd3JpdGUgdGhlIEhhcmR3YXJlIGFjY2VsZXJhdGVkIGNhcHMgdG8KICogIEhlbENhcHM6IFN0cnVjdHVyZSB0byB3cml0ZSB0aGUgZW11bGF0aW9uIGNhcHMgdG8KICoKICogUmV0dXJucwogKiAgVGhpcyBpbXBsZW1lbnRhdGlvbiByZXR1cm5zIEREX09LIG9ubHkKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0dldENhcHMoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgRERDQVBTICpEcml2ZXJDYXBzLAogICAgICAgICAgICAgICAgICAgICAgICBERENBUFMgKkhFTENhcHMpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBUUkFDRSgiKCVwKS0+KCVwLCVwKVxuIiwgVGhpcywgRHJpdmVyQ2FwcywgSEVMQ2Fwcyk7CgogICAgLyogT25lIHN0cnVjdHVyZSBtdXN0IGJlICE9IE5VTEwgKi8KICAgIGlmKCAoIURyaXZlckNhcHMpICYmICghSEVMQ2FwcykgKQogICAgewogICAgICAgIEVSUigiKCVwKSBJbnZhbGlkIHBhcmFtcyB0byBJRGlyZWN0RHJhd0ltcGxfR2V0Q2Fwc1xuIiwgVGhpcyk7CiAgICAgICAgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CiAgICB9CgogICAgaWYoRHJpdmVyQ2FwcykKICAgIHsKICAgICAgICBERF9TVFJVQ1RfQ09QWV9CWVNJWkUoRHJpdmVyQ2FwcywgJlRoaXMtPmNhcHMpOwogICAgICAgIGlmIChUUkFDRV9PTihkZHJhdykpCiAgICAgICAgewogICAgICAgICAgICBUUkFDRSgiRHJpdmVyIENhcHMgOlxuIik7CiAgICAgICAgICAgIEREUkFXX2R1bXBfRERDQVBTKERyaXZlckNhcHMpOwogICAgICAgIH0KCiAgICB9CiAgICBpZihIRUxDYXBzKQogICAgewogICAgICAgIEREX1NUUlVDVF9DT1BZX0JZU0laRShIRUxDYXBzLCAmVGhpcy0+Y2Fwcyk7CiAgICAgICAgaWYgKFRSQUNFX09OKGRkcmF3KSkKICAgICAgICB7CiAgICAgICAgICAgIFRSQUNFKCJIRUwgQ2FwcyA6XG4iKTsKICAgICAgICAgICAgRERSQVdfZHVtcF9ERENBUFMoSEVMQ2Fwcyk7CiAgICAgICAgfQogICAgfQoKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6Q29tcGFjdAogKgogKiBObyBpZGVhIHdoYXQgaXQgZG9lcywgTVNETiBzYXlzIGl0J3Mgbm90IGltcGxlbWVudGVkLgogKgogKiBSZXR1cm5zCiAqICBERF9PSywgYnV0IHRoaXMgaXMgdW5jaGVja2VkCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9Db21wYWN0KElEaXJlY3REcmF3NyAqaWZhY2UpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBUUkFDRSgiKCVwKVxuIiwgVGhpcyk7CgogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXREaXNwbGF5TW9kZQogKgogKiBSZXR1cm5zIGluZm9ybWF0aW9uIGFib3V0IHRoZSBjdXJyZW50IGRpc3BsYXkgbW9kZQogKgogKiBFeGlzdHMgaW4gVmVyc2lvbiAxLCAyLCA0IGFuZCA3CiAqCiAqIFBhcmFtczoKICogIEREU0Q6IEFkZHJlc3Mgb2YgYSBzdXJmYWNlIGRlc2NyaXB0aW9uIHN0cnVjdHVyZSB0byB3cml0ZSB0aGUgaW5mbyB0bwogKgogKiBSZXR1cm5zCiAqICBERF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfR2V0RGlzcGxheU1vZGUoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEREU1VSRkFDRURFU0MyICpERFNEKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgSFJFU1VMVCBocjsKICAgIFdJTkVEM0RESVNQTEFZTU9ERSBNb2RlOwogICAgRFdPUkQgU2l6ZTsKICAgIFRSQUNFKCIoJXApLT4oJXApOiBSZWxheVxuIiwgVGhpcywgRERTRCk7CgogICAgLyogVGhpcyBzZWVtcyBzYW5lICovCiAgICBpZighRERTRCkgCiAgICB7CiAgICAgICAgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CiAgICB9CgogICAgLyogVGhlIG5lY2Vzc2FyeSBtZW1iZXJzIG9mIExQRERTVVJGQUNFREVTQyBhbmQgTFBERFNVUkZBQ0VERVNDMiBhcmUgZXF1YWwsCiAgICAgKiBzbyBvbmUgbWV0aG9kIGNhbiBiZSB1c2VkIGZvciBhbGwgdmVyc2lvbnMgKEhvcGVmdWxseSkKICAgICAqLwogICAgaHIgPSBJV2luZUQzRERldmljZV9HZXREaXNwbGF5TW9kZShUaGlzLT53aW5lRDNERGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAgLyogc3dhcGNoYWluIDAgKi8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJk1vZGUpOwogICAgaWYoIGhyICE9IEQzRF9PSyApCiAgICB7CiAgICAgICAgRVJSKCIgKCVwKSBJV2luZUQzRERldmljZTo6R2V0RGlzcGxheU1vZGUgcmV0dXJuZWQgJTA4bHhcbiIsIFRoaXMsIGhyKTsKICAgICAgICByZXR1cm4gaHI7CiAgICB9CgogICAgU2l6ZSA9IEREU0QtPmR3U2l6ZTsKICAgIG1lbXNldChERFNELCAwLCBTaXplKTsKCiAgICBERFNELT5kd1NpemUgPSBTaXplOwogICAgRERTRC0+ZHdGbGFncyB8PSBERFNEX0hFSUdIVCB8IEREU0RfV0lEVEggfCBERFNEX1BJWEVMRk9STUFUIHwgRERTRF9QSVRDSCB8IEREU0RfUkVGUkVTSFJBVEU7CiAgICBERFNELT5kd1dpZHRoID0gTW9kZS5XaWR0aDsKICAgIEREU0QtPmR3SGVpZ2h0ID0gTW9kZS5IZWlnaHQ7IAogICAgRERTRC0+dTIuZHdSZWZyZXNoUmF0ZSA9IDYwOwogICAgRERTRC0+ZGRzQ2Fwcy5kd0NhcHMgPSAwOwogICAgRERTRC0+dTQuZGRwZlBpeGVsRm9ybWF0LmR3U2l6ZSA9IHNpemVvZihERFNELT51NC5kZHBmUGl4ZWxGb3JtYXQpOwogICAgUGl4ZWxGb3JtYXRfV2luZUQzRHRvREQoJkREU0QtPnU0LmRkcGZQaXhlbEZvcm1hdCwgTW9kZS5Gb3JtYXQpOwogICAgRERTRC0+dTEubFBpdGNoID0gTW9kZS5XaWR0aCAqIEREU0QtPnU0LmRkcGZQaXhlbEZvcm1hdC51MS5kd1JHQkJpdENvdW50IC8gODsKCiAgICBpZihUUkFDRV9PTihkZHJhdykpCiAgICB7CiAgICAgICAgVFJBQ0UoIlJldHVybmluZyBzdXJmYWNlIGRlc2MgOlxuIik7CiAgICAgICAgRERSQVdfZHVtcF9zdXJmYWNlX2Rlc2MoRERTRCk7CiAgICB9CgogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXRGb3VyQ0NDb2RlcwogKgogKiBSZXR1cm5zIGFuIGFycmF5IG9mIHN1cHBvcnRlZCBGb3VyQ0MgY29kZXMuCiAqCiAqIEV4aXN0cyBpbiBWZXJzaW9uIDEsIDIsIDQgYW5kIDcKICoKICogUGFyYW1zOgogKiAgTnVtQ29kZXM6IENvbnRhaW5zIHRoZSBudW1iZXIgb2YgQ29kZXMgdGhhdCBDb2RlcyBjYW4gY2FycnkuIFJldHVybnMgdGhlIG51bWJlcgogKiAgICAgICAgICAgIG9mIGVudW1lcmF0ZWQgY29kZXMKICogIENvZGVzOiBQb2ludGVyIHRvIGFuIGFycmF5IG9mIERXT1JEcyB3aGVyZSB0aGUgc3VwcG9ydGVkIGNvZGVzIGFyZSB3cml0dGVuCiAqICAgICAgICAgdG8KICoKICogUmV0dXJucwogKiAgQWx3YXlzIHJldHVybnMgRERfT0ssIGFzIGl0J3MgYSBzdHViIGZvciBub3cKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0dldEZvdXJDQ0NvZGVzKElEaXJlY3REcmF3NyAqaWZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCAqTnVtQ29kZXMsIERXT1JEICpDb2RlcykKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIEZJWE1FKCIoJXApLT4oJXAsICVwKTogU3R1YiFcbiIsIFRoaXMsIE51bUNvZGVzLCBDb2Rlcyk7CgogICAgaWYoTnVtQ29kZXMpICpOdW1Db2RlcyA9IDA7CgogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXRNb25pdG9yRnJlcXVlbmN5CiAqCiAqIFJldHVybnMgdGhlIG1vbml0b3IncyBmcmVxdWVuY3kKICoKICogRXhpc3RzIGluIFZlcnNpb24gMSwgMiwgNCBhbmQgNwogKgogKiBQYXJhbXM6CiAqICBGcmVxOiBQb2ludGVyIHRvIGEgRFdPUkQgdG8gd3JpdGUgdGhlIGZyZXF1ZW5jeSB0bwogKgogKiBSZXR1cm5zCiAqICBBbHdheXMgcmV0dXJucyBERF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfR2V0TW9uaXRvckZyZXF1ZW5jeShJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCAqRnJlcSkKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIFRSQUNFKCIoJXApLT4oJXApXG4iLCBUaGlzLCBGcmVxKTsKCiAgICAvKiBJZGVhbGx5IHRoaXMgc2hvdWxkIGJlIGluIFdpbmVEM0QsIGFzIGl0IGNvbmNlcm5zIHRoZSBzY3JlZW4gc2V0dXAsCiAgICAgKiBidXQgZm9yIG5vdyB0aGlzIHNob3VsZCBtYWtlIHRoZSBnYW1lcyBoYXBweQogICAgICovCiAgICAqRnJlcSA9IDYwOwogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXRWZXJ0aWNhbEJsYW5rU3RhdHVzCiAqCiAqIFJldHVybnMgdGhlIFZlcnRpY2FsIGJsYW5rIHN0YXR1cyBvZiB0aGUgbW9uaXRvci4gVGhpcyBzaG91bGQgYmUgaW4gV2luZUQzRAogKiB0b28gYmFzaWNhbGx5LCBidXQgYXMgaXQncyBhIHNlbWkgc3R1YiwgSSBkaWRuJ3QgY3JlYXRlIGEgZnVuY3Rpb24gdGhlcmUKICoKICogUGFyYW1zOgogKiAgc3RhdHVzOiBQb2ludGVyIHRvIGEgQk9PTCB0byBiZSBmaWxsZWQgd2l0aCB0aGUgdmVydGljYWwgYmxhbmsgc3RhdHVzCiAqCiAqIFJldHVybnMKICogIEREX09LIG9uIHN1Y2Nlc3MKICogIERERVJSX0lOVkFMSURQQVJBTVMgaWYgc3RhdHVzIGlzIE5VTEwKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0dldFZlcnRpY2FsQmxhbmtTdGF0dXMoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQk9PTCAqc3RhdHVzKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgVFJBQ0UoIiglcCktPiglcClcbiIsIFRoaXMsIHN0YXR1cyk7CgogICAgLyogVGhpcyBsb29rcyBzYW5lLCB0aGUgTVNETiBzdWdnZXN0cyBpdCB0b28gKi8KICAgIGlmKCFzdGF0dXMpIHJldHVybiBEREVSUl9JTlZBTElEUEFSQU1TOwoKICAgICpzdGF0dXMgPSBUaGlzLT5mYWtlX3ZibGFuazsKICAgIFRoaXMtPmZha2VfdmJsYW5rID0gIVRoaXMtPmZha2VfdmJsYW5rOwogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXRBdmFpbGFibGVWaWRNZW0KICoKICogUmV0dXJucyB0aGUgdG90YWwgYW5kIGZyZWUgdmlkZW8gbWVtb3J5CiAqCiAqIFBhcmFtczoKICogIENhcHM6IFNwZWNpZmllcyB0aGUgbWVtb3J5IHR5cGUgYXNrZWQgZm9yCiAqICB0b3RhbDogUG9pbnRlciB0byBhIERXT1JEIHRvIGJlIGZpbGxlZCB3aXRoIHRoZSB0b3RhbCBtZW1vcnkKICogIGZyZWU6IFBvaW50ZXIgdG8gYSBEV09SRCB0byBiZSBmaWxsZWQgd2l0aCB0aGUgZnJlZSBtZW1vcnkKICoKICogUmV0dXJucwogKiAgRERfT0sgb24gc3VjY2VzcwogKiAgRERFUlJfSU5WQUxJRFBBUkFNUyBvZiBmcmVlIGFuZCB0b3RhbCBhcmUgTlVMTAogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfR2V0QXZhaWxhYmxlVmlkTWVtKElEaXJlY3REcmF3NyAqaWZhY2UsIEREU0NBUFMyICpDYXBzLCBEV09SRCAqdG90YWwsIERXT1JEICpmcmVlKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgVFJBQ0UoIiglcCktPiglcCwgJXAsICVwKVxuIiwgVGhpcywgQ2FwcywgdG90YWwsIGZyZWUpOwoKICAgIGlmKFRSQUNFX09OKGRkcmF3KSkKICAgIHsKICAgICAgICBUUkFDRSgiKCVwKSBBc2tlZCBmb3IgbWVtb3J5IHdpdGggZGVzY3JpcHRpb246ICIsIFRoaXMpOwogICAgICAgIEREUkFXX2R1bXBfRERTQ0FQUzIoQ2Fwcyk7CiAgICAgICAgVFJBQ0UoIlxuIik7CiAgICB9CgogICAgLyogVG9kbzogU3lzdGVtIG1lbW9yeSB2cyBsb2NhbCB2aWRlbyBtZW1vcnkgdnMgbm9uLWxvY2FsIHZpZGVvIG1lbW9yeQogICAgICogVGhlIE1TRE4gYWxzbyBtZW50aW9ucyBkaWZmZXJlbmNlcyBiZXR3ZWVuIHRleHR1cmUgbWVtb3J5IGFuZCBvdGhlcgogICAgICogcmVzb3VyY2VzLCBidXQgdGhhdCdzIG5vdCBpbXBvcnRhbnQKICAgICAqLwoKICAgIGlmKCAoIXRvdGFsKSAmJiAoIWZyZWUpICkgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CgogICAgaWYodG90YWwpICp0b3RhbCA9IFRoaXMtPnRvdGFsX3ZpZG1lbTsKICAgIGlmKGZyZWUpICpmcmVlID0gSVdpbmVEM0REZXZpY2VfR2V0QXZhaWxhYmxlVGV4dHVyZU1lbShUaGlzLT53aW5lRDNERGV2aWNlKTsKCiAgICByZXR1cm4gRERfT0s7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OkluaXRpYWxpemUKICoKICogSW5pdGlhbGl6ZXMgYSBEaXJlY3REcmF3IGludGVyZmFjZS4KICoKICogUGFyYW1zOgogKiAgR1VJRDogSW50ZXJmYWNlIGlkZW50aWZpZXIuIFdlbGwsIGRvbid0IGtub3cgd2hhdCB0aGlzIGlzIHJlYWxseSBnb29kCiAqICAgZm9yCiAqCiAqIFJldHVybnMKICogIFJldHVybnMgRERfT0sgb24gdGhlIGZpcnN0IGNhbGwsCiAqICBEREVSUl9BTFJFQURZSU5JVElBTElaRUQgb24gcmVwZWF0ZWQgY2FsbHMKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0luaXRpYWxpemUoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgR1VJRCAqR3VpZCkKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIFRSQUNFKCIoJXApLT4oJXMpOiBOby1vcFxuIiwgVGhpcywgZGVidWdzdHJfZ3VpZChHdWlkKSk7CgogICAgaWYoVGhpcy0+aW5pdGlhbGl6ZWQpCiAgICB7CiAgICAgICAgcmV0dXJuIERERVJSX0FMUkVBRFlJTklUSUFMSVpFRDsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICByZXR1cm4gRERfT0s7CiAgICB9Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OkZsaXBUb0dESVN1cmZhY2UKICoKICogIk1ha2VzIHRoZSBzdXJmYWNlIHRoYXQgdGhlIEdESSB3cml0ZXMgdG8gdGhlIHByaW1hcnkgc3VyZmFjZSIKICogTG9va3MgbGlrZSBzb21lIHdpbmRvd3Mgc3BlY2lmaWMgdGhpbmcgd2UgZG9uJ3QgaGF2ZSB0byBjYXJlIGFib3V0LgogKiBBY2NvcmRpbmcgdG8gTVNETiBpdCBwZXJtaXRzIEdESSBkaWFsb2cgYm94ZXMgaW4gRlVMTFNDUkVFTiBtb2RlLiBHb29kIHRvCiAqIHNob3cgZXJyb3IgYm94ZXMgOykKICogV2VsbCwganVzdCByZXR1cm4gRERfT0suCiAqCiAqIFJldHVybnM6CiAqICBBbHdheXMgcmV0dXJucyBERF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfRmxpcFRvR0RJU3VyZmFjZShJRGlyZWN0RHJhdzcgKmlmYWNlKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgVFJBQ0UoIiglcClcbiIsIFRoaXMpOwoKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6V2FpdEZvclZlcnRpY2FsQmxhbmsKICoKICogVGhpcyBtZXRob2QgYWxsb3dzIGFwcGxpY2F0aW9ucyB0byBnZXQgaW4gc3luYyB3aXRoIHRoZSB2ZXJ0aWNhbCBibGFuawogKiBpbnRlcnZhbC4KICogVGhlIHdvcm1ob2xlIGRlbW8gaW4gdGhlIERpcmVjdFggNyBzZGsgdXNlcyB0aGlzIGNhbGwsIGFuZCBpdCBkb2Vzbid0CiAqIHJlZHJhdyB0aGUgc2NyZWVuLCBtb3N0IGxpa2VseSBiZWNhdXNlIG9mIHRoaXMgc3R1YgogKgogKiBQYXJhbWV0ZXJzOgogKiAgRmxhZ3M6IG9uZSBvZiBERFdBSVRWQl9CTE9DS0JFR0lOLCBERFdBSVRWQl9CTE9DS0JFR0lORVZFTlQKICogICAgICAgICBvciBERFdBSVRWQl9CTE9DS0VORAogKiAgaDogTm90IHVzZWQsIGFjY29yZGluZyB0byBNU0ROCiAqCiAqIFJldHVybnM6CiAqICBBbHdheXMgcmV0dXJucyBERF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovIApzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX1dhaXRGb3JWZXJ0aWNhbEJsYW5rKElEaXJlY3REcmF3NyAqaWZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBGbGFncywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEhBTkRMRSBoKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgRklYTUUoIiglcCktPiglbHgsJXApOiBTdHViXG4iLCBUaGlzLCBGbGFncywgaCk7CgogICAgLyogTVNETiBzYXlzIEREV0FJVFZCX0JMT0NLQkVHSU5FVkVOVCBpcyBub3Qgc3VwcG9ydGVkICovCiAgICBpZihGbGFncyAmIEREV0FJVFZCX0JMT0NLQkVHSU5FVkVOVCkKICAgICAgICByZXR1cm4gRERFUlJfVU5TVVBQT1JURUQ7IC8qIHVuY2hlY2tlZCAqLwoKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6R2V0U2NhbkxpbmUKICoKICogUmV0dXJucyB0aGUgc2NhbiBsaW5lIHRoYXQgaXMgYmVpbmcgZHJhd24gb24gdGhlIG1vbml0b3IKICoKICogUGFyYW1ldGVyczoKICogIFNjYW5saW5lOiBBZGRyZXNzIHRvIHdyaXRlIHRoZSBzY2FuIGxpbmUgdmFsdWUgdG8KICoKICogUmV0dXJuczoKICogIEFsd2F5cyByZXR1cm5zIEREX09LCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8gCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJRGlyZWN0RHJhd0ltcGxfR2V0U2NhbkxpbmUoSURpcmVjdERyYXc3ICppZmFjZSwgRFdPUkQgKlNjYW5saW5lKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgc3RhdGljIEJPT0wgaGlkZSA9IEZBTFNFOwogICAgV0lORUQzRERJU1BMQVlNT0RFIE1vZGU7CgogICAgLyogVGhpcyBmdW5jdGlvbiBpcyBjYWxsZWQgb2Z0ZW4sIHNvIHByaW50IHRoZSBmaXhtZSBvbmx5IG9uY2UgKi8KICAgIGlmKCFoaWRlKQogICAgewogICAgICAgIEZJWE1FKCIoJXApLT4oJXApOiBTZW1pLVN0dWJcbiIsIFRoaXMsIFNjYW5saW5lKTsKICAgICAgICBoaWRlID0gVFJVRTsKICAgIH0KCiAgICBJV2luZUQzRERldmljZV9HZXREaXNwbGF5TW9kZShUaGlzLT53aW5lRDNERGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZNb2RlKTsKCiAgICAvKiBGYWtlIHRoZSBsaW5lIHN3ZWVwaW5nIG9mIHRoZSBtb25pdG9yICovCiAgICAvKiBGSVhNRTogV2Ugc2hvdWxkIHN5bmNocm9uaXplIHdpdGggYSBzb3VyY2UgdG8ga2VlcCB0aGUgcmVmcmVzaCByYXRlICovIAogICAgKlNjYW5saW5lID0gVGhpcy0+Y3VyX3NjYW5saW5lKys7CiAgICAvKiBBc3N1bWUgMjAgc2NhbiBsaW5lcyBpbiB0aGUgdmVydGljYWwgYmxhbmsgKi8KICAgIGlmIChUaGlzLT5jdXJfc2NhbmxpbmUgPj0gTW9kZS5IZWlnaHQgKyAyMCkKICAgICAgICBUaGlzLT5jdXJfc2NhbmxpbmUgPSAwOwoKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6VGVzdENvb3BlcmF0aXZlTGV2ZWwKICoKICogSW5mb3JtcyB0aGUgYXBwbGljYXRpb24gYWJvdXQgdGhlIHN0YXRlIG9mIHRoZSB2aWRlbyBhZGFwdGVyLCBkZXBlbmRpbmcKICogb24gdGhlIGNvb3BlcmF0aXZlIGxldmVsCiAqCiAqIFJldHVybnM6CiAqICBERF9PSyBpZiB0aGUgZGV2aWNlIGlzIGluIGEgc2FuZSBzdGF0ZQogKiAgRERFUlJfTk9FWENMVVNJVkVNT0RFIG9yIERERVJSX0VYQ0xVU0lWRU1PREVBTFJFQURZU0VUCiAqICBpZiB0aGUgc3RhdGUgaXMgbm90IGNvcnJlY3QoU2VlIGJlbG93KQogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovIApzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX1Rlc3RDb29wZXJhdGl2ZUxldmVsKElEaXJlY3REcmF3NyAqaWZhY2UpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBIUkVTVUxUIGhyOwogICAgVFJBQ0UoIiglcClcbiIsIFRoaXMpOwoKICAgIC8qIERlc2NyaXB0aW9uIGZyb20gTVNETjoKICAgICAqIEZvciBmdWxsc2NyZWVuIGFwcHMgcmV0dXJuIERERVJSX05PRVhDTFVTSVZFTU9ERSBpZiB0aGUgdXNlciBzd2l0Y2hlZAogICAgICogYXdheSBmcm9tIHRoZSBhcHAgd2l0aCBlLmcuIGFsdC10YWIuIFdpbmRvd2VkIGFwcHMgcmVjZWl2ZSAKICAgICAqIERERVJSX0VYQ0xVU0lWRU1PREVBTFJFQURZU0VUIGlmIGFub3RoZXIgYXBwbGljYXRpb24gY3JlYXRlZCBhIAogICAgICogRGlyZWN0RHJhdyBvYmplY3QgaW4gZXhjbHVzaXZlIG1vZGUuIERERVJSX1dST05HTU9ERSBpcyByZXR1cm5lZCwKICAgICAqIHdoZW4gdGhlIHZpZGVvIG1vZGUgaGFzIGNoYW5nZWQKICAgICAqLwoKICAgIGhyID0gIElXaW5lRDNERGV2aWNlX1Rlc3RDb29wZXJhdGl2ZUxldmVsKFRoaXMtPndpbmVEM0REZXZpY2UpOwoKICAgIC8qIEZpeCB0aGUgcmVzdWx0IHZhbHVlLiBUaGVzZSB2YWx1ZXMgYXJlIG1hcHBlZCBmcm9tIHRoZWlyCiAgICAgKiBkM2Q5IGNvdW50ZXJwYXJ0LgogICAgICovCiAgICBzd2l0Y2goaHIpCiAgICB7CiAgICAgICAgY2FzZSBXSU5FRDNERVJSX0RFVklDRUxPU1Q6CiAgICAgICAgICAgIGlmKFRoaXMtPmNvb3BlcmF0aXZlX2xldmVsICYgRERTQ0xfRVhDTFVTSVZFKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICByZXR1cm4gRERFUlJfTk9FWENMVVNJVkVNT0RFOwogICAgICAgICAgICB9CiAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgcmV0dXJuIERERVJSX0VYQ0xVU0lWRU1PREVBTFJFQURZU0VUOwogICAgICAgICAgICB9CgogICAgICAgIGNhc2UgV0lORUQzREVSUl9ERVZJQ0VOT1RSRVNFVDoKICAgICAgICAgICAgcmV0dXJuIEREX09LOwoKICAgICAgICBjYXNlIFdJTkVEM0RfT0s6CiAgICAgICAgICAgIHJldHVybiBERF9PSzsKCiAgICAgICAgY2FzZSBXSU5FRDNERVJSX0RSSVZFUklOVEVSTkFMRVJST1I6CiAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgRVJSKCIoJXApIFVuZXhwZWN0ZWQgcmV0dXJuIHZhbHVlICUwOGx4IGZyb20gd2luZUQzRCwgIiBcCiAgICAgICAgICAgICAgICAiIHJldHVybmluZyBERF9PS1xuIiwgVGhpcywgaHIpOwogICAgfQoKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6R2V0R0RJU3VyZmFjZQogKgogKiBSZXR1cm5zIHRoZSBzdXJmYWNlIHRoYXQgR0RJIGlzIHRyZWF0aW5nIGFzIHRoZSBwcmltYXJ5IHN1cmZhY2UuCiAqIEZvciBXaW5lIHRoaXMgaXMgdGhlIGZyb250IGJ1ZmZlcgogKgogKiBQYXJhbXM6CiAqICBHRElTdXJmYWNlOiBBZGRyZXNzIHRvIHdyaXRlIHRoZSBzdXJmYWNlIHBvaW50ZXIgdG8KICoKICogUmV0dXJuczoKICogIEREX09LIGlmIHRoZSBzdXJmYWNlIHdhcyBmb3VuZAogKiAgRERFUlJfTk9URk9VTkQgaWYgdGhlIEdESSBzdXJmYWNlIHdhc24ndCBmb3VuZAogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovIApzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0dldEdESVN1cmZhY2UoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSURpcmVjdERyYXdTdXJmYWNlNyAqKkdESVN1cmZhY2UpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBJV2luZUQzRFN1cmZhY2UgKlN1cmY7CiAgICBJRGlyZWN0RHJhd1N1cmZhY2U3ICpkZHN1cmY7CiAgICBIUkVTVUxUIGhyOwogICAgRERTQ0FQUzIgZGRzQ2FwczsKICAgIFRSQUNFKCIoJXApLT4oJXApXG4iLCBUaGlzLCBHRElTdXJmYWNlKTsKCiAgICAvKiBHZXQgdGhlIGJhY2sgYnVmZmVyIGZyb20gdGhlIHdpbmVEM0REZXZpY2UgYW5kIHNlYXJjaCBpdHMKICAgICAqIGF0dGFjaGVkIHN1cmZhY2VzIGZvciB0aGUgZnJvbnQgYnVmZmVyCiAgICAgKi8KICAgIGhyID0gSVdpbmVEM0REZXZpY2VfR2V0QmFja0J1ZmZlcihUaGlzLT53aW5lRDNERGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsIC8qIFN3YXBDaGFpbiAqLwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsIC8qIGZpcnN0IGJhY2sgYnVmZmVyKi8KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBXSU5FRDNEQkFDS0JVRkZFUl9UWVBFX01PTk8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJlN1cmYpOwoKICAgIGlmKCAoaHIgIT0gRDNEX09LKSB8fAogICAgICAgICghU3VyZikgKQogICAgewogICAgICAgIEVSUigiSVdpbmVEM0REZXZpY2U6OkdldEJhY2tCdWZmZXIgZmFpbGVkXG4iKTsKICAgICAgICByZXR1cm4gRERFUlJfTk9URk9VTkQ7CiAgICB9CgogICAgLyogR2V0QmFja0J1ZmZlciBBZGRSZWYoKWVkIHRoZSBzdXJmYWNlLCByZWxlYXNlIGl0ICovCiAgICBJV2luZUQzRFN1cmZhY2VfUmVsZWFzZShTdXJmKTsKCiAgICBJV2luZUQzRFN1cmZhY2VfR2V0UGFyZW50KFN1cmYsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChJVW5rbm93biAqKikgJmRkc3VyZik7CiAgICBJRGlyZWN0RHJhd1N1cmZhY2U3X1JlbGVhc2UoZGRzdXJmKTsgIC8qIEZvciB0aGUgR2V0UGFyZW50ICovCgogICAgLyogRmluZCB0aGUgZnJvbnQgYnVmZmVyICovCiAgICBkZHNDYXBzLmR3Q2FwcyA9IEREU0NBUFNfRlJPTlRCVUZGRVI7CiAgICBociA9IElEaXJlY3REcmF3U3VyZmFjZTdfR2V0QXR0YWNoZWRTdXJmYWNlKGRkc3VyZiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJmRkc0NhcHMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEdESVN1cmZhY2UpOwogICAgaWYoaHIgIT0gRERfT0spCiAgICB7CiAgICAgICAgRVJSKCJJRGlyZWN0RHJhd1N1cmZhY2U3OjpHZXRBdHRhY2hlZFN1cmZhY2UgZmFpbGVkLCBociA9ICVseFxuIiwgaHIpOwogICAgfQoKICAgIC8qIFRoZSBBZGRSZWYgaXMgT0sgdGhpcyB0aW1lICovCiAgICByZXR1cm4gaHI7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhd0ltcGxfRW51bURpc3BsYXlNb2Rlc0NCCiAqCiAqIENhbGxiYWNrIGZ1bmN0aW9uIGZvciBJRGlyZWN0RHJhdzc6OkVudW1EaXNwbGF5TW9kZXMuIFRyYW5zbGF0ZXMKICogdGhlIHdpbmVEM0QgdmFsdWVzIHRvIGRkcmF3IHZhbHVlcyBhbmQgY2FsbHMgdGhlIGFwcGxpY2F0aW9uIGNhbGxiYWNrCiAqCiAqIFBhcmFtczoKICogIGRldmljZTogVGhlIElEaXJlY3REcmF3NyBpbnRlcmZhY2UgdG8gdGhlIGN1cnJlbnQgZGV2aWNlCiAqICBXaXRoLCBIZWlnaHQsIFBpeGVsZm9ybWF0LCBSZWZyZXNoOiBFbnVtZXJhdGVkIGRpc3BsYXkgbW9kZQogKiAgY29udGV4dDogdGhlIGNvbnRleHQgcG9pbnRlciBwYXNzZWQgdG8gSVdpbmVEM0REZXZpY2U6OkVudW1EaXNwbGF5TW9kZXMKICoKICogUmV0dXJuczoKICogIFRoZSByZXR1cm4gdmFsdWUgZnJvbSB0aGUgYXBwbGljYXRpb24gY2FsbGJhY2sKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLyAKc3RhdGljIEhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9FbnVtRGlzcGxheU1vZGVzQ0IoSVVua25vd24gKnBEZXZpY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUlOVCBXaWR0aCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVSU5UIEhlaWdodCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBXSU5FRDNERk9STUFUIFBpeGVsZm9ybWF0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEZMT0FUIFJlZnJlc2gsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCAqY29udGV4dCkKewogICAgRERTVVJGQUNFREVTQzIgY2FsbGJhY2tfc2Q7CiAgICBFbnVtRGlzcGxheU1vZGVzQ0JTICpjYnMgPSAoRW51bURpc3BsYXlNb2Rlc0NCUyAqKSBjb250ZXh0OwoKICAgIG1lbXNldCgmY2FsbGJhY2tfc2QsIDAsIHNpemVvZihjYWxsYmFja19zZCkpOwogICAgY2FsbGJhY2tfc2QuZHdTaXplID0gc2l6ZW9mKGNhbGxiYWNrX3NkKTsKICAgIGNhbGxiYWNrX3NkLnU0LmRkcGZQaXhlbEZvcm1hdC5kd1NpemUgPSBzaXplb2YoRERQSVhFTEZPUk1BVCk7CgogICAgY2FsbGJhY2tfc2QuZHdGbGFncyA9IEREU0RfSEVJR0hUfEREU0RfV0lEVEh8RERTRF9QSVhFTEZPUk1BVHxERFNEX1BJVENIOwogICAgaWYoUmVmcmVzaCA+IDAuMCkKICAgIHsKICAgICAgICBjYWxsYmFja19zZC5kd0ZsYWdzIHw9IEREU0RfUkVGUkVTSFJBVEU7CiAgICAgICAgY2FsbGJhY2tfc2QudTIuZHdSZWZyZXNoUmF0ZSA9IDYwLjA7CiAgICB9CgogICAgY2FsbGJhY2tfc2QuZHdIZWlnaHQgPSBIZWlnaHQ7CiAgICBjYWxsYmFja19zZC5kd1dpZHRoID0gV2lkdGg7CgogICAgUGl4ZWxGb3JtYXRfV2luZUQzRHRvREQoJmNhbGxiYWNrX3NkLnU0LmRkcGZQaXhlbEZvcm1hdCwgUGl4ZWxmb3JtYXQpOwogICAgcmV0dXJuIGNicy0+Y2FsbGJhY2soJmNhbGxiYWNrX3NkLCBjYnMtPmNvbnRleHQpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpFbnVtRGlzcGxheU1vZGVzCiAqCiAqIEVudW1lcmF0ZXMgdGhlIHN1cHBvcnRlZCBEaXNwbGF5IG1vZGVzLiBUaGUgbW9kZXMgY2FuIGJlIGZpbHRlcmVkIHdpdGgKICogdGhlIEREU0QgcGFyYW1ldGVyLgogKgogKiBQYXJhbXM6CiAqICBGbGFnczogY2FuIGJlIERERURNX1JFRlJFU0hSQVRFUyBhbmQgRERFRE1fU1RBTkRBUkRWR0FNT0RFUwogKiAgRERTRDogU3VyZmFjZSBkZXNjcmlwdGlvbiB0byBmaWx0ZXIgdGhlIG1vZGVzCiAqICBDb250ZXh0OiBQb2ludGVyIHBhc3NlZCBiYWNrIHRvIHRoZSBjYWxsYmFjayBmdW5jdGlvbgogKiAgY2I6IEFwcGxpY2F0aW9uLXByb3ZpZGVkIGNhbGxiYWNrIGZ1bmN0aW9uCiAqCiAqIFJldHVybnM6CiAqICBERF9PSyBvbiBzdWNjZXNzCiAqICBEREVSUl9JTlZBTElEUEFSQU1TIGlmIHRoZSBjYWxsYmFjayB3YXNuJ3Qgc2V0CiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8gCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfRW51bURpc3BsYXlNb2RlcyhJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBGbGFncywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRERTVVJGQUNFREVTQzIgKkREU0QsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZvaWQgKkNvbnRleHQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQRERFTlVNTU9ERVNDQUxMQkFDSzIgY2IpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBVSU5UIFdpZHRoID0gMCwgSGVpZ2h0ID0gMDsKICAgIFdJTkVEM0RGT1JNQVQgcGl4ZWxmb3JtYXQgPSBXSU5FRDNERk1UX1VOS05PV047CiAgICBFbnVtRGlzcGxheU1vZGVzQ0JTIGNiczsKCiAgICBUUkFDRSgiKCVwKS0+KCVwLCVwLCVwKTogUmVsYXlcbiIsIFRoaXMsIEREU0QsIENvbnRleHQsIGNiKTsKCiAgICAvKiBUaGlzIGxvb2tzIHNhbmUgKi8KICAgIGlmKCFjYikgcmV0dXJuIERERVJSX0lOVkFMSURQQVJBTVM7CgogICAgLyogVGhlIHByaXZhdGUgY2FsbGJhY2sgc3RydWN0dXJlICovCiAgICBjYnMuY2FsbGJhY2sgPSBjYjsKICAgIGNicy5jb250ZXh0ID0gQ29udGV4dDsKCiAgICBpZihERFNEKQogICAgewogICAgICAgIGlmIChERFNELT5kd0ZsYWdzICYgRERTRF9XSURUSCkKICAgICAgICAgICAgV2lkdGggPSBERFNELT5kd1dpZHRoOwogICAgICAgIGlmIChERFNELT5kd0ZsYWdzICYgRERTRF9IRUlHSFQpCiAgICAgICAgICAgIEhlaWdodCA9IEREU0QtPmR3SGVpZ2h0OwogICAgICAgIGlmICgoRERTRC0+ZHdGbGFncyAmIEREU0RfUElYRUxGT1JNQVQpICYmIChERFNELT51NC5kZHBmUGl4ZWxGb3JtYXQuZHdGbGFncyAmIEREUEZfUkdCKSApCiAgICAgICAgICAgIHBpeGVsZm9ybWF0ID0gUGl4ZWxGb3JtYXRfREQyV2luZUQzRCgmRERTRC0+dTQuZGRwZlBpeGVsRm9ybWF0KTsKICAgIH0KCiAgICByZXR1cm4gSVdpbmVEM0REZXZpY2VfRW51bURpc3BsYXlNb2RlcyhUaGlzLT53aW5lRDNERGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRmxhZ3MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBXaWR0aCwgSGVpZ2h0LCBwaXhlbGZvcm1hdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZjYnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJRGlyZWN0RHJhd0ltcGxfRW51bURpc3BsYXlNb2Rlc0NCKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6RXZhbHVhdGVNb2RlCiAqCiAqIFVzZWQgd2l0aCBJRGlyZWN0RHJhdzc6OlN0YXJ0TW9kZVRlc3QgdG8gdGVzdCB2aWRlbyBtb2Rlcy4KICogRXZhbHVhdGVNb2RlIGlzIHVzZWQgdG8gcGFzcyBvciBmYWlsIGEgbW9kZSwgYW5kIGNvbnRpbnVlIHdpdGggdGhlIG5leHQKICogbW9kZQogKgogKiBQYXJhbXM6CiAqICBGbGFnczogRERFTV9NT0RFUEFTU0VEIG9yIERERU1fTU9ERUZBSUxFRAogKiAgVGltZW91dDogUmV0dXJucyB0aGUgYW1vdW50IG9mIHNlY29uZHMgbGVmdCBiZWZvcmUgdGhlIG1vZGUgd291bGQgaGF2ZQogKiAgICAgICAgICAgYmVlbiBmYWlsZWQgYXV0b21hdGljYWxseQogKgogKiBSZXR1cm5zOgogKiAgVGhpcyBpbXBsZW1lbnRhdGlvbiBhbHdheXMgRERfT0ssIGJlY2F1c2UgaXQncyBhIHN0dWIKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0V2YWx1YXRlTW9kZShJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIEZsYWdzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEICpUaW1lb3V0KQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgRklYTUUoIiglcCktPiglbGQsJXApOiBTdHViIVxuIiwgVGhpcywgRmxhZ3MsIFRpbWVvdXQpOwoKICAgIC8qIFdoZW4gaW1wbGVtZW50aW5nIHRoaXMsIGltcGxlbWVudCBpdCBpbiBXaW5lRDNEICovCgogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXREZXZpY2VJZGVudGlmaWVyCiAqCiAqIFJldHVybnMgdGhlIGRldmljZSBpZGVudGlmaWVyLCB3aGljaCBnaXZlcyBpbmZvcm1hdGlvbiBhYm91dCB0aGUgZHJpdmVyCiAqIE91ciBkZXZpY2UgaWRlbnRpZmllciBpcyBkZWZpbmVkIGF0IHRoZSBiZWdpbm5pbmcgb2YgdGhpcyBmaWxlLgogKgogKiBQYXJhbXM6CiAqICBERERJOiBBZGRyZXNzIGZvciB0aGUgcmV0dXJuZWQgc3RydWN0dXJlCiAqICBGbGFnczogQ2FuIGJlIERER0RJX0dFVEhPU1RJREVOVElGSUVSCiAqCiAqIFJldHVybnM6CiAqICBPbiBzdWNjZXNzIGl0IHJldHVybnMgRERfT0sKICogIERERVJSX0lOVkFMSURQQVJBTVMgaWYgRERESSBpcyBOVUxMCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9HZXREZXZpY2VJZGVudGlmaWVyKElEaXJlY3REcmF3NyAqaWZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEREREVWSUNFSURFTlRJRklFUjIgKkREREksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIEZsYWdzKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgVFJBQ0UoIiglcCktPiglcCwlMDhseClcbiIsIFRoaXMsIEREREksIEZsYWdzKTsKCiAgICBpZighRERESSkKICAgICAgICByZXR1cm4gRERFUlJfSU5WQUxJRFBBUkFNUzsKCiAgICAvKiBUaGUgRERHRElfR0VUSE9TVElERU5USUZJRVIgcmV0dXJucyB0aGUgaW5mb3JtYXRpb24gYWJvdXQgdGhlIDJECiAgICAgKiBob3N0IGFkYXB0ZXIsIGlmIHRoZXJlJ3MgYSBzZWNvbmRhcnkgM0QgYWRhcHRlci4gVGhpcyBkb2Vzbid0IGFwcGx5CiAgICAgKiB0byBhbnkgbW9kZXJuIGhhcmR3YXJlLCBub3IgaXMgaXQgaW50ZXJlc3RpbmcgZm9yIFdpbmUsIHNvIGlnbm9yZSBpdAogICAgICovCgogICAgKkREREkgPSBkZXZpY2VpZGVudGlmaWVyOwogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXc3OjpHZXRTdXJmYWNlRnJvbURDCiAqCiAqIFJldHVybnMgdGhlIFN1cmZhY2UgZm9yIGEgR0RJIGRldmljZSBjb250ZXh0IGhhbmRsZS4KICogSXMgdGhpcyByZWxhdGVkIHRvIElEaXJlY3REcmF3U3VyZmFjZTo6R2V0REMgPz8/CiAqCiAqIFBhcmFtczoKICogIGhkYzogaGRjIHRvIHJldHVybiB0aGUgc3VyZmFjZSBmb3IKICogIFN1cmZhY2U6IEFkZHJlc3MgdG8gd3JpdGUgdGhlIHN1cmZhY2UgcG9pbnRlciB0bwogKgogKiBSZXR1cm5zOgogKiAgQWx3YXlzIHJldHVybnMgRERfT0sgYmVjYXVzZSBpdCdzIGEgc3R1YgogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfR2V0U3VyZmFjZUZyb21EQyhJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBIREMgaGRjLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJRGlyZWN0RHJhd1N1cmZhY2U3ICoqU3VyZmFjZSkKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIEZJWE1FKCIoJXApLT4oJXAsJXApOiBTdHViIVxuIiwgVGhpcywgaGRjLCBTdXJmYWNlKTsKCiAgICAvKiBJbXBsZW1lbnRhdGlvbiBpZGVhIGlmIG5lZWRlZDogTG9vcCB0aHJvdWdoIGFsbCBzdXJmYWNlcyBhbmQgY29tcGFyZQogICAgICogdGhlaXIgaGRjIHdpdGggaGRjLiBJbXBsZW1lbnQgaXQgaW4gV2luZUQzRCEgKi8KICAgIHJldHVybiBEREVSUl9OT1RGT1VORDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6UmVzdG9yZUFsbFN1cmZhY2VzCiAqCiAqIENhbGxzIHRoZSByZXN0b3JlIG1ldGhvZCBvZiBhbGwgc3VyZmFjZXMKICoKICogUGFyYW1zOgogKgogKiBSZXR1cm5zOgogKiAgQWx3YXlzIHJldHVybnMgRERfT0sgYmVjYXVzZSBpdCdzIGEgc3R1YgogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfUmVzdG9yZUFsbFN1cmZhY2VzKElEaXJlY3REcmF3NyAqaWZhY2UpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBGSVhNRSgiKCVwKTogU3R1YlxuIiwgVGhpcyk7CgogICAgLyogVGhpcyBpc24ndCBoYXJkIHRvIGltcGxlbWVudDogRW51bWVyYXRlIGFsbCBXaW5lRDNEIHN1cmZhY2VzLAogICAgICogZ2V0IHRoZWlyIHBhcmVudCBhbmQgY2FsbCB0aGVpciByZXN0b3JlIG1ldGhvZC4gRG8gbm90IGltcGxlbWVudAogICAgICogaXQgaW4gV2luZUQzRCwgYXMgcmVzdG9yaW5nIGEgc3VyZmFjZSBtZWFucyByZS1jcmVhdGluZyB0aGUKICAgICAqIFdpbmVEM0REU3VyZmFjZQogICAgICovCiAgICByZXR1cm4gRERfT0s7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhdzc6OlN0YXJ0TW9kZVRlc3QKICoKICogVGVzdHMgdGhlIHNwZWNpZmllZCB2aWRlbyBtb2RlcyB0byB1cGRhdGUgdGhlIHN5c3RlbSByZWdpc3RyeSB3aXRoCiAqIHJlZnJlc2ggcmF0ZSBpbmZvcm1hdGlvbi4gU3RhcnRNb2RlVGVzdCBzdGFydHMgdGhlIG1vZGUgdGVzdCwKICogRXZhbHVhdGVNb2RlIGlzIHVzZWQgdG8gZmFpbCBvciBwYXNzIGEgbW9kZS4gSWYgRXZhbHVhdGVNb2RlCiAqIGlzbid0IGNhbGxlZCB3aXRoaW4gMTUgc2Vjb25kcywgdGhlIG1vZGUgaXMgZmFpbGVkIGF1dG9tYXRpY2FsbHkKICoKICogQXMgcmVmcmVzaCByYXRlcyBhcmUgaGFuZGxlZCBieSB0aGUgWCBzZXJ2ZXIsIEkgZG9uJ3QgdGhpbmsgdGhpcwogKiBNZXRob2QgaXMgaW1wb3J0YW50CiAqCiAqIFBhcmFtczoKICogIE1vZGVzOiBBbiBhcnJheSBvZiBtb2RlIHNwZWNpZmljYXRpb25zCiAqICBOdW1Nb2RlczogVGhlIG51bWJlciBvZiBtb2RlcyBpbiBNb2RlcwogKiAgRmxhZ3M6IFNvbWUgZmxhZ3MuLi4KICoKICogUmV0dXJuczoKICogIFJldHVybnMgRERFUlJfVEVTVEZJTklTSEVEIGlmIGZsYWdzIGNvbnRhaW5zIEREU01UX0lTVEVTVFJFUVVJUkVELAogKiAgaWYgbm8gbW9kZXMgYXJlIHBhc3NlZCwgRERFUlJfSU5WQUxJRFBBUkFNUyBpcyByZXR1cm5lZCwKICogIG90aGVyd2lzZSBERF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfU3RhcnRNb2RlVGVzdChJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBTSVpFICpNb2RlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgTnVtTW9kZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIEZsYWdzKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgaWZhY2UpOwogICAgV0FSTigiKCVwKS0+KCVwLCAlbGQsICVseCk6IFNlbWktU3R1YiwgbW9zdCBsaWtlbHkgaGFybWxlc3NcbiIsIFRoaXMsIE1vZGVzLCBOdW1Nb2RlcywgRmxhZ3MpOwoKICAgIC8qIFRoaXMgbG9va3Mgc2FuZSAqLwogICAgaWYoICghTW9kZXMpIHx8IChOdW1Nb2RlcyA9PSAwKSApIHJldHVybiBEREVSUl9JTlZBTElEUEFSQU1TOwoKICAgIC8qIEREU01UX0lTVEVTVFJFUVVJUkVEIGFza3MgaWYgYSBtb2RlIHRlc3QgaXMgbmVjZXNzYXJ5LgogICAgICogQXMgaXQgaXMgbm90LCBEREVSUl9URVNURklOSVNIRUQgaXMgcmV0dXJuZWQKICAgICAqIChob3BlZnVsbHkgdGhhdCdzIGNvcnJlY3QKICAgICAqCiAgICBpZihGbGFncyAmIEREU01UX0lTVEVTVFJFUVVJUkVEKSByZXR1cm4gRERFUlJfVEVTVEZJTklTSEVEOwogICAgICogd2VsbCwgdGhhdCB2YWx1ZSBkb2Vzbid0ICh5ZXQpIGV4aXN0IGluIHRoZSB3aW5lIGhlYWRlcnMsIHNvIGlnbm9yZSBpdAogICAgICovCgogICAgcmV0dXJuIEREX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXdJbXBsX1JlY3JlYXRlU3VyZmFjZXNDYWxsYmFjawogKgogKiBFbnVtZXJhdGlvbiBjYWxsYmFjayBmb3IgSURpcmVjdERyYXdJbXBsX1JlY3JlYXRlQWxsU3VyZmFjZXMuCiAqIEl0IHJlLXJlY3JlYXRlcyB0aGUgV2luZUQzRFN1cmZhY2UuIEl0J3MgcHJldHR5IHN0cmFpZ2h0Zm9yd2FyZAogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCkhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9SZWNyZWF0ZVN1cmZhY2VzQ2FsbGJhY2soSURpcmVjdERyYXdTdXJmYWNlNyAqc3VyZiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBERFNVUkZBQ0VERVNDMiAqZGVzYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2b2lkICpDb250ZXh0KQp7CiAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICpzdXJmSW1wbCA9IElDT01fT0JKRUNUKElEaXJlY3REcmF3U3VyZmFjZUltcGwsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIElEaXJlY3REcmF3U3VyZmFjZTcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN1cmYpOwogICAgSURpcmVjdERyYXdJbXBsICpUaGlzID0gc3VyZkltcGwtPmRkcmF3OwogICAgSVVua25vd24gKlBhcmVudDsKICAgIElQYXJlbnRJbXBsICpwYXJJbXBsID0gTlVMTDsKICAgIElXaW5lRDNEU3VyZmFjZSAqd2luZUQzRFN1cmZhY2U7CiAgICBIUkVTVUxUIGhyOwogICAgdm9pZCAqdG1wOwoKICAgIFdJTkVEM0RTVVJGQUNFX0RFU0MgICAgIERlc2M7CiAgICBXSU5FRDNERk9STUFUICAgICAgICAgICBGb3JtYXQ7CiAgICBXSU5FRDNEUkVTT1VSQ0VUWVBFICAgICBUeXBlOwogICAgRFdPUkQgICAgICAgICAgICAgICAgICAgVXNhZ2U7CiAgICBXSU5FRDNEUE9PTCAgICAgICAgICAgICBQb29sOwogICAgVUlOVCAgICAgICAgICAgICAgICAgICAgU2l6ZTsKCiAgICBXSU5FRDNETVVMVElTQU1QTEVfVFlQRSBNdWx0aVNhbXBsZVR5cGU7CiAgICBEV09SRCAgICAgICAgICAgICAgICAgICBNdWx0aVNhbXBsZVF1YWxpdHk7CiAgICBVSU5UICAgICAgICAgICAgICAgICAgICBXaWR0aDsKICAgIFVJTlQgICAgICAgICAgICAgICAgICAgIEhlaWdodDsKCiAgICBUUkFDRSgiKCVwKTogRW51bWVyYXRlZCBTdXJmYWNlICVwXG4iLCBUaGlzLCBzdXJmSW1wbCk7CgogICAgLyogRm9yIHRoZSBlbnVtZXJhdGlvbiAqLwogICAgSURpcmVjdERyYXdTdXJmYWNlN19SZWxlYXNlKHN1cmYpOwoKICAgIGlmKHN1cmZJbXBsLT5JbXBsVHlwZSA9PSBUaGlzLT5JbXBsVHlwZSkgcmV0dXJuIERERU5VTVJFVF9PSzsgLyogQ29udGludWUgKi8KCiAgICAvKiBHZXQgdGhlIG9iamVjdHMgKi8KICAgIHdpbmVEM0RTdXJmYWNlID0gc3VyZkltcGwtPldpbmVEM0RTdXJmYWNlOwogICAgSVdpbmVEM0RTdXJmYWNlX0dldFBhcmVudCh3aW5lRDNEU3VyZmFjZSwgJlBhcmVudCk7CiAgICBJVW5rbm93bl9SZWxlYXNlKFBhcmVudCk7IC8qIEZvciB0aGUgZ2V0UGFyZW50ICovCgogICAgLyogSXMgdGhlIHBhcmVudCBhbiBJUGFyZW50IGludGVyZmFjZT8gKi8KICAgIGlmKElVbmtub3duX1F1ZXJ5SW50ZXJmYWNlKFBhcmVudCwgJklJRF9JUGFyZW50LCAmdG1wKSA9PSBTX09LKQogICAgewogICAgICAgIC8qIEl0IGlzIGEgSVBhcmVudCBpbnRlcmZhY2UhICovCiAgICAgICAgSVVua25vd25fUmVsZWFzZShQYXJlbnQpOyAvKiBGb3IgdGhlIFF1ZXJ5SW50ZXJmYWNlICovCiAgICAgICAgcGFySW1wbCA9IElDT01fT0JKRUNUKElQYXJlbnRJbXBsLCBJUGFyZW50LCBQYXJlbnQpOwogICAgICAgIC8qIFJlbGVhc2UgdGhlIHJlZmVyZW5jZSB0aGUgcGFyZW50IGludGVyZmFjZSBpcyBob2xkaW5nICovCiAgICAgICAgSVdpbmVEM0RTdXJmYWNlX1JlbGVhc2Uod2luZUQzRFN1cmZhY2UpOwogICAgfQoKCiAgICAvKiBHZXQgdGhlIHN1cmZhY2UgcHJvcGVydGllcyAqLwogICAgRGVzYy5Gb3JtYXQgPSAmRm9ybWF0OwogICAgRGVzYy5UeXBlID0gJlR5cGU7CiAgICBEZXNjLlVzYWdlID0gJlVzYWdlOwogICAgRGVzYy5Qb29sID0gJlBvb2w7CiAgICBEZXNjLlNpemUgPSAmU2l6ZTsKICAgIERlc2MuTXVsdGlTYW1wbGVUeXBlID0gJk11bHRpU2FtcGxlVHlwZTsKICAgIERlc2MuTXVsdGlTYW1wbGVRdWFsaXR5ID0gJk11bHRpU2FtcGxlUXVhbGl0eTsKICAgIERlc2MuV2lkdGggPSAmV2lkdGg7CiAgICBEZXNjLkhlaWdodCA9ICZIZWlnaHQ7CgogICAgaHIgPSBJV2luZUQzRFN1cmZhY2VfR2V0RGVzYyh3aW5lRDNEU3VyZmFjZSwgJkRlc2MpOwogICAgaWYoaHIgIT0gRDNEX09LKSByZXR1cm4gaHI7CgogICAgLyogQ3JlYXRlIHRoZSBuZXcgc3VyZmFjZSAqLwogICAgaHIgPSBJV2luZUQzRERldmljZV9DcmVhdGVTdXJmYWNlKFRoaXMtPndpbmVEM0REZXZpY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV2lkdGgsIEhlaWdodCwgRm9ybWF0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRSVUUgLyogTG9ja2FibGUgKi8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRkFMU0UgLyogRGlzY2FyZCAqLywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdXJmSW1wbC0+bWlwbWFwX2xldmVsLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZzdXJmSW1wbC0+V2luZUQzRFN1cmZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVc2FnZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQb29sLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIE11bHRpU2FtcGxlVHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBNdWx0aVNhbXBsZVF1YWxpdHksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCAvKiBTaGFyZWRIYW5kbGUgKi8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVGhpcy0+SW1wbFR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyZW50KTsKCiAgICBpZihociAhPSBEM0RfT0spCiAgICAgICAgcmV0dXJuIGhyOwoKICAgIC8qIFVwZGF0ZSB0aGUgSVBhcmVudCBpZiBpdCBleGlzdHMgKi8KICAgIGlmKHBhckltcGwpCiAgICB7CiAgICAgICAgcGFySW1wbC0+Y2hpbGQgPSAoSVVua25vd24gKikgc3VyZkltcGwtPldpbmVEM0RTdXJmYWNlOwogICAgICAgIC8qIEFkZCBhIHJlZmVyZW5jZSBmb3IgdGhlIElQYXJlbnQgKi8KICAgICAgICBJV2luZUQzRFN1cmZhY2VfQWRkUmVmKHN1cmZJbXBsLT5XaW5lRDNEU3VyZmFjZSk7CiAgICB9CiAgICAvKiBUT0RPOiBDb3B5IHRoZSBzdXJmYWNlIGNvbnRlbnQsIGV4Y2VwdCBmb3IgcmVuZGVyIHRhcmdldHMgKi8KCiAgICBpZihJV2luZUQzRFN1cmZhY2VfUmVsZWFzZSh3aW5lRDNEU3VyZmFjZSkgPT0gMCkKICAgICAgICBUUkFDRSgiU3VyZmFjZSByZWxlYXNlZCBzdWNjZXNzZnVsLCBuZXh0IHN1cmZhY2VcbiIpOwogICAgZWxzZQogICAgICAgIEVSUigiU29tZXRoaW5nJ3Mgc3RpbGwgaG9sZGluZyB0aGUgb2xkIFdpbmVEM0RTdXJmYWNlXG4iKTsKCiAgICBzdXJmSW1wbC0+SW1wbFR5cGUgPSBUaGlzLT5JbXBsVHlwZTsKCiAgICByZXR1cm4gRERFTlVNUkVUX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURpcmVjdERyYXdJbXBsX1JlY3JlYXRlQWxsU3VyZmFjZXMKICoKICogQSBmdW5jdGlvbiwgdGhhdCBjb252ZXJ0cyBhbGwgd2luZUQzRFN1cmZhY2VzIHRvIHRoZSBuZXcgaW1wbGVtZW50YXRpb24gdHlwZQogKiBJdCBlbnVtZXJhdGVzIGFsbCBzdXJmYWNlcyB3aXRoIElXaW5lRDNERGV2aWNlOjpFbnVtU3VyZmFjZXMsIGNyZWF0ZXMgYQogKiBuZXcgV2luZUQzRFN1cmZhY2UsIGNvcGllcyB0aGUgY29udGVudCBhbmQgcmVsZWFzZXMgdGhlIG9sZCBzdXJmYWNlCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIEhSRVNVTFQKSURpcmVjdERyYXdJbXBsX1JlY3JlYXRlQWxsU3VyZmFjZXMoSURpcmVjdERyYXdJbXBsICpUaGlzKQp7CiAgICBERFNVUkZBQ0VERVNDMiBkZXNjOwogICAgVFJBQ0UoIiglcCk6IFN3aXRjaCB0byBpbXBsZW1lbnRhdGlvbiAlZFxuIiwgVGhpcywgVGhpcy0+SW1wbFR5cGUpOwoKICAgIGlmKFRoaXMtPkltcGxUeXBlICE9IFNVUkZBQ0VfT1BFTkdMICYmIFRoaXMtPmQzZF9pbml0aWFsaXplZCkKICAgIHsKICAgICAgICAvKiBTaG91bGQgaGFwcGVuIGFsbW9zdCBuZXZlciAqLwogICAgICAgIEZJWE1FKCIoJXApIFN3aXRjaGluZyB0byBub24tb3BlbmdsIHN1cmZhY2VzIHdpdGggZDNkIHN0YXJ0ZWQuIElzIHRoaXMgYSBidWc/XG4iLCBUaGlzKTsKICAgICAgICAvKiBTaHV0ZG93biBkM2QgKi8KICAgICAgICBJV2luZUQzRERldmljZV9VbmluaXQzRChUaGlzLT53aW5lRDNERGV2aWNlKTsKICAgIH0KICAgIC8qIENvbnRyYXJ5OiBEM0Qgc3RhcnRpbmcgaXMgaGFuZGxlZCBieSB0aGUgY2FsbGVyLCBiZWNhdXNlIGl0IGtub3dzIHRoZSByZW5kZXIgdGFyZ2V0ICovCgogICAgbWVtc2V0KCZkZXNjLCAwLCBzaXplb2YoZGVzYykpOwogICAgZGVzYy5kd1NpemUgPSBzaXplb2YoZGVzYyk7CgogICAgcmV0dXJuIElEaXJlY3REcmF3N19FbnVtU3VyZmFjZXMoSUNPTV9JTlRFUkZBQ0UoVGhpcywgSURpcmVjdERyYXc3KSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmZGVzYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFRoaXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJRGlyZWN0RHJhd0ltcGxfUmVjcmVhdGVTdXJmYWNlc0NhbGxiYWNrKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIEQzRDdDQl9DcmVhdGVTdXJmYWNlCiAqCiAqIENhbGxiYWNrIGZ1bmN0aW9uIGZvciBJRGlyZWN0M0REZXZpY2VfQ3JlYXRlVGV4dHVyZS4gSXQgc2VhcmNoZXMgZm9yIHRoZQogKiBjb3JyZWN0IG1pcG1hcCBzdWJsZXZlbCwgYW5kIHJldHVybnMgaXQgdG8gV2luZUQzRC4KICogVGhlIHN1cmZhY2VzIGFyZSBjcmVhdGVkIGFscmVhZHkgYnkgSURpcmVjdERyYXc3OjpDcmVhdGVTdXJmYWNlCiAqCiAqIFBhcmFtczoKICogIFdpdGgsIEhlaWdodDogV2l0aCBhbmQgaGVpZ2h0IG9mIHRoZSBzdXJmYWNlCiAqICBGb3JtYXQ6IFRoZSByZXF1ZXN0ZWQgZm9ybWF0CiAqICBVc2FnZSwgUG9vbDogRDNEVVNBR0UgYW5kIEQzRFBPT0wgb2YgdGhlIHN1cmZhY2UKICogIGxldmVsOiBUaGUgbWlwbWFwIGxldmVsCiAqICBTdXJmYWNlOiBQb2ludGVyIHRvIHBhc3MgdGhlIGNyZWF0ZWQgc3VyZmFjZSBiYWNrIGF0CiAqICBTaGFyZWRIYW5kbGU6IE5VTEwKICoKICogUmV0dXJuczoKICogIEQzRF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpEM0Q3Q0JfQ3JlYXRlU3VyZmFjZShJVW5rbm93biAqZGV2aWNlLAogICAgICAgICAgICAgICAgICAgICBVSU5UIFdpZHRoLCBVSU5UIEhlaWdodCwKICAgICAgICAgICAgICAgICAgICAgV0lORUQzREZPUk1BVCBGb3JtYXQsCiAgICAgICAgICAgICAgICAgICAgIERXT1JEIFVzYWdlLCBXSU5FRDNEUE9PTCBQb29sLCBVSU5UIGxldmVsLAogICAgICAgICAgICAgICAgICAgICBJV2luZUQzRFN1cmZhY2UgKipTdXJmYWNlLAogICAgICAgICAgICAgICAgICAgICBIQU5ETEUgKlNoYXJlZEhhbmRsZSkKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGRldmljZSk7CiAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICpzdXJmID0gVGhpcy0+dGV4X3Jvb3Q7CiAgICBpbnQgaTsKICAgIFRSQUNFKCIoJXApIGNhbGwgYmFjay4gc3VyZj0lcFxuIiwgZGV2aWNlLCBzdXJmKTsKCiAgICAvKiBGaW5kIHRoZSB3YW50ZWQgbWlwbWFwLiBUaGVyZSBhcmUgZW5vdWdoIG1pcG1hcHMgaW4gdGhlIGNoYWluICovCiAgICBmb3IoaSA9IDA7IGkgPCBsZXZlbDsgaSsrKQogICAgICAgIHN1cmYgPSBzdXJmLT5uZXh0X2NvbXBsZXg7CgogICAgLyogUmV0dXJuIHRoZSBzdXJmYWNlICovCiAgICAqU3VyZmFjZSA9IHN1cmYtPldpbmVEM0RTdXJmYWNlOwoKICAgIFRSQUNFKCJSZXR1cm5pbmcgd2luZUQzRFN1cmZhY2UgJXAsIGl0IGJlbG9uZ3MgdG8gc3VyZmFjZSAlcFxuIiwgKlN1cmZhY2UsIHN1cmYpOwogICAgcmV0dXJuIEQzRF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3SW1wbF9DcmVhdGVOZXdTdXJmYWNlCiAqCiAqIEEgaGVscGVyIGZ1bmN0aW9uIGZvciBJRGlyZWN0RHJhdzc6OkNyZWF0ZVN1cmZhY2UuIEl0IGNyZWF0ZXMgYSBuZXcgc3VyZmFjZQogKiB3aXRoIHRoZSBwYXNzZWQgcGFyYW1ldGVycy4KICoKICogUGFyYW1zOgogKiAgRERTRDogRGVzY3JpcHRpb24gb2YgdGhlIHN1cmZhY2UgdG8gY3JlYXRlCiAqICBTdXJmOiBBZGRyZXNzIHRvIHN0b3JlIHRoZSBpbnRlcmZhY2UgcG9pbnRlciBhdAogKgogKiBSZXR1cm5zOgogKiAgRERfT0sgb24gc3VjY2VzcwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfQ3JlYXRlTmV3U3VyZmFjZShJRGlyZWN0RHJhd0ltcGwgKlRoaXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEREU1VSRkFDRURFU0MyICpwRERTRCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSURpcmVjdERyYXdTdXJmYWNlSW1wbCAqKnBwU3VyZiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUlOVCBsZXZlbCkKewogICAgSFJFU1VMVCBocjsKICAgIFVJTlQgV2lkdGggPSAwLCBIZWlnaHQgPSAwOwogICAgV0lORUQzREZPUk1BVCBGb3JtYXQgPSBXSU5FRDNERk1UX1VOS05PV047CiAgICBXSU5FRDNEUkVTT1VSQ0VUWVBFIFJlc1R5cGUgPSBXSU5FRDNEUlRZUEVfU1VSRkFDRTsKICAgIERXT1JEIFVzYWdlID0gMDsKICAgIFdJTkVEM0RTVVJGVFlQRSBJbXBsVHlwZSA9IFRoaXMtPkltcGxUeXBlOwogICAgV0lORUQzRFNVUkZBQ0VfREVTQyBEZXNjOwogICAgSVVua25vd24gKlBhcmVudDsKICAgIElQYXJlbnRJbXBsICpwYXJJbXBsID0gTlVMTDsKICAgIFdJTkVEM0RQT09MIFBvb2wgPSBXSU5FRDNEUE9PTF9ERUZBVUxUOwoKICAgIC8qIER1bW1pZXMgZm9yIEdldERlc2MgKi8KICAgIFdJTkVEM0RQT09MIGR1bW15X2QzZHBvb2w7CiAgICBXSU5FRDNETVVMVElTQU1QTEVfVFlQRSBkdW1teV9tc3Q7CiAgICBVSU5UIGR1bW15X3VpbnQ7CiAgICBEV09SRCBkdW1teV9kd29yZDsKCiAgICBpZiAoVFJBQ0VfT04oZGRyYXcpKQogICAgewogICAgICAgIFRSQUNFKCIgKCVwKSBSZXF1ZXN0aW5nIHN1cmZhY2UgZGVzYyA6XG4iLCBUaGlzKTsKICAgICAgICBERFJBV19kdW1wX3N1cmZhY2VfZGVzYyhwRERTRCk7CiAgICB9CgogICAgLyogU2VsZWN0IHRoZSBzdXJmYWNlIHR5cGUsIGlmIGl0IHdhc24ndCBjaG9vc2VuIHlldCAqLwogICAgaWYoSW1wbFR5cGUgPT0gU1VSRkFDRV9VTktOT1dOKQogICAgewogICAgICAgIC8qIFVzZSBHTCBTdXJmYWNlcyBpZiBhIEQzRERFVklDRSBTdXJmYWNlIGlzIHJlcXVlc3RlZCAqLwogICAgICAgIGlmKHBERFNELT5kZHNDYXBzLmR3Q2FwcyAmIEREU0NBUFNfM0RERVZJQ0UpCiAgICAgICAgewogICAgICAgICAgICBUUkFDRSgiKCVwKSBDaG9vc2luZyBHTCBzdXJmYWNlcyBiZWNhdXNlIGEgM0RERVZJQ0UgU3VyZmFjZSB3YXMgcmVxdWVzdGVkXG4iLCBUaGlzKTsKICAgICAgICAgICAgSW1wbFR5cGUgPSBTVVJGQUNFX09QRU5HTDsKICAgICAgICB9CgogICAgICAgIC8qIE90aGVyd2lzZSB1c2UgR0RJIHN1cmZhY2VzIGZvciBub3cgKi8KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICBUUkFDRSgiKCVwKSBDaG9vc2luZyBHREkgc3VyZmFjZXMgZm9yIDJEIHJlbmRlcmluZ1xuIiwgVGhpcyk7CiAgICAgICAgICAgIEltcGxUeXBlID0gU1VSRkFDRV9HREk7CiAgICAgICAgfQoKICAgICAgICAvKiBQb2xpY3kgaWYgYWxsIHN1cmZhY2UgaW1wbGVtZW50YXRpb25zIGFyZSBhdmFpbGFibGU6CiAgICAgICAgICogRmlyc3QsIGNoZWNrIGlmIGEgZGVmYXVsdCB0eXBlIHdhcyBzZXQgd2l0aCB3aW5lY2ZnLiBJZiBub3QsCiAgICAgICAgICogdHJ5IFhyZW5kZXIgc3VyZmFjZXMsIGFuZCB1c2UgdGhlbSBpZiB0aGV5IHdvcmsuIE5leHQsIGNoZWNrIGlmCiAgICAgICAgICogYWNjZWxlcmF0ZWQgT3BlbkdMIGlzIGF2YWlsYWJsZSwgYW5kIHVzZSBHTCBzdXJmYWNlcyBpbiB0aGlzCiAgICAgICAgICogY2FzZS4gSWYgYWxsIGVsc2UgZmFpbHMsIHVzZSBHREkgc3VyZmFjZXMuIElmIGEgM0RERVZJQ0Ugc3VyZmFjZQogICAgICAgICAqIHdhcyBjcmVhdGVkLCBhbHdheXMgdXNlIE9wZW5HTCBzdXJmYWNlcy4KICAgICAgICAgKgogICAgICAgICAqIChOb3RlOiBYcmVuZGVyIHN1cmZhY2VzIGFyZSBub3QgaW1wbGVtZW50ZWQgZm9yIG5vdywgdGhlCiAgICAgICAgICogdW5hY2NlbGVyYXRlZCBpbXBsZW1lbnRhdGlvbiB1c2VzIEdESSB0byByZW5kZXIgaW4gU29mdHdhcmUpCiAgICAgICAgICovCgogICAgICAgIC8qIFN0b3JlIHRoZSB0eXBlLiBJZiBpdCBuZWVkcyB0byBiZSBjaGFuZ2VkLCBhbGwgV2luZUQzRFN1cmZhY2VzIGhhdmUgdG8KICAgICAgICAgKiBiZSByZS1jcmVhdGVkLiBUaGlzIGNvdWxkIGJlIGRvbmUgd2l0aCBJRGlyZWN0RHJhd1N1cmZhY2U3OjpSZXN0b3JlCiAgICAgICAgICovCiAgICAgICAgVGhpcy0+SW1wbFR5cGUgPSBJbXBsVHlwZTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICAgaWYoKHBERFNELT5kZHNDYXBzLmR3Q2FwcyAmIEREU0NBUFNfM0RERVZJQ0UgKSAmJiAKICAgICAgICAgICAgKFRoaXMtPkltcGxUeXBlICE9IFNVUkZBQ0VfT1BFTkdMICkgJiYgRGVmYXVsdFN1cmZhY2VUeXBlID09IFNVUkZBQ0VfVU5LTk9XTikKICAgICAgICB7CiAgICAgICAgICAgIC8qIFdlIGhhdmUgdG8gY2hhbmdlIHRvIE9wZW5HTCwKICAgICAgICAgICAgICogYW5kIHJlLWNyZWF0ZSBhbGwgV2luZUQzRFN1cmZhY2VzCiAgICAgICAgICAgICAqLwogICAgICAgICAgICBJbXBsVHlwZSA9IFNVUkZBQ0VfT1BFTkdMOwogICAgICAgICAgICBUaGlzLT5JbXBsVHlwZSA9IEltcGxUeXBlOwogICAgICAgICAgICBUUkFDRSgiKCVwKSBSZS1jcmVhdGluZyBhbGwgc3VyZmFjZXNcbiIsIFRoaXMpOwogICAgICAgICAgICBJRGlyZWN0RHJhd0ltcGxfUmVjcmVhdGVBbGxTdXJmYWNlcyhUaGlzKTsKICAgICAgICAgICAgVFJBQ0UoIiglcCkgRG9uZSByZWNyZWF0aW5nIGFsbCBzdXJmYWNlc1xuIiwgVGhpcyk7CiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoVGhpcy0+SW1wbFR5cGUgIT0gU1VSRkFDRV9PUEVOR0wpCiAgICAgICAgewogICAgICAgICAgICBXQVJOKCJUaGUgYXBwbGljYXRpb24gcmVxdWVzdHMgYSAzRCBjYXBhYmxlIHN1cmZhY2UsIGJ1dCBhIG5vbi1vcGVuZ2wgc3VyZmFjZSB3YXMgc2V0IGluIHRoZSByZWdpc3RyeVxuIik7CiAgICAgICAgICAgIC8qIERvIG5vdCBmYWlsIHN1cmZhY2UgY3JlYXRpb24sIG9ubHkgZmFpbCAzRCBkZXZpY2UgY3JlYXRpb24gKi8KICAgICAgICB9CiAgICB9CgogICAgLyogR2V0IHRoZSBjb3JyZWN0IHdpbmVkM2QgdXNhZ2UgKi8KICAgIGlmIChwRERTRC0+ZGRzQ2Fwcy5kd0NhcHMgJiAoRERTQ0FQU19QUklNQVJZU1VSRkFDRSB8CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEREU0NBUFNfQkFDS0JVRkZFUiAgICAgfAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBERFNDQVBTXzNEREVWSUNFICAgICAgICkgKQogICAgewogICAgICAgIFVzYWdlIHw9IFdJTkVEM0RVU0FHRV9SRU5ERVJUQVJHRVQ7CgogICAgICAgIHBERFNELT5kZHNDYXBzLmR3Q2FwcyB8PSBERFNDQVBTX1ZJREVPTUVNT1JZIHwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRERTQ0FQU19WSVNJQkxFICAgICB8CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEREU0NBUFNfTE9DQUxWSURNRU07CiAgICB9CiAgICBpZihUaGlzLT5kZXB0aHN0ZW5jaWwpCiAgICB7CiAgICAgICAgLyogVGhlIGRlcHRoIHN0ZW5jaWwgY3JlYXRpb24gY2FsbGJhY2sgc2V0cyB0aGlzIGZsYWcuCiAgICAgICAgICogU2V0IHRoZSBXaW5lRDNEIHVzYWdlIHRvIGxldCBpdCBrbm93IHRoYXQgaXQncyBhIGRlcHRoCiAgICAgICAgICogU3RlbmNpbCBzdXJmYWNlLgogICAgICAgICAqLwogICAgICAgIFVzYWdlIHw9IFdJTkVEM0RVU0FHRV9ERVBUSFNURU5DSUw7CiAgICB9CiAgICBpZihwRERTRC0+ZGRzQ2Fwcy5kd0NhcHMgJiBERFNDQVBTX1NZU1RFTU1FTU9SWSkKICAgIHsKICAgICAgICBQb29sID0gV0lORUQzRFBPT0xfU1lTVEVNTUVNOwogICAgfQoKICAgIEZvcm1hdCA9IFBpeGVsRm9ybWF0X0REMldpbmVEM0QoJnBERFNELT51NC5kZHBmUGl4ZWxGb3JtYXQpOwogICAgaWYoRm9ybWF0ID09IFdJTkVEM0RGTVRfVU5LTk9XTikKICAgIHsKICAgICAgICBFUlIoIlVuc3VwcG9ydGVkIC8gVW5rbm93biBwaXhlbGZvcm1hdFxuIik7CiAgICAgICAgcmV0dXJuIERERVJSX0lOVkFMSURQSVhFTEZPUk1BVDsKICAgIH0KCiAgICAvKiBDcmVhdGUgdGhlIFN1cmZhY2Ugb2JqZWN0ICovCiAgICAqcHBTdXJmID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIEhFQVBfWkVST19NRU1PUlksIHNpemVvZihJRGlyZWN0RHJhd1N1cmZhY2VJbXBsKSk7CiAgICBpZighKnBwU3VyZikKICAgIHsKICAgICAgICBFUlIoIiglcCkgRXJyb3IgYWxsb2NhdGluZyBtZW1vcnkgZm9yIGEgc3VyZmFjZVxuIiwgVGhpcyk7CiAgICAgICAgcmV0dXJuIERERVJSX09VVE9GVklERU9NRU1PUlk7CiAgICB9CiAgICBJQ09NX0lOSVRfSU5URVJGQUNFKCpwcFN1cmYsIElEaXJlY3REcmF3U3VyZmFjZTcsIElEaXJlY3REcmF3U3VyZmFjZTdfVnRibCk7CiAgICBJQ09NX0lOSVRfSU5URVJGQUNFKCpwcFN1cmYsIElEaXJlY3REcmF3U3VyZmFjZTMsIElEaXJlY3REcmF3U3VyZmFjZTNfVnRibCk7CiAgICBJQ09NX0lOSVRfSU5URVJGQUNFKCpwcFN1cmYsIElEaXJlY3REcmF3R2FtbWFDb250cm9sLCBJRGlyZWN0RHJhd0dhbW1hQ29udHJvbF9WdGJsKTsKICAgIElDT01fSU5JVF9JTlRFUkZBQ0UoKnBwU3VyZiwgSURpcmVjdDNEVGV4dHVyZTIsIElEaXJlY3QzRFRleHR1cmUyX1Z0YmwpOwogICAgSUNPTV9JTklUX0lOVEVSRkFDRSgqcHBTdXJmLCBJRGlyZWN0M0RUZXh0dXJlLCBJRGlyZWN0M0RUZXh0dXJlMV9WdGJsKTsKICAgICgqcHBTdXJmKS0+cmVmID0gMTsKICAgICgqcHBTdXJmKS0+dmVyc2lvbiA9IDc7CiAgICAoKnBwU3VyZiktPmRkcmF3ID0gVGhpczsKICAgICgqcHBTdXJmKS0+c3VyZmFjZV9kZXNjLmR3U2l6ZSA9IHNpemVvZihERFNVUkZBQ0VERVNDMik7CiAgICAoKnBwU3VyZiktPnN1cmZhY2VfZGVzYy51NC5kZHBmUGl4ZWxGb3JtYXQuZHdTaXplID0gc2l6ZW9mKEREUElYRUxGT1JNQVQpOwogICAgRERfU1RSVUNUX0NPUFlfQllTSVpFKCYoKnBwU3VyZiktPnN1cmZhY2VfZGVzYywgcEREU0QpOwoKICAgIC8qIFN1cmZhY2UgYXR0YWNobWVudHMgKi8KICAgICgqcHBTdXJmKS0+bmV4dF9hdHRhY2hlZCA9IE5VTEw7CiAgICAoKnBwU3VyZiktPmZpcnN0X2F0dGFjaGVkID0gKnBwU3VyZjsKCiAgICAoKnBwU3VyZiktPm5leHRfY29tcGxleCA9IE5VTEw7CiAgICAoKnBwU3VyZiktPmZpcnN0X2NvbXBsZXggPSAqcHBTdXJmOwoKICAgIC8qIE5lZWRlZCB0byByZS1jcmVhdGUgdGhlIHN1cmZhY2Ugb24gYW4gaW1wbGVtZW50YXRpb24gY2hhbmdlICovCiAgICAoKnBwU3VyZiktPkltcGxUeXBlID0gSW1wbFR5cGU7CgogICAgLyogRm9yIEQzRERldmljZSBjcmVhdGlvbiAqLwogICAgKCpwcFN1cmYpLT5pc1JlbmRlclRhcmdldCA9IEZBTFNFOwoKICAgIC8qIEEgdHJhY2UgbWVzc2FnZSBmb3IgZGVidWdnaW5nICovCiAgICBUUkFDRSgiKCVwKSBDcmVhdGVkIElEaXJlY3REcmF3U3VyZmFjZSBpbXBsZW1lbnRhdGlvbiBzdHJ1Y3R1cmUgYXQgJXBcbiIsIFRoaXMsICpwcFN1cmYpOwoKICAgIGlmKHBERFNELT5kZHNDYXBzLmR3Q2FwcyAmICggRERTQ0FQU19QUklNQVJZU1VSRkFDRSB8IEREU0NBUFNfVEVYVFVSRSB8IEREU0NBUFNfM0RERVZJQ0UpICkKICAgIHsKICAgICAgICAvKiBSZW5kZXIgdGFyZ2V0cyBhbmQgdGV4dHVyZXMgbmVlZCBhIElQYXJlbnQgaW50ZXJmYWNlLAogICAgICAgICAqIGJlY2F1c2UgV2luZUQzRCB3aWxsIGRlc3Ryb3kgdGhlbSB3aGVuIHRoZSBzd2FwY2hhaW4KICAgICAgICAgKiBpcyByZWxlYXNlZAogICAgICAgICAqLwogICAgICAgIHBhckltcGwgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc2l6ZW9mKElQYXJlbnRJbXBsKSk7CiAgICAgICAgaWYoIXBhckltcGwpCiAgICAgICAgewogICAgICAgICAgICBFUlIoIk91dCBvZiBtZW1vcnkgd2hlbiBhbGxvY2F0aW5nIG1lbW9yeSBmb3IgYSBJUGFyZW50IGltcGxlbWVudGF0aW9uXG4iKTsKICAgICAgICAgICAgcmV0dXJuIERERVJSX09VVE9GTUVNT1JZOwogICAgICAgIH0KICAgICAgICBwYXJJbXBsLT5yZWYgPSAxOwogICAgICAgIElDT01fSU5JVF9JTlRFUkZBQ0UocGFySW1wbCwgSVBhcmVudCwgSVBhcmVudF9WdGJsKTsKICAgICAgICBQYXJlbnQgPSAoSVVua25vd24gKikgSUNPTV9JTlRFUkZBQ0UocGFySW1wbCwgSVBhcmVudCk7CiAgICAgICAgVFJBQ0UoIlVzaW5nIElQYXJlbnQgaW50ZXJmYWNlICVwIGFzIHBhcmVudFxuIiwgcGFySW1wbCk7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgLyogVXNlIHRoZSBzdXJmYWNlIGFzIHBhcmVudCAqLwogICAgICAgIFBhcmVudCA9IChJVW5rbm93biAqKSBJQ09NX0lOVEVSRkFDRSgqcHBTdXJmLCBJRGlyZWN0RHJhd1N1cmZhY2U3KTsKICAgICAgICBUUkFDRSgiVXNpbmcgU3VyZmFjZSBpbnRlcmZhY2UgJXAgYXMgcGFyZW50XG4iLCAqcHBTdXJmKTsKICAgIH0KCiAgICAvKiBOb3cgY3JlYXRlIHRoZSBXaW5lRDNEIFN1cmZhY2UgKi8KICAgIGhyID0gSVdpbmVEM0REZXZpY2VfQ3JlYXRlU3VyZmFjZShUaGlzLT53aW5lRDNERGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBERFNELT5kd1dpZHRoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBERFNELT5kd0hlaWdodCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGb3JtYXQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVFJVRSAvKiBMb2NrYWJsZSAqLywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGQUxTRSAvKiBEaXNjYXJkICovLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldmVsLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICYoKnBwU3VyZiktPldpbmVEM0RTdXJmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFJlc1R5cGUsIFVzYWdlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBvb2wsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV0lORUQzRE1VTFRJU0FNUExFX05PTkUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCAvKiBNdWx0aVNhbXBsZVF1YWxpdHkgKi8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCAvKiBTaGFyZWRIYW5kbGUgKi8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSW1wbFR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyZW50KTsKCiAgICBpZihociAhPSBEM0RfT0spCiAgICB7CiAgICAgICAgRVJSKCJJV2luZUQzRERldmljZTo6Q3JlYXRlU3VyZmFjZSBmYWlsZWQuIGhyID0gJTA4bHhcbiIsIGhyKTsKICAgICAgICByZXR1cm4gaHI7CiAgICB9CgogICAgLyogU2V0IHRoZSBjaGlsZCBvZiB0aGUgcGFyZW50IGltcGxlbWVudGF0aW9uIGlmIGl0IGV4aXN0cyAqLwogICAgaWYocGFySW1wbCkKICAgIHsKICAgICAgICBwYXJJbXBsLT5jaGlsZCA9IChJVW5rbm93biAqKSAoKnBwU3VyZiktPldpbmVEM0RTdXJmYWNlOwogICAgICAgIC8qIFRoZSBJUGFyZW50IHJlbGVhc2VzIHRoZSBXaW5lRDNEU3VyZmFjZSwgYW5kCiAgICAgICAgICogdGhlIGRkcmF3IHN1cmZhY2UgZG9lcyB0aGF0IHRvby4gSG9sZCBhIHJlZmVyZW5jZQogICAgICAgICAqLwogICAgICAgIElXaW5lRDNEU3VyZmFjZV9BZGRSZWYoKCpwcFN1cmYpLT5XaW5lRDNEU3VyZmFjZSk7CiAgICB9CgogICAgLyogSW5jcmVhc2UgdGhlIHN1cmZhY2UgY291bnRlciwgYW5kIGF0dGFjaCB0aGUgc3VyZmFjZSAqLwogICAgSW50ZXJsb2NrZWRJbmNyZW1lbnQoJlRoaXMtPnN1cmZhY2VzKTsKICAgIGlmKFRoaXMtPnN1cmZhY2VfbGlzdCkKICAgIHsKICAgICAgICBUaGlzLT5zdXJmYWNlX2xpc3QtPnByZXYgPSAqcHBTdXJmOwogICAgfQogICAgKCpwcFN1cmYpLT5uZXh0ID0gVGhpcy0+c3VyZmFjZV9saXN0OwogICAgVGhpcy0+c3VyZmFjZV9saXN0ID0gKnBwU3VyZjsKCiAgICAvKiBIZXJlIHdlIGNvdWxkIHN0b3JlIGFsbCBjcmVhdGVkIHN1cmZhY2VzIGluIHRoZSBEaXJlY3REcmF3SW1wbCBzdHJ1Y3R1cmUsCiAgICAgKiBCdXQgdGhpcyBjb3VsZCBhbHNvIGJlIGRlbGVnYXRlZCB0byBXaW5lRERyYXcsIGFzIGl0IGtlZXBzIHRyYWNrIG9mIGFsbCBpdHMKICAgICAqIHJlc291cmNlcy4gTm90IGltcGxlbWVudGVkIGZvciBub3csIGFzIHRoZXJlIGFyZSBtb3JlIGltcG9ydGFudCB0aGluZ3MgOykKICAgICAqLwoKICAgIC8qIEdldCB0aGUgcGl4ZWwgZm9ybWF0IG9mIHRoZSBXaW5lRDNEU3VyZmFjZSBhbmQgc3RvcmUgaXQuCiAgICAgKiBEb24ndCB1c2UgdGhlIEZvcm1hdCBjaG9vc2VuIGFib3ZlLCBXaW5lRDNEIG1pZ2h0IGhhdmUKICAgICAqIGNoYW5nZWQgaXQKICAgICAqLwogICAgRGVzYy5Gb3JtYXQgPSAmRm9ybWF0OwogICAgRGVzYy5UeXBlID0gJlJlc1R5cGU7CiAgICBEZXNjLlVzYWdlID0gJlVzYWdlOwogICAgRGVzYy5Qb29sID0gJmR1bW15X2QzZHBvb2w7CiAgICBEZXNjLlNpemUgPSAmZHVtbXlfdWludDsKICAgIERlc2MuTXVsdGlTYW1wbGVUeXBlID0gJmR1bW15X21zdDsKICAgIERlc2MuTXVsdGlTYW1wbGVRdWFsaXR5ID0gJmR1bW15X2R3b3JkOwogICAgRGVzYy5XaWR0aCA9ICZXaWR0aDsKICAgIERlc2MuSGVpZ2h0ID0gJkhlaWdodDsKCiAgICAoKnBwU3VyZiktPnN1cmZhY2VfZGVzYy5kd0ZsYWdzIHw9IEREU0RfUElYRUxGT1JNQVQ7CiAgICBociA9IElXaW5lRDNEU3VyZmFjZV9HZXREZXNjKCgqcHBTdXJmKS0+V2luZUQzRFN1cmZhY2UsICZEZXNjKTsKICAgIGlmKGhyICE9IEQzRF9PSykKICAgIHsKICAgICAgICBFUlIoIklXaW5lRDNEU3VyZmFjZTo6R2V0RGVzYyBmYWlsZWRcbiIpOwogICAgICAgIElEaXJlY3REcmF3U3VyZmFjZTdfUmVsZWFzZSggKElEaXJlY3REcmF3U3VyZmFjZTcgKikgKnBwU3VyZik7CiAgICAgICAgcmV0dXJuIGhyOwogICAgfQoKICAgIGlmKEZvcm1hdCA9PSBXSU5FRDNERk1UX1VOS05PV04pCiAgICB7CiAgICAgICAgRklYTUUoIklXaW5lRDNEU3VyZmFjZTo6R2V0RGVzYyByZXR1cm5lZCBXSU5FRDNERk1UX1VOS05PV05cbiIpOwogICAgfQogICAgUGl4ZWxGb3JtYXRfV2luZUQzRHRvREQoICYoKnBwU3VyZiktPnN1cmZhY2VfZGVzYy51NC5kZHBmUGl4ZWxGb3JtYXQsIEZvcm1hdCk7CgogICAgLyogQW5ubyAxNjAyIHN0b3JlcyB0aGUgcGl0Y2ggcmlnaHQgYWZ0ZXIgc3VyZmFjZSBjcmVhdGlvbiwgc28gbWFrZSBzdXJlIGl0J3MgdGhlcmUuCiAgICAgKiBJIGNhbid0IExvY2tSZWN0KCkgdGhlIHN1cmZhY2UgaGVyZSBiZWNhdXNlIGlmIE9wZW5HTCBzdXJmYWNlcyBhcmUgaW4gdXNlLCB0aGUKICAgICAqIFdpbmVEM0REZXZpY2UgbWlnaHQgbm90IGJlIHVzZWFibGUgZm9yIDNEIHlldCwgc28gYW4gZXh0cmEgbWV0aG9kIHdhcyBjcmVhdGVkCiAgICAgKi8KICAgICgqcHBTdXJmKS0+c3VyZmFjZV9kZXNjLmR3RmxhZ3MgfD0gRERTRF9QSVRDSDsKICAgICgqcHBTdXJmKS0+c3VyZmFjZV9kZXNjLnUxLmxQaXRjaCA9IElXaW5lRDNEU3VyZmFjZV9HZXRQaXRjaCgoKnBwU3VyZiktPldpbmVEM0RTdXJmYWNlKTsKCiAgICAvKiBBcHBsaWNhdGlvbiBwYXNzZWQgYSBjb2xvciBrZXk/IFNldCBpdCEgKi8KICAgIGlmKHBERFNELT5kd0ZsYWdzICYgRERTRF9DS0RFU1RPVkVSTEFZKQogICAgewogICAgICAgIElXaW5lRDNEU3VyZmFjZV9TZXRDb2xvcktleSgoKnBwU3VyZiktPldpbmVEM0RTdXJmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBERENLRVlfREVTVE9WRVJMQVksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZwRERTRC0+dTMuZGRja0NLRGVzdE92ZXJsYXkpOwogICAgfQogICAgaWYocEREU0QtPmR3RmxhZ3MgJiBERFNEX0NLREVTVEJMVCkKICAgIHsKICAgICAgICBJV2luZUQzRFN1cmZhY2VfU2V0Q29sb3JLZXkoKCpwcFN1cmYpLT5XaW5lRDNEU3VyZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRERDS0VZX0RFU1RCTFQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZwRERTRC0+ZGRja0NLRGVzdEJsdCk7CiAgICB9CiAgICBpZihwRERTRC0+ZHdGbGFncyAmIEREU0RfQ0tTUkNPVkVSTEFZKQogICAgewogICAgICAgIElXaW5lRDNEU3VyZmFjZV9TZXRDb2xvcktleSgoKnBwU3VyZiktPldpbmVEM0RTdXJmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBERENLRVlfU1JDT1ZFUkxBWSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJnBERFNELT5kZGNrQ0tTcmNPdmVybGF5KTsKICAgIH0KICAgIGlmKHBERFNELT5kd0ZsYWdzICYgRERTRF9DS1NSQ0JMVCkKICAgIHsKICAgICAgICBJV2luZUQzRFN1cmZhY2VfU2V0Q29sb3JLZXkoKCpwcFN1cmYpLT5XaW5lRDNEU3VyZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRERDS0VZX1NSQ0JMVCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJnBERFNELT5kZGNrQ0tTcmNCbHQpOwogICAgfQogICAgaWYgKCBwRERTRC0+ZHdGbGFncyAmIEREU0RfTFBTVVJGQUNFKQogICAgewogICAgICAgIGhyID0gSVdpbmVEM0RTdXJmYWNlX1NldE1lbSgoKnBwU3VyZiktPldpbmVEM0RTdXJmYWNlLCBwRERTRC0+bHBTdXJmYWNlKTsKICAgICAgICBpZihociAhPSBXSU5FRDNEX09LKQogICAgICAgIHsKICAgICAgICAgICAgLyogTm8gbmVlZCBmb3IgYSB0cmFjZSBoZXJlLCB3aW5lZDNkIGRvZXMgdGhhdCBmb3IgdXMgKi8KICAgICAgICAgICAgSURpcmVjdERyYXdTdXJmYWNlN19SZWxlYXNlKElDT01fSU5URVJGQUNFKCgqcHBTdXJmKSwgSURpcmVjdERyYXdTdXJmYWNlNykpOwogICAgICAgICAgICByZXR1cm4gaHI7CiAgICAgICAgfQogICAgfQoKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6Q3JlYXRlU3VyZmFjZQogKgogKiBDcmVhdGVzIGEgbmV3IElEaXJlY3REcmF3U3VyZmFjZSBvYmplY3QgYW5kIHJldHVybnMgaXRzIGludGVyZmFjZS4KICoKICogVGhlIHN1cmZhY2UgY29ubmVjdGlvbnMgd2l0aCB3aW5lZDNkIGFyZSBhIGJpdCB0cmlja3kuIEJhc2ljYWxseSBpdCB3b3JrcwogKiBsaWtlIHRoaXM6CiAqCiAqIHwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18ICAgICAgICAgICAgICAgfC0tLS0tLS0tLS0tLS0tLS0tfAogKiB8IEREcmF3IHN1cmZhY2UgICAgICAgICAgfCAgICAgICAgICAgICAgIHwgV2luZUQzRFN1cmZhY2UgIHwKICogfCAgICAgICAgICAgICAgICAgICAgICAgIHwgICAgICAgICAgICAgICB8ICAgICAgICAgICAgICAgICB8CiAqIHwgICAgICAgIFdpbmVEM0RTdXJmYWNlICB8LS0tLS0tLS0tLS0tLS0+fCAgICAgICAgICAgICAgICAgfAogKiB8ICAgICAgICBDaGlsZCAgICAgICAgICAgfDwtLS0tLS0tLS0tLS0tPnwgUGFyZW50ICAgICAgICAgIHwKICogfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwgICAgICAgICAgICAgICB8LS0tLS0tLS0tLS0tLS0tLS18CiAqCiAqIFRoZSBERHJhdyBzdXJmYWNlIGlzIHRoZSBwYXJlbnQgb2YgdGhlIHdpbmVkM2Qgc3VyZmFjZSwgYW5kIGl0IHJlbGVhc2VzCiAqIHRoZSBXaW5lRDNEU3VyZmFjZSB3aGVuIHRoZSBkZHJhdyBzdXJmYWNlIGlzIGRlc3Ryb3llZC4KICoKICogSG93ZXZlciwgZm9yIGFsbCBzdXJmYWNlcyB3aGljaCBjYW4gYmUgaW4gYSBjb250YWluZXIgaW4gV2luZUQzRCwKICogd2UgaGF2ZSB0byBkbyB0aGlzLiBUaGVzZSBzdXJmYWNlcyBhcmUgdXN1c2FsbHkgY29tcGxleCBzdXJmYWNlcywKICogc28gdGhpcyBjb25jZXJucyBwcmltYXJ5IHN1cmZhY2VzIHdpdGggYSBmcm9udCBhbmQgYSBiYWNrIGJ1ZmZlciwKICogYW5kIHRleHR1cmVzLgogKgogKiB8LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfCAgICAgICAgICAgICAgIHwtLS0tLS0tLS0tLS0tLS0tLXwKICogfCBERHJhdyBzdXJmYWNlICAgICAgICAgIHwgICAgICAgICAgICAgICB8IENvbnRhaW50ZXIgICAgICB8CiAqIHwgICAgICAgICAgICAgICAgICAgICAgICB8ICAgICAgICAgICAgICAgfCAgICAgICAgICAgICAgICAgfAogKiB8ICAgICAgICAgICAgICAgICAgQ2hpbGQgfDwtLS0tLS0tLS0tLS0tPnwgUGFyZW50ICAgICAgICAgIHwKICogfCAgICAgICAgICAgICAgICBUZXh0dXJlIHw8LS0tLS0tLS0tLS0tLT58ICAgICAgICAgICAgICAgICB8CiAqIHwgICAgICAgICBXaW5lRDNEU3VyZmFjZSB8PC0tLS18ICAgICAgICAgfCAgICAgICAgICBMZXZlbHMgfDwtLXwKICogfCBDb21wbGV4IGNvbm5lY3Rpb24gICAgIHwgICAgIHwgICAgICAgICB8ICAgICAgICAgICAgICAgICB8ICAgfAogKiB8LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfCAgICAgfCAgICAgICAgIHwtLS0tLS0tLS0tLS0tLS0tLXwgICB8CiAqICBeICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICogIHwgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAogKiAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CiAqICB8ICAgIHwtLS0tLS0tLS0tLS0tLS0tLS18ICAgICB8ICAgICAgICAgfC0tLS0tLS0tLS0tLS0tLS0tfCAgIHwKICogIHwgICAgfCBJUGFyZW50ICAgICAgICAgIHwgICAgIHwtLS0tLS0tLT58IFdpbmVEM0RTdXJmYWNlICB8ICAgfAogKiAgfCAgICB8ICAgICAgICAgICAgICAgICAgfCAgICAgICAgICAgICAgIHwgICAgICAgICAgICAgICAgIHwgICB8CiAqICB8ICAgIHwgICAgICAgICAgICBDaGlsZCB8PC0tLS0tLS0tLS0tLS0+fCBQYXJlbnQgICAgICAgICAgfCAgIHwKICogIHwgICAgfCAgICAgICAgICAgICAgICAgIHwgICAgICAgICAgICAgICB8ICAgICAgIENvbnRhaW5lciB8PC0tfAogKiAgfCAgICB8LS0tLS0tLS0tLS0tLS0tLS0tfCAgICAgICAgICAgICAgIHwtLS0tLS0tLS0tLS0tLS0tLXwgICB8CiAqICB8ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICogIHwgICB8LS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAogKiAgfCAgIHwgRERyYXcgc3VyZmFjZSAyICAgICAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CiAqICB8ICAgfCAgICAgICAgICAgICAgICAgICAgICB8ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICogIHw8LT58IENvbXBsZXggcm9vdCAgIENoaWxkIHwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAogKiAgfCAgIHwgICAgICAgICAgICAgIFRleHR1cmUgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CiAqICB8ICAgfCAgICAgICBXaW5lRDNEU3VyZmFjZSB8PC0tLS18ICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICogIHwgICB8LS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwgICAgIHwgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAogKiAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CiAqICB8ICAgIHwtLS0tLS0tLS0tLS0tLS0tLS0tLS18ICAgICB8ICAgICAgfC0tLS0tLS0tLS0tLS0tLS0tfCAgIHwKICogIHwgICAgfCBJUGFyZW50ICAgICAgICAgICAgIHwgICAgIHwtLS0tLT58IFdpbmVEM0RTdXJmYWNlICB8ICAgfAogKiAgfCAgICB8ICAgICAgICAgICAgICAgICAgICAgfCAgICAgICAgICAgIHwgICAgICAgICAgICAgICAgIHwgICB8CiAqICB8ICAgIHwgICAgICAgICAgICAgICBDaGlsZCB8PC0tLS0tLS0tLS0+fCBQYXJlbnQgICAgICAgICAgfCAgIHwKICogIHwgICAgfC0tLS0tLS0tLS0tLS0tLS0tLS0tLXwgICAgICAgICAgICB8ICAgICAgIENvbnRhaW5lciB8PC0tfAogKiAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwtLS0tLS0tLS0tLS0tLS0tLXwgICB8CiAqICB8ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICogIHwgICAgICAgICAgICAgLS0tTW9yZSBzdXJmYWNlcyBjYW4gZm9sbG93LS0tICAgICAgICAgICAgICAgICAgfAogKgogKiBUaGUgcmVhc29uIGlzIHRoYXQgdGhlIElXaW5lRDNEU3dhcGNoYWluKHJlbmRlciB0YXJnZXQgY29udGFpbmVyKQogKiBhbmQgdGhlIElXaW5lRDNEVGV4dXJlKFRleHR1cmUgY29udGFpbmVyKSByZWxlYXNlIHRoZSBwYXJlbnRzCiAqIG9mIHRoZWlyIHN1cmZhY2UncyBjaGlsZHJlbiwgYnV0IGJ5IHJlbGVhc2luZyB0aGUgY29tcGxleCByb290CiAqIHRoZSBzdXJmYWNlcyB3aGljaCBhcmUgY29tcGxleGx5IGF0dGFjaGVkIHRvIGl0IGFyZSBkZXN0cm95ZWQKICogdG9vLiBTZWUgSURpcmVjdERyYXdTdXJmYWNlOjpSZWxlYXNlIGZvciBhIG1vcmUgZGV0YWlsZWQKICogZXhwbGFuYXRpb24uCiAqCiAqIFBhcmFtczoKICogIEREU0Q6IERlc2NyaXB0aW9uIG9mIHRoZSBzdXJmYWNlIHRvIGNyZWF0ZQogKiAgU3VyZjogQWRkcmVzcyB0byBzdG9yZSB0aGUgaW50ZXJmYWNlIHBvaW50ZXIgYXQKICogIFVua091dGVyOiBCYXNpY2FsbHkgZm9yIGFnZ3JlZ2F0aW9uIHN1cHBvcnQsIGJ1dCBkZHJhdyBkb2Vzbid0IHN1cHBvcnQKICogICAgICAgICAgICBhZ2dyZWdhdGlvbiwgc28gaXQgaGFzIHRvIGJlIE5VTEwKICoKICogUmV0dXJuczoKICogIEREX09LIG9uIHN1Y2Nlc3MKICogIENMQVNTX0VfTk9BR0dSRUdBVElPTiBpZiBVbmtPdXRlciAhPSBOVUxMCiAqICBEREVSUl8qIGlmIGFuIGVycm9yIG9jY3VycwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfQ3JlYXRlU3VyZmFjZShJRGlyZWN0RHJhdzcgKmlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBERFNVUkZBQ0VERVNDMiAqRERTRCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSURpcmVjdERyYXdTdXJmYWNlNyAqKlN1cmYsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIElVbmtub3duICpVbmtPdXRlcikKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIElEaXJlY3REcmF3U3VyZmFjZUltcGwgKm9iamVjdCA9IE5VTEw7CiAgICBIUkVTVUxUIGhyOwogICAgTE9ORyBleHRyYV9zdXJmYWNlcyA9IDAsIGk7CiAgICBERFNVUkZBQ0VERVNDMiBkZXNjMjsKICAgIFVJTlQgbGV2ZWwgPSAwOwogICAgV0lORUQzRERJU1BMQVlNT0RFIE1vZGU7CgogICAgVFJBQ0UoIiglcCktPiglcCwlcCwlcClcbiIsIFRoaXMsIEREU0QsIFN1cmYsIFVua091dGVyKTsKCiAgICAvKiBTb21lIGNoZWNrcyBiZWZvcmUgd2Ugc3RhcnQgKi8KICAgIGlmIChUUkFDRV9PTihkZHJhdykpCiAgICB7CiAgICAgICAgVFJBQ0UoIiAoJXApIFJlcXVlc3Rpbmcgc3VyZmFjZSBkZXNjIDpcbiIsIFRoaXMpOwogICAgICAgIEREUkFXX2R1bXBfc3VyZmFjZV9kZXNjKEREU0QpOwogICAgfQoKICAgIGlmIChVbmtPdXRlciAhPSBOVUxMKQogICAgewogICAgICAgIEZJWE1FKCIoJXApIDogb3V0ZXIgIT0gTlVMTD9cbiIsIFRoaXMpOwogICAgICAgIHJldHVybiBDTEFTU19FX05PQUdHUkVHQVRJT047IC8qIHVuY2hlY2tlZCAqLwogICAgfQoKICAgIGlmICghKEREU0QtPmR3RmxhZ3MgJiBERFNEX0NBUFMpKQogICAgewogICAgICAgIC8qIERWSURFTy5ETEwgZG9lcyBmb3JnZXQgdGhlIEREU0RfQ0FQUyBmbGFnIC4uLiAqc2lnaCogKi8KICAgICAgICBERFNELT5kd0ZsYWdzIHw9IEREU0RfQ0FQUzsKICAgIH0KICAgIGlmIChERFNELT5kZHNDYXBzLmR3Q2FwcyA9PSAwKQogICAgewogICAgICAgIC8qIFRoaXMgaGFzIGJlZW4gY2hlY2tlZCBvbiByZWFsIFdpbmRvd3MgKi8KICAgICAgICBERFNELT5kZHNDYXBzLmR3Q2FwcyA9IEREU0NBUFNfTE9DQUxWSURNRU0gfCBERFNDQVBTX1ZJREVPTUVNT1JZOwogICAgfQoKICAgIGlmIChERFNELT5kZHNDYXBzLmR3Q2FwcyAmIEREU0NBUFNfQUxMT0NPTkxPQUQpCiAgICB7CiAgICAgICAgLyogSWYgdGhlIHN1cmZhY2UgaXMgb2YgdGhlICdhbGxvY29ubG9hZCcgdHlwZSwgaWdub3JlIHRoZSBMUFNVUkZBQ0UgZmllbGQgKi8KICAgICAgICBERFNELT5kd0ZsYWdzICY9IH5ERFNEX0xQU1VSRkFDRTsKICAgIH0KCiAgICBpZiAoKEREU0QtPmR3RmxhZ3MgJiBERFNEX0xQU1VSRkFDRSkgJiYgKEREU0QtPmxwU3VyZmFjZSA9PSBOVUxMKSkKICAgIHsKICAgICAgICAvKiBGcmFuayBIZXJiZXJ0J3MgRHVuZSBzcGVjaWZpZXMgYSBudWxsIHBvaW50ZXIgZm9yIHRoZSBzdXJmYWNlLCBpZ25vcmUgdGhlIExQU1VSRkFDRSBmaWVsZCAqLwogICAgICAgIFdBUk4oIiglcCkgTnVsbCBzdXJmYWNlIHBvaW50ZXIgc3BlY2lmaWVkLCBpZ25vcmUgaXQhXG4iLCBUaGlzKTsKICAgICAgICBERFNELT5kd0ZsYWdzICY9IH5ERFNEX0xQU1VSRkFDRTsKICAgIH0KCiAgICBpZiAoU3VyZiA9PSBOVUxMKQogICAgewogICAgICAgIEZJWE1FKCIoJXApIFlvdSB3YW50IHRvIGdldCBiYWNrIGEgc3VyZmFjZT8gRG9uJ3QgZ2l2ZSBOVUxMIHB0cnMhXG4iLCBUaGlzKTsKICAgICAgICByZXR1cm4gRV9QT0lOVEVSOyAvKiB1bmNoZWNrZWQgKi8KICAgIH0KCiAgICAvKiBNb2RpZnkgc29tZSBmbGFncyAqLwogICAgbWVtc2V0KCZkZXNjMiwgMCwgc2l6ZW9mKGRlc2MyKSk7CiAgICBkZXNjMi5kd1NpemUgPSBzaXplb2YoZGVzYzIpOyAgIC8qIEZvciB0aGUgc3RydWN0IGNvcHkgKi8KICAgIEREX1NUUlVDVF9DT1BZX0JZU0laRSgmZGVzYzIsIEREU0QpOwogICAgZGVzYzIuZHdTaXplID0gc2l6ZW9mKGRlc2MyKTsgICAvKiBUbyBvdmVycmlkZSBhIHBvc3NpYmx5IHNtYWxsZXIgc2l6ZSAqLwogICAgZGVzYzIudTQuZGRwZlBpeGVsRm9ybWF0LmR3U2l6ZT1zaXplb2YoRERQSVhFTEZPUk1BVCk7IC8qIEp1c3QgdG8gYmUgc3VyZSAqLwoKICAgIC8qIEdldCB0aGUgdmlkZW8gbW9kZSBmcm9tIFdpbmVEM0QgLSB3ZSB3aWxsIG5lZWQgaXQgKi8KICAgIGhyID0gSVdpbmVEM0REZXZpY2VfR2V0RGlzcGxheU1vZGUoVGhpcy0+d2luZUQzRERldmljZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgMCwgLyogU3dhcGNoYWluIDAgKi8KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJk1vZGUpOwogICAgaWYoRkFJTEVEKGhyKSkKICAgIHsKICAgICAgICBFUlIoIkZhaWxlZCB0byByZWFkIGRpc3BsYXkgbW9kZSBmcm9tIHdpbmVkM2RcbiIpOwogICAgICAgIHN3aXRjaChUaGlzLT5vcmlnX2JwcCkKICAgICAgICB7CiAgICAgICAgICAgIGNhc2UgODoKICAgICAgICAgICAgICAgIE1vZGUuRm9ybWF0ID0gV0lORUQzREZNVF9QODsKICAgICAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICAgICAgY2FzZSAxNToKICAgICAgICAgICAgICAgIE1vZGUuRm9ybWF0ID0gV0lORUQzREZNVF9YMVI1RzVCNTsKICAgICAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICAgICAgY2FzZSAxNjoKICAgICAgICAgICAgICAgIE1vZGUuRm9ybWF0ID0gV0lORUQzREZNVF9SNUc2QjU7CiAgICAgICAgICAgICAgICBicmVhazsKCiAgICAgICAgICAgIGNhc2UgMjQ6CiAgICAgICAgICAgICAgICBNb2RlLkZvcm1hdCA9IFdJTkVEM0RGTVRfUjhHOEI4OwogICAgICAgICAgICAgICAgYnJlYWs7CgogICAgICAgICAgICBjYXNlIDMyOgogICAgICAgICAgICAgICAgTW9kZS5Gb3JtYXQgPSBXSU5FRDNERk1UX1g4UjhHOEI4OwogICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgfQogICAgICAgIE1vZGUuV2lkdGggPSBUaGlzLT5vcmlnX3dpZHRoOwogICAgICAgIE1vZGUuSGVpZ2h0ID0gVGhpcy0+b3JpZ19oZWlnaHQ7CiAgICB9CgogICAgLyogTm8gcGl4ZWxmb3JtYXQgZ2l2ZW4/IFVzZSB0aGUgY3VycmVudCBzY3JlZW4gZm9ybWF0ICovCiAgICBpZighKGRlc2MyLmR3RmxhZ3MgJiBERFNEX1BJWEVMRk9STUFUKSkKICAgIHsKICAgICAgICBkZXNjMi5kd0ZsYWdzIHw9IEREU0RfUElYRUxGT1JNQVQ7CiAgICAgICAgZGVzYzIudTQuZGRwZlBpeGVsRm9ybWF0LmR3U2l6ZT1zaXplb2YoRERQSVhFTEZPUk1BVCk7CgogICAgICAgIC8qIFdhaXQ6IEl0IGNvdWxkIGJlIGEgWiBidWZmZXIgKi8KICAgICAgICBpZihkZXNjMi5kZHNDYXBzLmR3Q2FwcyAmIEREU0NBUFNfWkJVRkZFUikKICAgICAgICB7CiAgICAgICAgICAgIHN3aXRjaChkZXNjMi51Mi5kd01pcE1hcENvdW50KSAvKiBXaG8gaGFkIHRoaXMgZ2xvcmlvdXMgaWRlYT8gKi8KICAgICAgICAgICAgewogICAgICAgICAgICAgICAgY2FzZSAxNToKICAgICAgICAgICAgICAgICAgICBQaXhlbEZvcm1hdF9XaW5lRDNEdG9ERCgmZGVzYzIudTQuZGRwZlBpeGVsRm9ybWF0LCBXSU5FRDNERk1UX0QxNVMxKTsKICAgICAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgICAgIGNhc2UgMTY6CiAgICAgICAgICAgICAgICAgICAgUGl4ZWxGb3JtYXRfV2luZUQzRHRvREQoJmRlc2MyLnU0LmRkcGZQaXhlbEZvcm1hdCwgV0lORUQzREZNVF9EMTYpOwogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgY2FzZSAyNDoKICAgICAgICAgICAgICAgICAgICBQaXhlbEZvcm1hdF9XaW5lRDNEdG9ERCgmZGVzYzIudTQuZGRwZlBpeGVsRm9ybWF0LCBXSU5FRDNERk1UX0QyNFg4KTsKICAgICAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgICAgIGNhc2UgMzI6CiAgICAgICAgICAgICAgICAgICAgUGl4ZWxGb3JtYXRfV2luZUQzRHRvREQoJmRlc2MyLnU0LmRkcGZQaXhlbEZvcm1hdCwgV0lORUQzREZNVF9EMzIpOwogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgICAgICBFUlIoIlVua25vd24gWiBidWZmZXIgYml0IGRlcHRoXG4iKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICBQaXhlbEZvcm1hdF9XaW5lRDNEdG9ERCgmZGVzYzIudTQuZGRwZlBpeGVsRm9ybWF0LCBNb2RlLkZvcm1hdCk7CiAgICAgICAgfQogICAgfQoKICAgIC8qIE5vIFdpZHRoIG9yIG5vIEhlaWdodD8gVXNlIHRoZSBjdXJyZW50IHdpbmRvdyBzaXplIG9yCiAgICAgKiB0aGUgb3JpZ2luYWwgc2NyZWVuIHNpemUKICAgICAqLwogICAgaWYoIShkZXNjMi5kd0ZsYWdzICYgRERTRF9XSURUSCkgfHwKICAgICAgICEoZGVzYzIuZHdGbGFncyAmIEREU0RfSEVJR0hUKSApCiAgICB7CiAgICAgICAgSFdORCB3aW5kb3c7CgogICAgICAgIC8qIEZhbGxiYWNrOiBGcm9tIFdpbmVEM0QgLyBvcmlnaW5hbCBtb2RlICovCiAgICAgICAgZGVzYzIuZHdGbGFncyB8PSBERFNEX1dJRFRIIHwgRERTRF9IRUlHSFQ7CiAgICAgICAgZGVzYzIuZHdXaWR0aCA9IE1vZGUuV2lkdGg7CiAgICAgICAgZGVzYzIuZHdIZWlnaHQgPSBNb2RlLkhlaWdodDsKCiAgICAgICAgaHIgPSBJV2luZUQzRERldmljZV9HZXRIV05EKFRoaXMtPndpbmVEM0REZXZpY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZ3aW5kb3cpOwogICAgICAgIGlmKCAoaHIgPT0gRDNEX09LKSAmJiAod2luZG93ICE9IDApICkKICAgICAgICB7CiAgICAgICAgICAgIFJFQ1QgcmVjdDsKICAgICAgICAgICAgaWYoR2V0V2luZG93UmVjdCh3aW5kb3csICZyZWN0KSApCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIC8qIFRoaXMgaXMgYSBoYWNrIHVudGlsIEkgZmluZCBhIGJldHRlciBzb2x1dGlvbiAqLwogICAgICAgICAgICAgICAgaWYoIChyZWN0LnJpZ2h0IC0gcmVjdC5sZWZ0KSA8PSAxIHx8CiAgICAgICAgICAgICAgICAgICAgKHJlY3QuYm90dG9tIC0gcmVjdC50b3ApIDw9IDEgKQogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgIEZJWE1FKCJXYW50ZWQgdG8gZ2V0IHN1cmZhY2UgZGltZW5zaW9ucyBmcm9tIHdpbmRvdyAlcCwgYnV0IGl0IGhhcyBvbmx5IFwKICAgICAgICAgICAgICAgICAgICAgICAgICAgYSBzaXplIG9mICVsZHglbGQuIFVzaW5nIGZ1bGwgc2NyZWVuIGRpbWVuc2lvbnNcbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIHdpbmRvdywgcmVjdC5yaWdodCAtIHJlY3QubGVmdCwgcmVjdC5ib3R0b20gLSByZWN0LnRvcCk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgLyogTm90IHN1cmUgaWYgdGhpcyBpcyBjb3JyZWN0ICovCiAgICAgICAgICAgICAgICAgICAgZGVzYzIuZHdXaWR0aCA9IHJlY3QucmlnaHQgLSByZWN0LmxlZnQ7CiAgICAgICAgICAgICAgICAgICAgZGVzYzIuZHdIZWlnaHQgPSByZWN0LmJvdHRvbSAtIHJlY3QudG9wOwogICAgICAgICAgICAgICAgICAgIFRSQUNFKCJVc2luZyB3aW5kb3cgJXAncyBkaW1lbnNpb25zOiAlbGR4JWxkXG4iLCB3aW5kb3csIGRlc2MyLmR3V2lkdGgsIGRlc2MyLmR3SGVpZ2h0KTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICAvKiBNaXBtYXAgY291bnQgZml4ZXMgKi8KICAgIGlmKGRlc2MyLmRkc0NhcHMuZHdDYXBzICYgRERTQ0FQU19NSVBNQVApCiAgICB7CiAgICAgICAgaWYoZGVzYzIuZGRzQ2Fwcy5kd0NhcHMgJiBERFNDQVBTX0NPTVBMRVgpCiAgICAgICAgewogICAgICAgICAgICBpZihkZXNjMi5kd0ZsYWdzICYgRERTRF9NSVBNQVBDT1VOVCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgLyogTWlwbWFwIGNvdW50IGlzIGdpdmVuLCBub3RoaW5nIHRvIGRvICovCiAgICAgICAgICAgIH0KICAgICAgICAgICAgZWxzZQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAvKiBVbmRvY3VtZW50ZWQgZmVhdHVyZTogQ3JlYXRlIHN1YmxldmVscyB1bnRpbAogICAgICAgICAgICAgICAgICogZWl0aGVyIHRoZSB3aWR0aCBvciB0aGUgaGVpZ2h0IGlzIDEKICAgICAgICAgICAgICAgICAqLwogICAgICAgICAgICAgICAgRFdPUkQgbWluID0gZGVzYzIuZHdXaWR0aCA8IGRlc2MyLmR3SGVpZ2h0ID8KICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRlc2MyLmR3V2lkdGggOiBkZXNjMi5kd0hlaWdodDsKICAgICAgICAgICAgICAgIGRlc2MyLnUyLmR3TWlwTWFwQ291bnQgPSAwOwogICAgICAgICAgICAgICAgd2hpbGUoIG1pbiApCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgZGVzYzIudTIuZHdNaXBNYXBDb3VudCArPSAxOwogICAgICAgICAgICAgICAgICAgIG1pbiA+Pj0gMTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICAvKiBOb3QtY29tcGxleCBtaXBtYXAgLT4gTWlwbWFwY291bnQgPSAxICovCiAgICAgICAgICAgIGRlc2MyLnUyLmR3TWlwTWFwQ291bnQgPSAxOwogICAgICAgIH0KICAgICAgICBleHRyYV9zdXJmYWNlcyA9IGRlc2MyLnUyLmR3TWlwTWFwQ291bnQgLSAxOwoKICAgICAgICAvKiBUaGVyZSdzIGEgbWlwbWFwIGNvdW50IGluIHRoZSBjcmVhdGVkIHN1cmZhY2UgaW4gYW55IGNhc2UgKi8KICAgICAgICBkZXNjMi5kd0ZsYWdzIHw9IEREU0RfTUlQTUFQQ09VTlQ7CiAgICB9CiAgICAvKiBJZiBubyBtaXBtYXAgaXMgZ2l2ZW4sIHRoZSB0ZXh0dXJlIGhhcyBvbmx5IG9uZSBsZXZlbCAqLwoKICAgIC8qIFRoZSBmaXJzdCBzdXJmYWNlIGlzIGEgZnJvbnQgYnVmZmVyLCB0aGUgYmFjayBidWZmZXIgaXMgY3JlYXRlZCBhZnRlcndhcmRzICovCiAgICBpZiggKGRlc2MyLmR3RmxhZ3MgJiBERFNEX0NBUFMpICYmIChkZXNjMi5kZHNDYXBzLmR3Q2FwcyAmIEREU0NBUFNfUFJJTUFSWVNVUkZBQ0UpICkKICAgIHsKICAgICAgICBkZXNjMi5kZHNDYXBzLmR3Q2FwcyB8PSBERFNDQVBTX0ZST05UQlVGRkVSOwogICAgfQoKICAgIC8qIENyZWF0ZSB0aGUgZmlyc3Qgc3VyZmFjZSAqLwogICAgaHIgPSBJRGlyZWN0RHJhd0ltcGxfQ3JlYXRlTmV3U3VyZmFjZShUaGlzLCAmZGVzYzIsICZvYmplY3QsIDApOwogICAgaWYoIGhyICE9IEREX09LKQogICAgewogICAgICAgIEVSUigiSURpcmVjdERyYXdJbXBsX0NyZWF0ZU5ld1N1cmZhY2UgZmFpbGVkIHdpdGggJTA4bHhcbiIsIGhyKTsKICAgICAgICByZXR1cm4gaHI7CiAgICB9CgogICAgKlN1cmYgPSBJQ09NX0lOVEVSRkFDRShvYmplY3QsIElEaXJlY3REcmF3U3VyZmFjZTcpOwoKICAgIC8qIENyZWF0ZSBBZGRpdGlvbmFsIHN1cmZhY2VzIGlmIG5lY2Vzc2FyeQogICAgICogVGhpcyBhcHBsaWVzIHRvIFByaW1hcnkgc3VyZmFjZXMgd2hpY2ggaGF2ZSBhIGJhY2sgYnVmZmVyIGNvdW50CiAgICAgKiBzZXQsIGJ1dCBub3QgdG8gbWlwbWFwIHRleHR1cmVzLiBJbiBjYXNlIG9mIE1pcG1hcCB0ZXh0dXJlcywKICAgICAqIHdpbmVEM0QgdGFrZXMgY2FyZSBvZiB0aGUgY3JlYXRpb24gb2YgYWRkaXRpb25hbCBzdXJmYWNlcwogICAgICovCiAgICBpZihERFNELT5kd0ZsYWdzICYgRERTRF9CQUNLQlVGRkVSQ09VTlQpCiAgICB7CiAgICAgICAgZXh0cmFfc3VyZmFjZXMgPSBERFNELT5kd0JhY2tCdWZmZXJDb3VudDsKICAgICAgICBkZXNjMi5kZHNDYXBzLmR3Q2FwcyAmPSB+RERTQ0FQU19GUk9OVEJVRkZFUjsgLyogSXQncyBub3QgYSBmcm9udCBidWZmZXIgKi8KICAgICAgICBkZXNjMi5kZHNDYXBzLmR3Q2FwcyB8PSBERFNDQVBTX0JBQ0tCVUZGRVI7CiAgICB9CgogICAgZm9yKGkgPSAwOyBpIDwgZXh0cmFfc3VyZmFjZXM7IGkrKykKICAgIHsKICAgICAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICpvYmplY3QyID0gTlVMTDsKICAgICAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICppdGVyYXRvcjsKCiAgICAgICAgLyogaW5jcmVhc2UgdGhlIG1pcG1hcCBsZXZlbCwgYnV0IG9ubHkgaWYgYSBtaXBtYXAgaXMgY3JlYXRlZAogICAgICAgICAqIEluIHRoaXMgY2FzZSwgYWxzbyBoYWx2ZSB0aGUgc2l6ZQogICAgICAgICAqLwogICAgICAgIGlmKEREU0QtPmRkc0NhcHMuZHdDYXBzICYgRERTQ0FQU19NSVBNQVApCiAgICAgICAgewogICAgICAgICAgICBsZXZlbCsrOwogICAgICAgICAgICBkZXNjMi5kd1dpZHRoIC89IDI7CiAgICAgICAgICAgIGRlc2MyLmR3SGVpZ2h0IC89IDI7CiAgICAgICAgfQoKICAgICAgICBociA9IElEaXJlY3REcmF3SW1wbF9DcmVhdGVOZXdTdXJmYWNlKFRoaXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmZGVzYzIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmb2JqZWN0MiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldmVsKTsKICAgICAgICBpZihociAhPSBERF9PSykKICAgICAgICB7CiAgICAgICAgICAgIC8qIFRoaXMgZGVzdHJveXMgYW5kIHBvc3NpYmx5IGNyZWF0ZWQgc3VyZmFjZXMgdG9vICovCiAgICAgICAgICAgIElEaXJlY3REcmF3U3VyZmFjZV9SZWxlYXNlKCBJQ09NX0lOVEVSRkFDRShvYmplY3QsIElEaXJlY3REcmF3U3VyZmFjZTcpICk7CiAgICAgICAgICAgIHJldHVybiBocjsKICAgICAgICB9CgogICAgICAgIC8qIEFkZCB0aGUgbmV3IHN1cmZhY2UgdG8gdGhlIGNvbXBsZXggYXR0YWNobWVudCBsaXN0ICovCiAgICAgICAgb2JqZWN0Mi0+Zmlyc3RfY29tcGxleCA9IG9iamVjdDsKICAgICAgICBvYmplY3QyLT5uZXh0X2NvbXBsZXggPSBOVUxMOwogICAgICAgIGl0ZXJhdG9yID0gb2JqZWN0OwogICAgICAgIHdoaWxlKGl0ZXJhdG9yLT5uZXh0X2NvbXBsZXgpIGl0ZXJhdG9yID0gaXRlcmF0b3ItPm5leHRfY29tcGxleDsKICAgICAgICBpdGVyYXRvci0+bmV4dF9jb21wbGV4ID0gb2JqZWN0MjsKCiAgICAgICAgLyogUmVtb3ZlIHRoZSAocG9zc2libGUpIGJhY2sgYnVmZmVyIGNhcCBmcm9tIHRoZSBuZXcgc3VyZmFjZSBkZXNjcmlwdGlvbiwKICAgICAgICAgKiBiZWNhdXNlIG9ubHkgb25lIHN1cmZhY2UgaW4gdGhlIGZsaXBwaW5nIGNoYWluIGlzIGEgYmFjayBidWZmZXIsIG9uZQogICAgICAgICAqIGlzIGEgZnJvbnQgYnVmZmVyLCB0aGUgb3RoZXJzIGFyZSBqdXN0IHByaW1hcnkgc3VyZmFjZXMuCiAgICAgICAgICovCiAgICAgICAgZGVzYzIuZGRzQ2Fwcy5kd0NhcHMgJj0gfkREU0NBUFNfQkFDS0JVRkZFUjsKICAgIH0KCiAgICAvKiBBZGRyZWYgdGhlIGRkcmF3IGludGVyZmFjZSB0byBrZWVwIGFuIHJlZmVyZW5jZSBmb3IgZWFjaCBzdXJmYWNlICovCiAgICBJRGlyZWN0RHJhdzdfQWRkUmVmKGlmYWNlKTsKICAgIG9iamVjdC0+aWZhY2VUb1JlbGVhc2UgPSAoSVVua25vd24gKikgaWZhY2U7CgogICAgLyogSWYgdGhlIGltcGxlbWVudGF0aW9uIGlzIE9wZW5HTCBhbmQgdGhlcmUncyBubyBkM2RkZXZpY2UsIGF0dGFjaCBhIGQzZGRldmljZQogICAgICogQnV0IGF0dGFjaCB0aGUgZDNkZGV2aWNlIG9ubHkgaWYgdGhlIGN1cnJlbnRseSBjcmVhdGVkIHN1cmZhY2Ugd2FzCiAgICAgKiBhIHByaW1hcnkgc3VyZmFjZSAoMkQgYXBwIGluIDNEIG1vZGUpIG9yIGEgM0RERVZJQ0Ugc3VyZmFjZSAoM0QgYXBwKQogICAgICogVGhlIG9ubHkgY2FzZSBJIGNhbiB0aGluayBvZiB3aGVyZSB0aGlzIGRvZXNuJ3QgYXBwbHkgaXMgd2hlbiBhCiAgICAgKiAyRCBhcHAgd2FzIGNvbmZpZ3VyZWQgYnkgdGhlIHVzZXIgdG8gcnVuIHdpdGggT3BlbkdMIGFuZCBpdCBkaWRuJ3QgY3JlYXRlCiAgICAgKiB0aGUgcmVuZGVyIHRhcmdldCBhcyBmaXJzdCBzdXJmYWNlLiBJbiB0aGlzIGNhc2UgdGhlIHJlbmRlciB0YXJnZXQgY3JlYXRpb24KICAgICAqIHdpbGwgY2F1c2UgdGhlIDNEIGluaXQuCiAgICAgKi8KICAgIGlmKCAoVGhpcy0+SW1wbFR5cGUgPT0gU1VSRkFDRV9PUEVOR0wpICYmICEoVGhpcy0+ZDNkX2luaXRpYWxpemVkKSAmJgogICAgICAgIGRlc2MyLmRkc0NhcHMuZHdDYXBzICYgKEREU0NBUFNfUFJJTUFSWVNVUkZBQ0UgfCBERFNDQVBTXzNEREVWSUNFKSApCiAgICB7CiAgICAgICAgSURpcmVjdERyYXdTdXJmYWNlSW1wbCAqdGFyZ2V0ID0gVGhpcy0+c3VyZmFjZV9saXN0OwoKICAgICAgICAvKiBTZWFyY2ggZm9yIHRoZSBwcmltYXJ5IHRvIHVzZSBhcyByZW5kZXIgdGFyZ2V0ICovCiAgICAgICAgd2hpbGUodGFyZ2V0KQogICAgICAgIHsKICAgICAgICAgICAgaWYodGFyZ2V0LT5zdXJmYWNlX2Rlc2MuZGRzQ2Fwcy5kd0NhcHMgJiBERFNDQVBTX1BSSU1BUllTVVJGQUNFKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAvKiBmb3VuZCAqLwogICAgICAgICAgICAgICAgVFJBQ0UoIlVzaW5nIHByaW1hcnkgJXAgYXMgcmVuZGVyIHRhcmdldFxuIiwgdGFyZ2V0KTsKICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICB9CiAgICAgICAgICAgIHRhcmdldCA9IHRhcmdldC0+bmV4dDsKICAgICAgICB9CiAgICAgICAgLyogSWYgaXQncyBub3QgZm91bmQsIHVzZSB0aGUganVzdCBjcmVhdGVkIEREU0NBUFNfM0RERVZJQ0Ugc3VyZmFjZSAqLwogICAgICAgIGlmKCF0YXJnZXQpCiAgICAgICAgewogICAgICAgICAgICB0YXJnZXQgPSBvYmplY3Q7CiAgICAgICAgfQoKICAgICAgICBUUkFDRSgiKCVwKSBBdHRhY2hpbmcgYSBEM0REZXZpY2UsIHJlbmRlcnRhcmdldCA9ICVwXG4iLCBUaGlzLCB0YXJnZXQpOwogICAgICAgIGhyID0gSURpcmVjdERyYXdJbXBsX0F0dGFjaEQzRERldmljZShUaGlzLCB0YXJnZXQtPmZpcnN0X2NvbXBsZXgpOwogICAgICAgIGlmKGhyICE9IEQzRF9PSykKICAgICAgICB7CiAgICAgICAgICAgIEVSUigiSURpcmVjdERyYXdJbXBsX0F0dGFjaEQzRERldmljZSBmYWlsZWQsIGhyID0gJWx4XG4iLCBocik7CiAgICAgICAgfQogICAgfQoKICAgIC8qIENyZWF0ZSBhIFdpbmVEM0RUZXh0dXJlIGlmIGEgdGV4dHVyZSB3YXMgcmVxdWVzdGVkICovCiAgICBpZihERFNELT5kZHNDYXBzLmR3Q2FwcyAmIEREU0NBUFNfVEVYVFVSRSkKICAgIHsKICAgICAgICBVSU5UIGxldmVsczsKICAgICAgICBXSU5FRDNERk9STUFUIEZvcm1hdDsKICAgICAgICBXSU5FRDNEUE9PTCBQb29sID0gV0lORUQzRFBPT0xfREVGQVVMVDsKCiAgICAgICAgVGhpcy0+dGV4X3Jvb3QgPSBvYmplY3Q7CgogICAgICAgIGlmKGRlc2MyLmRkc0NhcHMuZHdDYXBzICYgRERTQ0FQU19NSVBNQVApCiAgICAgICAgewogICAgICAgICAgICAvKiBhIG1pcG1hcCBpcyBjcmVhdGVkLCBjcmVhdGUgZW5vdWdoIGxldmVscyAqLwogICAgICAgICAgICBsZXZlbHMgPSBkZXNjMi51Mi5kd01pcE1hcENvdW50OwogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICAvKiBObyBtaXBtYXAgaXMgY3JlYXRlZCwgY3JlYXRlIG9uZSBsZXZlbCAqLwogICAgICAgICAgICBsZXZlbHMgPSAxOwogICAgICAgIH0KCiAgICAgICAgLyogRERTQ0FQU19TWVNURU1NRU1PUlkgdGV4dHVyZXMgYXJlIGluIFdJTkVEM0RQT09MX1NZU1RFTU1FTSAqLwogICAgICAgIGlmKEREU0QtPmRkc0NhcHMuZHdDYXBzICYgRERTQ0FQU19TWVNURU1NRU1PUlkpCiAgICAgICAgewogICAgICAgICAgICBQb29sID0gV0lORUQzRFBPT0xfU1lTVEVNTUVNOwogICAgICAgIH0KICAgICAgICAvKiBTaG91bGQgSSBmb3J3YXJkIHRoZSBNQU5FR0VEIGNhcCB0byB0aGUgbWFuYWdlZCBwb29sID8gKi8KCiAgICAgICAgLyogR2V0IHRoZSBmb3JtYXQuIEl0J3Mgc2V0IGFscmVhZHkgYnkgQ3JlYXRlTmV3U3VyZmFjZSAqLwogICAgICAgIEZvcm1hdCA9IFBpeGVsRm9ybWF0X0REMldpbmVEM0QoJm9iamVjdC0+c3VyZmFjZV9kZXNjLnU0LmRkcGZQaXhlbEZvcm1hdCk7CgogICAgICAgIC8qIFRoZSBzdXJmYWNlcyBhcmUgYWxyZWFkeSBjcmVhdGVkLCB0aGUgY2FsbGJhY2sgb25seQogICAgICAgICAqIHBhc3NlcyB0aGUgSVdpbmVEM0RTdXJmYWNlIHRvIFdpbmVEM0QKICAgICAgICAgKi8KICAgICAgICBociA9IElXaW5lRDNERGV2aWNlX0NyZWF0ZVRleHR1cmUoIFRoaXMtPndpbmVEM0REZXZpY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBERFNELT5kd1dpZHRoLCBERFNELT5kd0hlaWdodCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldmVscywgLyogTWlwTWFwQ291bnQgPSBMZXZlbHMgKi8KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsIC8qIHVzYWdlICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGb3JtYXQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQb29sLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJm9iamVjdC0+d2luZUQzRFRleHR1cmUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAwLCAvKiBTaGFyZWRIYW5kbGUgKi8KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChJVW5rbm93biAqKSBJQ09NX0lOVEVSRkFDRShvYmplY3QsIElEaXJlY3REcmF3U3VyZmFjZTcpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRDNEN0NCX0NyZWF0ZVN1cmZhY2UgKTsKICAgICAgICBUaGlzLT50ZXhfcm9vdCA9IE5VTEw7CiAgICB9CgogICAgcmV0dXJuIGhyOwp9CgojZGVmaW5lIERERU5VTVNVUkZBQ0VTX1NFQVJDSFRZUEUgKERERU5VTVNVUkZBQ0VTX0NBTkJFQ1JFQVRFRHxEREVOVU1TVVJGQUNFU19ET0VTRVhJU1QpCiNkZWZpbmUgRERFTlVNU1VSRkFDRVNfTUFUQ0hUWVBFIChEREVOVU1TVVJGQUNFU19BTEx8RERFTlVNU1VSRkFDRVNfTUFUQ0h8RERFTlVNU1VSRkFDRVNfTk9NQVRDSCkKCnN0YXRpYyBCT09MCk1haW5fRGlyZWN0RHJhd19ERFBJWEVMRk9STUFUX01hdGNoKGNvbnN0IEREUElYRUxGT1JNQVQgKnJlcXVlc3RlZCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgRERQSVhFTEZPUk1BVCAqcHJvdmlkZWQpCnsKICAgIC8qIFNvbWUgZmxhZ3MgbXVzdCBiZSBwcmVzZW50IGluIGJvdGggb3IgbmVpdGhlciBmb3IgYSBtYXRjaC4gKi8KICAgIHN0YXRpYyBjb25zdCBEV09SRCBtdXN0X21hdGNoID0gRERQRl9QQUxFVFRFSU5ERVhFRDEgfCBERFBGX1BBTEVUVEVJTkRFWEVEMgogICAgICAgIHwgRERQRl9QQUxFVFRFSU5ERVhFRDQgfCBERFBGX1BBTEVUVEVJTkRFWEVEOCB8IEREUEZfRk9VUkNDCiAgICAgICAgfCBERFBGX1pCVUZGRVIgfCBERFBGX1NURU5DSUxCVUZGRVI7CgogICAgaWYgKChyZXF1ZXN0ZWQtPmR3RmxhZ3MgJiBwcm92aWRlZC0+ZHdGbGFncykgIT0gcmVxdWVzdGVkLT5kd0ZsYWdzKQogICAgICAgIHJldHVybiBGQUxTRTsKCiAgICBpZiAoKHJlcXVlc3RlZC0+ZHdGbGFncyAmIG11c3RfbWF0Y2gpICE9IChwcm92aWRlZC0+ZHdGbGFncyAmIG11c3RfbWF0Y2gpKQogICAgICAgIHJldHVybiBGQUxTRTsKCiAgICBpZiAocmVxdWVzdGVkLT5kd0ZsYWdzICYgRERQRl9GT1VSQ0MpCiAgICAgICAgaWYgKHJlcXVlc3RlZC0+ZHdGb3VyQ0MgIT0gcHJvdmlkZWQtPmR3Rm91ckNDKQogICAgICAgICAgICByZXR1cm4gRkFMU0U7CgogICAgaWYgKHJlcXVlc3RlZC0+ZHdGbGFncyAmIChERFBGX1JHQnxERFBGX1lVVnxERFBGX1pCVUZGRVJ8RERQRl9BTFBIQQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8RERQRl9MVU1JTkFOQ0V8RERQRl9CVU1QRFVEVikpCiAgICAgICAgaWYgKHJlcXVlc3RlZC0+dTEuZHdSR0JCaXRDb3VudCAhPSBwcm92aWRlZC0+dTEuZHdSR0JCaXRDb3VudCkKICAgICAgICAgICAgcmV0dXJuIEZBTFNFOwoKICAgIGlmIChyZXF1ZXN0ZWQtPmR3RmxhZ3MgJiAoRERQRl9SR0J8RERQRl9ZVVZ8RERQRl9TVEVOQ0lMQlVGRkVSCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHxERFBGX0xVTUlOQU5DRXxERFBGX0JVTVBEVURWKSkKICAgICAgICBpZiAocmVxdWVzdGVkLT51Mi5kd1JCaXRNYXNrICE9IHByb3ZpZGVkLT51Mi5kd1JCaXRNYXNrKQogICAgICAgICAgICByZXR1cm4gRkFMU0U7CgogICAgaWYgKHJlcXVlc3RlZC0+ZHdGbGFncyAmIChERFBGX1JHQnxERFBGX1lVVnxERFBGX1pCVUZGRVJ8RERQRl9CVU1QRFVEVikpCiAgICAgICAgaWYgKHJlcXVlc3RlZC0+dTMuZHdHQml0TWFzayAhPSBwcm92aWRlZC0+dTMuZHdHQml0TWFzaykKICAgICAgICAgICAgcmV0dXJuIEZBTFNFOwoKICAgIC8qIEkgY291bGQgYmUgd3JvbmcgYWJvdXQgdGhlIGJ1bXBtYXBwaW5nLiBNU0ROIGRvY3MgYXJlIHZhZ3VlLiAqLwogICAgaWYgKHJlcXVlc3RlZC0+ZHdGbGFncyAmIChERFBGX1JHQnxERFBGX1lVVnxERFBGX1NURU5DSUxCVUZGRVIKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfEREUEZfQlVNUERVRFYpKQogICAgICAgIGlmIChyZXF1ZXN0ZWQtPnU0LmR3QkJpdE1hc2sgIT0gcHJvdmlkZWQtPnU0LmR3QkJpdE1hc2spCiAgICAgICAgICAgIHJldHVybiBGQUxTRTsKCiAgICBpZiAocmVxdWVzdGVkLT5kd0ZsYWdzICYgKEREUEZfQUxQSEFQSVhFTFN8RERQRl9aUElYRUxTKSkKICAgICAgICBpZiAocmVxdWVzdGVkLT51NS5kd1JHQkFscGhhQml0TWFzayAhPSBwcm92aWRlZC0+dTUuZHdSR0JBbHBoYUJpdE1hc2spCiAgICAgICAgICAgIHJldHVybiBGQUxTRTsKCiAgICByZXR1cm4gVFJVRTsKfQoKc3RhdGljIEJPT0wKSURpcmVjdERyYXdJbXBsX0REU0RfTWF0Y2goY29uc3QgRERTVVJGQUNFREVTQzIqIHJlcXVlc3RlZCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgRERTVVJGQUNFREVTQzIqIHByb3ZpZGVkKQp7CiAgICBzdHJ1Y3QgY29tcGFyZV9pbmZvCiAgICB7CiAgICAgICAgRFdPUkQgZmxhZzsKICAgICAgICBwdHJkaWZmX3Qgb2Zmc2V0OwogICAgICAgIHNpemVfdCBzaXplOwogICAgfTsKCiNkZWZpbmUgQ01QKEZMQUcsIEZJRUxEKSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXAogICAgICAgIHsgRERTRF8jI0ZMQUcsIG9mZnNldG9mKEREU1VSRkFDRURFU0MyLCBGSUVMRCksIFwKICAgICAgICAgIHNpemVvZigoKEREU1VSRkFDRURFU0MyICopKE5VTEwpKS0+RklFTEQpIH0KCiAgICBzdGF0aWMgY29uc3Qgc3RydWN0IGNvbXBhcmVfaW5mbyBjb21wYXJlW10gPQogICAgewogICAgICAgIENNUChBTFBIQUJJVERFUFRILCBkd0FscGhhQml0RGVwdGgpLAogICAgICAgIENNUChCQUNLQlVGRkVSQ09VTlQsIGR3QmFja0J1ZmZlckNvdW50KSwKICAgICAgICBDTVAoQ0FQUywgZGRzQ2FwcyksCiAgICAgICAgQ01QKENLREVTVEJMVCwgZGRja0NLRGVzdEJsdCksCiAgICAgICAgQ01QKENLREVTVE9WRVJMQVksIHUzIC8qIGRkY2tDS0Rlc3RPdmVybGF5ICovKSwKICAgICAgICBDTVAoQ0tTUkNCTFQsIGRkY2tDS1NyY0JsdCksCiAgICAgICAgQ01QKENLU1JDT1ZFUkxBWSwgZGRja0NLU3JjT3ZlcmxheSksCiAgICAgICAgQ01QKEhFSUdIVCwgZHdIZWlnaHQpLAogICAgICAgIENNUChMSU5FQVJTSVpFLCB1MSAvKiBkd0xpbmVhclNpemUgKi8pLAogICAgICAgIENNUChMUFNVUkZBQ0UsIGxwU3VyZmFjZSksCiAgICAgICAgQ01QKE1JUE1BUENPVU5ULCB1MiAvKiBkd01pcE1hcENvdW50ICovKSwKICAgICAgICBDTVAoUElUQ0gsIHUxIC8qIGxQaXRjaCAqLyksCiAgICAgICAgLyogUElYRUxGT1JNQVQ6IG1hbnVhbCAqLwogICAgICAgIENNUChSRUZSRVNIUkFURSwgdTIgLyogZHdSZWZyZXNoUmF0ZSAqLyksCiAgICAgICAgQ01QKFRFWFRVUkVTVEFHRSwgZHdUZXh0dXJlU3RhZ2UpLAogICAgICAgIENNUChXSURUSCwgZHdXaWR0aCksCiAgICAgICAgLyogWkJVRkZFUkJJVERFUFRIOiAib2Jzb2xldGUiICovCiAgICB9OwoKI3VuZGVmIENNUAoKICAgIHVuc2lnbmVkIGludCBpOwoKICAgIGlmICgocmVxdWVzdGVkLT5kd0ZsYWdzICYgcHJvdmlkZWQtPmR3RmxhZ3MpICE9IHJlcXVlc3RlZC0+ZHdGbGFncykKICAgICAgICByZXR1cm4gRkFMU0U7CgogICAgZm9yIChpPTA7IGkgPCBzaXplb2YoY29tcGFyZSkvc2l6ZW9mKGNvbXBhcmVbMF0pOyBpKyspCiAgICB7CiAgICAgICAgaWYgKHJlcXVlc3RlZC0+ZHdGbGFncyAmIGNvbXBhcmVbaV0uZmxhZwogICAgICAgICAgICAmJiBtZW1jbXAoKGNvbnN0IGNoYXIgKilwcm92aWRlZCArIGNvbXBhcmVbaV0ub2Zmc2V0LAogICAgICAgICAgICAgICAgICAgICAgKGNvbnN0IGNoYXIgKilyZXF1ZXN0ZWQgKyBjb21wYXJlW2ldLm9mZnNldCwKICAgICAgICAgICAgICAgICAgICAgIGNvbXBhcmVbaV0uc2l6ZSkgIT0gMCkKICAgICAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmIChyZXF1ZXN0ZWQtPmR3RmxhZ3MgJiBERFNEX1BJWEVMRk9STUFUKQogICAgewogICAgICAgIGlmICghTWFpbl9EaXJlY3REcmF3X0REUElYRUxGT1JNQVRfTWF0Y2goJnJlcXVlc3RlZC0+dTQuZGRwZlBpeGVsRm9ybWF0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmcHJvdmlkZWQtPnU0LmRkcGZQaXhlbEZvcm1hdCkpCiAgICAgICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICByZXR1cm4gVFJVRTsKfQoKI3VuZGVmIERERU5VTVNVUkZBQ0VTX1NFQVJDSFRZUEUKI3VuZGVmIERERU5VTVNVUkZBQ0VTX01BVENIVFlQRQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6RW51bVN1cmZhY2VzCiAqCiAqIExvb3BzIHRocm91Z2ggYWxsIHN1cmZhY2VzIGF0dGFjaGVkIHRvIHRoaXMgZGV2aWNlIGFuZCBjYWxscyB0aGUKICogYXBwbGljYXRpb24gY2FsbGJhY2suIFRoaXMgY2FuJ3QgYmUgcmVsYXllZCB0byBXaW5lRDNERGV2aWNlLAogKiBiZWNhdXNlIHNvbWUgV2luZUQzRFN1cmZhY2VzJyBwYXJlbnRzIGFyZSBJUGFyZW50IG9iamVjdHMKICoKICogUGFyYW1zOgogKiAgRmxhZ3M6IFNvbWUgZmlsdGVyaW5nIGZsYWdzLiBTZWUgSURpcmVjdERyYXdJbXBsX0VudW1TdXJmYWNlc0NhbGxiYWNrCiAqICBERFNEOiBEZXNjcmlwdGlvbiB0byBmaWx0ZXIgZm9yCiAqICBDb250ZXh0OiBBcHBsaWNhdGlvbi1wcm92aWRlZCBwb2ludGVyLCBpdCdzIHBhc3NlZCB1bm1vZGlmaWVkIHRvIHRoZQogKiAgICAgICAgICAgQ2FsbGJhY2sgZnVuY3Rpb24KICogIENhbGxiYWNrOiBBZGRyZXNzIHRvIGNhbGwgZm9yIGVhY2ggc3VyZmFjZQogKgogKiBSZXR1cm5zOgogKiAgRERFUlJfSU5WQUxJRFBBUkFNUyBpZiB0aGUgY2FsbGJhY2sgaXMgTlVMTAogKiAgRERfT0sgb24gc3VjY2VzcwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfRW51bVN1cmZhY2VzKElEaXJlY3REcmF3NyAqaWZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgRmxhZ3MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRERTVVJGQUNFREVTQzIgKkREU0QsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCAqQ29udGV4dCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUERERU5VTVNVUkZBQ0VTQ0FMTEJBQ0s3IENhbGxiYWNrKQp7CiAgICAvKiBUaGUgc3VyZmFjZSBlbnVtZXJhdGlvbiBpcyBoYW5kbGVkIGJ5IFdpbmVERHJhdywKICAgICAqIGJlY2F1c2UgaXQga2VlcHMgdHJhY2sgb2YgYWxsIHN1cmZhY2VzIGF0dGFjaGVkIHRvCiAgICAgKiBpdC4gVGhlIGZpbHRlcmluZyBpcyBkb25lIGJ5IG91ciBjYWxsYmFjayBmdW5jdGlvbiwKICAgICAqIGJlY2F1c2UgV2luZUREcmF3IGRvZXNuJ3QgaGFuZGxlIGRkcmF3LWxpa2Ugc3VyZmFjZQogICAgICogY2FwcyBzdHJ1Y3R1cmVzCiAgICAgKi8KICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICpzdXJmOwogICAgQk9PTCBhbGwsIG5vbWF0Y2g7CiAgICBERFNVUkZBQ0VERVNDMiBkZXNjOwoKICAgIGFsbCA9IEZsYWdzICYgRERFTlVNU1VSRkFDRVNfQUxMOwogICAgbm9tYXRjaCA9IEZsYWdzICYgRERFTlVNU1VSRkFDRVNfTk9NQVRDSDsKCiAgICBUUkFDRSgiKCVwKS0+KCVseCwlcCwlcCwlcClcbiIsIFRoaXMsIEZsYWdzLCBERFNELCBDb250ZXh0LCBDYWxsYmFjayk7CgogICAgaWYoIUNhbGxiYWNrKQogICAgICAgIHJldHVybiBEREVSUl9JTlZBTElEUEFSQU1TOwoKICAgIGZvcihzdXJmID0gVGhpcy0+c3VyZmFjZV9saXN0OyBzdXJmOyBzdXJmID0gc3VyZi0+bmV4dCkKICAgIHsKICAgICAgICBpZiAoYWxsIHx8IChub21hdGNoICE9IElEaXJlY3REcmF3SW1wbF9ERFNEX01hdGNoKEREU0QsICZzdXJmLT5zdXJmYWNlX2Rlc2MpKSkKICAgICAgICB7CiAgICAgICAgICAgIGRlc2MgPSBzdXJmLT5zdXJmYWNlX2Rlc2M7CiAgICAgICAgICAgIElEaXJlY3REcmF3U3VyZmFjZTdfQWRkUmVmKElDT01fSU5URVJGQUNFKHN1cmYsIElEaXJlY3REcmF3U3VyZmFjZTcpKTsKICAgICAgICAgICAgaWYoQ2FsbGJhY2soIElDT01fSU5URVJGQUNFKHN1cmYsIElEaXJlY3REcmF3U3VyZmFjZTcpLCAmZGVzYywgQ29udGV4dCkgIT0gRERFTlVNUkVUX09LKQogICAgICAgICAgICAgICAgcmV0dXJuIEREX09LOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIEQzRDdDQl9DcmVhdGVSZW5kZXJUYXJnZXQKICoKICogQ2FsbGJhY2sgY2FsbGVkIGJ5IFdpbmVEM0QgdG8gY3JlYXRlIFN1cmZhY2VzIGZvciByZW5kZXIgdGFyZ2V0IHVzYWdlCiAqIFRoaXMgZnVuY3Rpb24gdGFrZXMgdGhlIEQzRCB0YXJnZXQgZnJvbSB0aGUgSURpcmVjdERyYXdJbXBsIHN0cnVjdHVyZSwKICogYW5kIHJldHVybnMgdGhlIFdpbmVEM0RTdXJmYWNlLiBUbyBhdm9pZCBkb3VibGUgdXNhZ2UsIHRoZSBzdXJmYWNlCiAqIGlzIG1hcmtlZCBhcyByZW5kZXIgdGFyZ2V0IGFmdGVyd2FyZHMKICoKICogUGFyYW1zCiAqICBkZXZpY2U6IFRoZSBXaW5lRDNERGV2aWNlJ3MgcGFyZW50CiAqICBXaWR0aCwgSGVpZ2h0LCBGb3JtYXQ6IERpbWVuc2lvbnMgYW5kIHBpeGVsZm9ybWF0IG9mIHRoZSByZW5kZXIgdGFyZ2V0CiAqICAgICAgICAgICAgICAgICAgICAgICAgIElnbm9yZWQsIGJlY2F1c2UgdGhlIHN1cmZhY2UgYWxyZWFkeSBleGlzdHMKICogIE11bHRpU2FtcGxlLCBNdWx0aXNhbXBsZVF1YWxpdHksIExvY2thYmxlOiBJZ25vcmVkIGZvciB0aGUgc2FtZSByZWFzb24KICogIExvY2thYmxlOiBpZ25vcmVkCiAqICBwcFN1cmZhY2U6IEFkZHJlc3MgdG8gcGFzcyB0aGUgc3VyZmFjZSBwb2ludGVyIGJhY2sgYXQKICogIHBTaGFyZWRIYW5kbGU6IElnbm9yZWQKICoKICogUmV0dXJuczoKICogIEFsd2F5cyByZXR1cm5zIEQzRF9PSwogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpEM0Q3Q0JfQ3JlYXRlUmVuZGVyVGFyZ2V0KElVbmtub3duICpkZXZpY2UsIFVJTlQgV2lkdGgsIFVJTlQgSGVpZ2h0LAogICAgICAgICAgICAgICAgICAgICAgICAgIFdJTkVEM0RGT1JNQVQgRm9ybWF0LAogICAgICAgICAgICAgICAgICAgICAgICAgIFdJTkVEM0RNVUxUSVNBTVBMRV9UWVBFIE11bHRpU2FtcGxlLAogICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIE11bHRpc2FtcGxlUXVhbGl0eSwKICAgICAgICAgICAgICAgICAgICAgICAgICBCT09MIExvY2thYmxlLAogICAgICAgICAgICAgICAgICAgICAgICAgIElXaW5lRDNEU3VyZmFjZSoqIHBwU3VyZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICBIQU5ETEUqIHBTaGFyZWRIYW5kbGUpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBkZXZpY2UpOwogICAgSURpcmVjdERyYXdTdXJmYWNlSW1wbCAqZDNkU3VyZmFjZSA9IChJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICopIFRoaXMtPmQzZF90YXJnZXQtPmZpcnN0X2NvbXBsZXg7CiAgICBUUkFDRSgiKCVwKSBjYWxsIGJhY2tcbiIsIGRldmljZSk7CgogICAgLyogTG9vcCB0aHJvdWdoIHRoZSBjb21wbGV4IGNoYWluIGFuZCB0cnkgdG8gZmluZCB1bnVzZWQgcHJpbWFyeSBzdXJmYWNlcyAqLwogICAgd2hpbGUoZDNkU3VyZmFjZS0+aXNSZW5kZXJUYXJnZXQpCiAgICB7CiAgICAgICAgZDNkU3VyZmFjZSA9IGQzZFN1cmZhY2UtPm5leHRfY29tcGxleDsKICAgICAgICBpZighZDNkU3VyZmFjZSkgYnJlYWs7CiAgICB9CiAgICBpZighZDNkU3VyZmFjZSkKICAgIHsKICAgICAgICBkM2RTdXJmYWNlID0gVGhpcy0+ZDNkX3RhcmdldDsKICAgICAgICBFUlIoIiAoJXApIDogTm8gRGlyZWN0RHJhd1N1cmZhY2UgZm91bmQgdG8gY3JlYXRlIHRoZSBiYWNrIGJ1ZmZlci4gVXNpbmcgdGhlIGZyb250IGJ1ZmZlciBhcyBiYWNrIGJ1ZmZlci4gVW5jZXJ0YWluIGNvbnNlcXVlbmNlc1xuIiwgVGhpcyk7CiAgICB9CgogICAgLyogVE9ETzogUmV0dXJuIGZhaWx1cmUgaWYgdGhlIGRpbWVuc2lvbnMgZG8gbm90IG1hdGNoLCBidXQgdGhpcyBzaG91bGRuJ3QgaGFwcGVuICovCgogICAgKnBwU3VyZmFjZSA9IGQzZFN1cmZhY2UtPldpbmVEM0RTdXJmYWNlOwogICAgZDNkU3VyZmFjZS0+aXNSZW5kZXJUYXJnZXQgPSBUUlVFOwogICAgVFJBQ0UoIlJldHVybmluZyB3aW5lRDNEU3VyZmFjZSAlcCwgaXQgYmVsb25ncyB0byBzdXJmYWNlICVwXG4iLCAqcHBTdXJmYWNlLCBkM2RTdXJmYWNlKTsKICAgIHJldHVybiBEM0RfT0s7Cn0KCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpEM0Q3Q0JfQ3JlYXRlRGVwdGhTdGVuY2lsU3VyZmFjZShJVW5rbm93biAqZGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVSU5UIFdpZHRoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVSU5UIEhlaWdodCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV0lORUQzREZPUk1BVCBGb3JtYXQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFdJTkVEM0RNVUxUSVNBTVBMRV9UWVBFIE11bHRpU2FtcGxlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBNdWx0aXNhbXBsZVF1YWxpdHksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEJPT0wgRGlzY2FyZCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSVdpbmVEM0RTdXJmYWNlKiogcHBTdXJmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBIQU5ETEUqIHBTaGFyZWRIYW5kbGUpCnsKICAgIC8qIENyZWF0ZSBhIERlcHRoIFN0ZW5jaWwgc3VyZmFjZSB0byBtYWtlIFdpbmVEM0QgaGFwcHkgKi8KICAgIEhSRVNVTFQgaHIgPSBEM0RfT0s7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgZGV2aWNlKTsKICAgIEREU1VSRkFDRURFU0MyIGRkc2Q7CgogICAgVFJBQ0UoIiglcCkgY2FsbCBiYWNrXG4iLCBkZXZpY2UpOwoKICAgICpwcFN1cmZhY2UgPSBOVUxMOwoKICAgIC8qIENyZWF0ZSBhIERpcmVjdERyYXcgc3VyZmFjZSAqLwogICAgbWVtc2V0KCZkZHNkLCAwLCBzaXplb2YoZGRzZCkpOwogICAgZGRzZC5kd1NpemUgPSBzaXplb2YoZGRzZCk7CiAgICBkZHNkLnU0LmRkcGZQaXhlbEZvcm1hdC5kd1NpemUgPSBzaXplb2YoRERQSVhFTEZPUk1BVCk7CiAgICBkZHNkLmR3RmxhZ3MgPSBERFNEX1BJWEVMRk9STUFUIHwgRERTRF9XSURUSCB8IEREU0RfSEVJR0hUIHwgRERTRF9DQVBTOwogICAgZGRzZC5kZHNDYXBzLmR3Q2FwcyA9IEREU0NBUFNfT0ZGU0NSRUVOUExBSU47CiAgICBkZHNkLmR3SGVpZ2h0ID0gSGVpZ2h0OwogICAgZGRzZC5kd1dpZHRoID0gV2lkdGg7CiAgICBpZihGb3JtYXQgIT0gMCkKICAgIHsKICAgICAgUGl4ZWxGb3JtYXRfV2luZUQzRHRvREQoJmRkc2QudTQuZGRwZlBpeGVsRm9ybWF0LCBGb3JtYXQpOwogICAgfQogICAgZWxzZQogICAgewogICAgICBkZHNkLmR3RmxhZ3MgXj0gRERTRF9QSVhFTEZPUk1BVDsKICAgIH0KCiAgICBUaGlzLT5kZXB0aHN0ZW5jaWwgPSBUUlVFOwogICAgaHIgPSBJRGlyZWN0RHJhdzdfQ3JlYXRlU3VyZmFjZSgoSURpcmVjdERyYXc3ICopIFRoaXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZkZHNkLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoSURpcmVjdERyYXdTdXJmYWNlNyAqKikgJlRoaXMtPkRlcHRoU3RlbmNpbEJ1ZmZlciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTlVMTCk7CiAgICBUaGlzLT5kZXB0aHN0ZW5jaWwgPSBGQUxTRTsKICAgIGlmKEZBSUxFRChocikpCiAgICB7CiAgICAgICAgRVJSKCIgKCVwKSBDcmVhdGluZyBhIERlcHRoU3RlbmNpbCBTdXJmYWNlIGZhaWxlZCwgcmVzdWx0ID0gJWx4XG4iLCBUaGlzLCBocik7CiAgICAgICAgcmV0dXJuIGhyOwogICAgfQogICAgKnBwU3VyZmFjZSA9IFRoaXMtPkRlcHRoU3RlbmNpbEJ1ZmZlci0+V2luZUQzRFN1cmZhY2U7CiAgICByZXR1cm4gRDNEX09LOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogRDNEN0NCX0NyZWF0ZUFkZGl0aW9uYWxTd2FwQ2hhaW4KICoKICogQ2FsbGJhY2sgZnVuY3Rpb24gZm9yIFdpbmVEM0Qgd2hpY2ggY3JlYXRlcyBhIG5ldyBXaW5lRDNEU3dhcGNoYWluCiAqIGludGVyZmFjZS4gSXQgYWxzbyBjcmVhdGVzIGFuIElQYXJlbnQgaW50ZXJmYWNlIHRvIHN0b3JlIHRoYXQgcG9pbnRlciwKICogc28gdGhlIFdpbmVEM0RTd2FwY2hhaW4gaGFzIGEgcGFyZW50IGFuZCBjYW4gYmUgcmVsZWFzZWQgd2hlbiB0aGUgRDNECiAqIGRldmljZSBpcyBkZXN0cm95ZWQKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKRDNEN0NCX0NyZWF0ZUFkZGl0aW9uYWxTd2FwQ2hhaW4oSVVua25vd24gKmRldmljZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgV0lORUQzRFBSRVNFTlRfUEFSQU1FVEVSUyogcFByZXNlbnRhdGlvblBhcmFtZXRlcnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIElXaW5lRDNEU3dhcENoYWluICoqIHBwU3dhcENoYWluKQp7CiAgICBJQ09NX1RISVNfRlJPTShJRGlyZWN0RHJhd0ltcGwsIElEaXJlY3REcmF3NywgZGV2aWNlKTsKICAgIElQYXJlbnRJbXBsICpvYmplY3QgPSBOVUxMOwogICAgSFJFU1VMVCByZXMgPSBEM0RfT0s7CiAgICBJV2luZUQzRFN3YXBDaGFpbiAqc3dhcGNoYWluOwogICAgVFJBQ0UoIiglcCkgY2FsbCBiYWNrXG4iLCBkZXZpY2UpOwoKICAgIG9iamVjdCA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAgSEVBUF9aRVJPX01FTU9SWSwgc2l6ZW9mKElQYXJlbnRJbXBsKSk7CiAgICBpZiAoTlVMTCA9PSBvYmplY3QpCiAgICB7CiAgICAgICAgRklYTUUoIkFsbG9jYXRpb24gb2YgbWVtb3J5IGZhaWxlZFxuIik7CiAgICAgICAgKnBwU3dhcENoYWluID0gTlVMTDsKICAgICAgICByZXR1cm4gRERFUlJfT1VUT0ZWSURFT01FTU9SWTsKICAgIH0KCiAgICBJQ09NX0lOSVRfSU5URVJGQUNFKG9iamVjdCwgSVBhcmVudCwgSVBhcmVudF9WdGJsKTsKICAgIG9iamVjdC0+cmVmID0gMTsKCiAgICByZXMgPSBJV2luZUQzRERldmljZV9DcmVhdGVBZGRpdGlvbmFsU3dhcENoYWluKFRoaXMtPndpbmVEM0REZXZpY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBQcmVzZW50YXRpb25QYXJhbWV0ZXJzLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJnN3YXBjaGFpbiwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChJVW5rbm93biopIElDT01fSU5URVJGQUNFKG9iamVjdCwgSVBhcmVudCksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEQzRDdDQl9DcmVhdGVSZW5kZXJUYXJnZXQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEQzRDdDQl9DcmVhdGVEZXB0aFN0ZW5jaWxTdXJmYWNlKTsKICAgIGlmIChyZXMgIT0gRDNEX09LKQogICAgewogICAgICAgIEZJWE1FKCIoJXApIGNhbGwgdG8gSVdpbmVEM0REZXZpY2VfQ3JlYXRlQWRkaXRpb25hbFN3YXBDaGFpbiBmYWlsZWRcbiIsIFRoaXMpOwogICAgICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAgLCBvYmplY3QpOwogICAgICAgICpwcFN3YXBDaGFpbiA9IE5VTEw7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgKnBwU3dhcENoYWluID0gc3dhcGNoYWluOwogICAgICAgIG9iamVjdC0+Y2hpbGQgPSAoSVVua25vd24gKikgc3dhcGNoYWluOwogICAgfQoKICAgIHJldHVybiByZXM7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGlyZWN0RHJhd0ltcGxfQXR0YWNoRDNERGV2aWNlCiAqCiAqIEluaXRpYWxpemVzIHRoZSBEM0QgY2FwYWJpbGl0aWVzIG9mIFdpbmVEM0QKICoKICogUGFyYW1zOgogKiAgcHJpbWFyeTogVGhlIHByaW1hcnkgc3VyZmFjZSBmb3IgRDNECiAqCiAqIFJldHVybnMKICogIEREX09LIG9uIHN1Y2Nlc3MsCiAqICBEREVSUl8qIG90aGVyd2lzZQogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSQpJRGlyZWN0RHJhd0ltcGxfQXR0YWNoRDNERGV2aWNlKElEaXJlY3REcmF3SW1wbCAqVGhpcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICpwcmltYXJ5KQp7CiAgICBIUkVTVUxUIGhyOwogICAgVUlOVCAgICAgICAgICAgICAgICAgIEJhY2tCdWZmZXJDb3VudCA9IDA7CiAgICBIV05EICAgICAgICAgICAgICAgICAgd2luZG93OwoKICAgIFdJTkVEM0RQUkVTRU5UX1BBUkFNRVRFUlMgbG9jYWxQYXJhbWV0ZXJzOwogICAgQk9PTCBpc1dpbmRvd2VkLCBFbmFibGVBdXRvRGVwdGhTdGVuY2lsOwogICAgV0lORUQzREZPUk1BVCBBdXRvRGVwdGhTdGVuY2lsRm9ybWF0OwogICAgV0lORUQzRE1VTFRJU0FNUExFX1RZUEUgTXVsdGlTYW1wbGVUeXBlOwogICAgV0lORUQzRFNXQVBFRkZFQ1QgIFN3YXBFZmZlY3Q7CiAgICBEV09SRCBGbGFncywgTXVsdGlTYW1wbGVRdWFsaXR5OwogICAgVUlOVCBGdWxsU2NyZWVuX1JlZnJlc2hSYXRlSW5IeiwgUHJlc2VudGF0aW9uSW50ZXJ2YWw7CiAgICBXSU5FRDNERElTUExBWU1PREUgTW9kZTsKCiAgICBUUkFDRSgiKCVwKS0+KCVwKVxuIiwgVGhpcywgcHJpbWFyeSk7CgogICAgLyogR2V0IHRoZSB3aW5kb3cgKi8KICAgIGhyID0gSVdpbmVEM0REZXZpY2VfR2V0SFdORChUaGlzLT53aW5lRDNERGV2aWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZ3aW5kb3cpOwogICAgaWYoaHIgIT0gRDNEX09LKQogICAgewogICAgICAgIEVSUigiSVdpbmVEM0REZXZpY2U6OkdldEhXTkQgZmFpbGVkXG4iKTsKICAgICAgICByZXR1cm4gaHI7CiAgICB9CgogICAgLyogSWYgdGhlcmUncyBubyB3aW5kb3csIGNyZWF0ZSBhIGhpZGRlbiB3aW5kb3cuIFdpbmVEM0QgbmVlZHMgaXQgKi8KICAgIGlmKHdpbmRvdyA9PSAwKQogICAgewogICAgICAgIHdpbmRvdyA9IENyZWF0ZVdpbmRvd0V4QSgwLCBUaGlzLT5jbGFzc25hbWUsICJIaWRkZW4gRDNEIFdpbmRvdyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFdTX0RJU0FCTEVELCAwLCAwLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBHZXRTeXN0ZW1NZXRyaWNzKFNNX0NYU0NSRUVOKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgR2V0U3lzdGVtTWV0cmljcyhTTV9DWVNDUkVFTiksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIE5VTEwsIE5VTEwsIEdldE1vZHVsZUhhbmRsZUEoMCksIE5VTEwpOwoKICAgICAgICBTaG93V2luZG93KHdpbmRvdywgU1dfSElERSk7ICAgLyogSnVzdCB0byBiZSBzdXJlICovCiAgICAgICAgV0FSTigiKCVwKSBObyB3aW5kb3cgZm9yIHRoZSBEaXJlY3QzRERldmljZSwgY3JlYXRlZCBhIGhpZGRlbiB3aW5kb3cuIEhXTkQ9JXBcbiIsIFRoaXMsIHdpbmRvdyk7CiAgICAgICAgVGhpcy0+ZDNkX3dpbmRvdyA9IHdpbmRvdzsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICBUUkFDRSgiKCVwKSBVc2luZyBleGlzdGluZyB3aW5kb3cgJXAgZm9yIERpcmVjdDNEIHJlbmRlcmluZ1xuIiwgVGhpcywgd2luZG93KTsKICAgIH0KCiAgICAvKiB1c2UgdGhlIHN1cmZhY2UgZGVzY3JpcHRpb24gZm9yIHRoZSBkZXZpY2UgcGFyYW1ldGVycywgbm90IHRoZQogICAgICogRGV2aWNlIHNldHRpbmdzLiBUaGUgYXBwIG1pZ2h0IHJlbmRlciB0byBhbiBvZmZzY3JlZW4gc3VyZmFjZQogICAgICovCiAgICBNb2RlLldpZHRoID0gcHJpbWFyeS0+c3VyZmFjZV9kZXNjLmR3V2lkdGg7CiAgICBNb2RlLkhlaWdodCA9IHByaW1hcnktPnN1cmZhY2VfZGVzYy5kd0hlaWdodDsKICAgIE1vZGUuRm9ybWF0ID0gUGl4ZWxGb3JtYXRfREQyV2luZUQzRCgmcHJpbWFyeS0+c3VyZmFjZV9kZXNjLnU0LmRkcGZQaXhlbEZvcm1hdCk7CgogICAgaWYocHJpbWFyeS0+c3VyZmFjZV9kZXNjLmR3RmxhZ3MgJiBERFNEX0JBQ0tCVUZGRVJDT1VOVCkKICAgIHsKICAgICAgICBCYWNrQnVmZmVyQ291bnQgPSBwcmltYXJ5LT5zdXJmYWNlX2Rlc2MuZHdCYWNrQnVmZmVyQ291bnQ7CiAgICB9CgogICAgLyogU3RvcmUgdGhlIGZ1dHVyZSBSZW5kZXIgVGFyZ2V0IHN1cmZhY2UgKi8KICAgIFRoaXMtPmQzZF90YXJnZXQgPSBwcmltYXJ5OwoKICAgIGlzV2luZG93ZWQgPSAhKFRoaXMtPmNvb3BlcmF0aXZlX2xldmVsICYgRERTQ0xfRlVMTFNDUkVFTik7CiAgICBFbmFibGVBdXRvRGVwdGhTdGVuY2lsID0gRkFMU0U7CiAgICBBdXRvRGVwdGhTdGVuY2lsRm9ybWF0ID0gV0lORUQzREZNVF9EMTY7CiAgICBNdWx0aVNhbXBsZVR5cGUgPSBXSU5FRDNETVVMVElTQU1QTEVfTk9ORTsKICAgIFN3YXBFZmZlY3QgPSBXSU5FRDNEU1dBUEVGRkVDVF9DT1BZOwogICAgRmxhZ3MgPSAwOwogICAgTXVsdGlTYW1wbGVRdWFsaXR5ID0gMDsKICAgIEZ1bGxTY3JlZW5fUmVmcmVzaFJhdGVJbkh6ID0gV0lORUQzRFBSRVNFTlRfUkFURV9ERUZBVUxUOyAvKiBEZWZhdWx0IHJhdGU6IEl0J3MgYWxyZWFkeSBzZXQgKi8KICAgIFByZXNlbnRhdGlvbkludGVydmFsID0gV0lORUQzRFBSRVNFTlRfSU5URVJWQUxfREVGQVVMVDsKCiAgICBUUkFDRSgiUGFzc2luZyBtb2RlICVkXG4iLCBNb2RlLkZvcm1hdCk7CgogICAgbG9jYWxQYXJhbWV0ZXJzLkJhY2tCdWZmZXJXaWR0aCAgICAgICAgICAgICAgICA9ICZNb2RlLldpZHRoOwogICAgbG9jYWxQYXJhbWV0ZXJzLkJhY2tCdWZmZXJIZWlnaHQgICAgICAgICAgICAgICA9ICZNb2RlLkhlaWdodDsKICAgIGxvY2FsUGFyYW1ldGVycy5CYWNrQnVmZmVyRm9ybWF0ICAgICAgICAgICAgICAgPSAoV0lORUQzREZPUk1BVCAqKSAmTW9kZS5Gb3JtYXQ7CiAgICBsb2NhbFBhcmFtZXRlcnMuQmFja0J1ZmZlckNvdW50ICAgICAgICAgICAgICAgID0gKFVJTlQgKikgJkJhY2tCdWZmZXJDb3VudDsKICAgIGxvY2FsUGFyYW1ldGVycy5NdWx0aVNhbXBsZVR5cGUgICAgICAgICAgICAgICAgPSAmTXVsdGlTYW1wbGVUeXBlOwogICAgbG9jYWxQYXJhbWV0ZXJzLk11bHRpU2FtcGxlUXVhbGl0eSAgICAgICAgICAgICA9ICZNdWx0aVNhbXBsZVF1YWxpdHk7CiAgICBsb2NhbFBhcmFtZXRlcnMuU3dhcEVmZmVjdCAgICAgICAgICAgICAgICAgICAgID0gJlN3YXBFZmZlY3Q7CiAgICBsb2NhbFBhcmFtZXRlcnMuaERldmljZVdpbmRvdyAgICAgICAgICAgICAgICAgID0gJndpbmRvdzsKICAgIGxvY2FsUGFyYW1ldGVycy5XaW5kb3dlZCAgICAgICAgICAgICAgICAgICAgICAgPSAmaXNXaW5kb3dlZDsKICAgIGxvY2FsUGFyYW1ldGVycy5FbmFibGVBdXRvRGVwdGhTdGVuY2lsICAgICAgICAgPSAmRW5hYmxlQXV0b0RlcHRoU3RlbmNpbDsKICAgIGxvY2FsUGFyYW1ldGVycy5BdXRvRGVwdGhTdGVuY2lsRm9ybWF0ICAgICAgICAgPSAmQXV0b0RlcHRoU3RlbmNpbEZvcm1hdDsKICAgIGxvY2FsUGFyYW1ldGVycy5GbGFncyAgICAgICAgICAgICAgICAgICAgICAgICAgPSAmRmxhZ3M7CiAgICBsb2NhbFBhcmFtZXRlcnMuRnVsbFNjcmVlbl9SZWZyZXNoUmF0ZUluSHogICAgID0gJkZ1bGxTY3JlZW5fUmVmcmVzaFJhdGVJbkh6OwogICAgbG9jYWxQYXJhbWV0ZXJzLlByZXNlbnRhdGlvbkludGVydmFsICAgICAgICAgICA9ICZQcmVzZW50YXRpb25JbnRlcnZhbDsKCiAgICAvKiBTZXQgdGhpcyBOT1csIG90aGVyd2lzZSBjcmVhdGluZyB0aGUgZGVwdGggc3RlbmNpbCBzdXJmYWNlIHdpbGwgY2F1c2UgYQogICAgICogcmVjdXJzaXZlIGxvb3AgdW50aWwgcmFtIG9yIGVtdWxhdGVkIHZpZGVvIG1lbW9yeSBpcyBmdWxsCiAgICAgKi8KICAgIFRoaXMtPmQzZF9pbml0aWFsaXplZCA9IFRSVUU7CgogICAgaHIgPSBJV2luZUQzRERldmljZV9Jbml0M0QoVGhpcy0+d2luZUQzRERldmljZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZsb2NhbFBhcmFtZXRlcnMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEM0Q3Q0JfQ3JlYXRlQWRkaXRpb25hbFN3YXBDaGFpbik7CiAgICBpZihGQUlMRUQoaHIpKQogICAgewogICAgICAgIFRoaXMtPndpbmVEM0REZXZpY2UgPSBOVUxMOwogICAgICAgIHJldHVybiBocjsKICAgIH0KCiAgICAvKiBDcmVhdGUgYW4gSW5kZXggQnVmZmVyIHBhcmVudCAqLwogICAgVFJBQ0UoIiglcCkgU3VjY2Vzc2Z1bGx5IGluaXRpYWxpemVkIDNEXG4iLCBUaGlzKTsKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIERpcmVjdERyYXdDcmVhdGVDbGlwcGVyIChERFJBVy5AKQogKgogKiBDcmVhdGVzIGEgbmV3IElEaXJlY3REcmF3Q2xpcHBlciBvYmplY3QuCiAqCiAqIFBhcmFtczoKICogIENsaXBwZXI6IEFkZHJlc3MgdG8gd3JpdGUgdGhlIGludGVyZmFjZSBwb2ludGVyIHRvCiAqICBVbmtPdXRlcjogRm9yIGFnZ3JlZ2F0aW9uIHN1cHBvcnQsIHdoaWNoIGRkcmF3IGRvZXNuJ3QgaGF2ZS4gSGFzIHRvIGJlCiAqICAgICAgICAgICAgTlVMTAogKgogKiBSZXR1cm5zOgogKiAgQ0xBU1NfRV9OT0FHR1JFR0FUSU9OIGlmIFVua091dGVyICE9IE5VTEwKICogIEVfT1VUT0ZNRU1PUlkgaWYgYWxsb2NhdGluZyB0aGUgb2JqZWN0IGZhaWxlZAogKgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCkhSRVNVTFQgV0lOQVBJCkRpcmVjdERyYXdDcmVhdGVDbGlwcGVyKERXT1JEIEZsYWdzLAogICAgICAgICAgICAgICAgICAgICAgICBJRGlyZWN0RHJhd0NsaXBwZXIgKipDbGlwcGVyLAogICAgICAgICAgICAgICAgICAgICAgICBJVW5rbm93biAqVW5rT3V0ZXIpCnsKICAgIElEaXJlY3REcmF3Q2xpcHBlckltcGwqIG9iamVjdDsKICAgIFRSQUNFKCIoJTA4bHgsJXAsJXApXG4iLCBGbGFncywgQ2xpcHBlciwgVW5rT3V0ZXIpOwoKICAgIGlmIChVbmtPdXRlciAhPSBOVUxMKSByZXR1cm4gQ0xBU1NfRV9OT0FHR1JFR0FUSU9OOwoKICAgIG9iamVjdCA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCBIRUFQX1pFUk9fTUVNT1JZLAogICAgICAgICAgICAgICAgICAgICBzaXplb2YoSURpcmVjdERyYXdDbGlwcGVySW1wbCkpOwogICAgaWYgKG9iamVjdCA9PSBOVUxMKSByZXR1cm4gRV9PVVRPRk1FTU9SWTsKCiAgICBJQ09NX0lOSVRfSU5URVJGQUNFKG9iamVjdCwgSURpcmVjdERyYXdDbGlwcGVyLCBJRGlyZWN0RHJhd0NsaXBwZXJfVnRibCk7CiAgICBvYmplY3QtPnJlZiA9IDE7CiAgICBvYmplY3QtPmhXbmQgPSAwOwogICAgb2JqZWN0LT5kZHJhd19vd25lciA9IE5VTEw7CgogICAgKkNsaXBwZXIgPSAoSURpcmVjdERyYXdDbGlwcGVyICopIG9iamVjdDsKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6Q3JlYXRlQ2xpcHBlcgogKgogKiBDcmVhdGVzIGEgRERyYXcgY2xpcHBlci4gU2VlIERpcmVjdERyYXdDcmVhdGVDbGlwcGVyIGZvciBkZXRhaWxzCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9DcmVhdGVDbGlwcGVyKElEaXJlY3REcmF3NyAqaWZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIEZsYWdzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJRGlyZWN0RHJhd0NsaXBwZXIgKipDbGlwcGVyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBJVW5rbm93biAqVW5rT3V0ZXIpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBUUkFDRSgiKCVwKS0+KCVseCwlcCwlcClcbiIsIFRoaXMsIEZsYWdzLCBDbGlwcGVyLCBVbmtPdXRlcik7CiAgICByZXR1cm4gRGlyZWN0RHJhd0NyZWF0ZUNsaXBwZXIoRmxhZ3MsIENsaXBwZXIsIFVua091dGVyKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6Q3JlYXRlUGFsZXR0ZQogKgogKiBDcmVhdGVzIGEgbmV3IElEaXJlY3REcmF3UGFsZXR0ZSBvYmplY3QKICoKICogUGFyYW1zOgogKiAgRmxhZ3M6IFRoZSBmbGFncyBmb3IgdGhlIG5ldyBjbGlwcGVyCiAqICBDb2xvclRhYmxlOiBDb2xvciB0YWJsZSB0byBhc3NpZ24gdG8gdGhlIG5ldyBjbGlwcGVyCiAqICBQYWxldHRlOiBBZGRyZXNzIHRvIHdyaXRlIHRoZSBpbnRlcmZhY2UgcG9pbnRlciB0bwogKiAgVW5rT3V0ZXI6IEZvciBhZ2dyZWdhdGlvbiBzdXBwb3J0LCB3aGljaCBkZHJhdyBkb2Vzbid0IGhhdmUuIEhhcyB0byBiZQogKiAgICAgICAgICAgIE5VTEwKICoKICogUmV0dXJuczoKICogIENMQVNTX0VfTk9BR0dSRUdBVElPTiBpZiBVbmtPdXRlciAhPSBOVUxMCiAqICBFX09VVE9GTUVNT1JZIGlmIGFsbG9jYXRpbmcgdGhlIG9iamVjdCBmYWlsZWQKICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkKSURpcmVjdERyYXdJbXBsX0NyZWF0ZVBhbGV0dGUoSURpcmVjdERyYXc3ICppZmFjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgRmxhZ3MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBBTEVUVEVFTlRSWSAqQ29sb3JUYWJsZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSURpcmVjdERyYXdQYWxldHRlICoqUGFsZXR0ZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSVVua25vd24gKnBVbmtPdXRlcikKewogICAgSUNPTV9USElTX0ZST00oSURpcmVjdERyYXdJbXBsLCBJRGlyZWN0RHJhdzcsIGlmYWNlKTsKICAgIElEaXJlY3REcmF3UGFsZXR0ZUltcGwgKm9iamVjdDsKICAgIEhSRVNVTFQgaHIgPSBEREVSUl9HRU5FUklDOwogICAgVFJBQ0UoIiglcCktPiglbHgsJXAsJXAsJXApXG4iLCBUaGlzLCBGbGFncywgQ29sb3JUYWJsZSwgUGFsZXR0ZSwgcFVua091dGVyKTsKCiAgICBpZihwVW5rT3V0ZXIgIT0gTlVMTCkKICAgIHsKICAgICAgICBXQVJOKCJwVW5rT3V0ZXIgaXMgJXAsIHJldHVybmluZyBDTEFTU19FX05PQUdHUkVHQVRJT05cbiIsIHBVbmtPdXRlcik7CiAgICAgICAgcmV0dXJuIENMQVNTX0VfTk9BR0dSRUdBVElPTjsKICAgIH0KCiAgICAvKiBUaGUgcmVmY291bnQgdGVzdCBzaG93cyB0aGF0IGEgY29vcGxldmVsIGlzIHJlcXVpcmVkIGZvciB0aGlzICovCiAgICBpZighVGhpcy0+Y29vcGVyYXRpdmVfbGV2ZWwpCiAgICB7CiAgICAgICAgV0FSTigiTm8gY29vcGVyYXRpdmUgbGV2ZWwgc2V0LCByZXR1cm5pbmcgRERFUlJfTk9DT09QRVJBVElWRUxFVkVMU0VUXG4iKTsKICAgICAgICByZXR1cm4gRERFUlJfTk9DT09QRVJBVElWRUxFVkVMU0VUOwogICAgfQoKICAgIG9iamVjdCA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBzaXplb2YoSURpcmVjdERyYXdQYWxldHRlSW1wbCkpOwogICAgaWYoIW9iamVjdCkKICAgIHsKICAgICAgICBFUlIoIk91dCBvZiBtZW1vcnkgd2hlbiBhbGxvY2F0aW5nIG1lbW9yeSBmb3IgYSBwYWxldHRlIGltcGxlbWVudGF0aW9uXG4iKTsKICAgICAgICByZXR1cm4gRV9PVVRPRk1FTU9SWTsKICAgIH0KCiAgICBJQ09NX0lOSVRfSU5URVJGQUNFKG9iamVjdCwgSURpcmVjdERyYXdQYWxldHRlLCBJRGlyZWN0RHJhd1BhbGV0dGVfVnRibCk7CiAgICBvYmplY3QtPnJlZiA9IDE7CiAgICBvYmplY3QtPmRkcmF3X293bmVyID0gVGhpczsKCiAgICBociA9IElXaW5lRDNERGV2aWNlX0NyZWF0ZVBhbGV0dGUoVGhpcy0+d2luZUQzRERldmljZSwgRmxhZ3MsIENvbG9yVGFibGUsICZvYmplY3QtPndpbmVEM0RQYWxldHRlLCAoSVVua25vd24gKikgSUNPTV9JTlRFUkZBQ0Uob2JqZWN0LCBJRGlyZWN0RHJhd1BhbGV0dGUpICk7CiAgICBpZihociAhPSBERF9PSykKICAgIHsKICAgICAgICBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBvYmplY3QpOwogICAgICAgIHJldHVybiBocjsKICAgIH0KCiAgICBJRGlyZWN0RHJhdzdfQWRkUmVmKGlmYWNlKTsKICAgIG9iamVjdC0+aWZhY2VUb1JlbGVhc2UgPSAoSVVua25vd24gKikgaWZhY2U7CiAgICAqUGFsZXR0ZSA9IElDT01fSU5URVJGQUNFKG9iamVjdCwgSURpcmVjdERyYXdQYWxldHRlKTsKICAgIHJldHVybiBERF9PSzsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3Nzo6RHVwbGljYXRlU3VyZmFjZQogKgogKiBEdXBsaWNhdGVzIGEgc3VyZmFjZS4gVGhlIHN1cmZhY2UgbWVtb3J5IHBvaW50cyB0byB0aGUgc2FtZSBtZW1vcnkgYXMKICogdGhlIG9yaWdpbmFsIHN1cmZhY2UsIGFuZCBpdCdzIHJlbGVhc2VkIHdoZW4gdGhlIGxhc3Qgc3VyZmFjZSByZWZlcmVuY2luZwogKiBpdCBpcyByZWxlYXNlZC4gSSBndWVzcyB0aGF0J3MgYmV5b25kIFdpbmUncyBzdXJmYWNlIG1hbmFnZW1lbnQgcmlnaHQgbm93CiAqIChJZGVhOiBjcmVhdGUgYSBuZXcgRERyYXcgc3VyZmFjZSB3aXRoIHRoZSBzYW1lIFdpbmVEM0RTdXJmYWNlLiBJIG5lZWQgYQogKiB0ZXN0IGFwcGxpY2F0aW9uIHRvIGltcGxlbWVudCB0aGlzKQogKgogKiBQYXJhbXM6CiAqICBTcmM6IEFkZHJlc3Mgb2YgdGhlIHNvdXJjZSBzdXJmYWNlCiAqICBEZXN0OiBBZGRyZXNzIHRvIHdyaXRlIHRoZSBuZXcgc3VyZmFjZSBwb2ludGVyIHRvCiAqCiAqIFJldHVybnM6CiAqICBTZWUgSURpcmVjdERyYXc3OjpDcmVhdGVTdXJmYWNlCiAqCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJCklEaXJlY3REcmF3SW1wbF9EdXBsaWNhdGVTdXJmYWNlKElEaXJlY3REcmF3NyAqaWZhY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIElEaXJlY3REcmF3U3VyZmFjZTcgKlNyYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSURpcmVjdERyYXdTdXJmYWNlNyAqKkRlc3QpCnsKICAgIElDT01fVEhJU19GUk9NKElEaXJlY3REcmF3SW1wbCwgSURpcmVjdERyYXc3LCBpZmFjZSk7CiAgICBJRGlyZWN0RHJhd1N1cmZhY2VJbXBsICpTdXJmID0gSUNPTV9PQkpFQ1QoSURpcmVjdERyYXdTdXJmYWNlSW1wbCwgSURpcmVjdERyYXdTdXJmYWNlNywgU3JjKTsKCiAgICBGSVhNRSgiKCVwKS0+KCVwLCVwKVxuIiwgVGhpcywgU3VyZiwgRGVzdCk7CgogICAgLyogRm9yIG5vdywgc2ltcGx5IGNyZWF0ZSBhIG5ldywgaW5kZXBlbmRlbnQgc3VyZmFjZSAqLwogICAgcmV0dXJuIElEaXJlY3REcmF3N19DcmVhdGVTdXJmYWNlKGlmYWNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICZTdXJmLT5zdXJmYWNlX2Rlc2MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRGVzdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOVUxMKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEaXJlY3REcmF3NyBWVGFibGUKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpjb25zdCBJRGlyZWN0RHJhdzdWdGJsIElEaXJlY3REcmF3N19WdGJsID0KewogICAgLyoqKiBJVW5rbm93biAqKiovCiAgICBJRGlyZWN0RHJhd0ltcGxfUXVlcnlJbnRlcmZhY2UsCiAgICBJRGlyZWN0RHJhd0ltcGxfQWRkUmVmLAogICAgSURpcmVjdERyYXdJbXBsX1JlbGVhc2UsCiAgICAvKioqIElEaXJlY3REcmF3ICoqKi8KICAgIElEaXJlY3REcmF3SW1wbF9Db21wYWN0LAogICAgSURpcmVjdERyYXdJbXBsX0NyZWF0ZUNsaXBwZXIsCiAgICBJRGlyZWN0RHJhd0ltcGxfQ3JlYXRlUGFsZXR0ZSwKICAgIElEaXJlY3REcmF3SW1wbF9DcmVhdGVTdXJmYWNlLAogICAgSURpcmVjdERyYXdJbXBsX0R1cGxpY2F0ZVN1cmZhY2UsCiAgICBJRGlyZWN0RHJhd0ltcGxfRW51bURpc3BsYXlNb2RlcywKICAgIElEaXJlY3REcmF3SW1wbF9FbnVtU3VyZmFjZXMsCiAgICBJRGlyZWN0RHJhd0ltcGxfRmxpcFRvR0RJU3VyZmFjZSwKICAgIElEaXJlY3REcmF3SW1wbF9HZXRDYXBzLAogICAgSURpcmVjdERyYXdJbXBsX0dldERpc3BsYXlNb2RlLAogICAgSURpcmVjdERyYXdJbXBsX0dldEZvdXJDQ0NvZGVzLAogICAgSURpcmVjdERyYXdJbXBsX0dldEdESVN1cmZhY2UsCiAgICBJRGlyZWN0RHJhd0ltcGxfR2V0TW9uaXRvckZyZXF1ZW5jeSwKICAgIElEaXJlY3REcmF3SW1wbF9HZXRTY2FuTGluZSwKICAgIElEaXJlY3REcmF3SW1wbF9HZXRWZXJ0aWNhbEJsYW5rU3RhdHVzLAogICAgSURpcmVjdERyYXdJbXBsX0luaXRpYWxpemUsCiAgICBJRGlyZWN0RHJhd0ltcGxfUmVzdG9yZURpc3BsYXlNb2RlLAogICAgSURpcmVjdERyYXdJbXBsX1NldENvb3BlcmF0aXZlTGV2ZWwsCiAgICBJRGlyZWN0RHJhd0ltcGxfU2V0RGlzcGxheU1vZGUsCiAgICBJRGlyZWN0RHJhd0ltcGxfV2FpdEZvclZlcnRpY2FsQmxhbmssCiAgICAvKioqIElEaXJlY3REcmF3MiAqKiovCiAgICBJRGlyZWN0RHJhd0ltcGxfR2V0QXZhaWxhYmxlVmlkTWVtLAogICAgLyoqKiBJRGlyZWN0RHJhdzcgKioqLwogICAgSURpcmVjdERyYXdJbXBsX0dldFN1cmZhY2VGcm9tREMsCiAgICBJRGlyZWN0RHJhd0ltcGxfUmVzdG9yZUFsbFN1cmZhY2VzLAogICAgSURpcmVjdERyYXdJbXBsX1Rlc3RDb29wZXJhdGl2ZUxldmVsLAogICAgSURpcmVjdERyYXdJbXBsX0dldERldmljZUlkZW50aWZpZXIsCiAgICAvKioqIElEaXJlY3REcmF3NyAqKiovCiAgICBJRGlyZWN0RHJhd0ltcGxfU3RhcnRNb2RlVGVzdCwKICAgIElEaXJlY3REcmF3SW1wbF9FdmFsdWF0ZU1vZGUKfTsK