Ly89PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLy8KLy8gU3BsYXNoT3V0cHV0RGV2LmgKLy8KLy8gQ29weXJpZ2h0IDIwMDMgR2x5cGggJiBDb2csIExMQwovLwovLz09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQoKLy89PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KLy8KLy8gTW9kaWZpZWQgdW5kZXIgdGhlIFBvcHBsZXIgcHJvamVjdCAtIGh0dHA6Ly9wb3BwbGVyLmZyZWVkZXNrdG9wLm9yZwovLwovLyBBbGwgY2hhbmdlcyBtYWRlIHVuZGVyIHRoZSBQb3BwbGVyIHByb2plY3QgdG8gdGhpcyBmaWxlIGFyZSBsaWNlbnNlZAovLyB1bmRlciBHUEwgdmVyc2lvbiAyIG9yIGxhdGVyCi8vCi8vIENvcHlyaWdodCAoQykgMjAwNSBUYWthc2hpIEl3YWkgPHRpd2FpQHN1c2UuZGU+Ci8vIENvcHlyaWdodCAoQykgMjAwOS0yMDE1IFRob21hcyBGcmVpdGFnIDxUaG9tYXMuRnJlaXRhZ0BhbGZhLmRlPgovLyBDb3B5cmlnaHQgKEMpIDIwMDkgQ2FybG9zIEdhcmNpYSBDYW1wb3MgPGNhcmxvc2djQGdub21lLm9yZz4KLy8gQ29weXJpZ2h0IChDKSAyMDEwIENocmlzdGlhbiBGZXVlcnPkbmdlciA8Y2ZldWVyc2FlbmdlckBnb29nbGVtYWlsLmNvbT4KLy8gQ29weXJpZ2h0IChDKSAyMDExIEFuZHJlYXMgSGFydG1ldHogPGFoYXJ0bWV0ekBnbWFpbC5jb20+Ci8vIENvcHlyaWdodCAoQykgMjAxMSBBbmRyZWEgQ2FuY2lhbmkgPHJhbm1hNDJAZ21haWwuY29tPgovLyBDb3B5cmlnaHQgKEMpIDIwMTEgQWRyaWFuIEpvaG5zb24gPGFqb2huc29uQHJlZG5lb24uY29tPgovLyBDb3B5cmlnaHQgKEMpIDIwMTIsIDIwMTUgQWxiZXJ0IEFzdGFscyBDaWQgPGFhY2lkQGtkZS5vcmc+Ci8vIENvcHlyaWdodCAoQykgMjAxNSBXaWxsaWFtIEJhZGVyIDx3aWxsaWFtYmFkZXJAaG90bWFpbC5jb20+Ci8vCi8vIFRvIHNlZSBhIGRlc2NyaXB0aW9uIG9mIHRoZSBjaGFuZ2VzIHBsZWFzZSBzZWUgdGhlIENoYW5nZWxvZyBmaWxlIHRoYXQKLy8gY2FtZSB3aXRoIHlvdXIgdGFyYmFsbCBvciB0eXBlIG1ha2UgQ2hhbmdlTG9nIGlmIHlvdSBhcmUgYnVpbGRpbmcgZnJvbSBnaXQKLy8KLy89PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCiNpZm5kZWYgU1BMQVNIT1VUUFVUREVWX0gKI2RlZmluZSBTUExBU0hPVVRQVVRERVZfSAoKI2lmZGVmIFVTRV9HQ0NfUFJBR01BUwojcHJhZ21hIGludGVyZmFjZQojZW5kaWYKCiNpbmNsdWRlICJnb28vZ3R5cGVzLmgiCiNpbmNsdWRlICJzcGxhc2gvU3BsYXNoVHlwZXMuaCIKI2luY2x1ZGUgInNwbGFzaC9TcGxhc2hQYXR0ZXJuLmgiCiNpbmNsdWRlICJwb3BwbGVyLWNvbmZpZy5oIgojaW5jbHVkZSAiT3V0cHV0RGV2LmgiCiNpbmNsdWRlICJHZnhTdGF0ZS5oIgojaW5jbHVkZSAiR2xvYmFsUGFyYW1zLmgiCgpjbGFzcyBQREZEb2M7CmNsYXNzIEdmeDhCaXRGb250OwpjbGFzcyBTcGxhc2hCaXRtYXA7CmNsYXNzIFNwbGFzaDsKY2xhc3MgU3BsYXNoUGF0aDsKY2xhc3MgU3BsYXNoRm9udEVuZ2luZTsKY2xhc3MgU3BsYXNoRm9udDsKY2xhc3MgVDNGb250Q2FjaGU7CnN0cnVjdCBUM0ZvbnRDYWNoZVRhZzsKc3RydWN0IFQzR2x5cGhTdGFjazsKc3RydWN0IFNwbGFzaFRyYW5zcGFyZW5jeUdyb3VwOwoKLy8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLy8gU3BsYXNoIGR5bmFtaWMgcGF0dGVybgovLy0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKY2xhc3MgU3BsYXNoVW5pdmFyaWF0ZVBhdHRlcm46IHB1YmxpYyBTcGxhc2hQYXR0ZXJuIHsKcHVibGljOgoKICBTcGxhc2hVbml2YXJpYXRlUGF0dGVybihTcGxhc2hDb2xvck1vZGUgY29sb3JNb2RlLCBHZnhTdGF0ZSAqc3RhdGUsIEdmeFVuaXZhcmlhdGVTaGFkaW5nICpzaGFkaW5nKTsKCiAgdmlydHVhbCB+U3BsYXNoVW5pdmFyaWF0ZVBhdHRlcm4oKTsKCiAgdmlydHVhbCBHQm9vbCBnZXRDb2xvcihpbnQgeCwgaW50IHksIFNwbGFzaENvbG9yUHRyIGMpOwoKICB2aXJ0dWFsIEdCb29sIHRlc3RQb3NpdGlvbihpbnQgeCwgaW50IHkpOwoKICB2aXJ0dWFsIEdCb29sIGlzU3RhdGljKCkgeyByZXR1cm4gZ0ZhbHNlOyB9CgogIHZpcnR1YWwgR0Jvb2wgZ2V0UGFyYW1ldGVyKGRvdWJsZSB4cywgZG91YmxlIHlzLCBkb3VibGUgKnQpID0gMDsKCiAgdmlydHVhbCBHZnhVbml2YXJpYXRlU2hhZGluZyAqZ2V0U2hhZGluZygpIHsgcmV0dXJuIHNoYWRpbmc7IH0KCiAgdmlydHVhbCBHQm9vbCBpc0NNWUsoKSB7IHJldHVybiBnZnhNb2RlID09IGNzRGV2aWNlQ01ZSzsgfQoKcHJvdGVjdGVkOgogIE1hdHJpeCBpY3RtOwogIGRvdWJsZSB0MCwgdDEsIGR0OwogIEdmeFVuaXZhcmlhdGVTaGFkaW5nICpzaGFkaW5nOwogIEdmeFN0YXRlICpzdGF0ZTsKICBTcGxhc2hDb2xvck1vZGUgY29sb3JNb2RlOwogIEdmeENvbG9yU3BhY2VNb2RlIGdmeE1vZGU7Cn07CgpjbGFzcyBTcGxhc2hBeGlhbFBhdHRlcm46IHB1YmxpYyBTcGxhc2hVbml2YXJpYXRlUGF0dGVybiB7CnB1YmxpYzoKCiAgU3BsYXNoQXhpYWxQYXR0ZXJuKFNwbGFzaENvbG9yTW9kZSBjb2xvck1vZGUsIEdmeFN0YXRlICpzdGF0ZSwgR2Z4QXhpYWxTaGFkaW5nICpzaGFkaW5nKTsKCiAgdmlydHVhbCBTcGxhc2hQYXR0ZXJuICpjb3B5KCkgeyByZXR1cm4gbmV3IFNwbGFzaEF4aWFsUGF0dGVybihjb2xvck1vZGUsIHN0YXRlLCAoR2Z4QXhpYWxTaGFkaW5nICopIHNoYWRpbmcpOyB9CgogIHZpcnR1YWwgflNwbGFzaEF4aWFsUGF0dGVybigpOwoKICB2aXJ0dWFsIEdCb29sIGdldFBhcmFtZXRlcihkb3VibGUgeHMsIGRvdWJsZSB5cywgZG91YmxlICp0KTsKCnByaXZhdGU6CiAgZG91YmxlIHgwLCB5MCwgeDEsIHkxOwogIGRvdWJsZSBkeCwgZHksIG11bDsKfTsKCi8vIHNlZSBHZnhTdGF0ZS5oLCBHZnhHb3VyYXVkVHJpYW5nbGVTaGFkaW5nCmNsYXNzIFNwbGFzaEdvdXJhdWRQYXR0ZXJuOiBwdWJsaWMgU3BsYXNoR291cmF1ZENvbG9yIHsKcHVibGljOgoKICBTcGxhc2hHb3VyYXVkUGF0dGVybihHQm9vbCBiRGlyZWN0Q29sb3JUcmFuc2xhdGlvbiwgR2Z4U3RhdGUgKnN0YXRlLCBHZnhHb3VyYXVkVHJpYW5nbGVTaGFkaW5nICpzaGFkaW5nLCBTcGxhc2hDb2xvck1vZGUgbW9kZSk7CgogIHZpcnR1YWwgU3BsYXNoUGF0dGVybiAqY29weSgpIHsgcmV0dXJuIG5ldyBTcGxhc2hHb3VyYXVkUGF0dGVybihiRGlyZWN0Q29sb3JUcmFuc2xhdGlvbiwgc3RhdGUsIHNoYWRpbmcsIG1vZGUpOyB9CgogIHZpcnR1YWwgflNwbGFzaEdvdXJhdWRQYXR0ZXJuKCk7CgogIHZpcnR1YWwgR0Jvb2wgZ2V0Q29sb3IoaW50IHgsIGludCB5LCBTcGxhc2hDb2xvclB0ciBjKSB7IHJldHVybiBnRmFsc2U7IH0KCiAgdmlydHVhbCBHQm9vbCB0ZXN0UG9zaXRpb24oaW50IHgsIGludCB5KSB7IHJldHVybiBnRmFsc2U7IH0KCiAgdmlydHVhbCBHQm9vbCBpc1N0YXRpYygpIHsgcmV0dXJuIGdGYWxzZTsgfQoKICB2aXJ0dWFsIEdCb29sIGlzQ01ZSygpIHsgcmV0dXJuIGdmeE1vZGUgPT0gY3NEZXZpY2VDTVlLOyB9CgogIHZpcnR1YWwgR0Jvb2wgaXNQYXJhbWV0ZXJpemVkKCkgeyByZXR1cm4gc2hhZGluZy0+aXNQYXJhbWV0ZXJpemVkKCk7IH0KICB2aXJ0dWFsIGludCBnZXROVHJpYW5nbGVzKCkgeyByZXR1cm4gc2hhZGluZy0+Z2V0TlRyaWFuZ2xlcygpOyB9CiAgdmlydHVhbCAgdm9pZCBnZXRUcmlhbmdsZShpbnQgaSwgZG91YmxlICp4MCwgZG91YmxlICp5MCwgZG91YmxlICpjb2xvcjAsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3VibGUgKngxLCBkb3VibGUgKnkxLCBkb3VibGUgKmNvbG9yMSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvdWJsZSAqeDIsIGRvdWJsZSAqeTIsIGRvdWJsZSAqY29sb3IyKQogIHsgcmV0dXJuIHNoYWRpbmctPmdldFRyaWFuZ2xlKGksIHgwLCB5MCwgY29sb3IwLCB4MSwgeTEsIGNvbG9yMSwgeDIsIHkyLCBjb2xvcjIpOyB9CgogIHZpcnR1YWwgdm9pZCBnZXRQYXJhbWV0ZXJpemVkQ29sb3IoZG91YmxlIHQsIFNwbGFzaENvbG9yTW9kZSBtb2RlLCBTcGxhc2hDb2xvclB0ciBjKTsKCnByaXZhdGU6CiAgR2Z4R291cmF1ZFRyaWFuZ2xlU2hhZGluZyAqc2hhZGluZzsKICBHZnhTdGF0ZSAqc3RhdGU7CiAgR0Jvb2wgYkRpcmVjdENvbG9yVHJhbnNsYXRpb247CiAgU3BsYXNoQ29sb3JNb2RlIG1vZGU7CiAgR2Z4Q29sb3JTcGFjZU1vZGUgZ2Z4TW9kZTsKfTsKCi8vIHNlZSBHZnhTdGF0ZS5oLCBHZnhSYWRpYWxTaGFkaW5nCmNsYXNzIFNwbGFzaFJhZGlhbFBhdHRlcm46IHB1YmxpYyBTcGxhc2hVbml2YXJpYXRlUGF0dGVybiB7CnB1YmxpYzoKCiAgU3BsYXNoUmFkaWFsUGF0dGVybihTcGxhc2hDb2xvck1vZGUgY29sb3JNb2RlLCBHZnhTdGF0ZSAqc3RhdGUsIEdmeFJhZGlhbFNoYWRpbmcgKnNoYWRpbmcpOwoKICB2aXJ0dWFsIFNwbGFzaFBhdHRlcm4gKmNvcHkoKSB7IHJldHVybiBuZXcgU3BsYXNoUmFkaWFsUGF0dGVybihjb2xvck1vZGUsIHN0YXRlLCAoR2Z4UmFkaWFsU2hhZGluZyAqKSBzaGFkaW5nKTsgfQoKICB2aXJ0dWFsIH5TcGxhc2hSYWRpYWxQYXR0ZXJuKCk7CgogIHZpcnR1YWwgR0Jvb2wgZ2V0UGFyYW1ldGVyKGRvdWJsZSB4cywgZG91YmxlIHlzLCBkb3VibGUgKnQpOwoKcHJpdmF0ZToKICBkb3VibGUgeDAsIHkwLCByMCwgZHgsIGR5LCBkcjsKICBkb3VibGUgYSwgaW52YTsKfTsKCi8vLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgovLyBudW1iZXIgb2YgVHlwZSAzIGZvbnRzIHRvIGNhY2hlCiNkZWZpbmUgc3BsYXNoT3V0VDNGb250Q2FjaGVTaXplIDgKCi8vLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi8vIFNwbGFzaE91dHB1dERldgovLy0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKY2xhc3MgU3BsYXNoT3V0cHV0RGV2OiBwdWJsaWMgT3V0cHV0RGV2IHsKcHVibGljOgoKICAvLyBDb25zdHJ1Y3Rvci4KICBTcGxhc2hPdXRwdXREZXYoU3BsYXNoQ29sb3JNb2RlIGNvbG9yTW9kZUEsIGludCBiaXRtYXBSb3dQYWRBLAoJCSAgR0Jvb2wgcmV2ZXJzZVZpZGVvQSwgU3BsYXNoQ29sb3JQdHIgcGFwZXJDb2xvckEsCgkJICBHQm9vbCBiaXRtYXBUb3BEb3duQSA9IGdUcnVlLAoJCSAgU3BsYXNoVGhpbkxpbmVNb2RlIHRoaW5MaW5lTW9kZSA9IHNwbGFzaFRoaW5MaW5lRGVmYXVsdCwKCQkgIEdCb29sIG92ZXJwcmludFByZXZpZXdBID0gZ2xvYmFsUGFyYW1zLT5nZXRPdmVycHJpbnRQcmV2aWV3KCkpOwoKICAvLyBEZXN0cnVjdG9yLgogIHZpcnR1YWwgflNwbGFzaE91dHB1dERldigpOwoKICAvLy0tLS0tIGdldCBpbmZvIGFib3V0IG91dHB1dCBkZXZpY2UKCiAgLy8gRG9lcyB0aGlzIGRldmljZSB1c2UgdGlsaW5nUGF0dGVybkZpbGwoKT8gIElmIHRoaXMgcmV0dXJucyBmYWxzZSwKICAvLyB0aWxpbmcgcGF0dGVybiBmaWxscyB3aWxsIGJlIHJlZHVjZWQgdG8gYSBzZXJpZXMgb2Ygb3RoZXIgZHJhd2luZwogIC8vIG9wZXJhdGlvbnMuCiAgdmlydHVhbCBHQm9vbCB1c2VUaWxpbmdQYXR0ZXJuRmlsbCgpIHsgcmV0dXJuIGdUcnVlOyB9CgogIC8vIERvZXMgdGhpcyBkZXZpY2UgdXNlIGZ1bmN0aW9uU2hhZGVkRmlsbCgpLCBheGlhbFNoYWRlZEZpbGwoKSwgYW5kCiAgLy8gcmFkaWFsU2hhZGVkRmlsbCgpPyAgSWYgdGhpcyByZXR1cm5zIGZhbHNlLCB0aGVzZSBzaGFkZWQgZmlsbHMKICAvLyB3aWxsIGJlIHJlZHVjZWQgdG8gYSBzZXJpZXMgb2Ygb3RoZXIgZHJhd2luZyBvcGVyYXRpb25zLgogIHZpcnR1YWwgR0Jvb2wgdXNlU2hhZGVkRmlsbHMoaW50IHR5cGUpCiAgeyByZXR1cm4gKHR5cGUgPj0gMiAmJiB0eXBlIDw9IDUpID8gZ1RydWUgOiBnRmFsc2U7IH0KCiAgLy8gRG9lcyB0aGlzIGRldmljZSB1c2UgdXBzaWRlLWRvd24gY29vcmRpbmF0ZXM/CiAgLy8gKFVwc2lkZS1kb3duIG1lYW5zICgwLDApIGlzIHRoZSB0b3AgbGVmdCBjb3JuZXIgb2YgdGhlIHBhZ2UuKQogIHZpcnR1YWwgR0Jvb2wgdXBzaWRlRG93bigpIHsgcmV0dXJuIGJpdG1hcFRvcERvd24gXiBiaXRtYXBVcHNpZGVEb3duOyB9CgogIC8vIERvZXMgdGhpcyBkZXZpY2UgdXNlIGRyYXdDaGFyKCkgb3IgZHJhd1N0cmluZygpPwogIHZpcnR1YWwgR0Jvb2wgdXNlRHJhd0NoYXIoKSB7IHJldHVybiBnVHJ1ZTsgfQoKICAvLyBEb2VzIHRoaXMgZGV2aWNlIHVzZSBiZWdpblR5cGUzQ2hhci9lbmRUeXBlM0NoYXI/ICBPdGhlcndpc2UsCiAgLy8gdGV4dCBpbiBUeXBlIDMgZm9udHMgd2lsbCBiZSBkcmF3biB3aXRoIGRyYXdDaGFyL2RyYXdTdHJpbmcuCiAgdmlydHVhbCBHQm9vbCBpbnRlcnByZXRUeXBlM0NoYXJzKCkgeyByZXR1cm4gZ1RydWU7IH0KCiAgLy8tLS0tLSBpbml0aWFsaXphdGlvbiBhbmQgY29udHJvbAoKICAvLyBTdGFydCBhIHBhZ2UuCiAgdmlydHVhbCB2b2lkIHN0YXJ0UGFnZShpbnQgcGFnZU51bSwgR2Z4U3RhdGUgKnN0YXRlLCBYUmVmICp4cmVmKTsKCiAgLy8gRW5kIGEgcGFnZS4KICB2aXJ0dWFsIHZvaWQgZW5kUGFnZSgpOwoKICAvLy0tLS0tIHNhdmUvcmVzdG9yZSBncmFwaGljcyBzdGF0ZQogIHZpcnR1YWwgdm9pZCBzYXZlU3RhdGUoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgcmVzdG9yZVN0YXRlKEdmeFN0YXRlICpzdGF0ZSk7CgogIC8vLS0tLS0gdXBkYXRlIGdyYXBoaWNzIHN0YXRlCiAgdmlydHVhbCB2b2lkIHVwZGF0ZUFsbChHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCB1cGRhdGVDVE0oR2Z4U3RhdGUgKnN0YXRlLCBkb3VibGUgbTExLCBkb3VibGUgbTEyLAoJCQkgZG91YmxlIG0yMSwgZG91YmxlIG0yMiwgZG91YmxlIG0zMSwgZG91YmxlIG0zMik7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZUxpbmVEYXNoKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZUZsYXRuZXNzKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZUxpbmVKb2luKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZUxpbmVDYXAoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgdXBkYXRlTWl0ZXJMaW1pdChHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCB1cGRhdGVMaW5lV2lkdGgoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgdXBkYXRlU3Ryb2tlQWRqdXN0KEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZUZpbGxDb2xvclNwYWNlKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZVN0cm9rZUNvbG9yU3BhY2UoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgdXBkYXRlRmlsbENvbG9yKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZVN0cm9rZUNvbG9yKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZUJsZW5kTW9kZShHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCB1cGRhdGVGaWxsT3BhY2l0eShHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCB1cGRhdGVTdHJva2VPcGFjaXR5KEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIHVwZGF0ZVBhdHRlcm5PcGFjaXR5KEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIGNsZWFyUGF0dGVybk9wYWNpdHkoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgdXBkYXRlRmlsbE92ZXJwcmludChHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCB1cGRhdGVTdHJva2VPdmVycHJpbnQoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgdXBkYXRlT3ZlcnByaW50TW9kZShHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCB1cGRhdGVUcmFuc2ZlcihHZnhTdGF0ZSAqc3RhdGUpOwoKICAvLy0tLS0tIHVwZGF0ZSB0ZXh0IHN0YXRlCiAgdmlydHVhbCB2b2lkIHVwZGF0ZUZvbnQoR2Z4U3RhdGUgKnN0YXRlKTsKCiAgLy8tLS0tLSBwYXRoIHBhaW50aW5nCiAgdmlydHVhbCB2b2lkIHN0cm9rZShHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCBmaWxsKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIGVvRmlsbChHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgR0Jvb2wgdGlsaW5nUGF0dGVybkZpbGwoR2Z4U3RhdGUgKnN0YXRlLCBHZnggKmdmeCwgQ2F0YWxvZyAqY2F0YWxvZywgT2JqZWN0ICpzdHIsCgkJCQkgIGRvdWJsZSAqcG1hdCwgaW50IHBhaW50VHlwZSwgaW50IHRpbGluZ1R5cGUsIERpY3QgKnJlc0RpY3QsCgkJCQkgIGRvdWJsZSAqbWF0LCBkb3VibGUgKmJib3gsCgkJCQkgIGludCB4MCwgaW50IHkwLCBpbnQgeDEsIGludCB5MSwKCQkJCSAgZG91YmxlIHhTdGVwLCBkb3VibGUgeVN0ZXApOwogIHZpcnR1YWwgR0Jvb2wgYXhpYWxTaGFkZWRGaWxsKEdmeFN0YXRlICpzdGF0ZSwgR2Z4QXhpYWxTaGFkaW5nICpzaGFkaW5nLCBkb3VibGUgdE1pbiwgZG91YmxlIHRNYXgpOwogIHZpcnR1YWwgR0Jvb2wgcmFkaWFsU2hhZGVkRmlsbChHZnhTdGF0ZSAqc3RhdGUsIEdmeFJhZGlhbFNoYWRpbmcgKnNoYWRpbmcsIGRvdWJsZSB0TWluLCBkb3VibGUgdE1heCk7CiAgdmlydHVhbCBHQm9vbCBnb3VyYXVkVHJpYW5nbGVTaGFkZWRGaWxsKEdmeFN0YXRlICpzdGF0ZSwgR2Z4R291cmF1ZFRyaWFuZ2xlU2hhZGluZyAqc2hhZGluZyk7CgogIC8vLS0tLS0gcGF0aCBjbGlwcGluZwogIHZpcnR1YWwgdm9pZCBjbGlwKEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIGVvQ2xpcChHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCBjbGlwVG9TdHJva2VQYXRoKEdmeFN0YXRlICpzdGF0ZSk7CgogIC8vLS0tLS0gdGV4dCBkcmF3aW5nCiAgdmlydHVhbCB2b2lkIGRyYXdDaGFyKEdmeFN0YXRlICpzdGF0ZSwgZG91YmxlIHgsIGRvdWJsZSB5LAoJCQlkb3VibGUgZHgsIGRvdWJsZSBkeSwKCQkJZG91YmxlIG9yaWdpblgsIGRvdWJsZSBvcmlnaW5ZLAoJCQlDaGFyQ29kZSBjb2RlLCBpbnQgbkJ5dGVzLCBVbmljb2RlICp1LCBpbnQgdUxlbik7CiAgdmlydHVhbCBHQm9vbCBiZWdpblR5cGUzQ2hhcihHZnhTdGF0ZSAqc3RhdGUsIGRvdWJsZSB4LCBkb3VibGUgeSwKCQkJICAgICAgIGRvdWJsZSBkeCwgZG91YmxlIGR5LAoJCQkgICAgICAgQ2hhckNvZGUgY29kZSwgVW5pY29kZSAqdSwgaW50IHVMZW4pOwogIHZpcnR1YWwgdm9pZCBlbmRUeXBlM0NoYXIoR2Z4U3RhdGUgKnN0YXRlKTsKICB2aXJ0dWFsIHZvaWQgYmVnaW5UZXh0T2JqZWN0KEdmeFN0YXRlICpzdGF0ZSk7CiAgdmlydHVhbCB2b2lkIGVuZFRleHRPYmplY3QoR2Z4U3RhdGUgKnN0YXRlKTsKCiAgLy8tLS0tLSBpbWFnZSBkcmF3aW5nCiAgdmlydHVhbCB2b2lkIGRyYXdJbWFnZU1hc2soR2Z4U3RhdGUgKnN0YXRlLCBPYmplY3QgKnJlZiwgU3RyZWFtICpzdHIsCgkJCSAgICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LCBHQm9vbCBpbnZlcnQsCgkJCSAgICAgR0Jvb2wgaW50ZXJwb2xhdGUsIEdCb29sIGlubGluZUltZyk7CiAgdmlydHVhbCB2b2lkIHNldFNvZnRNYXNrRnJvbUltYWdlTWFzayhHZnhTdGF0ZSAqc3RhdGUsCgkJCQkJT2JqZWN0ICpyZWYsIFN0cmVhbSAqc3RyLAoJCQkJCWludCB3aWR0aCwgaW50IGhlaWdodCwgR0Jvb2wgaW52ZXJ0LAoJCQkJCUdCb29sIGlubGluZUltZywgZG91YmxlICpiYXNlTWF0cml4KTsKICB2aXJ0dWFsIHZvaWQgdW5zZXRTb2Z0TWFza0Zyb21JbWFnZU1hc2soR2Z4U3RhdGUgKnN0YXRlLCBkb3VibGUgKmJhc2VNYXRyaXgpOwogIHZpcnR1YWwgdm9pZCBkcmF3SW1hZ2UoR2Z4U3RhdGUgKnN0YXRlLCBPYmplY3QgKnJlZiwgU3RyZWFtICpzdHIsCgkJCSBpbnQgd2lkdGgsIGludCBoZWlnaHQsIEdmeEltYWdlQ29sb3JNYXAgKmNvbG9yTWFwLAoJCQkgR0Jvb2wgaW50ZXJwb2xhdGUsIGludCAqbWFza0NvbG9ycywgR0Jvb2wgaW5saW5lSW1nKTsKICB2aXJ0dWFsIHZvaWQgZHJhd01hc2tlZEltYWdlKEdmeFN0YXRlICpzdGF0ZSwgT2JqZWN0ICpyZWYsIFN0cmVhbSAqc3RyLAoJCQkgICAgICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAoJCQkgICAgICAgR2Z4SW1hZ2VDb2xvck1hcCAqY29sb3JNYXAsCgkJCSAgICAgICBHQm9vbCBpbnRlcnBvbGF0ZSwKCQkJICAgICAgIFN0cmVhbSAqbWFza1N0ciwgaW50IG1hc2tXaWR0aCwgaW50IG1hc2tIZWlnaHQsCgkJCSAgICAgICBHQm9vbCBtYXNrSW52ZXJ0LCBHQm9vbCBtYXNrSW50ZXJwb2xhdGUpOwogIHZpcnR1YWwgdm9pZCBkcmF3U29mdE1hc2tlZEltYWdlKEdmeFN0YXRlICpzdGF0ZSwgT2JqZWN0ICpyZWYsIFN0cmVhbSAqc3RyLAoJCQkJICAgaW50IHdpZHRoLCBpbnQgaGVpZ2h0LAoJCQkJICAgR2Z4SW1hZ2VDb2xvck1hcCAqY29sb3JNYXAsCgkJCQkgICBHQm9vbCBpbnRlcnBvbGF0ZSwKCQkJCSAgIFN0cmVhbSAqbWFza1N0ciwKCQkJCSAgIGludCBtYXNrV2lkdGgsIGludCBtYXNrSGVpZ2h0LAoJCQkJICAgR2Z4SW1hZ2VDb2xvck1hcCAqbWFza0NvbG9yTWFwLAoJCQkJICAgR0Jvb2wgbWFza0ludGVycG9sYXRlKTsKCiAgLy8tLS0tLSBUeXBlIDMgZm9udCBvcGVyYXRvcnMKICB2aXJ0dWFsIHZvaWQgdHlwZTNEMChHZnhTdGF0ZSAqc3RhdGUsIGRvdWJsZSB3eCwgZG91YmxlIHd5KTsKICB2aXJ0dWFsIHZvaWQgdHlwZTNEMShHZnhTdGF0ZSAqc3RhdGUsIGRvdWJsZSB3eCwgZG91YmxlIHd5LAoJCSAgICAgICBkb3VibGUgbGx4LCBkb3VibGUgbGx5LCBkb3VibGUgdXJ4LCBkb3VibGUgdXJ5KTsKCiAgLy8tLS0tLSB0cmFuc3BhcmVuY3kgZ3JvdXBzIGFuZCBzb2Z0IG1hc2tzCiAgdmlydHVhbCBHQm9vbCBjaGVja1RyYW5zcGFyZW5jeUdyb3VwKEdmeFN0YXRlICpzdGF0ZSwgR0Jvb2wga25vY2tvdXQpOwogIHZpcnR1YWwgdm9pZCBiZWdpblRyYW5zcGFyZW5jeUdyb3VwKEdmeFN0YXRlICpzdGF0ZSwgZG91YmxlICpiYm94LAoJCQkJICAgICAgR2Z4Q29sb3JTcGFjZSAqYmxlbmRpbmdDb2xvclNwYWNlLAoJCQkJICAgICAgR0Jvb2wgaXNvbGF0ZWQsIEdCb29sIGtub2Nrb3V0LAoJCQkJICAgICAgR0Jvb2wgZm9yU29mdE1hc2spOwogIHZpcnR1YWwgdm9pZCBlbmRUcmFuc3BhcmVuY3lHcm91cChHZnhTdGF0ZSAqc3RhdGUpOwogIHZpcnR1YWwgdm9pZCBwYWludFRyYW5zcGFyZW5jeUdyb3VwKEdmeFN0YXRlICpzdGF0ZSwgZG91YmxlICpiYm94KTsKICB2aXJ0dWFsIHZvaWQgc2V0U29mdE1hc2soR2Z4U3RhdGUgKnN0YXRlLCBkb3VibGUgKmJib3gsIEdCb29sIGFscGhhLAoJCQkgICBGdW5jdGlvbiAqdHJhbnNmZXJGdW5jLCBHZnhDb2xvciAqYmFja2Ryb3BDb2xvcik7CiAgdmlydHVhbCB2b2lkIGNsZWFyU29mdE1hc2soR2Z4U3RhdGUgKnN0YXRlKTsKCiAgLy8tLS0tLSBzcGVjaWFsIGFjY2VzcwoKICAvLyBDYWxsZWQgdG8gaW5kaWNhdGUgdGhhdCBhIG5ldyBQREYgZG9jdW1lbnQgaGFzIGJlZW4gbG9hZGVkLgogIHZvaWQgc3RhcnREb2MoUERGRG9jICpkb2NBKTsKIAogIHZvaWQgc2V0UGFwZXJDb2xvcihTcGxhc2hDb2xvclB0ciBwYXBlckNvbG9yQSk7CgogIEdCb29sIGlzUmV2ZXJzZVZpZGVvKCkgeyByZXR1cm4gcmV2ZXJzZVZpZGVvOyB9CiAgdm9pZCBzZXRSZXZlcnNlVmlkZW8oR0Jvb2wgcmV2ZXJzZVZpZGVvQSkgeyByZXZlcnNlVmlkZW8gPSByZXZlcnNlVmlkZW9BOyB9CgogIC8vIEdldCB0aGUgYml0bWFwIGFuZCBpdHMgc2l6ZS4KICBTcGxhc2hCaXRtYXAgKmdldEJpdG1hcCgpIHsgcmV0dXJuIGJpdG1hcDsgfQogIGludCBnZXRCaXRtYXBXaWR0aCgpOwogIGludCBnZXRCaXRtYXBIZWlnaHQoKTsKCiAgLy8gUmV0dXJucyB0aGUgbGFzdCByYXN0ZXJpemVkIGJpdG1hcCwgdHJhbnNmZXJyaW5nIG93bmVyc2hpcCB0byB0aGUKICAvLyBjYWxsZXIuCiAgU3BsYXNoQml0bWFwICp0YWtlQml0bWFwKCk7CgogIC8vIFNldCB0aGlzIGZsYWcgdG8gdHJ1ZSB0byBnZW5lcmF0ZSBhbiB1cHNpZGUtZG93biBiaXRtYXAgKHVzZWZ1bAogIC8vIGZvciBXaW5kb3dzIEJNUCBmaWxlcykuCiAgdm9pZCBzZXRCaXRtYXBVcHNpZGVEb3duKEdCb29sIGYpIHsgYml0bWFwVXBzaWRlRG93biA9IGY7IH0KCiAgLy8gR2V0IHRoZSBTcGxhc2ggb2JqZWN0LgogIFNwbGFzaCAqZ2V0U3BsYXNoKCkgeyByZXR1cm4gc3BsYXNoOyB9CgogIC8vIEdldCB0aGUgbW9kaWZpZWQgcmVnaW9uLgogIHZvaWQgZ2V0TW9kUmVnaW9uKGludCAqeE1pbiwgaW50ICp5TWluLCBpbnQgKnhNYXgsIGludCAqeU1heCk7CgogIC8vIENsZWFyIHRoZSBtb2RpZmllZCByZWdpb24uCiAgdm9pZCBjbGVhck1vZFJlZ2lvbigpOwoKICBTcGxhc2hGb250ICpnZXRDdXJyZW50Rm9udCgpIHsgcmV0dXJuIGZvbnQ7IH0KCiAgLy8gSWYgPHNraXBUZXh0QT4gaXMgdHJ1ZSwgZG9uJ3QgZHJhdyBob3Jpem9udGFsIHRleHQuCiAgLy8gSWYgPHNraXBSb3RhdGVkVGV4dEE+IGlzIHRydWUsIGRvbid0IGRyYXcgcm90YXRlZCAobm9uLWhvcml6b250YWwpIHRleHQuCiAgdm9pZCBzZXRTa2lwVGV4dChHQm9vbCBza2lwSG9yaXpUZXh0QSwgR0Jvb2wgc2tpcFJvdGF0ZWRUZXh0QSkKICAgIHsgc2tpcEhvcml6VGV4dCA9IHNraXBIb3JpelRleHRBOyBza2lwUm90YXRlZFRleHQgPSBza2lwUm90YXRlZFRleHRBOyB9CgogIGludCBnZXROZXN0Q291bnQoKSB7IHJldHVybiBuZXN0Q291bnQ7IH0KCiNpZiAxIC8vfnRtcDogdHVybiBvZmYgYW50aS1hbGlhc2luZyB0ZW1wb3JhcmlseQogIHZpcnR1YWwgR0Jvb2wgZ2V0VmVjdG9yQW50aWFsaWFzKCk7CiAgdmlydHVhbCB2b2lkIHNldFZlY3RvckFudGlhbGlhcyhHQm9vbCB2YWEpOwojZW5kaWYKCiAgR0Jvb2wgZ2V0Rm9udEFudGlhbGlhcygpIHsgcmV0dXJuIGZvbnRBbnRpYWxpYXM7IH0KICB2b2lkIHNldEZvbnRBbnRpYWxpYXMoR0Jvb2wgYW50aSkgeyBmb250QW50aWFsaWFzID0gYW50aTsgfQoKICB2b2lkIHNldEZyZWVUeXBlSGludGluZyhHQm9vbCBlbmFibGUsIEdCb29sIGVuYWJsZVNsaWdodEhpbnRpbmcpOwoKcHJvdGVjdGVkOgogIHZvaWQgZG9VcGRhdGVGb250KEdmeFN0YXRlICpzdGF0ZSk7Cgpwcml2YXRlOgogIEdCb29sIHVuaXZhcmlhdGVTaGFkZWRGaWxsKEdmeFN0YXRlICpzdGF0ZSwgU3BsYXNoVW5pdmFyaWF0ZVBhdHRlcm4gKnBhdHRlcm4sIGRvdWJsZSB0TWluLCBkb3VibGUgdE1heCk7CgogIHZvaWQgc2V0dXBTY3JlZW5QYXJhbXMoZG91YmxlIGhEUEksIGRvdWJsZSB2RFBJKTsKICBTcGxhc2hQYXR0ZXJuICpnZXRDb2xvcihHZnhHcmF5IGdyYXkpOwogIFNwbGFzaFBhdHRlcm4gKmdldENvbG9yKEdmeFJHQiAqcmdiKTsKI2lmIFNQTEFTSF9DTVlLCiAgU3BsYXNoUGF0dGVybiAqZ2V0Q29sb3IoR2Z4Q01ZSyAqY215ayk7CiAgU3BsYXNoUGF0dGVybiAqZ2V0Q29sb3IoR2Z4Q29sb3IgKmRldmljZU4pOwojZW5kaWYKICB2b2lkIHNldE92ZXJwcmludE1hc2soR2Z4Q29sb3JTcGFjZSAqY29sb3JTcGFjZSwgR0Jvb2wgb3ZlcnByaW50RmxhZywKCQkJaW50IG92ZXJwcmludE1vZGUsIEdmeENvbG9yICpzaW5nbGVDb2xvciwgR0Jvb2wgZ3JheUluZGV4ZWQgPSBnRmFsc2UpOwogIFNwbGFzaFBhdGggKmNvbnZlcnRQYXRoKEdmeFN0YXRlICpzdGF0ZSwgR2Z4UGF0aCAqcGF0aCwKCQkJICBHQm9vbCBkcm9wRW1wdHlTdWJwYXRocyk7CiAgdm9pZCBkcmF3VHlwZTNHbHlwaChHZnhTdGF0ZSAqc3RhdGUsIFQzRm9udENhY2hlICp0M0ZvbnQsCgkJICAgICAgVDNGb250Q2FjaGVUYWcgKnRhZywgR3VjaGFyICpkYXRhKTsKI2lmZGVmIFVTRV9DTVMKICBHQm9vbCB1c2VJY2NJbWFnZVNyYyh2b2lkICpkYXRhKTsKICBzdGF0aWMgdm9pZCBpY2NUcmFuc2Zvcm0odm9pZCAqZGF0YSwgU3BsYXNoQml0bWFwICpiaXRtYXApOwogIHN0YXRpYyBHQm9vbCBpY2NJbWFnZVNyYyh2b2lkICpkYXRhLCBTcGxhc2hDb2xvclB0ciBjb2xvckxpbmUsCgkJCUd1Y2hhciAqYWxwaGFMaW5lKTsKI2VuZGlmCiAgc3RhdGljIEdCb29sIGltYWdlTWFza1NyYyh2b2lkICpkYXRhLCBTcGxhc2hDb2xvclB0ciBsaW5lKTsKICBzdGF0aWMgR0Jvb2wgaW1hZ2VTcmModm9pZCAqZGF0YSwgU3BsYXNoQ29sb3JQdHIgY29sb3JMaW5lLAoJCQlHdWNoYXIgKmFscGhhTGluZSk7CiAgc3RhdGljIEdCb29sIGFscGhhSW1hZ2VTcmModm9pZCAqZGF0YSwgU3BsYXNoQ29sb3JQdHIgbGluZSwKCQkJICAgICBHdWNoYXIgKmFscGhhTGluZSk7CiAgc3RhdGljIEdCb29sIG1hc2tlZEltYWdlU3JjKHZvaWQgKmRhdGEsIFNwbGFzaENvbG9yUHRyIGxpbmUsCgkJCSAgICAgIEd1Y2hhciAqYWxwaGFMaW5lKTsKICBzdGF0aWMgR0Jvb2wgdGlsaW5nQml0bWFwU3JjKHZvaWQgKmRhdGEsIFNwbGFzaENvbG9yUHRyIGxpbmUsCgkJCSAgICAgR3VjaGFyICphbHBoYUxpbmUpOwoKICBHQm9vbCBrZWVwQWxwaGFDaGFubmVsOwkvLyBkb24ndCBmaWxsIHdpdGggcGFwZXIgY29sb3IsIGtlZXAgYWxwaGEgY2hhbm5lbAoKICBTcGxhc2hDb2xvck1vZGUgY29sb3JNb2RlOwogIGludCBiaXRtYXBSb3dQYWQ7CiAgR0Jvb2wgYml0bWFwVG9wRG93bjsKICBHQm9vbCBiaXRtYXBVcHNpZGVEb3duOwogIEdCb29sIGZvbnRBbnRpYWxpYXM7CiAgR0Jvb2wgdmVjdG9yQW50aWFsaWFzOwogIEdCb29sIG92ZXJwcmludFByZXZpZXc7CiAgR0Jvb2wgZW5hYmxlRnJlZVR5cGVIaW50aW5nOwogIEdCb29sIGVuYWJsZVNsaWdodEhpbnRpbmc7CiAgR0Jvb2wgcmV2ZXJzZVZpZGVvOwkJLy8gcmV2ZXJzZSB2aWRlbyBtb2RlCiAgU3BsYXNoQ29sb3IgcGFwZXJDb2xvcjsJLy8gcGFwZXIgY29sb3IKICBTcGxhc2hTY3JlZW5QYXJhbXMgc2NyZWVuUGFyYW1zOwogIEdCb29sIHNraXBIb3JpelRleHQ7CiAgR0Jvb2wgc2tpcFJvdGF0ZWRUZXh0OwoKICBQREZEb2MgKmRvYzsJCQkvLyB0aGUgY3VycmVudCBkb2N1bWVudAogIFhSZWYgKnhyZWY7ICAgICAgIC8vIHRoZSB4cmVmIG9mIHRoZSBjdXJyZW50IGRvY3VtZW50CgogIFNwbGFzaEJpdG1hcCAqYml0bWFwOwogIFNwbGFzaCAqc3BsYXNoOwogIFNwbGFzaEZvbnRFbmdpbmUgKmZvbnRFbmdpbmU7CgogIFQzRm9udENhY2hlICoJCQkvLyBUeXBlIDMgZm9udCBjYWNoZQogICAgdDNGb250Q2FjaGVbc3BsYXNoT3V0VDNGb250Q2FjaGVTaXplXTsKICBpbnQgblQzRm9udHM7CQkJLy8gbnVtYmVyIG9mIHZhbGlkIGVudHJpZXMgaW4gdDNGb250Q2FjaGUKICBUM0dseXBoU3RhY2sgKnQzR2x5cGhTdGFjazsJLy8gVHlwZSAzIGdseXBoIGNvbnRleHQgc3RhY2sKICBHQm9vbCBoYXZlVDNEeDsJCS8vIHNldCBhZnRlciBzZWVpbmcgYSBkMC9kMSBvcGVyYXRvcgoKICBTcGxhc2hGb250ICpmb250OwkJLy8gY3VycmVudCBmb250CiAgR0Jvb2wgbmVlZEZvbnRVcGRhdGU7CQkvLyBzZXQgd2hlbiB0aGUgZm9udCBuZWVkcyB0byBiZSB1cGRhdGVkCiAgU3BsYXNoUGF0aCAqdGV4dENsaXBQYXRoOwkvLyBjbGlwcGluZyBwYXRoIGJ1aWx0IHdpdGggdGV4dCBvYmplY3QKCiAgU3BsYXNoVHJhbnNwYXJlbmN5R3JvdXAgKgkvLyB0cmFuc3BhcmVuY3kgZ3JvdXAgc3RhY2sKICAgIHRyYW5zcEdyb3VwU3RhY2s7CiAgU3BsYXNoQml0bWFwICptYXNrQml0bWFwOyAvLyBmb3IgaW1hZ2UgbWFza3MgaW4gcGF0dGVybiBjb2xvcnNwYWNlCiAgaW50IG5lc3RDb3VudDsKfTsKCiNlbmRpZgo=