LyoKICogQ29weXJpZ2h0IDIwMDItMjAwMyBNaWNoYWVsIEf8bm5ld2lnCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKICogbW9kaWZ5IGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIKICogdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAogKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgogKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlIEdOVQogKiBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgogKgogKiBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCiAqIExpY2Vuc2UgYWxvbmcgd2l0aCB0aGlzIGxpYnJhcnk7IGlmIG5vdCwgd3JpdGUgdG8gdGhlIEZyZWUgU29mdHdhcmUKICogRm91bmRhdGlvbiwgSW5jLiwgNTEgRnJhbmtsaW4gU3QsIEZpZnRoIEZsb29yLCBCb3N0b24sIE1BIDAyMTEwLTEzMDEsIFVTQQogKi8KCiNpbmNsdWRlIDxzdGRhcmcuaD4KCiNpbmNsdWRlICJ3aW5kZWYuaCIKI2luY2x1ZGUgIndpbmJhc2UuaCIKI2luY2x1ZGUgIndpbmdkaS5oIgojaW5jbHVkZSAid2ludXNlci5oIgojaW5jbHVkZSAidmZ3LmgiCgojaW5jbHVkZSAiYXZpZmlsZV9wcml2YXRlLmgiCgojaW5jbHVkZSAid2luZS9kZWJ1Zy5oIgoKV0lORV9ERUZBVUxUX0RFQlVHX0NIQU5ORUwoYXZpZmlsZSk7CgojaWZuZGVmIERJQlBUUgojZGVmaW5lIERJQlBUUihscCkgICAgICAoKExQQllURSkobHApICsgKGxwKS0+YmlTaXplICsgXAogICAgICAgICAgICAgICAgICAgICAgICAgKGxwKS0+YmlDbHJVc2VkICogc2l6ZW9mKFJHQlFVQUQpKQojZW5kaWYKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJR2V0RnJhbWVfZm5RdWVyeUludGVyZmFjZShJR2V0RnJhbWUgKmlmYWNlLAoJCQkJCQkgUkVGSUlEIHJlZmlpZCwgTFBWT0lEICpvYmopOwpzdGF0aWMgVUxPTkcgICBXSU5BUEkgSUdldEZyYW1lX2ZuQWRkUmVmKElHZXRGcmFtZSAqaWZhY2UpOwpzdGF0aWMgVUxPTkcgICBXSU5BUEkgSUdldEZyYW1lX2ZuUmVsZWFzZShJR2V0RnJhbWUgKmlmYWNlKTsKc3RhdGljIExQVk9JRCAgV0lOQVBJIElHZXRGcmFtZV9mbkdldEZyYW1lKElHZXRGcmFtZSAqaWZhY2UsIExPTkcgbFBvcyk7CnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJR2V0RnJhbWVfZm5CZWdpbihJR2V0RnJhbWUgKmlmYWNlLCBMT05HIGxTdGFydCwKCQkJCQlMT05HIGxFbmQsIExPTkcgbFJhdGUpOwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSUdldEZyYW1lX2ZuRW5kKElHZXRGcmFtZSAqaWZhY2UpOwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSUdldEZyYW1lX2ZuU2V0Rm9ybWF0KElHZXRGcmFtZSAqaWZhY2UsCgkJCQkJICAgIExQQklUTUFQSU5GT0hFQURFUiBscGJpLAoJCQkJCSAgICBMUFZPSUQgbHBCaXRzLCBJTlQgeCwgSU5UIHksCgkJCQkJICAgIElOVCBkeCwgSU5UIGR5KTsKCnN0YXRpYyBjb25zdCBzdHJ1Y3QgSUdldEZyYW1lVnRibCBpZ2V0ZnJhbWVWdGJsID0gewogIElHZXRGcmFtZV9mblF1ZXJ5SW50ZXJmYWNlLAogIElHZXRGcmFtZV9mbkFkZFJlZiwKICBJR2V0RnJhbWVfZm5SZWxlYXNlLAogIElHZXRGcmFtZV9mbkdldEZyYW1lLAogIElHZXRGcmFtZV9mbkJlZ2luLAogIElHZXRGcmFtZV9mbkVuZCwKICBJR2V0RnJhbWVfZm5TZXRGb3JtYXQKfTsKCnR5cGVkZWYgc3RydWN0IF9JR2V0RnJhbWVJbXBsIHsKICAvKiBJVW5rbm93biBzdHVmZiAqLwogIGNvbnN0IElHZXRGcmFtZVZ0YmwgKmxwVnRibDsKICBMT05HICAgICAgICAgICAgICAgcmVmOwoKICAvKiBJR2V0RnJhbWUgc3R1ZmYgKi8KICBCT09MICAgICAgICAgICAgICAgYkZpeGVkU3RyZWFtOwogIFBBVklTVFJFQU0gICAgICAgICBwU3RyZWFtOwoKICBMUFZPSUQgICAgICAgICAgICAgbHBJbkJ1ZmZlcjsKICBMT05HICAgICAgICAgICAgICAgY2JJbkJ1ZmZlcjsKICBMUEJJVE1BUElORk9IRUFERVIgbHBJbkZvcm1hdDsKICBMT05HICAgICAgICAgICAgICAgY2JJbkZvcm1hdDsKCiAgTE9ORyAgICAgICAgICAgICAgIGxDdXJyZW50RnJhbWU7CiAgTFBCSVRNQVBJTkZPSEVBREVSIGxwT3V0Rm9ybWF0OwogIExQVk9JRCAgICAgICAgICAgICBscE91dEJ1ZmZlcjsKCiAgSElDICAgICAgICAgICAgICAgIGhpYzsKICBCT09MICAgICAgICAgICAgICAgYlJlc2l6ZTsKICBEV09SRCAgICAgICAgICAgICAgeDsKICBEV09SRCAgICAgICAgICAgICAgeTsKICBEV09SRCAgICAgICAgICAgICAgZHg7CiAgRFdPUkQgICAgICAgICAgICAgIGR5OwoKICBCT09MICAgICAgICAgICAgICAgYkZvcm1hdENoYW5nZXM7CiAgRFdPUkQgICAgICAgICAgICAgIGR3Rm9ybWF0Q2hhbmdlQ291bnQ7CiAgRFdPUkQgICAgICAgICAgICAgIGR3RWRpdENvdW50Owp9IElHZXRGcmFtZUltcGw7CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCgpzdGF0aWMgdm9pZCBBVklGSUxFX0Nsb3NlQ29tcHJlc3NvcihJR2V0RnJhbWVJbXBsICpUaGlzKQp7CiAgaWYgKFRoaXMtPmxwSW5Gb3JtYXQgIT0gVGhpcy0+bHBPdXRGb3JtYXQpIHsKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIFRoaXMtPmxwT3V0Rm9ybWF0KTsKICAgIFRoaXMtPmxwT3V0Rm9ybWF0ID0gTlVMTDsKICB9CiAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgVGhpcy0+bHBJbkZvcm1hdCk7CiAgVGhpcy0+bHBJbkZvcm1hdCA9IE5VTEw7CiAgaWYgKFRoaXMtPmhpYyAhPSBOVUxMKSB7CiAgICBpZiAoVGhpcy0+YlJlc2l6ZSkKICAgICAgSUNEZWNvbXByZXNzRXhFbmQoVGhpcy0+aGljKTsKICAgIGVsc2UKICAgICAgSUNEZWNvbXByZXNzRW5kKFRoaXMtPmhpYyk7CiAgICBJQ0Nsb3NlKFRoaXMtPmhpYyk7CiAgICBUaGlzLT5oaWMgPSBOVUxMOwogIH0KfQoKUEdFVEZSQU1FIEFWSUZJTEVfQ3JlYXRlR2V0RnJhbWUoUEFWSVNUUkVBTSBwU3RyZWFtKQp7CiAgSUdldEZyYW1lSW1wbCAqcGc7CgogIC8qIGNoZWNrIHBhcmFtZXRlciAqLwogIGlmIChwU3RyZWFtID09IE5VTEwpCiAgICByZXR1cm4gTlVMTDsKCiAgcGcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgSEVBUF9aRVJPX01FTU9SWSwgc2l6ZW9mKElHZXRGcmFtZUltcGwpKTsKICBpZiAocGcgIT0gTlVMTCkgewogICAgcGctPmxwVnRibCAgICAgICAgPSAmaWdldGZyYW1lVnRibDsKICAgIHBnLT5yZWYgICAgICAgICAgID0gMTsKICAgIHBnLT5sQ3VycmVudEZyYW1lID0gLTE7CiAgICBwZy0+cFN0cmVhbSAgICAgICA9IHBTdHJlYW07CiAgICBJQVZJU3RyZWFtX0FkZFJlZihwU3RyZWFtKTsKICB9CgogIHJldHVybiAoUEdFVEZSQU1FKXBnOwp9CgpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSUdldEZyYW1lX2ZuUXVlcnlJbnRlcmZhY2UoSUdldEZyYW1lICppZmFjZSwKCQkJCQkJIFJFRklJRCByZWZpaWQsIExQVk9JRCAqb2JqKQp7CiAgSUdldEZyYW1lSW1wbCAqVGhpcyA9IChJR2V0RnJhbWVJbXBsICopaWZhY2U7CgogIFRSQUNFKCIoJXAsJXMsJXApXG4iLCBUaGlzLCBkZWJ1Z3N0cl9ndWlkKHJlZmlpZCksIG9iaik7CgogIGlmIChJc0VxdWFsR1VJRCgmSUlEX0lVbmtub3duLCByZWZpaWQpIHx8CiAgICAgIElzRXF1YWxHVUlEKCZJSURfSUdldEZyYW1lLCByZWZpaWQpKSB7CiAgICAqb2JqID0gaWZhY2U7CiAgICByZXR1cm4gU19PSzsKICB9CgogIHJldHVybiBPTEVfRV9FTlVNX05PTU9SRTsKfQoKc3RhdGljIFVMT05HICAgV0lOQVBJIElHZXRGcmFtZV9mbkFkZFJlZihJR2V0RnJhbWUgKmlmYWNlKQp7CiAgSUdldEZyYW1lSW1wbCAqVGhpcyA9IChJR2V0RnJhbWVJbXBsICopaWZhY2U7CiAgVUxPTkcgcmVmID0gSW50ZXJsb2NrZWRJbmNyZW1lbnQoJlRoaXMtPnJlZik7CgogIFRSQUNFKCIoJXApXG4iLCBpZmFjZSk7CgogIHJldHVybiByZWY7Cn0KCnN0YXRpYyBVTE9ORyAgIFdJTkFQSSBJR2V0RnJhbWVfZm5SZWxlYXNlKElHZXRGcmFtZSAqaWZhY2UpCnsKICBJR2V0RnJhbWVJbXBsICpUaGlzID0gKElHZXRGcmFtZUltcGwgKilpZmFjZTsKICBVTE9ORyByZWYgPSBJbnRlcmxvY2tlZERlY3JlbWVudCgmVGhpcy0+cmVmKTsKCiAgVFJBQ0UoIiglcClcbiIsIGlmYWNlKTsKCiAgaWYgKCFyZWYpIHsKICAgIEFWSUZJTEVfQ2xvc2VDb21wcmVzc29yKFRoaXMpOwogICAgaWYgKFRoaXMtPnBTdHJlYW0gIT0gTlVMTCkgewogICAgICBJQVZJU3RyZWFtX1JlbGVhc2UoVGhpcy0+cFN0cmVhbSk7CiAgICAgIFRoaXMtPnBTdHJlYW0gPSBOVUxMOwogICAgfQoKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGlmYWNlKTsKICAgIHJldHVybiAwOwogIH0KCiAgcmV0dXJuIHJlZjsKfQoKc3RhdGljIExQVk9JRCAgV0lOQVBJIElHZXRGcmFtZV9mbkdldEZyYW1lKElHZXRGcmFtZSAqaWZhY2UsIExPTkcgbFBvcykKewogIElHZXRGcmFtZUltcGwgKlRoaXMgPSAoSUdldEZyYW1lSW1wbCAqKWlmYWNlOwoKICBMT05HIHJlYWRCeXRlczsKICBMT05HIHJlYWRTYW1wbGVzOwoKICBUUkFDRSgiKCVwLCVkKVxuIiwgaWZhY2UsIGxQb3MpOwoKICAvKiBXZSBkb24ndCB3YW50IG5lZ2F0aXZlIHN0YXJ0IHZhbHVlcyEgLS0gbWFya3MgaW52YWxpZCBidWZmZXIgY29udGVudCAqLwogIGlmIChsUG9zIDwgMCkKICAgIHJldHVybiBOVUxMOwoKICAvKiBjaGVjayBzdGF0ZSAqLwogIGlmIChUaGlzLT5wU3RyZWFtID09IE5VTEwpCiAgICByZXR1cm4gTlVMTDsKICBpZiAoVGhpcy0+bHBJbkZvcm1hdCA9PSBOVUxMKQogICAgcmV0dXJuIE5VTEw7CgogIC8qIENvdWxkIHN0cmVhbSBoYXZlIGNoYW5nZWQ/ICovCiAgaWYgKCEgVGhpcy0+YkZpeGVkU3RyZWFtKSB7CiAgICBBVklTVFJFQU1JTkZPVyBzSW5mbzsKCiAgICBJQVZJU3RyZWFtX0luZm8oVGhpcy0+cFN0cmVhbSwgJnNJbmZvLCBzaXplb2Yoc0luZm8pKTsKCiAgICBpZiAoc0luZm8uZHdFZGl0Q291bnQgIT0gVGhpcy0+ZHdFZGl0Q291bnQpIHsKICAgICAgVGhpcy0+ZHdFZGl0Q291bnQgICA9IHNJbmZvLmR3RWRpdENvdW50OwogICAgICBUaGlzLT5sQ3VycmVudEZyYW1lID0gLTE7CiAgICB9CgogICAgaWYgKHNJbmZvLmR3Rm9ybWF0Q2hhbmdlQ291bnQgIT0gVGhpcy0+ZHdGb3JtYXRDaGFuZ2VDb3VudCkgewogICAgICAvKiBzdHJlYW0gaGFzIGNoYW5nZWQgKi8KICAgICAgaWYgKFRoaXMtPmxwT3V0Rm9ybWF0ICE9IE5VTEwpIHsKCUJJVE1BUElORk9IRUFERVIgYmk7CgoJbWVtY3B5KCZiaSwgVGhpcy0+bHBPdXRGb3JtYXQsIHNpemVvZihiaSkpOwoJQVZJRklMRV9DbG9zZUNvbXByZXNzb3IoVGhpcyk7CgoJaWYgKEZBSUxFRChJR2V0RnJhbWVfU2V0Rm9ybWF0KGlmYWNlLCAmYmksIE5VTEwsIDAsIDAsIC0xLCAtMSkpKSB7CgkgIGlmIChGQUlMRUQoSUdldEZyYW1lX1NldEZvcm1hdChpZmFjZSwgTlVMTCwgTlVMTCwgMCwgMCwgLTEsIC0xKSkpCgkgICAgcmV0dXJuIE5VTEw7Cgl9CiAgICAgIH0gZWxzZSBpZiAoRkFJTEVEKElHZXRGcmFtZV9TZXRGb3JtYXQoaWZhY2UsIE5VTEwsIE5VTEwsIDAsIDAsIC0xLCAtMSkpKQoJcmV0dXJuIE5VTEw7CiAgICB9CiAgfQoKICBpZiAobFBvcyAhPSBUaGlzLT5sQ3VycmVudEZyYW1lKSB7CiAgICBMT05HIGxOZXh0ID0gSUFWSVN0cmVhbV9GaW5kU2FtcGxlKFRoaXMtPnBTdHJlYW0sbFBvcyxGSU5EX0tFWXxGSU5EX1BSRVYpOwoKICAgIGlmIChsTmV4dCA9PSAtMSkKICAgICAgcmV0dXJuIE5VTEw7IC8qIGZyYW1lIGRvZXNuJ3QgZXhpc3QgKi8KICAgIGlmIChsTmV4dCA8PSBUaGlzLT5sQ3VycmVudEZyYW1lICYmIFRoaXMtPmxDdXJyZW50RnJhbWUgPCBsUG9zKQogICAgICBsTmV4dCA9IFRoaXMtPmxDdXJyZW50RnJhbWUgKyAxOwoKICAgIGZvciAoOyBsTmV4dCA8PSBsUG9zOyBsTmV4dCsrKSB7CiAgICAgIC8qIG5ldyBmb3JtYXQgZm9yIHRoaXMgZnJhbWU/ICovCiAgICAgIGlmIChUaGlzLT5iRm9ybWF0Q2hhbmdlcykgewoJSUFWSVN0cmVhbV9SZWFkRm9ybWF0KFRoaXMtPnBTdHJlYW0sIGxOZXh0LAoJCQkgICAgICBUaGlzLT5scEluRm9ybWF0LCAmVGhpcy0+Y2JJbkZvcm1hdCk7CglpZiAoVGhpcy0+bHBPdXRGb3JtYXQgIT0gTlVMTCkgewoJICBpZiAoVGhpcy0+bHBPdXRGb3JtYXQtPmJpQml0Q291bnQgPD0gOCkKCSAgICBJQ0RlY29tcHJlc3NHZXRQYWxldHRlKFRoaXMtPmhpYywgVGhpcy0+bHBJbkZvcm1hdCwKCQkJCSAgIFRoaXMtPmxwT3V0Rm9ybWF0KTsKCX0KICAgICAgfQoKICAgICAgLyogcmVhZCBpbnB1dCBmcmFtZSAqLwogICAgICB3aGlsZSAoRkFJTEVEKEFWSVN0cmVhbVJlYWQoVGhpcy0+cFN0cmVhbSwgbE5leHQsIDEsIFRoaXMtPmxwSW5CdWZmZXIsCgkJCQkgIFRoaXMtPmNiSW5CdWZmZXIsICZyZWFkQnl0ZXMsICZyZWFkU2FtcGxlcykpKSB7CgkvKiBub3QgZW5vdWdoIG1lbW9yeSBmb3IgaW5wdXQgYnVmZmVyPyAqLwoJcmVhZEJ5dGVzID0gMDsKCWlmIChGQUlMRUQoQVZJU3RyZWFtU2FtcGxlU2l6ZShUaGlzLT5wU3RyZWFtLCBsTmV4dCwgJnJlYWRCeXRlcykpKQoJICByZXR1cm4gTlVMTDsgLyogYmFkIHRoaW5nLCBidXQgYmFkIHRoaW5ncyB3aWxsIGhhcHBlbiAqLwoJaWYgKHJlYWRCeXRlcyA8PSAwKSB7CgkgIEVSUigiOiBJQVZJU3RyZWFtOjpSRWFkIGRvZXNuJ3QgcmV0dXJuIG5lZWRlZCBieXRlcyFcbiIpOwoJICByZXR1cm4gTlVMTDsKCX0KCgkvKiBJQVZJU3RyZWFtOjpSZWFkIGZhaWxlZCBiZWNhdXNlIG9mIG90aGVyIHJlYXNvbnMgbm90IGJ1ZmZlcnNpemU/ICovCglpZiAoVGhpcy0+Y2JJbkJ1ZmZlciA+PSByZWFkQnl0ZXMpCgkgIGJyZWFrOwoJVGhpcy0+Y2JJbkJ1ZmZlciA9IFRoaXMtPmNiSW5Gb3JtYXQgKyByZWFkQnl0ZXM7CglUaGlzLT5scEluRm9ybWF0ID0gSGVhcFJlQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgVGhpcy0+bHBJbkZvcm1hdCwgVGhpcy0+Y2JJbkJ1ZmZlcik7CglpZiAoVGhpcy0+bHBJbkZvcm1hdCA9PSBOVUxMKQoJICByZXR1cm4gTlVMTDsgLyogb3V0IG9mIG1lbW9yeSAqLwoJVGhpcy0+bHBJbkJ1ZmZlciA9IChCWVRFKilUaGlzLT5scEluRm9ybWF0ICsgVGhpcy0+Y2JJbkZvcm1hdDsKICAgICAgfQoKICAgICAgaWYgKHJlYWRTYW1wbGVzICE9IDEpIHsKCUVSUigiOiBubyBmcmFtZXMgcmVhZFxuIik7CglyZXR1cm4gTlVMTDsKICAgICAgfQogICAgICBpZiAocmVhZEJ5dGVzICE9IDApIHsKCVRoaXMtPmxwSW5Gb3JtYXQtPmJpU2l6ZUltYWdlID0gcmVhZEJ5dGVzOwoKCS8qIG5vdGhpbmcgdG8gZGVjb21wcmVzcz8gKi8KCWlmIChUaGlzLT5oaWMgPT0gTlVMTCkgewoJICBUaGlzLT5sQ3VycmVudEZyYW1lID0gbFBvczsKCSAgcmV0dXJuIFRoaXMtPmxwSW5Gb3JtYXQ7Cgl9CgoJaWYgKFRoaXMtPmJSZXNpemUpIHsKCSAgSUNEZWNvbXByZXNzRXgoVGhpcy0+aGljLDAsVGhpcy0+bHBJbkZvcm1hdCxUaGlzLT5scEluQnVmZmVyLDAsMCwKCQkJIFRoaXMtPmxwSW5Gb3JtYXQtPmJpV2lkdGgsVGhpcy0+bHBJbkZvcm1hdC0+YmlIZWlnaHQsCgkJCSBUaGlzLT5scE91dEZvcm1hdCxUaGlzLT5scE91dEJ1ZmZlcixUaGlzLT54LFRoaXMtPnksCgkJCSBUaGlzLT5keCxUaGlzLT5keSk7Cgl9IGVsc2UgewoJICBJQ0RlY29tcHJlc3MoVGhpcy0+aGljLCAwLCBUaGlzLT5scEluRm9ybWF0LCBUaGlzLT5scEluQnVmZmVyLAoJCSAgICAgICBUaGlzLT5scE91dEZvcm1hdCwgVGhpcy0+bHBPdXRCdWZmZXIpOwoJfQogICAgICB9CiAgICB9IC8qIGZvciAobE5leHQgPCBsUG9zKSAqLwogIH0gLyogaWYgKFRoaXMtPmxDdXJyZW50RnJhbWUgIT0gbFBvcykgKi8KCiAgcmV0dXJuIChUaGlzLT5oaWMgPT0gTlVMTCA/IFRoaXMtPmxwSW5Gb3JtYXQgOiBUaGlzLT5scE91dEZvcm1hdCk7Cn0KCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJR2V0RnJhbWVfZm5CZWdpbihJR2V0RnJhbWUgKmlmYWNlLCBMT05HIGxTdGFydCwKCQkJCQlMT05HIGxFbmQsIExPTkcgbFJhdGUpCnsKICBJR2V0RnJhbWVJbXBsICpUaGlzID0gKElHZXRGcmFtZUltcGwgKilpZmFjZTsKCiAgVFJBQ0UoIiglcCwlZCwlZCwlZClcbiIsIGlmYWNlLCBsU3RhcnQsIGxFbmQsIGxSYXRlKTsKCiAgVGhpcy0+YkZpeGVkU3RyZWFtID0gVFJVRTsKCiAgcmV0dXJuIChJR2V0RnJhbWVfR2V0RnJhbWUoaWZhY2UsIGxTdGFydCkgPyBBVklFUlJfT0sgOiBBVklFUlJfRVJST1IpOwp9CgpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSUdldEZyYW1lX2ZuRW5kKElHZXRGcmFtZSAqaWZhY2UpCnsKICBJR2V0RnJhbWVJbXBsICpUaGlzID0gKElHZXRGcmFtZUltcGwgKilpZmFjZTsKCiAgVFJBQ0UoIiglcClcbiIsIGlmYWNlKTsKCiAgVGhpcy0+YkZpeGVkU3RyZWFtID0gRkFMU0U7CgogIHJldHVybiBBVklFUlJfT0s7Cn0KCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJR2V0RnJhbWVfZm5TZXRGb3JtYXQoSUdldEZyYW1lICppZmFjZSwKCQkJCQkgICAgTFBCSVRNQVBJTkZPSEVBREVSIGxwYmlXYW50ZWQsCgkJCQkJICAgIExQVk9JRCBscEJpdHMsIElOVCB4LCBJTlQgeSwKCQkJCQkgICAgSU5UIGR4LCBJTlQgZHkpCnsKICBJR2V0RnJhbWVJbXBsICpUaGlzID0gKElHZXRGcmFtZUltcGwgKilpZmFjZTsKCiAgQVZJU1RSRUFNSU5GT1cgICAgIHNJbmZvOwogIExQQklUTUFQSU5GT0hFQURFUiBscGJpICAgICAgICAgPSBscGJpV2FudGVkOwogIEJPT0wgICAgICAgICAgICAgICBiQmVzdERpc3BsYXkgPSBGQUxTRTsKCiAgVFJBQ0UoIiglcCwlcCwlcCwlZCwlZCwlZCwlZClcbiIsIGlmYWNlLCBscGJpV2FudGVkLCBscEJpdHMsCgl4LCB5LCBkeCwgZHkpOwoKICBpZiAoVGhpcy0+cFN0cmVhbSA9PSBOVUxMKQogICAgcmV0dXJuIEFWSUVSUl9FUlJPUjsKCiAgaWYgKGxwYmlXYW50ZWQgPT0gKExQQklUTUFQSU5GT0hFQURFUilBVklHRVRGUkFNRUZfQkVTVERJU1BMQVlGTVQpIHsKICAgIGxwYmkgPSBOVUxMOwogICAgYkJlc3REaXNwbGF5ID0gVFJVRTsKICB9CgogIElBVklTdHJlYW1fSW5mbyhUaGlzLT5wU3RyZWFtLCAmc0luZm8sIHNpemVvZihzSW5mbykpOwogIGlmIChzSW5mby5mY2NUeXBlICE9IHN0cmVhbXR5cGVWSURFTykKICAgIHJldHVybiBBVklFUlJfVU5TVVBQT1JURUQ7CgogIFRoaXMtPmJGb3JtYXRDaGFuZ2VzID0KICAgIChzSW5mby5kd0ZsYWdzICYgQVZJU1RSRUFNSU5GT19GT1JNQVRDSEFOR0VTID8gVFJVRSA6IEZBTFNFICk7CiAgVGhpcy0+ZHdGb3JtYXRDaGFuZ2VDb3VudCA9IHNJbmZvLmR3Rm9ybWF0Q2hhbmdlQ291bnQ7CiAgVGhpcy0+ZHdFZGl0Q291bnQgICAgICAgICA9IHNJbmZvLmR3RWRpdENvdW50OwogIFRoaXMtPmxDdXJyZW50RnJhbWUgICAgICAgPSAtMTsKCiAgLyogZ2V0IGlucHV0IGZvcm1hdCBmcm9tIHN0cmVhbSAqLwogIGlmIChUaGlzLT5scEluRm9ybWF0ID09IE5VTEwpIHsKICAgIEhSRVNVTFQgaHI7CgogICAgVGhpcy0+Y2JJbkJ1ZmZlciA9IChMT05HKXNJbmZvLmR3U3VnZ2VzdGVkQnVmZmVyU2l6ZTsKICAgIGlmIChUaGlzLT5jYkluQnVmZmVyID09IDApCiAgICAgIFRoaXMtPmNiSW5CdWZmZXIgPSAxMDI0OwoKICAgIElBVklTdHJlYW1fUmVhZEZvcm1hdChUaGlzLT5wU3RyZWFtLCBzSW5mby5kd1N0YXJ0LAoJCQkgIE5VTEwsICZUaGlzLT5jYkluRm9ybWF0KTsKCiAgICBUaGlzLT5scEluRm9ybWF0ID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIFRoaXMtPmNiSW5Gb3JtYXQgKyBUaGlzLT5jYkluQnVmZmVyKTsKICAgIGlmIChUaGlzLT5scEluRm9ybWF0ID09IE5VTEwpIHsKICAgICAgQVZJRklMRV9DbG9zZUNvbXByZXNzb3IoVGhpcyk7CiAgICAgIHJldHVybiBBVklFUlJfTUVNT1JZOwogICAgfQoKICAgIGhyID0gSUFWSVN0cmVhbV9SZWFkRm9ybWF0KFRoaXMtPnBTdHJlYW0sIHNJbmZvLmR3U3RhcnQsIFRoaXMtPmxwSW5Gb3JtYXQsICZUaGlzLT5jYkluRm9ybWF0KTsKICAgIGlmIChGQUlMRUQoaHIpKSB7CiAgICAgIEFWSUZJTEVfQ2xvc2VDb21wcmVzc29yKFRoaXMpOwogICAgICByZXR1cm4gaHI7CiAgICB9CgogICAgVGhpcy0+bHBJbkJ1ZmZlciA9ICgoTFBCWVRFKVRoaXMtPmxwSW5Gb3JtYXQpICsgVGhpcy0+Y2JJbkZvcm1hdDsKICB9CgogIC8qIGNoZWNrIGlucHV0IGZvcm1hdCAqLwogIGlmIChUaGlzLT5scEluRm9ybWF0LT5iaUNsclVzZWQgPT0gMCAmJiBUaGlzLT5scEluRm9ybWF0LT5iaUJpdENvdW50IDw9IDgpCiAgICBUaGlzLT5scEluRm9ybWF0LT5iaUNsclVzZWQgPSAxdSA8PCBUaGlzLT5scEluRm9ybWF0LT5iaUJpdENvdW50OwogIGlmIChUaGlzLT5scEluRm9ybWF0LT5iaVNpemVJbWFnZSA9PSAwICYmCiAgICAgIFRoaXMtPmxwSW5Gb3JtYXQtPmJpQ29tcHJlc3Npb24gPT0gQklfUkdCKSB7CiAgICBUaGlzLT5scEluRm9ybWF0LT5iaVNpemVJbWFnZSA9CiAgICAgIERJQldJRFRIQllURVMoKlRoaXMtPmxwSW5Gb3JtYXQpICogVGhpcy0+bHBJbkZvcm1hdC0+YmlIZWlnaHQ7CiAgfQoKICAvKiBvbmx5IHRvIHBhc3MgdGhyb3VnaD8gKi8KICBpZiAoVGhpcy0+bHBJbkZvcm1hdC0+YmlDb21wcmVzc2lvbiA9PSBCSV9SR0IgJiYgbHBCaXRzID09IE5VTEwpIHsKICAgIGlmIChscGJpID09IE5VTEwgfHwgCgkobHBiaS0+YmlDb21wcmVzc2lvbiA9PSBCSV9SR0IgJiYKCSBscGJpLT5iaVdpZHRoID09IFRoaXMtPmxwSW5Gb3JtYXQtPmJpV2lkdGggJiYKCSBscGJpLT5iaUhlaWdodCA9PSBUaGlzLT5scEluRm9ybWF0LT5iaUhlaWdodCAmJgoJIGxwYmktPmJpQml0Q291bnQgPT0gVGhpcy0+bHBJbkZvcm1hdC0+YmlCaXRDb3VudCkpIHsKICAgICAgVGhpcy0+bHBPdXRGb3JtYXQgPSBUaGlzLT5scEluRm9ybWF0OwogICAgICBUaGlzLT5scE91dEJ1ZmZlciA9IERJQlBUUihUaGlzLT5scEluRm9ybWF0KTsKICAgICAgcmV0dXJuIEFWSUVSUl9PSzsKICAgIH0KICB9CgogIC8qIG5lZWQgbWVtb3J5IGZvciBvdXRwdXQgZm9ybWF0PyAqLwogIGlmIChUaGlzLT5scE91dEZvcm1hdCA9PSBOVUxMKSB7CiAgICBUaGlzLT5scE91dEZvcm1hdCA9CiAgICAgIEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBzaXplb2YoQklUTUFQSU5GT0hFQURFUikgKyAyNTYgKiBzaXplb2YoUkdCUVVBRCkpOwogICAgaWYgKFRoaXMtPmxwT3V0Rm9ybWF0ID09IE5VTEwpIHsKICAgICAgQVZJRklMRV9DbG9zZUNvbXByZXNzb3IoVGhpcyk7CiAgICAgIHJldHVybiBBVklFUlJfTUVNT1JZOwogICAgfQogIH0KCiAgLyogbmVlZCBoYW5kbGUgdG8gdmlkZW8gY29tcHJlc3NvciAqLwogIGlmIChUaGlzLT5oaWMgPT0gTlVMTCkgewogICAgRk9VUkNDIGZjY0hhbmRsZXI7CgogICAgaWYgKFRoaXMtPmxwSW5Gb3JtYXQtPmJpQ29tcHJlc3Npb24gPT0gQklfUkdCKQogICAgICBmY2NIYW5kbGVyID0gY29tcHR5cGVESUI7CiAgICBlbHNlIGlmIChUaGlzLT5scEluRm9ybWF0LT5iaUNvbXByZXNzaW9uID09IEJJX1JMRTgpCiAgICAgIGZjY0hhbmRsZXIgPSBtbWlvRk9VUkNDKCdSJywnTCcsJ0UnLCcgJyk7CiAgICBlbHNlCiAgICAgIGZjY0hhbmRsZXIgPSBzSW5mby5mY2NIYW5kbGVyOwoKICAgIGlmIChscGJpICE9IE5VTEwpIHsKICAgICAgaWYgKGxwYmktPmJpV2lkdGggPT0gMCkKCWxwYmktPmJpV2lkdGggPSBUaGlzLT5scEluRm9ybWF0LT5iaVdpZHRoOwogICAgICBpZiAobHBiaS0+YmlIZWlnaHQgPT0gMCkKCWxwYmktPmJpSGVpZ2h0ID0gVGhpcy0+bHBJbkZvcm1hdC0+YmlIZWlnaHQ7CiAgICB9CgogICAgVGhpcy0+aGljID0gSUNMb2NhdGUoSUNUWVBFX1ZJREVPLCBmY2NIYW5kbGVyLCBUaGlzLT5scEluRm9ybWF0LCBscGJpLCBJQ01PREVfREVDT01QUkVTUyk7CiAgICBpZiAoVGhpcy0+aGljID09IE5VTEwpIHsKICAgICAgQVZJRklMRV9DbG9zZUNvbXByZXNzb3IoVGhpcyk7CiAgICAgIHJldHVybiBBVklFUlJfTk9DT01QUkVTU09SOwogICAgfQogIH0KCiAgLyogb3V0cHV0IGZvcm1hdCBnaXZlbj8gKi8KICBpZiAobHBiaSAhPSBOVUxMKSB7CiAgICAvKiBjaGVjayB0aGUgZ2l2ZW4gb3V0cHV0IGZvcm1hdCAuLi4gKi8KICAgIGlmIChscGJpLT5iaUNsclVzZWQgPT0gMCAmJiBscGJpLT5iaUJpdENvdW50IDw9IDgpCiAgICAgIGxwYmktPmJpQ2xyVXNlZCA9IDF1IDw8IGxwYmktPmJpQml0Q291bnQ7CgogICAgLyogLi4uIGFuZCByZW1lbWJlciBpdCAqLwogICAgbWVtY3B5KFRoaXMtPmxwT3V0Rm9ybWF0LCBscGJpLAoJICAgbHBiaS0+YmlTaXplICsgbHBiaS0+YmlDbHJVc2VkICogc2l6ZW9mKFJHQlFVQUQpKTsKICAgIGlmIChscGJpLT5iaUJpdENvdW50IDw9IDgpCiAgICAgIElDRGVjb21wcmVzc0dldFBhbGV0dGUoVGhpcy0+aGljLCBUaGlzLT5scEluRm9ybWF0LCBUaGlzLT5scE91dEZvcm1hdCk7CgogICAgcmV0dXJuIEFWSUVSUl9PSzsKICB9IGVsc2UgewogICAgaWYgKGJCZXN0RGlzcGxheSkgewogICAgICBJQ0dldERpc3BsYXlGb3JtYXQoVGhpcy0+aGljLCBUaGlzLT5scEluRm9ybWF0LAoJCQkgVGhpcy0+bHBPdXRGb3JtYXQsIDAsIGR4LCBkeSk7CiAgICB9IGVsc2UgaWYgKElDRGVjb21wcmVzc0dldEZvcm1hdChUaGlzLT5oaWMsIFRoaXMtPmxwSW5Gb3JtYXQsCgkJCQkgICAgIFRoaXMtPmxwT3V0Rm9ybWF0KSA8IDApIHsKICAgICAgQVZJRklMRV9DbG9zZUNvbXByZXNzb3IoVGhpcyk7CiAgICAgIHJldHVybiBBVklFUlJfTk9DT01QUkVTU09SOwogICAgfQoKICAgIC8qIGNoZWNrIG91dHB1dCBmb3JtYXQgKi8KICAgIGlmIChUaGlzLT5scE91dEZvcm1hdC0+YmlDbHJVc2VkID09IDAgJiYKCVRoaXMtPmxwT3V0Rm9ybWF0LT5iaUJpdENvdW50IDw9IDgpCiAgICAgIFRoaXMtPmxwT3V0Rm9ybWF0LT5iaUNsclVzZWQgPSAxdSA8PCBUaGlzLT5scE91dEZvcm1hdC0+YmlCaXRDb3VudDsKICAgIGlmIChUaGlzLT5scE91dEZvcm1hdC0+YmlTaXplSW1hZ2UgPT0gMCAmJgoJVGhpcy0+bHBPdXRGb3JtYXQtPmJpQ29tcHJlc3Npb24gPT0gQklfUkdCKSB7CiAgICAgIFRoaXMtPmxwT3V0Rm9ybWF0LT5iaVNpemVJbWFnZSA9CglESUJXSURUSEJZVEVTKCpUaGlzLT5scE91dEZvcm1hdCkgKiBUaGlzLT5scE91dEZvcm1hdC0+YmlIZWlnaHQ7CiAgICB9CgogICAgaWYgKGxwQml0cyA9PSBOVUxMKSB7CiAgICAgIHJlZ2lzdGVyIERXT1JEIHNpemUgPSBUaGlzLT5scE91dEZvcm1hdC0+YmlDbHJVc2VkICogc2l6ZW9mKFJHQlFVQUQpOwoKICAgICAgc2l6ZSArPSBUaGlzLT5scE91dEZvcm1hdC0+YmlTaXplICsgVGhpcy0+bHBPdXRGb3JtYXQtPmJpU2l6ZUltYWdlOwogICAgICBUaGlzLT5scE91dEZvcm1hdCA9IEhlYXBSZUFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIFRoaXMtPmxwT3V0Rm9ybWF0LCBzaXplKTsKICAgICAgaWYgKFRoaXMtPmxwT3V0Rm9ybWF0ID09IE5VTEwpIHsKCUFWSUZJTEVfQ2xvc2VDb21wcmVzc29yKFRoaXMpOwoJcmV0dXJuIEFWSUVSUl9NRU1PUlk7CiAgICAgIH0KICAgICAgVGhpcy0+bHBPdXRCdWZmZXIgPSBESUJQVFIoVGhpcy0+bHBPdXRGb3JtYXQpOwogICAgfSBlbHNlCiAgICAgIFRoaXMtPmxwT3V0QnVmZmVyID0gbHBCaXRzOwoKICAgIC8qIGZvciB1c2VyIHNpemUgd2FzIGlycmVsZXZhbnQgKi8KICAgIGlmIChkeCA9PSAtMSkKICAgICAgZHggPSBUaGlzLT5scE91dEZvcm1hdC0+YmlXaWR0aDsKICAgIGlmIChkeSA9PSAtMSkKICAgICAgZHkgPSBUaGlzLT5scE91dEZvcm1hdC0+YmlIZWlnaHQ7CgogICAgLyogbmVlZCB0byByZXNpemU/ICovCiAgICBpZiAoeCAhPSAwIHx8IHkgIT0gMCkgewogICAgICBpZiAoZHkgPT0gVGhpcy0+bHBPdXRGb3JtYXQtPmJpSGVpZ2h0ICYmCgkgIGR4ID09IFRoaXMtPmxwT3V0Rm9ybWF0LT5iaVdpZHRoKQoJVGhpcy0+YlJlc2l6ZSA9IEZBTFNFOwogICAgICBlbHNlCglUaGlzLT5iUmVzaXplID0gVFJVRTsKICAgIH0KCiAgICBpZiAoVGhpcy0+YlJlc2l6ZSkgewogICAgICBUaGlzLT54ICA9IHg7CiAgICAgIFRoaXMtPnkgID0geTsKICAgICAgVGhpcy0+ZHggPSBkeDsKICAgICAgVGhpcy0+ZHkgPSBkeTsKCiAgICAgIGlmIChJQ0RlY29tcHJlc3NFeEJlZ2luKFRoaXMtPmhpYywwLFRoaXMtPmxwSW5Gb3JtYXQsVGhpcy0+bHBJbkJ1ZmZlciwwLAoJCQkgICAgICAwLFRoaXMtPmxwSW5Gb3JtYXQtPmJpV2lkdGgsCgkJCSAgICAgIFRoaXMtPmxwSW5Gb3JtYXQtPmJpSGVpZ2h0LFRoaXMtPmxwT3V0Rm9ybWF0LAoJCQkgICAgICBUaGlzLT5scE91dEJ1ZmZlciwgeCwgeSwgZHgsIGR5KSA9PSBJQ0VSUl9PSykKCXJldHVybiBBVklFUlJfT0s7CiAgICB9IGVsc2UgaWYgKElDRGVjb21wcmVzc0JlZ2luKFRoaXMtPmhpYywgVGhpcy0+bHBJbkZvcm1hdCwKCQkJCSBUaGlzLT5scE91dEZvcm1hdCkgPT0gSUNFUlJfT0spCiAgICAgIHJldHVybiBBVklFUlJfT0s7CgogICAgQVZJRklMRV9DbG9zZUNvbXByZXNzb3IoVGhpcyk7CgogICAgcmV0dXJuIEFWSUVSUl9DT01QUkVTU09SOwogIH0KfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwo=