LyoKICogbGludXgvZHJpdmVycy9pZGUvaWRlLWNkLmMKICoKICogQ29weXJpZ2h0IChDKSAxOTk0LCAxOTk1LCAxOTk2ICBzY290dCBzbnlkZXIgIDxzbnlkZXJAZm5hbGQwLmZuYWwuZ292PgogKiBDb3B5cmlnaHQgKEMpIDE5OTYtMTk5OCAgRXJpayBBbmRlcnNlbiA8YW5kZXJzZWVAZGViaWFuLm9yZz4KICogQ29weXJpZ2h0IChDKSAxOTk4LTIwMDAgIEplbnMgQXhib2UgPGF4Ym9lQHN1c2UuZGU+CiAqCiAqIE1heSBiZSBjb3BpZWQgb3IgbW9kaWZpZWQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMKICogTGljZW5zZS4gIFNlZSBsaW51eC9DT1BZSU5HIGZvciBtb3JlIGluZm9ybWF0aW9uLgogKgogKiBBVEFQSSBDRC1ST00gZHJpdmVyLiAgVG8gYmUgdXNlZCB3aXRoIGlkZS5jLgogKiBTZWUgRG9jdW1lbnRhdGlvbi9jZHJvbS9pZGUtY2QgZm9yIHVzYWdlIGluZm9ybWF0aW9uLgogKgogKiBTdWdnZXN0aW9ucyBhcmUgd2VsY29tZS4gUGF0Y2hlcyB0aGF0IHdvcmsgYXJlIG1vcmUgd2VsY29tZSB0aG91Z2guIDstKQogKiBGb3IgdGhvc2Ugd2lzaGluZyB0byB3b3JrIG9uIHRoaXMgZHJpdmVyLCBwbGVhc2UgYmUgc3VyZSB5b3UgZG93bmxvYWQKICogYW5kIGNvbXBseSB3aXRoIHRoZSBsYXRlc3QgTXQuIEZ1amkgKFNGRjgwOTAgdmVyc2lvbiA0KSBhbmQgQVRBUEkgCiAqIChTRkYtODAyMGkgcmV2IDIuNikgc3RhbmRhcmRzLiBUaGVzZSBkb2N1bWVudHMgY2FuIGJlIG9idGFpbmVkIGJ5IAogKiBhbm9ueW1vdXMgZnRwIGZyb206CiAqIGZ0cDovL2Zpc3Npb24uZHQud2RjLmNvbS9wdWIvc3RhbmRhcmRzL1NGRl9hdGFwaS9zcGVjL1NGRjgwMjAtcjIuNi9QUy84MDIwcjI2LnBzCiAqIGZ0cDovL2Z0cC5hdmMtcGlvbmVlci5jb20vTXRmdWppNC9TcGVjL0Z1amk0cjEwLnBkZgogKgogKiBEcml2ZXMgdGhhdCBkZXZpYXRlIGZyb20gdGhlc2Ugc3RhbmRhcmRzIHdpbGwgYmUgYWNjb21tb2RhdGVkIGFzIG11Y2gKICogYXMgcG9zc2libGUgdmlhIGNvbXBpbGUgdGltZSBvciBjb21tYW5kLWxpbmUgb3B0aW9ucy4gIFNpbmNlIEkgb25seSBoYXZlCiAqIGEgZmV3IGRyaXZlcywgeW91IGdlbmVyYWxseSBuZWVkIHRvIHNlbmQgbWUgcGF0Y2hlcy4uLgogKgogKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAqIFRPIERPIExJU1Q6CiAqIC1NYWtlIGl0IHNvIHRoYXQgUGlvbmVlciBDRCBEUi1BMjRYIGFuZCBmcmllbmRzIGRvbid0IGdldCBzY3Jld2VkIHVwIG9uCiAqICAgYm9vdAogKgogKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAqIDEuMDAgIE9jdCAzMSwgMTk5NCAtLSBJbml0aWFsIHZlcnNpb24uCiAqIDEuMDEgIE5vdiAgMiwgMTk5NCAtLSBGaXhlZCBwcm9ibGVtIHdpdGggc3RhcnRpbmcgcmVxdWVzdCBpbgogKiAgICAgICAgICAgICAgICAgICAgICAgY2Ryb21fY2hlY2tfc3RhdHVzLgogKiAxLjAzICBOb3YgMjUsIDE5OTQgLS0gbGVhdmluZyB1bm1hc2tfaW50cltdIGFzIGEgdXNlci1zZXR0aW5nIChhcyBmb3IgZGlza3MpCiAqIChmcm9tIG1sb3JkKSAgICAgICAtLSBtaW5vciBjaGFuZ2VzIHRvIGNkcm9tX3NldHVwKCkKICogICAgICAgICAgICAgICAgICAgIC0tIHJlbmFtZWQgaWRlX2Rldl9zIHRvIGlkZV9kcml2ZV90LCBlbmFibGUgaXJxIG9uIGNvbW1hbmQKICogMi4wMCAgTm92IDI3LCAxOTk0IC0tIEdlbmVyYWxpemUgcGFja2V0IGNvbW1hbmQgaW50ZXJmYWNlOwogKiAgICAgICAgICAgICAgICAgICAgICAgYWRkIGF1ZGlvIGlvY3Rscy4KICogMi4wMSAgRGVjICAzLCAxOTk0IC0tIFJld29yayBwYWNrZXQgY29tbWFuZCBpbnRlcmZhY2UgdG8gaGFuZGxlIGRldmljZXMKICogICAgICAgICAgICAgICAgICAgICAgIHdoaWNoIHNlbmQgYW4gaW50ZXJydXB0IHdoZW4gcmVhZHkgZm9yIGEgY29tbWFuZC4KICogMi4wMiAgRGVjIDExLCAxOTk0IC0tIENhY2hlIHRoZSBUT0MgaW4gdGhlIGRyaXZlci4KICogICAgICAgICAgICAgICAgICAgICAgIERvbid0IHVzZSBTQ01EX1BMQVlBVURJT19USTsgaXQncyBub3QgaW5jbHVkZWQKICogICAgICAgICAgICAgICAgICAgICAgIGluIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgQVRBUEkuCiAqICAgICAgICAgICAgICAgICAgICAgICBUcnkgdG8gdXNlIExCQSBpbnN0ZWFkIG9mIHRyYWNrIG9yIE1TRiBhZGRyZXNzaW5nCiAqICAgICAgICAgICAgICAgICAgICAgICB3aGVuIHBvc3NpYmxlLgogKiAgICAgICAgICAgICAgICAgICAgICAgRG9uJ3Qgd2FpdCBmb3IgUkVBRFlfU1RBVC4KICogMi4wMyAgSmFuIDEwLCAxOTk1IC0tIFJld3JpdGUgYmxvY2sgcmVhZCByb3V0aW5lcyB0byBoYW5kbGUgYmxvY2sgc2l6ZXMKICogICAgICAgICAgICAgICAgICAgICAgIG90aGVyIHRoYW4gMmsgYW5kIHRvIG1vdmUgbXVsdGlwbGUgc2VjdG9ycyBpbiBhCiAqICAgICAgICAgICAgICAgICAgICAgICBzaW5nbGUgdHJhbnNhY3Rpb24uCiAqIDIuMDQgIEFwciAyMSwgMTk5NSAtLSBBZGQgd29yay1hcm91bmQgZm9yIENyZWF0aXZlIExhYnMgQ0QyMjBFIGRyaXZlcy4KICogICAgICAgICAgICAgICAgICAgICAgIFRoYW5rcyB0byBOaWNrIFNhdyA8Y3dzYXdAcHRzNy5wdHMubW90LmNvbT4gZm9yCiAqICAgICAgICAgICAgICAgICAgICAgICBoZWxwIGluIGZpZ3VyaW5nIHRoaXMgb3V0LiAgRGl0dG8gZm9yIEFjZXIgYW5kCiAqICAgICAgICAgICAgICAgICAgICAgICBBenRlY2ggZHJpdmVzLCB3aGljaCBzZWVtIHRvIGhhdmUgdGhlIHNhbWUgcHJvYmxlbS4KICogMi4wNGIgTWF5IDMwLCAxOTk1IC0tIEZpeCB0byBtYXRjaCBjaGFuZ2VzIGluIGlkZS5jIHZlcnNpb24gMy4xNiAtbWwKICogMi4wNSAgSnVuICA4LCAxOTk1IC0tIERvbid0IGF0dGVtcHQgdG8gcmV0cnkgYWZ0ZXIgYW4gaWxsZWdhbCByZXF1ZXN0CiAqICAgICAgICAgICAgICAgICAgICAgICAgb3IgZGF0YSBwcm90ZWN0IGVycm9yLgogKiAgICAgICAgICAgICAgICAgICAgICAgVXNlIEhXSUYgYW5kIERFVl9IV0lGIG1hY3JvcyBhcyBpbiBpZGUuYy4KICogICAgICAgICAgICAgICAgICAgICAgIEFsd2F5cyB0cnkgdG8gZG8gYSByZXF1ZXN0X3NlbnNlIGFmdGVyCiAqICAgICAgICAgICAgICAgICAgICAgICAgYSBmYWlsZWQgY29tbWFuZC4KICogICAgICAgICAgICAgICAgICAgICAgIEluY2x1ZGUgYW4gb3B0aW9uIHRvIGdpdmUgdGV4dHVhbCBkZXNjcmlwdGlvbnMKICogICAgICAgICAgICAgICAgICAgICAgICBvZiBBVEFQSSBlcnJvcnMuCiAqICAgICAgICAgICAgICAgICAgICAgICBGaXggYSBidWcgaW4gaGFuZGxpbmcgdGhlIHNlY3RvciBjYWNoZSB3aGljaAogKiAgICAgICAgICAgICAgICAgICAgICAgIHNob3dlZCB1cCBpZiB0aGUgZHJpdmUgcmV0dXJuZWQgZGF0YSBpbiA1MTIgYnl0ZQogKiAgICAgICAgICAgICAgICAgICAgICAgIGJsb2NrcyAobGlrZSBQaW9uZWVyIGRyaXZlcykuICBUaGFua3MgdG8KICogICAgICAgICAgICAgICAgICAgICAgICBSaWNoYXJkIEhpcnN0IDxzcmhAZ3B0LmNvLnVrPiBmb3IgZGlhZ25vc2luZyB0aGlzLgogKiAgICAgICAgICAgICAgICAgICAgICAgUHJvcGVybHkgc3VwcGx5IHRoZSBwYWdlIG51bWJlciBmaWVsZCBpbiB0aGUKICogICAgICAgICAgICAgICAgICAgICAgICBNT0RFX1NFTEVDVCBjb21tYW5kLgogKiAgICAgICAgICAgICAgICAgICAgICAgUExBWUFVRElPMTIgaXMgYnJva2VuIG9uIHRoZSBBenRlY2g7IHdvcmsgYXJvdW5kIGl0LgogKiAyLjA1eCBBdWcgMTEsIDE5OTUgLS0gbG90cyBvZiBkYXRhIHN0cnVjdHVyZSByZW5hbWluZy9yZXN0cnVjdHVyaW5nIGluIGlkZS5jCiAqICAgICAgICAgICAgICAgICAgICAgICAobXkgYXBvbG9naWVzIHRvIFNjb3R0LCBidXQgbm93IGlkZS1jZC5jIGlzIGluZGVwZW5kZW50KQogKiAzLjAwICBBdWcgMjIsIDE5OTUgLS0gSW1wbGVtZW50IENEUk9NTVVMVElTRVNTSU9OIGlvY3RsLgogKiAgICAgICAgICAgICAgICAgICAgICAgSW1wbGVtZW50IENEUk9NUkVBREFVRElPIGlvY3RsIChVTlRFU1RFRCkuCiAqICAgICAgICAgICAgICAgICAgICAgICBVc2UgaW5wdXRfaWRlX2RhdGEoKSBhbmQgb3V0cHV0X2lkZV9kYXRhKCkuCiAqICAgICAgICAgICAgICAgICAgICAgICBBZGQgZG9vciBsb2NraW5nLgogKiAgICAgICAgICAgICAgICAgICAgICAgRml4IHVzYWdlIGNvdW50IGxlYWsgaW4gY2Ryb21fb3Blbiwgd2hpY2ggaGFwcGVuZWQKICogICAgICAgICAgICAgICAgICAgICAgICB3aGVuIGEgcmVhZC13cml0ZSBtb3VudCB3YXMgYXR0ZW1wdGVkLgogKiAgICAgICAgICAgICAgICAgICAgICAgVHJ5IHRvIGxvYWQgdGhlIGRpc2sgb24gb3Blbi4KICogICAgICAgICAgICAgICAgICAgICAgIEltcGxlbWVudCBDRFJPTUVKRUNUX1NXIGlvY3RsIChvZmYgYnkgZGVmYXVsdCkuCiAqICAgICAgICAgICAgICAgICAgICAgICBSZWFkIHRvdGFsIGNkcm9tIGNhcGFjaXR5IGR1cmluZyBvcGVuLgogKiAgICAgICAgICAgICAgICAgICAgICAgUmVhcnJhbmdlIGxvZ2ljIGluIGNkcm9tX2RlY29kZV9zdGF0dXMuICBJc3N1ZQogKiAgICAgICAgICAgICAgICAgICAgICAgIHJlcXVlc3Qgc2Vuc2UgY29tbWFuZHMgZm9yIGZhaWxlZCBwYWNrZXQgY29tbWFuZHMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIGhlcmUgaW5zdGVhZCBvZiBmcm9tIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kLgogKiAgICAgICAgICAgICAgICAgICAgICAgIEZpeCBhIHJhY2UgY29uZGl0aW9uIGluIHJldHJpZXZpbmcgZXJyb3IgaW5mb3JtYXRpb24uCiAqICAgICAgICAgICAgICAgICAgICAgICBTdXBwcmVzcyBwcmludGluZyBub3JtYWwgdW5pdCBhdHRlbnRpb24gZXJyb3JzIGFuZAogKiAgICAgICAgICAgICAgICAgICAgICAgIHNvbWUgZHJpdmUgbm90IHJlYWR5IGVycm9ycy4KICogICAgICAgICAgICAgICAgICAgICAgIEltcGxlbWVudCBDRFJPTVZPTFJFQUQgaW9jdGwuCiAqICAgICAgICAgICAgICAgICAgICAgICBJbXBsZW1lbnQgQ0RST01SRUFETU9ERTEvMiBpb2N0bHMuCiAqICAgICAgICAgICAgICAgICAgICAgICBGaXggcmFjZSBjb25kaXRpb24gaW4gc2V0dGluZyB1cCBpbnRlcnJ1cHQgaGFuZGxlcnMKICogICAgICAgICAgICAgICAgICAgICAgICB3aGVuIHRoZSBgc2VyaWFsaXplJyBvcHRpb24gaXMgdXNlZC4KICogMy4wMSAgU2VwICAyLCAxOTk1IC0tIEZpeCBvcmRlcmluZyBvZiByZWVuYWJsaW5nIGludGVycnVwdHMgaW4KICogICAgICAgICAgICAgICAgICAgICAgICBjZHJvbV9xdWV1ZV9yZXF1ZXN0LgogKiAgICAgICAgICAgICAgICAgICAgICAgQW5vdGhlciB0cnkgYXQgdXNpbmcgaWRlX1tpbnB1dCxvdXRwdXRdX2RhdGEuCiAqIDMuMDIgIFNlcCAxNiwgMTk5NSAtLSBTdGljayB0b3RhbCBkaXNrIGNhcGFjaXR5IGluIHBhcnRpdGlvbiB0YWJsZSBhcyB3ZWxsLgogKiAgICAgICAgICAgICAgICAgICAgICAgTWFrZSBWRVJCT1NFX0lERV9DRF9FUlJPUlMgZHVtcCBmYWlsZWQgY29tbWFuZCBhZ2Fpbi4KICogICAgICAgICAgICAgICAgICAgICAgIER1bXAgb3V0IG1vcmUgaW5mb3JtYXRpb24gZm9yIElMTEVHQUwgUkVRVUVTVCBlcnJzLgogKiAgICAgICAgICAgICAgICAgICAgICAgRml4IGhhbmRsaW5nIG9mIGVycm9ycyBvY2N1cnJpbmcgYmVmb3JlIHRoZQogKiAgICAgICAgICAgICAgICAgICAgICAgIHBhY2tldCBjb21tYW5kIGlzIHRyYW5zZmVycmVkLgogKiAgICAgICAgICAgICAgICAgICAgICAgRml4IHRyYW5zZmVycyB3aXRoIG9kZCBieXRlbGVuZ3Rocy4KICogMy4wMyAgT2N0IDI3LCAxOTk1IC0tIFNvbWUgQ3JlYXRpdmUgZHJpdmVzIGhhdmUgYW4gaWQgb2YganVzdCBgQ0QnLgogKiAgICAgICAgICAgICAgICAgICAgICAgYERDSS0yUzEwJyBkcml2ZXMgYXJlIGJyb2tlbiB0b28uCiAqIDMuMDQgIE5vdiAyMCwgMTk5NSAtLSBTbyBhcmUgVmVydG9zIGRyaXZlcy4KICogMy4wNSAgRGVjICAxLCAxOTk1IC0tIENoYW5nZXMgdG8gZ28gd2l0aCBvdmVyaGF1bCBvZiBpZGUuYyBhbmQgaWRlLXRhcGUuYwogKiAzLjA2ICBEZWMgMTYsIDE5OTUgLS0gQWRkIHN1cHBvcnQgbmVlZGVkIGZvciBwYXJ0aXRpb25zLgogKiAgICAgICAgICAgICAgICAgICAgICAgTW9yZSB3b3JrYXJvdW5kcyBmb3IgVmVydG9zIGJ1Z3MgKGJhc2VkIG9uIHBhdGNoZXMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIEhvbGdlciBEaWV0emUgPGRpZXR6ZUBhaXg1MjAuaW5mb3JtYXRpay51bmktbGVpcHppZy5kZT4pLgogKiAgICAgICAgICAgICAgICAgICAgICAgVHJ5IHRvIGVsaW1pbmF0ZSBieXRlb3JkZXIgYXNzdW1wdGlvbnMuCiAqICAgICAgICAgICAgICAgICAgICAgICBVc2UgYXRhcGlfY2Ryb21fc3ViY2hubCBzdHJ1Y3QgZGVmaW5pdGlvbi4KICogICAgICAgICAgICAgICAgICAgICAgIEFkZCBTVEFOREFSRF9BVEFQSSBjb21waWxhdGlvbiBvcHRpb24uCiAqIDMuMDcgIEphbiAyOSwgMTk5NiAtLSBNb3JlIHR3aWRkbGluZyBmb3IgYnJva2VuIGRyaXZlczogU29ueSA1NUQsCiAqICAgICAgICAgICAgICAgICAgICAgICAgVmVydG9zIDMwMC4KICogICAgICAgICAgICAgICAgICAgICAgIEFkZCBOT19ET09SX0xPQ0tJTkcgY29uZmlndXJhdGlvbiBvcHRpb24uCiAqICAgICAgICAgICAgICAgICAgICAgICBIYW5kbGUgZHJpdmVfY21kIHJlcXVlc3RzIHcvTlVMTCBhcmdzIChmb3IgaGRwYXJtIC10KS4KICogICAgICAgICAgICAgICAgICAgICAgIFdvcmsgYXJvdW5kIHNwb3JhZGljIFNvbnk1NWUgYXVkaW8gcGxheSBwcm9ibGVtLgogKiAzLjA3YSBGZWIgMTEsIDE5OTYgLS0gY2hlY2sgZHJpdmUtPmlkIGZvciBOVUxMIGJlZm9yZSBkZXJlZmVyZW5jaW5nLCB0byBmaXgKICogICAgICAgICAgICAgICAgICAgICAgICBwcm9ibGVtIHdpdGggImhkZT1jZHJvbSIgd2l0aCBubyBkcml2ZSBwcmVzZW50LiAgLW1sCiAqIDMuMDggIE1hciAgNiwgMTk5NiAtLSBNb3JlIFZlcnRvcyB3b3JrYXJvdW5kcy4KICogMy4wOSAgQXByICA1LCAxOTk2IC0tIEFkZCBDRFJPTUNMT1NFVFJBWSBpb2N0bC4KICogICAgICAgICAgICAgICAgICAgICAgIFN3aXRjaCB0byB1c2luZyBNU0YgYWRkcmVzc2luZyBmb3IgYXVkaW8gY29tbWFuZHMuCiAqICAgICAgICAgICAgICAgICAgICAgICBSZWZvcm1hdCB0byBtYXRjaCBrZXJuZWwgdGFiYmluZyBzdHlsZS4KICogICAgICAgICAgICAgICAgICAgICAgIEFkZCBDRFJPTV9HRVRfVVBDIGlvY3RsLgogKiAzLjEwICBBcHIgMTAsIDE5OTYgLS0gRml4IGNvbXBpbGF0aW9uIGVycm9yIHdpdGggU1RBTkRBUkRfQVRBUEkuCiAqIDMuMTEgIEFwciAyOSwgMTk5NiAtLSBQYXRjaCBmcm9tIEhlaWtvIEVpc3NmZWxkdCA8aGVpa29AY29sb3NzdXMuZXNjYXBlLmRlPgogKiAgICAgICAgICAgICAgICAgICAgICAgdG8gcmVtb3ZlIHJlZHVuZGFudCB2ZXJpZnlfYXJlYSBjYWxscy4KICogMy4xMiAgTWF5ICA3LCAxOTk2IC0tIFJ1ZGltZW50YXJ5IGNoYW5nZXIgc3VwcG9ydC4gIEJhc2VkIG9uIHBhdGNoZXMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIEdlcmhhcmQgWnViZXIgPHp1YmVyQGJlcmxpbi5zbmFmdS5kZT4uCiAqICAgICAgICAgICAgICAgICAgICAgICBMZXQgb3BlbiBzdWNjZWVkIGV2ZW4gaWYgdGhlcmUncyBubyBsb2FkZWQgZGlzYy4KICogMy4xMyAgTWF5IDE5LCAxOTk2IC0tIEZpeGVzIGZvciBjaGFuZ2VyIGNvZGUuCiAqIDMuMTQgIE1heSAyOSwgMTk5NiAtLSBBZGQgd29yay1hcm91bmQgZm9yIFZlcnRvcyA2MDAuCiAqICAgICAgICAgICAgICAgICAgICAgICAgKEZyb20gSGVubnVzIEJlcmdtYW4gPGhlbm51c0Bza3kub3cubmw+LikKICogMy4xNSAgSnVseSAyLCAxOTk2IC0tIEFkZGVkIHN1cHBvcnQgZm9yIFNhbnlvIDMgQ0QgY2hhbmdlcnMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIEJlbiBHYWxsaWFydCA8YmdhbGxpYUBsdWMuZWR1PiB3aXRoIAogKiAgICAgICAgICAgICAgICAgICAgICAgIHNwZWNpYWwgaGVscCBmcm9tIEplZmYgTGlnaHRmb290IAogKiAgICAgICAgICAgICAgICAgICAgICAgIDxqZWZmbWxAcG9ib3guY29tPgogKiAzLjE1YSBKdWx5IDksIDE5OTYgLS0gSW1wcm92ZWQgU2FueW8gMyBDRCBjaGFuZ2VyIGlkZW50aWZpY2F0aW9uCiAqIDMuMTYgIEp1bCAyOCwgMTk5NiAtLSBGaXggZnJvbSBHYWRpIHRvIHJlZHVjZSBrZXJuZWwgc3RhY2sgdXNhZ2UgZm9yIGlvY3RsLgogKiAzLjE3ICBTZXAgMTcsIDE5OTYgLS0gVHdlYWsgYXVkaW8gcmVhZHMgZm9yIHNvbWUgZHJpdmVzLgogKiAgICAgICAgICAgICAgICAgICAgICAgU3RhcnQgY2hhbmdpbmcgQ0RST01MT0FERlJPTVNMT1QgdG8gQ0RST01fU0VMRUNUX0RJU0MuCiAqIDMuMTggIE9jdCAzMSwgMTk5NiAtLSBBZGRlZCBtb2R1bGUgYW5kIERNQSBzdXBwb3J0LgogKiAgICAgICAgICAgICAgICAgICAgICAgCiAqICAgICAgICAgICAgICAgICAgICAgICAKICogNC4wMCAgTm92IDUsIDE5OTYgICAtLSBOZXcgaWRlLWNkIG1haW50YWluZXIsCiAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRXJpayBCLiBBbmRlcnNlbiA8YW5kZXJzZWVAZGViaWFuLm9yZz4KICogICAgICAgICAgICAgICAgICAgICAtLSBOZXdlciBDcmVhdGl2ZSBkcml2ZXMgZG9uJ3QgYWx3YXlzIHNldCB0aGUgZXJyb3IKICogICAgICAgICAgICAgICAgICAgICAgICAgIHJlZ2lzdGVyIGNvcnJlY3RseS4gIE1ha2Ugc3VyZSB3ZSBzZWUgbWVkaWEgY2hhbmdlcwogKiAgICAgICAgICAgICAgICAgICAgICAgICAgcmVnYXJkbGVzcy4KICogICAgICAgICAgICAgICAgICAgICAtLSBJbnRlZ3JhdGUgd2l0aCBnZW5lcmljIGNkcm9tIGRyaXZlci4KICogICAgICAgICAgICAgICAgICAgICAtLSBDRFJPTUdFVFNQSU5ET1dOIGFuZCBDRFJPTVNFVFNQSU5ET1dOIGlvY3RscywgYmFzZWQgb24KICogICAgICAgICAgICAgICAgICAgICAgICAgIGEgcGF0Y2ggZnJvbSBDaXJvIENhdHR1dG8gPD4uCiAqICAgICAgICAgICAgICAgICAgICAgLS0gQ2FsbCBzZXRfZGV2aWNlX3JvLgogKiAgICAgICAgICAgICAgICAgICAgIC0tIEltcGxlbWVudCBDRFJPTU1FQ0hBTklTTVNUQVRVUyBhbmQgQ0RST01TTE9UVEFCTEUKICogICAgICAgICAgICAgICAgICAgICAgICAgIGlvY3RscywgYmFzZWQgb24gcGF0Y2ggYnkgRXJpayBBbmRlcnNlbgogKiAgICAgICAgICAgICAgICAgICAgIC0tIEFkZCBzb21lIHByb2JlcyBvZiBkcml2ZSBjYXBhYmlsaXR5IGR1cmluZyBzZXR1cC4KICoKICogNC4wMSAgTm92IDExLCAxOTk2ICAtLSBTcGxpdCBpbnRvIGlkZS1jZC5jIGFuZCBpZGUtY2QuaAogKiAgICAgICAgICAgICAgICAgICAgIC0tIFJlbW92ZWQgQ0RST01NRUNIQU5JU01TVEFUVVMgYW5kIENEUk9NU0xPVFRBQkxFIAogKiAgICAgICAgICAgICAgICAgICAgICAgICAgaW9jdGxzIGluIGZhdm9yIG9mIGEgZ2VuZXJhbGl6ZWQgYXBwcm9hY2ggCiAqICAgICAgICAgICAgICAgICAgICAgICAgICB1c2luZyB0aGUgZ2VuZXJpYyBjZHJvbSBkcml2ZXIuCiAqICAgICAgICAgICAgICAgICAgICAgLS0gRnVsbHkgaW50ZWdyYXRlZCB3aXRoIHRoZSAyLjEuWCBrZXJuZWwuCiAqICAgICAgICAgICAgICAgICAgICAgLS0gT3RoZXIgc3R1ZmYgdGhhdCBJIGZvcmdvdCAobG90cyBvZiBjaGFuZ2VzKQogKgogKiA0LjAyICBEZWMgMDEsIDE5OTYgIC0tIEFwcGxpZWQgcGF0Y2ggZnJvbSBHYWRpIE94bWFuIDxnYWRpb0BuZXR2aXNpb24ubmV0LmlsPgogKiAgICAgICAgICAgICAgICAgICAgICAgICAgdG8gZml4IHRoZSBkcml2ZSBkb29yIGxvY2tpbmcgcHJvYmxlbXMuCiAqCiAqIDQuMDMgIERlYyAwNCwgMTk5NiAgLS0gQWRkZWQgRFNDIG92ZXJsYXAgc3VwcG9ydC4KICogNC4wNCAgRGVjIDI5LCAxOTk2ICAtLSBBZGRlZCBDRFJPTVJFQURSQVcgaW9jbHQgYmFzZWQgb24gcGF0Y2ggCiAqICAgICAgICAgICAgICAgICAgICAgICAgICBieSBBbGVzIE1ha2Fyb3YgKHhtYWthcm92QHN1bi5mZWxrLmN2dXQuY3opCiAqCiAqIDQuMDUgIE5vdiAyMCwgMTk5NyAgLS0gTW9kaWZpZWQgdG8gcHJpbnQgbW9yZSBkcml2ZSBpbmZvIG9uIGluaXQKICogICAgICAgICAgICAgICAgICAgICAgICBNaW5vciBvdGhlciBjaGFuZ2VzCiAqICAgICAgICAgICAgICAgICAgICAgICAgRml4IGVycm9ycyBvbiBDRFJPTVNUT1AgKElmIHlvdSBoYXZlIGEgIkRvbHBoaW4iLAogKiAgICAgICAgICAgICAgICAgICAgICAgICAgeW91IG11c3QgZGVmaW5lIElIQVZFQURPTFBISU4pCiAqICAgICAgICAgICAgICAgICAgICAgICAgQWRkZWQgaWRlbnRpZmllciBzbyBuZXcgU2FueW8gQ0QtY2hhbmdlciB3b3JrcwogKiAgICAgICAgICAgICAgICAgICAgICAgIEJldHRlciBkZXRlY3Rpb24gaWYgZG9vciBsb2NraW5nIGlzbid0IHN1cHBvcnRlZAogKgogKiA0LjA2ICBEZWMgMTcsIDE5OTcgIC0tIGZpeGVkIGVuZGxlc3MgInRyYXkgb3BlbiIgbWVzc2FnZXMgIC1tbAogKiA0LjA3ICBEZWMgMTcsIDE5OTcgIC0tIGZhbGxiYWNrIHRvIHNldCBwYy0+c3RhdCBvbiAidHJheSBvcGVuIgogKiA0LjA4ICBEZWMgMTgsIDE5OTcgIC0tIHNwZXcgbGVzcyBub2lzZSB3aGVuIHRyYXkgaXMgZW1wdHkKICogICAgICAgICAgICAgICAgICAgICAtLSBmaXggc3BlZWQgZGlzcGxheSBmb3IgQUNFUiAyNFgsIDE4WAogKiA0LjA5ICBKYW4gMDQsIDE5OTggIC0tIGZpeCBoYW5kbGluZyBvZiB0aGUgbGFzdCBibG9jayBzbyB3ZSByZXR1cm4KICogICAgICAgICAgICAgICAgICAgICAgICAgYW4gZW5kIG9mIGZpbGUgaW5zdGVhZCBvZiBhbiBJL08gZXJyb3IgKEdhZGkpCiAqIDQuMTAgIEphbiAyNCwgMTk5OCAgLS0gZml4ZWQgYSBidWcgc28gbm93IGNoYW5nZXJzIGNhbiBjaGFuZ2UgdG8gYSBuZXcKICogICAgICAgICAgICAgICAgICAgICAgICAgc2xvdCB3aGVuIHRoZXJlIGlzIG5vIGRpc2MgaW4gdGhlIGN1cnJlbnQgc2xvdC4KICogICAgICAgICAgICAgICAgICAgICAtLSBGaXhlZCBhIG1lbW9yeSBsZWFrIHdoZXJlIGluZm8tPmNoYW5nZXJfaW5mbyB3YXMKICogICAgICAgICAgICAgICAgICAgICAgICAgbWFsbG9jJ2VkIGJ1dCBuZXZlciBmcmVlJ2Qgd2hlbiBjbG9zaW5nIHRoZSBkZXZpY2UuCiAqICAgICAgICAgICAgICAgICAgICAgLS0gQ2xlYW5lZCB1cCB0aGUgZ2xvYmFsIG5hbWVzcGFjZSBhIGJpdCBieSBtYWtpbmcgbW9yZQogKiAgICAgICAgICAgICAgICAgICAgICAgICBmdW5jdGlvbnMgc3RhdGljIHRoYXQgc2hvdWxkIGFscmVhZHkgaGF2ZSBiZWVuLgogKiA0LjExICBNYXIgMTIsIDE5OTggIC0tIEFkZGVkIHN1cHBvcnQgZm9yIHRoZSBDRFJPTV9TRUxFQ1RfU1BFRUQgaW9jdGwKICogICAgICAgICAgICAgICAgICAgICAgICAgYmFzZWQgb24gYSBwYXRjaCBmb3IgMi4wLjMzIGJ5IEplbGxlIEZva3MgCiAqICAgICAgICAgICAgICAgICAgICAgICAgIDxqZWxsZUBzY2ludGlsbGEudXR3ZW50ZS5ubD4sIGEgcGF0Y2ggZm9yIDIuMC4zMwogKiAgICAgICAgICAgICAgICAgICAgICAgICBieSBUb25pIEdpb3JnaW5vIDx0b25pQHBjYXBlMi5waS5pbmZuLml0PiwgdGhlIFNDU0kKICogICAgICAgICAgICAgICAgICAgICAgICAgdmVyc2lvbiwgYW5kIG15IG93biBlZmZvcnRzLiAgLWVyaWsKICogICAgICAgICAgICAgICAgICAgICAtLSBGaXhlZCBhIHN0dXBpZCBidWcgd2hpY2ggZWdjcyB3YXMga2luZCBlbm91Z2ggdG8KICogICAgICAgICAgICAgICAgICAgICAgICAgaW5mb3JtIG1lIG9mIHdoZXJlICJJbGxlZ2FsIG1vZGUgZm9yIHRoaXMgdHJhY2siCiAqICAgICAgICAgICAgICAgICAgICAgICAgIHdhcyBuZXZlciByZXR1cm5lZCBkdWUgdG8gYSBjb21wYXJpc29uIG9uIGRhdGEKICogICAgICAgICAgICAgICAgICAgICAgICAgdHlwZXMgb2YgbGltaXRlZCByYW5nZS4KICogNC4xMiAgTWFyIDI5LCAxOTk4ICAtLSBGaXhlZCBidWcgaW4gQ0RST01fU0VMRUNUX1NQRUVEIHNvIHdyaXRlIHNwZWVkIGlzIAogKiAgICAgICAgICAgICAgICAgICAgICAgICBub3cgc2V0IGlvbmx5IGZvciBDRC1SIGFuZCBDRC1SVyBkcml2ZXMuICBJIGhhZCAKICogICAgICAgICAgICAgICAgICAgICAgICAgcmVtb3ZlZCB0aGlzIHN1cHBvcnQgYmVjYXVzZSBpdCBwcm9kdWNlZCBlcnJvcnMuCiAqICAgICAgICAgICAgICAgICAgICAgICAgIEl0IHByb2R1Y2VkIGVycm9ycyBfb25seV8gZm9yIG5vbi13cml0ZXJzLiBkdWguCiAqIDQuMTMgIE1heSAwNSwgMTk5OCAgLS0gU3VwcHJlc3MgdXNlbGVzcyAiaW4gcHJvZ3Jlc3Mgb2YgYmVjb21pbmcgcmVhZHkiCiAqICAgICAgICAgICAgICAgICAgICAgICAgIG1lc3NhZ2VzLCBzaW5jZSB0aGlzIGlzIG5vdCBhbiBlcnJvci4KICogICAgICAgICAgICAgICAgICAgICAtLSBDaGFuZ2UgZXJyb3IgbWVzc2FnZXMgdG8gYmUgY29uc3QKICogICAgICAgICAgICAgICAgICAgICAtLSBSZW1vdmUgYSAiXHQiIHdoaWNoIGxvb2tzIHVnbHkgaW4gdGhlIHN5c2xvZ3MKICogNC4xNCAgSnVseSAxNywgMTk5OCAtLSBDaGFuZ2UgdG8gcG9pbnRpbmcgdG8gLnBzIHZlcnNpb24gb2YgQVRBUEkgc3BlYwogKiAgICAgICAgICAgICAgICAgICAgICAgICBzaW5jZSB0aGUgLnBkZiB2ZXJzaW9uIGRvZXNuJ3Qgc2VlbSB0byB3b3JrLi4uCiAqICAgICAgICAgICAgICAgICAgICAgLS0gVXBkYXRlZCB0aGUgVE9ETyBsaXN0IHRvIHNvbWV0aGluZyBtb3JlIGN1cnJlbnQuCiAqCiAqIDQuMTUgIEF1ZyAyNSwgMTk5OCAgLS0gVXBkYXRlZCBpZGUtY2QuaCB0byByZXNwZWN0IG1lY2hpbmUgZW5kaWFuZXNzLCAKICogICAgICAgICAgICAgICAgICAgICAgICAgcGF0Y2ggdGhhbmtzIHRvICJFZGRpZSBDLiBEb3N0IiA8ZWNkQHNreW5ldC5iZT4KICoKICogNC41MCAgT2N0IDE5LCAxOTk4ICAtLSBOZXcgbWFpbnRhaW5lcnMhCiAqICAgICAgICAgICAgICAgICAgICAgICAgIEplbnMgQXhib2UgPGF4Ym9lQGltYWdlLmRrPgogKiAgICAgICAgICAgICAgICAgICAgICAgICBDaHJpcyBad2lsbGluZyA8Y2hyaXNAY2xvdWRuZXQuY29tPgogKgogKiA0LjUxICBEZWMgMjMsIDE5OTggIC0tIEplbnMgQXhib2UgPGF4Ym9lQGltYWdlLmRrPgogKiAgICAgICAgICAgICAgICAgICAgICAtIGlkZV9jZHJvbV9yZXNldCBlbmFibGVkIHNpbmNlIHRoZSBpZGUgc3Vic3lzdGVtCiAqICAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZXMgcmVzZXRzIGZpbmUgbm93LiA8YXhib2VAaW1hZ2UuZGs+CiAqICAgICAgICAgICAgICAgICAgICAgIC0gVHJhbnNmZXIgc2l6ZSBmaXggZm9yIFNhbXN1bmcgQ0QtUk9NcywgdGhhbmtzIHRvCiAqICAgICAgICAgICAgICAgICAgICAgICAgIlZpbGxlIEhhbGxpayIgPHZpbGxlLmhhbGxpa0BtYWlsLmVlPi4KICogICAgICAgICAgICAgICAgICAgICAgLSBvdGhlciBtaW5vciBzdHVmZi4KICoKICogNC41MiAgSmFuIDE5LCAxOTk5ICAtLSBKZW5zIEF4Ym9lIDxheGJvZUBpbWFnZS5kaz4KICogICAgICAgICAgICAgICAgICAgICAgLSBEZXRlY3QgRFZELVJPTS9SQU0gZHJpdmVzCiAqCiAqIDQuNTMgIEZlYiAyMiwgMTk5OSAgIC0gSW5jbHVkZSBvdGhlciBtb2RlbCBTYW1zdW5nIGFuZCBvbmUgR29sZHN0YXIKICogICAgICAgICAgICAgICAgICAgICAgICAgZHJpdmUgaW4gdHJhbnNmZXIgc2l6ZSBsaW1pdC4KICogICAgICAgICAgICAgICAgICAgICAgLSBGaXggdGhlIEkvTyBlcnJvciB3aGVuIGRvaW5nIGVqZWN0IHdpdGhvdXQgYSBtZWRpdW0KICogICAgICAgICAgICAgICAgICAgICAgICAgbG9hZGVkIG9uIHNvbWUgZHJpdmVzLgogKiAgICAgICAgICAgICAgICAgICAgICAtIENEUk9NUkVBRE1PREUyIGlzIG5vdyBpbXBsZW1lbnRlZCB0aHJvdWdoCiAqICAgICAgICAgICAgICAgICAgICAgICAgIENEUk9NUkVBRFJBVywgc2luY2UgbWFueSBkcml2ZXMgZG9uJ3Qgc3VwcG9ydAogKiAgICAgICAgICAgICAgICAgICAgICAgICBNT0RFMiAoZXZlbiB0aG91Z2ggQVRBUEkgMi42IHNheXMgdGhleSBtdXN0KS4KICogICAgICAgICAgICAgICAgICAgICAgLSBBZGRlZCBpZ25vcmUgcGFyYW1ldGVyIHRvIGlkZS1jZCAoYXMgYSBtb2R1bGUpLCBlZwogKiAgICAgICAgICAgICAgICAgICAgICAgICAJaW5zbW9kIGlkZS1jZCBpZ25vcmU9J2hkYSBoZGInCiAqICAgICAgICAgICAgICAgICAgICAgICAgIFVzZWZ1bCB3aGVuIHVzaW5nIGlkZS1jZCBpbiBjb25qdW5jdGlvbiB3aXRoCiAqICAgICAgICAgICAgICAgICAgICAgICAgIGlkZS1zY3NpLiBUT0RPOiBub24tbW9kdWxhciB3YXkgb2YgZG9pbmcgdGhlCiAqICAgICAgICAgICAgICAgICAgICAgICAgIHNhbWUuCiAqCiAqIDQuNTQgIEF1ZyA1LCAxOTk5CS0gU3VwcG9ydCBmb3IgTU1DMiBjbGFzcyBjb21tYW5kcyB0aHJvdWdoIHRoZSBnZW5lcmljCiAqCQkJICBwYWNrZXQgaW50ZXJmYWNlIHRvIGNkcm9tLmMuCiAqCQkJLSBVbmlmaWVkIGF1ZGlvIGlvY3RsIHN1cHBvcnQsIG1vc3Qgb2YgaXQuCiAqCQkJLSBjbGVhbmVkIHVwIHZhcmlvdXMgZGVwcmVjYXRlZCB2ZXJpZnlfYXJlYSgpLgogKgkJCS0gQWRkZWQgaWRlX2Nkcm9tX3BhY2tldCgpIGFzIHRoZSBpbnRlcmZhY2UgZm9yCiAqCQkJICB0aGUgVW5pZm9ybSBnZW5lcmljX3BhY2tldCgpLgogKgkJCS0gYnVuY2ggb2Ygb3RoZXIgc3R1ZmYsIHdpbGwgZmlsbCBpbiBsb2dzIGxhdGVyLgogKgkJCS0gcmVwb3J0IDEgc2xvdCBmb3Igbm9uLWNoYW5nZXJzLCBsaWtlIHRoZSBvdGhlcgogKgkJCSAgY2Qtcm9tIGRyaXZlcnMuIGRvbid0IHJlcG9ydCBzZWxlY3QgZGlzYyBmb3IKICoJCQkgIG5vbi1jaGFuZ2VycyBhcyB3ZWxsLgogKgkJCS0gbWFzayBvdXQgYXVkaW8gcGxheWluZywgaWYgdGhlIGRldmljZSBjYW4ndCBkbyBpdC4KICoKICogNC41NSAgU2VwIDEsIDE5OTkJLSBFbGltaW5hdGVkIHRoZSByZXN0IG9mIHRoZSBhdWRpbyBpb2N0bHMsIGV4Y2VwdAogKgkJCSAgZm9yIENEUk9NUkVBRFRPQ1tFTlRSWXxIRUFERVJdLiBTb21lIG9mIHRoZSBkcml2ZXJzCiAqCQkJICB1c2UgdGhpcyBpbmRlcGVuZGVudGx5IG9mIHRoZSBhY3R1YWwgYXVkaW8gaGFuZGxpbmcuCiAqCQkJICBUaGV5IHdpbGwgZGlzYXBwZWFyIGxhdGVyIHdoZW4gSSBnZXQgdGhlIHRpbWUgdG8KICoJCQkgIGRvIGl0IGNsZWFubHkuCiAqCQkJLSBNaW5pbWl6ZSB0aGUgVE9DIHJlYWRpbmcgLSBvbmx5IGRvIGl0IHdoZW4gd2UKICoJCQkgIGtub3cgYSBtZWRpYSBjaGFuZ2UgaGFzIG9jY3VycmVkLgogKgkJCS0gTW92ZWQgYWxsIHRoZSBDRFJPTVJFQUR4IGlvY3RscyB0byB0aGUgVW5pZm9ybSBsYXllci4KICoJCQktIEhlaWtvIEVpc3NmZWxkdCA8aGVpa29AY29sb3NzdXMuZXNjYXBlLmRlPiBzdXBwbGllZAogKgkJCSAgc29tZSBmaXhlcyBmb3IgQ0RJLgogKgkJCS0gQ0QtUk9NIGxlYXZpbmcgZG9vciBsb2NrZWQgZml4IGZyb20gQW5kcmllcwogKgkJCSAgQnJvdXdlciA8QW5kcmllcy5Ccm91d2VyQGN3aS5ubD4KICoJCQktIEVyaWsgQW5kZXJzZW4gPGFuZGVyc2VuQHhtaXNzaW9uLmNvbT4gdW5pZmllZAogKgkJCSAgY29tbWFuZHMgYWNyb3NzIHRoZSB2YXJpb3VzIGRyaXZlcnMgYW5kIGhvdwogKgkJCSAgc2Vuc2UgZXJyb3JzIGFyZSBoYW5kbGVkLgogKgogKiA0LjU2ICBTZXAgMTIsIDE5OTkJLSBSZW1vdmVkIGNoYW5nZXIgc3VwcG9ydCAtIGl0IGlzIG5vdyBpbiB0aGUKICoJCQkgIFVuaWZvcm0gbGF5ZXIuCiAqCQkJLSBBZGRlZCBwYXJ0aXRpb24gYmFzZWQgbXVsdGlzZXNzaW9uIGhhbmRsaW5nLgogKgkJCS0gTW9kZSBzZW5zZSBhbmQgbW9kZSBzZWxlY3QgbW92ZWQgdG8gdGhlCiAqCQkJICBVbmlmb3JtIGxheWVyLgogKgkJCS0gRml4ZWQgYSBwcm9ibGVtIHdpdGggV1BJIENEUy0zMlggZHJpdmUgLSBpdAogKgkJCSAgZmFpbGVkIHRoZSBjYXBhYmlsaXRpZXMgCiAqCiAqIDQuNTcgIEFwciA3LCAyMDAwCS0gRml4ZWQgc2Vuc2UgcmVwb3J0aW5nLgogKgkJCS0gRml4ZWQgcG9zc2libGUgb29wcyBpbiBpZGVfY2Ryb21fZ2V0X2xhc3Rfc2Vzc2lvbigpCiAqCQkJLSBGaXggbG9ja2luZyBtYW5pYSBhbmQgbWFrZSBpZGVfY2Ryb21fcmVzZXQgcmVsb2NrCiAqCQkJLSBTdG9wIHNwZXdpbmcgZXJyb3JzIHRvIGxvZyB3aGVuIG1hZ2ljZGV2IHBvbGxzIHdpdGgKICoJCQkgIFRFU1RfVU5JVF9SRUFEWSBvbiBzb21lIGRyaXZlcy4KICoJCQktIFZhcmlvdXMgZml4ZXMgZnJvbSBUb2JpYXMgUmluZ3N0cm9tOgogKgkJCSAgdHJheSBpZiBpdCB3YXMgbG9ja2VkIHByaW9yIHRvIHRoZSByZXNldC4KICoJCQkgIC0gY2Ryb21fcmVhZF9jYXBhY2l0eSByZXR1cm5zIG9uZSBmcmFtZSB0b28gbGl0dGxlLgogKgkJCSAgLSBGaXggcmVhbCBjYXBhY2l0eSByZXBvcnRpbmcuCiAqCiAqIDQuNTggIE1heSAxLCAyMDAwCS0gQ2xlYW4gdXAgQUNFUjUwIHN0dWZmLgogKgkJCS0gRml4IHNtYWxsIHByb2JsZW0gd2l0aCBpZGVfY2Ryb21fY2FwYWNpdHkKICoKICogNC41OSAgQXVnIDExLCAyMDAwCS0gRml4IGNoYW5nZXIgcHJvYmxlbSBpbiBjZHJvbV9yZWFkX3RvYywgd2Ugd2VyZW4ndAogKgkJCSAgY29ycmVjdGx5IHNlbnNpbmcgYSBkaXNjIGNoYW5nZS4KICoJCQktIFJlYXJyYW5nZWQgc29tZSBjb2RlCiAqCQkJLSBVc2UgZXh0ZW5kZWQgc2Vuc2Ugb24gZHJpdmVzIHRoYXQgc3VwcG9ydCBpdCBmb3IKICoJCQkgIGNvcnJlY3RseSByZXBvcnRpbmcgdHJheSBzdGF0dXMgLS0gZnJvbQogKgkJCSAgTWljaGFlbCBEIEpvaG5zb24gPGpvaG5zb21Ab3JzdC5lZHU+CiAqIDQuNjAgIERlYyAxNywgMjAwMwktIEFkZCBtdCByYWluaWVyIHN1cHBvcnQKICoJCQktIEJ1bXAgdGltZW91dCBmb3IgcGFja2V0IGNvbW1hbmRzLCBtYXRjaGVzIHNyCiAqCQkJLSBPZGQgc3R1ZmYKICogNC42MSAgSmFuIDIyLCAyMDA0CS0gc3VwcG9ydCBoYXJkd2FyZSBzZWN0b3Igc2l6ZXMgb3RoZXIgdGhhbiAya0IsCiAqCQkJICBQYXNjYWwgU2NobWlkdCA8ZGVyLmVyZW1pdEBlbWFpbC5kZT4KICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCiAKI2RlZmluZSBJREVDRF9WRVJTSU9OICI0LjYxIgoKI2luY2x1ZGUgPGxpbnV4L21vZHVsZS5oPgojaW5jbHVkZSA8bGludXgvdHlwZXMuaD4KI2luY2x1ZGUgPGxpbnV4L2tlcm5lbC5oPgojaW5jbHVkZSA8bGludXgvZGVsYXkuaD4KI2luY2x1ZGUgPGxpbnV4L3RpbWVyLmg+CiNpbmNsdWRlIDxsaW51eC9zbGFiLmg+CiNpbmNsdWRlIDxsaW51eC9pbnRlcnJ1cHQuaD4KI2luY2x1ZGUgPGxpbnV4L2Vycm5vLmg+CiNpbmNsdWRlIDxsaW51eC9jZHJvbS5oPgojaW5jbHVkZSA8bGludXgvaWRlLmg+CiNpbmNsdWRlIDxsaW51eC9jb21wbGV0aW9uLmg+CiNpbmNsdWRlIDxsaW51eC9tdXRleC5oPgoKI2luY2x1ZGUgPHNjc2kvc2NzaS5oPgkvKiBGb3IgU0NTSSAtPiBBVEFQSSBjb21tYW5kIGNvbnZlcnNpb24gKi8KCiNpbmNsdWRlIDxhc20vaXJxLmg+CiNpbmNsdWRlIDxhc20vaW8uaD4KI2luY2x1ZGUgPGFzbS9ieXRlb3JkZXIuaD4KI2luY2x1ZGUgPGFzbS91YWNjZXNzLmg+CiNpbmNsdWRlIDxhc20vdW5hbGlnbmVkLmg+CgojaW5jbHVkZSAiaWRlLWNkLmgiCgpzdGF0aWMgREVGSU5FX01VVEVYKGlkZWNkX3JlZl9tdXRleCk7CgojZGVmaW5lIHRvX2lkZV9jZChvYmopIGNvbnRhaW5lcl9vZihvYmosIHN0cnVjdCBjZHJvbV9pbmZvLCBrcmVmKSAKCiNkZWZpbmUgaWRlX2NkX2coZGlzaykgXAoJY29udGFpbmVyX29mKChkaXNrKS0+cHJpdmF0ZV9kYXRhLCBzdHJ1Y3QgY2Ryb21faW5mbywgZHJpdmVyKQoKc3RhdGljIHN0cnVjdCBjZHJvbV9pbmZvICppZGVfY2RfZ2V0KHN0cnVjdCBnZW5kaXNrICpkaXNrKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqY2QgPSBOVUxMOwoKCW11dGV4X2xvY2soJmlkZWNkX3JlZl9tdXRleCk7CgljZCA9IGlkZV9jZF9nKGRpc2spOwoJaWYgKGNkKQoJCWtyZWZfZ2V0KCZjZC0+a3JlZik7CgltdXRleF91bmxvY2soJmlkZWNkX3JlZl9tdXRleCk7CglyZXR1cm4gY2Q7Cn0KCnN0YXRpYyB2b2lkIGlkZV9jZF9yZWxlYXNlKHN0cnVjdCBrcmVmICopOwoKc3RhdGljIHZvaWQgaWRlX2NkX3B1dChzdHJ1Y3QgY2Ryb21faW5mbyAqY2QpCnsKCW11dGV4X2xvY2soJmlkZWNkX3JlZl9tdXRleCk7CglrcmVmX3B1dCgmY2QtPmtyZWYsIGlkZV9jZF9yZWxlYXNlKTsKCW11dGV4X3VubG9jaygmaWRlY2RfcmVmX211dGV4KTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogR2VuZXJpYyBwYWNrZXQgY29tbWFuZCBzdXBwb3J0IGFuZCBlcnJvciBoYW5kbGluZyByb3V0aW5lcy4KICovCgovKiBNYXJrIHRoYXQgd2UndmUgc2VlbiBhIG1lZGlhIGNoYW5nZSwgYW5kIGludmFsaWRhdGUgb3VyIGludGVybmFsCiAgIGJ1ZmZlcnMuICovCnN0YXRpYyB2b2lkIGNkcm9tX3Nhd19tZWRpYV9jaGFuZ2UgKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgkKCUNEUk9NX1NUQVRFX0ZMQUdTIChkcml2ZSktPm1lZGlhX2NoYW5nZWQgPSAxOwoJQ0RST01fU1RBVEVfRkxBR1MgKGRyaXZlKS0+dG9jX3ZhbGlkID0gMDsKCWluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkID0gMDsKfQoKc3RhdGljIGludCBjZHJvbV9sb2dfc2Vuc2UoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEsCgkJCSAgIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJaW50IGxvZyA9IDA7CgoJaWYgKCFzZW5zZSB8fCAhcnEgfHwgKHJxLT5jbWRfZmxhZ3MgJiBSRVFfUVVJRVQpKQoJCXJldHVybiAwOwoKCXN3aXRjaCAoc2Vuc2UtPnNlbnNlX2tleSkgewoJCWNhc2UgTk9fU0VOU0U6IGNhc2UgUkVDT1ZFUkVEX0VSUk9SOgoJCQlicmVhazsKCQljYXNlIE5PVF9SRUFEWToKCQkJLyoKCQkJICogZG9uJ3QgY2FyZSBhYm91dCB0cmF5IHN0YXRlIG1lc3NhZ2VzIGZvcgoJCQkgKiBlLmcuIGNhcGFjaXR5IGNvbW1hbmRzIG9yIGluLXByb2dyZXNzIG9yCgkJCSAqIGJlY29taW5nIHJlYWR5CgkJCSAqLwoJCQlpZiAoc2Vuc2UtPmFzYyA9PSAweDNhIHx8IHNlbnNlLT5hc2MgPT0gMHgwNCkKCQkJCWJyZWFrOwoJCQlsb2cgPSAxOwoJCQlicmVhazsKCQljYXNlIElMTEVHQUxfUkVRVUVTVDoKCQkJLyoKCQkJICogZG9uJ3QgbG9nIFNUQVJUX1NUT1AgdW5pdCB3aXRoIExvRWogc2V0LCBzaW5jZQoJCQkgKiB3ZSBjYW5ub3QgcmVsaWFibHkgY2hlY2sgaWYgZHJpdmUgY2FuIGF1dG8tY2xvc2UKCQkJICovCgkJCWlmIChycS0+Y21kWzBdID09IEdQQ01EX1NUQVJUX1NUT1BfVU5JVCAmJiBzZW5zZS0+YXNjID09IDB4MjQpCgkJCQlicmVhazsKCQkJbG9nID0gMTsKCQkJYnJlYWs7CgkJY2FzZSBVTklUX0FUVEVOVElPTjoKCQkJLyoKCQkJICogTWFrZSBnb29kIGFuZCBzdXJlIHdlJ3ZlIHNlZW4gdGhpcyBwb3RlbnRpYWwgbWVkaWEKCQkJICogY2hhbmdlLiBTb21lIGRyaXZlcyAoaS5lLiBDcmVhdGl2ZSkgZmFpbCB0byBwcmVzZW50CgkJCSAqIHRoZSBjb3JyZWN0IHNlbnNlIGtleSBpbiB0aGUgZXJyb3IgcmVnaXN0ZXIuCgkJCSAqLwoJCQljZHJvbV9zYXdfbWVkaWFfY2hhbmdlKGRyaXZlKTsKCQkJYnJlYWs7CgkJZGVmYXVsdDoKCQkJbG9nID0gMTsKCQkJYnJlYWs7Cgl9CglyZXR1cm4gbG9nOwp9CgpzdGF0aWMKdm9pZCBjZHJvbV9hbmFseXplX3NlbnNlX2RhdGEoaWRlX2RyaXZlX3QgKmRyaXZlLAoJCQkgICAgICBzdHJ1Y3QgcmVxdWVzdCAqZmFpbGVkX2NvbW1hbmQsCgkJCSAgICAgIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJdW5zaWduZWQgbG9uZyBzZWN0b3I7Cgl1bnNpZ25lZCBsb25nIGJpb19zZWN0b3JzOwoJdW5zaWduZWQgbG9uZyB2YWxpZDsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoKCWlmICghY2Ryb21fbG9nX3NlbnNlKGRyaXZlLCBmYWlsZWRfY29tbWFuZCwgc2Vuc2UpKQoJCXJldHVybjsKCgkvKgoJICogSWYgYSByZWFkIHRvYyBpcyBleGVjdXRlZCBmb3IgYSBDRC1SIG9yIENELVJXIG1lZGl1bSB3aGVyZQoJICogdGhlIGZpcnN0IHRvYyBoYXMgbm90IGJlZW4gcmVjb3JkZWQgeWV0LCBpdCB3aWxsIGZhaWwgd2l0aAoJICogMDUvMjQvMDAgKHdoaWNoIGlzIGEgY29uZnVzaW5nIGVycm9yKQoJICovCglpZiAoZmFpbGVkX2NvbW1hbmQgJiYgZmFpbGVkX2NvbW1hbmQtPmNtZFswXSA9PSBHUENNRF9SRUFEX1RPQ19QTUFfQVRJUCkKCQlpZiAoc2Vuc2UtPnNlbnNlX2tleSA9PSAweDA1ICYmIHNlbnNlLT5hc2MgPT0gMHgyNCkKCQkJcmV0dXJuOwoKIAlpZiAoc2Vuc2UtPmVycm9yX2NvZGUgPT0gMHg3MCkgewkvKiBDdXJyZW50IEVycm9yICovCiAJCXN3aXRjaChzZW5zZS0+c2Vuc2Vfa2V5KSB7CgkJY2FzZSBNRURJVU1fRVJST1I6CgkJY2FzZSBWT0xVTUVfT1ZFUkZMT1c6CgkJY2FzZSBJTExFR0FMX1JFUVVFU1Q6CgkJCWlmICghc2Vuc2UtPnZhbGlkKQoJCQkJYnJlYWs7CgkJCWlmIChmYWlsZWRfY29tbWFuZCA9PSBOVUxMIHx8CgkJCQkJIWJsa19mc19yZXF1ZXN0KGZhaWxlZF9jb21tYW5kKSkKCQkJCWJyZWFrOwoJCQlzZWN0b3IgPSAoc2Vuc2UtPmluZm9ybWF0aW9uWzBdIDw8IDI0KSB8CgkJCQkgKHNlbnNlLT5pbmZvcm1hdGlvblsxXSA8PCAxNikgfAoJCQkJIChzZW5zZS0+aW5mb3JtYXRpb25bMl0gPDwgIDgpIHwKCQkJCSAoc2Vuc2UtPmluZm9ybWF0aW9uWzNdKTsKCgkJCWJpb19zZWN0b3JzID0gYmlvX3NlY3RvcnMoZmFpbGVkX2NvbW1hbmQtPmJpbyk7CgkJCWlmIChiaW9fc2VjdG9ycyA8IDQpCgkJCQliaW9fc2VjdG9ycyA9IDQ7CgkJCWlmIChkcml2ZS0+cXVldWUtPmhhcmRzZWN0X3NpemUgPT0gMjA0OCkKCQkJCXNlY3RvciA8PD0gMjsJLyogRGV2aWNlIHNlY3RvciBzaXplIGlzIDJLICovCgkJCXNlY3RvciAmPSB+KGJpb19zZWN0b3JzIC0xKTsKCQkJdmFsaWQgPSAoc2VjdG9yIC0gZmFpbGVkX2NvbW1hbmQtPnNlY3RvcikgPDwgOTsKCgkJCWlmICh2YWxpZCA8IDApCgkJCQl2YWxpZCA9IDA7CgkJCWlmIChzZWN0b3IgPCBnZXRfY2FwYWNpdHkoaW5mby0+ZGlzaykgJiYKCQkJCWRyaXZlLT5wcm9iZWRfY2FwYWNpdHkgLSBzZWN0b3IgPCA0ICogNzUpIHsKCQkJCXNldF9jYXBhY2l0eShpbmZvLT5kaXNrLCBzZWN0b3IpOwoJCQl9CiAJCX0KIAl9CiNpZiBWRVJCT1NFX0lERV9DRF9FUlJPUlMKCXsKCQlpbnQgaTsKCQljb25zdCBjaGFyICpzID0gImJhZCBzZW5zZSBrZXkhIjsKCQljaGFyIGJ1Zls4MF07CgoJCXByaW50ayAoIkFUQVBJIGRldmljZSAlczpcbiIsIGRyaXZlLT5uYW1lKTsKCQlpZiAoc2Vuc2UtPmVycm9yX2NvZGU9PTB4NzApCgkJCXByaW50aygiICBFcnJvcjogIik7CgkJZWxzZSBpZiAoc2Vuc2UtPmVycm9yX2NvZGU9PTB4NzEpCgkJCXByaW50aygiICBEZWZlcnJlZCBFcnJvcjogIik7CgkJZWxzZSBpZiAoc2Vuc2UtPmVycm9yX2NvZGUgPT0gMHg3ZikKCQkJcHJpbnRrKCIgIFZlbmRvci1zcGVjaWZpYyBFcnJvcjogIik7CgkJZWxzZQoJCQlwcmludGsoIiAgVW5rbm93biBFcnJvciBUeXBlOiAiKTsKCgkJaWYgKHNlbnNlLT5zZW5zZV9rZXkgPCBBUllfTEVOKHNlbnNlX2tleV90ZXh0cykpCgkJCXMgPSBzZW5zZV9rZXlfdGV4dHNbc2Vuc2UtPnNlbnNlX2tleV07CgoJCXByaW50aygiJXMgLS0gKFNlbnNlIGtleT0weCUwMngpXG4iLCBzLCBzZW5zZS0+c2Vuc2Vfa2V5KTsKCgkJaWYgKHNlbnNlLT5hc2MgPT0gMHg0MCkgewoJCQlzcHJpbnRmKGJ1ZiwgIkRpYWdub3N0aWMgZmFpbHVyZSBvbiBjb21wb25lbnQgMHglMDJ4IiwKCQkJCSBzZW5zZS0+YXNjcSk7CgkJCXMgPSBidWY7CgkJfSBlbHNlIHsKCQkJaW50IGxvID0gMCwgbWlkLCBoaSA9IEFSWV9MRU4oc2Vuc2VfZGF0YV90ZXh0cyk7CgkJCXVuc2lnbmVkIGxvbmcga2V5ID0gKHNlbnNlLT5zZW5zZV9rZXkgPDwgMTYpOwoJCQlrZXkgfD0gKHNlbnNlLT5hc2MgPDwgOCk7CgkJCWlmICghKHNlbnNlLT5hc2NxID49IDB4ODAgJiYgc2Vuc2UtPmFzY3EgPD0gMHhkZCkpCgkJCQlrZXkgfD0gc2Vuc2UtPmFzY3E7CgkJCXMgPSBOVUxMOwoKCQkJd2hpbGUgKGhpID4gbG8pIHsKCQkJCW1pZCA9IChsbyArIGhpKSAvIDI7CgkJCQlpZiAoc2Vuc2VfZGF0YV90ZXh0c1ttaWRdLmFzY19hc2NxID09IGtleSB8fAoJCQkJICAgIHNlbnNlX2RhdGFfdGV4dHNbbWlkXS5hc2NfYXNjcSA9PSAoMHhmZjAwMDB8a2V5KSkgewoJCQkJCXMgPSBzZW5zZV9kYXRhX3RleHRzW21pZF0udGV4dDsKCQkJCQlicmVhazsKCQkJCX0KCQkJCWVsc2UgaWYgKHNlbnNlX2RhdGFfdGV4dHNbbWlkXS5hc2NfYXNjcSA+IGtleSkKCQkJCQloaSA9IG1pZDsKCQkJCWVsc2UKCQkJCQlsbyA9IG1pZCsxOwoJCQl9CgkJfQoKCQlpZiAocyA9PSBOVUxMKSB7CgkJCWlmIChzZW5zZS0+YXNjID4gMHg4MCkKCQkJCXMgPSAiKHZlbmRvci1zcGVjaWZpYyBlcnJvcikiOwoJCQllbHNlCgkJCQlzID0gIihyZXNlcnZlZCBlcnJvciBjb2RlKSI7CgkJfQoKCQlwcmludGsoS0VSTl9FUlIgIiAgJXMgLS0gKGFzYz0weCUwMngsIGFzY3E9MHglMDJ4KVxuIiwKCQkJcywgc2Vuc2UtPmFzYywgc2Vuc2UtPmFzY3EpOwoKCQlpZiAoZmFpbGVkX2NvbW1hbmQgIT0gTlVMTCkgewoKCQkJaW50IGxvPTAsIG1pZCwgaGk9IEFSWV9MRU4gKHBhY2tldF9jb21tYW5kX3RleHRzKTsKCQkJcyA9IE5VTEw7CgoJCQl3aGlsZSAoaGkgPiBsbykgewoJCQkJbWlkID0gKGxvICsgaGkpIC8gMjsKCQkJCWlmIChwYWNrZXRfY29tbWFuZF90ZXh0c1ttaWRdLnBhY2tldF9jb21tYW5kID09CgkJCQkgICAgZmFpbGVkX2NvbW1hbmQtPmNtZFswXSkgewoJCQkJCXMgPSBwYWNrZXRfY29tbWFuZF90ZXh0c1ttaWRdLnRleHQ7CgkJCQkJYnJlYWs7CgkJCQl9CgkJCQlpZiAocGFja2V0X2NvbW1hbmRfdGV4dHNbbWlkXS5wYWNrZXRfY29tbWFuZCA+CgkJCQkgICAgZmFpbGVkX2NvbW1hbmQtPmNtZFswXSkKCQkJCQloaSA9IG1pZDsKCQkJCWVsc2UKCQkJCQlsbyA9IG1pZCsxOwoJCQl9CgoJCQlwcmludGsgKEtFUk5fRVJSICIgIFRoZSBmYWlsZWQgXCIlc1wiIHBhY2tldCBjb21tYW5kIHdhczogXG4gIFwiIiwgcyk7CgkJCWZvciAoaT0wOyBpPHNpemVvZiAoZmFpbGVkX2NvbW1hbmQtPmNtZCk7IGkrKykKCQkJCXByaW50ayAoIiUwMnggIiwgZmFpbGVkX2NvbW1hbmQtPmNtZFtpXSk7CgkJCXByaW50ayAoIlwiXG4iKTsKCQl9CgoJCS8qIFRoZSBTS1NWIGJpdCBzcGVjaWZpZXMgdmFsaWRpdHkgb2YgdGhlIHNlbnNlX2tleV9zcGVjaWZpYwoJCSAqIGluIHRoZSBuZXh0IHR3byBjb21tYW5kcy4gSXQgaXMgYml0IDcgb2YgdGhlIGZpcnN0IGJ5dGUuCgkJICogSW4gdGhlIGNhc2Ugb2YgTk9UX1JFQURZLCBpZiBTS1NWIGlzIHNldCB0aGUgZHJpdmUgY2FuCgkJICogZ2l2ZSB1cyBuaWNlIEVUQSByZWFkaW5ncy4KCQkgKi8KCQlpZiAoc2Vuc2UtPnNlbnNlX2tleSA9PSBOT1RfUkVBRFkgJiYgKHNlbnNlLT5za3NbMF0gJiAweDgwKSkgewoJCQlpbnQgcHJvZ3Jlc3MgPSAoc2Vuc2UtPnNrc1sxXSA8PCA4IHwgc2Vuc2UtPnNrc1syXSkgKiAxMDA7CgkJCXByaW50ayhLRVJOX0VSUiAiICBDb21tYW5kIGlzICUwMmQlJSBjb21wbGV0ZVxuIiwgcHJvZ3Jlc3MgLyAweGZmZmYpOwoKCQl9CgoJCWlmIChzZW5zZS0+c2Vuc2Vfa2V5ID09IElMTEVHQUxfUkVRVUVTVCAmJgoJCSAgICAoc2Vuc2UtPnNrc1swXSAmIDB4ODApICE9IDApIHsKCQkJcHJpbnRrKEtFUk5fRVJSICIgIEVycm9yIGluICVzIGJ5dGUgJWQiLAoJCQkJKHNlbnNlLT5za3NbMF0gJiAweDQwKSAhPSAwID8KCQkJCSJjb21tYW5kIHBhY2tldCIgOiAiY29tbWFuZCBkYXRhIiwKCQkJCShzZW5zZS0+c2tzWzFdIDw8IDgpICsgc2Vuc2UtPnNrc1syXSk7CgoJCQlpZiAoKHNlbnNlLT5za3NbMF0gJiAweDQwKSAhPSAwKQoJCQkJcHJpbnRrICgiIGJpdCAlZCIsIHNlbnNlLT5za3NbMF0gJiAweDA3KTsKCgkJCXByaW50ayAoIlxuIik7CgkJfQoJfQoKI2Vsc2UgLyogbm90IFZFUkJPU0VfSURFX0NEX0VSUk9SUyAqLwoKCS8qIFN1cHByZXNzIHByaW50aW5nIHVuaXQgYXR0ZW50aW9uIGFuZCBgaW4gcHJvZ3Jlc3Mgb2YgYmVjb21pbmcgcmVhZHknCgkgICBlcnJvcnMgd2hlbiB3ZSdyZSBub3QgYmVpbmcgdmVyYm9zZS4gKi8KCglpZiAoc2Vuc2UtPnNlbnNlX2tleSA9PSBVTklUX0FUVEVOVElPTiB8fAoJICAgIChzZW5zZS0+c2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSAmJiAoc2Vuc2UtPmFzYyA9PSA0IHx8CgkJCQkJCXNlbnNlLT5hc2MgPT0gMHgzYSkpKQoJCXJldHVybjsKCglwcmludGsoS0VSTl9FUlIgIiVzOiBlcnJvciBjb2RlOiAweCUwMnggIHNlbnNlX2tleTogMHglMDJ4ICBhc2M6IDB4JTAyeCAgYXNjcTogMHglMDJ4XG4iLAoJCWRyaXZlLT5uYW1lLAoJCXNlbnNlLT5lcnJvcl9jb2RlLCBzZW5zZS0+c2Vuc2Vfa2V5LAoJCXNlbnNlLT5hc2MsIHNlbnNlLT5hc2NxKTsKI2VuZGlmIC8qIG5vdCBWRVJCT1NFX0lERV9DRF9FUlJPUlMgKi8KfQoKLyoKICogSW5pdGlhbGl6ZSBhIGlkZS1jZCBwYWNrZXQgY29tbWFuZCByZXF1ZXN0CiAqLwpzdGF0aWMgdm9pZCBjZHJvbV9wcmVwYXJlX3JlcXVlc3QoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICpjZCA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCglpZGVfaW5pdF9kcml2ZV9jbWQocnEpOwoJcnEtPmNtZF90eXBlID0gUkVRX1RZUEVfQVRBX1BDOwoJcnEtPnJxX2Rpc2sgPSBjZC0+ZGlzazsKfQoKc3RhdGljIHZvaWQgY2Ryb21fcXVldWVfcmVxdWVzdF9zZW5zZShpZGVfZHJpdmVfdCAqZHJpdmUsIHZvaWQgKnNlbnNlLAoJCQkJICAgICAgc3RydWN0IHJlcXVlc3QgKmZhaWxlZF9jb21tYW5kKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbwkJPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgcmVxdWVzdCAqcnEJCT0gJmluZm8tPnJlcXVlc3Rfc2Vuc2VfcmVxdWVzdDsKCglpZiAoc2Vuc2UgPT0gTlVMTCkKCQlzZW5zZSA9ICZpbmZvLT5zZW5zZV9kYXRhOwoKCS8qIHN0dWZmIHRoZSBzZW5zZSByZXF1ZXN0IGluIGZyb250IG9mIG91ciBjdXJyZW50IHJlcXVlc3QgKi8KCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgcnEpOwoKCXJxLT5kYXRhID0gc2Vuc2U7CglycS0+Y21kWzBdID0gR1BDTURfUkVRVUVTVF9TRU5TRTsKCXJxLT5jbWRbNF0gPSBycS0+ZGF0YV9sZW4gPSAxODsKCglycS0+Y21kX3R5cGUgPSBSRVFfVFlQRV9TRU5TRTsKCgkvKiBOT1RFISBTYXZlIHRoZSBmYWlsZWQgY29tbWFuZCBpbiAicnEtPmJ1ZmZlciIgKi8KCXJxLT5idWZmZXIgPSAodm9pZCAqKSBmYWlsZWRfY29tbWFuZDsKCgkodm9pZCkgaWRlX2RvX2RyaXZlX2NtZChkcml2ZSwgcnEsIGlkZV9wcmVlbXB0KTsKfQoKc3RhdGljIHZvaWQgY2Ryb21fZW5kX3JlcXVlc3QgKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IHVwdG9kYXRlKQp7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CglpbnQgbnNlY3RvcnMgPSBycS0+aGFyZF9jdXJfc2VjdG9yczsKCglpZiAoYmxrX3NlbnNlX3JlcXVlc3QocnEpICYmIHVwdG9kYXRlKSB7CgkJLyoKCQkgKiBGb3IgUkVRX1RZUEVfU0VOU0UsICJycS0+YnVmZmVyIiBwb2ludHMgdG8gdGhlIG9yaWdpbmFsCgkJICogZmFpbGVkIHJlcXVlc3QKCQkgKi8KCQlzdHJ1Y3QgcmVxdWVzdCAqZmFpbGVkID0gKHN0cnVjdCByZXF1ZXN0ICopIHJxLT5idWZmZXI7CgkJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgkJdm9pZCAqc2Vuc2UgPSAmaW5mby0+c2Vuc2VfZGF0YTsKCQl1bnNpZ25lZCBsb25nIGZsYWdzOwoKCQlpZiAoZmFpbGVkKSB7CgkJCWlmIChmYWlsZWQtPnNlbnNlKSB7CgkJCQlzZW5zZSA9IGZhaWxlZC0+c2Vuc2U7CgkJCQlmYWlsZWQtPnNlbnNlX2xlbiA9IHJxLT5zZW5zZV9sZW47CgkJCX0KCQkJY2Ryb21fYW5hbHl6ZV9zZW5zZV9kYXRhKGRyaXZlLCBmYWlsZWQsIHNlbnNlKTsKCQkJLyoKCQkJICogbm93IGVuZCBmYWlsZWQgcmVxdWVzdAoJCQkgKi8KCQkJaWYgKGJsa19mc19yZXF1ZXN0KGZhaWxlZCkpIHsKCQkJCWlmIChpZGVfZW5kX2RlcXVldWVkX3JlcXVlc3QoZHJpdmUsIGZhaWxlZCwgMCwKCQkJCQkJZmFpbGVkLT5oYXJkX25yX3NlY3RvcnMpKQoJCQkJCUJVRygpOwoJCQl9IGVsc2UgewoJCQkJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CgkJCQllbmRfdGhhdF9yZXF1ZXN0X2NodW5rKGZhaWxlZCwgMCwKCQkJCQkJCWZhaWxlZC0+ZGF0YV9sZW4pOwoJCQkJZW5kX3RoYXRfcmVxdWVzdF9sYXN0KGZhaWxlZCwgMCk7CgkJCQlzcGluX3VubG9ja19pcnFyZXN0b3JlKCZpZGVfbG9jaywgZmxhZ3MpOwoJCQl9CgkJfSBlbHNlCgkJCWNkcm9tX2FuYWx5emVfc2Vuc2VfZGF0YShkcml2ZSwgTlVMTCwgc2Vuc2UpOwoJfQoKCWlmICghcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyAmJiBibGtfZnNfcmVxdWVzdChycSkpCgkJdXB0b2RhdGUgPSAxOwoJLyogbWFrZSBzdXJlIGl0J3MgZnVsbHkgZW5kZWQgKi8KCWlmIChibGtfcGNfcmVxdWVzdChycSkpCgkJbnNlY3RvcnMgPSAocnEtPmRhdGFfbGVuICsgNTExKSA+PiA5OwoJaWYgKCFuc2VjdG9ycykKCQluc2VjdG9ycyA9IDE7CgoJaWRlX2VuZF9yZXF1ZXN0KGRyaXZlLCB1cHRvZGF0ZSwgbnNlY3RvcnMpOwp9CgpzdGF0aWMgdm9pZCBpZGVfZHVtcF9zdGF0dXNfbm9fc2Vuc2UoaWRlX2RyaXZlX3QgKmRyaXZlLCBjb25zdCBjaGFyICptc2csIHU4IHN0YXQpCnsKCWlmIChzdGF0ICYgMHg4MCkKCQlyZXR1cm47CglpZGVfZHVtcF9zdGF0dXMoZHJpdmUsIG1zZywgc3RhdCk7Cn0KCi8qIFJldHVybnMgMCBpZiB0aGUgcmVxdWVzdCBzaG91bGQgYmUgY29udGludWVkLgogICBSZXR1cm5zIDEgaWYgdGhlIHJlcXVlc3Qgd2FzIGVuZGVkLiAqLwpzdGF0aWMgaW50IGNkcm9tX2RlY29kZV9zdGF0dXMoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgZ29vZF9zdGF0LCBpbnQgKnN0YXRfcmV0KQp7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CglpbnQgc3RhdCwgZXJyLCBzZW5zZV9rZXk7CgkKCS8qIENoZWNrIGZvciBlcnJvcnMuICovCglzdGF0ID0gSFdJRihkcml2ZSktPklOQihJREVfU1RBVFVTX1JFRyk7CglpZiAoc3RhdF9yZXQpCgkJKnN0YXRfcmV0ID0gc3RhdDsKCglpZiAoT0tfU1RBVChzdGF0LCBnb29kX3N0YXQsIEJBRF9SX1NUQVQpKQoJCXJldHVybiAwOwoKCS8qIEdldCB0aGUgSURFIGVycm9yIHJlZ2lzdGVyLiAqLwoJZXJyID0gSFdJRihkcml2ZSktPklOQihJREVfRVJST1JfUkVHKTsKCXNlbnNlX2tleSA9IGVyciA+PiA0OwoKCWlmIChycSA9PSBOVUxMKSB7CgkJcHJpbnRrKCIlczogbWlzc2luZyBycSBpbiBjZHJvbV9kZWNvZGVfc3RhdHVzXG4iLCBkcml2ZS0+bmFtZSk7CgkJcmV0dXJuIDE7Cgl9CgoJaWYgKGJsa19zZW5zZV9yZXF1ZXN0KHJxKSkgewoJCS8qIFdlIGdvdCBhbiBlcnJvciB0cnlpbmcgdG8gZ2V0IHNlbnNlIGluZm8KCQkgICBmcm9tIHRoZSBkcml2ZSAocHJvYmFibHkgd2hpbGUgdHJ5aW5nCgkJICAgdG8gcmVjb3ZlciBmcm9tIGEgZm9ybWVyIGVycm9yKS4gIEp1c3QgZ2l2ZSB1cC4gKi8KCgkJcnEtPmNtZF9mbGFncyB8PSBSRVFfRkFJTEVEOwoJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQlpZGVfZXJyb3IoZHJpdmUsICJyZXF1ZXN0IHNlbnNlIGZhaWx1cmUiLCBzdGF0KTsKCQlyZXR1cm4gMTsKCgl9IGVsc2UgaWYgKGJsa19wY19yZXF1ZXN0KHJxKSB8fCBycS0+Y21kX3R5cGUgPT0gUkVRX1RZUEVfQVRBX1BDKSB7CgkJLyogQWxsIG90aGVyIGZ1bmN0aW9ucywgZXhjZXB0IGZvciBSRUFELiAqLwoJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJCS8qCgkJICogaWYgd2UgaGF2ZSBhbiBlcnJvciwgcGFzcyBiYWNrIENIRUNLX0NPTkRJVElPTiBhcyB0aGUKCQkgKiBzY3NpIHN0YXR1cyBieXRlCgkJICovCgkJaWYgKGJsa19wY19yZXF1ZXN0KHJxKSAmJiAhcnEtPmVycm9ycykKCQkJcnEtPmVycm9ycyA9IFNBTV9TVEFUX0NIRUNLX0NPTkRJVElPTjsKCgkJLyogQ2hlY2sgZm9yIHRyYXkgb3Blbi4gKi8KCQlpZiAoc2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSkgewoJCQljZHJvbV9zYXdfbWVkaWFfY2hhbmdlIChkcml2ZSk7CgkJfSBlbHNlIGlmIChzZW5zZV9rZXkgPT0gVU5JVF9BVFRFTlRJT04pIHsKCQkJLyogQ2hlY2sgZm9yIG1lZGlhIGNoYW5nZS4gKi8KCQkJY2Ryb21fc2F3X21lZGlhX2NoYW5nZSAoZHJpdmUpOwoJCQkvKnByaW50aygiJXM6IG1lZGlhIGNoYW5nZWRcbiIsZHJpdmUtPm5hbWUpOyovCgkJCXJldHVybiAwOwogCQl9IGVsc2UgaWYgKChzZW5zZV9rZXkgPT0gSUxMRUdBTF9SRVFVRVNUKSAmJgogCQkJICAgKHJxLT5jbWRbMF0gPT0gR1BDTURfU1RBUlRfU1RPUF9VTklUKSkgewogCQkJLyoKIAkJCSAqIERvbid0IHByaW50IGVycm9yIG1lc3NhZ2UgZm9yIHRoaXMgY29uZGl0aW9uLS0KIAkJCSAqIFNGRjgwOTBpIGluZGljYXRlcyB0aGF0IDUvMjQvMDAgaXMgdGhlIGNvcnJlY3QKIAkJCSAqIHJlc3BvbnNlIHRvIGEgcmVxdWVzdCB0byBjbG9zZSB0aGUgdHJheSBpZiB0aGUKIAkJCSAqIGRyaXZlIGRvZXNuJ3QgaGF2ZSB0aGF0IGNhcGFiaWxpdHkuCiAJCQkgKiBjZHJvbV9sb2dfc2Vuc2UoKSBrbm93cyB0aGlzIQogCQkJICovCgkJfSBlbHNlIGlmICghKHJxLT5jbWRfZmxhZ3MgJiBSRVFfUVVJRVQpKSB7CgkJCS8qIE90aGVyd2lzZSwgcHJpbnQgYW4gZXJyb3IuICovCgkJCWlkZV9kdW1wX3N0YXR1cyhkcml2ZSwgInBhY2tldCBjb21tYW5kIGVycm9yIiwgc3RhdCk7CgkJfQoJCQoJCXJxLT5jbWRfZmxhZ3MgfD0gUkVRX0ZBSUxFRDsKCgkJLyoKCQkgKiBpbnN0ZWFkIG9mIHBsYXlpbmcgZ2FtZXMgd2l0aCBtb3ZpbmcgY29tcGxldGlvbnMgYXJvdW5kLAoJCSAqIHJlbW92ZSBmYWlsZWQgcmVxdWVzdCBjb21wbGV0ZWx5IGFuZCBlbmQgaXQgd2hlbiB0aGUKCQkgKiByZXF1ZXN0IHNlbnNlIGhhcyBjb21wbGV0ZWQKCQkgKi8KCQlpZiAoc3RhdCAmIEVSUl9TVEFUKSB7CgkJCXNwaW5fbG9ja19pcnFzYXZlKCZpZGVfbG9jaywgZmxhZ3MpOwoJCQlibGtkZXZfZGVxdWV1ZV9yZXF1ZXN0KHJxKTsKCQkJSFdHUk9VUChkcml2ZSktPnJxID0gTlVMTDsKCQkJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmaWRlX2xvY2ssIGZsYWdzKTsKCgkJCWNkcm9tX3F1ZXVlX3JlcXVlc3Rfc2Vuc2UoZHJpdmUsIHJxLT5zZW5zZSwgcnEpOwoJCX0gZWxzZQoJCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7CgoJfSBlbHNlIGlmIChibGtfZnNfcmVxdWVzdChycSkpIHsKCQlpbnQgZG9fZW5kX3JlcXVlc3QgPSAwOwoKCQkvKiBIYW5kbGUgZXJyb3JzIGZyb20gUkVBRCBhbmQgV1JJVEUgcmVxdWVzdHMuICovCgoJCWlmIChibGtfbm9yZXRyeV9yZXF1ZXN0KHJxKSkKCQkJZG9fZW5kX3JlcXVlc3QgPSAxOwoKCQlpZiAoc2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSkgewoJCQkvKiBUcmF5IG9wZW4uICovCgkJCWlmIChycV9kYXRhX2RpcihycSkgPT0gUkVBRCkgewoJCQkJY2Ryb21fc2F3X21lZGlhX2NoYW5nZSAoZHJpdmUpOwoKCQkJCS8qIEZhaWwgdGhlIHJlcXVlc3QuICovCgkJCQlwcmludGsgKCIlczogdHJheSBvcGVuXG4iLCBkcml2ZS0+bmFtZSk7CgkJCQlkb19lbmRfcmVxdWVzdCA9IDE7CgkJCX0gZWxzZSB7CgkJCQlzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCgkJCQkvKiBhbGxvdyB0aGUgZHJpdmUgNSBzZWNvbmRzIHRvIHJlY292ZXIsIHNvbWUKCQkJCSAqIGRldmljZXMgd2lsbCByZXR1cm4gdGhpcyBlcnJvciB3aGlsZSBmbHVzaGluZwoJCQkJICogZGF0YSBmcm9tIGNhY2hlICovCgkJCQlpZiAoIXJxLT5lcnJvcnMpCgkJCQkJaW5mby0+d3JpdGVfdGltZW91dCA9IGppZmZpZXMgKyBBVEFQSV9XQUlUX1dSSVRFX0JVU1k7CgkJCQlycS0+ZXJyb3JzID0gMTsKCQkJCWlmICh0aW1lX2FmdGVyKGppZmZpZXMsIGluZm8tPndyaXRlX3RpbWVvdXQpKQoJCQkJCWRvX2VuZF9yZXF1ZXN0ID0gMTsKCQkJCWVsc2UgewoJCQkJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJCQkJCS8qCgkJCQkJICogdGFrZSBhIGJyZWF0aGVyIHJlbHlpbmcgb24gdGhlCgkJCQkJICogdW5wbHVnIHRpbWVyIHRvIGtpY2sgdXMgYWdhaW4KCQkJCQkgKi8KCQkJCQlzcGluX2xvY2tfaXJxc2F2ZSgmaWRlX2xvY2ssIGZsYWdzKTsKCQkJCQlibGtfcGx1Z19kZXZpY2UoZHJpdmUtPnF1ZXVlKTsKCQkJCQlzcGluX3VubG9ja19pcnFyZXN0b3JlKCZpZGVfbG9jayxmbGFncyk7CgkJCQkJcmV0dXJuIDE7CgkJCQl9CgkJCX0KCQl9IGVsc2UgaWYgKHNlbnNlX2tleSA9PSBVTklUX0FUVEVOVElPTikgewoJCQkvKiBNZWRpYSBjaGFuZ2UuICovCgkJCWNkcm9tX3Nhd19tZWRpYV9jaGFuZ2UgKGRyaXZlKTsKCgkJCS8qIEFycmFuZ2UgdG8gcmV0cnkgdGhlIHJlcXVlc3QuCgkJCSAgIEJ1dCBiZSBzdXJlIHRvIGdpdmUgdXAgaWYgd2UndmUgcmV0cmllZAoJCQkgICB0b28gbWFueSB0aW1lcy4gKi8KCQkJaWYgKCsrcnEtPmVycm9ycyA+IEVSUk9SX01BWCkKCQkJCWRvX2VuZF9yZXF1ZXN0ID0gMTsKCQl9IGVsc2UgaWYgKHNlbnNlX2tleSA9PSBJTExFR0FMX1JFUVVFU1QgfHwKCQkJICAgc2Vuc2Vfa2V5ID09IERBVEFfUFJPVEVDVCkgewoJCQkvKiBObyBwb2ludCBpbiByZXRyeWluZyBhZnRlciBhbiBpbGxlZ2FsCgkJCSAgIHJlcXVlc3Qgb3IgZGF0YSBwcm90ZWN0IGVycm9yLiovCgkJCWlkZV9kdW1wX3N0YXR1c19ub19zZW5zZSAoZHJpdmUsICJjb21tYW5kIGVycm9yIiwgc3RhdCk7CgkJCWRvX2VuZF9yZXF1ZXN0ID0gMTsKCQl9IGVsc2UgaWYgKHNlbnNlX2tleSA9PSBNRURJVU1fRVJST1IpIHsKCQkJLyogTm8gcG9pbnQgaW4gcmUtdHJ5aW5nIGEgemlsbGlvbiB0aW1lcyBvbiBhIGJhZCAKCQkJICogc2VjdG9yLi4uICBJZiB3ZSBnb3QgaGVyZSB0aGUgZXJyb3IgaXMgbm90IGNvcnJlY3RhYmxlICovCgkJCWlkZV9kdW1wX3N0YXR1c19ub19zZW5zZSAoZHJpdmUsICJtZWRpYSBlcnJvciAoYmFkIHNlY3RvcikiLCBzdGF0KTsKCQkJZG9fZW5kX3JlcXVlc3QgPSAxOwoJCX0gZWxzZSBpZiAoc2Vuc2Vfa2V5ID09IEJMQU5LX0NIRUNLKSB7CgkJCS8qIERpc2sgYXBwZWFycyBibGFuayA/PyAqLwoJCQlpZGVfZHVtcF9zdGF0dXNfbm9fc2Vuc2UgKGRyaXZlLCAibWVkaWEgZXJyb3IgKGJsYW5rKSIsIHN0YXQpOwoJCQlkb19lbmRfcmVxdWVzdCA9IDE7CgkJfSBlbHNlIGlmICgoZXJyICYgfkFCUlRfRVJSKSAhPSAwKSB7CgkJCS8qIEdvIHRvIHRoZSBkZWZhdWx0IGhhbmRsZXIKCQkJICAgZm9yIG90aGVyIGVycm9ycy4gKi8KCQkJaWRlX2Vycm9yKGRyaXZlLCAiY2Ryb21fZGVjb2RlX3N0YXR1cyIsIHN0YXQpOwoJCQlyZXR1cm4gMTsKCQl9IGVsc2UgaWYgKCgrK3JxLT5lcnJvcnMgPiBFUlJPUl9NQVgpKSB7CgkJCS8qIFdlJ3ZlIHJhY2tlZCB1cCB0b28gbWFueSByZXRyaWVzLiAgQWJvcnQuICovCgkJCWRvX2VuZF9yZXF1ZXN0ID0gMTsKCQl9CgoJCS8qIEVuZCBhIHJlcXVlc3QgdGhyb3VnaCByZXF1ZXN0IHNlbnNlIGFuYWx5c2lzIHdoZW4gd2UgaGF2ZQoJCSAgIHNlbnNlIGRhdGEuIFdlIG5lZWQgdGhpcyBpbiBvcmRlciB0byBwZXJmb3JtIGVuZCBvZiBtZWRpYQoJCSAgIHByb2Nlc3NpbmcgKi8KCgkJaWYgKGRvX2VuZF9yZXF1ZXN0KSB7CgkJCWlmIChzdGF0ICYgRVJSX1NUQVQpIHsKCQkJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgkJCQlzcGluX2xvY2tfaXJxc2F2ZSgmaWRlX2xvY2ssIGZsYWdzKTsKCQkJCWJsa2Rldl9kZXF1ZXVlX3JlcXVlc3QocnEpOwoJCQkJSFdHUk9VUChkcml2ZSktPnJxID0gTlVMTDsKCQkJCXNwaW5fdW5sb2NrX2lycXJlc3RvcmUoJmlkZV9sb2NrLCBmbGFncyk7CgoJCQkJY2Ryb21fcXVldWVfcmVxdWVzdF9zZW5zZShkcml2ZSwgcnEtPnNlbnNlLCBycSk7CgkJCX0gZWxzZQoJCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJCX0gZWxzZSB7CgkJCS8qIElmIHdlIGdvdCBhIENIRUNLX0NPTkRJVElPTiBzdGF0dXMsCgkJCSAgIHF1ZXVlIGEgcmVxdWVzdCBzZW5zZSBjb21tYW5kLiAqLwoJCQlpZiAoc3RhdCAmIEVSUl9TVEFUKQoJCQkJY2Ryb21fcXVldWVfcmVxdWVzdF9zZW5zZShkcml2ZSwgTlVMTCwgTlVMTCk7CgkJfQoJfSBlbHNlIHsKCQlibGtfZHVtcF9ycV9mbGFncyhycSwgImlkZS1jZDogYmFkIHJxIik7CgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJfQoKCS8qIFJldHJ5LCBvciBoYW5kbGUgdGhlIG5leHQgcmVxdWVzdC4gKi8KCXJldHVybiAxOwp9CgpzdGF0aWMgaW50IGNkcm9tX3RpbWVyX2V4cGlyeShpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCXVuc2lnbmVkIGxvbmcgd2FpdCA9IDA7CgoJLyoKCSAqIFNvbWUgY29tbWFuZHMgYXJlICpzbG93KiBhbmQgbm9ybWFsbHkgdGFrZSBhIGxvbmcgdGltZSB0bwoJICogY29tcGxldGUuIFVzdWFsbHkgd2UgY2FuIHVzZSB0aGUgQVRBUEkgImRpc2Nvbm5lY3QiIHRvIGJ5cGFzcwoJICogdGhpcywgYnV0IG5vdCBhbGwgY29tbWFuZHMvZHJpdmVzIHN1cHBvcnQgdGhhdC4gTGV0CgkgKiBpZGVfdGltZXJfZXhwaXJ5IGtlZXAgcG9sbGluZyB1cyBmb3IgdGhlc2UuCgkgKi8KCXN3aXRjaCAocnEtPmNtZFswXSkgewoJCWNhc2UgR1BDTURfQkxBTks6CgkJY2FzZSBHUENNRF9GT1JNQVRfVU5JVDoKCQljYXNlIEdQQ01EX1JFU0VSVkVfUlpPTkVfVFJBQ0s6CgkJY2FzZSBHUENNRF9DTE9TRV9UUkFDSzoKCQljYXNlIEdQQ01EX0ZMVVNIX0NBQ0hFOgoJCQl3YWl0ID0gQVRBUElfV0FJVF9QQzsKCQkJYnJlYWs7CgkJZGVmYXVsdDoKCQkJaWYgKCEocnEtPmNtZF9mbGFncyAmIFJFUV9RVUlFVCkpCgkJCQlwcmludGsoS0VSTl9JTkZPICJpZGUtY2Q6IGNtZCAweCV4IHRpbWVkIG91dFxuIiwgcnEtPmNtZFswXSk7CgkJCXdhaXQgPSAwOwoJCQlicmVhazsKCX0KCXJldHVybiB3YWl0Owp9CgovKiBTZXQgdXAgdGhlIGRldmljZSByZWdpc3RlcnMgZm9yIHRyYW5zZmVycmluZyBhIHBhY2tldCBjb21tYW5kIG9uIERFViwKICAgZXhwZWN0aW5nIHRvIGxhdGVyIHRyYW5zZmVyIFhGRVJMRU4gYnl0ZXMuICBIQU5ETEVSIGlzIHRoZSByb3V0aW5lCiAgIHdoaWNoIGFjdHVhbGx5IHRyYW5zZmVycyB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUuICBJZiB0aGlzIGlzIGEKICAgZHJxX2ludGVycnVwdCBkZXZpY2UsIHRoaXMgcm91dGluZSB3aWxsIGFycmFuZ2UgZm9yIEhBTkRMRVIgdG8gYmUKICAgY2FsbGVkIHdoZW4gdGhlIGludGVycnVwdCBmcm9tIHRoZSBkcml2ZSBhcnJpdmVzLiAgT3RoZXJ3aXNlLCBIQU5ETEVSCiAgIHdpbGwgYmUgY2FsbGVkIGltbWVkaWF0ZWx5IGFmdGVyIHRoZSBkcml2ZSBpcyBwcmVwYXJlZCBmb3IgdGhlIHRyYW5zZmVyLiAqLwoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9zdGFydF9wYWNrZXRfY29tbWFuZChpZGVfZHJpdmVfdCAqZHJpdmUsCgkJCQkJCSAgaW50IHhmZXJsZW4sCgkJCQkJCSAgaWRlX2hhbmRsZXJfdCAqaGFuZGxlcikKewoJaWRlX3N0YXJ0c3RvcF90IHN0YXJ0c3RvcDsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJaWRlX2h3aWZfdCAqaHdpZiA9IGRyaXZlLT5od2lmOwoKCS8qIFdhaXQgZm9yIHRoZSBjb250cm9sbGVyIHRvIGJlIGlkbGUuICovCglpZiAoaWRlX3dhaXRfc3RhdCgmc3RhcnRzdG9wLCBkcml2ZSwgMCwgQlVTWV9TVEFULCBXQUlUX1JFQURZKSkKCQlyZXR1cm4gc3RhcnRzdG9wOwoKCWlmIChpbmZvLT5kbWEpCgkJaW5mby0+ZG1hID0gIWh3aWYtPmRtYV9zZXR1cChkcml2ZSk7CgoJLyogU2V0IHVwIHRoZSBjb250cm9sbGVyIHJlZ2lzdGVycy4gKi8KCS8qIEZJWE1FOiBmb3IgVmlydHVhbCBETUEgd2UgbXVzdCBjaGVjayBoYXJkZXIgKi8KCUhXSUYoZHJpdmUpLT5PVVRCKGluZm8tPmRtYSwgSURFX0ZFQVRVUkVfUkVHKTsKCUhXSUYoZHJpdmUpLT5PVVRCKDAsIElERV9JUkVBU09OX1JFRyk7CglIV0lGKGRyaXZlKS0+T1VUQigwLCBJREVfU0VDVE9SX1JFRyk7CgoJSFdJRihkcml2ZSktPk9VVEIoeGZlcmxlbiAmIDB4ZmYsIElERV9CQ09VTlRMX1JFRyk7CglIV0lGKGRyaXZlKS0+T1VUQih4ZmVybGVuID4+IDggICwgSURFX0JDT1VOVEhfUkVHKTsKCWlmIChJREVfQ09OVFJPTF9SRUcpCgkJSFdJRihkcml2ZSktPk9VVEIoZHJpdmUtPmN0bCwgSURFX0NPTlRST0xfUkVHKTsKIAoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyAoZHJpdmUpLT5kcnFfaW50ZXJydXB0KSB7CgkJLyogd2FpdGluZyBmb3IgQ0RCIGludGVycnVwdCwgbm90IERNQSB5ZXQuICovCgkJaWYgKGluZm8tPmRtYSkKCQkJZHJpdmUtPndhaXRpbmdfZm9yX2RtYSA9IDA7CgoJCS8qIHBhY2tldCBjb21tYW5kICovCgkJaWRlX2V4ZWN1dGVfY29tbWFuZChkcml2ZSwgV0lOX1BBQ0tFVENNRCwgaGFuZGxlciwgQVRBUElfV0FJVF9QQywgY2Ryb21fdGltZXJfZXhwaXJ5KTsKCQlyZXR1cm4gaWRlX3N0YXJ0ZWQ7Cgl9IGVsc2UgewoJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJCS8qIHBhY2tldCBjb21tYW5kICovCgkJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CgkJaHdpZi0+T1VUQlNZTkMoZHJpdmUsIFdJTl9QQUNLRVRDTUQsIElERV9DT01NQU5EX1JFRyk7CgkJbmRlbGF5KDQwMCk7CgkJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmaWRlX2xvY2ssIGZsYWdzKTsKCgkJcmV0dXJuICgqaGFuZGxlcikgKGRyaXZlKTsKCX0KfQoKLyogU2VuZCBhIHBhY2tldCBjb21tYW5kIHRvIERSSVZFIGRlc2NyaWJlZCBieSBDTURfQlVGIGFuZCBDTURfTEVOLgogICBUaGUgZGV2aWNlIHJlZ2lzdGVycyBtdXN0IGhhdmUgYWxyZWFkeSBiZWVuIHByZXBhcmVkCiAgIGJ5IGNkcm9tX3N0YXJ0X3BhY2tldF9jb21tYW5kLgogICBIQU5ETEVSIGlzIHRoZSBpbnRlcnJ1cHQgaGFuZGxlciB0byBjYWxsIHdoZW4gdGhlIGNvbW1hbmQgY29tcGxldGVzCiAgIG9yIHRoZXJlJ3MgZGF0YSByZWFkeS4gKi8KLyoKICogY2hhbmdlZCA1IHBhcmFtZXRlcnMgdG8gMyBmb3IgZHZkLXJhbQogKiBzdHJ1Y3QgcGFja2V0X2NvbW1hbmQgKnBjOyBub3cgcGFja2V0X2NvbW1hbmRfdCAqcGM7CiAqLwojZGVmaW5lIEFUQVBJX01JTl9DREJfQllURVMgMTIKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV90cmFuc2Zlcl9wYWNrZXRfY29tbWFuZCAoaWRlX2RyaXZlX3QgKmRyaXZlLAoJCQkJCSAgc3RydWN0IHJlcXVlc3QgKnJxLAoJCQkJCSAgaWRlX2hhbmRsZXJfdCAqaGFuZGxlcikKewoJaWRlX2h3aWZfdCAqaHdpZiA9IGRyaXZlLT5od2lmOwoJaW50IGNtZF9sZW47CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCWlkZV9zdGFydHN0b3BfdCBzdGFydHN0b3A7CgoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmRycV9pbnRlcnJ1cHQpIHsKCQkvKiBIZXJlIHdlIHNob3VsZCBoYXZlIGJlZW4gY2FsbGVkIGFmdGVyIHJlY2VpdmluZyBhbiBpbnRlcnJ1cHQKCQkgICBmcm9tIHRoZSBkZXZpY2UuICBEUlEgc2hvdWxkIGhvdyBiZSBzZXQuICovCgoJCS8qIENoZWNrIGZvciBlcnJvcnMuICovCgkJaWYgKGNkcm9tX2RlY29kZV9zdGF0dXMoZHJpdmUsIERSUV9TVEFULCBOVUxMKSkKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCQkvKiBPaywgbmV4dCBpbnRlcnJ1cHQgd2lsbCBiZSBETUEgaW50ZXJydXB0LiAqLwoJCWlmIChpbmZvLT5kbWEpCgkJCWRyaXZlLT53YWl0aW5nX2Zvcl9kbWEgPSAxOwoJfSBlbHNlIHsKCQkvKiBPdGhlcndpc2UsIHdlIG11c3Qgd2FpdCBmb3IgRFJRIHRvIGdldCBzZXQuICovCgkJaWYgKGlkZV93YWl0X3N0YXQoJnN0YXJ0c3RvcCwgZHJpdmUsIERSUV9TVEFULAoJCQkJQlVTWV9TVEFULCBXQUlUX1JFQURZKSkKCQkJcmV0dXJuIHN0YXJ0c3RvcDsKCX0KCgkvKiBBcm0gdGhlIGludGVycnVwdCBoYW5kbGVyLiAqLwoJaWRlX3NldF9oYW5kbGVyKGRyaXZlLCBoYW5kbGVyLCBycS0+dGltZW91dCwgY2Ryb21fdGltZXJfZXhwaXJ5KTsKCgkvKiBBVEFQSSBjb21tYW5kcyBnZXQgcGFkZGVkIG91dCB0byAxMiBieXRlcyBtaW5pbXVtICovCgljbWRfbGVuID0gQ09NTUFORF9TSVpFKHJxLT5jbWRbMF0pOwoJaWYgKGNtZF9sZW4gPCBBVEFQSV9NSU5fQ0RCX0JZVEVTKQoJCWNtZF9sZW4gPSBBVEFQSV9NSU5fQ0RCX0JZVEVTOwoKCS8qIFNlbmQgdGhlIGNvbW1hbmQgdG8gdGhlIGRldmljZS4gKi8KCUhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXMoZHJpdmUsIHJxLT5jbWQsIGNtZF9sZW4pOwoKCS8qIFN0YXJ0IHRoZSBETUEgaWYgbmVlZCBiZSAqLwoJaWYgKGluZm8tPmRtYSkKCQlod2lmLT5kbWFfc3RhcnQoZHJpdmUpOwoKCXJldHVybiBpZGVfc3RhcnRlZDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQmxvY2sgcmVhZCBmdW5jdGlvbnMuCiAqLwoKLyoKICogQnVmZmVyIHVwIHRvIFNFQ1RPUlNfVE9fVFJBTlNGRVIgc2VjdG9ycyBmcm9tIHRoZSBkcml2ZSBpbiBvdXIgc2VjdG9yCiAqIGJ1ZmZlci4gIE9uY2UgdGhlIGZpcnN0IHNlY3RvciBpcyBhZGRlZCwgYW55IHN1YnNlcXVlbnQgc2VjdG9ycyBhcmUKICogYXNzdW1lZCB0byBiZSBjb250aW51b3VzICh1bnRpbCB0aGUgYnVmZmVyIGlzIGNsZWFyZWQpLiAgRm9yIHRoZSBmaXJzdAogKiBzZWN0b3IgYWRkZWQsIFNFQ1RPUiBpcyBpdHMgc2VjdG9yIG51bWJlci4gIChTRUNUT1IgaXMgdGhlbiBpZ25vcmVkIHVudGlsCiAqIHRoZSBidWZmZXIgaXMgY2xlYXJlZC4pCiAqLwpzdGF0aWMgdm9pZCBjZHJvbV9idWZmZXJfc2VjdG9ycyAoaWRlX2RyaXZlX3QgKmRyaXZlLCB1bnNpZ25lZCBsb25nIHNlY3RvciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBzZWN0b3JzX3RvX3RyYW5zZmVyKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCgkvKiBOdW1iZXIgb2Ygc2VjdG9ycyB0byByZWFkIGludG8gdGhlIGJ1ZmZlci4gKi8KCWludCBzZWN0b3JzX3RvX2J1ZmZlciA9IG1pbl90KGludCwgc2VjdG9yc190b190cmFuc2ZlciwKCQkJCSAgICAgKFNFQ1RPUl9CVUZGRVJfU0laRSA+PiBTRUNUT1JfQklUUykgLQoJCQkJICAgICAgIGluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkKTsKCgljaGFyICpkZXN0OwoKCS8qIElmIHdlIGNvdWxkbid0IGdldCBhIGJ1ZmZlciwgZG9uJ3QgdHJ5IHRvIGJ1ZmZlciBhbnl0aGluZy4uLiAqLwoJaWYgKGluZm8tPmJ1ZmZlciA9PSBOVUxMKQoJCXNlY3RvcnNfdG9fYnVmZmVyID0gMDsKCgkvKiBJZiB0aGlzIGlzIHRoZSBmaXJzdCBzZWN0b3IgaW4gdGhlIGJ1ZmZlciwgcmVtZW1iZXIgaXRzIG51bWJlci4gKi8KCWlmIChpbmZvLT5uc2VjdG9yc19idWZmZXJlZCA9PSAwKQoJCWluZm8tPnNlY3Rvcl9idWZmZXJlZCA9IHNlY3RvcjsKCgkvKiBSZWFkIHRoZSBkYXRhIGludG8gdGhlIGJ1ZmZlci4gKi8KCWRlc3QgPSBpbmZvLT5idWZmZXIgKyBpbmZvLT5uc2VjdG9yc19idWZmZXJlZCAqIFNFQ1RPUl9TSVpFOwoJd2hpbGUgKHNlY3RvcnNfdG9fYnVmZmVyID4gMCkgewoJCUhXSUYoZHJpdmUpLT5hdGFwaV9pbnB1dF9ieXRlcyhkcml2ZSwgZGVzdCwgU0VDVE9SX1NJWkUpOwoJCS0tc2VjdG9yc190b19idWZmZXI7CgkJLS1zZWN0b3JzX3RvX3RyYW5zZmVyOwoJCSsraW5mby0+bnNlY3RvcnNfYnVmZmVyZWQ7CgkJZGVzdCArPSBTRUNUT1JfU0laRTsKCX0KCgkvKiBUaHJvdyBhd2F5IGFueSByZW1haW5pbmcgZGF0YS4gKi8KCXdoaWxlIChzZWN0b3JzX3RvX3RyYW5zZmVyID4gMCkgewoJCXN0YXRpYyBjaGFyIGR1bVtTRUNUT1JfU0laRV07CgkJSFdJRihkcml2ZSktPmF0YXBpX2lucHV0X2J5dGVzKGRyaXZlLCBkdW0sIHNpemVvZiAoZHVtKSk7CgkJLS1zZWN0b3JzX3RvX3RyYW5zZmVyOwoJfQp9CgovKgogKiBDaGVjayB0aGUgY29udGVudHMgb2YgdGhlIGludGVycnVwdCByZWFzb24gcmVnaXN0ZXIgZnJvbSB0aGUgY2Ryb20KICogYW5kIGF0dGVtcHQgdG8gcmVjb3ZlciBpZiB0aGVyZSBhcmUgcHJvYmxlbXMuICBSZXR1cm5zICAwIGlmIGV2ZXJ5dGhpbmcncwogKiBvazsgbm9uemVybyBpZiB0aGUgcmVxdWVzdCBoYXMgYmVlbiB0ZXJtaW5hdGVkLgogKi8Kc3RhdGljCmludCBjZHJvbV9yZWFkX2NoZWNrX2lyZWFzb24gKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IGxlbiwgaW50IGlyZWFzb24pCnsKCWlmIChpcmVhc29uID09IDIpCgkJcmV0dXJuIDA7CgllbHNlIGlmIChpcmVhc29uID09IDApIHsKCQkvKiBXaG9vcHMuLi4gVGhlIGRyaXZlIGlzIGV4cGVjdGluZyB0byByZWNlaXZlIGRhdGEgZnJvbSB1cyEgKi8KCQlwcmludGsoS0VSTl9FUlIgIiVzOiByZWFkX2ludHI6IERyaXZlIHdhbnRzIHRvIHRyYW5zZmVyIGRhdGEgdGhlICIKCQkJCQkJIndyb25nIHdheSFcbiIsIGRyaXZlLT5uYW1lKTsKCgkJLyogVGhyb3cgc29tZSBkYXRhIGF0IHRoZSBkcml2ZSBzbyBpdCBkb2Vzbid0IGhhbmcKCQkgICBhbmQgcXVpdCB0aGlzIHJlcXVlc3QuICovCgkJd2hpbGUgKGxlbiA+IDApIHsKCQkJaW50IGR1bSA9IDA7CgkJCUhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXMoZHJpdmUsICZkdW0sIHNpemVvZiAoZHVtKSk7CgkJCWxlbiAtPSBzaXplb2YgKGR1bSk7CgkJfQoJfSBlbHNlICBpZiAoaXJlYXNvbiA9PSAxKSB7CgkJLyogU29tZSBkcml2ZXMgKEFTVVMpIHNlZW0gdG8gdGVsbCB1cyB0aGF0IHN0YXR1cwoJCSAqIGluZm8gaXMgYXZhaWxhYmxlLiBqdXN0IGdldCBpdCBhbmQgaWdub3JlLgoJCSAqLwoJCSh2b2lkKSBIV0lGKGRyaXZlKS0+SU5CKElERV9TVEFUVVNfUkVHKTsKCQlyZXR1cm4gMDsKCX0gZWxzZSB7CgkJLyogRHJpdmUgd2FudHMgYSBjb21tYW5kIHBhY2tldCwgb3IgaW52YWxpZCBpcmVhc29uLi4uICovCgkJcHJpbnRrKEtFUk5fRVJSICIlczogcmVhZF9pbnRyOiBiYWQgaW50ZXJydXB0IHJlYXNvbiAleFxuIiwgZHJpdmUtPm5hbWUsCgkJCQkJCQkJaXJlYXNvbik7Cgl9CgoJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJcmV0dXJuIC0xOwp9CgovKgogKiBJbnRlcnJ1cHQgcm91dGluZS4gIENhbGxlZCB3aGVuIGEgcmVhZCByZXF1ZXN0IGhhcyBjb21wbGV0ZWQuCiAqLwpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3JlYWRfaW50ciAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglpbnQgc3RhdDsKCWludCBpcmVhc29uLCBsZW4sIHNlY3RvcnNfdG9fdHJhbnNmZXIsIG5za2lwOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7Cgl1OCBsb3djeWwgPSAwLCBoaWdoY3lsID0gMDsKCWludCBkbWEgPSBpbmZvLT5kbWEsIGRtYV9lcnJvciA9IDA7CgoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoKCS8qCgkgKiBoYW5kbGUgZG1hIGNhc2UKCSAqLwoJaWYgKGRtYSkgewoJCWluZm8tPmRtYSA9IDA7CgkJaWYgKChkbWFfZXJyb3IgPSBIV0lGKGRyaXZlKS0+aWRlX2RtYV9lbmQoZHJpdmUpKSkKCQkJaWRlX2RtYV9vZmYoZHJpdmUpOwoJfQoKCWlmIChjZHJvbV9kZWNvZGVfc3RhdHVzKGRyaXZlLCAwLCAmc3RhdCkpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCWlmIChkbWEpIHsKCQlpZiAoIWRtYV9lcnJvcikgewoJCQlpZGVfZW5kX3JlcXVlc3QoZHJpdmUsIDEsIHJxLT5ucl9zZWN0b3JzKTsKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJCX0gZWxzZQoJCQlyZXR1cm4gaWRlX2Vycm9yKGRyaXZlLCAiZG1hIGVycm9yIiwgc3RhdCk7Cgl9CgoJLyogUmVhZCB0aGUgaW50ZXJydXB0IHJlYXNvbiBhbmQgdGhlIHRyYW5zZmVyIGxlbmd0aC4gKi8KCWlyZWFzb24gPSBIV0lGKGRyaXZlKS0+SU5CKElERV9JUkVBU09OX1JFRykgJiAweDM7Cglsb3djeWwgID0gSFdJRihkcml2ZSktPklOQihJREVfQkNPVU5UTF9SRUcpOwoJaGlnaGN5bCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVEhfUkVHKTsKCglsZW4gPSBsb3djeWwgKyAoMjU2ICogaGlnaGN5bCk7CgoJLyogSWYgRFJRIGlzIGNsZWFyLCB0aGUgY29tbWFuZCBoYXMgY29tcGxldGVkLiAqLwoJaWYgKChzdGF0ICYgRFJRX1NUQVQpID09IDApIHsKCQkvKiBJZiB3ZSdyZSBub3QgZG9uZSBmaWxsaW5nIHRoZSBjdXJyZW50IGJ1ZmZlciwgY29tcGxhaW4uCgkJICAgT3RoZXJ3aXNlLCBjb21wbGV0ZSB0aGUgY29tbWFuZCBub3JtYWxseS4gKi8KCQlpZiAocnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA+IDApIHsKCQkJcHJpbnRrIChLRVJOX0VSUiAiJXM6IGNkcm9tX3JlYWRfaW50cjogZGF0YSB1bmRlcnJ1biAoJWQgYmxvY2tzKVxuIiwKCQkJCWRyaXZlLT5uYW1lLCBycS0+Y3VycmVudF9ucl9zZWN0b3JzKTsKCQkJcnEtPmNtZF9mbGFncyB8PSBSRVFfRkFJTEVEOwoJCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7CgkJfSBlbHNlCgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAxKTsKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7Cgl9CgoJLyogQ2hlY2sgdGhhdCB0aGUgZHJpdmUgaXMgZXhwZWN0aW5nIHRvIGRvIHRoZSBzYW1lIHRoaW5nIHdlIGFyZS4gKi8KCWlmIChjZHJvbV9yZWFkX2NoZWNrX2lyZWFzb24gKGRyaXZlLCBsZW4sIGlyZWFzb24pKQoJCXJldHVybiBpZGVfc3RvcHBlZDsKCgkvKiBBc3N1bWUgdGhhdCB0aGUgZHJpdmUgd2lsbCBhbHdheXMgcHJvdmlkZSBkYXRhIGluIG11bHRpcGxlcwoJICAgb2YgYXQgbGVhc3QgU0VDVE9SX1NJWkUsIGFzIGl0IGdldHMgaGFpcnkgdG8ga2VlcCB0cmFjawoJICAgb2YgdGhlIHRyYW5zZmVycyBvdGhlcndpc2UuICovCglpZiAoKGxlbiAlIFNFQ1RPUl9TSVpFKSAhPSAwKSB7CgkJcHJpbnRrIChLRVJOX0VSUiAiJXM6IGNkcm9tX3JlYWRfaW50cjogQmFkIHRyYW5zZmVyIHNpemUgJWRcbiIsCgkJCWRyaXZlLT5uYW1lLCBsZW4pOwoJCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5saW1pdF9uZnJhbWVzKQoJCQlwcmludGsgKEtFUk5fRVJSICIgIFRoaXMgZHJpdmUgaXMgbm90IHN1cHBvcnRlZCBieSB0aGlzIHZlcnNpb24gb2YgdGhlIGRyaXZlclxuIik7CgkJZWxzZSB7CgkJCXByaW50ayAoS0VSTl9FUlIgIiAgVHJ5aW5nIHRvIGxpbWl0IHRyYW5zZmVyIHNpemVzXG4iKTsKCQkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bGltaXRfbmZyYW1lcyA9IDE7CgkJfQoJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7Cgl9CgoJLyogVGhlIG51bWJlciBvZiBzZWN0b3JzIHdlIG5lZWQgdG8gcmVhZCBmcm9tIHRoZSBkcml2ZS4gKi8KCXNlY3RvcnNfdG9fdHJhbnNmZXIgPSBsZW4gLyBTRUNUT1JfU0laRTsKCgkvKiBGaXJzdCwgZmlndXJlIG91dCBpZiB3ZSBuZWVkIHRvIGJpdC1idWNrZXQKCSAgIGFueSBvZiB0aGUgbGVhZGluZyBzZWN0b3JzLiAqLwoJbnNraXAgPSBtaW5fdChpbnQsIHJxLT5jdXJyZW50X25yX3NlY3RvcnMgLSBiaW9fY3VyX3NlY3RvcnMocnEtPmJpbyksIHNlY3RvcnNfdG9fdHJhbnNmZXIpOwoKCXdoaWxlIChuc2tpcCA+IDApIHsKCQkvKiBXZSBuZWVkIHRvIHRocm93IGF3YXkgYSBzZWN0b3IuICovCgkJc3RhdGljIGNoYXIgZHVtW1NFQ1RPUl9TSVpFXTsKCQlIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXMoZHJpdmUsIGR1bSwgc2l6ZW9mIChkdW0pKTsKCgkJLS1ycS0+Y3VycmVudF9ucl9zZWN0b3JzOwoJCS0tbnNraXA7CgkJLS1zZWN0b3JzX3RvX3RyYW5zZmVyOwoJfQoKCS8qIE5vdyBsb29wIHdoaWxlIHdlIHN0aWxsIGhhdmUgZGF0YSB0byByZWFkIGZyb20gdGhlIGRyaXZlLiAqLwoJd2hpbGUgKHNlY3RvcnNfdG9fdHJhbnNmZXIgPiAwKSB7CgkJaW50IHRoaXNfdHJhbnNmZXI7CgoJCS8qIElmIHdlJ3ZlIGZpbGxlZCB0aGUgcHJlc2VudCBidWZmZXIgYnV0IHRoZXJlJ3MgYW5vdGhlcgoJCSAgIGNoYWluZWQgYnVmZmVyIGFmdGVyIGl0LCBtb3ZlIG9uLiAqLwoJCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzID09IDAgJiYgcnEtPm5yX3NlY3RvcnMpCgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAxKTsKCgkJLyogSWYgdGhlIGJ1ZmZlcnMgYXJlIGZ1bGwsIGNhY2hlIHRoZSByZXN0IG9mIHRoZSBkYXRhIGluIG91cgoJCSAgIGludGVybmFsIGJ1ZmZlci4gKi8KCQlpZiAocnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA9PSAwKSB7CgkJCWNkcm9tX2J1ZmZlcl9zZWN0b3JzKGRyaXZlLCBycS0+c2VjdG9yLCBzZWN0b3JzX3RvX3RyYW5zZmVyKTsKCQkJc2VjdG9yc190b190cmFuc2ZlciA9IDA7CgkJfSBlbHNlIHsKCQkJLyogVHJhbnNmZXIgZGF0YSB0byB0aGUgYnVmZmVycy4KCQkJICAgRmlndXJlIG91dCBob3cgbWFueSBzZWN0b3JzIHdlIGNhbiB0cmFuc2ZlcgoJCQkgICB0byB0aGUgY3VycmVudCBidWZmZXIuICovCgkJCXRoaXNfdHJhbnNmZXIgPSBtaW5fdChpbnQsIHNlY3RvcnNfdG9fdHJhbnNmZXIsCgkJCQkJICAgICBycS0+Y3VycmVudF9ucl9zZWN0b3JzKTsKCgkJCS8qIFJlYWQgdGhpc190cmFuc2ZlciBzZWN0b3JzCgkJCSAgIGludG8gdGhlIGN1cnJlbnQgYnVmZmVyLiAqLwoJCQl3aGlsZSAodGhpc190cmFuc2ZlciA+IDApIHsKCQkJCUhXSUYoZHJpdmUpLT5hdGFwaV9pbnB1dF9ieXRlcyhkcml2ZSwgcnEtPmJ1ZmZlciwgU0VDVE9SX1NJWkUpOwoJCQkJcnEtPmJ1ZmZlciArPSBTRUNUT1JfU0laRTsKCQkJCS0tcnEtPm5yX3NlY3RvcnM7CgkJCQktLXJxLT5jdXJyZW50X25yX3NlY3RvcnM7CgkJCQkrK3JxLT5zZWN0b3I7CgkJCQktLXRoaXNfdHJhbnNmZXI7CgkJCQktLXNlY3RvcnNfdG9fdHJhbnNmZXI7CgkJCX0KCQl9Cgl9CgoJLyogRG9uZSBtb3ZpbmcgZGF0YSEgIFdhaXQgZm9yIGFub3RoZXIgaW50ZXJydXB0LiAqLwoJaWRlX3NldF9oYW5kbGVyKGRyaXZlLCAmY2Ryb21fcmVhZF9pbnRyLCBBVEFQSV9XQUlUX1BDLCBOVUxMKTsKCXJldHVybiBpZGVfc3RhcnRlZDsKfQoKLyoKICogVHJ5IHRvIHNhdGlzZnkgc29tZSBvZiB0aGUgY3VycmVudCByZWFkIHJlcXVlc3QgZnJvbSBvdXIgY2FjaGVkIGRhdGEuCiAqIFJldHVybnMgbm9uemVybyBpZiB0aGUgcmVxdWVzdCBoYXMgYmVlbiBjb21wbGV0ZWQsIHplcm8gb3RoZXJ3aXNlLgogKi8Kc3RhdGljIGludCBjZHJvbV9yZWFkX2Zyb21fYnVmZmVyIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJdW5zaWduZWQgc2hvcnQgc2VjdG9yc19wZXJfZnJhbWU7CgoJc2VjdG9yc19wZXJfZnJhbWUgPSBxdWV1ZV9oYXJkc2VjdF9zaXplKGRyaXZlLT5xdWV1ZSkgPj4gU0VDVE9SX0JJVFM7CgoJLyogQ2FuJ3QgZG8gYW55dGhpbmcgaWYgdGhlcmUncyBubyBidWZmZXIuICovCglpZiAoaW5mby0+YnVmZmVyID09IE5VTEwpIHJldHVybiAwOwoKCS8qIExvb3Agd2hpbGUgdGhpcyByZXF1ZXN0IG5lZWRzIGRhdGEgYW5kIHRoZSBuZXh0IGJsb2NrIGlzIHByZXNlbnQKCSAgIGluIG91ciBjYWNoZS4gKi8KCXdoaWxlIChycS0+bnJfc2VjdG9ycyA+IDAgJiYKCSAgICAgICBycS0+c2VjdG9yID49IGluZm8tPnNlY3Rvcl9idWZmZXJlZCAmJgoJICAgICAgIHJxLT5zZWN0b3IgPCBpbmZvLT5zZWN0b3JfYnVmZmVyZWQgKyBpbmZvLT5uc2VjdG9yc19idWZmZXJlZCkgewoJCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzID09IDApCgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAxKTsKCgkJbWVtY3B5IChycS0+YnVmZmVyLAoJCQlpbmZvLT5idWZmZXIgKwoJCQkocnEtPnNlY3RvciAtIGluZm8tPnNlY3Rvcl9idWZmZXJlZCkgKiBTRUNUT1JfU0laRSwKCQkJU0VDVE9SX1NJWkUpOwoJCXJxLT5idWZmZXIgKz0gU0VDVE9SX1NJWkU7CgkJLS1ycS0+Y3VycmVudF9ucl9zZWN0b3JzOwoJCS0tcnEtPm5yX3NlY3RvcnM7CgkJKytycS0+c2VjdG9yOwoJfQoKCS8qIElmIHdlJ3ZlIHNhdGlzZmllZCB0aGUgY3VycmVudCByZXF1ZXN0LAoJICAgdGVybWluYXRlIGl0IHN1Y2Nlc3NmdWxseS4gKi8KCWlmIChycS0+bnJfc2VjdG9ycyA9PSAwKSB7CgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoJCXJldHVybiAtMTsKCX0KCgkvKiBNb3ZlIG9uIHRvIHRoZSBuZXh0IGJ1ZmZlciBpZiBuZWVkZWQuICovCglpZiAocnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA9PSAwKQoJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAxKTsKCgkvKiBJZiB0aGlzIGNvbmRpdGlvbiBkb2VzIG5vdCBob2xkLCB0aGVuIHRoZSBrbHVnZSBpIHVzZSB0bwoJICAgcmVwcmVzZW50IHRoZSBudW1iZXIgb2Ygc2VjdG9ycyB0byBza2lwIGF0IHRoZSBzdGFydCBvZiBhIHRyYW5zZmVyCgkgICB3aWxsIGZhaWwuICBJIHRoaW5rIHRoYXQgdGhpcyB3aWxsIG5ldmVyIGhhcHBlbiwgYnV0IGxldCdzIGJlCgkgICBwYXJhbm9pZCBhbmQgY2hlY2suICovCglpZiAocnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA8IGJpb19jdXJfc2VjdG9ycyhycS0+YmlvKSAmJgoJICAgIChycS0+c2VjdG9yICYgKHNlY3RvcnNfcGVyX2ZyYW1lIC0gMSkpKSB7CgkJcHJpbnRrKEtFUk5fRVJSICIlczogY2Ryb21fcmVhZF9mcm9tX2J1ZmZlcjogYnVmZmVyIGJvdGNoICglbGQpXG4iLAoJCQlkcml2ZS0+bmFtZSwgKGxvbmcpcnEtPnNlY3Rvcik7CgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJCXJldHVybiAtMTsKCX0KCglyZXR1cm4gMDsKfQoKLyoKICogUm91dGluZSB0byBzZW5kIGEgcmVhZCBwYWNrZXQgY29tbWFuZCB0byB0aGUgZHJpdmUuCiAqIFRoaXMgaXMgdXN1YWxseSBjYWxsZWQgZGlyZWN0bHkgZnJvbSBjZHJvbV9zdGFydF9yZWFkLgogKiBIb3dldmVyLCBmb3IgZHJxX2ludGVycnVwdCBkZXZpY2VzLCBpdCBpcyBjYWxsZWQgZnJvbSBhbiBpbnRlcnJ1cHQKICogd2hlbiB0aGUgZHJpdmUgaXMgcmVhZHkgdG8gYWNjZXB0IHRoZSBjb21tYW5kLgogKi8Kc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9zdGFydF9yZWFkX2NvbnRpbnVhdGlvbiAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7Cgl1bnNpZ25lZCBzaG9ydCBzZWN0b3JzX3Blcl9mcmFtZTsKCWludCBuc2tpcDsKCglzZWN0b3JzX3Blcl9mcmFtZSA9IHF1ZXVlX2hhcmRzZWN0X3NpemUoZHJpdmUtPnF1ZXVlKSA+PiBTRUNUT1JfQklUUzsKCgkvKiBJZiB0aGUgcmVxdWVzdGVkIHNlY3RvciBkb2Vzbid0IHN0YXJ0IG9uIGEgY2Ryb20gYmxvY2sgYm91bmRhcnksCgkgICB3ZSBtdXN0IGFkanVzdCB0aGUgc3RhcnQgb2YgdGhlIHRyYW5zZmVyIHNvIHRoYXQgaXQgZG9lcywKCSAgIGFuZCByZW1lbWJlciB0byBza2lwIHRoZSBmaXJzdCBmZXcgc2VjdG9ycy4KCSAgIElmIHRoZSBDVVJSRU5UX05SX1NFQ1RPUlMgZmllbGQgaXMgbGFyZ2VyIHRoYW4gdGhlIHNpemUKCSAgIG9mIHRoZSBidWZmZXIsIGl0IHdpbGwgbWVhbiB0aGF0IHdlJ3JlIHRvIHNraXAgYSBudW1iZXIKCSAgIG9mIHNlY3RvcnMgZXF1YWwgdG8gdGhlIGFtb3VudCBieSB3aGljaCBDVVJSRU5UX05SX1NFQ1RPUlMKCSAgIGlzIGxhcmdlciB0aGFuIHRoZSBidWZmZXIgc2l6ZS4gKi8KCW5za2lwID0gcnEtPnNlY3RvciAmIChzZWN0b3JzX3Blcl9mcmFtZSAtIDEpOwoJaWYgKG5za2lwID4gMCkgewoJCS8qIFNhbml0eSBjaGVjay4uLiAqLwoJCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzICE9IGJpb19jdXJfc2VjdG9ycyhycS0+YmlvKSAmJgoJCQkocnEtPnNlY3RvciAmIChzZWN0b3JzX3Blcl9mcmFtZSAtIDEpKSkgewoJCQlwcmludGsoS0VSTl9FUlIgIiVzOiBjZHJvbV9zdGFydF9yZWFkX2NvbnRpbnVhdGlvbjogYnVmZmVyIGJvdGNoICgldSlcbiIsCgkJCQlkcml2ZS0+bmFtZSwgcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyk7CgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJCX0KCQlycS0+Y3VycmVudF9ucl9zZWN0b3JzICs9IG5za2lwOwoJfQoKCS8qIFNldCB1cCB0aGUgY29tbWFuZCAqLwoJcnEtPnRpbWVvdXQgPSBBVEFQSV9XQUlUX1BDOwoKCS8qIFNlbmQgdGhlIGNvbW1hbmQgdG8gdGhlIGRyaXZlIGFuZCByZXR1cm4uICovCglyZXR1cm4gY2Ryb21fdHJhbnNmZXJfcGFja2V0X2NvbW1hbmQoZHJpdmUsIHJxLCAmY2Ryb21fcmVhZF9pbnRyKTsKfQoKCiNkZWZpbmUgSURFQ0RfU0VFS19USFJFU0hPTEQJKDEwMDApCQkJLyogMTAwMCBibG9ja3MgKi8KI2RlZmluZSBJREVDRF9TRUVLX1RJTUVSCSg1ICogV0FJVF9NSU5fU0xFRVApCS8qIDEwMCBtcyAqLwojZGVmaW5lIElERUNEX1NFRUtfVElNRU9VVAkoMiAqIFdBSVRfQ01EKQkJLyogMjAgc2VjICovCgpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3NlZWtfaW50ciAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCWludCBzdGF0OwoJc3RhdGljIGludCByZXRyeSA9IDEwOwoKCWlmIChjZHJvbV9kZWNvZGVfc3RhdHVzKGRyaXZlLCAwLCAmc3RhdCkpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+c2Vla2luZyA9IDE7CgoJaWYgKHJldHJ5ICYmIHRpbWVfYWZ0ZXIoamlmZmllcywgaW5mby0+c3RhcnRfc2VlayArIElERUNEX1NFRUtfVElNRVIpKSB7CgkJaWYgKC0tcmV0cnkgPT0gMCkgewoJCQkvKgoJCQkgKiB0aGlzIGNvbmRpdGlvbiBpcyBmYXIgdG9vIGNvbW1vbiwgdG8gYm90aGVyCgkJCSAqIHVzZXJzIGFib3V0IGl0CgkJCSAqLwoJCQkvKiBwcmludGsoIiVzOiBkaXNhYmxlZCBEU0Mgc2VlayBvdmVybGFwXG4iLCBkcml2ZS0+bmFtZSk7Ki8gCgkJCWRyaXZlLT5kc2Nfb3ZlcmxhcCA9IDA7CgkJfQoJfQoJcmV0dXJuIGlkZV9zdG9wcGVkOwp9CgpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3N0YXJ0X3NlZWtfY29udGludWF0aW9uIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCXNlY3Rvcl90IGZyYW1lID0gcnEtPnNlY3RvcjsKCglzZWN0b3JfZGl2KGZyYW1lLCBxdWV1ZV9oYXJkc2VjdF9zaXplKGRyaXZlLT5xdWV1ZSkgPj4gU0VDVE9SX0JJVFMpOwoKCW1lbXNldChycS0+Y21kLCAwLCBzaXplb2YocnEtPmNtZCkpOwoJcnEtPmNtZFswXSA9IEdQQ01EX1NFRUs7CglwdXRfdW5hbGlnbmVkKGNwdV90b19iZTMyKGZyYW1lKSwgKHVuc2lnbmVkIGludCAqKSAmcnEtPmNtZFsyXSk7CgoJcnEtPnRpbWVvdXQgPSBBVEFQSV9XQUlUX1BDOwoJcmV0dXJuIGNkcm9tX3RyYW5zZmVyX3BhY2tldF9jb21tYW5kKGRyaXZlLCBycSwgJmNkcm9tX3NlZWtfaW50cik7Cn0KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc3RhcnRfc2VlayAoaWRlX2RyaXZlX3QgKmRyaXZlLCB1bnNpZ25lZCBpbnQgYmxvY2spCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoKCWluZm8tPmRtYSA9IDA7CglpbmZvLT5zdGFydF9zZWVrID0gamlmZmllczsKCXJldHVybiBjZHJvbV9zdGFydF9wYWNrZXRfY29tbWFuZChkcml2ZSwgMCwgY2Ryb21fc3RhcnRfc2Vla19jb250aW51YXRpb24pOwp9CgovKiBGaXggdXAgYSBwb3NzaWJseSBwYXJ0aWFsbHktcHJvY2Vzc2VkIHJlcXVlc3Qgc28gdGhhdCB3ZSBjYW4KICAgc3RhcnQgaXQgb3ZlciBlbnRpcmVseSwgb3IgZXZlbiBwdXQgaXQgYmFjayBvbiB0aGUgcmVxdWVzdCBxdWV1ZS4gKi8Kc3RhdGljIHZvaWQgcmVzdG9yZV9yZXF1ZXN0IChzdHJ1Y3QgcmVxdWVzdCAqcnEpCnsKCWlmIChycS0+YnVmZmVyICE9IGJpb19kYXRhKHJxLT5iaW8pKSB7CgkJc2VjdG9yX3QgbiA9IChycS0+YnVmZmVyIC0gKGNoYXIgKikgYmlvX2RhdGEocnEtPmJpbykpIC8gU0VDVE9SX1NJWkU7CgoJCXJxLT5idWZmZXIgPSBiaW9fZGF0YShycS0+YmlvKTsKCQlycS0+bnJfc2VjdG9ycyArPSBuOwoJCXJxLT5zZWN0b3IgLT0gbjsKCX0KCXJxLT5oYXJkX2N1cl9zZWN0b3JzID0gcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA9IGJpb19jdXJfc2VjdG9ycyhycS0+YmlvKTsKCXJxLT5oYXJkX25yX3NlY3RvcnMgPSBycS0+bnJfc2VjdG9yczsKCXJxLT5oYXJkX3NlY3RvciA9IHJxLT5zZWN0b3I7CglycS0+cS0+cHJlcF9ycV9mbihycS0+cSwgcnEpOwp9CgovKgogKiBTdGFydCBhIHJlYWQgcmVxdWVzdCBmcm9tIHRoZSBDRC1ST00uCiAqLwpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3N0YXJ0X3JlYWQgKGlkZV9kcml2ZV90ICpkcml2ZSwgdW5zaWduZWQgaW50IGJsb2NrKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCXVuc2lnbmVkIHNob3J0IHNlY3RvcnNfcGVyX2ZyYW1lOwoKCXNlY3RvcnNfcGVyX2ZyYW1lID0gcXVldWVfaGFyZHNlY3Rfc2l6ZShkcml2ZS0+cXVldWUpID4+IFNFQ1RPUl9CSVRTOwoKCS8qIFdlIG1heSBiZSByZXRyeWluZyB0aGlzIHJlcXVlc3QgYWZ0ZXIgYW4gZXJyb3IuICBGaXggdXAKCSAgIGFueSB3ZWlyZG5lc3Mgd2hpY2ggbWlnaHQgYmUgcHJlc2VudCBpbiB0aGUgcmVxdWVzdCBwYWNrZXQuICovCglyZXN0b3JlX3JlcXVlc3QocnEpOwoKCS8qIFNhdGlzZnkgd2hhdGV2ZXIgd2UgY2FuIG9mIHRoaXMgcmVxdWVzdCBmcm9tIG91ciBjYWNoZWQgc2VjdG9yLiAqLwoJaWYgKGNkcm9tX3JlYWRfZnJvbV9idWZmZXIoZHJpdmUpKQoJCXJldHVybiBpZGVfc3RvcHBlZDsKCgkvKiBDbGVhciB0aGUgbG9jYWwgc2VjdG9yIGJ1ZmZlci4gKi8KCWluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkID0gMDsKCgkvKiB1c2UgZG1hLCBpZiBwb3NzaWJsZS4gKi8KCWluZm8tPmRtYSA9IGRyaXZlLT51c2luZ19kbWE7CglpZiAoKHJxLT5zZWN0b3IgJiAoc2VjdG9yc19wZXJfZnJhbWUgLSAxKSkgfHwKCSAgICAocnEtPm5yX3NlY3RvcnMgJiAoc2VjdG9yc19wZXJfZnJhbWUgLSAxKSkpCgkJaW5mby0+ZG1hID0gMDsKCgkvKiBTdGFydCBzZW5kaW5nIHRoZSByZWFkIHJlcXVlc3QgdG8gdGhlIGRyaXZlLiAqLwoJcmV0dXJuIGNkcm9tX3N0YXJ0X3BhY2tldF9jb21tYW5kKGRyaXZlLCAzMjc2OCwgY2Ryb21fc3RhcnRfcmVhZF9jb250aW51YXRpb24pOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBFeGVjdXRlIGFsbCBvdGhlciBwYWNrZXQgY29tbWFuZHMuCiAqLwoKLyogSW50ZXJydXB0IHJvdXRpbmUgZm9yIHBhY2tldCBjb21tYW5kIGNvbXBsZXRpb24uICovCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fcGNfaW50ciAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglpbnQgaXJlYXNvbiwgbGVuLCB0aGlzbGVuOwoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJdTggbG93Y3lsID0gMCwgaGlnaGN5bCA9IDA7CglpbnQgc3RhdDsKCgkvKiBDaGVjayBmb3IgZXJyb3JzLiAqLwoJaWYgKGNkcm9tX2RlY29kZV9zdGF0dXMoZHJpdmUsIDAsICZzdGF0KSkKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CgoJLyogUmVhZCB0aGUgaW50ZXJydXB0IHJlYXNvbiBhbmQgdGhlIHRyYW5zZmVyIGxlbmd0aC4gKi8KCWlyZWFzb24gPSBIV0lGKGRyaXZlKS0+SU5CKElERV9JUkVBU09OX1JFRyk7Cglsb3djeWwgID0gSFdJRihkcml2ZSktPklOQihJREVfQkNPVU5UTF9SRUcpOwoJaGlnaGN5bCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVEhfUkVHKTsKCglsZW4gPSBsb3djeWwgKyAoMjU2ICogaGlnaGN5bCk7CgoJLyogSWYgRFJRIGlzIGNsZWFyLCB0aGUgY29tbWFuZCBoYXMgY29tcGxldGVkLgoJICAgQ29tcGxhaW4gaWYgd2Ugc3RpbGwgaGF2ZSBkYXRhIGxlZnQgdG8gdHJhbnNmZXIuICovCglpZiAoKHN0YXQgJiBEUlFfU1RBVCkgPT0gMCkgewoJCS8qIFNvbWUgb2YgdGhlIHRyYWlsaW5nIHJlcXVlc3Qgc2Vuc2UgZmllbGRzIGFyZSBvcHRpb25hbCwgYW5kCgkJICAgc29tZSBkcml2ZXMgZG9uJ3Qgc2VuZCB0aGVtLiAgU2lnaC4gKi8KCQlpZiAocnEtPmNtZFswXSA9PSBHUENNRF9SRVFVRVNUX1NFTlNFICYmCgkJICAgIHJxLT5kYXRhX2xlbiA+IDAgJiYKCQkgICAgcnEtPmRhdGFfbGVuIDw9IDUpIHsKCQkJd2hpbGUgKHJxLT5kYXRhX2xlbiA+IDApIHsKCQkJCSoodW5zaWduZWQgY2hhciAqKXJxLT5kYXRhKysgPSAwOwoJCQkJLS1ycS0+ZGF0YV9sZW47CgkJCX0KCQl9CgoJCWlmIChycS0+ZGF0YV9sZW4gPT0gMCkKCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoJCWVsc2UgewoJCQkvKiBDb21tZW50IHRoaXMgb3V0LCBiZWNhdXNlIHRoaXMgYWx3YXlzIGhhcHBlbnMgCgkJCSAgIHJpZ2h0IGFmdGVyIGEgcmVzZXQgb2NjdXJzLCBhbmQgaXQgaXMgYW5ub3lpbmcgdG8gCgkJCSAgIGFsd2F5cyBwcmludCBleHBlY3RlZCBzdHVmZi4gICovCgkJCS8qCgkJCXByaW50ayAoIiVzOiBjZHJvbV9wY19pbnRyOiBkYXRhIHVuZGVycnVuICVkXG4iLAoJCQkJZHJpdmUtPm5hbWUsIHBjLT5idWZsZW4pOwoJCQkqLwoJCQlycS0+Y21kX2ZsYWdzIHw9IFJFUV9GQUlMRUQ7CgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQl9CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCS8qIEZpZ3VyZSBvdXQgaG93IG11Y2ggZGF0YSB0byB0cmFuc2Zlci4gKi8KCXRoaXNsZW4gPSBycS0+ZGF0YV9sZW47CglpZiAodGhpc2xlbiA+IGxlbikgdGhpc2xlbiA9IGxlbjsKCgkvKiBUaGUgZHJpdmUgd2FudHMgdG8gYmUgd3JpdHRlbiB0by4gKi8KCWlmICgoaXJlYXNvbiAmIDMpID09IDApIHsKCQlpZiAoIXJxLT5kYXRhKSB7CgkJCWJsa19kdW1wX3JxX2ZsYWdzKHJxLCAiY2Ryb21fcGNfaW50ciwgd3JpdGUiKTsKCQkJZ290byBjb25mdXNlZDsKCQl9CgkJLyogVHJhbnNmZXIgdGhlIGRhdGEuICovCgkJSFdJRihkcml2ZSktPmF0YXBpX291dHB1dF9ieXRlcyhkcml2ZSwgcnEtPmRhdGEsIHRoaXNsZW4pOwoKCQkvKiBJZiB3ZSBoYXZlbid0IG1vdmVkIGVub3VnaCBkYXRhIHRvIHNhdGlzZnkgdGhlIGRyaXZlLAoJCSAgIGFkZCBzb21lIHBhZGRpbmcuICovCgkJd2hpbGUgKGxlbiA+IHRoaXNsZW4pIHsKCQkJaW50IGR1bSA9IDA7CgkJCUhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXMoZHJpdmUsICZkdW0sIHNpemVvZihkdW0pKTsKCQkJbGVuIC09IHNpemVvZihkdW0pOwoJCX0KCgkJLyogS2VlcCBjb3VudCBvZiBob3cgbXVjaCBkYXRhIHdlJ3ZlIG1vdmVkLiAqLwoJCXJxLT5kYXRhICs9IHRoaXNsZW47CgkJcnEtPmRhdGFfbGVuIC09IHRoaXNsZW47Cgl9CgoJLyogU2FtZSBkcmlsbCBmb3IgcmVhZGluZy4gKi8KCWVsc2UgaWYgKChpcmVhc29uICYgMykgPT0gMikgewoJCWlmICghcnEtPmRhdGEpIHsKCQkJYmxrX2R1bXBfcnFfZmxhZ3MocnEsICJjZHJvbV9wY19pbnRyLCB3cml0ZSIpOwoJCQlnb3RvIGNvbmZ1c2VkOwoJCX0KCQkvKiBUcmFuc2ZlciB0aGUgZGF0YS4gKi8KCQlIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXMoZHJpdmUsIHJxLT5kYXRhLCB0aGlzbGVuKTsKCgkJLyogSWYgd2UgaGF2ZW4ndCBtb3ZlZCBlbm91Z2ggZGF0YSB0byBzYXRpc2Z5IHRoZSBkcml2ZSwKCQkgICBhZGQgc29tZSBwYWRkaW5nLiAqLwoJCXdoaWxlIChsZW4gPiB0aGlzbGVuKSB7CgkJCWludCBkdW0gPSAwOwoJCQlIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXMoZHJpdmUsICZkdW0sIHNpemVvZihkdW0pKTsKCQkJbGVuIC09IHNpemVvZihkdW0pOwoJCX0KCgkJLyogS2VlcCBjb3VudCBvZiBob3cgbXVjaCBkYXRhIHdlJ3ZlIG1vdmVkLiAqLwoJCXJxLT5kYXRhICs9IHRoaXNsZW47CgkJcnEtPmRhdGFfbGVuIC09IHRoaXNsZW47CgoJCWlmIChibGtfc2Vuc2VfcmVxdWVzdChycSkpCgkJCXJxLT5zZW5zZV9sZW4gKz0gdGhpc2xlbjsKCX0gZWxzZSB7CmNvbmZ1c2VkOgoJCXByaW50ayAoS0VSTl9FUlIgIiVzOiBjZHJvbV9wY19pbnRyOiBUaGUgZHJpdmUgIgoJCQkiYXBwZWFycyBjb25mdXNlZCAoaXJlYXNvbiA9IDB4JTAyeCkuICIKCQkJIlRyeWluZyB0byByZWNvdmVyIGJ5IGVuZGluZyByZXF1ZXN0LlxuIiwKCQkJZHJpdmUtPm5hbWUsIGlyZWFzb24pOwoJCXJxLT5jbWRfZmxhZ3MgfD0gUkVRX0ZBSUxFRDsKCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCS8qIE5vdyB3ZSB3YWl0IGZvciBhbm90aGVyIGludGVycnVwdC4gKi8KCWlkZV9zZXRfaGFuZGxlcihkcml2ZSwgJmNkcm9tX3BjX2ludHIsIEFUQVBJX1dBSVRfUEMsIGNkcm9tX3RpbWVyX2V4cGlyeSk7CglyZXR1cm4gaWRlX3N0YXJ0ZWQ7Cn0KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fZG9fcGNfY29udGludWF0aW9uIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCglpZiAoIXJxLT50aW1lb3V0KQoJCXJxLT50aW1lb3V0ID0gQVRBUElfV0FJVF9QQzsKCgkvKiBTZW5kIHRoZSBjb21tYW5kIHRvIHRoZSBkcml2ZSBhbmQgcmV0dXJuLiAqLwoJcmV0dXJuIGNkcm9tX3RyYW5zZmVyX3BhY2tldF9jb21tYW5kKGRyaXZlLCBycSwgJmNkcm9tX3BjX2ludHIpOwp9CgoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9kb19wYWNrZXRfY29tbWFuZCAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglpbnQgbGVuOwoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJaW5mby0+ZG1hID0gMDsKCXJxLT5jbWRfZmxhZ3MgJj0gflJFUV9GQUlMRUQ7CglsZW4gPSBycS0+ZGF0YV9sZW47CgoJLyogU3RhcnQgc2VuZGluZyB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUuICovCglyZXR1cm4gY2Ryb21fc3RhcnRfcGFja2V0X2NvbW1hbmQoZHJpdmUsIGxlbiwgY2Ryb21fZG9fcGNfY29udGludWF0aW9uKTsKfQoKCnN0YXRpYyBpbnQgY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEpCnsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIHNlbnNlOwoJaW50IHJldHJpZXMgPSAxMDsKCXVuc2lnbmVkIGludCBmbGFncyA9IHJxLT5jbWRfZmxhZ3M7CgoJaWYgKHJxLT5zZW5zZSA9PSBOVUxMKQoJCXJxLT5zZW5zZSA9ICZzZW5zZTsKCgkvKiBTdGFydCBvZiByZXRyeSBsb29wLiAqLwoJZG8gewoJCWludCBlcnJvcjsKCQl1bnNpZ25lZCBsb25nIHRpbWUgPSBqaWZmaWVzOwoJCXJxLT5jbWRfZmxhZ3MgPSBmbGFnczsKCgkJZXJyb3IgPSBpZGVfZG9fZHJpdmVfY21kKGRyaXZlLCBycSwgaWRlX3dhaXQpOwoJCXRpbWUgPSBqaWZmaWVzIC0gdGltZTsKCgkJLyogRklYTUU6IHdlIHNob3VsZCBwcm9iYWJseSBhYm9ydC9yZXRyeSBvciBzb21ldGhpbmcgCgkJICogaW4gY2FzZSBvZiBmYWlsdXJlICovCgkJaWYgKHJxLT5jbWRfZmxhZ3MgJiBSRVFfRkFJTEVEKSB7CgkJCS8qIFRoZSByZXF1ZXN0IGZhaWxlZC4gIFJldHJ5IGlmIGl0IHdhcyBkdWUgdG8gYSB1bml0CgkJCSAgIGF0dGVudGlvbiBzdGF0dXMKCQkJICAgKHVzdWFsbHkgbWVhbnMgbWVkaWEgd2FzIGNoYW5nZWQpLiAqLwoJCQlzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqcmVxYnVmID0gcnEtPnNlbnNlOwoKCQkJaWYgKHJlcWJ1Zi0+c2Vuc2Vfa2V5ID09IFVOSVRfQVRURU5USU9OKQoJCQkJY2Ryb21fc2F3X21lZGlhX2NoYW5nZShkcml2ZSk7CgkJCWVsc2UgaWYgKHJlcWJ1Zi0+c2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSAmJgoJCQkJIHJlcWJ1Zi0+YXNjID09IDQgJiYgcmVxYnVmLT5hc2NxICE9IDQpIHsKCQkJCS8qIFRoZSBkcml2ZSBpcyBpbiB0aGUgcHJvY2VzcyBvZiBsb2FkaW5nCgkJCQkgICBhIGRpc2suICBSZXRyeSwgYnV0IHdhaXQgYSBsaXR0bGUgdG8gZ2l2ZQoJCQkJICAgdGhlIGRyaXZlIHRpbWUgdG8gY29tcGxldGUgdGhlIGxvYWQuICovCgkJCQlzc2xlZXAoMik7CgkJCX0gZWxzZSB7CgkJCQkvKiBPdGhlcndpc2UsIGRvbid0IHJldHJ5LiAqLwoJCQkJcmV0cmllcyA9IDA7CgkJCX0KCQkJLS1yZXRyaWVzOwoJCX0KCgkJLyogRW5kIG9mIHJldHJ5IGxvb3AuICovCgl9IHdoaWxlICgocnEtPmNtZF9mbGFncyAmIFJFUV9GQUlMRUQpICYmIHJldHJpZXMgPj0gMCk7CgoJLyogUmV0dXJuIGFuIGVycm9yIGlmIHRoZSBjb21tYW5kIGZhaWxlZC4gKi8KCXJldHVybiAocnEtPmNtZF9mbGFncyAmIFJFUV9GQUlMRUQpID8gLUVJTyA6IDA7Cn0KCi8qCiAqIFdyaXRlIGhhbmRsaW5nCiAqLwpzdGF0aWMgaW50IGNkcm9tX3dyaXRlX2NoZWNrX2lyZWFzb24oaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgbGVuLCBpbnQgaXJlYXNvbikKewoJLyogVHdvIG5vdGVzIGFib3V0IElERSBpbnRlcnJ1cHQgcmVhc29uIGhlcmUgLSAwIG1lYW5zIHRoYXQKCSAqIHRoZSBkcml2ZSB3YW50cyB0byByZWNlaXZlIGRhdGEgZnJvbSB1cywgMiBtZWFucyB0aGF0CgkgKiB0aGUgZHJpdmUgaXMgZXhwZWN0aW5nIHRvIHRyYW5zZmVyIGRhdGEgdG8gdXMuCgkgKi8KCWlmIChpcmVhc29uID09IDApCgkJcmV0dXJuIDA7CgllbHNlIGlmIChpcmVhc29uID09IDIpIHsKCQkvKiBXaG9vcHMuLi4gVGhlIGRyaXZlIHdhbnRzIHRvIHNlbmQgZGF0YS4gKi8KCQlwcmludGsoS0VSTl9FUlIgIiVzOiB3cml0ZV9pbnRyOiB3cm9uZyB0cmFuc2ZlciBkaXJlY3Rpb24hXG4iLAoJCQkJCQkJZHJpdmUtPm5hbWUpOwoKCQl3aGlsZSAobGVuID4gMCkgewoJCQlpbnQgZHVtID0gMDsKCQkJSFdJRihkcml2ZSktPmF0YXBpX2lucHV0X2J5dGVzKGRyaXZlLCAmZHVtLCBzaXplb2YoZHVtKSk7CgkJCWxlbiAtPSBzaXplb2YoZHVtKTsKCQl9Cgl9IGVsc2UgewoJCS8qIERyaXZlIHdhbnRzIGEgY29tbWFuZCBwYWNrZXQsIG9yIGludmFsaWQgaXJlYXNvbi4uLiAqLwoJCXByaW50ayhLRVJOX0VSUiAiJXM6IHdyaXRlX2ludHI6IGJhZCBpbnRlcnJ1cHQgcmVhc29uICV4XG4iLAoJCQkJCQkJZHJpdmUtPm5hbWUsIGlyZWFzb24pOwoJfQoKCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCXJldHVybiAxOwp9CgpzdGF0aWMgdm9pZCBwb3N0X3RyYW5zZm9ybV9jb21tYW5kKHN0cnVjdCByZXF1ZXN0ICpyZXEpCnsKCXU4ICpjID0gcmVxLT5jbWQ7CgljaGFyICppYnVmOwoKCWlmICghYmxrX3BjX3JlcXVlc3QocmVxKSkKCQlyZXR1cm47CgoJaWYgKHJlcS0+YmlvKQoJCWlidWYgPSBiaW9fZGF0YShyZXEtPmJpbyk7CgllbHNlCgkJaWJ1ZiA9IHJlcS0+ZGF0YTsKCglpZiAoIWlidWYpCgkJcmV0dXJuOwoKCS8qCgkgKiBzZXQgYW5zaS1yZXZpc2lvbiBhbmQgcmVzcG9uc2UgZGF0YSBhcyBhdGFwaQoJICovCglpZiAoY1swXSA9PSBHUENNRF9JTlFVSVJZKSB7CgkJaWJ1ZlsyXSB8PSAyOwoJCWlidWZbM10gPSAoaWJ1ZlszXSAmIDB4ZjApIHwgMjsKCX0KfQoKdHlwZWRlZiB2b2lkICh4ZmVyX2Z1bmNfdCkoaWRlX2RyaXZlX3QgKiwgdm9pZCAqLCB1MzIpOwoKLyoKICogYmVzdCB3YXkgdG8gZGVhbCB3aXRoIGRtYSB0aGF0IGlzIG5vdCBzZWN0b3IgYWxpZ25lZCByaWdodCBub3cuLi4gbm90ZQogKiB0aGF0IGluIHRoaXMgcGF0aCB3ZSBhcmUgbm90IHVzaW5nIC0+ZGF0YSBvciAtPmJ1ZmZlciBhdCBhbGwuIHRoaXMgaXJzCiAqIGNhbiByZXBsYWNlIGNkcm9tX3BjX2ludHIsIGNkcm9tX3JlYWRfaW50ciwgYW5kIGNkcm9tX3dyaXRlX2ludHIgaW4gdGhlCiAqIGZ1dHVyZS4KICovCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fbmV3cGNfaW50cihpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJaW50IGRtYV9lcnJvciwgZG1hLCBzdGF0LCBpcmVhc29uLCBsZW4sIHRoaXNsZW47Cgl1OCBsb3djeWwsIGhpZ2hjeWw7Cgl4ZmVyX2Z1bmNfdCAqeGZlcmZ1bmM7Cgl1bnNpZ25lZCBsb25nIGZsYWdzOwoKCS8qIENoZWNrIGZvciBlcnJvcnMuICovCglkbWFfZXJyb3IgPSAwOwoJZG1hID0gaW5mby0+ZG1hOwoJaWYgKGRtYSkgewoJCWluZm8tPmRtYSA9IDA7CgkJZG1hX2Vycm9yID0gSFdJRihkcml2ZSktPmlkZV9kbWFfZW5kKGRyaXZlKTsKCX0KCglpZiAoY2Ryb21fZGVjb2RlX3N0YXR1cyhkcml2ZSwgMCwgJnN0YXQpKQoJCXJldHVybiBpZGVfc3RvcHBlZDsKCgkvKgoJICogdXNpbmcgZG1hLCB0cmFuc2ZlciBpcyBjb21wbGV0ZSBub3cKCSAqLwoJaWYgKGRtYSkgewoJCWlmIChkbWFfZXJyb3IpIHsKCQkJcHJpbnRrKEtFUk5fRVJSICJpZGUtY2Q6IGRtYSBlcnJvclxuIik7CgkJCWlkZV9kbWFfb2ZmKGRyaXZlKTsKCQkJcmV0dXJuIGlkZV9lcnJvcihkcml2ZSwgImRtYSBlcnJvciIsIHN0YXQpOwoJCX0KCgkJZW5kX3RoYXRfcmVxdWVzdF9jaHVuayhycSwgMSwgcnEtPmRhdGFfbGVuKTsKCQlycS0+ZGF0YV9sZW4gPSAwOwoJCWdvdG8gZW5kX3JlcXVlc3Q7Cgl9CgoJLyoKCSAqIG9rIHdlIGZhbGwgdG8gcGlvIDovCgkgKi8KCWlyZWFzb24gPSBIV0lGKGRyaXZlKS0+SU5CKElERV9JUkVBU09OX1JFRykgJiAweDM7Cglsb3djeWwgID0gSFdJRihkcml2ZSktPklOQihJREVfQkNPVU5UTF9SRUcpOwoJaGlnaGN5bCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVEhfUkVHKTsKCglsZW4gPSBsb3djeWwgKyAoMjU2ICogaGlnaGN5bCk7Cgl0aGlzbGVuID0gcnEtPmRhdGFfbGVuOwoJaWYgKHRoaXNsZW4gPiBsZW4pCgkJdGhpc2xlbiA9IGxlbjsKCgkvKgoJICogSWYgRFJRIGlzIGNsZWFyLCB0aGUgY29tbWFuZCBoYXMgY29tcGxldGVkLgoJICovCglpZiAoKHN0YXQgJiBEUlFfU1RBVCkgPT0gMCkKCQlnb3RvIGVuZF9yZXF1ZXN0OwoKCS8qCgkgKiBjaGVjayB3aGljaCB3YXkgdG8gdHJhbnNmZXIgZGF0YQoJICovCglpZiAocnFfZGF0YV9kaXIocnEpID09IFdSSVRFKSB7CgkJLyoKCQkgKiB3cml0ZSB0byBkcml2ZQoJCSAqLwoJCWlmIChjZHJvbV93cml0ZV9jaGVja19pcmVhc29uKGRyaXZlLCBsZW4sIGlyZWFzb24pKQoJCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CgoJCXhmZXJmdW5jID0gSFdJRihkcml2ZSktPmF0YXBpX291dHB1dF9ieXRlczsKCX0gZWxzZSAgewoJCS8qCgkJICogcmVhZCBmcm9tIGRyaXZlCgkJICovCgkJaWYgKGNkcm9tX3JlYWRfY2hlY2tfaXJlYXNvbihkcml2ZSwgbGVuLCBpcmVhc29uKSkKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCQl4ZmVyZnVuYyA9IEhXSUYoZHJpdmUpLT5hdGFwaV9pbnB1dF9ieXRlczsKCX0KCgkvKgoJICogdHJhbnNmZXIgZGF0YQoJICovCgl3aGlsZSAodGhpc2xlbiA+IDApIHsKCQlpbnQgYmxlbiA9IGJsZW4gPSBycS0+ZGF0YV9sZW47CgkJY2hhciAqcHRyID0gcnEtPmRhdGE7CgoJCS8qCgkJICogYmlvIGJhY2tlZD8KCQkgKi8KCQlpZiAocnEtPmJpbykgewoJCQlwdHIgPSBiaW9fZGF0YShycS0+YmlvKTsKCQkJYmxlbiA9IGJpb19pb3ZlYyhycS0+YmlvKS0+YnZfbGVuOwoJCX0KCgkJaWYgKCFwdHIpIHsKCQkJcHJpbnRrKEtFUk5fRVJSICIlczogY29uZnVzZWQsIG1pc3NpbmcgZGF0YVxuIiwgZHJpdmUtPm5hbWUpOwoJCQlicmVhazsKCQl9CgoJCWlmIChibGVuID4gdGhpc2xlbikKCQkJYmxlbiA9IHRoaXNsZW47CgoJCXhmZXJmdW5jKGRyaXZlLCBwdHIsIGJsZW4pOwoKCQl0aGlzbGVuIC09IGJsZW47CgkJbGVuIC09IGJsZW47CgkJcnEtPmRhdGFfbGVuIC09IGJsZW47CgoJCWlmIChycS0+YmlvKQoJCQllbmRfdGhhdF9yZXF1ZXN0X2NodW5rKHJxLCAxLCBibGVuKTsKCQllbHNlCgkJCXJxLT5kYXRhICs9IGJsZW47Cgl9CgoJLyoKCSAqIHBhZCwgaWYgbmVjZXNzYXJ5CgkgKi8KCWlmIChsZW4gPiAwKSB7CgkJd2hpbGUgKGxlbiA+IDApIHsKCQkJaW50IHBhZCA9IDA7CgoJCQl4ZmVyZnVuYyhkcml2ZSwgJnBhZCwgc2l6ZW9mKHBhZCkpOwoJCQlsZW4gLT0gc2l6ZW9mKHBhZCk7CgkJfQoJfQoKCUJVR19PTihIV0dST1VQKGRyaXZlKS0+aGFuZGxlciAhPSBOVUxMKTsKCglpZGVfc2V0X2hhbmRsZXIoZHJpdmUsIGNkcm9tX25ld3BjX2ludHIsIHJxLT50aW1lb3V0LCBOVUxMKTsKCXJldHVybiBpZGVfc3RhcnRlZDsKCmVuZF9yZXF1ZXN0OgoJaWYgKCFycS0+ZGF0YV9sZW4pCgkJcG9zdF90cmFuc2Zvcm1fY29tbWFuZChycSk7CgoJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CglibGtkZXZfZGVxdWV1ZV9yZXF1ZXN0KHJxKTsKCWVuZF90aGF0X3JlcXVlc3RfbGFzdChycSwgMSk7CglIV0dST1VQKGRyaXZlKS0+cnEgPSBOVUxMOwoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmaWRlX2xvY2ssIGZsYWdzKTsKCXJldHVybiBpZGVfc3RvcHBlZDsKfQoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV93cml0ZV9pbnRyKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJaW50IHN0YXQsIGlyZWFzb24sIGxlbiwgc2VjdG9yc190b190cmFuc2ZlciwgdXB0b2RhdGU7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCWludCBkbWFfZXJyb3IgPSAwLCBkbWEgPSBpbmZvLT5kbWE7Cgl1OCBsb3djeWwgPSAwLCBoaWdoY3lsID0gMDsKCglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CgoJLyogQ2hlY2sgZm9yIGVycm9ycy4gKi8KCWlmIChkbWEpIHsKCQlpbmZvLT5kbWEgPSAwOwoJCWlmICgoZG1hX2Vycm9yID0gSFdJRihkcml2ZSktPmlkZV9kbWFfZW5kKGRyaXZlKSkpIHsKCQkJcHJpbnRrKEtFUk5fRVJSICJpZGUtY2Q6IHdyaXRlIGRtYSBlcnJvclxuIik7CgkJCWlkZV9kbWFfb2ZmKGRyaXZlKTsKCQl9Cgl9CgoJaWYgKGNkcm9tX2RlY29kZV9zdGF0dXMoZHJpdmUsIDAsICZzdGF0KSkKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CgoJLyoKCSAqIHVzaW5nIGRtYSwgdHJhbnNmZXIgaXMgY29tcGxldGUgbm93CgkgKi8KCWlmIChkbWEpIHsKCQlpZiAoZG1hX2Vycm9yKQoJCQlyZXR1cm4gaWRlX2Vycm9yKGRyaXZlLCAiZG1hIGVycm9yIiwgc3RhdCk7CgoJCWlkZV9lbmRfcmVxdWVzdChkcml2ZSwgMSwgcnEtPm5yX3NlY3RvcnMpOwoJCXJldHVybiBpZGVfc3RvcHBlZDsKCX0KCgkvKiBSZWFkIHRoZSBpbnRlcnJ1cHQgcmVhc29uIGFuZCB0aGUgdHJhbnNmZXIgbGVuZ3RoLiAqLwoJaXJlYXNvbiA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0lSRUFTT05fUkVHKTsKCWxvd2N5bCAgPSBIV0lGKGRyaXZlKS0+SU5CKElERV9CQ09VTlRMX1JFRyk7CgloaWdoY3lsID0gSFdJRihkcml2ZSktPklOQihJREVfQkNPVU5USF9SRUcpOwoKCWxlbiA9IGxvd2N5bCArICgyNTYgKiBoaWdoY3lsKTsKCgkvKiBJZiBEUlEgaXMgY2xlYXIsIHRoZSBjb21tYW5kIGhhcyBjb21wbGV0ZWQuICovCglpZiAoKHN0YXQgJiBEUlFfU1RBVCkgPT0gMCkgewoJCS8qIElmIHdlJ3JlIG5vdCBkb25lIHdyaXRpbmcsIGNvbXBsYWluLgoJCSAqIE90aGVyd2lzZSwgY29tcGxldGUgdGhlIGNvbW1hbmQgbm9ybWFsbHkuCgkJICovCgkJdXB0b2RhdGUgPSAxOwoJCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzID4gMCkgewoJCQlwcmludGsoS0VSTl9FUlIgIiVzOiB3cml0ZV9pbnRyOiBkYXRhIHVuZGVycnVuICglZCBibG9ja3MpXG4iLAoJCQlkcml2ZS0+bmFtZSwgcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyk7CgkJCXVwdG9kYXRlID0gMDsKCQl9CgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIHVwdG9kYXRlKTsKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7Cgl9CgoJLyogQ2hlY2sgdGhhdCB0aGUgZHJpdmUgaXMgZXhwZWN0aW5nIHRvIGRvIHRoZSBzYW1lIHRoaW5nIHdlIGFyZS4gKi8KCWlmIChjZHJvbV93cml0ZV9jaGVja19pcmVhc29uKGRyaXZlLCBsZW4sIGlyZWFzb24pKQoJCXJldHVybiBpZGVfc3RvcHBlZDsKCglzZWN0b3JzX3RvX3RyYW5zZmVyID0gbGVuIC8gU0VDVE9SX1NJWkU7CgoJLyoKCSAqIG5vdyBsb29wIGFuZCB3cml0ZSBvdXQgdGhlIGRhdGEKCSAqLwoJd2hpbGUgKHNlY3RvcnNfdG9fdHJhbnNmZXIgPiAwKSB7CgkJaW50IHRoaXNfdHJhbnNmZXI7CgoJCWlmICghcnEtPmN1cnJlbnRfbnJfc2VjdG9ycykgewoJCQlwcmludGsoS0VSTl9FUlIgImlkZS1jZDogd3JpdGVfaW50cjogb29wc1xuIik7CgkJCWJyZWFrOwoJCX0KCgkJLyoKCQkgKiBGaWd1cmUgb3V0IGhvdyBtYW55IHNlY3RvcnMgd2UgY2FuIHRyYW5zZmVyCgkJICovCgkJdGhpc190cmFuc2ZlciA9IG1pbl90KGludCwgc2VjdG9yc190b190cmFuc2ZlciwgcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyk7CgoJCXdoaWxlICh0aGlzX3RyYW5zZmVyID4gMCkgewoJCQlIV0lGKGRyaXZlKS0+YXRhcGlfb3V0cHV0X2J5dGVzKGRyaXZlLCBycS0+YnVmZmVyLCBTRUNUT1JfU0laRSk7CgkJCXJxLT5idWZmZXIgKz0gU0VDVE9SX1NJWkU7CgkJCS0tcnEtPm5yX3NlY3RvcnM7CgkJCS0tcnEtPmN1cnJlbnRfbnJfc2VjdG9yczsKCQkJKytycS0+c2VjdG9yOwoJCQktLXRoaXNfdHJhbnNmZXI7CgkJCS0tc2VjdG9yc190b190cmFuc2ZlcjsKCQl9CgoJCS8qCgkJICogY3VycmVudCBidWZmZXIgY29tcGxldGUsIG1vdmUgb24KCQkgKi8KCQlpZiAocnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA9PSAwICYmIHJxLT5ucl9zZWN0b3JzKQoJCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMSk7Cgl9CgoJLyogcmUtYXJtIGhhbmRsZXIgKi8KCWlkZV9zZXRfaGFuZGxlcihkcml2ZSwgJmNkcm9tX3dyaXRlX2ludHIsIEFUQVBJX1dBSVRfUEMsIE5VTEwpOwoJcmV0dXJuIGlkZV9zdGFydGVkOwp9CgpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3N0YXJ0X3dyaXRlX2NvbnQoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CgojaWYgMAkvKiB0aGUgaW1tZWRpYXRlIGJpdCAqLwoJcnEtPmNtZFsxXSA9IDEgPDwgMzsKI2VuZGlmCglycS0+dGltZW91dCA9IEFUQVBJX1dBSVRfUEM7CgoJcmV0dXJuIGNkcm9tX3RyYW5zZmVyX3BhY2tldF9jb21tYW5kKGRyaXZlLCBycSwgY2Ryb21fd3JpdGVfaW50cik7Cn0KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc3RhcnRfd3JpdGUoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IGdlbmRpc2sgKmcgPSBpbmZvLT5kaXNrOwoJdW5zaWduZWQgc2hvcnQgc2VjdG9yc19wZXJfZnJhbWUgPSBxdWV1ZV9oYXJkc2VjdF9zaXplKGRyaXZlLT5xdWV1ZSkgPj4gU0VDVE9SX0JJVFM7CgoJLyoKCSAqIHdyaXRlcyAqbXVzdCogYmUgaGFyZHdhcmUgZnJhbWUgYWxpZ25lZAoJICovCglpZiAoKHJxLT5ucl9zZWN0b3JzICYgKHNlY3RvcnNfcGVyX2ZyYW1lIC0gMSkpIHx8CgkgICAgKHJxLT5zZWN0b3IgJiAoc2VjdG9yc19wZXJfZnJhbWUgLSAxKSkpIHsKCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCS8qCgkgKiBkaXNrIGhhcyBiZWNvbWUgd3JpdGUgcHJvdGVjdGVkCgkgKi8KCWlmIChnLT5wb2xpY3kpIHsKCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCWluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkID0gMDsKCgkvKiB1c2UgZG1hLCBpZiBwb3NzaWJsZS4gd2UgZG9uJ3QgbmVlZCB0byBjaGVjayBtb3JlLCBzaW5jZSB3ZQoJICoga25vdyB0aGF0IHRoZSB0cmFuc2ZlciBpcyBhbHdheXMgKGF0IGxlYXN0ISkgZnJhbWUgYWxpZ25lZCAqLwoJaW5mby0+ZG1hID0gZHJpdmUtPnVzaW5nX2RtYSA/IDEgOiAwOwoKCWluZm8tPmRldmluZm8ubWVkaWFfd3JpdHRlbiA9IDE7CgoJLyogU3RhcnQgc2VuZGluZyB0aGUgd3JpdGUgcmVxdWVzdCB0byB0aGUgZHJpdmUuICovCglyZXR1cm4gY2Ryb21fc3RhcnRfcGFja2V0X2NvbW1hbmQoZHJpdmUsIDMyNzY4LCBjZHJvbV9zdGFydF93cml0ZV9jb250KTsKfQoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9kb19uZXdwY19jb250KGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoKCWlmICghcnEtPnRpbWVvdXQpCgkJcnEtPnRpbWVvdXQgPSBBVEFQSV9XQUlUX1BDOwoKCXJldHVybiBjZHJvbV90cmFuc2Zlcl9wYWNrZXRfY29tbWFuZChkcml2ZSwgcnEsIGNkcm9tX25ld3BjX2ludHIpOwp9CgpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX2RvX2Jsb2NrX3BjKGlkZV9kcml2ZV90ICpkcml2ZSwgc3RydWN0IHJlcXVlc3QgKnJxKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCglycS0+Y21kX2ZsYWdzIHw9IFJFUV9RVUlFVDsKCglpbmZvLT5kbWEgPSAwOwoKCS8qCgkgKiBzZyByZXF1ZXN0CgkgKi8KCWlmIChycS0+YmlvKSB7CgkJaW50IG1hc2sgPSBkcml2ZS0+cXVldWUtPmRtYV9hbGlnbm1lbnQ7CgkJdW5zaWduZWQgbG9uZyBhZGRyID0gKHVuc2lnbmVkIGxvbmcpIHBhZ2VfYWRkcmVzcyhiaW9fcGFnZShycS0+YmlvKSk7CgoJCWluZm8tPmRtYSA9IGRyaXZlLT51c2luZ19kbWE7CgoJCS8qCgkJICogY2hlY2sgaWYgZG1hIGlzIHNhZmUKCQkgKgoJCSAqIE5PVEUhIFRoZSAibGVuIiBhbmQgImFkZHIiIGNoZWNrcyBzaG91bGQgcG9zc2libHkgaGF2ZQoJCSAqIHNlcGFyYXRlIG1hc2tzLgoJCSAqLwoJCWlmICgocnEtPmRhdGFfbGVuICYgMTUpIHx8IChhZGRyICYgbWFzaykpCgkJCWluZm8tPmRtYSA9IDA7Cgl9CgoJLyogU3RhcnQgc2VuZGluZyB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUuICovCglyZXR1cm4gY2Ryb21fc3RhcnRfcGFja2V0X2NvbW1hbmQoZHJpdmUsIHJxLT5kYXRhX2xlbiwgY2Ryb21fZG9fbmV3cGNfY29udCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIGNkcm9tIGRyaXZlciByZXF1ZXN0IHJvdXRpbmUuCiAqLwpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90CmlkZV9kb19yd19jZHJvbSAoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEsIHNlY3Rvcl90IGJsb2NrKQp7CglpZGVfc3RhcnRzdG9wX3QgYWN0aW9uOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJaWYgKGJsa19mc19yZXF1ZXN0KHJxKSkgewoJCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zZWVraW5nKSB7CgkJCXVuc2lnbmVkIGxvbmcgZWxhcHNlZCA9IGppZmZpZXMgLSBpbmZvLT5zdGFydF9zZWVrOwoJCQlpbnQgc3RhdCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX1NUQVRVU19SRUcpOwoKCQkJaWYgKChzdGF0ICYgU0VFS19TVEFUKSAhPSBTRUVLX1NUQVQpIHsKCQkJCWlmIChlbGFwc2VkIDwgSURFQ0RfU0VFS19USU1FT1VUKSB7CgkJCQkJaWRlX3N0YWxsX3F1ZXVlKGRyaXZlLCBJREVDRF9TRUVLX1RJTUVSKTsKCQkJCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CgkJCQl9CgkJCQlwcmludGsgKEtFUk5fRVJSICIlczogRFNDIHRpbWVvdXRcbiIsIGRyaXZlLT5uYW1lKTsKCQkJfQoJCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zZWVraW5nID0gMDsKCQl9CgkJaWYgKChycV9kYXRhX2RpcihycSkgPT0gUkVBRCkgJiYgSURFX0xBUkdFX1NFRUsoaW5mby0+bGFzdF9ibG9jaywgYmxvY2ssIElERUNEX1NFRUtfVEhSRVNIT0xEKSAmJiBkcml2ZS0+ZHNjX292ZXJsYXApIHsKCQkJYWN0aW9uID0gY2Ryb21fc3RhcnRfc2Vlayhkcml2ZSwgYmxvY2spOwoJCX0gZWxzZSB7CgkJCWlmIChycV9kYXRhX2RpcihycSkgPT0gUkVBRCkKCQkJCWFjdGlvbiA9IGNkcm9tX3N0YXJ0X3JlYWQoZHJpdmUsIGJsb2NrKTsKCQkJZWxzZQoJCQkJYWN0aW9uID0gY2Ryb21fc3RhcnRfd3JpdGUoZHJpdmUsIHJxKTsKCQl9CgkJaW5mby0+bGFzdF9ibG9jayA9IGJsb2NrOwoJCXJldHVybiBhY3Rpb247Cgl9IGVsc2UgaWYgKHJxLT5jbWRfdHlwZSA9PSBSRVFfVFlQRV9TRU5TRSB8fAoJCSAgIHJxLT5jbWRfdHlwZSA9PSBSRVFfVFlQRV9BVEFfUEMpIHsKCQlyZXR1cm4gY2Ryb21fZG9fcGFja2V0X2NvbW1hbmQoZHJpdmUpOwoJfSBlbHNlIGlmIChibGtfcGNfcmVxdWVzdChycSkpIHsKCQlyZXR1cm4gY2Ryb21fZG9fYmxvY2tfcGMoZHJpdmUsIHJxKTsKCX0gZWxzZSBpZiAoYmxrX3NwZWNpYWxfcmVxdWVzdChycSkpIHsKCQkvKgoJCSAqIHJpZ2h0IG5vdyB0aGlzIGNhbiBvbmx5IGJlIGEgcmVzZXQuLi4KCQkgKi8KCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMSk7CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCWJsa19kdW1wX3JxX2ZsYWdzKHJxLCAiaWRlLWNkIGJhZCBmbGFncyIpOwoJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJcmV0dXJuIGlkZV9zdG9wcGVkOwp9CgoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElvY3RsIGhhbmRsaW5nLgogKgogKiBSb3V0aW5lcyB3aGljaCBxdWV1ZSBwYWNrZXQgY29tbWFuZHMgdGFrZSBhcyBhIGZpbmFsIGFyZ3VtZW50IGEgcG9pbnRlcgogKiB0byBhIHJlcXVlc3Rfc2Vuc2Ugc3RydWN0LiAgSWYgZXhlY3V0aW9uIG9mIHRoZSBjb21tYW5kIHJlc3VsdHMKICogaW4gYW4gZXJyb3Igd2l0aCBhIENIRUNLIENPTkRJVElPTiBzdGF0dXMsIHRoaXMgc3RydWN0dXJlIHdpbGwgYmUgZmlsbGVkCiAqIHdpdGggdGhlIHJlc3VsdHMgb2YgdGhlIHN1YnNlcXVlbnQgcmVxdWVzdCBzZW5zZSBjb21tYW5kLiAgVGhlIHBvaW50ZXIKICogY2FuIGFsc28gYmUgTlVMTCwgaW4gd2hpY2ggY2FzZSBubyBzZW5zZSBpbmZvcm1hdGlvbiBpcyByZXR1cm5lZC4KICovCgojaWYgISBTVEFOREFSRF9BVEFQSQpzdGF0aWMgaW5saW5lCmludCBiaW4yYmNkIChpbnQgeCkKewoJcmV0dXJuICh4JTEwKSB8ICgoeC8xMCkgPDwgNCk7Cn0KCgpzdGF0aWMgaW5saW5lCmludCBiY2QyYmluIChpbnQgeCkKewoJcmV0dXJuICh4ID4+IDQpICogMTAgKyAoeCAmIDB4MGYpOwp9CgpzdGF0aWMKdm9pZCBtc2ZfZnJvbV9iY2QgKHN0cnVjdCBhdGFwaV9tc2YgKm1zZikKewoJbXNmLT5taW51dGUgPSBiY2QyYmluIChtc2YtPm1pbnV0ZSk7Cgltc2YtPnNlY29uZCA9IGJjZDJiaW4gKG1zZi0+c2Vjb25kKTsKCW1zZi0+ZnJhbWUgID0gYmNkMmJpbiAobXNmLT5mcmFtZSk7Cn0KCiNlbmRpZiAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCgpzdGF0aWMgaW5saW5lCnZvaWQgbGJhX3RvX21zZiAoaW50IGxiYSwgYnl0ZSAqbSwgYnl0ZSAqcywgYnl0ZSAqZikKewoJbGJhICs9IENEX01TRl9PRkZTRVQ7CglsYmEgJj0gMHhmZmZmZmY7ICAvKiBuZWdhdGl2ZSBsYmFzIHVzZSBvbmx5IDI0IGJpdHMgKi8KCSptID0gbGJhIC8gKENEX1NFQ1MgKiBDRF9GUkFNRVMpOwoJbGJhICU9IChDRF9TRUNTICogQ0RfRlJBTUVTKTsKCSpzID0gbGJhIC8gQ0RfRlJBTUVTOwoJKmYgPSBsYmEgJSBDRF9GUkFNRVM7Cn0KCgpzdGF0aWMgaW5saW5lCmludCBtc2ZfdG9fbGJhIChieXRlIG0sIGJ5dGUgcywgYnl0ZSBmKQp7CglyZXR1cm4gKCgobSAqIENEX1NFQ1MpICsgcykgKiBDRF9GUkFNRVMgKyBmKSAtIENEX01TRl9PRkZTRVQ7Cn0KCnN0YXRpYyBpbnQgY2Ryb21fY2hlY2tfc3RhdHVzKGlkZV9kcml2ZV90ICpkcml2ZSwgc3RydWN0IHJlcXVlc3Rfc2Vuc2UgKnNlbnNlKQp7CglzdHJ1Y3QgcmVxdWVzdCByZXE7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpID0gJmluZm8tPmRldmluZm87CgoJY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGRyaXZlLCAmcmVxKTsKCglyZXEuc2Vuc2UgPSBzZW5zZTsKCXJlcS5jbWRbMF0gPSBHUENNRF9URVNUX1VOSVRfUkVBRFk7CglyZXEuY21kX2ZsYWdzIHw9IFJFUV9RVUlFVDsKCiNpZiAhIFNUQU5EQVJEX0FUQVBJCiAgICAgICAgLyogdGhlIFNhbnlvIDMgQ0QgY2hhbmdlciB1c2VzIGJ5dGUgNyBvZiBURVNUX1VOSVRfUkVBRFkgdG8gCiAgICAgICAgICAgc3dpdGNoIENEcyBpbnN0ZWFkIG9mIHN1cHBvcnRpbmcgdGhlIExPQURfVU5MT0FEIG9wY29kZSAgICovCgoJcmVxLmNtZFs3XSA9IGNkaS0+c2FueW9fc2xvdCAlIDM7CiNlbmRpZiAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCglyZXR1cm4gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwp9CgoKLyogTG9jayB0aGUgZG9vciBpZiBMT0NLRkxBRyBpcyBub256ZXJvOyB1bmxvY2sgaXQgb3RoZXJ3aXNlLiAqLwpzdGF0aWMgaW50CmNkcm9tX2xvY2tkb29yKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IGxvY2tmbGFnLCBzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqc2Vuc2UpCnsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIG15X3NlbnNlOwoJc3RydWN0IHJlcXVlc3QgcmVxOwoJaW50IHN0YXQ7CgoJaWYgKHNlbnNlID09IE5VTEwpCgkJc2Vuc2UgPSAmbXlfc2Vuc2U7CgoJLyogSWYgdGhlIGRyaXZlIGNhbm5vdCBsb2NrIHRoZSBkb29yLCBqdXN0IHByZXRlbmQuICovCglpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZG9vcmxvY2spIHsKCQlzdGF0ID0gMDsKCX0gZWxzZSB7CgkJY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGRyaXZlLCAmcmVxKTsKCQlyZXEuc2Vuc2UgPSBzZW5zZTsKCQlyZXEuY21kWzBdID0gR1BDTURfUFJFVkVOVF9BTExPV19NRURJVU1fUkVNT1ZBTDsKCQlyZXEuY21kWzRdID0gbG9ja2ZsYWcgPyAxIDogMDsKCQlzdGF0ID0gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwoJfQoKCS8qIElmIHdlIGdvdCBhbiBpbGxlZ2FsIGZpZWxkIGVycm9yLCB0aGUgZHJpdmUKCSAgIHByb2JhYmx5IGNhbm5vdCBsb2NrIHRoZSBkb29yLiAqLwoJaWYgKHN0YXQgIT0gMCAmJgoJICAgIHNlbnNlLT5zZW5zZV9rZXkgPT0gSUxMRUdBTF9SRVFVRVNUICYmCgkgICAgKHNlbnNlLT5hc2MgPT0gMHgyNCB8fCBzZW5zZS0+YXNjID09IDB4MjApKSB7CgkJcHJpbnRrIChLRVJOX0VSUiAiJXM6IGRvb3IgbG9ja2luZyBub3Qgc3VwcG9ydGVkXG4iLAoJCQlkcml2ZS0+bmFtZSk7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZG9vcmxvY2sgPSAxOwoJCXN0YXQgPSAwOwoJfQoJCgkvKiBubyBtZWRpdW0sIHRoYXQncyBhbHJpZ2h0LiAqLwoJaWYgKHN0YXQgIT0gMCAmJiBzZW5zZS0+c2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSAmJiBzZW5zZS0+YXNjID09IDB4M2EpCgkJc3RhdCA9IDA7CgoJaWYgKHN0YXQgPT0gMCkKCQlDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmRvb3JfbG9ja2VkID0gbG9ja2ZsYWc7CgoJcmV0dXJuIHN0YXQ7Cn0KCgovKiBFamVjdCB0aGUgZGlzayBpZiBFSkVDVEZMQUcgaXMgMC4KICAgSWYgRUpFQ1RGTEFHIGlzIDEsIHRyeSB0byByZWxvYWQgdGhlIGRpc2suICovCnN0YXRpYyBpbnQgY2Ryb21fZWplY3QoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgZWplY3RmbGFnLAoJCSAgICAgICBzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqc2Vuc2UpCnsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCWNoYXIgbG9laiA9IDB4MDI7CgoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2VqZWN0ICYmICFlamVjdGZsYWcpCgkJcmV0dXJuIC1FRFJJVkVfQ0FOVF9ET19USElTOwoJCgkvKiByZWxvYWQgZmFpbHMgb24gc29tZSBkcml2ZXMsIGlmIHRoZSB0cmF5IGlzIGxvY2tlZCAqLwoJaWYgKENEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+ZG9vcl9sb2NrZWQgJiYgZWplY3RmbGFnKQoJCXJldHVybiAwOwoKCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgoJLyogb25seSB0ZWxsIGRyaXZlIHRvIGNsb3NlIHRyYXkgaWYgb3BlbiwgaWYgaXQgY2FuIGRvIHRoYXQgKi8KCWlmIChlamVjdGZsYWcgJiYgIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNsb3NlX3RyYXkpCgkJbG9laiA9IDA7CgoJcmVxLnNlbnNlID0gc2Vuc2U7CglyZXEuY21kWzBdID0gR1BDTURfU1RBUlRfU1RPUF9VTklUOwoJcmVxLmNtZFs0XSA9IGxvZWogfCAoZWplY3RmbGFnICE9IDApOwoJcmV0dXJuIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKfQoKc3RhdGljIGludCBjZHJvbV9yZWFkX2NhcGFjaXR5KGlkZV9kcml2ZV90ICpkcml2ZSwgdW5zaWduZWQgbG9uZyAqY2FwYWNpdHksCgkJCSAgICAgICB1bnNpZ25lZCBsb25nICpzZWN0b3JzX3Blcl9mcmFtZSwKCQkJICAgICAgIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJc3RydWN0IHsKCQlfX3UzMiBsYmE7CgkJX191MzIgYmxvY2tsZW47Cgl9IGNhcGJ1ZjsKCglpbnQgc3RhdDsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsICZyZXEpOwoKCXJlcS5zZW5zZSA9IHNlbnNlOwoJcmVxLmNtZFswXSA9IEdQQ01EX1JFQURfQ0RWRF9DQVBBQ0lUWTsKCXJlcS5kYXRhID0gKGNoYXIgKikmY2FwYnVmOwoJcmVxLmRhdGFfbGVuID0gc2l6ZW9mKGNhcGJ1Zik7CglyZXEuY21kX2ZsYWdzIHw9IFJFUV9RVUlFVDsKCglzdGF0ID0gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwoJaWYgKHN0YXQgPT0gMCkgewoJCSpjYXBhY2l0eSA9IDEgKyBiZTMyX3RvX2NwdShjYXBidWYubGJhKTsKCQkqc2VjdG9yc19wZXJfZnJhbWUgPQoJCQliZTMyX3RvX2NwdShjYXBidWYuYmxvY2tsZW4pID4+IFNFQ1RPUl9CSVRTOwoJfQoKCXJldHVybiBzdGF0Owp9CgpzdGF0aWMgaW50IGNkcm9tX3JlYWRfdG9jZW50cnkoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgdHJhY2tubywgaW50IG1zZl9mbGFnLAoJCQkJaW50IGZvcm1hdCwgY2hhciAqYnVmLCBpbnQgYnVmbGVuLAoJCQkJc3RydWN0IHJlcXVlc3Rfc2Vuc2UgKnNlbnNlKQp7CglzdHJ1Y3QgcmVxdWVzdCByZXE7CgoJY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGRyaXZlLCAmcmVxKTsKCglyZXEuc2Vuc2UgPSBzZW5zZTsKCXJlcS5kYXRhID0gIGJ1ZjsKCXJlcS5kYXRhX2xlbiA9IGJ1ZmxlbjsKCXJlcS5jbWRfZmxhZ3MgfD0gUkVRX1FVSUVUOwoJcmVxLmNtZFswXSA9IEdQQ01EX1JFQURfVE9DX1BNQV9BVElQOwoJcmVxLmNtZFs2XSA9IHRyYWNrbm87CglyZXEuY21kWzddID0gKGJ1ZmxlbiA+PiA4KTsKCXJlcS5jbWRbOF0gPSAoYnVmbGVuICYgMHhmZik7CglyZXEuY21kWzldID0gKGZvcm1hdCA8PCA2KTsKCglpZiAobXNmX2ZsYWcpCgkJcmVxLmNtZFsxXSA9IDI7CgoJcmV0dXJuIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKfQoKCi8qIFRyeSB0byByZWFkIHRoZSBlbnRpcmUgVE9DIGZvciB0aGUgZGlzayBpbnRvIG91ciBpbnRlcm5hbCBidWZmZXIuICovCnN0YXRpYyBpbnQgY2Ryb21fcmVhZF90b2MoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqc2Vuc2UpCnsKCWludCBzdGF0LCBudHJhY2tzLCBpOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSA9ICZpbmZvLT5kZXZpbmZvOwoJc3RydWN0IGF0YXBpX3RvYyAqdG9jID0gaW5mby0+dG9jOwoJc3RydWN0IHsKCQlzdHJ1Y3QgYXRhcGlfdG9jX2hlYWRlciBoZHI7CgkJc3RydWN0IGF0YXBpX3RvY19lbnRyeSAgZW50OwoJfSBtc190bXA7Cglsb25nIGxhc3Rfd3JpdHRlbjsKCXVuc2lnbmVkIGxvbmcgc2VjdG9yc19wZXJfZnJhbWUgPSBTRUNUT1JTX1BFUl9GUkFNRTsKCglpZiAodG9jID09IE5VTEwpIHsKCQkvKiBUcnkgdG8gYWxsb2NhdGUgc3BhY2UuICovCgkJdG9jID0ga21hbGxvYyhzaXplb2Yoc3RydWN0IGF0YXBpX3RvYyksIEdGUF9LRVJORUwpOwoJCWlmICh0b2MgPT0gTlVMTCkgewoJCQlwcmludGsgKEtFUk5fRVJSICIlczogTm8gY2Ryb20gVE9DIGJ1ZmZlciFcbiIsIGRyaXZlLT5uYW1lKTsKCQkJcmV0dXJuIC1FTk9NRU07CgkJfQoJCWluZm8tPnRvYyA9IHRvYzsKCX0KCgkvKiBDaGVjayB0byBzZWUgaWYgdGhlIGV4aXN0aW5nIGRhdGEgaXMgc3RpbGwgdmFsaWQuCgkgICBJZiBpdCBpcywganVzdCByZXR1cm4uICovCgkodm9pZCkgY2Ryb21fY2hlY2tfc3RhdHVzKGRyaXZlLCBzZW5zZSk7CgoJaWYgKENEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkKQoJCXJldHVybiAwOwoKCS8qIFRyeSB0byBnZXQgdGhlIHRvdGFsIGNkcm9tIGNhcGFjaXR5IGFuZCBzZWN0b3Igc2l6ZS4gKi8KCXN0YXQgPSBjZHJvbV9yZWFkX2NhcGFjaXR5KGRyaXZlLCAmdG9jLT5jYXBhY2l0eSwgJnNlY3RvcnNfcGVyX2ZyYW1lLAoJCQkJICAgc2Vuc2UpOwoJaWYgKHN0YXQpCgkJdG9jLT5jYXBhY2l0eSA9IDB4MWZmZmZmOwoKCXNldF9jYXBhY2l0eShpbmZvLT5kaXNrLCB0b2MtPmNhcGFjaXR5ICogc2VjdG9yc19wZXJfZnJhbWUpOwoJLyogU2F2ZSBhIHByaXZhdGUgY29weSBvZiB0ZSBUT0MgY2FwYWNpdHkgZm9yIGVycm9yIGhhbmRsaW5nICovCglkcml2ZS0+cHJvYmVkX2NhcGFjaXR5ID0gdG9jLT5jYXBhY2l0eSAqIHNlY3RvcnNfcGVyX2ZyYW1lOwoKCWJsa19xdWV1ZV9oYXJkc2VjdF9zaXplKGRyaXZlLT5xdWV1ZSwKCQkJCXNlY3RvcnNfcGVyX2ZyYW1lIDw8IFNFQ1RPUl9CSVRTKTsKCgkvKiBGaXJzdCByZWFkIGp1c3QgdGhlIGhlYWRlciwgc28gd2Uga25vdyBob3cgbG9uZyB0aGUgVE9DIGlzLiAqLwoJc3RhdCA9IGNkcm9tX3JlYWRfdG9jZW50cnkoZHJpdmUsIDAsIDEsIDAsIChjaGFyICopICZ0b2MtPmhkciwKCQkJCSAgICBzaXplb2Yoc3RydWN0IGF0YXBpX3RvY19oZWFkZXIpLCBzZW5zZSk7CglpZiAoc3RhdCkKCQlyZXR1cm4gc3RhdDsKCiNpZiAhIFNUQU5EQVJEX0FUQVBJCglpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jdHJhY2tzX2FzX2JjZCkgewoJCXRvYy0+aGRyLmZpcnN0X3RyYWNrID0gYmNkMmJpbih0b2MtPmhkci5maXJzdF90cmFjayk7CgkJdG9jLT5oZHIubGFzdF90cmFjayAgPSBiY2QyYmluKHRvYy0+aGRyLmxhc3RfdHJhY2spOwoJfQojZW5kaWYgIC8qIG5vdCBTVEFOREFSRF9BVEFQSSAqLwoKCW50cmFja3MgPSB0b2MtPmhkci5sYXN0X3RyYWNrIC0gdG9jLT5oZHIuZmlyc3RfdHJhY2sgKyAxOwoJaWYgKG50cmFja3MgPD0gMCkKCQlyZXR1cm4gLUVJTzsKCWlmIChudHJhY2tzID4gTUFYX1RSQUNLUykKCQludHJhY2tzID0gTUFYX1RSQUNLUzsKCgkvKiBOb3cgcmVhZCB0aGUgd2hvbGUgc2NobWVlci4gKi8KCXN0YXQgPSBjZHJvbV9yZWFkX3RvY2VudHJ5KGRyaXZlLCB0b2MtPmhkci5maXJzdF90cmFjaywgMSwgMCwKCQkJCSAgKGNoYXIgKikmdG9jLT5oZHIsCgkJCQkgICBzaXplb2Yoc3RydWN0IGF0YXBpX3RvY19oZWFkZXIpICsKCQkJCSAgIChudHJhY2tzICsgMSkgKgoJCQkJICAgc2l6ZW9mKHN0cnVjdCBhdGFwaV90b2NfZW50cnkpLCBzZW5zZSk7CgoJaWYgKHN0YXQgJiYgdG9jLT5oZHIuZmlyc3RfdHJhY2sgPiAxKSB7CgkJLyogQ2RzIHdpdGggQ0RJIHRyYWNrcyBvbmx5IGRvbid0IGhhdmUgYW55IFRPQyBlbnRyaWVzLAoJCSAgIGRlc3BpdGUgb2YgdGhpcyB0aGUgcmV0dXJuZWQgdmFsdWVzIGFyZQoJCSAgIGZpcnN0X3RyYWNrID09IGxhc3RfdHJhY2sgPSBudW1iZXIgb2YgQ0RJIHRyYWNrcyArIDEsCgkJICAgc28gdGhhdCB0aGlzIGNhc2UgaXMgaW5kaXN0aW5ndWlzaGFibGUgZnJvbSB0aGUgc2FtZQoJCSAgIGxheW91dCBwbHVzIGFuIGFkZGl0aW9uYWwgYXVkaW8gdHJhY2suCgkJICAgSWYgd2UgZ2V0IGFuIGVycm9yIGZvciB0aGUgcmVndWxhciBjYXNlLCB3ZSBhc3N1bWUKCQkgICBhIENESSB3aXRob3V0IGFkZGl0aW9uYWwgYXVkaW8gdHJhY2tzLiBJbiB0aGlzIGNhc2UKCQkgICB0aGUgcmVhZGFibGUgVE9DIGlzIGVtcHR5IChDREkgdHJhY2tzIGFyZSBub3QgaW5jbHVkZWQpCgkJICAgYW5kIG9ubHkgaG9sZHMgdGhlIExlYWRvdXQgZW50cnkuIEhlaWtvIEVp32ZlbGR0ICovCgkJbnRyYWNrcyA9IDA7CgkJc3RhdCA9IGNkcm9tX3JlYWRfdG9jZW50cnkoZHJpdmUsIENEUk9NX0xFQURPVVQsIDEsIDAsCgkJCQkJICAgKGNoYXIgKikmdG9jLT5oZHIsCgkJCQkJICAgc2l6ZW9mKHN0cnVjdCBhdGFwaV90b2NfaGVhZGVyKSArCgkJCQkJICAgKG50cmFja3MgKyAxKSAqCgkJCQkJICAgc2l6ZW9mKHN0cnVjdCBhdGFwaV90b2NfZW50cnkpLAoJCQkJCSAgIHNlbnNlKTsKCQlpZiAoc3RhdCkgewoJCQlyZXR1cm4gc3RhdDsKCQl9CiNpZiAhIFNUQU5EQVJEX0FUQVBJCgkJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY3RyYWNrc19hc19iY2QpIHsKCQkJdG9jLT5oZHIuZmlyc3RfdHJhY2sgPSBiaW4yYmNkKENEUk9NX0xFQURPVVQpOwoJCQl0b2MtPmhkci5sYXN0X3RyYWNrID0gYmluMmJjZChDRFJPTV9MRUFET1VUKTsKCQl9IGVsc2UKI2VuZGlmICAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCQl7CgkJCXRvYy0+aGRyLmZpcnN0X3RyYWNrID0gQ0RST01fTEVBRE9VVDsKCQkJdG9jLT5oZHIubGFzdF90cmFjayA9IENEUk9NX0xFQURPVVQ7CgkJfQoJfQoKCWlmIChzdGF0KQoJCXJldHVybiBzdGF0OwoKCXRvYy0+aGRyLnRvY19sZW5ndGggPSBudG9ocyAodG9jLT5oZHIudG9jX2xlbmd0aCk7CgojaWYgISBTVEFOREFSRF9BVEFQSQoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY3RyYWNrc19hc19iY2QpIHsKCQl0b2MtPmhkci5maXJzdF90cmFjayA9IGJjZDJiaW4odG9jLT5oZHIuZmlyc3RfdHJhY2spOwoJCXRvYy0+aGRyLmxhc3RfdHJhY2sgID0gYmNkMmJpbih0b2MtPmhkci5sYXN0X3RyYWNrKTsKCX0KI2VuZGlmICAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCglmb3IgKGk9MDsgaTw9bnRyYWNrczsgaSsrKSB7CiNpZiAhIFNUQU5EQVJEX0FUQVBJCgkJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY2FkZHJfYXNfYmNkKSB7CgkJCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50b2N0cmFja3NfYXNfYmNkKQoJCQkJdG9jLT5lbnRbaV0udHJhY2sgPSBiY2QyYmluKHRvYy0+ZW50W2ldLnRyYWNrKTsKCQkJbXNmX2Zyb21fYmNkKCZ0b2MtPmVudFtpXS5hZGRyLm1zZik7CgkJfQojZW5kaWYgIC8qIG5vdCBTVEFOREFSRF9BVEFQSSAqLwoJCXRvYy0+ZW50W2ldLmFkZHIubGJhID0gbXNmX3RvX2xiYSAodG9jLT5lbnRbaV0uYWRkci5tc2YubWludXRlLAoJCQkJCQkgICB0b2MtPmVudFtpXS5hZGRyLm1zZi5zZWNvbmQsCgkJCQkJCSAgIHRvYy0+ZW50W2ldLmFkZHIubXNmLmZyYW1lKTsKCX0KCgkvKiBSZWFkIHRoZSBtdWx0aXNlc3Npb24gaW5mb3JtYXRpb24uICovCglpZiAodG9jLT5oZHIuZmlyc3RfdHJhY2sgIT0gQ0RST01fTEVBRE9VVCkgewoJCS8qIFJlYWQgdGhlIG11bHRpc2Vzc2lvbiBpbmZvcm1hdGlvbi4gKi8KCQlzdGF0ID0gY2Ryb21fcmVhZF90b2NlbnRyeShkcml2ZSwgMCwgMCwgMSwgKGNoYXIgKikmbXNfdG1wLAoJCQkJCSAgIHNpemVvZihtc190bXApLCBzZW5zZSk7CgkJaWYgKHN0YXQpCgkJCXJldHVybiBzdGF0OwoKCQl0b2MtPmxhc3Rfc2Vzc2lvbl9sYmEgPSBiZTMyX3RvX2NwdShtc190bXAuZW50LmFkZHIubGJhKTsKCX0gZWxzZSB7CgkJbXNfdG1wLmhkci5maXJzdF90cmFjayA9IG1zX3RtcC5oZHIubGFzdF90cmFjayA9IENEUk9NX0xFQURPVVQ7CgkJdG9jLT5sYXN0X3Nlc3Npb25fbGJhID0gbXNmX3RvX2xiYSgwLCAyLCAwKTsgLyogMG0gMnMgMGYgKi8KCX0KCiNpZiAhIFNUQU5EQVJEX0FUQVBJCglpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jYWRkcl9hc19iY2QpIHsKCQkvKiBSZS1yZWFkIG11bHRpc2Vzc2lvbiBpbmZvcm1hdGlvbiB1c2luZyBNU0YgZm9ybWF0ICovCgkJc3RhdCA9IGNkcm9tX3JlYWRfdG9jZW50cnkoZHJpdmUsIDAsIDEsIDEsIChjaGFyICopJm1zX3RtcCwKCQkJCQkgICBzaXplb2YobXNfdG1wKSwgc2Vuc2UpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCgkJbXNmX2Zyb21fYmNkICgmbXNfdG1wLmVudC5hZGRyLm1zZik7CgkJdG9jLT5sYXN0X3Nlc3Npb25fbGJhID0gbXNmX3RvX2xiYShtc190bXAuZW50LmFkZHIubXNmLm1pbnV0ZSwKCQkJCQkgIAkgICBtc190bXAuZW50LmFkZHIubXNmLnNlY29uZCwKCQkJCQkJICAgbXNfdG1wLmVudC5hZGRyLm1zZi5mcmFtZSk7Cgl9CiNlbmRpZiAgLyogbm90IFNUQU5EQVJEX0FUQVBJICovCgoJdG9jLT54YV9mbGFnID0gKG1zX3RtcC5oZHIuZmlyc3RfdHJhY2sgIT0gbXNfdG1wLmhkci5sYXN0X3RyYWNrKTsKCgkvKiBOb3cgdHJ5IHRvIGdldCB0aGUgdG90YWwgY2Ryb20gY2FwYWNpdHkuICovCglzdGF0ID0gY2Ryb21fZ2V0X2xhc3Rfd3JpdHRlbihjZGksICZsYXN0X3dyaXR0ZW4pOwoJaWYgKCFzdGF0ICYmIChsYXN0X3dyaXR0ZW4gPiB0b2MtPmNhcGFjaXR5KSkgewoJCXRvYy0+Y2FwYWNpdHkgPSBsYXN0X3dyaXR0ZW47CgkJc2V0X2NhcGFjaXR5KGluZm8tPmRpc2ssIHRvYy0+Y2FwYWNpdHkgKiBzZWN0b3JzX3Blcl9mcmFtZSk7CgkJZHJpdmUtPnByb2JlZF9jYXBhY2l0eSA9IHRvYy0+Y2FwYWNpdHkgKiBzZWN0b3JzX3Blcl9mcmFtZTsKCX0KCgkvKiBSZW1lbWJlciB0aGF0IHdlJ3ZlIHJlYWQgdGhpcyBzdHVmZi4gKi8KCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkID0gMTsKCglyZXR1cm4gMDsKfQoKCnN0YXRpYyBpbnQgY2Ryb21fcmVhZF9zdWJjaGFubmVsKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IGZvcm1hdCwgY2hhciAqYnVmLAoJCQkJIGludCBidWZsZW4sIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJc3RydWN0IHJlcXVlc3QgcmVxOwoKCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgoJcmVxLnNlbnNlID0gc2Vuc2U7CglyZXEuZGF0YSA9IGJ1ZjsKCXJlcS5kYXRhX2xlbiA9IGJ1ZmxlbjsKCXJlcS5jbWRbMF0gPSBHUENNRF9SRUFEX1NVQkNIQU5ORUw7CglyZXEuY21kWzFdID0gMjsgICAgIC8qIE1TRiBhZGRyZXNzaW5nICovCglyZXEuY21kWzJdID0gMHg0MDsgIC8qIHJlcXVlc3Qgc3ViUSBkYXRhICovCglyZXEuY21kWzNdID0gZm9ybWF0OwoJcmVxLmNtZFs3XSA9IChidWZsZW4gPj4gOCk7CglyZXEuY21kWzhdID0gKGJ1ZmxlbiAmIDB4ZmYpOwoJcmV0dXJuIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKfQoKLyogQVRBUEkgY2Ryb20gZHJpdmVzIGFyZSBmcmVlIHRvIHNlbGVjdCB0aGUgc3BlZWQgeW91IHJlcXVlc3Qgb3IgYW55IHNsb3dlcgogICByYXRlIDotKCBSZXF1ZXN0aW5nIHRvbyBmYXN0IGEgc3BlZWQgd2lsbCBfbm90XyBwcm9kdWNlIGFuIGVycm9yLiAqLwpzdGF0aWMgaW50IGNkcm9tX3NlbGVjdF9zcGVlZChpZGVfZHJpdmVfdCAqZHJpdmUsIGludCBzcGVlZCwKCQkJICAgICAgc3RydWN0IHJlcXVlc3Rfc2Vuc2UgKnNlbnNlKQp7CglzdHJ1Y3QgcmVxdWVzdCByZXE7CgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsICZyZXEpOwoKCXJlcS5zZW5zZSA9IHNlbnNlOwoJaWYgKHNwZWVkID09IDApCgkJc3BlZWQgPSAweGZmZmY7IC8qIHNldCB0byBtYXggKi8KCWVsc2UKCQlzcGVlZCAqPSAxNzc7ICAgLyogTnggdG8ga2J5dGVzL3MgKi8KCglyZXEuY21kWzBdID0gR1BDTURfU0VUX1NQRUVEOwoJLyogUmVhZCBEcml2ZSBzcGVlZCBpbiBrYnl0ZXMvc2Vjb25kIE1TQiAqLwoJcmVxLmNtZFsyXSA9IChzcGVlZCA+PiA4KSAmIDB4ZmY7CQoJLyogUmVhZCBEcml2ZSBzcGVlZCBpbiBrYnl0ZXMvc2Vjb25kIExTQiAqLwoJcmVxLmNtZFszXSA9IHNwZWVkICYgMHhmZjsKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yIHx8CgkgICAgQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2RfcncgfHwKCSAgICBDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcikgewoJCS8qIFdyaXRlIERyaXZlIHNwZWVkIGluIGtieXRlcy9zZWNvbmQgTVNCICovCgkJcmVxLmNtZFs0XSA9IChzcGVlZCA+PiA4KSAmIDB4ZmY7CgkJLyogV3JpdGUgRHJpdmUgc3BlZWQgaW4ga2J5dGVzL3NlY29uZCBMU0IgKi8KCQlyZXEuY21kWzVdID0gc3BlZWQgJiAweGZmOwogICAgICAgfQoKCXJldHVybiBjZHJvbV9xdWV1ZV9wYWNrZXRfY29tbWFuZChkcml2ZSwgJnJlcSk7Cn0KCnN0YXRpYyBpbnQgY2Ryb21fcGxheV9hdWRpbyhpZGVfZHJpdmVfdCAqZHJpdmUsIGludCBsYmFfc3RhcnQsIGludCBsYmFfZW5kKQp7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsICZyZXEpOwoKCXJlcS5zZW5zZSA9ICZzZW5zZTsKCXJlcS5jbWRbMF0gPSBHUENNRF9QTEFZX0FVRElPX01TRjsKCWxiYV90b19tc2YobGJhX3N0YXJ0LCAmcmVxLmNtZFszXSwgJnJlcS5jbWRbNF0sICZyZXEuY21kWzVdKTsKCWxiYV90b19tc2YobGJhX2VuZC0xLCAmcmVxLmNtZFs2XSwgJnJlcS5jbWRbN10sICZyZXEuY21kWzhdKTsKCglyZXR1cm4gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwp9CgpzdGF0aWMgaW50IGNkcm9tX2dldF90b2NfZW50cnkoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgdHJhY2ssCgkJCQlzdHJ1Y3QgYXRhcGlfdG9jX2VudHJ5ICoqZW50KQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBhdGFwaV90b2MgKnRvYyA9IGluZm8tPnRvYzsKCWludCBudHJhY2tzOwoKCS8qCgkgKiBkb24ndCBzZXJ2ZSBjYWNoZWQgZGF0YSwgaWYgdGhlIHRvYyBpc24ndCB2YWxpZAoJICovCglpZiAoIUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkKQoJCXJldHVybiAtRUlOVkFMOwoKCS8qIENoZWNrIHZhbGlkaXR5IG9mIHJlcXVlc3RlZCB0cmFjayBudW1iZXIuICovCgludHJhY2tzID0gdG9jLT5oZHIubGFzdF90cmFjayAtIHRvYy0+aGRyLmZpcnN0X3RyYWNrICsgMTsKCWlmICh0b2MtPmhkci5maXJzdF90cmFjayA9PSBDRFJPTV9MRUFET1VUKSBudHJhY2tzID0gMDsKCWlmICh0cmFjayA9PSBDRFJPTV9MRUFET1VUKQoJCSplbnQgPSAmdG9jLT5lbnRbbnRyYWNrc107CgllbHNlIGlmICh0cmFjayA8IHRvYy0+aGRyLmZpcnN0X3RyYWNrIHx8CgkJIHRyYWNrID4gdG9jLT5oZHIubGFzdF90cmFjaykKCQlyZXR1cm4gLUVJTlZBTDsKCWVsc2UKCQkqZW50ID0gJnRvYy0+ZW50W3RyYWNrIC0gdG9jLT5oZHIuZmlyc3RfdHJhY2tdOwoKCXJldHVybiAwOwp9CgovKiB0aGUgZ2VuZXJpYyBwYWNrZXQgaW50ZXJmYWNlIHRvIGNkcm9tLmMgKi8Kc3RhdGljIGludCBpZGVfY2Ryb21fcGFja2V0KHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLAoJCQkgICAgc3RydWN0IHBhY2tldF9jb21tYW5kICpjZ2MpCnsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoKCWlmIChjZ2MtPnRpbWVvdXQgPD0gMCkKCQljZ2MtPnRpbWVvdXQgPSBBVEFQSV9XQUlUX1BDOwoKCS8qIGhlcmUgd2UgcXVldWUgdGhlIGNvbW1hbmRzIGZyb20gdGhlIHVuaWZvcm0gQ0QtUk9NCgkgICBsYXllci4gdGhlIHBhY2tldCBtdXN0IGJlIGNvbXBsZXRlLCBhcyB3ZSBkbyBub3QKCSAgIHRvdWNoIGl0IGF0IGFsbC4gKi8KCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgltZW1jcHkocmVxLmNtZCwgY2djLT5jbWQsIENEUk9NX1BBQ0tFVF9TSVpFKTsKCWlmIChjZ2MtPnNlbnNlKQoJCW1lbXNldChjZ2MtPnNlbnNlLCAwLCBzaXplb2Yoc3RydWN0IHJlcXVlc3Rfc2Vuc2UpKTsKCXJlcS5kYXRhID0gY2djLT5idWZmZXI7CglyZXEuZGF0YV9sZW4gPSBjZ2MtPmJ1ZmxlbjsKCXJlcS50aW1lb3V0ID0gY2djLT50aW1lb3V0OwoKCWlmIChjZ2MtPnF1aWV0KQoJCXJlcS5jbWRfZmxhZ3MgfD0gUkVRX1FVSUVUOwoKCXJlcS5zZW5zZSA9IGNnYy0+c2Vuc2U7CgljZ2MtPnN0YXQgPSBjZHJvbV9xdWV1ZV9wYWNrZXRfY29tbWFuZChkcml2ZSwgJnJlcSk7CglpZiAoIWNnYy0+c3RhdCkKCQljZ2MtPmJ1ZmxlbiAtPSByZXEuZGF0YV9sZW47CglyZXR1cm4gY2djLT5zdGF0Owp9CgpzdGF0aWMKaW50IGlkZV9jZHJvbV9hdWRpb19pb2N0bCAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksCgkJCSAgIHVuc2lnbmVkIGludCBjbWQsIHZvaWQgKmFyZykKCQkJICAgCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglpbnQgc3RhdDsKCglzd2l0Y2ggKGNtZCkgewoJLyoKCSAqIGVtdWxhdGUgUExBWV9BVURJT19USSBjb21tYW5kIHdpdGggUExBWV9BVURJT18xMCwgc2luY2UKCSAqIGF0YXBpIGRvZXNuJ3Qgc3VwcG9ydCBpdAoJICovCgljYXNlIENEUk9NUExBWVRSS0lORDogewoJCXVuc2lnbmVkIGxvbmcgbGJhX3N0YXJ0LCBsYmFfZW5kOwoJCXN0cnVjdCBjZHJvbV90aSAqdGkgPSBhcmc7CgkJc3RydWN0IGF0YXBpX3RvY19lbnRyeSAqZmlyc3RfdG9jLCAqbGFzdF90b2M7CgoJCXN0YXQgPSBjZHJvbV9nZXRfdG9jX2VudHJ5KGRyaXZlLCB0aS0+Y2R0aV90cmswLCAmZmlyc3RfdG9jKTsKCQlpZiAoc3RhdCkKCQkJcmV0dXJuIHN0YXQ7CgoJCXN0YXQgPSBjZHJvbV9nZXRfdG9jX2VudHJ5KGRyaXZlLCB0aS0+Y2R0aV90cmsxLCAmbGFzdF90b2MpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCgkJaWYgKHRpLT5jZHRpX3RyazEgIT0gQ0RST01fTEVBRE9VVCkKCQkJKytsYXN0X3RvYzsKCQlsYmFfc3RhcnQgPSBmaXJzdF90b2MtPmFkZHIubGJhOwoJCWxiYV9lbmQgICA9IGxhc3RfdG9jLT5hZGRyLmxiYTsKCgkJaWYgKGxiYV9lbmQgPD0gbGJhX3N0YXJ0KQoJCQlyZXR1cm4gLUVJTlZBTDsKCgkJcmV0dXJuIGNkcm9tX3BsYXlfYXVkaW8oZHJpdmUsIGxiYV9zdGFydCwgbGJhX2VuZCk7Cgl9CgoJY2FzZSBDRFJPTVJFQURUT0NIRFI6IHsKCQlzdHJ1Y3QgY2Ryb21fdG9jaGRyICp0b2NoZHIgPSBhcmc7CgkJc3RydWN0IGF0YXBpX3RvYyAqdG9jOwoKCQkvKiBNYWtlIHN1cmUgb3VyIHNhdmVkIFRPQyBpcyB2YWxpZC4gKi8KCQlzdGF0ID0gY2Ryb21fcmVhZF90b2MoZHJpdmUsIE5VTEwpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCgkJdG9jID0gaW5mby0+dG9jOwoJCXRvY2hkci0+Y2R0aF90cmswID0gdG9jLT5oZHIuZmlyc3RfdHJhY2s7CgkJdG9jaGRyLT5jZHRoX3RyazEgPSB0b2MtPmhkci5sYXN0X3RyYWNrOwoKCQlyZXR1cm4gMDsKCX0KCgljYXNlIENEUk9NUkVBRFRPQ0VOVFJZOiB7CgkJc3RydWN0IGNkcm9tX3RvY2VudHJ5ICp0b2NlbnRyeSA9IGFyZzsKCQlzdHJ1Y3QgYXRhcGlfdG9jX2VudHJ5ICp0b2NlOwoKCQlzdGF0ID0gY2Ryb21fZ2V0X3RvY19lbnRyeShkcml2ZSwgdG9jZW50cnktPmNkdGVfdHJhY2ssICZ0b2NlKTsKCQlpZiAoc3RhdCkKCQkJcmV0dXJuIHN0YXQ7CgoJCXRvY2VudHJ5LT5jZHRlX2N0cmwgPSB0b2NlLT5jb250cm9sOwoJCXRvY2VudHJ5LT5jZHRlX2FkciAgPSB0b2NlLT5hZHI7CgkJaWYgKHRvY2VudHJ5LT5jZHRlX2Zvcm1hdCA9PSBDRFJPTV9NU0YpIHsKCQkJbGJhX3RvX21zZiAodG9jZS0+YWRkci5sYmEsCgkJCQkgICAmdG9jZW50cnktPmNkdGVfYWRkci5tc2YubWludXRlLAoJCQkJICAgJnRvY2VudHJ5LT5jZHRlX2FkZHIubXNmLnNlY29uZCwKCQkJCSAgICZ0b2NlbnRyeS0+Y2R0ZV9hZGRyLm1zZi5mcmFtZSk7CgkJfSBlbHNlCgkJCXRvY2VudHJ5LT5jZHRlX2FkZHIubGJhID0gdG9jZS0+YWRkci5sYmE7CgoJCXJldHVybiAwOwoJfQoKCWRlZmF1bHQ6CgkJcmV0dXJuIC1FSU5WQUw7Cgl9Cn0KCnN0YXRpYwppbnQgaWRlX2Nkcm9tX3Jlc2V0IChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSkKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCWludCByZXQ7CgoJY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGRyaXZlLCAmcmVxKTsKCXJlcS5jbWRfdHlwZSA9IFJFUV9UWVBFX1NQRUNJQUw7CglyZXEuY21kX2ZsYWdzID0gUkVRX1FVSUVUOwoJcmV0ID0gaWRlX2RvX2RyaXZlX2NtZChkcml2ZSwgJnJlcSwgaWRlX3dhaXQpOwoKCS8qCgkgKiBBIHJlc2V0IHdpbGwgdW5sb2NrIHRoZSBkb29yLiBJZiBpdCB3YXMgcHJldmlvdXNseSBsb2NrZWQsCgkgKiBsb2NrIGl0IGFnYWluLgoJICovCglpZiAoQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT5kb29yX2xvY2tlZCkKCQkodm9pZCkgY2Ryb21fbG9ja2Rvb3IoZHJpdmUsIDEsICZzZW5zZSk7CgoJcmV0dXJuIHJldDsKfQoKCnN0YXRpYwppbnQgaWRlX2Nkcm9tX3RyYXlfbW92ZSAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksIGludCBwb3NpdGlvbikKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCglpZiAocG9zaXRpb24pIHsKCQlpbnQgc3RhdCA9IGNkcm9tX2xvY2tkb29yKGRyaXZlLCAwLCAmc2Vuc2UpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCX0KCglyZXR1cm4gY2Ryb21fZWplY3QoZHJpdmUsICFwb3NpdGlvbiwgJnNlbnNlKTsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fbG9ja19kb29yIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwgaW50IGxvY2spCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoJcmV0dXJuIGNkcm9tX2xvY2tkb29yKGRyaXZlLCBsb2NrLCBOVUxMKTsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fZ2V0X2NhcGFiaWxpdGllcyhpZGVfZHJpdmVfdCAqZHJpdmUsIHN0cnVjdCBhdGFwaV9jYXBhYmlsaXRpZXNfcGFnZSAqY2FwKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpID0gJmluZm8tPmRldmluZm87CglzdHJ1Y3QgcGFja2V0X2NvbW1hbmQgY2djOwoJaW50IHN0YXQsIGF0dGVtcHRzID0gMywgc2l6ZSA9IHNpemVvZigqY2FwKTsKCgkvKgoJICogQUNFUjUwIChhbmQgb3RoZXJzPykgcmVxdWlyZSB0aGUgZnVsbCBzcGVjIGxlbmd0aCBtb2RlIHNlbnNlCgkgKiBwYWdlIGNhcGFiaWxpdGllcyBzaXplLCBidXQgb2xkZXIgZHJpdmVzIGJyZWFrLgoJICovCglpZiAoISghc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJBVEFQSSBDRCBST00gRFJJVkUgNTBYIE1BWCIpIHx8CgkgICAgIXN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiV1BJIENEUy0zMlgiKSkpCgkJc2l6ZSAtPSBzaXplb2YoY2FwLT5wYWQpOwoKCWluaXRfY2Ryb21fY29tbWFuZCgmY2djLCBjYXAsIHNpemUsIENHQ19EQVRBX1VOS05PV04pOwoJZG8geyAvKiB3ZSBzZWVtIHRvIGdldCBzdGF0PTB4MDEsZXJyPTB4MDAgdGhlIGZpcnN0IHRpbWUgKD8/KSAqLwoJCXN0YXQgPSBjZHJvbV9tb2RlX3NlbnNlKGNkaSwgJmNnYywgR1BNT0RFX0NBUEFCSUxJVElFU19QQUdFLCAwKTsKCQlpZiAoIXN0YXQpCgkJCWJyZWFrOwoJfSB3aGlsZSAoLS1hdHRlbXB0cyk7CglyZXR1cm4gc3RhdDsKfQoKc3RhdGljCnZvaWQgaWRlX2Nkcm9tX3VwZGF0ZV9zcGVlZCAoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgYXRhcGlfY2FwYWJpbGl0aWVzX3BhZ2UgKmNhcCkKewoJLyogVGhlIEFDRVIvQU9wZW4gMjRYIGNkcm9tIGhhcyB0aGUgc3BlZWQgZmllbGRzIGJ5dGUtc3dhcHBlZCAqLwoJaWYgKCFkcml2ZS0+aWQtPm1vZGVsWzBdICYmCgkgICAgIXN0cm5jbXAoZHJpdmUtPmlkLT5md19yZXYsICIyNDFOIiwgNCkpIHsKCQlDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmN1cnJlbnRfc3BlZWQgID0KCQkJKCgodW5zaWduZWQgaW50KWNhcC0+Y3Vyc3BlZWQpICsgKDE3Ni8yKSkgLyAxNzY7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bWF4X3NwZWVkID0KCQkJKCgodW5zaWduZWQgaW50KWNhcC0+bWF4c3BlZWQpICsgKDE3Ni8yKSkgLyAxNzY7Cgl9IGVsc2UgewoJCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+Y3VycmVudF9zcGVlZCAgPQoJCQkobnRvaHMoY2FwLT5jdXJzcGVlZCkgKyAoMTc2LzIpKSAvIDE3NjsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5tYXhfc3BlZWQgPQoJCQkobnRvaHMoY2FwLT5tYXhzcGVlZCkgKyAoMTc2LzIpKSAvIDE3NjsKCX0KfQoKc3RhdGljCmludCBpZGVfY2Ryb21fc2VsZWN0X3NwZWVkIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwgaW50IHNwZWVkKQp7CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBjZGktPmhhbmRsZTsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIHNlbnNlOwoJc3RydWN0IGF0YXBpX2NhcGFiaWxpdGllc19wYWdlIGNhcDsKCWludCBzdGF0OwoKCWlmICgoc3RhdCA9IGNkcm9tX3NlbGVjdF9zcGVlZChkcml2ZSwgc3BlZWQsICZzZW5zZSkpIDwgMCkKCQlyZXR1cm4gc3RhdDsKCglpZiAoIWlkZV9jZHJvbV9nZXRfY2FwYWJpbGl0aWVzKGRyaXZlLCAmY2FwKSkgewoJCWlkZV9jZHJvbV91cGRhdGVfc3BlZWQoZHJpdmUsICZjYXApOwoJCWNkaS0+c3BlZWQgPSBDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmN1cnJlbnRfc3BlZWQ7Cgl9CiAgICAgICAgcmV0dXJuIDA7Cn0KCi8qCiAqIGFkZCBsb2dpYyB0byB0cnkgR0VUX0VWRU5UIGNvbW1hbmQgZmlyc3QgdG8gY2hlY2sgZm9yIG1lZGlhIGFuZCB0cmF5CiAqIHN0YXR1cy4gdGhpcyBzaG91bGQgYmUgc3VwcG9ydGVkIGJ5IG5ld2VyIGNkLXIvdyBhbmQgYWxsIERWRCBldGMKICogZHJpdmVzCiAqLwpzdGF0aWMKaW50IGlkZV9jZHJvbV9kcml2ZV9zdGF0dXMgKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLCBpbnQgc2xvdF9ucikKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CglzdHJ1Y3QgbWVkaWFfZXZlbnRfZGVzYyBtZWQ7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCWludCBzdGF0OwoKCWlmIChzbG90X25yICE9IENEU0xfQ1VSUkVOVCkKCQlyZXR1cm4gLUVJTlZBTDsKCglzdGF0ID0gY2Ryb21fY2hlY2tfc3RhdHVzKGRyaXZlLCAmc2Vuc2UpOwoJaWYgKCFzdGF0IHx8IHNlbnNlLnNlbnNlX2tleSA9PSBVTklUX0FUVEVOVElPTikKCQlyZXR1cm4gQ0RTX0RJU0NfT0s7CgoJaWYgKCFjZHJvbV9nZXRfbWVkaWFfZXZlbnQoY2RpLCAmbWVkKSkgewoJCWlmIChtZWQubWVkaWFfcHJlc2VudCkKCQkJcmV0dXJuIENEU19ESVNDX09LOwoJCWVsc2UgaWYgKG1lZC5kb29yX29wZW4pCgkJCXJldHVybiBDRFNfVFJBWV9PUEVOOwoJCWVsc2UKCQkJcmV0dXJuIENEU19OT19ESVNDOwoJfQoKCWlmIChzZW5zZS5zZW5zZV9rZXkgPT0gTk9UX1JFQURZICYmIHNlbnNlLmFzYyA9PSAweDA0ICYmIHNlbnNlLmFzY3EgPT0gMHgwNCkKCQlyZXR1cm4gQ0RTX0RJU0NfT0s7CgoJLyoKCSAqIElmIG5vdCB1c2luZyBNdCBGdWppIGV4dGVuZGVkIG1lZGlhIHRyYXkgcmVwb3J0cywKCSAqIGp1c3QgcmV0dXJuIFRSQVlfT1BFTiBzaW5jZSBBVEFQSSBkb2Vzbid0IHByb3ZpZGUKCSAqIGFueSBvdGhlciB3YXkgdG8gZGV0ZWN0IHRoaXMuLi4KCSAqLwoJaWYgKHNlbnNlLnNlbnNlX2tleSA9PSBOT1RfUkVBRFkpIHsKCQlpZiAoc2Vuc2UuYXNjID09IDB4M2EgJiYgc2Vuc2UuYXNjcSA9PSAxKQoJCQlyZXR1cm4gQ0RTX05PX0RJU0M7CgkJZWxzZQoJCQlyZXR1cm4gQ0RTX1RSQVlfT1BFTjsKCX0KCXJldHVybiBDRFNfRFJJVkVfTk9UX1JFQURZOwp9CgpzdGF0aWMKaW50IGlkZV9jZHJvbV9nZXRfbGFzdF9zZXNzaW9uIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwKCQkJCXN0cnVjdCBjZHJvbV9tdWx0aXNlc3Npb24gKm1zX2luZm8pCnsKCXN0cnVjdCBhdGFwaV90b2MgKnRvYzsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCWludCByZXQ7CgoJaWYgKCFDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPnRvY192YWxpZCB8fCBpbmZvLT50b2MgPT0gTlVMTCkKCQlpZiAoKHJldCA9IGNkcm9tX3JlYWRfdG9jKGRyaXZlLCAmc2Vuc2UpKSkKCQkJcmV0dXJuIHJldDsKCgl0b2MgPSBpbmZvLT50b2M7Cgltc19pbmZvLT5hZGRyLmxiYSA9IHRvYy0+bGFzdF9zZXNzaW9uX2xiYTsKCW1zX2luZm8tPnhhX2ZsYWcgPSB0b2MtPnhhX2ZsYWc7CgoJcmV0dXJuIDA7Cn0KCnN0YXRpYwppbnQgaWRlX2Nkcm9tX2dldF9tY24gKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLAoJCSAgICAgICBzdHJ1Y3QgY2Ryb21fbWNuICptY25faW5mbykKewoJaW50IHN0YXQ7CgljaGFyIG1jbmJ1ZlsyNF07CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBjZGktPmhhbmRsZTsKCi8qIGdldCBNQ04gKi8KCWlmICgoc3RhdCA9IGNkcm9tX3JlYWRfc3ViY2hhbm5lbChkcml2ZSwgMiwgbWNuYnVmLCBzaXplb2YgKG1jbmJ1ZiksIE5VTEwpKSkKCQlyZXR1cm4gc3RhdDsKCgltZW1jcHkgKG1jbl9pbmZvLT5tZWRpdW1fY2F0YWxvZ19udW1iZXIsIG1jbmJ1Zis5LAoJCXNpemVvZiAobWNuX2luZm8tPm1lZGl1bV9jYXRhbG9nX251bWJlciktMSk7CgltY25faW5mby0+bWVkaXVtX2NhdGFsb2dfbnVtYmVyW3NpemVvZiAobWNuX2luZm8tPm1lZGl1bV9jYXRhbG9nX251bWJlciktMV0KCQk9ICdcMCc7CgoJcmV0dXJuIDA7Cn0KCgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogT3RoZXIgZHJpdmVyIHJlcXVlc3RzIChvcGVuLCBjbG9zZSwgY2hlY2sgbWVkaWEgY2hhbmdlKS4KICovCgpzdGF0aWMKaW50IGlkZV9jZHJvbV9jaGVja19tZWRpYV9jaGFuZ2VfcmVhbCAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksCgkJCQkgICAgICAgaW50IHNsb3RfbnIpCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoJaW50IHJldHZhbDsKCQoJaWYgKHNsb3RfbnIgPT0gQ0RTTF9DVVJSRU5UKSB7CgkJKHZvaWQpIGNkcm9tX2NoZWNrX3N0YXR1cyhkcml2ZSwgTlVMTCk7CgkJcmV0dmFsID0gQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT5tZWRpYV9jaGFuZ2VkOwoJCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+bWVkaWFfY2hhbmdlZCA9IDA7CgkJcmV0dXJuIHJldHZhbDsKCX0gZWxzZSB7CgkJcmV0dXJuIC1FSU5WQUw7Cgl9Cn0KCgpzdGF0aWMKaW50IGlkZV9jZHJvbV9vcGVuX3JlYWwgKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLCBpbnQgcHVycG9zZSkKewoJcmV0dXJuIDA7Cn0KCi8qCiAqIENsb3NlIGRvd24gdGhlIGRldmljZS4gIEludmFsaWRhdGUgYWxsIGNhY2hlZCBibG9ja3MuCiAqLwoKc3RhdGljCnZvaWQgaWRlX2Nkcm9tX3JlbGVhc2VfcmVhbCAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGkpCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoKCWlmICghY2RpLT51c2VfY291bnQpCgkJQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT50b2NfdmFsaWQgPSAwOwp9CgoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIERldmljZSBpbml0aWFsaXphdGlvbi4KICovCnN0YXRpYyBzdHJ1Y3QgY2Ryb21fZGV2aWNlX29wcyBpZGVfY2Ryb21fZG9wcyA9IHsKCS5vcGVuCQkJPSBpZGVfY2Ryb21fb3Blbl9yZWFsLAoJLnJlbGVhc2UJCT0gaWRlX2Nkcm9tX3JlbGVhc2VfcmVhbCwKCS5kcml2ZV9zdGF0dXMJCT0gaWRlX2Nkcm9tX2RyaXZlX3N0YXR1cywKCS5tZWRpYV9jaGFuZ2VkCQk9IGlkZV9jZHJvbV9jaGVja19tZWRpYV9jaGFuZ2VfcmVhbCwKCS50cmF5X21vdmUJCT0gaWRlX2Nkcm9tX3RyYXlfbW92ZSwKCS5sb2NrX2Rvb3IJCT0gaWRlX2Nkcm9tX2xvY2tfZG9vciwKCS5zZWxlY3Rfc3BlZWQJCT0gaWRlX2Nkcm9tX3NlbGVjdF9zcGVlZCwKCS5nZXRfbGFzdF9zZXNzaW9uCT0gaWRlX2Nkcm9tX2dldF9sYXN0X3Nlc3Npb24sCgkuZ2V0X21jbgkJPSBpZGVfY2Ryb21fZ2V0X21jbiwKCS5yZXNldAkJCT0gaWRlX2Nkcm9tX3Jlc2V0LAoJLmF1ZGlvX2lvY3RsCQk9IGlkZV9jZHJvbV9hdWRpb19pb2N0bCwKCS5jYXBhYmlsaXR5CQk9IENEQ19DTE9TRV9UUkFZIHwgQ0RDX09QRU5fVFJBWSB8IENEQ19MT0NLIHwKCQkJCUNEQ19TRUxFQ1RfU1BFRUQgfCBDRENfU0VMRUNUX0RJU0MgfAoJCQkJQ0RDX01VTFRJX1NFU1NJT04gfCBDRENfTUNOIHwKCQkJCUNEQ19NRURJQV9DSEFOR0VEIHwgQ0RDX1BMQVlfQVVESU8gfCBDRENfUkVTRVQgfAoJCQkJQ0RDX0RSSVZFX1NUQVRVUyB8IENEQ19DRF9SIHwKCQkJCUNEQ19DRF9SVyB8IENEQ19EVkQgfCBDRENfRFZEX1J8IENEQ19EVkRfUkFNIHwKCQkJCUNEQ19HRU5FUklDX1BBQ0tFVCB8IENEQ19NT19EUklWRSB8IENEQ19NUlcgfAoJCQkJQ0RDX01SV19XIHwgQ0RDX1JBTSwKCS5nZW5lcmljX3BhY2tldAkJPSBpZGVfY2Ryb21fcGFja2V0LAp9OwoKc3RhdGljIGludCBpZGVfY2Ryb21fcmVnaXN0ZXIgKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IG5zbG90cykKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmRldmluZm8gPSAmaW5mby0+ZGV2aW5mbzsKCglkZXZpbmZvLT5vcHMgPSAmaWRlX2Nkcm9tX2RvcHM7CglkZXZpbmZvLT5tYXNrID0gMDsKCWRldmluZm8tPnNwZWVkID0gQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT5jdXJyZW50X3NwZWVkOwoJZGV2aW5mby0+Y2FwYWNpdHkgPSBuc2xvdHM7CglkZXZpbmZvLT5oYW5kbGUgPSBkcml2ZTsKCXN0cmNweShkZXZpbmZvLT5uYW1lLCBkcml2ZS0+bmFtZSk7CgkKCS8qIHNldCBjYXBhYmlsaXR5IG1hc2sgdG8gbWF0Y2ggdGhlIHByb2JlLiAqLwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yKQoJCWRldmluZm8tPm1hc2sgfD0gQ0RDX0NEX1I7CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3J3KQoJCWRldmluZm8tPm1hc2sgfD0gQ0RDX0NEX1JXOwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmQpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfRFZEOwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcikKCQlkZXZpbmZvLT5tYXNrIHw9IENEQ19EVkRfUjsKCWlmICghQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+ZHZkX3JhbSkKCQlkZXZpbmZvLT5tYXNrIHw9IENEQ19EVkRfUkFNOwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5pc19jaGFuZ2VyKQoJCWRldmluZm8tPm1hc2sgfD0gQ0RDX1NFTEVDVF9ESVNDOwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5hdWRpb19wbGF5KQoJCWRldmluZm8tPm1hc2sgfD0gQ0RDX1BMQVlfQVVESU87CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNsb3NlX3RyYXkpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfQ0xPU0VfVFJBWTsKCWlmICghQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bW9fZHJpdmUpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfTU9fRFJJVkU7CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnJhbSkKCQlkZXZpbmZvLT5tYXNrIHw9IENEQ19SQU07CgoJZGV2aW5mby0+ZGlzayA9IGluZm8tPmRpc2s7CglyZXR1cm4gcmVnaXN0ZXJfY2Ryb20oZGV2aW5mbyk7Cn0KCnN0YXRpYwppbnQgaWRlX2Nkcm9tX3Byb2JlX2NhcGFiaWxpdGllcyAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpID0gJmluZm8tPmRldmluZm87CglzdHJ1Y3QgYXRhcGlfY2FwYWJpbGl0aWVzX3BhZ2UgY2FwOwoJaW50IG5zbG90cyA9IDE7CgoJaWYgKGRyaXZlLT5tZWRpYSA9PSBpZGVfb3B0aWNhbCkgewoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm1vX2RyaXZlID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5yYW0gPSAxOwoJCXByaW50ayhLRVJOX0VSUiAiJXM6IEFUQVBJIG1hZ25ldG8tb3B0aWNhbCBkcml2ZVxuIiwgZHJpdmUtPm5hbWUpOwoJCXJldHVybiBuc2xvdHM7Cgl9CgoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5lYzI2MCB8fAoJICAgICFzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwiU1RJTkdSQVkgODQyMiBJREUgOFggQ0QtUk9NIDctMjctOTUiKSkgewoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2VqZWN0ID0gMDsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5hdWRpb19wbGF5ID0gMTsKCQlyZXR1cm4gbnNsb3RzOwoJfQoKCS8qCgkgKiB3ZSBoYXZlIHRvIGNoZWF0IGEgbGl0dGxlIGhlcmUuIHRoZSBwYWNrZXQgd2lsbCBldmVudHVhbGx5CgkgKiBiZSBxdWV1ZWQgd2l0aCBpZGVfY2Ryb21fcGFja2V0KCksIHdoaWNoIGV4dHJhY3RzIHRoZQoJICogZHJpdmUgZnJvbSBjZGktPmhhbmRsZS4gU2luY2UgdGhpcyBkZXZpY2UgaGFzbid0IGJlZW4KCSAqIHJlZ2lzdGVyZWQgd2l0aCB0aGUgVW5pZm9ybSBsYXllciB5ZXQsIGl0IGNhbid0IGRvIHRoaXMuCgkgKiBTYW1lIGdvZXMgZm9yIGNkaS0+b3BzLgoJICovCgljZGktPmhhbmRsZSA9IGRyaXZlOwoJY2RpLT5vcHMgPSAmaWRlX2Nkcm9tX2RvcHM7CgoJaWYgKGlkZV9jZHJvbV9nZXRfY2FwYWJpbGl0aWVzKGRyaXZlLCAmY2FwKSkKCQlyZXR1cm4gMDsKCglpZiAoY2FwLmxvY2sgPT0gMCkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5ub19kb29ybG9jayA9IDE7CglpZiAoY2FwLmVqZWN0KQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2VqZWN0ID0gMDsKCWlmIChjYXAuY2Rfcl93cml0ZSkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yID0gMTsKCWlmIChjYXAuY2Rfcndfd3JpdGUpIHsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9ydyA9IDE7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+cmFtID0gMTsKCX0KCWlmIChjYXAudGVzdF93cml0ZSkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50ZXN0X3dyaXRlID0gMTsKCWlmIChjYXAuZHZkX3JhbV9yZWFkIHx8IGNhcC5kdmRfcl9yZWFkIHx8IGNhcC5kdmRfcm9tKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZCA9IDE7CglpZiAoY2FwLmR2ZF9yYW1fd3JpdGUpIHsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcmFtID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5yYW0gPSAxOwoJfQoJaWYgKGNhcC5kdmRfcl93cml0ZSkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfciA9IDE7CglpZiAoY2FwLmF1ZGlvX3BsYXkpCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+YXVkaW9fcGxheSA9IDE7CglpZiAoY2FwLm1lY2h0eXBlID09IG1lY2h0eXBlX2NhZGR5IHx8IGNhcC5tZWNodHlwZSA9PSBtZWNodHlwZV9wb3B1cCkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jbG9zZV90cmF5ID0gMDsKCgkvKiBTb21lIGRyaXZlcyB1c2VkIGJ5IEFwcGxlIGRvbid0IGFkdmVydGlzZSBhdWRpbyBwbGF5CgkgKiBidXQgdGhleSBkbyBzdXBwb3J0IHJlYWRpbmcgVE9DICYgYXVkaW8gZGF0YXMKCSAqLwoJaWYgKHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiTUFUU0hJVEFEVkQtUk9NIFNSLTgxODciKSA9PSAwIHx8CgkgICAgc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJNQVRTSElUQURWRC1ST00gU1ItODE4NiIpID09IDAgfHwKCSAgICBzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIk1BVFNISVRBRFZELVJPTSBTUi04MTc2IikgPT0gMCB8fAoJICAgIHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiTUFUU0hJVEFEVkQtUk9NIFNSLTgxNzQiKSA9PSAwKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmF1ZGlvX3BsYXkgPSAxOwoKI2lmICEgU1RBTkRBUkRfQVRBUEkKCWlmIChjZGktPnNhbnlvX3Nsb3QgPiAwKSB7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+aXNfY2hhbmdlciA9IDE7CgkJbnNsb3RzID0gMzsKCX0KCgllbHNlCiNlbmRpZiAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCWlmIChjYXAubWVjaHR5cGUgPT0gbWVjaHR5cGVfaW5kaXZpZHVhbF9jaGFuZ2VyIHx8CgkgICAgY2FwLm1lY2h0eXBlID09IG1lY2h0eXBlX2NhcnRyaWRnZV9jaGFuZ2VyKSB7CgkJaWYgKChuc2xvdHMgPSBjZHJvbV9udW1iZXJfb2Zfc2xvdHMoY2RpKSkgPiAxKSB7CgkJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmlzX2NoYW5nZXIgPSAxOwoJCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zdXBwX2Rpc2NfcHJlc2VudCA9IDE7CgkJfQoJfQoKCWlkZV9jZHJvbV91cGRhdGVfc3BlZWQoZHJpdmUsICZjYXApOwoJLyogZG9uJ3QgcHJpbnQgc3BlZWQgaWYgdGhlIGRyaXZlIHJlcG9ydGVkIDAuCgkgKi8KCXByaW50ayhLRVJOX0lORk8gIiVzOiBBVEFQSSIsIGRyaXZlLT5uYW1lKTsKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5tYXhfc3BlZWQpCgkJcHJpbnRrKCIgJWRYIiwgQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bWF4X3NwZWVkKTsKCXByaW50aygiICVzIiwgQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+ZHZkID8gIkRWRC1ST00iIDogIkNELVJPTSIpOwoKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcnxDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcmFtKQogICAgICAgIAlwcmludGsoIiBEVkQlcyVzIiwgCiAgICAgICAgCShDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcik/ICItUiIgOiAiIiwgCiAgICAgICAgCShDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcmFtKT8gIi1SQU0iIDogIiIpOwoKICAgICAgICBpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2RfcnxDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9ydykgCiAgICAgICAgCXByaW50aygiIENEJXMlcyIsIAogICAgICAgIAkoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2Rfcik/ICItUiIgOiAiIiwgCiAgICAgICAgCShDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9ydyk/ICIvUlciIDogIiIpOwoKICAgICAgICBpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+aXNfY2hhbmdlcikgCiAgICAgICAgCXByaW50aygiIGNoYW5nZXIgdy8lZCBzbG90cyIsIG5zbG90cyk7CiAgICAgICAgZWxzZSAJCiAgICAgICAgCXByaW50aygiIGRyaXZlIik7CgoJcHJpbnRrKCIsICVka0IgQ2FjaGUiLCBiZTE2X3RvX2NwdShjYXAuYnVmZmVyX3NpemUpKTsKCglpZiAoZHJpdmUtPnVzaW5nX2RtYSkKCQlpZGVfZG1hX3ZlcmJvc2UoZHJpdmUpOwoKCXByaW50aygiXG4iKTsKCglyZXR1cm4gbnNsb3RzOwp9CgojaWZkZWYgQ09ORklHX0lERV9QUk9DX0ZTCnN0YXRpYyB2b2lkIGlkZV9jZHJvbV9hZGRfc2V0dGluZ3MoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglpZGVfYWRkX3NldHRpbmcoZHJpdmUsICJkc2Nfb3ZlcmxhcCIsIFNFVFRJTkdfUlcsIFRZUEVfQllURSwgMCwgMSwgMSwgMSwgJmRyaXZlLT5kc2Nfb3ZlcmxhcCwgTlVMTCk7Cn0KI2Vsc2UKc3RhdGljIGlubGluZSB2b2lkIGlkZV9jZHJvbV9hZGRfc2V0dGluZ3MoaWRlX2RyaXZlX3QgKmRyaXZlKSB7IDsgfQojZW5kaWYKCi8qCiAqIHN0YW5kYXJkIHByZXBfcnFfZm4gdGhhdCBidWlsZHMgMTAgYnl0ZSBjbWRzCiAqLwpzdGF0aWMgaW50IGlkZV9jZHJvbV9wcmVwX2ZzKHJlcXVlc3RfcXVldWVfdCAqcSwgc3RydWN0IHJlcXVlc3QgKnJxKQp7CglpbnQgaGFyZF9zZWN0ID0gcXVldWVfaGFyZHNlY3Rfc2l6ZShxKTsKCWxvbmcgYmxvY2sgPSAobG9uZylycS0+aGFyZF9zZWN0b3IgLyAoaGFyZF9zZWN0ID4+IDkpOwoJdW5zaWduZWQgbG9uZyBibG9ja3MgPSBycS0+aGFyZF9ucl9zZWN0b3JzIC8gKGhhcmRfc2VjdCA+PiA5KTsKCgltZW1zZXQocnEtPmNtZCwgMCwgc2l6ZW9mKHJxLT5jbWQpKTsKCglpZiAocnFfZGF0YV9kaXIocnEpID09IFJFQUQpCgkJcnEtPmNtZFswXSA9IEdQQ01EX1JFQURfMTA7CgllbHNlCgkJcnEtPmNtZFswXSA9IEdQQ01EX1dSSVRFXzEwOwoKCS8qCgkgKiBmaWxsIGluIGxiYQoJICovCglycS0+Y21kWzJdID0gKGJsb2NrID4+IDI0KSAmIDB4ZmY7CglycS0+Y21kWzNdID0gKGJsb2NrID4+IDE2KSAmIDB4ZmY7CglycS0+Y21kWzRdID0gKGJsb2NrID4+ICA4KSAmIDB4ZmY7CglycS0+Y21kWzVdID0gYmxvY2sgJiAweGZmOwoKCS8qCgkgKiBhbmQgdHJhbnNmZXIgbGVuZ3RoCgkgKi8KCXJxLT5jbWRbN10gPSAoYmxvY2tzID4+IDgpICYgMHhmZjsKCXJxLT5jbWRbOF0gPSBibG9ja3MgJiAweGZmOwoJcnEtPmNtZF9sZW4gPSAxMDsKCXJldHVybiBCTEtQUkVQX09LOwp9CgovKgogKiBNb3N0IG9mIHRoZSBTQ1NJIGNvbW1hbmRzIGFyZSBzdXBwb3J0ZWQgZGlyZWN0bHkgYnkgQVRBUEkgZGV2aWNlcy4KICogVGhpcyB0cmFuc2Zvcm0gaGFuZGxlcyB0aGUgZmV3IGV4Y2VwdGlvbnMuCiAqLwpzdGF0aWMgaW50IGlkZV9jZHJvbV9wcmVwX3BjKHN0cnVjdCByZXF1ZXN0ICpycSkKewoJdTggKmMgPSBycS0+Y21kOwoKCS8qCgkgKiBUcmFuc2Zvcm0gNi1ieXRlIHJlYWQvd3JpdGUgY29tbWFuZHMgdG8gdGhlIDEwLWJ5dGUgdmVyc2lvbgoJICovCglpZiAoY1swXSA9PSBSRUFEXzYgfHwgY1swXSA9PSBXUklURV82KSB7CgkJY1s4XSA9IGNbNF07CgkJY1s1XSA9IGNbM107CgkJY1s0XSA9IGNbMl07CgkJY1szXSA9IGNbMV0gJiAweDFmOwoJCWNbMl0gPSAwOwoJCWNbMV0gJj0gMHhlMDsKCQljWzBdICs9IChSRUFEXzEwIC0gUkVBRF82KTsKCQlycS0+Y21kX2xlbiA9IDEwOwoJCXJldHVybiBCTEtQUkVQX09LOwoJfQoKCS8qCgkgKiBpdCdzIHNpbGx5IHRvIHByZXRlbmQgd2UgdW5kZXJzdGFuZCA2LWJ5dGUgc2Vuc2UgY29tbWFuZHMsIGp1c3QKCSAqIHJlamVjdCB3aXRoIElMTEVHQUxfUkVRVUVTVCBhbmQgdGhlIGNhbGxlciBzaG91bGQgdGFrZSB0aGUKCSAqIGFwcHJvcHJpYXRlIGFjdGlvbgoJICovCglpZiAoY1swXSA9PSBNT0RFX1NFTlNFIHx8IGNbMF0gPT0gTU9ERV9TRUxFQ1QpIHsKCQlycS0+ZXJyb3JzID0gSUxMRUdBTF9SRVFVRVNUOwoJCXJldHVybiBCTEtQUkVQX0tJTEw7Cgl9CgkKCXJldHVybiBCTEtQUkVQX09LOwp9CgpzdGF0aWMgaW50IGlkZV9jZHJvbV9wcmVwX2ZuKHJlcXVlc3RfcXVldWVfdCAqcSwgc3RydWN0IHJlcXVlc3QgKnJxKQp7CglpZiAoYmxrX2ZzX3JlcXVlc3QocnEpKQoJCXJldHVybiBpZGVfY2Ryb21fcHJlcF9mcyhxLCBycSk7CgllbHNlIGlmIChibGtfcGNfcmVxdWVzdChycSkpCgkJcmV0dXJuIGlkZV9jZHJvbV9wcmVwX3BjKHJxKTsKCglyZXR1cm4gMDsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fc2V0dXAgKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSA9ICZpbmZvLT5kZXZpbmZvOwoJaW50IG5zbG90czsKCglibGtfcXVldWVfcHJlcF9ycShkcml2ZS0+cXVldWUsIGlkZV9jZHJvbV9wcmVwX2ZuKTsKCWJsa19xdWV1ZV9kbWFfYWxpZ25tZW50KGRyaXZlLT5xdWV1ZSwgMzEpOwoJZHJpdmUtPnF1ZXVlLT51bnBsdWdfZGVsYXkgPSAoMSAqIEhaKSAvIDEwMDA7CglpZiAoIWRyaXZlLT5xdWV1ZS0+dW5wbHVnX2RlbGF5KQoJCWRyaXZlLT5xdWV1ZS0+dW5wbHVnX2RlbGF5ID0gMTsKCglkcml2ZS0+c3BlY2lhbC5hbGwJPSAwOwoKCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+bWVkaWFfY2hhbmdlZCA9IDE7CglDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPnRvY192YWxpZCAgICAgPSAwOwoJQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT5kb29yX2xvY2tlZCAgID0gMDsKCiNpZiBOT19ET09SX0xPQ0tJTkcKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2Rvb3Jsb2NrID0gMTsKI2Vsc2UKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2Rvb3Jsb2NrID0gMDsKI2VuZGlmCgoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+ZHJxX2ludGVycnVwdCA9ICgoZHJpdmUtPmlkLT5jb25maWcgJiAweDAwNjApID09IDB4MjApOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+aXNfY2hhbmdlciA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3J3ID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRlc3Rfd3JpdGUgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+ZHZkID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yYW0gPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZWplY3QgPSAxOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+c3VwcF9kaXNjX3ByZXNlbnQgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+YXVkaW9fcGxheSA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jbG9zZV90cmF5ID0gMTsKCQoJLyogbGltaXQgdHJhbnNmZXIgc2l6ZSBwZXIgaW50ZXJydXB0LiAqLwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bGltaXRfbmZyYW1lcyA9IDA7CgkvKiBhIHRlc3RhbWVudCB0byB0aGUgbmljZSBxdWFsaXR5IG9mIFNhbXN1bmcgZHJpdmVzLi4uICovCglpZiAoIXN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiU0FNU1VORyBDRC1ST00gU0NSLTI0MzAiKSkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5saW1pdF9uZnJhbWVzID0gMTsKCWVsc2UgaWYgKCFzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIlNBTVNVTkcgQ0QtUk9NIFNDUi0yNDMyIikpCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bGltaXRfbmZyYW1lcyA9IDE7CgkvKiB0aGUgMzIzMSBtb2RlbCBkb2VzIG5vdCBzdXBwb3J0IHRoZSBTRVRfQ0RfU1BFRUQgY29tbWFuZCAqLwoJZWxzZSBpZiAoIXN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiU0FNU1VORyBDRC1ST00gU0NSLTMyMzEiKSkKCQljZGktPm1hc2sgfD0gQ0RDX1NFTEVDVF9TUEVFRDsKCiNpZiAhIFNUQU5EQVJEX0FUQVBJCgkvKiBieSBkZWZhdWx0IFNhbnlvIDMgQ0QgY2hhbmdlciBzdXBwb3J0IGlzIHR1cm5lZCBvZmYgYW5kCiAgICAgICAgICAgQVRBUEkgUmV2IDIuMisgc3RhbmRhcmQgc3VwcG9ydCBmb3IgQ0QgY2hhbmdlcnMgaXMgdXNlZCAqLwoJY2RpLT5zYW55b19zbG90ID0gMDsKCglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5uZWMyNjAgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jdHJhY2tzX2FzX2JjZCA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50b2NhZGRyX2FzX2JjZCA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5wbGF5bXNmX2FzX2JjZCA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zdWJjaGFuX2FzX2JjZCA9IDA7CgoJaWYgKHN0cmNtcCAoZHJpdmUtPmlkLT5tb2RlbCwgIlYwMDNTMERTIikgPT0gMCAmJgoJICAgIGRyaXZlLT5pZC0+ZndfcmV2WzRdID09ICcxJyAmJgoJICAgIGRyaXZlLT5pZC0+ZndfcmV2WzZdIDw9ICcyJykgewoJCS8qIFZlcnRvcyAzMDAuCgkJICAgU29tZSB2ZXJzaW9ucyBvZiB0aGlzIGRyaXZlIGxpa2UgdG8gdGFsayBCQ0QuICovCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jdHJhY2tzX2FzX2JjZCA9IDE7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jYWRkcl9hc19iY2QgPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnBsYXltc2ZfYXNfYmNkID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zdWJjaGFuX2FzX2JjZCA9IDE7Cgl9CgoJZWxzZSBpZiAoc3RyY21wIChkcml2ZS0+aWQtPm1vZGVsLCAiVjAwNkUwRFMiKSA9PSAwICYmCgkgICAgZHJpdmUtPmlkLT5md19yZXZbNF0gPT0gJzEnICYmCgkgICAgZHJpdmUtPmlkLT5md19yZXZbNl0gPD0gJzInKSB7CgkJLyogVmVydG9zIDYwMCBFU0QuICovCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jdHJhY2tzX2FzX2JjZCA9IDE7Cgl9CgllbHNlIGlmIChzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIk5FQyBDRC1ST00gRFJJVkU6MjYwIikgPT0gMCAmJgoJCSBzdHJuY21wKGRyaXZlLT5pZC0+ZndfcmV2LCAiMS4wMSIsIDQpID09IDApIHsgLyogRklYTUUgKi8KCQkvKiBPbGQgTkVDMjYwIChub3QgUikuCgkJICAgVGhpcyBkcml2ZSB3YXMgcmVsZWFzZWQgYmVmb3JlIHRoZSAxLjIgdmVyc2lvbgoJCSAgIG9mIHRoZSBzcGVjLiAqLwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY2FkZHJfYXNfYmNkID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5wbGF5bXNmX2FzX2JjZCA9IDE7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+c3ViY2hhbl9hc19iY2QgPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5lYzI2MCAgICAgICAgID0gMTsKCX0KCWVsc2UgaWYgKHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiV0VBUk5FUyBDREQtMTIwIikgPT0gMCAmJgoJCSBzdHJuY21wKGRyaXZlLT5pZC0+ZndfcmV2LCAiQTEuMSIsIDQpID09IDApIHsgLyogRklYTUUgKi8KCQkvKiBXZWFybmVzICovCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+cGxheW1zZl9hc19iY2QgPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnN1YmNoYW5fYXNfYmNkID0gMTsKCX0KICAgICAgICAvKiBTYW55byAzIENEIGNoYW5nZXIgdXNlcyBhIG5vbi1zdGFuZGFyZCBjb21tYW5kCiAgICAgICAgICAgZm9yIENEIGNoYW5naW5nICovCiAgICAgICAgZWxzZSBpZiAoKHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiQ0QtUk9NIENEUi1DMyBHIikgPT0gMCkgfHwKICAgICAgICAgICAgICAgICAoc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJDRC1ST00gQ0RSLUMzRyIpID09IDApIHx8CiAgICAgICAgICAgICAgICAgKHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiQ0QtUk9NIENEUl9DMzYiKSA9PSAwKSkgewogICAgICAgICAgICAgICAgIC8qIHVzZXMgQ0QgaW4gc2xvdCAwIHdoZW4gdmFsdWUgaXMgc2V0IHRvIDMgKi8KICAgICAgICAgICAgICAgICBjZGktPnNhbnlvX3Nsb3QgPSAzOwogICAgICAgIH0KI2VuZGlmIC8qIG5vdCBTVEFOREFSRF9BVEFQSSAqLwoKCWluZm8tPnRvYwkJPSBOVUxMOwoJaW5mby0+YnVmZmVyCQk9IE5VTEw7CglpbmZvLT5zZWN0b3JfYnVmZmVyZWQJPSAwOwoJaW5mby0+bnNlY3RvcnNfYnVmZmVyZWQJPSAwOwoJaW5mby0+Y2hhbmdlcl9pbmZvICAgICAgPSBOVUxMOwoJaW5mby0+bGFzdF9ibG9jawk9IDA7CglpbmZvLT5zdGFydF9zZWVrCT0gMDsKCgluc2xvdHMgPSBpZGVfY2Ryb21fcHJvYmVfY2FwYWJpbGl0aWVzIChkcml2ZSk7CgoJLyoKCSAqIHNldCBjb3JyZWN0IGJsb2NrIHNpemUKCSAqLwoJYmxrX3F1ZXVlX2hhcmRzZWN0X3NpemUoZHJpdmUtPnF1ZXVlLCBDRF9GUkFNRVNJWkUpOwoKCWlmIChkcml2ZS0+YXV0b3R1bmUgPT0gSURFX1RVTkVfREVGQVVMVCB8fAoJICAgIGRyaXZlLT5hdXRvdHVuZSA9PSBJREVfVFVORV9BVVRPKQoJCWRyaXZlLT5kc2Nfb3ZlcmxhcCA9IChkcml2ZS0+bmV4dCAhPSBkcml2ZSk7CgoJaWYgKGlkZV9jZHJvbV9yZWdpc3Rlcihkcml2ZSwgbnNsb3RzKSkgewoJCXByaW50ayAoS0VSTl9FUlIgIiVzOiBpZGVfY2Ryb21fc2V0dXAgZmFpbGVkIHRvIHJlZ2lzdGVyIGRldmljZSB3aXRoIHRoZSBjZHJvbSBkcml2ZXIuXG4iLCBkcml2ZS0+bmFtZSk7CgkJaW5mby0+ZGV2aW5mby5oYW5kbGUgPSBOVUxMOwoJCXJldHVybiAxOwoJfQoJaWRlX2Nkcm9tX2FkZF9zZXR0aW5ncyhkcml2ZSk7CglyZXR1cm4gMDsKfQoKI2lmZGVmIENPTkZJR19JREVfUFJPQ19GUwpzdGF0aWMKc2VjdG9yX3QgaWRlX2Nkcm9tX2NhcGFjaXR5IChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXVuc2lnbmVkIGxvbmcgY2FwYWNpdHksIHNlY3RvcnNfcGVyX2ZyYW1lOwoKCWlmIChjZHJvbV9yZWFkX2NhcGFjaXR5KGRyaXZlLCAmY2FwYWNpdHksICZzZWN0b3JzX3Blcl9mcmFtZSwgTlVMTCkpCgkJcmV0dXJuIDA7CgoJcmV0dXJuIGNhcGFjaXR5ICogc2VjdG9yc19wZXJfZnJhbWU7Cn0KI2VuZGlmCgpzdGF0aWMgdm9pZCBpZGVfY2RfcmVtb3ZlKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJaWRlX3Byb2NfdW5yZWdpc3Rlcl9kcml2ZXIoZHJpdmUsIGluZm8tPmRyaXZlcik7CgoJZGVsX2dlbmRpc2soaW5mby0+ZGlzayk7CgoJaWRlX2NkX3B1dChpbmZvKTsKfQoKc3RhdGljIHZvaWQgaWRlX2NkX3JlbGVhc2Uoc3RydWN0IGtyZWYgKmtyZWYpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gdG9faWRlX2NkKGtyZWYpOwoJc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpkZXZpbmZvID0gJmluZm8tPmRldmluZm87CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBpbmZvLT5kcml2ZTsKCXN0cnVjdCBnZW5kaXNrICpnID0gaW5mby0+ZGlzazsKCglrZnJlZShpbmZvLT5idWZmZXIpOwoJa2ZyZWUoaW5mby0+dG9jKTsKCWtmcmVlKGluZm8tPmNoYW5nZXJfaW5mbyk7CglpZiAoZGV2aW5mby0+aGFuZGxlID09IGRyaXZlICYmIHVucmVnaXN0ZXJfY2Ryb20oZGV2aW5mbykpCgkJcHJpbnRrKEtFUk5fRVJSICIlczogJXMgZmFpbGVkIHRvIHVucmVnaXN0ZXIgZGV2aWNlIGZyb20gdGhlIGNkcm9tICIKCQkJCSJkcml2ZXIuXG4iLCBfX0ZVTkNUSU9OX18sIGRyaXZlLT5uYW1lKTsKCWRyaXZlLT5kc2Nfb3ZlcmxhcCA9IDA7Cglkcml2ZS0+ZHJpdmVyX2RhdGEgPSBOVUxMOwoJYmxrX3F1ZXVlX3ByZXBfcnEoZHJpdmUtPnF1ZXVlLCBOVUxMKTsKCWctPnByaXZhdGVfZGF0YSA9IE5VTEw7CglwdXRfZGlzayhnKTsKCWtmcmVlKGluZm8pOwp9CgpzdGF0aWMgaW50IGlkZV9jZF9wcm9iZShpZGVfZHJpdmVfdCAqKTsKCiNpZmRlZiBDT05GSUdfSURFX1BST0NfRlMKc3RhdGljIGludCBwcm9jX2lkZWNkX3JlYWRfY2FwYWNpdHkKCShjaGFyICpwYWdlLCBjaGFyICoqc3RhcnQsIG9mZl90IG9mZiwgaW50IGNvdW50LCBpbnQgKmVvZiwgdm9pZCAqZGF0YSkKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gZGF0YTsKCWludCBsZW47CgoJbGVuID0gc3ByaW50ZihwYWdlLCIlbGx1XG4iLCAobG9uZyBsb25nKWlkZV9jZHJvbV9jYXBhY2l0eShkcml2ZSkpOwoJUFJPQ19JREVfUkVBRF9SRVRVUk4ocGFnZSxzdGFydCxvZmYsY291bnQsZW9mLGxlbik7Cn0KCnN0YXRpYyBpZGVfcHJvY19lbnRyeV90IGlkZWNkX3Byb2NbXSA9IHsKCXsgImNhcGFjaXR5IiwgU19JRlJFR3xTX0lSVUdPLCBwcm9jX2lkZWNkX3JlYWRfY2FwYWNpdHksIE5VTEwgfSwKCXsgTlVMTCwgMCwgTlVMTCwgTlVMTCB9Cn07CiNlbmRpZgoKc3RhdGljIGlkZV9kcml2ZXJfdCBpZGVfY2Ryb21fZHJpdmVyID0gewoJLmdlbl9kcml2ZXIgPSB7CgkJLm93bmVyCQk9IFRISVNfTU9EVUxFLAoJCS5uYW1lCQk9ICJpZGUtY2Ryb20iLAoJCS5idXMJCT0gJmlkZV9idXNfdHlwZSwKCX0sCgkucHJvYmUJCQk9IGlkZV9jZF9wcm9iZSwKCS5yZW1vdmUJCQk9IGlkZV9jZF9yZW1vdmUsCgkudmVyc2lvbgkJPSBJREVDRF9WRVJTSU9OLAoJLm1lZGlhCQkJPSBpZGVfY2Ryb20sCgkuc3VwcG9ydHNfZHNjX292ZXJsYXAJPSAxLAoJLmRvX3JlcXVlc3QJCT0gaWRlX2RvX3J3X2Nkcm9tLAoJLmVuZF9yZXF1ZXN0CQk9IGlkZV9lbmRfcmVxdWVzdCwKCS5lcnJvcgkJCT0gX19pZGVfZXJyb3IsCgkuYWJvcnQJCQk9IF9faWRlX2Fib3J0LAojaWZkZWYgQ09ORklHX0lERV9QUk9DX0ZTCgkucHJvYwkJCT0gaWRlY2RfcHJvYywKI2VuZGlmCn07CgpzdGF0aWMgaW50IGlkZWNkX29wZW4oc3RydWN0IGlub2RlICogaW5vZGUsIHN0cnVjdCBmaWxlICogZmlsZSkKewoJc3RydWN0IGdlbmRpc2sgKmRpc2sgPSBpbm9kZS0+aV9iZGV2LT5iZF9kaXNrOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm87CglpbnQgcmMgPSAtRU5PTUVNOwoKCWlmICghKGluZm8gPSBpZGVfY2RfZ2V0KGRpc2spKSkKCQlyZXR1cm4gLUVOWElPOwoKCWlmICghaW5mby0+YnVmZmVyKQoJCWluZm8tPmJ1ZmZlciA9IGttYWxsb2MoU0VDVE9SX0JVRkZFUl9TSVpFLCBHRlBfS0VSTkVMfF9fR0ZQX1JFUEVBVCk7CgoJaWYgKGluZm8tPmJ1ZmZlcikKCQlyYyA9IGNkcm9tX29wZW4oJmluZm8tPmRldmluZm8sIGlub2RlLCBmaWxlKTsKCglpZiAocmMgPCAwKQoJCWlkZV9jZF9wdXQoaW5mbyk7CgoJcmV0dXJuIHJjOwp9CgpzdGF0aWMgaW50IGlkZWNkX3JlbGVhc2Uoc3RydWN0IGlub2RlICogaW5vZGUsIHN0cnVjdCBmaWxlICogZmlsZSkKewoJc3RydWN0IGdlbmRpc2sgKmRpc2sgPSBpbm9kZS0+aV9iZGV2LT5iZF9kaXNrOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBpZGVfY2RfZyhkaXNrKTsKCgljZHJvbV9yZWxlYXNlICgmaW5mby0+ZGV2aW5mbywgZmlsZSk7CgoJaWRlX2NkX3B1dChpbmZvKTsKCglyZXR1cm4gMDsKfQoKc3RhdGljIGludCBpZGVjZF9zZXRfc3BpbmRvd24oc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksIHVuc2lnbmVkIGxvbmcgYXJnKQp7CglzdHJ1Y3QgcGFja2V0X2NvbW1hbmQgY2djOwoJY2hhciBidWZmZXJbMTZdOwoJaW50IHN0YXQ7CgljaGFyIHNwaW5kb3duOwoKCWlmIChjb3B5X2Zyb21fdXNlcigmc3BpbmRvd24sICh2b2lkIF9fdXNlciAqKWFyZywgc2l6ZW9mKGNoYXIpKSkKCQlyZXR1cm4gLUVGQVVMVDsKCglpbml0X2Nkcm9tX2NvbW1hbmQoJmNnYywgYnVmZmVyLCBzaXplb2YoYnVmZmVyKSwgQ0dDX0RBVEFfVU5LTk9XTik7CgoJc3RhdCA9IGNkcm9tX21vZGVfc2Vuc2UoY2RpLCAmY2djLCBHUE1PREVfQ0RST01fUEFHRSwgMCk7CglpZiAoc3RhdCkKCQlyZXR1cm4gc3RhdDsKCglidWZmZXJbMTFdID0gKGJ1ZmZlclsxMV0gJiAweGYwKSB8IChzcGluZG93biAmIDB4MGYpOwoJcmV0dXJuIGNkcm9tX21vZGVfc2VsZWN0KGNkaSwgJmNnYyk7Cn0KCnN0YXRpYyBpbnQgaWRlY2RfZ2V0X3NwaW5kb3duKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLCB1bnNpZ25lZCBsb25nIGFyZykKewoJc3RydWN0IHBhY2tldF9jb21tYW5kIGNnYzsKCWNoYXIgYnVmZmVyWzE2XTsKCWludCBzdGF0OwogCWNoYXIgc3BpbmRvd247CgoJaW5pdF9jZHJvbV9jb21tYW5kKCZjZ2MsIGJ1ZmZlciwgc2l6ZW9mKGJ1ZmZlciksIENHQ19EQVRBX1VOS05PV04pOwoKCXN0YXQgPSBjZHJvbV9tb2RlX3NlbnNlKGNkaSwgJmNnYywgR1BNT0RFX0NEUk9NX1BBR0UsIDApOwoJaWYgKHN0YXQpCgkJcmV0dXJuIHN0YXQ7CgoJc3BpbmRvd24gPSBidWZmZXJbMTFdICYgMHgwZjsKCWlmIChjb3B5X3RvX3VzZXIoKHZvaWQgX191c2VyICopYXJnLCAmc3BpbmRvd24sIHNpemVvZiAoY2hhcikpKQoJCXJldHVybiAtRUZBVUxUOwoJcmV0dXJuIDA7Cn0KCnN0YXRpYyBpbnQgaWRlY2RfaW9jdGwgKHN0cnVjdCBpbm9kZSAqaW5vZGUsIHN0cnVjdCBmaWxlICpmaWxlLAoJCQl1bnNpZ25lZCBpbnQgY21kLCB1bnNpZ25lZCBsb25nIGFyZykKewoJc3RydWN0IGJsb2NrX2RldmljZSAqYmRldiA9IGlub2RlLT5pX2JkZXY7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGlkZV9jZF9nKGJkZXYtPmJkX2Rpc2spOwoJaW50IGVycjsKCglzd2l0Y2ggKGNtZCkgewogCWNhc2UgQ0RST01TRVRTUElORE9XTjoKCQlyZXR1cm4gaWRlY2Rfc2V0X3NwaW5kb3duKCZpbmZvLT5kZXZpbmZvLCBhcmcpOwogCWNhc2UgQ0RST01HRVRTUElORE9XTjoKCQlyZXR1cm4gaWRlY2RfZ2V0X3NwaW5kb3duKCZpbmZvLT5kZXZpbmZvLCBhcmcpOwoJZGVmYXVsdDoKCQlicmVhazsKIAl9CgoJZXJyID0gZ2VuZXJpY19pZGVfaW9jdGwoaW5mby0+ZHJpdmUsIGZpbGUsIGJkZXYsIGNtZCwgYXJnKTsKCWlmIChlcnIgPT0gLUVJTlZBTCkKCQllcnIgPSBjZHJvbV9pb2N0bChmaWxlLCAmaW5mby0+ZGV2aW5mbywgaW5vZGUsIGNtZCwgYXJnKTsKCglyZXR1cm4gZXJyOwp9CgpzdGF0aWMgaW50IGlkZWNkX21lZGlhX2NoYW5nZWQoc3RydWN0IGdlbmRpc2sgKmRpc2spCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gaWRlX2NkX2coZGlzayk7CglyZXR1cm4gY2Ryb21fbWVkaWFfY2hhbmdlZCgmaW5mby0+ZGV2aW5mbyk7Cn0KCnN0YXRpYyBpbnQgaWRlY2RfcmV2YWxpZGF0ZV9kaXNrKHN0cnVjdCBnZW5kaXNrICpkaXNrKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGlkZV9jZF9nKGRpc2spOwoJc3RydWN0IHJlcXVlc3Rfc2Vuc2Ugc2Vuc2U7CgljZHJvbV9yZWFkX3RvYyhpbmZvLT5kcml2ZSwgJnNlbnNlKTsKCXJldHVybiAgMDsKfQoKc3RhdGljIHN0cnVjdCBibG9ja19kZXZpY2Vfb3BlcmF0aW9ucyBpZGVjZF9vcHMgPSB7Cgkub3duZXIJCT0gVEhJU19NT0RVTEUsCgkub3BlbgkJPSBpZGVjZF9vcGVuLAoJLnJlbGVhc2UJPSBpZGVjZF9yZWxlYXNlLAoJLmlvY3RsCQk9IGlkZWNkX2lvY3RsLAoJLm1lZGlhX2NoYW5nZWQJPSBpZGVjZF9tZWRpYV9jaGFuZ2VkLAoJLnJldmFsaWRhdGVfZGlzaz0gaWRlY2RfcmV2YWxpZGF0ZV9kaXNrCn07CgovKiBvcHRpb25zICovCnN0YXRpYyBjaGFyICppZ25vcmUgPSBOVUxMOwoKbW9kdWxlX3BhcmFtKGlnbm9yZSwgY2hhcnAsIDA0MDApOwpNT0RVTEVfREVTQ1JJUFRJT04oIkFUQVBJIENELVJPTSBEcml2ZXIiKTsKCnN0YXRpYyBpbnQgaWRlX2NkX3Byb2JlKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm87CglzdHJ1Y3QgZ2VuZGlzayAqZzsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIHNlbnNlOwoKCWlmICghc3Ryc3RyKCJpZGUtY2Ryb20iLCBkcml2ZS0+ZHJpdmVyX3JlcSkpCgkJZ290byBmYWlsZWQ7CglpZiAoIWRyaXZlLT5wcmVzZW50KQoJCWdvdG8gZmFpbGVkOwoJaWYgKGRyaXZlLT5tZWRpYSAhPSBpZGVfY2Ryb20gJiYgZHJpdmUtPm1lZGlhICE9IGlkZV9vcHRpY2FsKQoJCWdvdG8gZmFpbGVkOwoJLyogc2tpcCBkcml2ZXMgdGhhdCB3ZSB3ZXJlIHRvbGQgdG8gaWdub3JlICovCglpZiAoaWdub3JlICE9IE5VTEwpIHsKCQlpZiAoc3Ryc3RyKGlnbm9yZSwgZHJpdmUtPm5hbWUpKSB7CgkJCXByaW50ayhLRVJOX0lORk8gImlkZS1jZDogaWdub3JpbmcgZHJpdmUgJXNcbiIsIGRyaXZlLT5uYW1lKTsKCQkJZ290byBmYWlsZWQ7CgkJfQoJfQoJaWYgKGRyaXZlLT5zY3NpKSB7CgkJcHJpbnRrKEtFUk5fSU5GTyAiaWRlLWNkOiBwYXNzaW5nIGRyaXZlICVzIHRvIGlkZS1zY3NpIGVtdWxhdGlvbi5cbiIsIGRyaXZlLT5uYW1lKTsKCQlnb3RvIGZhaWxlZDsKCX0KCWluZm8gPSBremFsbG9jKHNpemVvZihzdHJ1Y3QgY2Ryb21faW5mbyksIEdGUF9LRVJORUwpOwoJaWYgKGluZm8gPT0gTlVMTCkgewoJCXByaW50ayhLRVJOX0VSUiAiJXM6IENhbid0IGFsbG9jYXRlIGEgY2Ryb20gc3RydWN0dXJlXG4iLCBkcml2ZS0+bmFtZSk7CgkJZ290byBmYWlsZWQ7Cgl9CgoJZyA9IGFsbG9jX2Rpc2soMSA8PCBQQVJUTl9CSVRTKTsKCWlmICghZykKCQlnb3RvIG91dF9mcmVlX2NkOwoKCWlkZV9pbml0X2Rpc2soZywgZHJpdmUpOwoKCWlkZV9wcm9jX3JlZ2lzdGVyX2RyaXZlcihkcml2ZSwgJmlkZV9jZHJvbV9kcml2ZXIpOwoKCWtyZWZfaW5pdCgmaW5mby0+a3JlZik7CgoJaW5mby0+ZHJpdmUgPSBkcml2ZTsKCWluZm8tPmRyaXZlciA9ICZpZGVfY2Ryb21fZHJpdmVyOwoJaW5mby0+ZGlzayA9IGc7CgoJZy0+cHJpdmF0ZV9kYXRhID0gJmluZm8tPmRyaXZlcjsKCglkcml2ZS0+ZHJpdmVyX2RhdGEgPSBpbmZvOwoKCWctPm1pbm9ycyA9IDE7CglnLT5kcml2ZXJmc19kZXYgPSAmZHJpdmUtPmdlbmRldjsKCWctPmZsYWdzID0gR0VOSERfRkxfQ0QgfCBHRU5IRF9GTF9SRU1PVkFCTEU7CglpZiAoaWRlX2Nkcm9tX3NldHVwKGRyaXZlKSkgewoJCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqZGV2aW5mbyA9ICZpbmZvLT5kZXZpbmZvOwoJCWlkZV9wcm9jX3VucmVnaXN0ZXJfZHJpdmVyKGRyaXZlLCAmaWRlX2Nkcm9tX2RyaXZlcik7CgkJa2ZyZWUoaW5mby0+YnVmZmVyKTsKCQlrZnJlZShpbmZvLT50b2MpOwoJCWtmcmVlKGluZm8tPmNoYW5nZXJfaW5mbyk7CgkJaWYgKGRldmluZm8tPmhhbmRsZSA9PSBkcml2ZSAmJiB1bnJlZ2lzdGVyX2Nkcm9tKGRldmluZm8pKQoJCQlwcmludGsgKEtFUk5fRVJSICIlczogaWRlX2Nkcm9tX2NsZWFudXAgZmFpbGVkIHRvIHVucmVnaXN0ZXIgZGV2aWNlIGZyb20gdGhlIGNkcm9tIGRyaXZlci5cbiIsIGRyaXZlLT5uYW1lKTsKCQlrZnJlZShpbmZvKTsKCQlkcml2ZS0+ZHJpdmVyX2RhdGEgPSBOVUxMOwoJCWdvdG8gZmFpbGVkOwoJfQoKCWNkcm9tX3JlYWRfdG9jKGRyaXZlLCAmc2Vuc2UpOwoJZy0+Zm9wcyA9ICZpZGVjZF9vcHM7CglnLT5mbGFncyB8PSBHRU5IRF9GTF9SRU1PVkFCTEU7CglhZGRfZGlzayhnKTsKCXJldHVybiAwOwoKb3V0X2ZyZWVfY2Q6CglrZnJlZShpbmZvKTsKZmFpbGVkOgoJcmV0dXJuIC1FTk9ERVY7Cn0KCnN0YXRpYyB2b2lkIF9fZXhpdCBpZGVfY2Ryb21fZXhpdCh2b2lkKQp7Cglkcml2ZXJfdW5yZWdpc3RlcigmaWRlX2Nkcm9tX2RyaXZlci5nZW5fZHJpdmVyKTsKfQoKc3RhdGljIGludCBfX2luaXQgaWRlX2Nkcm9tX2luaXQodm9pZCkKewoJcmV0dXJuIGRyaXZlcl9yZWdpc3RlcigmaWRlX2Nkcm9tX2RyaXZlci5nZW5fZHJpdmVyKTsKfQoKTU9EVUxFX0FMSUFTKCJpZGU6Km0tY2Ryb20qIik7Cm1vZHVsZV9pbml0KGlkZV9jZHJvbV9pbml0KTsKbW9kdWxlX2V4aXQoaWRlX2Nkcm9tX2V4aXQpOwpNT0RVTEVfTElDRU5TRSgiR1BMIik7Cg==