Lyogdmk6IHNldCBzdz00IHRzPTQ6ICovCi8qCiAqICByZWFkcHJvZmlsZS5jIC0gdXNlZCB0byByZWFkIC9wcm9jL3Byb2ZpbGUKICoKICogIENvcHlyaWdodCAoQykgMTk5NCwxOTk2IEFsZXNzYW5kcm8gUnViaW5pIChydWJpbmlAaXB2dmlzLnVuaXB2Lml0KQogKgogKiAgIFRoaXMgcHJvZ3JhbSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IgbW9kaWZ5CiAqICAgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkKICogICB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAyIG9mIHRoZSBMaWNlbnNlLCBvcgogKiAgIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiAqCiAqICAgVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCiAqICAgYnV0IFdJVEhPVVQgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YKICogICBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlCiAqICAgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KICoKICogICBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQogKiAgIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtOyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCiAqICAgRm91bmRhdGlvbiwgSW5jLiwgNjc1IE1hc3MgQXZlLCBDYW1icmlkZ2UsIE1BIDAyMTM5LCBVU0EuCiAqLwoKLyoKICogMTk5OS0wMi0yMiBBcmthZGl1c3ogTWm2a2lld2ljeiA8bWlzaWVrQHBsZC5PUkcuUEw+CiAqIC0gYWRkZWQgTmF0aXZlIExhbmd1YWdlIFN1cHBvcnQKICogMTk5OS0wOS0wMSBTdGVwaGFuZSBFcmFuaWFuIDxlcmFuaWFuQGNlbGxvLmhwbC5ocC5jb20+CiAqIC0gNjRiaXQgY2xlYW4gcGF0Y2gKICogM0ZlYjIwMDEgQW5kcmV3IE1vcnRvbiA8YW5kcmV3bUB1b3cuZWR1LmF1PgogKiAtIC1NIG9wdGlvbiB0byB3cml0ZSBwcm9maWxlIG11bHRpcGxpZXIuCiAqIDIwMDEtMTEtMDcgV2VybmVyIEFsbWVzYmVyZ2VyIDx3YUBhbG1lc2Jlcmdlci5uZXQ+CiAqIC0gYnl0ZSBvcmRlciBhdXRvLWRldGVjdGlvbiBhbmQgLW4gb3B0aW9uCiAqIDIwMDEtMTEtMDkgV2VybmVyIEFsbWVzYmVyZ2VyIDx3YUBhbG1lc2Jlcmdlci5uZXQ+CiAqIC0gc2tpcCBzdGVwIHNpemUgKGluZGV4IDApCiAqIDIwMDItMDMtMDkgSm9obiBMZXZvbiA8bW96QGNvbXBzb2MubWFuLmFjLnVrPgogKiAtIG1ha2UgbWFwbGluZW5vIGRvIHNvbWV0aGluZwogKiAyMDAyLTExLTI4IE1hZHMgTWFydGluIEpvZXJnZW5zZW4gKwogKiAtIGFsc28gdHJ5IC9ib290L1N5c3RlbS5tYXAtYHVuYW1lIC1yYAogKiAyMDAzLTA0LTA5IFdlcm5lciBBbG1lc2JlcmdlciA8d2FAYWxtZXNiZXJnZXIubmV0PgogKiAtIGZpeGVkIG9mZi1ieSBlaWdodCBlcnJvciBhbmQgaW1wcm92ZWQgaGV1cmlzdGljcyBpbiBieXRlIG9yZGVyIGRldGVjdGlvbgogKiAyMDAzLTA4LTEyIE5pa2l0YSBEYW5pbG92IDxOaWtpdGFATmFtZXN5cy5DT00+CiAqIC0gYWRkZWQgLXMgb3B0aW9uOyBleGFtcGxlIG9mIHVzZToKICogInJlYWRwcm9maWxlIC1zIC1tIC9ib290L1N5c3RlbS5tYXAtdGVzdCB8IGdyZXAgX19kX2xvb2t1cCB8IHNvcnQgLW4gLWszIgogKgogKiBUYWtlbiBmcm9tIHV0aWwtbGludXggYW5kIGFkYXB0ZWQgZm9yIGJ1c3lib3ggYnkKICogUGF1bCBNdW5kdCA8bGV0aGFsQGxpbnV4LXNoLm9yZz4uCiAqLwoKI2luY2x1ZGUgPGVycm5vLmg+CiNpbmNsdWRlIDxzdGRpby5oPgojaW5jbHVkZSA8ZmNudGwuaD4KI2luY2x1ZGUgPHN0ZGxpYi5oPgojaW5jbHVkZSA8dW5pc3RkLmg+CiNpbmNsdWRlIDxzdHJpbmcuaD4KI2luY2x1ZGUgPHN5cy90eXBlcy5oPgojaW5jbHVkZSA8c3lzL3N0YXQuaD4KI2luY2x1ZGUgPHN5cy91dHNuYW1lLmg+CgojaW5jbHVkZSAiYnVzeWJveC5oIgoKI2RlZmluZSBTX0xFTiAxMjgKCi8qIFRoZXNlIGFyZSB0aGUgZGVmYXVsdHMgKi8Kc3RhdGljIGNvbnN0IGNoYXIgZGVmYXVsdG1hcFtdPSIvYm9vdC9TeXN0ZW0ubWFwIjsKc3RhdGljIGNvbnN0IGNoYXIgZGVmYXVsdHByb1tdPSIvcHJvYy9wcm9maWxlIjsKCmludCByZWFkcHJvZmlsZV9tYWluKGludCBhcmdjLCBjaGFyICoqYXJndikKewoJRklMRSAqbWFwOwoJaW50IHByb0ZkOwoJY29uc3QgY2hhciAqbWFwRmlsZSwgKnByb0ZpbGUsICptdWx0PTA7Cgl1bnNpZ25lZCBsb25nIGxlbj0wLCBpbmR4PTE7Cgl1bnNpZ25lZCBsb25nIGxvbmcgYWRkMD0wOwoJdW5zaWduZWQgaW50IHN0ZXA7Cgl1bnNpZ25lZCBpbnQgKmJ1ZiwgdG90YWwsIGZuX2xlbjsKCXVuc2lnbmVkIGxvbmcgbG9uZyBmbl9hZGQsIG5leHRfYWRkOyAgICAgICAgICAvKiBjdXJyZW50IGFuZCBuZXh0IGFkZHJlc3MgKi8KCWNoYXIgZm5fbmFtZVtTX0xFTl0sIG5leHRfbmFtZVtTX0xFTl07ICAgLyogY3VycmVudCBhbmQgbmV4dCBuYW1lICovCgljaGFyIG1vZGVbOF07CglpbnQgYzsKCWludCBvcHRBbGw9MCwgb3B0SW5mbz0wLCBvcHRSZXNldD0wLCBvcHRWZXJib3NlPTAsIG9wdE5hdGl2ZT0wOwoJaW50IG9wdEJpbnM9MCwgb3B0U3ViPTA7CgljaGFyIG1hcGxpbmVbU19MRU5dOwoJaW50IG1hcGxpbmVubz0xOwoJaW50IGhlYWRlcl9wcmludGVkOwoKI2RlZmluZSBuZXh0IChjdXJyZW50XjEpCgoJcHJvRmlsZSA9IGRlZmF1bHRwcm87CgltYXBGaWxlID0gZGVmYXVsdG1hcDsKCgl3aGlsZSAoKGMgPSBnZXRvcHQoYXJnYywgYXJndiwgIk06bTpucDppdHZhclZicyIpKSAhPSAtMSkgewoJCXN3aXRjaChjKSB7CgkJY2FzZSAnbSc6CgkJCW1hcEZpbGUgPSBvcHRhcmc7CgkJCWJyZWFrOwoJCWNhc2UgJ24nOgoJCQlvcHROYXRpdmUrKzsKCQkJYnJlYWs7CgkJY2FzZSAncCc6CgkJCXByb0ZpbGUgPSBvcHRhcmc7CgkJCWJyZWFrOwoJCWNhc2UgJ2EnOgoJCQlvcHRBbGwrKzsKCQkJYnJlYWs7CgkJY2FzZSAnYic6CgkJCW9wdEJpbnMrKzsKCQkJYnJlYWs7CgkJY2FzZSAncyc6CgkJCW9wdFN1YisrOwoJCQlicmVhazsKCQljYXNlICdpJzoKCQkJb3B0SW5mbysrOwoJCQlicmVhazsKCQljYXNlICdNJzoKCQkJbXVsdCA9IG9wdGFyZzsKCQkJYnJlYWs7CgkJY2FzZSAncic6CgkJCW9wdFJlc2V0Kys7CgkJCWJyZWFrOwoJCWNhc2UgJ3YnOgoJCQlvcHRWZXJib3NlKys7CgkJCWJyZWFrOwoJCWRlZmF1bHQ6CgkJCWJiX3Nob3dfdXNhZ2UoKTsKCQl9Cgl9CgoJaWYgKG9wdFJlc2V0IHx8IG11bHQpIHsKCQlpbnQgbXVsdGlwbGllciwgZmQsIHRvX3dyaXRlOwoKCQkvKgoJCSAqIFdoZW4gd3JpdGluZyB0aGUgbXVsdGlwbGllciwgaWYgdGhlIGxlbmd0aCBvZiB0aGUgd3JpdGUgaXMKCQkgKiBub3Qgc2l6ZW9mKGludCksIHRoZSBtdWx0aXBsaWVyIGlzIG5vdCBjaGFuZ2VkCgkJICovCgkJaWYgKG11bHQpIHsKCQkJbXVsdGlwbGllciA9IHN0cnRvdWwobXVsdCwgMCwgMTApOwoJCQl0b193cml0ZSA9IHNpemVvZihpbnQpOwoJCX0gZWxzZSB7CgkJCW11bHRpcGxpZXIgPSAwOwoJCQl0b193cml0ZSA9IDE7CS8qIHN0aCBkaWZmZXJlbnQgZnJvbSBzaXplb2YoaW50KSAqLwoJCX0KCgkJZmQgPSBiYl94b3BlbihkZWZhdWx0cHJvLE9fV1JPTkxZKTsKCgkJaWYgKHdyaXRlKGZkLCAmbXVsdGlwbGllciwgdG9fd3JpdGUpICE9IHRvX3dyaXRlKQoJCQliYl9wZXJyb3JfbXNnX2FuZF9kaWUoImVycm9yIHdyaXRpbmcgJXMiLCBkZWZhdWx0cHJvKTsKCgkJY2xvc2UoZmQpOwoJCXJldHVybiBFWElUX1NVQ0NFU1M7Cgl9CgoJLyoKCSAqIFVzZSBhbiBmZCBmb3IgdGhlIHByb2ZpbGluZyBidWZmZXIsIHRvIHNraXAgc3RkaW8gb3ZlcmhlYWQKCSAqLwoKCXByb0ZkID0gYmJfeG9wZW4ocHJvRmlsZSxPX1JET05MWSk7CgoJaWYgKCgoaW50KShsZW49bHNlZWsocHJvRmQsMCxTRUVLX0VORCkpIDwgMCkKCSAgICB8fCAobHNlZWsocHJvRmQsMCxTRUVLX1NFVCkgPCAwKSkKCQliYl9wZXJyb3JfbXNnX2FuZF9kaWUocHJvRmlsZSk7CgoJYnVmID0geG1hbGxvYyhsZW4pOwoKCWlmIChyZWFkKHByb0ZkLGJ1ZixsZW4pICE9IGxlbikKCQliYl9wZXJyb3JfbXNnX2FuZF9kaWUocHJvRmlsZSk7CgoJY2xvc2UocHJvRmQpOwoKCWlmICghb3B0TmF0aXZlKSB7CgkJaW50IGVudHJpZXMgPSBsZW4vc2l6ZW9mKCpidWYpOwoJCWludCBiaWcgPSAwLHNtYWxsID0gMCxpOwoJCXVuc2lnbmVkICpwOwoKCQlmb3IgKHAgPSBidWYrMTsgcCA8IGJ1ZitlbnRyaWVzOyBwKyspIHsKCQkJaWYgKCpwICYgfjBVIDw8IChzaXplb2YoKmJ1ZikqNCkpCgkJCQliaWcrKzsKCQkJaWYgKCpwICYgKCgxIDw8IChzaXplb2YoKmJ1ZikqNCkpLTEpKQoJCQkJc21hbGwrKzsKCQl9CgkJaWYgKGJpZyA+IHNtYWxsKSB7CgkJCWJiX2Vycm9yX21zZygiQXNzdW1pbmcgcmV2ZXJzZWQgYnl0ZSBvcmRlci4gIgoJCQkJIlVzZSAtbiB0byBmb3JjZSBuYXRpdmUgYnl0ZSBvcmRlci4iKTsKCQkJZm9yIChwID0gYnVmOyBwIDwgYnVmK2VudHJpZXM7IHArKykKCQkJCWZvciAoaSA9IDA7IGkgPCBzaXplb2YoKmJ1ZikvMjsgaSsrKSB7CgkJCQkJdW5zaWduZWQgY2hhciAqYiA9ICh1bnNpZ25lZCBjaGFyICopIHA7CgkJCQkJdW5zaWduZWQgY2hhciB0bXA7CgoJCQkJCXRtcCA9IGJbaV07CgkJCQkJYltpXSA9IGJbc2l6ZW9mKCpidWYpLWktMV07CgkJCQkJYltzaXplb2YoKmJ1ZiktaS0xXSA9IHRtcDsKCQkJCX0KCQl9Cgl9CgoJc3RlcCA9IGJ1ZlswXTsKCWlmIChvcHRJbmZvKSB7CgkJcHJpbnRmKCJTYW1wbGluZ19zdGVwOiAlaVxuIiwgc3RlcCk7CgkJcmV0dXJuIEVYSVRfU1VDQ0VTUzsKCX0KCgl0b3RhbCA9IDA7CgoJbWFwID0gYmJfeGZvcGVuKG1hcEZpbGUsICJyIik7CgoJd2hpbGUgKGZnZXRzKG1hcGxpbmUsU19MRU4sbWFwKSkgewoJCWlmIChzc2NhbmYobWFwbGluZSwiJWxseCAlcyAlcyIsJmZuX2FkZCxtb2RlLGZuX25hbWUpICE9IDMpCgkJCWJiX2Vycm9yX21zZ19hbmRfZGllKCIlcyglaSk6IHdyb25nIG1hcCBsaW5lIiwKCQkJCQkgICAgIG1hcEZpbGUsIG1hcGxpbmVubyk7CgoJCWlmICghc3RyY21wKGZuX25hbWUsIl9zdGV4dCIpKSAvKiBvbmx5IGVsZiB3b3JrcyBsaWtlIHRoaXMgKi8gewoJCQlhZGQwID0gZm5fYWRkOwoJCQlicmVhazsKCQl9CgkJbWFwbGluZW5vKys7Cgl9CgoJaWYgKCFhZGQwKQoJCWJiX2Vycm9yX21zZ19hbmRfZGllKCJjYW4ndCBmaW5kIFwiX3N0ZXh0XCIgaW4gJXMiLCBtYXBGaWxlKTsKCgkvKgoJICogTWFpbiBsb29wLgoJICovCgl3aGlsZSAoZmdldHMobWFwbGluZSxTX0xFTixtYXApKSB7CgkJdW5zaWduZWQgaW50IHRoaXMgPSAwOwoKCQlpZiAoc3NjYW5mKG1hcGxpbmUsIiVsbHggJXMgJXMiLCZuZXh0X2FkZCxtb2RlLG5leHRfbmFtZSkgIT0gMykKCQkJYmJfZXJyb3JfbXNnX2FuZF9kaWUoIiVzKCVpKTogd3JvbmcgbWFwIGxpbmUiLAoJCQkJCSAgICAgbWFwRmlsZSwgbWFwbGluZW5vKTsKCgkJaGVhZGVyX3ByaW50ZWQgPSAwOwoKCQkvKiBpZ25vcmUgYW55IExFQURJTkcgKGJlZm9yZSBhICdbdFRdJyBzeW1ib2wgaXMgZm91bmQpCgkJICAgQWJzb2x1dGUgc3ltYm9scyAqLwoJCWlmICgoKm1vZGUgPT0gJ0EnIHx8ICptb2RlID09ICc/JykgJiYgdG90YWwgPT0gMCkgY29udGludWU7CgkJaWYgKCptb2RlICE9ICdUJyAmJiAqbW9kZSAhPSAndCcgJiYKCQkgICAgKm1vZGUgIT0gJ1cnICYmICptb2RlICE9ICd3JykKCQkJYnJlYWs7CS8qIG9ubHkgdGV4dCBpcyBwcm9maWxlZCAqLwoKCQlpZiAoaW5keCA+PSBsZW4gLyBzaXplb2YoKmJ1ZikpCgkJCWJiX2Vycm9yX21zZ19hbmRfZGllKCJwcm9maWxlIGFkZHJlc3Mgb3V0IG9mIHJhbmdlLiAiCgkJCQkJICAgICAiV3JvbmcgbWFwIGZpbGU/Iik7CgoJCXdoaWxlIChpbmR4IDwgKG5leHRfYWRkLWFkZDApL3N0ZXApIHsKCQkJaWYgKG9wdEJpbnMgJiYgKGJ1ZltpbmR4XSB8fCBvcHRBbGwpKSB7CgkJCQlpZiAoIWhlYWRlcl9wcmludGVkKSB7CgkJCQkJcHJpbnRmICgiJXM6XG4iLCBmbl9uYW1lKTsKCQkJCQloZWFkZXJfcHJpbnRlZCA9IDE7CgkJCQl9CgkJCQlwcmludGYgKCJcdCVsbHhcdCV1XG4iLCAoaW5keCAtIDEpKnN0ZXAgKyBhZGQwLCBidWZbaW5keF0pOwoJCQl9CgkJCXRoaXMgKz0gYnVmW2luZHgrK107CgkJfQoJCXRvdGFsICs9IHRoaXM7CgoJCWlmIChvcHRCaW5zKSB7CgkJCWlmIChvcHRWZXJib3NlIHx8IHRoaXMgPiAwKQoJCQkJcHJpbnRmICgiICB0b3RhbFx0XHRcdFx0JXVcbiIsIHRoaXMpOwoJCX0gZWxzZSBpZiAoKHRoaXMgfHwgb3B0QWxsKSAmJgoJCQkgICAoZm5fbGVuID0gbmV4dF9hZGQtZm5fYWRkKSAhPSAwKSB7CgkJCWlmIChvcHRWZXJib3NlKQoJCQkJcHJpbnRmKCIlMDE2bGx4ICUtNDBzICU2aSAlOC40ZlxuIiwgZm5fYWRkLAoJCQkJICAgICAgIGZuX25hbWUsdGhpcyx0aGlzLyhkb3VibGUpZm5fbGVuKTsKCQkJZWxzZQoJCQkJcHJpbnRmKCIlNmkgJS00MHMgJTguNGZcbiIsCgkJCQkgICAgICAgdGhpcyxmbl9uYW1lLHRoaXMvKGRvdWJsZSlmbl9sZW4pOwoJCQlpZiAob3B0U3ViKSB7CgkJCQl1bnNpZ25lZCBsb25nIGxvbmcgc2NhbjsKCgkJCQlmb3IgKHNjYW4gPSAoZm5fYWRkLWFkZDApL3N0ZXAgKyAxOwoJCQkJICAgICBzY2FuIDwgKG5leHRfYWRkLWFkZDApL3N0ZXA7IHNjYW4rKykgewoJCQkJCXVuc2lnbmVkIGxvbmcgbG9uZyBhZGRyOwoKCQkJCQlhZGRyID0gKHNjYW4gLSAxKSpzdGVwICsgYWRkMDsKCQkJCQlwcmludGYoIlx0JSNsbHhcdCVzKyUjbGx4XHQldVxuIiwKCQkJCQkgICAgICAgYWRkciwgZm5fbmFtZSwgYWRkciAtIGZuX2FkZCwKCQkJCQkgICAgICAgYnVmW3NjYW5dKTsKCQkJCX0KCQkJfQoJCX0KCgkJZm5fYWRkID0gbmV4dF9hZGQ7CgkJc3RyY3B5KGZuX25hbWUsbmV4dF9uYW1lKTsKCgkJbWFwbGluZW5vKys7Cgl9CgoJLyogY2xvY2sgdGlja3MsIG91dCBvZiBrZXJuZWwgdGV4dCAtIHByb2JhYmx5IG1vZHVsZXMgKi8KCXByaW50ZigiJTZpICVzXG4iLCBidWZbbGVuL3NpemVvZigqYnVmKS0xXSwgIip1bmtub3duKiIpOwoKCS8qIHRyYWlsZXIgKi8KCWlmIChvcHRWZXJib3NlKQoJCXByaW50ZigiJTAxNnggJS00MHMgJTZpICU4LjRmXG4iLAoJCSAgICAgICAwLCJ0b3RhbCIsdG90YWwsdG90YWwvKGRvdWJsZSkoZm5fYWRkLWFkZDApKTsKCWVsc2UKCQlwcmludGYoIiU2aSAlLTQwcyAlOC40ZlxuIiwKCQkgICAgICAgdG90YWwsInRvdGFsIix0b3RhbC8oZG91YmxlKShmbl9hZGQtYWRkMCkpOwoKCWZjbG9zZShtYXApOwoJZnJlZShidWYpOwoKCXJldHVybiBFWElUX1NVQ0NFU1M7Cn0K