LyoKICogbGludXgvZHJpdmVycy9pZGUvaWRlLWNkLmMKICoKICogQ29weXJpZ2h0IChDKSAxOTk0LCAxOTk1LCAxOTk2ICBzY290dCBzbnlkZXIgIDxzbnlkZXJAZm5hbGQwLmZuYWwuZ292PgogKiBDb3B5cmlnaHQgKEMpIDE5OTYtMTk5OCAgRXJpayBBbmRlcnNlbiA8YW5kZXJzZWVAZGViaWFuLm9yZz4KICogQ29weXJpZ2h0IChDKSAxOTk4LTIwMDAgIEplbnMgQXhib2UgPGF4Ym9lQHN1c2UuZGU+CiAqCiAqIE1heSBiZSBjb3BpZWQgb3IgbW9kaWZpZWQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMKICogTGljZW5zZS4gIFNlZSBsaW51eC9DT1BZSU5HIGZvciBtb3JlIGluZm9ybWF0aW9uLgogKgogKiBBVEFQSSBDRC1ST00gZHJpdmVyLiAgVG8gYmUgdXNlZCB3aXRoIGlkZS5jLgogKiBTZWUgRG9jdW1lbnRhdGlvbi9jZHJvbS9pZGUtY2QgZm9yIHVzYWdlIGluZm9ybWF0aW9uLgogKgogKiBTdWdnZXN0aW9ucyBhcmUgd2VsY29tZS4gUGF0Y2hlcyB0aGF0IHdvcmsgYXJlIG1vcmUgd2VsY29tZSB0aG91Z2guIDstKQogKiBGb3IgdGhvc2Ugd2lzaGluZyB0byB3b3JrIG9uIHRoaXMgZHJpdmVyLCBwbGVhc2UgYmUgc3VyZSB5b3UgZG93bmxvYWQKICogYW5kIGNvbXBseSB3aXRoIHRoZSBsYXRlc3QgTXQuIEZ1amkgKFNGRjgwOTAgdmVyc2lvbiA0KSBhbmQgQVRBUEkgCiAqIChTRkYtODAyMGkgcmV2IDIuNikgc3RhbmRhcmRzLiBUaGVzZSBkb2N1bWVudHMgY2FuIGJlIG9idGFpbmVkIGJ5IAogKiBhbm9ueW1vdXMgZnRwIGZyb206CiAqIGZ0cDovL2Zpc3Npb24uZHQud2RjLmNvbS9wdWIvc3RhbmRhcmRzL1NGRl9hdGFwaS9zcGVjL1NGRjgwMjAtcjIuNi9QUy84MDIwcjI2LnBzCiAqIGZ0cDovL2Z0cC5hdmMtcGlvbmVlci5jb20vTXRmdWppNC9TcGVjL0Z1amk0cjEwLnBkZgogKgogKiBEcml2ZXMgdGhhdCBkZXZpYXRlIGZyb20gdGhlc2Ugc3RhbmRhcmRzIHdpbGwgYmUgYWNjb21tb2RhdGVkIGFzIG11Y2gKICogYXMgcG9zc2libGUgdmlhIGNvbXBpbGUgdGltZSBvciBjb21tYW5kLWxpbmUgb3B0aW9ucy4gIFNpbmNlIEkgb25seSBoYXZlCiAqIGEgZmV3IGRyaXZlcywgeW91IGdlbmVyYWxseSBuZWVkIHRvIHNlbmQgbWUgcGF0Y2hlcy4uLgogKgogKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAqIFRPIERPIExJU1Q6CiAqIC1NYWtlIGl0IHNvIHRoYXQgUGlvbmVlciBDRCBEUi1BMjRYIGFuZCBmcmllbmRzIGRvbid0IGdldCBzY3Jld2VkIHVwIG9uCiAqICAgYm9vdAogKgogKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAqIDEuMDAgIE9jdCAzMSwgMTk5NCAtLSBJbml0aWFsIHZlcnNpb24uCiAqIDEuMDEgIE5vdiAgMiwgMTk5NCAtLSBGaXhlZCBwcm9ibGVtIHdpdGggc3RhcnRpbmcgcmVxdWVzdCBpbgogKiAgICAgICAgICAgICAgICAgICAgICAgY2Ryb21fY2hlY2tfc3RhdHVzLgogKiAxLjAzICBOb3YgMjUsIDE5OTQgLS0gbGVhdmluZyB1bm1hc2tfaW50cltdIGFzIGEgdXNlci1zZXR0aW5nIChhcyBmb3IgZGlza3MpCiAqIChmcm9tIG1sb3JkKSAgICAgICAtLSBtaW5vciBjaGFuZ2VzIHRvIGNkcm9tX3NldHVwKCkKICogICAgICAgICAgICAgICAgICAgIC0tIHJlbmFtZWQgaWRlX2Rldl9zIHRvIGlkZV9kcml2ZV90LCBlbmFibGUgaXJxIG9uIGNvbW1hbmQKICogMi4wMCAgTm92IDI3LCAxOTk0IC0tIEdlbmVyYWxpemUgcGFja2V0IGNvbW1hbmQgaW50ZXJmYWNlOwogKiAgICAgICAgICAgICAgICAgICAgICAgYWRkIGF1ZGlvIGlvY3Rscy4KICogMi4wMSAgRGVjICAzLCAxOTk0IC0tIFJld29yayBwYWNrZXQgY29tbWFuZCBpbnRlcmZhY2UgdG8gaGFuZGxlIGRldmljZXMKICogICAgICAgICAgICAgICAgICAgICAgIHdoaWNoIHNlbmQgYW4gaW50ZXJydXB0IHdoZW4gcmVhZHkgZm9yIGEgY29tbWFuZC4KICogMi4wMiAgRGVjIDExLCAxOTk0IC0tIENhY2hlIHRoZSBUT0MgaW4gdGhlIGRyaXZlci4KICogICAgICAgICAgICAgICAgICAgICAgIERvbid0IHVzZSBTQ01EX1BMQVlBVURJT19USTsgaXQncyBub3QgaW5jbHVkZWQKICogICAgICAgICAgICAgICAgICAgICAgIGluIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgQVRBUEkuCiAqICAgICAgICAgICAgICAgICAgICAgICBUcnkgdG8gdXNlIExCQSBpbnN0ZWFkIG9mIHRyYWNrIG9yIE1TRiBhZGRyZXNzaW5nCiAqICAgICAgICAgICAgICAgICAgICAgICB3aGVuIHBvc3NpYmxlLgogKiAgICAgICAgICAgICAgICAgICAgICAgRG9uJ3Qgd2FpdCBmb3IgUkVBRFlfU1RBVC4KICogMi4wMyAgSmFuIDEwLCAxOTk1IC0tIFJld3JpdGUgYmxvY2sgcmVhZCByb3V0aW5lcyB0byBoYW5kbGUgYmxvY2sgc2l6ZXMKICogICAgICAgICAgICAgICAgICAgICAgIG90aGVyIHRoYW4gMmsgYW5kIHRvIG1vdmUgbXVsdGlwbGUgc2VjdG9ycyBpbiBhCiAqICAgICAgICAgICAgICAgICAgICAgICBzaW5nbGUgdHJhbnNhY3Rpb24uCiAqIDIuMDQgIEFwciAyMSwgMTk5NSAtLSBBZGQgd29yay1hcm91bmQgZm9yIENyZWF0aXZlIExhYnMgQ0QyMjBFIGRyaXZlcy4KICogICAgICAgICAgICAgICAgICAgICAgIFRoYW5rcyB0byBOaWNrIFNhdyA8Y3dzYXdAcHRzNy5wdHMubW90LmNvbT4gZm9yCiAqICAgICAgICAgICAgICAgICAgICAgICBoZWxwIGluIGZpZ3VyaW5nIHRoaXMgb3V0LiAgRGl0dG8gZm9yIEFjZXIgYW5kCiAqICAgICAgICAgICAgICAgICAgICAgICBBenRlY2ggZHJpdmVzLCB3aGljaCBzZWVtIHRvIGhhdmUgdGhlIHNhbWUgcHJvYmxlbS4KICogMi4wNGIgTWF5IDMwLCAxOTk1IC0tIEZpeCB0byBtYXRjaCBjaGFuZ2VzIGluIGlkZS5jIHZlcnNpb24gMy4xNiAtbWwKICogMi4wNSAgSnVuICA4LCAxOTk1IC0tIERvbid0IGF0dGVtcHQgdG8gcmV0cnkgYWZ0ZXIgYW4gaWxsZWdhbCByZXF1ZXN0CiAqICAgICAgICAgICAgICAgICAgICAgICAgb3IgZGF0YSBwcm90ZWN0IGVycm9yLgogKiAgICAgICAgICAgICAgICAgICAgICAgVXNlIEhXSUYgYW5kIERFVl9IV0lGIG1hY3JvcyBhcyBpbiBpZGUuYy4KICogICAgICAgICAgICAgICAgICAgICAgIEFsd2F5cyB0cnkgdG8gZG8gYSByZXF1ZXN0X3NlbnNlIGFmdGVyCiAqICAgICAgICAgICAgICAgICAgICAgICAgYSBmYWlsZWQgY29tbWFuZC4KICogICAgICAgICAgICAgICAgICAgICAgIEluY2x1ZGUgYW4gb3B0aW9uIHRvIGdpdmUgdGV4dHVhbCBkZXNjcmlwdGlvbnMKICogICAgICAgICAgICAgICAgICAgICAgICBvZiBBVEFQSSBlcnJvcnMuCiAqICAgICAgICAgICAgICAgICAgICAgICBGaXggYSBidWcgaW4gaGFuZGxpbmcgdGhlIHNlY3RvciBjYWNoZSB3aGljaAogKiAgICAgICAgICAgICAgICAgICAgICAgIHNob3dlZCB1cCBpZiB0aGUgZHJpdmUgcmV0dXJuZWQgZGF0YSBpbiA1MTIgYnl0ZQogKiAgICAgICAgICAgICAgICAgICAgICAgIGJsb2NrcyAobGlrZSBQaW9uZWVyIGRyaXZlcykuICBUaGFua3MgdG8KICogICAgICAgICAgICAgICAgICAgICAgICBSaWNoYXJkIEhpcnN0IDxzcmhAZ3B0LmNvLnVrPiBmb3IgZGlhZ25vc2luZyB0aGlzLgogKiAgICAgICAgICAgICAgICAgICAgICAgUHJvcGVybHkgc3VwcGx5IHRoZSBwYWdlIG51bWJlciBmaWVsZCBpbiB0aGUKICogICAgICAgICAgICAgICAgICAgICAgICBNT0RFX1NFTEVDVCBjb21tYW5kLgogKiAgICAgICAgICAgICAgICAgICAgICAgUExBWUFVRElPMTIgaXMgYnJva2VuIG9uIHRoZSBBenRlY2g7IHdvcmsgYXJvdW5kIGl0LgogKiAyLjA1eCBBdWcgMTEsIDE5OTUgLS0gbG90cyBvZiBkYXRhIHN0cnVjdHVyZSByZW5hbWluZy9yZXN0cnVjdHVyaW5nIGluIGlkZS5jCiAqICAgICAgICAgICAgICAgICAgICAgICAobXkgYXBvbG9naWVzIHRvIFNjb3R0LCBidXQgbm93IGlkZS1jZC5jIGlzIGluZGVwZW5kZW50KQogKiAzLjAwICBBdWcgMjIsIDE5OTUgLS0gSW1wbGVtZW50IENEUk9NTVVMVElTRVNTSU9OIGlvY3RsLgogKiAgICAgICAgICAgICAgICAgICAgICAgSW1wbGVtZW50IENEUk9NUkVBREFVRElPIGlvY3RsIChVTlRFU1RFRCkuCiAqICAgICAgICAgICAgICAgICAgICAgICBVc2UgaW5wdXRfaWRlX2RhdGEoKSBhbmQgb3V0cHV0X2lkZV9kYXRhKCkuCiAqICAgICAgICAgICAgICAgICAgICAgICBBZGQgZG9vciBsb2NraW5nLgogKiAgICAgICAgICAgICAgICAgICAgICAgRml4IHVzYWdlIGNvdW50IGxlYWsgaW4gY2Ryb21fb3Blbiwgd2hpY2ggaGFwcGVuZWQKICogICAgICAgICAgICAgICAgICAgICAgICB3aGVuIGEgcmVhZC13cml0ZSBtb3VudCB3YXMgYXR0ZW1wdGVkLgogKiAgICAgICAgICAgICAgICAgICAgICAgVHJ5IHRvIGxvYWQgdGhlIGRpc2sgb24gb3Blbi4KICogICAgICAgICAgICAgICAgICAgICAgIEltcGxlbWVudCBDRFJPTUVKRUNUX1NXIGlvY3RsIChvZmYgYnkgZGVmYXVsdCkuCiAqICAgICAgICAgICAgICAgICAgICAgICBSZWFkIHRvdGFsIGNkcm9tIGNhcGFjaXR5IGR1cmluZyBvcGVuLgogKiAgICAgICAgICAgICAgICAgICAgICAgUmVhcnJhbmdlIGxvZ2ljIGluIGNkcm9tX2RlY29kZV9zdGF0dXMuICBJc3N1ZQogKiAgICAgICAgICAgICAgICAgICAgICAgIHJlcXVlc3Qgc2Vuc2UgY29tbWFuZHMgZm9yIGZhaWxlZCBwYWNrZXQgY29tbWFuZHMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIGhlcmUgaW5zdGVhZCBvZiBmcm9tIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kLgogKiAgICAgICAgICAgICAgICAgICAgICAgIEZpeCBhIHJhY2UgY29uZGl0aW9uIGluIHJldHJpZXZpbmcgZXJyb3IgaW5mb3JtYXRpb24uCiAqICAgICAgICAgICAgICAgICAgICAgICBTdXBwcmVzcyBwcmludGluZyBub3JtYWwgdW5pdCBhdHRlbnRpb24gZXJyb3JzIGFuZAogKiAgICAgICAgICAgICAgICAgICAgICAgIHNvbWUgZHJpdmUgbm90IHJlYWR5IGVycm9ycy4KICogICAgICAgICAgICAgICAgICAgICAgIEltcGxlbWVudCBDRFJPTVZPTFJFQUQgaW9jdGwuCiAqICAgICAgICAgICAgICAgICAgICAgICBJbXBsZW1lbnQgQ0RST01SRUFETU9ERTEvMiBpb2N0bHMuCiAqICAgICAgICAgICAgICAgICAgICAgICBGaXggcmFjZSBjb25kaXRpb24gaW4gc2V0dGluZyB1cCBpbnRlcnJ1cHQgaGFuZGxlcnMKICogICAgICAgICAgICAgICAgICAgICAgICB3aGVuIHRoZSBgc2VyaWFsaXplJyBvcHRpb24gaXMgdXNlZC4KICogMy4wMSAgU2VwICAyLCAxOTk1IC0tIEZpeCBvcmRlcmluZyBvZiByZWVuYWJsaW5nIGludGVycnVwdHMgaW4KICogICAgICAgICAgICAgICAgICAgICAgICBjZHJvbV9xdWV1ZV9yZXF1ZXN0LgogKiAgICAgICAgICAgICAgICAgICAgICAgQW5vdGhlciB0cnkgYXQgdXNpbmcgaWRlX1tpbnB1dCxvdXRwdXRdX2RhdGEuCiAqIDMuMDIgIFNlcCAxNiwgMTk5NSAtLSBTdGljayB0b3RhbCBkaXNrIGNhcGFjaXR5IGluIHBhcnRpdGlvbiB0YWJsZSBhcyB3ZWxsLgogKiAgICAgICAgICAgICAgICAgICAgICAgTWFrZSBWRVJCT1NFX0lERV9DRF9FUlJPUlMgZHVtcCBmYWlsZWQgY29tbWFuZCBhZ2Fpbi4KICogICAgICAgICAgICAgICAgICAgICAgIER1bXAgb3V0IG1vcmUgaW5mb3JtYXRpb24gZm9yIElMTEVHQUwgUkVRVUVTVCBlcnJzLgogKiAgICAgICAgICAgICAgICAgICAgICAgRml4IGhhbmRsaW5nIG9mIGVycm9ycyBvY2N1cnJpbmcgYmVmb3JlIHRoZQogKiAgICAgICAgICAgICAgICAgICAgICAgIHBhY2tldCBjb21tYW5kIGlzIHRyYW5zZmVycmVkLgogKiAgICAgICAgICAgICAgICAgICAgICAgRml4IHRyYW5zZmVycyB3aXRoIG9kZCBieXRlbGVuZ3Rocy4KICogMy4wMyAgT2N0IDI3LCAxOTk1IC0tIFNvbWUgQ3JlYXRpdmUgZHJpdmVzIGhhdmUgYW4gaWQgb2YganVzdCBgQ0QnLgogKiAgICAgICAgICAgICAgICAgICAgICAgYERDSS0yUzEwJyBkcml2ZXMgYXJlIGJyb2tlbiB0b28uCiAqIDMuMDQgIE5vdiAyMCwgMTk5NSAtLSBTbyBhcmUgVmVydG9zIGRyaXZlcy4KICogMy4wNSAgRGVjICAxLCAxOTk1IC0tIENoYW5nZXMgdG8gZ28gd2l0aCBvdmVyaGF1bCBvZiBpZGUuYyBhbmQgaWRlLXRhcGUuYwogKiAzLjA2ICBEZWMgMTYsIDE5OTUgLS0gQWRkIHN1cHBvcnQgbmVlZGVkIGZvciBwYXJ0aXRpb25zLgogKiAgICAgICAgICAgICAgICAgICAgICAgTW9yZSB3b3JrYXJvdW5kcyBmb3IgVmVydG9zIGJ1Z3MgKGJhc2VkIG9uIHBhdGNoZXMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIEhvbGdlciBEaWV0emUgPGRpZXR6ZUBhaXg1MjAuaW5mb3JtYXRpay51bmktbGVpcHppZy5kZT4pLgogKiAgICAgICAgICAgICAgICAgICAgICAgVHJ5IHRvIGVsaW1pbmF0ZSBieXRlb3JkZXIgYXNzdW1wdGlvbnMuCiAqICAgICAgICAgICAgICAgICAgICAgICBVc2UgYXRhcGlfY2Ryb21fc3ViY2hubCBzdHJ1Y3QgZGVmaW5pdGlvbi4KICogICAgICAgICAgICAgICAgICAgICAgIEFkZCBTVEFOREFSRF9BVEFQSSBjb21waWxhdGlvbiBvcHRpb24uCiAqIDMuMDcgIEphbiAyOSwgMTk5NiAtLSBNb3JlIHR3aWRkbGluZyBmb3IgYnJva2VuIGRyaXZlczogU29ueSA1NUQsCiAqICAgICAgICAgICAgICAgICAgICAgICAgVmVydG9zIDMwMC4KICogICAgICAgICAgICAgICAgICAgICAgIEFkZCBOT19ET09SX0xPQ0tJTkcgY29uZmlndXJhdGlvbiBvcHRpb24uCiAqICAgICAgICAgICAgICAgICAgICAgICBIYW5kbGUgZHJpdmVfY21kIHJlcXVlc3RzIHcvTlVMTCBhcmdzIChmb3IgaGRwYXJtIC10KS4KICogICAgICAgICAgICAgICAgICAgICAgIFdvcmsgYXJvdW5kIHNwb3JhZGljIFNvbnk1NWUgYXVkaW8gcGxheSBwcm9ibGVtLgogKiAzLjA3YSBGZWIgMTEsIDE5OTYgLS0gY2hlY2sgZHJpdmUtPmlkIGZvciBOVUxMIGJlZm9yZSBkZXJlZmVyZW5jaW5nLCB0byBmaXgKICogICAgICAgICAgICAgICAgICAgICAgICBwcm9ibGVtIHdpdGggImhkZT1jZHJvbSIgd2l0aCBubyBkcml2ZSBwcmVzZW50LiAgLW1sCiAqIDMuMDggIE1hciAgNiwgMTk5NiAtLSBNb3JlIFZlcnRvcyB3b3JrYXJvdW5kcy4KICogMy4wOSAgQXByICA1LCAxOTk2IC0tIEFkZCBDRFJPTUNMT1NFVFJBWSBpb2N0bC4KICogICAgICAgICAgICAgICAgICAgICAgIFN3aXRjaCB0byB1c2luZyBNU0YgYWRkcmVzc2luZyBmb3IgYXVkaW8gY29tbWFuZHMuCiAqICAgICAgICAgICAgICAgICAgICAgICBSZWZvcm1hdCB0byBtYXRjaCBrZXJuZWwgdGFiYmluZyBzdHlsZS4KICogICAgICAgICAgICAgICAgICAgICAgIEFkZCBDRFJPTV9HRVRfVVBDIGlvY3RsLgogKiAzLjEwICBBcHIgMTAsIDE5OTYgLS0gRml4IGNvbXBpbGF0aW9uIGVycm9yIHdpdGggU1RBTkRBUkRfQVRBUEkuCiAqIDMuMTEgIEFwciAyOSwgMTk5NiAtLSBQYXRjaCBmcm9tIEhlaWtvIEVpc3NmZWxkdCA8aGVpa29AY29sb3NzdXMuZXNjYXBlLmRlPgogKiAgICAgICAgICAgICAgICAgICAgICAgdG8gcmVtb3ZlIHJlZHVuZGFudCB2ZXJpZnlfYXJlYSBjYWxscy4KICogMy4xMiAgTWF5ICA3LCAxOTk2IC0tIFJ1ZGltZW50YXJ5IGNoYW5nZXIgc3VwcG9ydC4gIEJhc2VkIG9uIHBhdGNoZXMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIEdlcmhhcmQgWnViZXIgPHp1YmVyQGJlcmxpbi5zbmFmdS5kZT4uCiAqICAgICAgICAgICAgICAgICAgICAgICBMZXQgb3BlbiBzdWNjZWVkIGV2ZW4gaWYgdGhlcmUncyBubyBsb2FkZWQgZGlzYy4KICogMy4xMyAgTWF5IDE5LCAxOTk2IC0tIEZpeGVzIGZvciBjaGFuZ2VyIGNvZGUuCiAqIDMuMTQgIE1heSAyOSwgMTk5NiAtLSBBZGQgd29yay1hcm91bmQgZm9yIFZlcnRvcyA2MDAuCiAqICAgICAgICAgICAgICAgICAgICAgICAgKEZyb20gSGVubnVzIEJlcmdtYW4gPGhlbm51c0Bza3kub3cubmw+LikKICogMy4xNSAgSnVseSAyLCAxOTk2IC0tIEFkZGVkIHN1cHBvcnQgZm9yIFNhbnlvIDMgQ0QgY2hhbmdlcnMKICogICAgICAgICAgICAgICAgICAgICAgICBmcm9tIEJlbiBHYWxsaWFydCA8YmdhbGxpYUBsdWMuZWR1PiB3aXRoIAogKiAgICAgICAgICAgICAgICAgICAgICAgIHNwZWNpYWwgaGVscCBmcm9tIEplZmYgTGlnaHRmb290IAogKiAgICAgICAgICAgICAgICAgICAgICAgIDxqZWZmbWxAcG9ib3guY29tPgogKiAzLjE1YSBKdWx5IDksIDE5OTYgLS0gSW1wcm92ZWQgU2FueW8gMyBDRCBjaGFuZ2VyIGlkZW50aWZpY2F0aW9uCiAqIDMuMTYgIEp1bCAyOCwgMTk5NiAtLSBGaXggZnJvbSBHYWRpIHRvIHJlZHVjZSBrZXJuZWwgc3RhY2sgdXNhZ2UgZm9yIGlvY3RsLgogKiAzLjE3ICBTZXAgMTcsIDE5OTYgLS0gVHdlYWsgYXVkaW8gcmVhZHMgZm9yIHNvbWUgZHJpdmVzLgogKiAgICAgICAgICAgICAgICAgICAgICAgU3RhcnQgY2hhbmdpbmcgQ0RST01MT0FERlJPTVNMT1QgdG8gQ0RST01fU0VMRUNUX0RJU0MuCiAqIDMuMTggIE9jdCAzMSwgMTk5NiAtLSBBZGRlZCBtb2R1bGUgYW5kIERNQSBzdXBwb3J0LgogKiAgICAgICAgICAgICAgICAgICAgICAgCiAqICAgICAgICAgICAgICAgICAgICAgICAKICogNC4wMCAgTm92IDUsIDE5OTYgICAtLSBOZXcgaWRlLWNkIG1haW50YWluZXIsCiAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRXJpayBCLiBBbmRlcnNlbiA8YW5kZXJzZWVAZGViaWFuLm9yZz4KICogICAgICAgICAgICAgICAgICAgICAtLSBOZXdlciBDcmVhdGl2ZSBkcml2ZXMgZG9uJ3QgYWx3YXlzIHNldCB0aGUgZXJyb3IKICogICAgICAgICAgICAgICAgICAgICAgICAgIHJlZ2lzdGVyIGNvcnJlY3RseS4gIE1ha2Ugc3VyZSB3ZSBzZWUgbWVkaWEgY2hhbmdlcwogKiAgICAgICAgICAgICAgICAgICAgICAgICAgcmVnYXJkbGVzcy4KICogICAgICAgICAgICAgICAgICAgICAtLSBJbnRlZ3JhdGUgd2l0aCBnZW5lcmljIGNkcm9tIGRyaXZlci4KICogICAgICAgICAgICAgICAgICAgICAtLSBDRFJPTUdFVFNQSU5ET1dOIGFuZCBDRFJPTVNFVFNQSU5ET1dOIGlvY3RscywgYmFzZWQgb24KICogICAgICAgICAgICAgICAgICAgICAgICAgIGEgcGF0Y2ggZnJvbSBDaXJvIENhdHR1dG8gPD4uCiAqICAgICAgICAgICAgICAgICAgICAgLS0gQ2FsbCBzZXRfZGV2aWNlX3JvLgogKiAgICAgICAgICAgICAgICAgICAgIC0tIEltcGxlbWVudCBDRFJPTU1FQ0hBTklTTVNUQVRVUyBhbmQgQ0RST01TTE9UVEFCTEUKICogICAgICAgICAgICAgICAgICAgICAgICAgIGlvY3RscywgYmFzZWQgb24gcGF0Y2ggYnkgRXJpayBBbmRlcnNlbgogKiAgICAgICAgICAgICAgICAgICAgIC0tIEFkZCBzb21lIHByb2JlcyBvZiBkcml2ZSBjYXBhYmlsaXR5IGR1cmluZyBzZXR1cC4KICoKICogNC4wMSAgTm92IDExLCAxOTk2ICAtLSBTcGxpdCBpbnRvIGlkZS1jZC5jIGFuZCBpZGUtY2QuaAogKiAgICAgICAgICAgICAgICAgICAgIC0tIFJlbW92ZWQgQ0RST01NRUNIQU5JU01TVEFUVVMgYW5kIENEUk9NU0xPVFRBQkxFIAogKiAgICAgICAgICAgICAgICAgICAgICAgICAgaW9jdGxzIGluIGZhdm9yIG9mIGEgZ2VuZXJhbGl6ZWQgYXBwcm9hY2ggCiAqICAgICAgICAgICAgICAgICAgICAgICAgICB1c2luZyB0aGUgZ2VuZXJpYyBjZHJvbSBkcml2ZXIuCiAqICAgICAgICAgICAgICAgICAgICAgLS0gRnVsbHkgaW50ZWdyYXRlZCB3aXRoIHRoZSAyLjEuWCBrZXJuZWwuCiAqICAgICAgICAgICAgICAgICAgICAgLS0gT3RoZXIgc3R1ZmYgdGhhdCBJIGZvcmdvdCAobG90cyBvZiBjaGFuZ2VzKQogKgogKiA0LjAyICBEZWMgMDEsIDE5OTYgIC0tIEFwcGxpZWQgcGF0Y2ggZnJvbSBHYWRpIE94bWFuIDxnYWRpb0BuZXR2aXNpb24ubmV0LmlsPgogKiAgICAgICAgICAgICAgICAgICAgICAgICAgdG8gZml4IHRoZSBkcml2ZSBkb29yIGxvY2tpbmcgcHJvYmxlbXMuCiAqCiAqIDQuMDMgIERlYyAwNCwgMTk5NiAgLS0gQWRkZWQgRFNDIG92ZXJsYXAgc3VwcG9ydC4KICogNC4wNCAgRGVjIDI5LCAxOTk2ICAtLSBBZGRlZCBDRFJPTVJFQURSQVcgaW9jbHQgYmFzZWQgb24gcGF0Y2ggCiAqICAgICAgICAgICAgICAgICAgICAgICAgICBieSBBbGVzIE1ha2Fyb3YgKHhtYWthcm92QHN1bi5mZWxrLmN2dXQuY3opCiAqCiAqIDQuMDUgIE5vdiAyMCwgMTk5NyAgLS0gTW9kaWZpZWQgdG8gcHJpbnQgbW9yZSBkcml2ZSBpbmZvIG9uIGluaXQKICogICAgICAgICAgICAgICAgICAgICAgICBNaW5vciBvdGhlciBjaGFuZ2VzCiAqICAgICAgICAgICAgICAgICAgICAgICAgRml4IGVycm9ycyBvbiBDRFJPTVNUT1AgKElmIHlvdSBoYXZlIGEgIkRvbHBoaW4iLAogKiAgICAgICAgICAgICAgICAgICAgICAgICAgeW91IG11c3QgZGVmaW5lIElIQVZFQURPTFBISU4pCiAqICAgICAgICAgICAgICAgICAgICAgICAgQWRkZWQgaWRlbnRpZmllciBzbyBuZXcgU2FueW8gQ0QtY2hhbmdlciB3b3JrcwogKiAgICAgICAgICAgICAgICAgICAgICAgIEJldHRlciBkZXRlY3Rpb24gaWYgZG9vciBsb2NraW5nIGlzbid0IHN1cHBvcnRlZAogKgogKiA0LjA2ICBEZWMgMTcsIDE5OTcgIC0tIGZpeGVkIGVuZGxlc3MgInRyYXkgb3BlbiIgbWVzc2FnZXMgIC1tbAogKiA0LjA3ICBEZWMgMTcsIDE5OTcgIC0tIGZhbGxiYWNrIHRvIHNldCBwYy0+c3RhdCBvbiAidHJheSBvcGVuIgogKiA0LjA4ICBEZWMgMTgsIDE5OTcgIC0tIHNwZXcgbGVzcyBub2lzZSB3aGVuIHRyYXkgaXMgZW1wdHkKICogICAgICAgICAgICAgICAgICAgICAtLSBmaXggc3BlZWQgZGlzcGxheSBmb3IgQUNFUiAyNFgsIDE4WAogKiA0LjA5ICBKYW4gMDQsIDE5OTggIC0tIGZpeCBoYW5kbGluZyBvZiB0aGUgbGFzdCBibG9jayBzbyB3ZSByZXR1cm4KICogICAgICAgICAgICAgICAgICAgICAgICAgYW4gZW5kIG9mIGZpbGUgaW5zdGVhZCBvZiBhbiBJL08gZXJyb3IgKEdhZGkpCiAqIDQuMTAgIEphbiAyNCwgMTk5OCAgLS0gZml4ZWQgYSBidWcgc28gbm93IGNoYW5nZXJzIGNhbiBjaGFuZ2UgdG8gYSBuZXcKICogICAgICAgICAgICAgICAgICAgICAgICAgc2xvdCB3aGVuIHRoZXJlIGlzIG5vIGRpc2MgaW4gdGhlIGN1cnJlbnQgc2xvdC4KICogICAgICAgICAgICAgICAgICAgICAtLSBGaXhlZCBhIG1lbW9yeSBsZWFrIHdoZXJlIGluZm8tPmNoYW5nZXJfaW5mbyB3YXMKICogICAgICAgICAgICAgICAgICAgICAgICAgbWFsbG9jJ2VkIGJ1dCBuZXZlciBmcmVlJ2Qgd2hlbiBjbG9zaW5nIHRoZSBkZXZpY2UuCiAqICAgICAgICAgICAgICAgICAgICAgLS0gQ2xlYW5lZCB1cCB0aGUgZ2xvYmFsIG5hbWVzcGFjZSBhIGJpdCBieSBtYWtpbmcgbW9yZQogKiAgICAgICAgICAgICAgICAgICAgICAgICBmdW5jdGlvbnMgc3RhdGljIHRoYXQgc2hvdWxkIGFscmVhZHkgaGF2ZSBiZWVuLgogKiA0LjExICBNYXIgMTIsIDE5OTggIC0tIEFkZGVkIHN1cHBvcnQgZm9yIHRoZSBDRFJPTV9TRUxFQ1RfU1BFRUQgaW9jdGwKICogICAgICAgICAgICAgICAgICAgICAgICAgYmFzZWQgb24gYSBwYXRjaCBmb3IgMi4wLjMzIGJ5IEplbGxlIEZva3MgCiAqICAgICAgICAgICAgICAgICAgICAgICAgIDxqZWxsZUBzY2ludGlsbGEudXR3ZW50ZS5ubD4sIGEgcGF0Y2ggZm9yIDIuMC4zMwogKiAgICAgICAgICAgICAgICAgICAgICAgICBieSBUb25pIEdpb3JnaW5vIDx0b25pQHBjYXBlMi5waS5pbmZuLml0PiwgdGhlIFNDU0kKICogICAgICAgICAgICAgICAgICAgICAgICAgdmVyc2lvbiwgYW5kIG15IG93biBlZmZvcnRzLiAgLWVyaWsKICogICAgICAgICAgICAgICAgICAgICAtLSBGaXhlZCBhIHN0dXBpZCBidWcgd2hpY2ggZWdjcyB3YXMga2luZCBlbm91Z2ggdG8KICogICAgICAgICAgICAgICAgICAgICAgICAgaW5mb3JtIG1lIG9mIHdoZXJlICJJbGxlZ2FsIG1vZGUgZm9yIHRoaXMgdHJhY2siCiAqICAgICAgICAgICAgICAgICAgICAgICAgIHdhcyBuZXZlciByZXR1cm5lZCBkdWUgdG8gYSBjb21wYXJpc29uIG9uIGRhdGEKICogICAgICAgICAgICAgICAgICAgICAgICAgdHlwZXMgb2YgbGltaXRlZCByYW5nZS4KICogNC4xMiAgTWFyIDI5LCAxOTk4ICAtLSBGaXhlZCBidWcgaW4gQ0RST01fU0VMRUNUX1NQRUVEIHNvIHdyaXRlIHNwZWVkIGlzIAogKiAgICAgICAgICAgICAgICAgICAgICAgICBub3cgc2V0IGlvbmx5IGZvciBDRC1SIGFuZCBDRC1SVyBkcml2ZXMuICBJIGhhZCAKICogICAgICAgICAgICAgICAgICAgICAgICAgcmVtb3ZlZCB0aGlzIHN1cHBvcnQgYmVjYXVzZSBpdCBwcm9kdWNlZCBlcnJvcnMuCiAqICAgICAgICAgICAgICAgICAgICAgICAgIEl0IHByb2R1Y2VkIGVycm9ycyBfb25seV8gZm9yIG5vbi13cml0ZXJzLiBkdWguCiAqIDQuMTMgIE1heSAwNSwgMTk5OCAgLS0gU3VwcHJlc3MgdXNlbGVzcyAiaW4gcHJvZ3Jlc3Mgb2YgYmVjb21pbmcgcmVhZHkiCiAqICAgICAgICAgICAgICAgICAgICAgICAgIG1lc3NhZ2VzLCBzaW5jZSB0aGlzIGlzIG5vdCBhbiBlcnJvci4KICogICAgICAgICAgICAgICAgICAgICAtLSBDaGFuZ2UgZXJyb3IgbWVzc2FnZXMgdG8gYmUgY29uc3QKICogICAgICAgICAgICAgICAgICAgICAtLSBSZW1vdmUgYSAiXHQiIHdoaWNoIGxvb2tzIHVnbHkgaW4gdGhlIHN5c2xvZ3MKICogNC4xNCAgSnVseSAxNywgMTk5OCAtLSBDaGFuZ2UgdG8gcG9pbnRpbmcgdG8gLnBzIHZlcnNpb24gb2YgQVRBUEkgc3BlYwogKiAgICAgICAgICAgICAgICAgICAgICAgICBzaW5jZSB0aGUgLnBkZiB2ZXJzaW9uIGRvZXNuJ3Qgc2VlbSB0byB3b3JrLi4uCiAqICAgICAgICAgICAgICAgICAgICAgLS0gVXBkYXRlZCB0aGUgVE9ETyBsaXN0IHRvIHNvbWV0aGluZyBtb3JlIGN1cnJlbnQuCiAqCiAqIDQuMTUgIEF1ZyAyNSwgMTk5OCAgLS0gVXBkYXRlZCBpZGUtY2QuaCB0byByZXNwZWN0IG1lY2hpbmUgZW5kaWFuZXNzLCAKICogICAgICAgICAgICAgICAgICAgICAgICAgcGF0Y2ggdGhhbmtzIHRvICJFZGRpZSBDLiBEb3N0IiA8ZWNkQHNreW5ldC5iZT4KICoKICogNC41MCAgT2N0IDE5LCAxOTk4ICAtLSBOZXcgbWFpbnRhaW5lcnMhCiAqICAgICAgICAgICAgICAgICAgICAgICAgIEplbnMgQXhib2UgPGF4Ym9lQGltYWdlLmRrPgogKiAgICAgICAgICAgICAgICAgICAgICAgICBDaHJpcyBad2lsbGluZyA8Y2hyaXNAY2xvdWRuZXQuY29tPgogKgogKiA0LjUxICBEZWMgMjMsIDE5OTggIC0tIEplbnMgQXhib2UgPGF4Ym9lQGltYWdlLmRrPgogKiAgICAgICAgICAgICAgICAgICAgICAtIGlkZV9jZHJvbV9yZXNldCBlbmFibGVkIHNpbmNlIHRoZSBpZGUgc3Vic3lzdGVtCiAqICAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZXMgcmVzZXRzIGZpbmUgbm93LiA8YXhib2VAaW1hZ2UuZGs+CiAqICAgICAgICAgICAgICAgICAgICAgIC0gVHJhbnNmZXIgc2l6ZSBmaXggZm9yIFNhbXN1bmcgQ0QtUk9NcywgdGhhbmtzIHRvCiAqICAgICAgICAgICAgICAgICAgICAgICAgIlZpbGxlIEhhbGxpayIgPHZpbGxlLmhhbGxpa0BtYWlsLmVlPi4KICogICAgICAgICAgICAgICAgICAgICAgLSBvdGhlciBtaW5vciBzdHVmZi4KICoKICogNC41MiAgSmFuIDE5LCAxOTk5ICAtLSBKZW5zIEF4Ym9lIDxheGJvZUBpbWFnZS5kaz4KICogICAgICAgICAgICAgICAgICAgICAgLSBEZXRlY3QgRFZELVJPTS9SQU0gZHJpdmVzCiAqCiAqIDQuNTMgIEZlYiAyMiwgMTk5OSAgIC0gSW5jbHVkZSBvdGhlciBtb2RlbCBTYW1zdW5nIGFuZCBvbmUgR29sZHN0YXIKICogICAgICAgICAgICAgICAgICAgICAgICAgZHJpdmUgaW4gdHJhbnNmZXIgc2l6ZSBsaW1pdC4KICogICAgICAgICAgICAgICAgICAgICAgLSBGaXggdGhlIEkvTyBlcnJvciB3aGVuIGRvaW5nIGVqZWN0IHdpdGhvdXQgYSBtZWRpdW0KICogICAgICAgICAgICAgICAgICAgICAgICAgbG9hZGVkIG9uIHNvbWUgZHJpdmVzLgogKiAgICAgICAgICAgICAgICAgICAgICAtIENEUk9NUkVBRE1PREUyIGlzIG5vdyBpbXBsZW1lbnRlZCB0aHJvdWdoCiAqICAgICAgICAgICAgICAgICAgICAgICAgIENEUk9NUkVBRFJBVywgc2luY2UgbWFueSBkcml2ZXMgZG9uJ3Qgc3VwcG9ydAogKiAgICAgICAgICAgICAgICAgICAgICAgICBNT0RFMiAoZXZlbiB0aG91Z2ggQVRBUEkgMi42IHNheXMgdGhleSBtdXN0KS4KICogICAgICAgICAgICAgICAgICAgICAgLSBBZGRlZCBpZ25vcmUgcGFyYW1ldGVyIHRvIGlkZS1jZCAoYXMgYSBtb2R1bGUpLCBlZwogKiAgICAgICAgICAgICAgICAgICAgICAgICAJaW5zbW9kIGlkZS1jZCBpZ25vcmU9J2hkYSBoZGInCiAqICAgICAgICAgICAgICAgICAgICAgICAgIFVzZWZ1bCB3aGVuIHVzaW5nIGlkZS1jZCBpbiBjb25qdW5jdGlvbiB3aXRoCiAqICAgICAgICAgICAgICAgICAgICAgICAgIGlkZS1zY3NpLiBUT0RPOiBub24tbW9kdWxhciB3YXkgb2YgZG9pbmcgdGhlCiAqICAgICAgICAgICAgICAgICAgICAgICAgIHNhbWUuCiAqCiAqIDQuNTQgIEF1ZyA1LCAxOTk5CS0gU3VwcG9ydCBmb3IgTU1DMiBjbGFzcyBjb21tYW5kcyB0aHJvdWdoIHRoZSBnZW5lcmljCiAqCQkJICBwYWNrZXQgaW50ZXJmYWNlIHRvIGNkcm9tLmMuCiAqCQkJLSBVbmlmaWVkIGF1ZGlvIGlvY3RsIHN1cHBvcnQsIG1vc3Qgb2YgaXQuCiAqCQkJLSBjbGVhbmVkIHVwIHZhcmlvdXMgZGVwcmVjYXRlZCB2ZXJpZnlfYXJlYSgpLgogKgkJCS0gQWRkZWQgaWRlX2Nkcm9tX3BhY2tldCgpIGFzIHRoZSBpbnRlcmZhY2UgZm9yCiAqCQkJICB0aGUgVW5pZm9ybSBnZW5lcmljX3BhY2tldCgpLgogKgkJCS0gYnVuY2ggb2Ygb3RoZXIgc3R1ZmYsIHdpbGwgZmlsbCBpbiBsb2dzIGxhdGVyLgogKgkJCS0gcmVwb3J0IDEgc2xvdCBmb3Igbm9uLWNoYW5nZXJzLCBsaWtlIHRoZSBvdGhlcgogKgkJCSAgY2Qtcm9tIGRyaXZlcnMuIGRvbid0IHJlcG9ydCBzZWxlY3QgZGlzYyBmb3IKICoJCQkgIG5vbi1jaGFuZ2VycyBhcyB3ZWxsLgogKgkJCS0gbWFzayBvdXQgYXVkaW8gcGxheWluZywgaWYgdGhlIGRldmljZSBjYW4ndCBkbyBpdC4KICoKICogNC41NSAgU2VwIDEsIDE5OTkJLSBFbGltaW5hdGVkIHRoZSByZXN0IG9mIHRoZSBhdWRpbyBpb2N0bHMsIGV4Y2VwdAogKgkJCSAgZm9yIENEUk9NUkVBRFRPQ1tFTlRSWXxIRUFERVJdLiBTb21lIG9mIHRoZSBkcml2ZXJzCiAqCQkJICB1c2UgdGhpcyBpbmRlcGVuZGVudGx5IG9mIHRoZSBhY3R1YWwgYXVkaW8gaGFuZGxpbmcuCiAqCQkJICBUaGV5IHdpbGwgZGlzYXBwZWFyIGxhdGVyIHdoZW4gSSBnZXQgdGhlIHRpbWUgdG8KICoJCQkgIGRvIGl0IGNsZWFubHkuCiAqCQkJLSBNaW5pbWl6ZSB0aGUgVE9DIHJlYWRpbmcgLSBvbmx5IGRvIGl0IHdoZW4gd2UKICoJCQkgIGtub3cgYSBtZWRpYSBjaGFuZ2UgaGFzIG9jY3VycmVkLgogKgkJCS0gTW92ZWQgYWxsIHRoZSBDRFJPTVJFQUR4IGlvY3RscyB0byB0aGUgVW5pZm9ybSBsYXllci4KICoJCQktIEhlaWtvIEVpc3NmZWxkdCA8aGVpa29AY29sb3NzdXMuZXNjYXBlLmRlPiBzdXBwbGllZAogKgkJCSAgc29tZSBmaXhlcyBmb3IgQ0RJLgogKgkJCS0gQ0QtUk9NIGxlYXZpbmcgZG9vciBsb2NrZWQgZml4IGZyb20gQW5kcmllcwogKgkJCSAgQnJvdXdlciA8QW5kcmllcy5Ccm91d2VyQGN3aS5ubD4KICoJCQktIEVyaWsgQW5kZXJzZW4gPGFuZGVyc2VuQHhtaXNzaW9uLmNvbT4gdW5pZmllZAogKgkJCSAgY29tbWFuZHMgYWNyb3NzIHRoZSB2YXJpb3VzIGRyaXZlcnMgYW5kIGhvdwogKgkJCSAgc2Vuc2UgZXJyb3JzIGFyZSBoYW5kbGVkLgogKgogKiA0LjU2ICBTZXAgMTIsIDE5OTkJLSBSZW1vdmVkIGNoYW5nZXIgc3VwcG9ydCAtIGl0IGlzIG5vdyBpbiB0aGUKICoJCQkgIFVuaWZvcm0gbGF5ZXIuCiAqCQkJLSBBZGRlZCBwYXJ0aXRpb24gYmFzZWQgbXVsdGlzZXNzaW9uIGhhbmRsaW5nLgogKgkJCS0gTW9kZSBzZW5zZSBhbmQgbW9kZSBzZWxlY3QgbW92ZWQgdG8gdGhlCiAqCQkJICBVbmlmb3JtIGxheWVyLgogKgkJCS0gRml4ZWQgYSBwcm9ibGVtIHdpdGggV1BJIENEUy0zMlggZHJpdmUgLSBpdAogKgkJCSAgZmFpbGVkIHRoZSBjYXBhYmlsaXRpZXMgCiAqCiAqIDQuNTcgIEFwciA3LCAyMDAwCS0gRml4ZWQgc2Vuc2UgcmVwb3J0aW5nLgogKgkJCS0gRml4ZWQgcG9zc2libGUgb29wcyBpbiBpZGVfY2Ryb21fZ2V0X2xhc3Rfc2Vzc2lvbigpCiAqCQkJLSBGaXggbG9ja2luZyBtYW5pYSBhbmQgbWFrZSBpZGVfY2Ryb21fcmVzZXQgcmVsb2NrCiAqCQkJLSBTdG9wIHNwZXdpbmcgZXJyb3JzIHRvIGxvZyB3aGVuIG1hZ2ljZGV2IHBvbGxzIHdpdGgKICoJCQkgIFRFU1RfVU5JVF9SRUFEWSBvbiBzb21lIGRyaXZlcy4KICoJCQktIFZhcmlvdXMgZml4ZXMgZnJvbSBUb2JpYXMgUmluZ3N0cm9tOgogKgkJCSAgdHJheSBpZiBpdCB3YXMgbG9ja2VkIHByaW9yIHRvIHRoZSByZXNldC4KICoJCQkgIC0gY2Ryb21fcmVhZF9jYXBhY2l0eSByZXR1cm5zIG9uZSBmcmFtZSB0b28gbGl0dGxlLgogKgkJCSAgLSBGaXggcmVhbCBjYXBhY2l0eSByZXBvcnRpbmcuCiAqCiAqIDQuNTggIE1heSAxLCAyMDAwCS0gQ2xlYW4gdXAgQUNFUjUwIHN0dWZmLgogKgkJCS0gRml4IHNtYWxsIHByb2JsZW0gd2l0aCBpZGVfY2Ryb21fY2FwYWNpdHkKICoKICogNC41OSAgQXVnIDExLCAyMDAwCS0gRml4IGNoYW5nZXIgcHJvYmxlbSBpbiBjZHJvbV9yZWFkX3RvYywgd2Ugd2VyZW4ndAogKgkJCSAgY29ycmVjdGx5IHNlbnNpbmcgYSBkaXNjIGNoYW5nZS4KICoJCQktIFJlYXJyYW5nZWQgc29tZSBjb2RlCiAqCQkJLSBVc2UgZXh0ZW5kZWQgc2Vuc2Ugb24gZHJpdmVzIHRoYXQgc3VwcG9ydCBpdCBmb3IKICoJCQkgIGNvcnJlY3RseSByZXBvcnRpbmcgdHJheSBzdGF0dXMgLS0gZnJvbQogKgkJCSAgTWljaGFlbCBEIEpvaG5zb24gPGpvaG5zb21Ab3JzdC5lZHU+CiAqIDQuNjAgIERlYyAxNywgMjAwMwktIEFkZCBtdCByYWluaWVyIHN1cHBvcnQKICoJCQktIEJ1bXAgdGltZW91dCBmb3IgcGFja2V0IGNvbW1hbmRzLCBtYXRjaGVzIHNyCiAqCQkJLSBPZGQgc3R1ZmYKICogNC42MSAgSmFuIDIyLCAyMDA0CS0gc3VwcG9ydCBoYXJkd2FyZSBzZWN0b3Igc2l6ZXMgb3RoZXIgdGhhbiAya0IsCiAqCQkJICBQYXNjYWwgU2NobWlkdCA8ZGVyLmVyZW1pdEBlbWFpbC5kZT4KICoKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCiAKI2RlZmluZSBJREVDRF9WRVJTSU9OICI0LjYxIgoKI2luY2x1ZGUgPGxpbnV4L2NvbmZpZy5oPgojaW5jbHVkZSA8bGludXgvbW9kdWxlLmg+CiNpbmNsdWRlIDxsaW51eC90eXBlcy5oPgojaW5jbHVkZSA8bGludXgva2VybmVsLmg+CiNpbmNsdWRlIDxsaW51eC9kZWxheS5oPgojaW5jbHVkZSA8bGludXgvdGltZXIuaD4KI2luY2x1ZGUgPGxpbnV4L3NsYWIuaD4KI2luY2x1ZGUgPGxpbnV4L2ludGVycnVwdC5oPgojaW5jbHVkZSA8bGludXgvZXJybm8uaD4KI2luY2x1ZGUgPGxpbnV4L2Nkcm9tLmg+CiNpbmNsdWRlIDxsaW51eC9pZGUuaD4KI2luY2x1ZGUgPGxpbnV4L2NvbXBsZXRpb24uaD4KCiNpbmNsdWRlIDxzY3NpL3Njc2kuaD4JLyogRm9yIFNDU0kgLT4gQVRBUEkgY29tbWFuZCBjb252ZXJzaW9uICovCgojaW5jbHVkZSA8YXNtL2lycS5oPgojaW5jbHVkZSA8YXNtL2lvLmg+CiNpbmNsdWRlIDxhc20vYnl0ZW9yZGVyLmg+CiNpbmNsdWRlIDxhc20vdWFjY2Vzcy5oPgojaW5jbHVkZSA8YXNtL3VuYWxpZ25lZC5oPgoKI2luY2x1ZGUgImlkZS1jZC5oIgoKc3RhdGljIERFQ0xBUkVfTVVURVgoaWRlY2RfcmVmX3NlbSk7CgojZGVmaW5lIHRvX2lkZV9jZChvYmopIGNvbnRhaW5lcl9vZihvYmosIHN0cnVjdCBjZHJvbV9pbmZvLCBrcmVmKSAKCiNkZWZpbmUgaWRlX2NkX2coZGlzaykgXAoJY29udGFpbmVyX29mKChkaXNrKS0+cHJpdmF0ZV9kYXRhLCBzdHJ1Y3QgY2Ryb21faW5mbywgZHJpdmVyKQoKc3RhdGljIHN0cnVjdCBjZHJvbV9pbmZvICppZGVfY2RfZ2V0KHN0cnVjdCBnZW5kaXNrICpkaXNrKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqY2QgPSBOVUxMOwoKCWRvd24oJmlkZWNkX3JlZl9zZW0pOwoJY2QgPSBpZGVfY2RfZyhkaXNrKTsKCWlmIChjZCkKCQlrcmVmX2dldCgmY2QtPmtyZWYpOwoJdXAoJmlkZWNkX3JlZl9zZW0pOwoJcmV0dXJuIGNkOwp9CgpzdGF0aWMgdm9pZCBpZGVfY2RfcmVsZWFzZShzdHJ1Y3Qga3JlZiAqKTsKCnN0YXRpYyB2b2lkIGlkZV9jZF9wdXQoc3RydWN0IGNkcm9tX2luZm8gKmNkKQp7Cglkb3duKCZpZGVjZF9yZWZfc2VtKTsKCWtyZWZfcHV0KCZjZC0+a3JlZiwgaWRlX2NkX3JlbGVhc2UpOwoJdXAoJmlkZWNkX3JlZl9zZW0pOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBHZW5lcmljIHBhY2tldCBjb21tYW5kIHN1cHBvcnQgYW5kIGVycm9yIGhhbmRsaW5nIHJvdXRpbmVzLgogKi8KCi8qIE1hcmsgdGhhdCB3ZSd2ZSBzZWVuIGEgbWVkaWEgY2hhbmdlLCBhbmQgaW52YWxpZGF0ZSBvdXIgaW50ZXJuYWwKICAgYnVmZmVycy4gKi8Kc3RhdGljIHZvaWQgY2Ryb21fc2F3X21lZGlhX2NoYW5nZSAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCQoJQ0RST01fU1RBVEVfRkxBR1MgKGRyaXZlKS0+bWVkaWFfY2hhbmdlZCA9IDE7CglDRFJPTV9TVEFURV9GTEFHUyAoZHJpdmUpLT50b2NfdmFsaWQgPSAwOwoJaW5mby0+bnNlY3RvcnNfYnVmZmVyZWQgPSAwOwp9CgpzdGF0aWMgaW50IGNkcm9tX2xvZ19zZW5zZShpZGVfZHJpdmVfdCAqZHJpdmUsIHN0cnVjdCByZXF1ZXN0ICpycSwKCQkJICAgc3RydWN0IHJlcXVlc3Rfc2Vuc2UgKnNlbnNlKQp7CglpbnQgbG9nID0gMDsKCglpZiAoIXNlbnNlIHx8ICFycSB8fCAocnEtPmZsYWdzICYgUkVRX1FVSUVUKSkKCQlyZXR1cm4gMDsKCglzd2l0Y2ggKHNlbnNlLT5zZW5zZV9rZXkpIHsKCQljYXNlIE5PX1NFTlNFOiBjYXNlIFJFQ09WRVJFRF9FUlJPUjoKCQkJYnJlYWs7CgkJY2FzZSBOT1RfUkVBRFk6CgkJCS8qCgkJCSAqIGRvbid0IGNhcmUgYWJvdXQgdHJheSBzdGF0ZSBtZXNzYWdlcyBmb3IKCQkJICogZS5nLiBjYXBhY2l0eSBjb21tYW5kcyBvciBpbi1wcm9ncmVzcyBvcgoJCQkgKiBiZWNvbWluZyByZWFkeQoJCQkgKi8KCQkJaWYgKHNlbnNlLT5hc2MgPT0gMHgzYSB8fCBzZW5zZS0+YXNjID09IDB4MDQpCgkJCQlicmVhazsKCQkJbG9nID0gMTsKCQkJYnJlYWs7CgkJY2FzZSBJTExFR0FMX1JFUVVFU1Q6CgkJCS8qCgkJCSAqIGRvbid0IGxvZyBTVEFSVF9TVE9QIHVuaXQgd2l0aCBMb0VqIHNldCwgc2luY2UKCQkJICogd2UgY2Fubm90IHJlbGlhYmx5IGNoZWNrIGlmIGRyaXZlIGNhbiBhdXRvLWNsb3NlCgkJCSAqLwoJCQlpZiAocnEtPmNtZFswXSA9PSBHUENNRF9TVEFSVF9TVE9QX1VOSVQgJiYgc2Vuc2UtPmFzYyA9PSAweDI0KQoJCQkJbG9nID0gMDsKCQkJYnJlYWs7CgkJY2FzZSBVTklUX0FUVEVOVElPTjoKCQkJLyoKCQkJICogTWFrZSBnb29kIGFuZCBzdXJlIHdlJ3ZlIHNlZW4gdGhpcyBwb3RlbnRpYWwgbWVkaWEKCQkJICogY2hhbmdlLiBTb21lIGRyaXZlcyAoaS5lLiBDcmVhdGl2ZSkgZmFpbCB0byBwcmVzZW50CgkJCSAqIHRoZSBjb3JyZWN0IHNlbnNlIGtleSBpbiB0aGUgZXJyb3IgcmVnaXN0ZXIuCgkJCSAqLwoJCQljZHJvbV9zYXdfbWVkaWFfY2hhbmdlKGRyaXZlKTsKCQkJYnJlYWs7CgkJZGVmYXVsdDoKCQkJbG9nID0gMTsKCQkJYnJlYWs7Cgl9CglyZXR1cm4gbG9nOwp9CgpzdGF0aWMKdm9pZCBjZHJvbV9hbmFseXplX3NlbnNlX2RhdGEoaWRlX2RyaXZlX3QgKmRyaXZlLAoJCQkgICAgICBzdHJ1Y3QgcmVxdWVzdCAqZmFpbGVkX2NvbW1hbmQsCgkJCSAgICAgIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJaWYgKCFjZHJvbV9sb2dfc2Vuc2UoZHJpdmUsIGZhaWxlZF9jb21tYW5kLCBzZW5zZSkpCgkJcmV0dXJuOwoKCS8qCgkgKiBJZiBhIHJlYWQgdG9jIGlzIGV4ZWN1dGVkIGZvciBhIENELVIgb3IgQ0QtUlcgbWVkaXVtIHdoZXJlCgkgKiB0aGUgZmlyc3QgdG9jIGhhcyBub3QgYmVlbiByZWNvcmRlZCB5ZXQsIGl0IHdpbGwgZmFpbCB3aXRoCgkgKiAwNS8yNC8wMCAod2hpY2ggaXMgYSBjb25mdXNpbmcgZXJyb3IpCgkgKi8KCWlmIChmYWlsZWRfY29tbWFuZCAmJiBmYWlsZWRfY29tbWFuZC0+Y21kWzBdID09IEdQQ01EX1JFQURfVE9DX1BNQV9BVElQKQoJCWlmIChzZW5zZS0+c2Vuc2Vfa2V5ID09IDB4MDUgJiYgc2Vuc2UtPmFzYyA9PSAweDI0KQoJCQlyZXR1cm47CgojaWYgVkVSQk9TRV9JREVfQ0RfRVJST1JTCgl7CgkJaW50IGk7CgkJY29uc3QgY2hhciAqcyA9ICJiYWQgc2Vuc2Uga2V5ISI7CgkJY2hhciBidWZbODBdOwoKCQlwcmludGsgKCJBVEFQSSBkZXZpY2UgJXM6XG4iLCBkcml2ZS0+bmFtZSk7CgkJaWYgKHNlbnNlLT5lcnJvcl9jb2RlPT0weDcwKQoJCQlwcmludGsoIiAgRXJyb3I6ICIpOwoJCWVsc2UgaWYgKHNlbnNlLT5lcnJvcl9jb2RlPT0weDcxKQoJCQlwcmludGsoIiAgRGVmZXJyZWQgRXJyb3I6ICIpOwoJCWVsc2UgaWYgKHNlbnNlLT5lcnJvcl9jb2RlID09IDB4N2YpCgkJCXByaW50aygiICBWZW5kb3Itc3BlY2lmaWMgRXJyb3I6ICIpOwoJCWVsc2UKCQkJcHJpbnRrKCIgIFVua25vd24gRXJyb3IgVHlwZTogIik7CgoJCWlmIChzZW5zZS0+c2Vuc2Vfa2V5IDwgQVJZX0xFTihzZW5zZV9rZXlfdGV4dHMpKQoJCQlzID0gc2Vuc2Vfa2V5X3RleHRzW3NlbnNlLT5zZW5zZV9rZXldOwoKCQlwcmludGsoIiVzIC0tIChTZW5zZSBrZXk9MHglMDJ4KVxuIiwgcywgc2Vuc2UtPnNlbnNlX2tleSk7CgoJCWlmIChzZW5zZS0+YXNjID09IDB4NDApIHsKCQkJc3ByaW50ZihidWYsICJEaWFnbm9zdGljIGZhaWx1cmUgb24gY29tcG9uZW50IDB4JTAyeCIsCgkJCQkgc2Vuc2UtPmFzY3EpOwoJCQlzID0gYnVmOwoJCX0gZWxzZSB7CgkJCWludCBsbyA9IDAsIG1pZCwgaGkgPSBBUllfTEVOKHNlbnNlX2RhdGFfdGV4dHMpOwoJCQl1bnNpZ25lZCBsb25nIGtleSA9IChzZW5zZS0+c2Vuc2Vfa2V5IDw8IDE2KTsKCQkJa2V5IHw9IChzZW5zZS0+YXNjIDw8IDgpOwoJCQlpZiAoIShzZW5zZS0+YXNjcSA+PSAweDgwICYmIHNlbnNlLT5hc2NxIDw9IDB4ZGQpKQoJCQkJa2V5IHw9IHNlbnNlLT5hc2NxOwoJCQlzID0gTlVMTDsKCgkJCXdoaWxlIChoaSA+IGxvKSB7CgkJCQltaWQgPSAobG8gKyBoaSkgLyAyOwoJCQkJaWYgKHNlbnNlX2RhdGFfdGV4dHNbbWlkXS5hc2NfYXNjcSA9PSBrZXkgfHwKCQkJCSAgICBzZW5zZV9kYXRhX3RleHRzW21pZF0uYXNjX2FzY3EgPT0gKDB4ZmYwMDAwfGtleSkpIHsKCQkJCQlzID0gc2Vuc2VfZGF0YV90ZXh0c1ttaWRdLnRleHQ7CgkJCQkJYnJlYWs7CgkJCQl9CgkJCQllbHNlIGlmIChzZW5zZV9kYXRhX3RleHRzW21pZF0uYXNjX2FzY3EgPiBrZXkpCgkJCQkJaGkgPSBtaWQ7CgkJCQllbHNlCgkJCQkJbG8gPSBtaWQrMTsKCQkJfQoJCX0KCgkJaWYgKHMgPT0gTlVMTCkgewoJCQlpZiAoc2Vuc2UtPmFzYyA+IDB4ODApCgkJCQlzID0gIih2ZW5kb3Itc3BlY2lmaWMgZXJyb3IpIjsKCQkJZWxzZQoJCQkJcyA9ICIocmVzZXJ2ZWQgZXJyb3IgY29kZSkiOwoJCX0KCgkJcHJpbnRrKEtFUk5fRVJSICIgICVzIC0tIChhc2M9MHglMDJ4LCBhc2NxPTB4JTAyeClcbiIsCgkJCXMsIHNlbnNlLT5hc2MsIHNlbnNlLT5hc2NxKTsKCgkJaWYgKGZhaWxlZF9jb21tYW5kICE9IE5VTEwpIHsKCgkJCWludCBsbz0wLCBtaWQsIGhpPSBBUllfTEVOIChwYWNrZXRfY29tbWFuZF90ZXh0cyk7CgkJCXMgPSBOVUxMOwoKCQkJd2hpbGUgKGhpID4gbG8pIHsKCQkJCW1pZCA9IChsbyArIGhpKSAvIDI7CgkJCQlpZiAocGFja2V0X2NvbW1hbmRfdGV4dHNbbWlkXS5wYWNrZXRfY29tbWFuZCA9PQoJCQkJICAgIGZhaWxlZF9jb21tYW5kLT5jbWRbMF0pIHsKCQkJCQlzID0gcGFja2V0X2NvbW1hbmRfdGV4dHNbbWlkXS50ZXh0OwoJCQkJCWJyZWFrOwoJCQkJfQoJCQkJaWYgKHBhY2tldF9jb21tYW5kX3RleHRzW21pZF0ucGFja2V0X2NvbW1hbmQgPgoJCQkJICAgIGZhaWxlZF9jb21tYW5kLT5jbWRbMF0pCgkJCQkJaGkgPSBtaWQ7CgkJCQllbHNlCgkJCQkJbG8gPSBtaWQrMTsKCQkJfQoKCQkJcHJpbnRrIChLRVJOX0VSUiAiICBUaGUgZmFpbGVkIFwiJXNcIiBwYWNrZXQgY29tbWFuZCB3YXM6IFxuICBcIiIsIHMpOwoJCQlmb3IgKGk9MDsgaTxzaXplb2YgKGZhaWxlZF9jb21tYW5kLT5jbWQpOyBpKyspCgkJCQlwcmludGsgKCIlMDJ4ICIsIGZhaWxlZF9jb21tYW5kLT5jbWRbaV0pOwoJCQlwcmludGsgKCJcIlxuIik7CgkJfQoKCQkvKiBUaGUgU0tTViBiaXQgc3BlY2lmaWVzIHZhbGlkaXR5IG9mIHRoZSBzZW5zZV9rZXlfc3BlY2lmaWMKCQkgKiBpbiB0aGUgbmV4dCB0d28gY29tbWFuZHMuIEl0IGlzIGJpdCA3IG9mIHRoZSBmaXJzdCBieXRlLgoJCSAqIEluIHRoZSBjYXNlIG9mIE5PVF9SRUFEWSwgaWYgU0tTViBpcyBzZXQgdGhlIGRyaXZlIGNhbgoJCSAqIGdpdmUgdXMgbmljZSBFVEEgcmVhZGluZ3MuCgkJICovCgkJaWYgKHNlbnNlLT5zZW5zZV9rZXkgPT0gTk9UX1JFQURZICYmIChzZW5zZS0+c2tzWzBdICYgMHg4MCkpIHsKCQkJaW50IHByb2dyZXNzID0gKHNlbnNlLT5za3NbMV0gPDwgOCB8IHNlbnNlLT5za3NbMl0pICogMTAwOwoJCQlwcmludGsoS0VSTl9FUlIgIiAgQ29tbWFuZCBpcyAlMDJkJSUgY29tcGxldGVcbiIsIHByb2dyZXNzIC8gMHhmZmZmKTsKCgkJfQoKCQlpZiAoc2Vuc2UtPnNlbnNlX2tleSA9PSBJTExFR0FMX1JFUVVFU1QgJiYKCQkgICAgKHNlbnNlLT5za3NbMF0gJiAweDgwKSAhPSAwKSB7CgkJCXByaW50ayhLRVJOX0VSUiAiICBFcnJvciBpbiAlcyBieXRlICVkIiwKCQkJCShzZW5zZS0+c2tzWzBdICYgMHg0MCkgIT0gMCA/CgkJCQkiY29tbWFuZCBwYWNrZXQiIDogImNvbW1hbmQgZGF0YSIsCgkJCQkoc2Vuc2UtPnNrc1sxXSA8PCA4KSArIHNlbnNlLT5za3NbMl0pOwoKCQkJaWYgKChzZW5zZS0+c2tzWzBdICYgMHg0MCkgIT0gMCkKCQkJCXByaW50ayAoIiBiaXQgJWQiLCBzZW5zZS0+c2tzWzBdICYgMHgwNyk7CgoJCQlwcmludGsgKCJcbiIpOwoJCX0KCX0KCiNlbHNlIC8qIG5vdCBWRVJCT1NFX0lERV9DRF9FUlJPUlMgKi8KCgkvKiBTdXBwcmVzcyBwcmludGluZyB1bml0IGF0dGVudGlvbiBhbmQgYGluIHByb2dyZXNzIG9mIGJlY29taW5nIHJlYWR5JwoJICAgZXJyb3JzIHdoZW4gd2UncmUgbm90IGJlaW5nIHZlcmJvc2UuICovCgoJaWYgKHNlbnNlLT5zZW5zZV9rZXkgPT0gVU5JVF9BVFRFTlRJT04gfHwKCSAgICAoc2Vuc2UtPnNlbnNlX2tleSA9PSBOT1RfUkVBRFkgJiYgKHNlbnNlLT5hc2MgPT0gNCB8fAoJCQkJCQlzZW5zZS0+YXNjID09IDB4M2EpKSkKCQlyZXR1cm47CgoJcHJpbnRrKEtFUk5fRVJSICIlczogZXJyb3IgY29kZTogMHglMDJ4ICBzZW5zZV9rZXk6IDB4JTAyeCAgYXNjOiAweCUwMnggIGFzY3E6IDB4JTAyeFxuIiwKCQlkcml2ZS0+bmFtZSwKCQlzZW5zZS0+ZXJyb3JfY29kZSwgc2Vuc2UtPnNlbnNlX2tleSwKCQlzZW5zZS0+YXNjLCBzZW5zZS0+YXNjcSk7CiNlbmRpZiAvKiBub3QgVkVSQk9TRV9JREVfQ0RfRVJST1JTICovCn0KCi8qCiAqIEluaXRpYWxpemUgYSBpZGUtY2QgcGFja2V0IGNvbW1hbmQgcmVxdWVzdAogKi8Kc3RhdGljIHZvaWQgY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGlkZV9kcml2ZV90ICpkcml2ZSwgc3RydWN0IHJlcXVlc3QgKnJxKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqY2QgPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJaWRlX2luaXRfZHJpdmVfY21kKHJxKTsKCXJxLT5mbGFncyA9IFJFUV9QQzsKCXJxLT5ycV9kaXNrID0gY2QtPmRpc2s7Cn0KCnN0YXRpYyB2b2lkIGNkcm9tX3F1ZXVlX3JlcXVlc3Rfc2Vuc2UoaWRlX2RyaXZlX3QgKmRyaXZlLCB2b2lkICpzZW5zZSwKCQkJCSAgICAgIHN0cnVjdCByZXF1ZXN0ICpmYWlsZWRfY29tbWFuZCkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8JCT0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IHJlcXVlc3QgKnJxCQk9ICZpbmZvLT5yZXF1ZXN0X3NlbnNlX3JlcXVlc3Q7CgoJaWYgKHNlbnNlID09IE5VTEwpCgkJc2Vuc2UgPSAmaW5mby0+c2Vuc2VfZGF0YTsKCgkvKiBzdHVmZiB0aGUgc2Vuc2UgcmVxdWVzdCBpbiBmcm9udCBvZiBvdXIgY3VycmVudCByZXF1ZXN0ICovCgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsIHJxKTsKCglycS0+ZGF0YSA9IHNlbnNlOwoJcnEtPmNtZFswXSA9IEdQQ01EX1JFUVVFU1RfU0VOU0U7CglycS0+Y21kWzRdID0gcnEtPmRhdGFfbGVuID0gMTg7CgoJcnEtPmZsYWdzID0gUkVRX1NFTlNFOwoKCS8qIE5PVEUhIFNhdmUgdGhlIGZhaWxlZCBjb21tYW5kIGluICJycS0+YnVmZmVyIiAqLwoJcnEtPmJ1ZmZlciA9ICh2b2lkICopIGZhaWxlZF9jb21tYW5kOwoKCSh2b2lkKSBpZGVfZG9fZHJpdmVfY21kKGRyaXZlLCBycSwgaWRlX3ByZWVtcHQpOwp9CgpzdGF0aWMgdm9pZCBjZHJvbV9lbmRfcmVxdWVzdCAoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgdXB0b2RhdGUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCWludCBuc2VjdG9ycyA9IHJxLT5oYXJkX2N1cl9zZWN0b3JzOwoKCWlmICgocnEtPmZsYWdzICYgUkVRX1NFTlNFKSAmJiB1cHRvZGF0ZSkgewoJCS8qCgkJICogRm9yIFJFUV9TRU5TRSwgInJxLT5idWZmZXIiIHBvaW50cyB0byB0aGUgb3JpZ2luYWwgZmFpbGVkCgkJICogcmVxdWVzdAoJCSAqLwoJCXN0cnVjdCByZXF1ZXN0ICpmYWlsZWQgPSAoc3RydWN0IHJlcXVlc3QgKikgcnEtPmJ1ZmZlcjsKCQlzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCQl2b2lkICpzZW5zZSA9ICZpbmZvLT5zZW5zZV9kYXRhOwoJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJCWlmIChmYWlsZWQpIHsKCQkJaWYgKGZhaWxlZC0+c2Vuc2UpIHsKCQkJCXNlbnNlID0gZmFpbGVkLT5zZW5zZTsKCQkJCWZhaWxlZC0+c2Vuc2VfbGVuID0gcnEtPnNlbnNlX2xlbjsKCQkJfQoKCQkJLyoKCQkJICogbm93IGVuZCBmYWlsZWQgcmVxdWVzdAoJCQkgKi8KCQkJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CgkJCWVuZF90aGF0X3JlcXVlc3RfY2h1bmsoZmFpbGVkLCAwLCBmYWlsZWQtPmRhdGFfbGVuKTsKCQkJZW5kX3RoYXRfcmVxdWVzdF9sYXN0KGZhaWxlZCwgMCk7CgkJCXNwaW5fdW5sb2NrX2lycXJlc3RvcmUoJmlkZV9sb2NrLCBmbGFncyk7CgkJfQoKCQljZHJvbV9hbmFseXplX3NlbnNlX2RhdGEoZHJpdmUsIGZhaWxlZCwgc2Vuc2UpOwoJfQoKCWlmICghcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyAmJiBibGtfZnNfcmVxdWVzdChycSkpCgkJdXB0b2RhdGUgPSAxOwoJLyogbWFrZSBzdXJlIGl0J3MgZnVsbHkgZW5kZWQgKi8KCWlmIChibGtfcGNfcmVxdWVzdChycSkpCgkJbnNlY3RvcnMgPSAocnEtPmRhdGFfbGVuICsgNTExKSA+PiA5OwoJaWYgKCFuc2VjdG9ycykKCQluc2VjdG9ycyA9IDE7CgoJaWRlX2VuZF9yZXF1ZXN0KGRyaXZlLCB1cHRvZGF0ZSwgbnNlY3RvcnMpOwp9CgovKiBSZXR1cm5zIDAgaWYgdGhlIHJlcXVlc3Qgc2hvdWxkIGJlIGNvbnRpbnVlZC4KICAgUmV0dXJucyAxIGlmIHRoZSByZXF1ZXN0IHdhcyBlbmRlZC4gKi8Kc3RhdGljIGludCBjZHJvbV9kZWNvZGVfc3RhdHVzKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IGdvb2Rfc3RhdCwgaW50ICpzdGF0X3JldCkKewoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJaW50IHN0YXQsIGVyciwgc2Vuc2Vfa2V5OwoJCgkvKiBDaGVjayBmb3IgZXJyb3JzLiAqLwoJc3RhdCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX1NUQVRVU19SRUcpOwoJaWYgKHN0YXRfcmV0KQoJCSpzdGF0X3JldCA9IHN0YXQ7CgoJaWYgKE9LX1NUQVQoc3RhdCwgZ29vZF9zdGF0LCBCQURfUl9TVEFUKSkKCQlyZXR1cm4gMDsKCgkvKiBHZXQgdGhlIElERSBlcnJvciByZWdpc3Rlci4gKi8KCWVyciA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0VSUk9SX1JFRyk7CglzZW5zZV9rZXkgPSBlcnIgPj4gNDsKCglpZiAocnEgPT0gTlVMTCkgewoJCXByaW50aygiJXM6IG1pc3NpbmcgcnEgaW4gY2Ryb21fZGVjb2RlX3N0YXR1c1xuIiwgZHJpdmUtPm5hbWUpOwoJCXJldHVybiAxOwoJfQoKCWlmIChycS0+ZmxhZ3MgJiBSRVFfU0VOU0UpIHsKCQkvKiBXZSBnb3QgYW4gZXJyb3IgdHJ5aW5nIHRvIGdldCBzZW5zZSBpbmZvCgkJICAgZnJvbSB0aGUgZHJpdmUgKHByb2JhYmx5IHdoaWxlIHRyeWluZwoJCSAgIHRvIHJlY292ZXIgZnJvbSBhIGZvcm1lciBlcnJvcikuICBKdXN0IGdpdmUgdXAuICovCgoJCXJxLT5mbGFncyB8PSBSRVFfRkFJTEVEOwoJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQlpZGVfZXJyb3IoZHJpdmUsICJyZXF1ZXN0IHNlbnNlIGZhaWx1cmUiLCBzdGF0KTsKCQlyZXR1cm4gMTsKCgl9IGVsc2UgaWYgKHJxLT5mbGFncyAmIChSRVFfUEMgfCBSRVFfQkxPQ0tfUEMpKSB7CgkJLyogQWxsIG90aGVyIGZ1bmN0aW9ucywgZXhjZXB0IGZvciBSRUFELiAqLwoJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJCS8qCgkJICogaWYgd2UgaGF2ZSBhbiBlcnJvciwgcGFzcyBiYWNrIENIRUNLX0NPTkRJVElPTiBhcyB0aGUKCQkgKiBzY3NpIHN0YXR1cyBieXRlCgkJICovCgkJaWYgKChycS0+ZmxhZ3MgJiBSRVFfQkxPQ0tfUEMpICYmICFycS0+ZXJyb3JzKQoJCQlycS0+ZXJyb3JzID0gU0FNX1NUQVRfQ0hFQ0tfQ09ORElUSU9OOwoKCQkvKiBDaGVjayBmb3IgdHJheSBvcGVuLiAqLwoJCWlmIChzZW5zZV9rZXkgPT0gTk9UX1JFQURZKSB7CgkJCWNkcm9tX3Nhd19tZWRpYV9jaGFuZ2UgKGRyaXZlKTsKCQl9IGVsc2UgaWYgKHNlbnNlX2tleSA9PSBVTklUX0FUVEVOVElPTikgewoJCQkvKiBDaGVjayBmb3IgbWVkaWEgY2hhbmdlLiAqLwoJCQljZHJvbV9zYXdfbWVkaWFfY2hhbmdlIChkcml2ZSk7CgkJCS8qcHJpbnRrKCIlczogbWVkaWEgY2hhbmdlZFxuIixkcml2ZS0+bmFtZSk7Ki8KCQkJcmV0dXJuIDA7CgkJfSBlbHNlIGlmICghKHJxLT5mbGFncyAmIFJFUV9RVUlFVCkpIHsKCQkJLyogT3RoZXJ3aXNlLCBwcmludCBhbiBlcnJvci4gKi8KCQkJaWRlX2R1bXBfc3RhdHVzKGRyaXZlLCAicGFja2V0IGNvbW1hbmQgZXJyb3IiLCBzdGF0KTsKCQl9CgkJCgkJcnEtPmZsYWdzIHw9IFJFUV9GQUlMRUQ7CgoJCS8qCgkJICogaW5zdGVhZCBvZiBwbGF5aW5nIGdhbWVzIHdpdGggbW92aW5nIGNvbXBsZXRpb25zIGFyb3VuZCwKCQkgKiByZW1vdmUgZmFpbGVkIHJlcXVlc3QgY29tcGxldGVseSBhbmQgZW5kIGl0IHdoZW4gdGhlCgkJICogcmVxdWVzdCBzZW5zZSBoYXMgY29tcGxldGVkCgkJICovCgkJaWYgKHN0YXQgJiBFUlJfU1RBVCkgewoJCQlzcGluX2xvY2tfaXJxc2F2ZSgmaWRlX2xvY2ssIGZsYWdzKTsKCQkJYmxrZGV2X2RlcXVldWVfcmVxdWVzdChycSk7CgkJCUhXR1JPVVAoZHJpdmUpLT5ycSA9IE5VTEw7CgkJCXNwaW5fdW5sb2NrX2lycXJlc3RvcmUoJmlkZV9sb2NrLCBmbGFncyk7CgoJCQljZHJvbV9xdWV1ZV9yZXF1ZXN0X3NlbnNlKGRyaXZlLCBycS0+c2Vuc2UsIHJxKTsKCQl9IGVsc2UKCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoKCX0gZWxzZSBpZiAoYmxrX2ZzX3JlcXVlc3QocnEpKSB7CgkJaW50IGRvX2VuZF9yZXF1ZXN0ID0gMDsKCgkJLyogSGFuZGxlIGVycm9ycyBmcm9tIFJFQUQgYW5kIFdSSVRFIHJlcXVlc3RzLiAqLwoKCQlpZiAoYmxrX25vcmV0cnlfcmVxdWVzdChycSkpCgkJCWRvX2VuZF9yZXF1ZXN0ID0gMTsKCgkJaWYgKHNlbnNlX2tleSA9PSBOT1RfUkVBRFkpIHsKCQkJLyogVHJheSBvcGVuLiAqLwoJCQlpZiAocnFfZGF0YV9kaXIocnEpID09IFJFQUQpIHsKCQkJCWNkcm9tX3Nhd19tZWRpYV9jaGFuZ2UgKGRyaXZlKTsKCgkJCQkvKiBGYWlsIHRoZSByZXF1ZXN0LiAqLwoJCQkJcHJpbnRrICgiJXM6IHRyYXkgb3BlblxuIiwgZHJpdmUtPm5hbWUpOwoJCQkJZG9fZW5kX3JlcXVlc3QgPSAxOwoJCQl9IGVsc2UgewoJCQkJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJCQkJLyogYWxsb3cgdGhlIGRyaXZlIDUgc2Vjb25kcyB0byByZWNvdmVyLCBzb21lCgkJCQkgKiBkZXZpY2VzIHdpbGwgcmV0dXJuIHRoaXMgZXJyb3Igd2hpbGUgZmx1c2hpbmcKCQkJCSAqIGRhdGEgZnJvbSBjYWNoZSAqLwoJCQkJaWYgKCFycS0+ZXJyb3JzKQoJCQkJCWluZm8tPndyaXRlX3RpbWVvdXQgPSBqaWZmaWVzICsgQVRBUElfV0FJVF9XUklURV9CVVNZOwoJCQkJcnEtPmVycm9ycyA9IDE7CgkJCQlpZiAodGltZV9hZnRlcihqaWZmaWVzLCBpbmZvLT53cml0ZV90aW1lb3V0KSkKCQkJCQlkb19lbmRfcmVxdWVzdCA9IDE7CgkJCQllbHNlIHsKCQkJCQl1bnNpZ25lZCBsb25nIGZsYWdzOwoKCQkJCQkvKgoJCQkJCSAqIHRha2UgYSBicmVhdGhlciByZWx5aW5nIG9uIHRoZQoJCQkJCSAqIHVucGx1ZyB0aW1lciB0byBraWNrIHVzIGFnYWluCgkJCQkJICovCgkJCQkJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CgkJCQkJYmxrX3BsdWdfZGV2aWNlKGRyaXZlLT5xdWV1ZSk7CgkJCQkJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmaWRlX2xvY2ssZmxhZ3MpOwoJCQkJCXJldHVybiAxOwoJCQkJfQoJCQl9CgkJfSBlbHNlIGlmIChzZW5zZV9rZXkgPT0gVU5JVF9BVFRFTlRJT04pIHsKCQkJLyogTWVkaWEgY2hhbmdlLiAqLwoJCQljZHJvbV9zYXdfbWVkaWFfY2hhbmdlIChkcml2ZSk7CgoJCQkvKiBBcnJhbmdlIHRvIHJldHJ5IHRoZSByZXF1ZXN0LgoJCQkgICBCdXQgYmUgc3VyZSB0byBnaXZlIHVwIGlmIHdlJ3ZlIHJldHJpZWQKCQkJICAgdG9vIG1hbnkgdGltZXMuICovCgkJCWlmICgrK3JxLT5lcnJvcnMgPiBFUlJPUl9NQVgpCgkJCQlkb19lbmRfcmVxdWVzdCA9IDE7CgkJfSBlbHNlIGlmIChzZW5zZV9rZXkgPT0gSUxMRUdBTF9SRVFVRVNUIHx8CgkJCSAgIHNlbnNlX2tleSA9PSBEQVRBX1BST1RFQ1QpIHsKCQkJLyogTm8gcG9pbnQgaW4gcmV0cnlpbmcgYWZ0ZXIgYW4gaWxsZWdhbAoJCQkgICByZXF1ZXN0IG9yIGRhdGEgcHJvdGVjdCBlcnJvci4qLwoJCQlpZGVfZHVtcF9zdGF0dXMgKGRyaXZlLCAiY29tbWFuZCBlcnJvciIsIHN0YXQpOwoJCQlkb19lbmRfcmVxdWVzdCA9IDE7CgkJfSBlbHNlIGlmIChzZW5zZV9rZXkgPT0gTUVESVVNX0VSUk9SKSB7CgkJCS8qIE5vIHBvaW50IGluIHJlLXRyeWluZyBhIHppbGxpb24gdGltZXMgb24gYSBiYWQgCgkJCSAqIHNlY3Rvci4uLiAgSWYgd2UgZ290IGhlcmUgdGhlIGVycm9yIGlzIG5vdCBjb3JyZWN0YWJsZSAqLwoJCQlpZGVfZHVtcF9zdGF0dXMgKGRyaXZlLCAibWVkaWEgZXJyb3IgKGJhZCBzZWN0b3IpIiwgc3RhdCk7CgkJCWRvX2VuZF9yZXF1ZXN0ID0gMTsKCQl9IGVsc2UgaWYgKHNlbnNlX2tleSA9PSBCTEFOS19DSEVDSykgewoJCQkvKiBEaXNrIGFwcGVhcnMgYmxhbmsgPz8gKi8KCQkJaWRlX2R1bXBfc3RhdHVzIChkcml2ZSwgIm1lZGlhIGVycm9yIChibGFuaykiLCBzdGF0KTsKCQkJZG9fZW5kX3JlcXVlc3QgPSAxOwoJCX0gZWxzZSBpZiAoKGVyciAmIH5BQlJUX0VSUikgIT0gMCkgewoJCQkvKiBHbyB0byB0aGUgZGVmYXVsdCBoYW5kbGVyCgkJCSAgIGZvciBvdGhlciBlcnJvcnMuICovCgkJCWlkZV9lcnJvcihkcml2ZSwgImNkcm9tX2RlY29kZV9zdGF0dXMiLCBzdGF0KTsKCQkJcmV0dXJuIDE7CgkJfSBlbHNlIGlmICgoKytycS0+ZXJyb3JzID4gRVJST1JfTUFYKSkgewoJCQkvKiBXZSd2ZSByYWNrZWQgdXAgdG9vIG1hbnkgcmV0cmllcy4gIEFib3J0LiAqLwoJCQlkb19lbmRfcmVxdWVzdCA9IDE7CgkJfQoKCQlpZiAoZG9fZW5kX3JlcXVlc3QpCgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCgkJLyogSWYgd2UgZ290IGEgQ0hFQ0tfQ09ORElUSU9OIHN0YXR1cywKCQkgICBxdWV1ZSBhIHJlcXVlc3Qgc2Vuc2UgY29tbWFuZC4gKi8KCQlpZiAoKHN0YXQgJiBFUlJfU1RBVCkgIT0gMCkKCQkJY2Ryb21fcXVldWVfcmVxdWVzdF9zZW5zZShkcml2ZSwgTlVMTCwgTlVMTCk7Cgl9IGVsc2UgewoJCWJsa19kdW1wX3JxX2ZsYWdzKHJxLCAiaWRlLWNkOiBiYWQgcnEiKTsKCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7Cgl9CgoJLyogUmV0cnksIG9yIGhhbmRsZSB0aGUgbmV4dCByZXF1ZXN0LiAqLwoJcmV0dXJuIDE7Cn0KCnN0YXRpYyBpbnQgY2Ryb21fdGltZXJfZXhwaXJ5KGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJdW5zaWduZWQgbG9uZyB3YWl0ID0gMDsKCgkvKgoJICogU29tZSBjb21tYW5kcyBhcmUgKnNsb3cqIGFuZCBub3JtYWxseSB0YWtlIGEgbG9uZyB0aW1lIHRvCgkgKiBjb21wbGV0ZS4gVXN1YWxseSB3ZSBjYW4gdXNlIHRoZSBBVEFQSSAiZGlzY29ubmVjdCIgdG8gYnlwYXNzCgkgKiB0aGlzLCBidXQgbm90IGFsbCBjb21tYW5kcy9kcml2ZXMgc3VwcG9ydCB0aGF0LiBMZXQKCSAqIGlkZV90aW1lcl9leHBpcnkga2VlcCBwb2xsaW5nIHVzIGZvciB0aGVzZS4KCSAqLwoJc3dpdGNoIChycS0+Y21kWzBdKSB7CgkJY2FzZSBHUENNRF9CTEFOSzoKCQljYXNlIEdQQ01EX0ZPUk1BVF9VTklUOgoJCWNhc2UgR1BDTURfUkVTRVJWRV9SWk9ORV9UUkFDSzoKCQljYXNlIEdQQ01EX0NMT1NFX1RSQUNLOgoJCWNhc2UgR1BDTURfRkxVU0hfQ0FDSEU6CgkJCXdhaXQgPSBBVEFQSV9XQUlUX1BDOwoJCQlicmVhazsKCQlkZWZhdWx0OgoJCQlpZiAoIShycS0+ZmxhZ3MgJiBSRVFfUVVJRVQpKQoJCQkJcHJpbnRrKEtFUk5fSU5GTyAiaWRlLWNkOiBjbWQgMHgleCB0aW1lZCBvdXRcbiIsIHJxLT5jbWRbMF0pOwoJCQl3YWl0ID0gMDsKCQkJYnJlYWs7Cgl9CglyZXR1cm4gd2FpdDsKfQoKLyogU2V0IHVwIHRoZSBkZXZpY2UgcmVnaXN0ZXJzIGZvciB0cmFuc2ZlcnJpbmcgYSBwYWNrZXQgY29tbWFuZCBvbiBERVYsCiAgIGV4cGVjdGluZyB0byBsYXRlciB0cmFuc2ZlciBYRkVSTEVOIGJ5dGVzLiAgSEFORExFUiBpcyB0aGUgcm91dGluZQogICB3aGljaCBhY3R1YWxseSB0cmFuc2ZlcnMgdGhlIGNvbW1hbmQgdG8gdGhlIGRyaXZlLiAgSWYgdGhpcyBpcyBhCiAgIGRycV9pbnRlcnJ1cHQgZGV2aWNlLCB0aGlzIHJvdXRpbmUgd2lsbCBhcnJhbmdlIGZvciBIQU5ETEVSIHRvIGJlCiAgIGNhbGxlZCB3aGVuIHRoZSBpbnRlcnJ1cHQgZnJvbSB0aGUgZHJpdmUgYXJyaXZlcy4gIE90aGVyd2lzZSwgSEFORExFUgogICB3aWxsIGJlIGNhbGxlZCBpbW1lZGlhdGVseSBhZnRlciB0aGUgZHJpdmUgaXMgcHJlcGFyZWQgZm9yIHRoZSB0cmFuc2Zlci4gKi8KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc3RhcnRfcGFja2V0X2NvbW1hbmQoaWRlX2RyaXZlX3QgKmRyaXZlLAoJCQkJCQkgIGludCB4ZmVybGVuLAoJCQkJCQkgIGlkZV9oYW5kbGVyX3QgKmhhbmRsZXIpCnsKCWlkZV9zdGFydHN0b3BfdCBzdGFydHN0b3A7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCWlkZV9od2lmX3QgKmh3aWYgPSBkcml2ZS0+aHdpZjsKCgkvKiBXYWl0IGZvciB0aGUgY29udHJvbGxlciB0byBiZSBpZGxlLiAqLwoJaWYgKGlkZV93YWl0X3N0YXQoJnN0YXJ0c3RvcCwgZHJpdmUsIDAsIEJVU1lfU1RBVCwgV0FJVF9SRUFEWSkpCgkJcmV0dXJuIHN0YXJ0c3RvcDsKCglpZiAoaW5mby0+ZG1hKQoJCWluZm8tPmRtYSA9ICFod2lmLT5kbWFfc2V0dXAoZHJpdmUpOwoKCS8qIFNldCB1cCB0aGUgY29udHJvbGxlciByZWdpc3RlcnMuICovCgkvKiBGSVhNRTogZm9yIFZpcnR1YWwgRE1BIHdlIG11c3QgY2hlY2sgaGFyZGVyICovCglIV0lGKGRyaXZlKS0+T1VUQihpbmZvLT5kbWEsIElERV9GRUFUVVJFX1JFRyk7CglIV0lGKGRyaXZlKS0+T1VUQigwLCBJREVfSVJFQVNPTl9SRUcpOwoJSFdJRihkcml2ZSktPk9VVEIoMCwgSURFX1NFQ1RPUl9SRUcpOwoKCUhXSUYoZHJpdmUpLT5PVVRCKHhmZXJsZW4gJiAweGZmLCBJREVfQkNPVU5UTF9SRUcpOwoJSFdJRihkcml2ZSktPk9VVEIoeGZlcmxlbiA+PiA4ICAsIElERV9CQ09VTlRIX1JFRyk7CglpZiAoSURFX0NPTlRST0xfUkVHKQoJCUhXSUYoZHJpdmUpLT5PVVRCKGRyaXZlLT5jdGwsIElERV9DT05UUk9MX1JFRyk7CiAKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MgKGRyaXZlKS0+ZHJxX2ludGVycnVwdCkgewoJCS8qIHBhY2tldCBjb21tYW5kICovCgkJaWRlX2V4ZWN1dGVfY29tbWFuZChkcml2ZSwgV0lOX1BBQ0tFVENNRCwgaGFuZGxlciwgQVRBUElfV0FJVF9QQywgY2Ryb21fdGltZXJfZXhwaXJ5KTsKCQlyZXR1cm4gaWRlX3N0YXJ0ZWQ7Cgl9IGVsc2UgewoJCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJCS8qIHBhY2tldCBjb21tYW5kICovCgkJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CgkJaHdpZi0+T1VUQlNZTkMoZHJpdmUsIFdJTl9QQUNLRVRDTUQsIElERV9DT01NQU5EX1JFRyk7CgkJbmRlbGF5KDQwMCk7CgkJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmaWRlX2xvY2ssIGZsYWdzKTsKCgkJcmV0dXJuICgqaGFuZGxlcikgKGRyaXZlKTsKCX0KfQoKLyogU2VuZCBhIHBhY2tldCBjb21tYW5kIHRvIERSSVZFIGRlc2NyaWJlZCBieSBDTURfQlVGIGFuZCBDTURfTEVOLgogICBUaGUgZGV2aWNlIHJlZ2lzdGVycyBtdXN0IGhhdmUgYWxyZWFkeSBiZWVuIHByZXBhcmVkCiAgIGJ5IGNkcm9tX3N0YXJ0X3BhY2tldF9jb21tYW5kLgogICBIQU5ETEVSIGlzIHRoZSBpbnRlcnJ1cHQgaGFuZGxlciB0byBjYWxsIHdoZW4gdGhlIGNvbW1hbmQgY29tcGxldGVzCiAgIG9yIHRoZXJlJ3MgZGF0YSByZWFkeS4gKi8KLyoKICogY2hhbmdlZCA1IHBhcmFtZXRlcnMgdG8gMyBmb3IgZHZkLXJhbQogKiBzdHJ1Y3QgcGFja2V0X2NvbW1hbmQgKnBjOyBub3cgcGFja2V0X2NvbW1hbmRfdCAqcGM7CiAqLwojZGVmaW5lIEFUQVBJX01JTl9DREJfQllURVMgMTIKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV90cmFuc2Zlcl9wYWNrZXRfY29tbWFuZCAoaWRlX2RyaXZlX3QgKmRyaXZlLAoJCQkJCSAgc3RydWN0IHJlcXVlc3QgKnJxLAoJCQkJCSAgaWRlX2hhbmRsZXJfdCAqaGFuZGxlcikKewoJaWRlX2h3aWZfdCAqaHdpZiA9IGRyaXZlLT5od2lmOwoJaW50IGNtZF9sZW47CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCWlkZV9zdGFydHN0b3BfdCBzdGFydHN0b3A7CgoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmRycV9pbnRlcnJ1cHQpIHsKCQkvKiBIZXJlIHdlIHNob3VsZCBoYXZlIGJlZW4gY2FsbGVkIGFmdGVyIHJlY2VpdmluZyBhbiBpbnRlcnJ1cHQKCQkgICBmcm9tIHRoZSBkZXZpY2UuICBEUlEgc2hvdWxkIGhvdyBiZSBzZXQuICovCgoJCS8qIENoZWNrIGZvciBlcnJvcnMuICovCgkJaWYgKGNkcm9tX2RlY29kZV9zdGF0dXMoZHJpdmUsIERSUV9TVEFULCBOVUxMKSkKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfSBlbHNlIHsKCQkvKiBPdGhlcndpc2UsIHdlIG11c3Qgd2FpdCBmb3IgRFJRIHRvIGdldCBzZXQuICovCgkJaWYgKGlkZV93YWl0X3N0YXQoJnN0YXJ0c3RvcCwgZHJpdmUsIERSUV9TVEFULAoJCQkJQlVTWV9TVEFULCBXQUlUX1JFQURZKSkKCQkJcmV0dXJuIHN0YXJ0c3RvcDsKCX0KCgkvKiBBcm0gdGhlIGludGVycnVwdCBoYW5kbGVyLiAqLwoJaWRlX3NldF9oYW5kbGVyKGRyaXZlLCBoYW5kbGVyLCBycS0+dGltZW91dCwgY2Ryb21fdGltZXJfZXhwaXJ5KTsKCgkvKiBBVEFQSSBjb21tYW5kcyBnZXQgcGFkZGVkIG91dCB0byAxMiBieXRlcyBtaW5pbXVtICovCgljbWRfbGVuID0gQ09NTUFORF9TSVpFKHJxLT5jbWRbMF0pOwoJaWYgKGNtZF9sZW4gPCBBVEFQSV9NSU5fQ0RCX0JZVEVTKQoJCWNtZF9sZW4gPSBBVEFQSV9NSU5fQ0RCX0JZVEVTOwoKCS8qIFNlbmQgdGhlIGNvbW1hbmQgdG8gdGhlIGRldmljZS4gKi8KCUhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXMoZHJpdmUsIHJxLT5jbWQsIGNtZF9sZW4pOwoKCS8qIFN0YXJ0IHRoZSBETUEgaWYgbmVlZCBiZSAqLwoJaWYgKGluZm8tPmRtYSkKCQlod2lmLT5kbWFfc3RhcnQoZHJpdmUpOwoKCXJldHVybiBpZGVfc3RhcnRlZDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQmxvY2sgcmVhZCBmdW5jdGlvbnMuCiAqLwoKLyoKICogQnVmZmVyIHVwIHRvIFNFQ1RPUlNfVE9fVFJBTlNGRVIgc2VjdG9ycyBmcm9tIHRoZSBkcml2ZSBpbiBvdXIgc2VjdG9yCiAqIGJ1ZmZlci4gIE9uY2UgdGhlIGZpcnN0IHNlY3RvciBpcyBhZGRlZCwgYW55IHN1YnNlcXVlbnQgc2VjdG9ycyBhcmUKICogYXNzdW1lZCB0byBiZSBjb250aW51b3VzICh1bnRpbCB0aGUgYnVmZmVyIGlzIGNsZWFyZWQpLiAgRm9yIHRoZSBmaXJzdAogKiBzZWN0b3IgYWRkZWQsIFNFQ1RPUiBpcyBpdHMgc2VjdG9yIG51bWJlci4gIChTRUNUT1IgaXMgdGhlbiBpZ25vcmVkIHVudGlsCiAqIHRoZSBidWZmZXIgaXMgY2xlYXJlZC4pCiAqLwpzdGF0aWMgdm9pZCBjZHJvbV9idWZmZXJfc2VjdG9ycyAoaWRlX2RyaXZlX3QgKmRyaXZlLCB1bnNpZ25lZCBsb25nIHNlY3RvciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludCBzZWN0b3JzX3RvX3RyYW5zZmVyKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCgkvKiBOdW1iZXIgb2Ygc2VjdG9ycyB0byByZWFkIGludG8gdGhlIGJ1ZmZlci4gKi8KCWludCBzZWN0b3JzX3RvX2J1ZmZlciA9IG1pbl90KGludCwgc2VjdG9yc190b190cmFuc2ZlciwKCQkJCSAgICAgKFNFQ1RPUl9CVUZGRVJfU0laRSA+PiBTRUNUT1JfQklUUykgLQoJCQkJICAgICAgIGluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkKTsKCgljaGFyICpkZXN0OwoKCS8qIElmIHdlIGNvdWxkbid0IGdldCBhIGJ1ZmZlciwgZG9uJ3QgdHJ5IHRvIGJ1ZmZlciBhbnl0aGluZy4uLiAqLwoJaWYgKGluZm8tPmJ1ZmZlciA9PSBOVUxMKQoJCXNlY3RvcnNfdG9fYnVmZmVyID0gMDsKCgkvKiBJZiB0aGlzIGlzIHRoZSBmaXJzdCBzZWN0b3IgaW4gdGhlIGJ1ZmZlciwgcmVtZW1iZXIgaXRzIG51bWJlci4gKi8KCWlmIChpbmZvLT5uc2VjdG9yc19idWZmZXJlZCA9PSAwKQoJCWluZm8tPnNlY3Rvcl9idWZmZXJlZCA9IHNlY3RvcjsKCgkvKiBSZWFkIHRoZSBkYXRhIGludG8gdGhlIGJ1ZmZlci4gKi8KCWRlc3QgPSBpbmZvLT5idWZmZXIgKyBpbmZvLT5uc2VjdG9yc19idWZmZXJlZCAqIFNFQ1RPUl9TSVpFOwoJd2hpbGUgKHNlY3RvcnNfdG9fYnVmZmVyID4gMCkgewoJCUhXSUYoZHJpdmUpLT5hdGFwaV9pbnB1dF9ieXRlcyhkcml2ZSwgZGVzdCwgU0VDVE9SX1NJWkUpOwoJCS0tc2VjdG9yc190b19idWZmZXI7CgkJLS1zZWN0b3JzX3RvX3RyYW5zZmVyOwoJCSsraW5mby0+bnNlY3RvcnNfYnVmZmVyZWQ7CgkJZGVzdCArPSBTRUNUT1JfU0laRTsKCX0KCgkvKiBUaHJvdyBhd2F5IGFueSByZW1haW5pbmcgZGF0YS4gKi8KCXdoaWxlIChzZWN0b3JzX3RvX3RyYW5zZmVyID4gMCkgewoJCXN0YXRpYyBjaGFyIGR1bVtTRUNUT1JfU0laRV07CgkJSFdJRihkcml2ZSktPmF0YXBpX2lucHV0X2J5dGVzKGRyaXZlLCBkdW0sIHNpemVvZiAoZHVtKSk7CgkJLS1zZWN0b3JzX3RvX3RyYW5zZmVyOwoJfQp9CgovKgogKiBDaGVjayB0aGUgY29udGVudHMgb2YgdGhlIGludGVycnVwdCByZWFzb24gcmVnaXN0ZXIgZnJvbSB0aGUgY2Ryb20KICogYW5kIGF0dGVtcHQgdG8gcmVjb3ZlciBpZiB0aGVyZSBhcmUgcHJvYmxlbXMuICBSZXR1cm5zICAwIGlmIGV2ZXJ5dGhpbmcncwogKiBvazsgbm9uemVybyBpZiB0aGUgcmVxdWVzdCBoYXMgYmVlbiB0ZXJtaW5hdGVkLgogKi8Kc3RhdGljIGlubGluZQppbnQgY2Ryb21fcmVhZF9jaGVja19pcmVhc29uIChpZGVfZHJpdmVfdCAqZHJpdmUsIGludCBsZW4sIGludCBpcmVhc29uKQp7CglpZiAoaXJlYXNvbiA9PSAyKQoJCXJldHVybiAwOwoJZWxzZSBpZiAoaXJlYXNvbiA9PSAwKSB7CgkJLyogV2hvb3BzLi4uIFRoZSBkcml2ZSBpcyBleHBlY3RpbmcgdG8gcmVjZWl2ZSBkYXRhIGZyb20gdXMhICovCgkJcHJpbnRrKEtFUk5fRVJSICIlczogcmVhZF9pbnRyOiBEcml2ZSB3YW50cyB0byB0cmFuc2ZlciBkYXRhIHRoZSAiCgkJCQkJCSJ3cm9uZyB3YXkhXG4iLCBkcml2ZS0+bmFtZSk7CgoJCS8qIFRocm93IHNvbWUgZGF0YSBhdCB0aGUgZHJpdmUgc28gaXQgZG9lc24ndCBoYW5nCgkJICAgYW5kIHF1aXQgdGhpcyByZXF1ZXN0LiAqLwoJCXdoaWxlIChsZW4gPiAwKSB7CgkJCWludCBkdW0gPSAwOwoJCQlIV0lGKGRyaXZlKS0+YXRhcGlfb3V0cHV0X2J5dGVzKGRyaXZlLCAmZHVtLCBzaXplb2YgKGR1bSkpOwoJCQlsZW4gLT0gc2l6ZW9mIChkdW0pOwoJCX0KCX0gZWxzZSAgaWYgKGlyZWFzb24gPT0gMSkgewoJCS8qIFNvbWUgZHJpdmVzIChBU1VTKSBzZWVtIHRvIHRlbGwgdXMgdGhhdCBzdGF0dXMKCQkgKiBpbmZvIGlzIGF2YWlsYWJsZS4ganVzdCBnZXQgaXQgYW5kIGlnbm9yZS4KCQkgKi8KCQkodm9pZCkgSFdJRihkcml2ZSktPklOQihJREVfU1RBVFVTX1JFRyk7CgkJcmV0dXJuIDA7Cgl9IGVsc2UgewoJCS8qIERyaXZlIHdhbnRzIGEgY29tbWFuZCBwYWNrZXQsIG9yIGludmFsaWQgaXJlYXNvbi4uLiAqLwoJCXByaW50ayhLRVJOX0VSUiAiJXM6IHJlYWRfaW50cjogYmFkIGludGVycnVwdCByZWFzb24gJXhcbiIsIGRyaXZlLT5uYW1lLAoJCQkJCQkJCWlyZWFzb24pOwoJfQoKCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCXJldHVybiAtMTsKfQoKLyoKICogSW50ZXJydXB0IHJvdXRpbmUuICBDYWxsZWQgd2hlbiBhIHJlYWQgcmVxdWVzdCBoYXMgY29tcGxldGVkLgogKi8Kc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9yZWFkX2ludHIgKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJaW50IHN0YXQ7CglpbnQgaXJlYXNvbiwgbGVuLCBzZWN0b3JzX3RvX3RyYW5zZmVyLCBuc2tpcDsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJdTggbG93Y3lsID0gMCwgaGlnaGN5bCA9IDA7CglpbnQgZG1hID0gaW5mby0+ZG1hLCBkbWFfZXJyb3IgPSAwOwoKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCgkvKgoJICogaGFuZGxlIGRtYSBjYXNlCgkgKi8KCWlmIChkbWEpIHsKCQlpbmZvLT5kbWEgPSAwOwoJCWlmICgoZG1hX2Vycm9yID0gSFdJRihkcml2ZSktPmlkZV9kbWFfZW5kKGRyaXZlKSkpCgkJCV9faWRlX2RtYV9vZmYoZHJpdmUpOwoJfQoKCWlmIChjZHJvbV9kZWNvZGVfc3RhdHVzKGRyaXZlLCAwLCAmc3RhdCkpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCWlmIChkbWEpIHsKCQlpZiAoIWRtYV9lcnJvcikgewoJCQlpZGVfZW5kX3JlcXVlc3QoZHJpdmUsIDEsIHJxLT5ucl9zZWN0b3JzKTsKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJCX0gZWxzZQoJCQlyZXR1cm4gaWRlX2Vycm9yKGRyaXZlLCAiZG1hIGVycm9yIiwgc3RhdCk7Cgl9CgoJLyogUmVhZCB0aGUgaW50ZXJydXB0IHJlYXNvbiBhbmQgdGhlIHRyYW5zZmVyIGxlbmd0aC4gKi8KCWlyZWFzb24gPSBIV0lGKGRyaXZlKS0+SU5CKElERV9JUkVBU09OX1JFRykgJiAweDM7Cglsb3djeWwgID0gSFdJRihkcml2ZSktPklOQihJREVfQkNPVU5UTF9SRUcpOwoJaGlnaGN5bCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVEhfUkVHKTsKCglsZW4gPSBsb3djeWwgKyAoMjU2ICogaGlnaGN5bCk7CgoJLyogSWYgRFJRIGlzIGNsZWFyLCB0aGUgY29tbWFuZCBoYXMgY29tcGxldGVkLiAqLwoJaWYgKChzdGF0ICYgRFJRX1NUQVQpID09IDApIHsKCQkvKiBJZiB3ZSdyZSBub3QgZG9uZSBmaWxsaW5nIHRoZSBjdXJyZW50IGJ1ZmZlciwgY29tcGxhaW4uCgkJICAgT3RoZXJ3aXNlLCBjb21wbGV0ZSB0aGUgY29tbWFuZCBub3JtYWxseS4gKi8KCQlpZiAocnEtPmN1cnJlbnRfbnJfc2VjdG9ycyA+IDApIHsKCQkJcHJpbnRrIChLRVJOX0VSUiAiJXM6IGNkcm9tX3JlYWRfaW50cjogZGF0YSB1bmRlcnJ1biAoJWQgYmxvY2tzKVxuIiwKCQkJCWRyaXZlLT5uYW1lLCBycS0+Y3VycmVudF9ucl9zZWN0b3JzKTsKCQkJcnEtPmZsYWdzIHw9IFJFUV9GQUlMRUQ7CgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQl9IGVsc2UKCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoJCXJldHVybiBpZGVfc3RvcHBlZDsKCX0KCgkvKiBDaGVjayB0aGF0IHRoZSBkcml2ZSBpcyBleHBlY3RpbmcgdG8gZG8gdGhlIHNhbWUgdGhpbmcgd2UgYXJlLiAqLwoJaWYgKGNkcm9tX3JlYWRfY2hlY2tfaXJlYXNvbiAoZHJpdmUsIGxlbiwgaXJlYXNvbikpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCS8qIEFzc3VtZSB0aGF0IHRoZSBkcml2ZSB3aWxsIGFsd2F5cyBwcm92aWRlIGRhdGEgaW4gbXVsdGlwbGVzCgkgICBvZiBhdCBsZWFzdCBTRUNUT1JfU0laRSwgYXMgaXQgZ2V0cyBoYWlyeSB0byBrZWVwIHRyYWNrCgkgICBvZiB0aGUgdHJhbnNmZXJzIG90aGVyd2lzZS4gKi8KCWlmICgobGVuICUgU0VDVE9SX1NJWkUpICE9IDApIHsKCQlwcmludGsgKEtFUk5fRVJSICIlczogY2Ryb21fcmVhZF9pbnRyOiBCYWQgdHJhbnNmZXIgc2l6ZSAlZFxuIiwKCQkJZHJpdmUtPm5hbWUsIGxlbik7CgkJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmxpbWl0X25mcmFtZXMpCgkJCXByaW50ayAoS0VSTl9FUlIgIiAgVGhpcyBkcml2ZSBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoaXMgdmVyc2lvbiBvZiB0aGUgZHJpdmVyXG4iKTsKCQllbHNlIHsKCQkJcHJpbnRrIChLRVJOX0VSUiAiICBUcnlpbmcgdG8gbGltaXQgdHJhbnNmZXIgc2l6ZXNcbiIpOwoJCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5saW1pdF9uZnJhbWVzID0gMTsKCQl9CgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJCXJldHVybiBpZGVfc3RvcHBlZDsKCX0KCgkvKiBUaGUgbnVtYmVyIG9mIHNlY3RvcnMgd2UgbmVlZCB0byByZWFkIGZyb20gdGhlIGRyaXZlLiAqLwoJc2VjdG9yc190b190cmFuc2ZlciA9IGxlbiAvIFNFQ1RPUl9TSVpFOwoKCS8qIEZpcnN0LCBmaWd1cmUgb3V0IGlmIHdlIG5lZWQgdG8gYml0LWJ1Y2tldAoJICAgYW55IG9mIHRoZSBsZWFkaW5nIHNlY3RvcnMuICovCgluc2tpcCA9IG1pbl90KGludCwgcnEtPmN1cnJlbnRfbnJfc2VjdG9ycyAtIGJpb19jdXJfc2VjdG9ycyhycS0+YmlvKSwgc2VjdG9yc190b190cmFuc2Zlcik7CgoJd2hpbGUgKG5za2lwID4gMCkgewoJCS8qIFdlIG5lZWQgdG8gdGhyb3cgYXdheSBhIHNlY3Rvci4gKi8KCQlzdGF0aWMgY2hhciBkdW1bU0VDVE9SX1NJWkVdOwoJCUhXSUYoZHJpdmUpLT5hdGFwaV9pbnB1dF9ieXRlcyhkcml2ZSwgZHVtLCBzaXplb2YgKGR1bSkpOwoKCQktLXJxLT5jdXJyZW50X25yX3NlY3RvcnM7CgkJLS1uc2tpcDsKCQktLXNlY3RvcnNfdG9fdHJhbnNmZXI7Cgl9CgoJLyogTm93IGxvb3Agd2hpbGUgd2Ugc3RpbGwgaGF2ZSBkYXRhIHRvIHJlYWQgZnJvbSB0aGUgZHJpdmUuICovCgl3aGlsZSAoc2VjdG9yc190b190cmFuc2ZlciA+IDApIHsKCQlpbnQgdGhpc190cmFuc2ZlcjsKCgkJLyogSWYgd2UndmUgZmlsbGVkIHRoZSBwcmVzZW50IGJ1ZmZlciBidXQgdGhlcmUncyBhbm90aGVyCgkJICAgY2hhaW5lZCBidWZmZXIgYWZ0ZXIgaXQsIG1vdmUgb24uICovCgkJaWYgKHJxLT5jdXJyZW50X25yX3NlY3RvcnMgPT0gMCAmJiBycS0+bnJfc2VjdG9ycykKCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoKCQkvKiBJZiB0aGUgYnVmZmVycyBhcmUgZnVsbCwgY2FjaGUgdGhlIHJlc3Qgb2YgdGhlIGRhdGEgaW4gb3VyCgkJICAgaW50ZXJuYWwgYnVmZmVyLiAqLwoJCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzID09IDApIHsKCQkJY2Ryb21fYnVmZmVyX3NlY3RvcnMoZHJpdmUsIHJxLT5zZWN0b3IsIHNlY3RvcnNfdG9fdHJhbnNmZXIpOwoJCQlzZWN0b3JzX3RvX3RyYW5zZmVyID0gMDsKCQl9IGVsc2UgewoJCQkvKiBUcmFuc2ZlciBkYXRhIHRvIHRoZSBidWZmZXJzLgoJCQkgICBGaWd1cmUgb3V0IGhvdyBtYW55IHNlY3RvcnMgd2UgY2FuIHRyYW5zZmVyCgkJCSAgIHRvIHRoZSBjdXJyZW50IGJ1ZmZlci4gKi8KCQkJdGhpc190cmFuc2ZlciA9IG1pbl90KGludCwgc2VjdG9yc190b190cmFuc2ZlciwKCQkJCQkgICAgIHJxLT5jdXJyZW50X25yX3NlY3RvcnMpOwoKCQkJLyogUmVhZCB0aGlzX3RyYW5zZmVyIHNlY3RvcnMKCQkJICAgaW50byB0aGUgY3VycmVudCBidWZmZXIuICovCgkJCXdoaWxlICh0aGlzX3RyYW5zZmVyID4gMCkgewoJCQkJSFdJRihkcml2ZSktPmF0YXBpX2lucHV0X2J5dGVzKGRyaXZlLCBycS0+YnVmZmVyLCBTRUNUT1JfU0laRSk7CgkJCQlycS0+YnVmZmVyICs9IFNFQ1RPUl9TSVpFOwoJCQkJLS1ycS0+bnJfc2VjdG9yczsKCQkJCS0tcnEtPmN1cnJlbnRfbnJfc2VjdG9yczsKCQkJCSsrcnEtPnNlY3RvcjsKCQkJCS0tdGhpc190cmFuc2ZlcjsKCQkJCS0tc2VjdG9yc190b190cmFuc2ZlcjsKCQkJfQoJCX0KCX0KCgkvKiBEb25lIG1vdmluZyBkYXRhISAgV2FpdCBmb3IgYW5vdGhlciBpbnRlcnJ1cHQuICovCglpZGVfc2V0X2hhbmRsZXIoZHJpdmUsICZjZHJvbV9yZWFkX2ludHIsIEFUQVBJX1dBSVRfUEMsIE5VTEwpOwoJcmV0dXJuIGlkZV9zdGFydGVkOwp9CgovKgogKiBUcnkgdG8gc2F0aXNmeSBzb21lIG9mIHRoZSBjdXJyZW50IHJlYWQgcmVxdWVzdCBmcm9tIG91ciBjYWNoZWQgZGF0YS4KICogUmV0dXJucyBub256ZXJvIGlmIHRoZSByZXF1ZXN0IGhhcyBiZWVuIGNvbXBsZXRlZCwgemVybyBvdGhlcndpc2UuCiAqLwpzdGF0aWMgaW50IGNkcm9tX3JlYWRfZnJvbV9idWZmZXIgKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7Cgl1bnNpZ25lZCBzaG9ydCBzZWN0b3JzX3Blcl9mcmFtZTsKCglzZWN0b3JzX3Blcl9mcmFtZSA9IHF1ZXVlX2hhcmRzZWN0X3NpemUoZHJpdmUtPnF1ZXVlKSA+PiBTRUNUT1JfQklUUzsKCgkvKiBDYW4ndCBkbyBhbnl0aGluZyBpZiB0aGVyZSdzIG5vIGJ1ZmZlci4gKi8KCWlmIChpbmZvLT5idWZmZXIgPT0gTlVMTCkgcmV0dXJuIDA7CgoJLyogTG9vcCB3aGlsZSB0aGlzIHJlcXVlc3QgbmVlZHMgZGF0YSBhbmQgdGhlIG5leHQgYmxvY2sgaXMgcHJlc2VudAoJICAgaW4gb3VyIGNhY2hlLiAqLwoJd2hpbGUgKHJxLT5ucl9zZWN0b3JzID4gMCAmJgoJICAgICAgIHJxLT5zZWN0b3IgPj0gaW5mby0+c2VjdG9yX2J1ZmZlcmVkICYmCgkgICAgICAgcnEtPnNlY3RvciA8IGluZm8tPnNlY3Rvcl9idWZmZXJlZCArIGluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkKSB7CgkJaWYgKHJxLT5jdXJyZW50X25yX3NlY3RvcnMgPT0gMCkKCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoKCQltZW1jcHkgKHJxLT5idWZmZXIsCgkJCWluZm8tPmJ1ZmZlciArCgkJCShycS0+c2VjdG9yIC0gaW5mby0+c2VjdG9yX2J1ZmZlcmVkKSAqIFNFQ1RPUl9TSVpFLAoJCQlTRUNUT1JfU0laRSk7CgkJcnEtPmJ1ZmZlciArPSBTRUNUT1JfU0laRTsKCQktLXJxLT5jdXJyZW50X25yX3NlY3RvcnM7CgkJLS1ycS0+bnJfc2VjdG9yczsKCQkrK3JxLT5zZWN0b3I7Cgl9CgoJLyogSWYgd2UndmUgc2F0aXNmaWVkIHRoZSBjdXJyZW50IHJlcXVlc3QsCgkgICB0ZXJtaW5hdGUgaXQgc3VjY2Vzc2Z1bGx5LiAqLwoJaWYgKHJxLT5ucl9zZWN0b3JzID09IDApIHsKCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMSk7CgkJcmV0dXJuIC0xOwoJfQoKCS8qIE1vdmUgb24gdG8gdGhlIG5leHQgYnVmZmVyIGlmIG5lZWRlZC4gKi8KCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzID09IDApCgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoKCS8qIElmIHRoaXMgY29uZGl0aW9uIGRvZXMgbm90IGhvbGQsIHRoZW4gdGhlIGtsdWdlIGkgdXNlIHRvCgkgICByZXByZXNlbnQgdGhlIG51bWJlciBvZiBzZWN0b3JzIHRvIHNraXAgYXQgdGhlIHN0YXJ0IG9mIGEgdHJhbnNmZXIKCSAgIHdpbGwgZmFpbC4gIEkgdGhpbmsgdGhhdCB0aGlzIHdpbGwgbmV2ZXIgaGFwcGVuLCBidXQgbGV0J3MgYmUKCSAgIHBhcmFub2lkIGFuZCBjaGVjay4gKi8KCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzIDwgYmlvX2N1cl9zZWN0b3JzKHJxLT5iaW8pICYmCgkgICAgKHJxLT5zZWN0b3IgJiAoc2VjdG9yc19wZXJfZnJhbWUgLSAxKSkpIHsKCQlwcmludGsoS0VSTl9FUlIgIiVzOiBjZHJvbV9yZWFkX2Zyb21fYnVmZmVyOiBidWZmZXIgYm90Y2ggKCVsZClcbiIsCgkJCWRyaXZlLT5uYW1lLCAobG9uZylycS0+c2VjdG9yKTsKCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgMCk7CgkJcmV0dXJuIC0xOwoJfQoKCXJldHVybiAwOwp9CgovKgogKiBSb3V0aW5lIHRvIHNlbmQgYSByZWFkIHBhY2tldCBjb21tYW5kIHRvIHRoZSBkcml2ZS4KICogVGhpcyBpcyB1c3VhbGx5IGNhbGxlZCBkaXJlY3RseSBmcm9tIGNkcm9tX3N0YXJ0X3JlYWQuCiAqIEhvd2V2ZXIsIGZvciBkcnFfaW50ZXJydXB0IGRldmljZXMsIGl0IGlzIGNhbGxlZCBmcm9tIGFuIGludGVycnVwdAogKiB3aGVuIHRoZSBkcml2ZSBpcyByZWFkeSB0byBhY2NlcHQgdGhlIGNvbW1hbmQuCiAqLwpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3N0YXJ0X3JlYWRfY29udGludWF0aW9uIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCXVuc2lnbmVkIHNob3J0IHNlY3RvcnNfcGVyX2ZyYW1lOwoJaW50IG5za2lwOwoKCXNlY3RvcnNfcGVyX2ZyYW1lID0gcXVldWVfaGFyZHNlY3Rfc2l6ZShkcml2ZS0+cXVldWUpID4+IFNFQ1RPUl9CSVRTOwoKCS8qIElmIHRoZSByZXF1ZXN0ZWQgc2VjdG9yIGRvZXNuJ3Qgc3RhcnQgb24gYSBjZHJvbSBibG9jayBib3VuZGFyeSwKCSAgIHdlIG11c3QgYWRqdXN0IHRoZSBzdGFydCBvZiB0aGUgdHJhbnNmZXIgc28gdGhhdCBpdCBkb2VzLAoJICAgYW5kIHJlbWVtYmVyIHRvIHNraXAgdGhlIGZpcnN0IGZldyBzZWN0b3JzLgoJICAgSWYgdGhlIENVUlJFTlRfTlJfU0VDVE9SUyBmaWVsZCBpcyBsYXJnZXIgdGhhbiB0aGUgc2l6ZQoJICAgb2YgdGhlIGJ1ZmZlciwgaXQgd2lsbCBtZWFuIHRoYXQgd2UncmUgdG8gc2tpcCBhIG51bWJlcgoJICAgb2Ygc2VjdG9ycyBlcXVhbCB0byB0aGUgYW1vdW50IGJ5IHdoaWNoIENVUlJFTlRfTlJfU0VDVE9SUwoJICAgaXMgbGFyZ2VyIHRoYW4gdGhlIGJ1ZmZlciBzaXplLiAqLwoJbnNraXAgPSBycS0+c2VjdG9yICYgKHNlY3RvcnNfcGVyX2ZyYW1lIC0gMSk7CglpZiAobnNraXAgPiAwKSB7CgkJLyogU2FuaXR5IGNoZWNrLi4uICovCgkJaWYgKHJxLT5jdXJyZW50X25yX3NlY3RvcnMgIT0gYmlvX2N1cl9zZWN0b3JzKHJxLT5iaW8pICYmCgkJCShycS0+c2VjdG9yICYgKHNlY3RvcnNfcGVyX2ZyYW1lIC0gMSkpKSB7CgkJCXByaW50ayhLRVJOX0VSUiAiJXM6IGNkcm9tX3N0YXJ0X3JlYWRfY29udGludWF0aW9uOiBidWZmZXIgYm90Y2ggKCV1KVxuIiwKCQkJCWRyaXZlLT5uYW1lLCBycS0+Y3VycmVudF9ucl9zZWN0b3JzKTsKCQkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CgkJfQoJCXJxLT5jdXJyZW50X25yX3NlY3RvcnMgKz0gbnNraXA7Cgl9CgoJLyogU2V0IHVwIHRoZSBjb21tYW5kICovCglycS0+dGltZW91dCA9IEFUQVBJX1dBSVRfUEM7CgoJLyogU2VuZCB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUgYW5kIHJldHVybi4gKi8KCXJldHVybiBjZHJvbV90cmFuc2Zlcl9wYWNrZXRfY29tbWFuZChkcml2ZSwgcnEsICZjZHJvbV9yZWFkX2ludHIpOwp9CgoKI2RlZmluZSBJREVDRF9TRUVLX1RIUkVTSE9MRAkoMTAwMCkJCQkvKiAxMDAwIGJsb2NrcyAqLwojZGVmaW5lIElERUNEX1NFRUtfVElNRVIJKDUgKiBXQUlUX01JTl9TTEVFUCkJLyogMTAwIG1zICovCiNkZWZpbmUgSURFQ0RfU0VFS19USU1FT1VUCSgyICogV0FJVF9DTUQpCQkvKiAyMCBzZWMgKi8KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc2Vla19pbnRyIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJaW50IHN0YXQ7CglzdGF0aWMgaW50IHJldHJ5ID0gMTA7CgoJaWYgKGNkcm9tX2RlY29kZV9zdGF0dXMoZHJpdmUsIDAsICZzdGF0KSkKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zZWVraW5nID0gMTsKCglpZiAocmV0cnkgJiYgdGltZV9hZnRlcihqaWZmaWVzLCBpbmZvLT5zdGFydF9zZWVrICsgSURFQ0RfU0VFS19USU1FUikpIHsKCQlpZiAoLS1yZXRyeSA9PSAwKSB7CgkJCS8qCgkJCSAqIHRoaXMgY29uZGl0aW9uIGlzIGZhciB0b28gY29tbW9uLCB0byBib3RoZXIKCQkJICogdXNlcnMgYWJvdXQgaXQKCQkJICovCgkJCS8qIHByaW50aygiJXM6IGRpc2FibGVkIERTQyBzZWVrIG92ZXJsYXBcbiIsIGRyaXZlLT5uYW1lKTsqLyAKCQkJZHJpdmUtPmRzY19vdmVybGFwID0gMDsKCQl9Cgl9CglyZXR1cm4gaWRlX3N0b3BwZWQ7Cn0KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc3RhcnRfc2Vla19jb250aW51YXRpb24gKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJc2VjdG9yX3QgZnJhbWUgPSBycS0+c2VjdG9yOwoKCXNlY3Rvcl9kaXYoZnJhbWUsIHF1ZXVlX2hhcmRzZWN0X3NpemUoZHJpdmUtPnF1ZXVlKSA+PiBTRUNUT1JfQklUUyk7CgoJbWVtc2V0KHJxLT5jbWQsIDAsIHNpemVvZihycS0+Y21kKSk7CglycS0+Y21kWzBdID0gR1BDTURfU0VFSzsKCXB1dF91bmFsaWduZWQoY3B1X3RvX2JlMzIoZnJhbWUpLCAodW5zaWduZWQgaW50ICopICZycS0+Y21kWzJdKTsKCglycS0+dGltZW91dCA9IEFUQVBJX1dBSVRfUEM7CglyZXR1cm4gY2Ryb21fdHJhbnNmZXJfcGFja2V0X2NvbW1hbmQoZHJpdmUsIHJxLCAmY2Ryb21fc2Vla19pbnRyKTsKfQoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9zdGFydF9zZWVrIChpZGVfZHJpdmVfdCAqZHJpdmUsIHVuc2lnbmVkIGludCBibG9jaykKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJaW5mby0+ZG1hID0gMDsKCWluZm8tPnN0YXJ0X3NlZWsgPSBqaWZmaWVzOwoJcmV0dXJuIGNkcm9tX3N0YXJ0X3BhY2tldF9jb21tYW5kKGRyaXZlLCAwLCBjZHJvbV9zdGFydF9zZWVrX2NvbnRpbnVhdGlvbik7Cn0KCi8qIEZpeCB1cCBhIHBvc3NpYmx5IHBhcnRpYWxseS1wcm9jZXNzZWQgcmVxdWVzdCBzbyB0aGF0IHdlIGNhbgogICBzdGFydCBpdCBvdmVyIGVudGlyZWx5LCBvciBldmVuIHB1dCBpdCBiYWNrIG9uIHRoZSByZXF1ZXN0IHF1ZXVlLiAqLwpzdGF0aWMgdm9pZCByZXN0b3JlX3JlcXVlc3QgKHN0cnVjdCByZXF1ZXN0ICpycSkKewoJaWYgKHJxLT5idWZmZXIgIT0gYmlvX2RhdGEocnEtPmJpbykpIHsKCQlzZWN0b3JfdCBuID0gKHJxLT5idWZmZXIgLSAoY2hhciAqKSBiaW9fZGF0YShycS0+YmlvKSkgLyBTRUNUT1JfU0laRTsKCgkJcnEtPmJ1ZmZlciA9IGJpb19kYXRhKHJxLT5iaW8pOwoJCXJxLT5ucl9zZWN0b3JzICs9IG47CgkJcnEtPnNlY3RvciAtPSBuOwoJfQoJcnEtPmhhcmRfY3VyX3NlY3RvcnMgPSBycS0+Y3VycmVudF9ucl9zZWN0b3JzID0gYmlvX2N1cl9zZWN0b3JzKHJxLT5iaW8pOwoJcnEtPmhhcmRfbnJfc2VjdG9ycyA9IHJxLT5ucl9zZWN0b3JzOwoJcnEtPmhhcmRfc2VjdG9yID0gcnEtPnNlY3RvcjsKCXJxLT5xLT5wcmVwX3JxX2ZuKHJxLT5xLCBycSk7Cn0KCi8qCiAqIFN0YXJ0IGEgcmVhZCByZXF1ZXN0IGZyb20gdGhlIENELVJPTS4KICovCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc3RhcnRfcmVhZCAoaWRlX2RyaXZlX3QgKmRyaXZlLCB1bnNpZ25lZCBpbnQgYmxvY2spCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IHJlcXVlc3QgKnJxID0gSFdHUk9VUChkcml2ZSktPnJxOwoJdW5zaWduZWQgc2hvcnQgc2VjdG9yc19wZXJfZnJhbWU7CgoJc2VjdG9yc19wZXJfZnJhbWUgPSBxdWV1ZV9oYXJkc2VjdF9zaXplKGRyaXZlLT5xdWV1ZSkgPj4gU0VDVE9SX0JJVFM7CgoJLyogV2UgbWF5IGJlIHJldHJ5aW5nIHRoaXMgcmVxdWVzdCBhZnRlciBhbiBlcnJvci4gIEZpeCB1cAoJICAgYW55IHdlaXJkbmVzcyB3aGljaCBtaWdodCBiZSBwcmVzZW50IGluIHRoZSByZXF1ZXN0IHBhY2tldC4gKi8KCXJlc3RvcmVfcmVxdWVzdChycSk7CgoJLyogU2F0aXNmeSB3aGF0ZXZlciB3ZSBjYW4gb2YgdGhpcyByZXF1ZXN0IGZyb20gb3VyIGNhY2hlZCBzZWN0b3IuICovCglpZiAoY2Ryb21fcmVhZF9mcm9tX2J1ZmZlcihkcml2ZSkpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCWJsa19hdHRlbXB0X3JlbWVyZ2UoZHJpdmUtPnF1ZXVlLCBycSk7CgoJLyogQ2xlYXIgdGhlIGxvY2FsIHNlY3RvciBidWZmZXIuICovCglpbmZvLT5uc2VjdG9yc19idWZmZXJlZCA9IDA7CgoJLyogdXNlIGRtYSwgaWYgcG9zc2libGUuICovCglpbmZvLT5kbWEgPSBkcml2ZS0+dXNpbmdfZG1hOwoJaWYgKChycS0+c2VjdG9yICYgKHNlY3RvcnNfcGVyX2ZyYW1lIC0gMSkpIHx8CgkgICAgKHJxLT5ucl9zZWN0b3JzICYgKHNlY3RvcnNfcGVyX2ZyYW1lIC0gMSkpKQoJCWluZm8tPmRtYSA9IDA7CgoJLyogU3RhcnQgc2VuZGluZyB0aGUgcmVhZCByZXF1ZXN0IHRvIHRoZSBkcml2ZS4gKi8KCXJldHVybiBjZHJvbV9zdGFydF9wYWNrZXRfY29tbWFuZChkcml2ZSwgMzI3NjgsIGNkcm9tX3N0YXJ0X3JlYWRfY29udGludWF0aW9uKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogRXhlY3V0ZSBhbGwgb3RoZXIgcGFja2V0IGNvbW1hbmRzLgogKi8KCi8qIEludGVycnVwdCByb3V0aW5lIGZvciBwYWNrZXQgY29tbWFuZCBjb21wbGV0aW9uLiAqLwpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX3BjX2ludHIgKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJaW50IGlyZWFzb24sIGxlbiwgdGhpc2xlbjsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCXU4IGxvd2N5bCA9IDAsIGhpZ2hjeWwgPSAwOwoJaW50IHN0YXQ7CgoJLyogQ2hlY2sgZm9yIGVycm9ycy4gKi8KCWlmIChjZHJvbV9kZWNvZGVfc3RhdHVzKGRyaXZlLCAwLCAmc3RhdCkpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCS8qIFJlYWQgdGhlIGludGVycnVwdCByZWFzb24gYW5kIHRoZSB0cmFuc2ZlciBsZW5ndGguICovCglpcmVhc29uID0gSFdJRihkcml2ZSktPklOQihJREVfSVJFQVNPTl9SRUcpOwoJbG93Y3lsICA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVExfUkVHKTsKCWhpZ2hjeWwgPSBIV0lGKGRyaXZlKS0+SU5CKElERV9CQ09VTlRIX1JFRyk7CgoJbGVuID0gbG93Y3lsICsgKDI1NiAqIGhpZ2hjeWwpOwoKCS8qIElmIERSUSBpcyBjbGVhciwgdGhlIGNvbW1hbmQgaGFzIGNvbXBsZXRlZC4KCSAgIENvbXBsYWluIGlmIHdlIHN0aWxsIGhhdmUgZGF0YSBsZWZ0IHRvIHRyYW5zZmVyLiAqLwoJaWYgKChzdGF0ICYgRFJRX1NUQVQpID09IDApIHsKCQkvKiBTb21lIG9mIHRoZSB0cmFpbGluZyByZXF1ZXN0IHNlbnNlIGZpZWxkcyBhcmUgb3B0aW9uYWwsIGFuZAoJCSAgIHNvbWUgZHJpdmVzIGRvbid0IHNlbmQgdGhlbS4gIFNpZ2guICovCgkJaWYgKHJxLT5jbWRbMF0gPT0gR1BDTURfUkVRVUVTVF9TRU5TRSAmJgoJCSAgICBycS0+ZGF0YV9sZW4gPiAwICYmCgkJICAgIHJxLT5kYXRhX2xlbiA8PSA1KSB7CgkJCXdoaWxlIChycS0+ZGF0YV9sZW4gPiAwKSB7CgkJCQkqKHVuc2lnbmVkIGNoYXIgKilycS0+ZGF0YSsrID0gMDsKCQkJCS0tcnEtPmRhdGFfbGVuOwoJCQl9CgkJfQoKCQlpZiAocnEtPmRhdGFfbGVuID09IDApCgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAxKTsKCQllbHNlIHsKCQkJLyogQ29tbWVudCB0aGlzIG91dCwgYmVjYXVzZSB0aGlzIGFsd2F5cyBoYXBwZW5zIAoJCQkgICByaWdodCBhZnRlciBhIHJlc2V0IG9jY3VycywgYW5kIGl0IGlzIGFubm95aW5nIHRvIAoJCQkgICBhbHdheXMgcHJpbnQgZXhwZWN0ZWQgc3R1ZmYuICAqLwoJCQkvKgoJCQlwcmludGsgKCIlczogY2Ryb21fcGNfaW50cjogZGF0YSB1bmRlcnJ1biAlZFxuIiwKCQkJCWRyaXZlLT5uYW1lLCBwYy0+YnVmbGVuKTsKCQkJKi8KCQkJcnEtPmZsYWdzIHw9IFJFUV9GQUlMRUQ7CgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQl9CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCS8qIEZpZ3VyZSBvdXQgaG93IG11Y2ggZGF0YSB0byB0cmFuc2Zlci4gKi8KCXRoaXNsZW4gPSBycS0+ZGF0YV9sZW47CglpZiAodGhpc2xlbiA+IGxlbikgdGhpc2xlbiA9IGxlbjsKCgkvKiBUaGUgZHJpdmUgd2FudHMgdG8gYmUgd3JpdHRlbiB0by4gKi8KCWlmICgoaXJlYXNvbiAmIDMpID09IDApIHsKCQlpZiAoIXJxLT5kYXRhKSB7CgkJCWJsa19kdW1wX3JxX2ZsYWdzKHJxLCAiY2Ryb21fcGNfaW50ciwgd3JpdGUiKTsKCQkJZ290byBjb25mdXNlZDsKCQl9CgkJLyogVHJhbnNmZXIgdGhlIGRhdGEuICovCgkJSFdJRihkcml2ZSktPmF0YXBpX291dHB1dF9ieXRlcyhkcml2ZSwgcnEtPmRhdGEsIHRoaXNsZW4pOwoKCQkvKiBJZiB3ZSBoYXZlbid0IG1vdmVkIGVub3VnaCBkYXRhIHRvIHNhdGlzZnkgdGhlIGRyaXZlLAoJCSAgIGFkZCBzb21lIHBhZGRpbmcuICovCgkJd2hpbGUgKGxlbiA+IHRoaXNsZW4pIHsKCQkJaW50IGR1bSA9IDA7CgkJCUhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXMoZHJpdmUsICZkdW0sIHNpemVvZihkdW0pKTsKCQkJbGVuIC09IHNpemVvZihkdW0pOwoJCX0KCgkJLyogS2VlcCBjb3VudCBvZiBob3cgbXVjaCBkYXRhIHdlJ3ZlIG1vdmVkLiAqLwoJCXJxLT5kYXRhICs9IHRoaXNsZW47CgkJcnEtPmRhdGFfbGVuIC09IHRoaXNsZW47Cgl9CgoJLyogU2FtZSBkcmlsbCBmb3IgcmVhZGluZy4gKi8KCWVsc2UgaWYgKChpcmVhc29uICYgMykgPT0gMikgewoJCWlmICghcnEtPmRhdGEpIHsKCQkJYmxrX2R1bXBfcnFfZmxhZ3MocnEsICJjZHJvbV9wY19pbnRyLCB3cml0ZSIpOwoJCQlnb3RvIGNvbmZ1c2VkOwoJCX0KCQkvKiBUcmFuc2ZlciB0aGUgZGF0YS4gKi8KCQlIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXMoZHJpdmUsIHJxLT5kYXRhLCB0aGlzbGVuKTsKCgkJLyogSWYgd2UgaGF2ZW4ndCBtb3ZlZCBlbm91Z2ggZGF0YSB0byBzYXRpc2Z5IHRoZSBkcml2ZSwKCQkgICBhZGQgc29tZSBwYWRkaW5nLiAqLwoJCXdoaWxlIChsZW4gPiB0aGlzbGVuKSB7CgkJCWludCBkdW0gPSAwOwoJCQlIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXMoZHJpdmUsICZkdW0sIHNpemVvZihkdW0pKTsKCQkJbGVuIC09IHNpemVvZihkdW0pOwoJCX0KCgkJLyogS2VlcCBjb3VudCBvZiBob3cgbXVjaCBkYXRhIHdlJ3ZlIG1vdmVkLiAqLwoJCXJxLT5kYXRhICs9IHRoaXNsZW47CgkJcnEtPmRhdGFfbGVuIC09IHRoaXNsZW47CgoJCWlmIChycS0+ZmxhZ3MgJiBSRVFfU0VOU0UpCgkJCXJxLT5zZW5zZV9sZW4gKz0gdGhpc2xlbjsKCX0gZWxzZSB7CmNvbmZ1c2VkOgoJCXByaW50ayAoS0VSTl9FUlIgIiVzOiBjZHJvbV9wY19pbnRyOiBUaGUgZHJpdmUgIgoJCQkiYXBwZWFycyBjb25mdXNlZCAoaXJlYXNvbiA9IDB4JTAyeClcbiIsCgkJCWRyaXZlLT5uYW1lLCBpcmVhc29uKTsKCQlycS0+ZmxhZ3MgfD0gUkVRX0ZBSUxFRDsKCX0KCgkvKiBOb3cgd2Ugd2FpdCBmb3IgYW5vdGhlciBpbnRlcnJ1cHQuICovCglpZGVfc2V0X2hhbmRsZXIoZHJpdmUsICZjZHJvbV9wY19pbnRyLCBBVEFQSV9XQUlUX1BDLCBjZHJvbV90aW1lcl9leHBpcnkpOwoJcmV0dXJuIGlkZV9zdGFydGVkOwp9CgpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90IGNkcm9tX2RvX3BjX2NvbnRpbnVhdGlvbiAoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CgoJaWYgKCFycS0+dGltZW91dCkKCQlycS0+dGltZW91dCA9IEFUQVBJX1dBSVRfUEM7CgoJLyogU2VuZCB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUgYW5kIHJldHVybi4gKi8KCXJldHVybiBjZHJvbV90cmFuc2Zlcl9wYWNrZXRfY29tbWFuZChkcml2ZSwgcnEsICZjZHJvbV9wY19pbnRyKTsKfQoKCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fZG9fcGFja2V0X2NvbW1hbmQgKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJaW50IGxlbjsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoKCWluZm8tPmRtYSA9IDA7CglycS0+ZmxhZ3MgJj0gflJFUV9GQUlMRUQ7CglsZW4gPSBycS0+ZGF0YV9sZW47CgoJLyogU3RhcnQgc2VuZGluZyB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUuICovCglyZXR1cm4gY2Ryb21fc3RhcnRfcGFja2V0X2NvbW1hbmQoZHJpdmUsIGxlbiwgY2Ryb21fZG9fcGNfY29udGludWF0aW9uKTsKfQoKCnN0YXRpYwppbnQgY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEpCnsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIHNlbnNlOwoJaW50IHJldHJpZXMgPSAxMDsKCXVuc2lnbmVkIGludCBmbGFncyA9IHJxLT5mbGFnczsKCglpZiAocnEtPnNlbnNlID09IE5VTEwpCgkJcnEtPnNlbnNlID0gJnNlbnNlOwoKCS8qIFN0YXJ0IG9mIHJldHJ5IGxvb3AuICovCglkbyB7CgkJaW50IGVycm9yOwoJCXVuc2lnbmVkIGxvbmcgdGltZSA9IGppZmZpZXM7CgkJcnEtPmZsYWdzID0gZmxhZ3M7CgoJCWVycm9yID0gaWRlX2RvX2RyaXZlX2NtZChkcml2ZSwgcnEsIGlkZV93YWl0KTsKCQl0aW1lID0gamlmZmllcyAtIHRpbWU7CgoJCS8qIEZJWE1FOiB3ZSBzaG91bGQgcHJvYmFibHkgYWJvcnQvcmV0cnkgb3Igc29tZXRoaW5nIAoJCSAqIGluIGNhc2Ugb2YgZmFpbHVyZSAqLwoJCWlmIChycS0+ZmxhZ3MgJiBSRVFfRkFJTEVEKSB7CgkJCS8qIFRoZSByZXF1ZXN0IGZhaWxlZC4gIFJldHJ5IGlmIGl0IHdhcyBkdWUgdG8gYSB1bml0CgkJCSAgIGF0dGVudGlvbiBzdGF0dXMKCQkJICAgKHVzdWFsbHkgbWVhbnMgbWVkaWEgd2FzIGNoYW5nZWQpLiAqLwoJCQlzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqcmVxYnVmID0gcnEtPnNlbnNlOwoKCQkJaWYgKHJlcWJ1Zi0+c2Vuc2Vfa2V5ID09IFVOSVRfQVRURU5USU9OKQoJCQkJY2Ryb21fc2F3X21lZGlhX2NoYW5nZShkcml2ZSk7CgkJCWVsc2UgaWYgKHJlcWJ1Zi0+c2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSAmJgoJCQkJIHJlcWJ1Zi0+YXNjID09IDQgJiYgcmVxYnVmLT5hc2NxICE9IDQpIHsKCQkJCS8qIFRoZSBkcml2ZSBpcyBpbiB0aGUgcHJvY2VzcyBvZiBsb2FkaW5nCgkJCQkgICBhIGRpc2suICBSZXRyeSwgYnV0IHdhaXQgYSBsaXR0bGUgdG8gZ2l2ZQoJCQkJICAgdGhlIGRyaXZlIHRpbWUgdG8gY29tcGxldGUgdGhlIGxvYWQuICovCgkJCQlzc2xlZXAoMik7CgkJCX0gZWxzZSB7CgkJCQkvKiBPdGhlcndpc2UsIGRvbid0IHJldHJ5LiAqLwoJCQkJcmV0cmllcyA9IDA7CgkJCX0KCQkJLS1yZXRyaWVzOwoJCX0KCgkJLyogRW5kIG9mIHJldHJ5IGxvb3AuICovCgl9IHdoaWxlICgocnEtPmZsYWdzICYgUkVRX0ZBSUxFRCkgJiYgcmV0cmllcyA+PSAwKTsKCgkvKiBSZXR1cm4gYW4gZXJyb3IgaWYgdGhlIGNvbW1hbmQgZmFpbGVkLiAqLwoJcmV0dXJuIChycS0+ZmxhZ3MgJiBSRVFfRkFJTEVEKSA/IC1FSU8gOiAwOwp9CgovKgogKiBXcml0ZSBoYW5kbGluZwogKi8Kc3RhdGljIGlubGluZSBpbnQgY2Ryb21fd3JpdGVfY2hlY2tfaXJlYXNvbihpZGVfZHJpdmVfdCAqZHJpdmUsIGludCBsZW4sIGludCBpcmVhc29uKQp7CgkvKiBUd28gbm90ZXMgYWJvdXQgSURFIGludGVycnVwdCByZWFzb24gaGVyZSAtIDAgbWVhbnMgdGhhdAoJICogdGhlIGRyaXZlIHdhbnRzIHRvIHJlY2VpdmUgZGF0YSBmcm9tIHVzLCAyIG1lYW5zIHRoYXQKCSAqIHRoZSBkcml2ZSBpcyBleHBlY3RpbmcgdG8gdHJhbnNmZXIgZGF0YSB0byB1cy4KCSAqLwoJaWYgKGlyZWFzb24gPT0gMCkKCQlyZXR1cm4gMDsKCWVsc2UgaWYgKGlyZWFzb24gPT0gMikgewoJCS8qIFdob29wcy4uLiBUaGUgZHJpdmUgd2FudHMgdG8gc2VuZCBkYXRhLiAqLwoJCXByaW50ayhLRVJOX0VSUiAiJXM6IHdyaXRlX2ludHI6IHdyb25nIHRyYW5zZmVyIGRpcmVjdGlvbiFcbiIsCgkJCQkJCQlkcml2ZS0+bmFtZSk7CgoJCXdoaWxlIChsZW4gPiAwKSB7CgkJCWludCBkdW0gPSAwOwoJCQlIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXMoZHJpdmUsICZkdW0sIHNpemVvZihkdW0pKTsKCQkJbGVuIC09IHNpemVvZihkdW0pOwoJCX0KCX0gZWxzZSB7CgkJLyogRHJpdmUgd2FudHMgYSBjb21tYW5kIHBhY2tldCwgb3IgaW52YWxpZCBpcmVhc29uLi4uICovCgkJcHJpbnRrKEtFUk5fRVJSICIlczogd3JpdGVfaW50cjogYmFkIGludGVycnVwdCByZWFzb24gJXhcbiIsCgkJCQkJCQlkcml2ZS0+bmFtZSwgaXJlYXNvbik7Cgl9CgoJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDApOwoJcmV0dXJuIDE7Cn0KCnN0YXRpYyB2b2lkIHBvc3RfdHJhbnNmb3JtX2NvbW1hbmQoc3RydWN0IHJlcXVlc3QgKnJlcSkKewoJdTggKmMgPSByZXEtPmNtZDsKCWNoYXIgKmlidWY7CgoJaWYgKCFibGtfcGNfcmVxdWVzdChyZXEpKQoJCXJldHVybjsKCglpZiAocmVxLT5iaW8pCgkJaWJ1ZiA9IGJpb19kYXRhKHJlcS0+YmlvKTsKCWVsc2UKCQlpYnVmID0gcmVxLT5kYXRhOwoKCWlmICghaWJ1ZikKCQlyZXR1cm47CgoJLyoKCSAqIHNldCBhbnNpLXJldmlzaW9uIGFuZCByZXNwb25zZSBkYXRhIGFzIGF0YXBpCgkgKi8KCWlmIChjWzBdID09IEdQQ01EX0lOUVVJUlkpIHsKCQlpYnVmWzJdIHw9IDI7CgkJaWJ1ZlszXSA9IChpYnVmWzNdICYgMHhmMCkgfCAyOwoJfQp9Cgp0eXBlZGVmIHZvaWQgKHhmZXJfZnVuY190KShpZGVfZHJpdmVfdCAqLCB2b2lkICosIHUzMik7CgovKgogKiBiZXN0IHdheSB0byBkZWFsIHdpdGggZG1hIHRoYXQgaXMgbm90IHNlY3RvciBhbGlnbmVkIHJpZ2h0IG5vdy4uLiBub3RlCiAqIHRoYXQgaW4gdGhpcyBwYXRoIHdlIGFyZSBub3QgdXNpbmcgLT5kYXRhIG9yIC0+YnVmZmVyIGF0IGFsbC4gdGhpcyBpcnMKICogY2FuIHJlcGxhY2UgY2Ryb21fcGNfaW50ciwgY2Ryb21fcmVhZF9pbnRyLCBhbmQgY2Ryb21fd3JpdGVfaW50ciBpbiB0aGUKICogZnV0dXJlLgogKi8Kc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9uZXdwY19pbnRyKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CglpbnQgZG1hX2Vycm9yLCBkbWEsIHN0YXQsIGlyZWFzb24sIGxlbiwgdGhpc2xlbjsKCXU4IGxvd2N5bCwgaGlnaGN5bDsKCXhmZXJfZnVuY190ICp4ZmVyZnVuYzsKCXVuc2lnbmVkIGxvbmcgZmxhZ3M7CgoJLyogQ2hlY2sgZm9yIGVycm9ycy4gKi8KCWRtYV9lcnJvciA9IDA7CglkbWEgPSBpbmZvLT5kbWE7CglpZiAoZG1hKSB7CgkJaW5mby0+ZG1hID0gMDsKCQlkbWFfZXJyb3IgPSBIV0lGKGRyaXZlKS0+aWRlX2RtYV9lbmQoZHJpdmUpOwoJfQoKCWlmIChjZHJvbV9kZWNvZGVfc3RhdHVzKGRyaXZlLCAwLCAmc3RhdCkpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCS8qCgkgKiB1c2luZyBkbWEsIHRyYW5zZmVyIGlzIGNvbXBsZXRlIG5vdwoJICovCglpZiAoZG1hKSB7CgkJaWYgKGRtYV9lcnJvcikgewoJCQlwcmludGsoS0VSTl9FUlIgImlkZS1jZDogZG1hIGVycm9yXG4iKTsKCQkJX19pZGVfZG1hX29mZihkcml2ZSk7CgkJCXJldHVybiBpZGVfZXJyb3IoZHJpdmUsICJkbWEgZXJyb3IiLCBzdGF0KTsKCQl9CgoJCWVuZF90aGF0X3JlcXVlc3RfY2h1bmsocnEsIDEsIHJxLT5kYXRhX2xlbik7CgkJcnEtPmRhdGFfbGVuID0gMDsKCQlnb3RvIGVuZF9yZXF1ZXN0OwoJfQoKCS8qCgkgKiBvayB3ZSBmYWxsIHRvIHBpbyA6LwoJICovCglpcmVhc29uID0gSFdJRihkcml2ZSktPklOQihJREVfSVJFQVNPTl9SRUcpICYgMHgzOwoJbG93Y3lsICA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVExfUkVHKTsKCWhpZ2hjeWwgPSBIV0lGKGRyaXZlKS0+SU5CKElERV9CQ09VTlRIX1JFRyk7CgoJbGVuID0gbG93Y3lsICsgKDI1NiAqIGhpZ2hjeWwpOwoJdGhpc2xlbiA9IHJxLT5kYXRhX2xlbjsKCWlmICh0aGlzbGVuID4gbGVuKQoJCXRoaXNsZW4gPSBsZW47CgoJLyoKCSAqIElmIERSUSBpcyBjbGVhciwgdGhlIGNvbW1hbmQgaGFzIGNvbXBsZXRlZC4KCSAqLwoJaWYgKChzdGF0ICYgRFJRX1NUQVQpID09IDApCgkJZ290byBlbmRfcmVxdWVzdDsKCgkvKgoJICogY2hlY2sgd2hpY2ggd2F5IHRvIHRyYW5zZmVyIGRhdGEKCSAqLwoJaWYgKHJxX2RhdGFfZGlyKHJxKSA9PSBXUklURSkgewoJCS8qCgkJICogd3JpdGUgdG8gZHJpdmUKCQkgKi8KCQlpZiAoY2Ryb21fd3JpdGVfY2hlY2tfaXJlYXNvbihkcml2ZSwgbGVuLCBpcmVhc29uKSkKCQkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCQl4ZmVyZnVuYyA9IEhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXM7Cgl9IGVsc2UgIHsKCQkvKgoJCSAqIHJlYWQgZnJvbSBkcml2ZQoJCSAqLwoJCWlmIChjZHJvbV9yZWFkX2NoZWNrX2lyZWFzb24oZHJpdmUsIGxlbiwgaXJlYXNvbikpCgkJCXJldHVybiBpZGVfc3RvcHBlZDsKCgkJeGZlcmZ1bmMgPSBIV0lGKGRyaXZlKS0+YXRhcGlfaW5wdXRfYnl0ZXM7Cgl9CgoJLyoKCSAqIHRyYW5zZmVyIGRhdGEKCSAqLwoJd2hpbGUgKHRoaXNsZW4gPiAwKSB7CgkJaW50IGJsZW4gPSBibGVuID0gcnEtPmRhdGFfbGVuOwoJCWNoYXIgKnB0ciA9IHJxLT5kYXRhOwoKCQkvKgoJCSAqIGJpbyBiYWNrZWQ/CgkJICovCgkJaWYgKHJxLT5iaW8pIHsKCQkJcHRyID0gYmlvX2RhdGEocnEtPmJpbyk7CgkJCWJsZW4gPSBiaW9faW92ZWMocnEtPmJpbyktPmJ2X2xlbjsKCQl9CgoJCWlmICghcHRyKSB7CgkJCXByaW50ayhLRVJOX0VSUiAiJXM6IGNvbmZ1c2VkLCBtaXNzaW5nIGRhdGFcbiIsIGRyaXZlLT5uYW1lKTsKCQkJYnJlYWs7CgkJfQoKCQlpZiAoYmxlbiA+IHRoaXNsZW4pCgkJCWJsZW4gPSB0aGlzbGVuOwoKCQl4ZmVyZnVuYyhkcml2ZSwgcHRyLCBibGVuKTsKCgkJdGhpc2xlbiAtPSBibGVuOwoJCWxlbiAtPSBibGVuOwoJCXJxLT5kYXRhX2xlbiAtPSBibGVuOwoKCQlpZiAocnEtPmJpbykKCQkJZW5kX3RoYXRfcmVxdWVzdF9jaHVuayhycSwgMSwgYmxlbik7CgkJZWxzZQoJCQlycS0+ZGF0YSArPSBibGVuOwoJfQoKCS8qCgkgKiBwYWQsIGlmIG5lY2Vzc2FyeQoJICovCglpZiAobGVuID4gMCkgewoJCXdoaWxlIChsZW4gPiAwKSB7CgkJCWludCBwYWQgPSAwOwoKCQkJeGZlcmZ1bmMoZHJpdmUsICZwYWQsIHNpemVvZihwYWQpKTsKCQkJbGVuIC09IHNpemVvZihwYWQpOwoJCX0KCX0KCglpZiAoSFdHUk9VUChkcml2ZSktPmhhbmRsZXIgIT0gTlVMTCkKCQlCVUcoKTsKCglpZGVfc2V0X2hhbmRsZXIoZHJpdmUsIGNkcm9tX25ld3BjX2ludHIsIHJxLT50aW1lb3V0LCBOVUxMKTsKCXJldHVybiBpZGVfc3RhcnRlZDsKCmVuZF9yZXF1ZXN0OgoJaWYgKCFycS0+ZGF0YV9sZW4pCgkJcG9zdF90cmFuc2Zvcm1fY29tbWFuZChycSk7CgoJc3Bpbl9sb2NrX2lycXNhdmUoJmlkZV9sb2NrLCBmbGFncyk7CglibGtkZXZfZGVxdWV1ZV9yZXF1ZXN0KHJxKTsKCWVuZF90aGF0X3JlcXVlc3RfbGFzdChycSwgMSk7CglIV0dST1VQKGRyaXZlKS0+cnEgPSBOVUxMOwoJc3Bpbl91bmxvY2tfaXJxcmVzdG9yZSgmaWRlX2xvY2ssIGZsYWdzKTsKCXJldHVybiBpZGVfc3RvcHBlZDsKfQoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV93cml0ZV9pbnRyKGlkZV9kcml2ZV90ICpkcml2ZSkKewoJaW50IHN0YXQsIGlyZWFzb24sIGxlbiwgc2VjdG9yc190b190cmFuc2ZlciwgdXB0b2RhdGU7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCWludCBkbWFfZXJyb3IgPSAwLCBkbWEgPSBpbmZvLT5kbWE7Cgl1OCBsb3djeWwgPSAwLCBoaWdoY3lsID0gMDsKCglzdHJ1Y3QgcmVxdWVzdCAqcnEgPSBIV0dST1VQKGRyaXZlKS0+cnE7CgoJLyogQ2hlY2sgZm9yIGVycm9ycy4gKi8KCWlmIChkbWEpIHsKCQlpbmZvLT5kbWEgPSAwOwoJCWlmICgoZG1hX2Vycm9yID0gSFdJRihkcml2ZSktPmlkZV9kbWFfZW5kKGRyaXZlKSkpIHsKCQkJcHJpbnRrKEtFUk5fRVJSICJpZGUtY2Q6IHdyaXRlIGRtYSBlcnJvclxuIik7CgkJCV9faWRlX2RtYV9vZmYoZHJpdmUpOwoJCX0KCX0KCglpZiAoY2Ryb21fZGVjb2RlX3N0YXR1cyhkcml2ZSwgMCwgJnN0YXQpKQoJCXJldHVybiBpZGVfc3RvcHBlZDsKCgkvKgoJICogdXNpbmcgZG1hLCB0cmFuc2ZlciBpcyBjb21wbGV0ZSBub3cKCSAqLwoJaWYgKGRtYSkgewoJCWlmIChkbWFfZXJyb3IpCgkJCXJldHVybiBpZGVfZXJyb3IoZHJpdmUsICJkbWEgZXJyb3IiLCBzdGF0KTsKCgkJaWRlX2VuZF9yZXF1ZXN0KGRyaXZlLCAxLCBycS0+bnJfc2VjdG9ycyk7CgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoJfQoKCS8qIFJlYWQgdGhlIGludGVycnVwdCByZWFzb24gYW5kIHRoZSB0cmFuc2ZlciBsZW5ndGguICovCglpcmVhc29uID0gSFdJRihkcml2ZSktPklOQihJREVfSVJFQVNPTl9SRUcpOwoJbG93Y3lsICA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX0JDT1VOVExfUkVHKTsKCWhpZ2hjeWwgPSBIV0lGKGRyaXZlKS0+SU5CKElERV9CQ09VTlRIX1JFRyk7CgoJbGVuID0gbG93Y3lsICsgKDI1NiAqIGhpZ2hjeWwpOwoKCS8qIElmIERSUSBpcyBjbGVhciwgdGhlIGNvbW1hbmQgaGFzIGNvbXBsZXRlZC4gKi8KCWlmICgoc3RhdCAmIERSUV9TVEFUKSA9PSAwKSB7CgkJLyogSWYgd2UncmUgbm90IGRvbmUgd3JpdGluZywgY29tcGxhaW4uCgkJICogT3RoZXJ3aXNlLCBjb21wbGV0ZSB0aGUgY29tbWFuZCBub3JtYWxseS4KCQkgKi8KCQl1cHRvZGF0ZSA9IDE7CgkJaWYgKHJxLT5jdXJyZW50X25yX3NlY3RvcnMgPiAwKSB7CgkJCXByaW50ayhLRVJOX0VSUiAiJXM6IHdyaXRlX2ludHI6IGRhdGEgdW5kZXJydW4gKCVkIGJsb2NrcylcbiIsCgkJCWRyaXZlLT5uYW1lLCBycS0+Y3VycmVudF9ucl9zZWN0b3JzKTsKCQkJdXB0b2RhdGUgPSAwOwoJCX0KCQljZHJvbV9lbmRfcmVxdWVzdChkcml2ZSwgdXB0b2RhdGUpOwoJCXJldHVybiBpZGVfc3RvcHBlZDsKCX0KCgkvKiBDaGVjayB0aGF0IHRoZSBkcml2ZSBpcyBleHBlY3RpbmcgdG8gZG8gdGhlIHNhbWUgdGhpbmcgd2UgYXJlLiAqLwoJaWYgKGNkcm9tX3dyaXRlX2NoZWNrX2lyZWFzb24oZHJpdmUsIGxlbiwgaXJlYXNvbikpCgkJcmV0dXJuIGlkZV9zdG9wcGVkOwoKCXNlY3RvcnNfdG9fdHJhbnNmZXIgPSBsZW4gLyBTRUNUT1JfU0laRTsKCgkvKgoJICogbm93IGxvb3AgYW5kIHdyaXRlIG91dCB0aGUgZGF0YQoJICovCgl3aGlsZSAoc2VjdG9yc190b190cmFuc2ZlciA+IDApIHsKCQlpbnQgdGhpc190cmFuc2ZlcjsKCgkJaWYgKCFycS0+Y3VycmVudF9ucl9zZWN0b3JzKSB7CgkJCXByaW50ayhLRVJOX0VSUiAiaWRlLWNkOiB3cml0ZV9pbnRyOiBvb3BzXG4iKTsKCQkJYnJlYWs7CgkJfQoKCQkvKgoJCSAqIEZpZ3VyZSBvdXQgaG93IG1hbnkgc2VjdG9ycyB3ZSBjYW4gdHJhbnNmZXIKCQkgKi8KCQl0aGlzX3RyYW5zZmVyID0gbWluX3QoaW50LCBzZWN0b3JzX3RvX3RyYW5zZmVyLCBycS0+Y3VycmVudF9ucl9zZWN0b3JzKTsKCgkJd2hpbGUgKHRoaXNfdHJhbnNmZXIgPiAwKSB7CgkJCUhXSUYoZHJpdmUpLT5hdGFwaV9vdXRwdXRfYnl0ZXMoZHJpdmUsIHJxLT5idWZmZXIsIFNFQ1RPUl9TSVpFKTsKCQkJcnEtPmJ1ZmZlciArPSBTRUNUT1JfU0laRTsKCQkJLS1ycS0+bnJfc2VjdG9yczsKCQkJLS1ycS0+Y3VycmVudF9ucl9zZWN0b3JzOwoJCQkrK3JxLT5zZWN0b3I7CgkJCS0tdGhpc190cmFuc2ZlcjsKCQkJLS1zZWN0b3JzX3RvX3RyYW5zZmVyOwoJCX0KCgkJLyoKCQkgKiBjdXJyZW50IGJ1ZmZlciBjb21wbGV0ZSwgbW92ZSBvbgoJCSAqLwoJCWlmIChycS0+Y3VycmVudF9ucl9zZWN0b3JzID09IDAgJiYgcnEtPm5yX3NlY3RvcnMpCgkJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAxKTsKCX0KCgkvKiByZS1hcm0gaGFuZGxlciAqLwoJaWRlX3NldF9oYW5kbGVyKGRyaXZlLCAmY2Ryb21fd3JpdGVfaW50ciwgQVRBUElfV0FJVF9QQywgTlVMTCk7CglyZXR1cm4gaWRlX3N0YXJ0ZWQ7Cn0KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fc3RhcnRfd3JpdGVfY29udChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCiNpZiAwCS8qIHRoZSBpbW1lZGlhdGUgYml0ICovCglycS0+Y21kWzFdID0gMSA8PCAzOwojZW5kaWYKCXJxLT50aW1lb3V0ID0gQVRBUElfV0FJVF9QQzsKCglyZXR1cm4gY2Ryb21fdHJhbnNmZXJfcGFja2V0X2NvbW1hbmQoZHJpdmUsIHJxLCBjZHJvbV93cml0ZV9pbnRyKTsKfQoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9zdGFydF93cml0ZShpZGVfZHJpdmVfdCAqZHJpdmUsIHN0cnVjdCByZXF1ZXN0ICpycSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgZ2VuZGlzayAqZyA9IGluZm8tPmRpc2s7Cgl1bnNpZ25lZCBzaG9ydCBzZWN0b3JzX3Blcl9mcmFtZSA9IHF1ZXVlX2hhcmRzZWN0X3NpemUoZHJpdmUtPnF1ZXVlKSA+PiBTRUNUT1JfQklUUzsKCgkvKgoJICogd3JpdGVzICptdXN0KiBiZSBoYXJkd2FyZSBmcmFtZSBhbGlnbmVkCgkgKi8KCWlmICgocnEtPm5yX3NlY3RvcnMgJiAoc2VjdG9yc19wZXJfZnJhbWUgLSAxKSkgfHwKCSAgICAocnEtPnNlY3RvciAmIChzZWN0b3JzX3Blcl9mcmFtZSAtIDEpKSkgewoJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7Cgl9CgoJLyoKCSAqIGRpc2sgaGFzIGJlY29tZSB3cml0ZSBwcm90ZWN0ZWQKCSAqLwoJaWYgKGctPnBvbGljeSkgewoJCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCQlyZXR1cm4gaWRlX3N0b3BwZWQ7Cgl9CgoJLyoKCSAqIGZvciBkdmQtcmFtIGFuZCBzdWNoIG1lZGlhLCBpdCdzIGEgcmVhbGx5IGJpZyBkZWFsIHRvIGdldAoJICogYmlnIHdyaXRlcyBhbGwgdGhlIHRpbWUuIHNvIHNjb3VyIHRoZSBxdWV1ZSBhbmQgYXR0ZW1wdCB0bwoJICogcmVtZXJnZSByZXF1ZXN0cywgb2Z0ZW4gdGhlIHBsdWdnaW5nIHdpbGwgbm90IGhhdmUgaGFkIHRpbWUKCSAqIHRvIGRvIHRoaXMgcHJvcGVybHkKCSAqLwoJYmxrX2F0dGVtcHRfcmVtZXJnZShkcml2ZS0+cXVldWUsIHJxKTsKCglpbmZvLT5uc2VjdG9yc19idWZmZXJlZCA9IDA7CgoJLyogdXNlIGRtYSwgaWYgcG9zc2libGUuIHdlIGRvbid0IG5lZWQgdG8gY2hlY2sgbW9yZSwgc2luY2Ugd2UKCSAqIGtub3cgdGhhdCB0aGUgdHJhbnNmZXIgaXMgYWx3YXlzIChhdCBsZWFzdCEpIGZyYW1lIGFsaWduZWQgKi8KCWluZm8tPmRtYSA9IGRyaXZlLT51c2luZ19kbWEgPyAxIDogMDsKCglpbmZvLT5kZXZpbmZvLm1lZGlhX3dyaXR0ZW4gPSAxOwoKCS8qIFN0YXJ0IHNlbmRpbmcgdGhlIHdyaXRlIHJlcXVlc3QgdG8gdGhlIGRyaXZlLiAqLwoJcmV0dXJuIGNkcm9tX3N0YXJ0X3BhY2tldF9jb21tYW5kKGRyaXZlLCAzMjc2OCwgY2Ryb21fc3RhcnRfd3JpdGVfY29udCk7Cn0KCnN0YXRpYyBpZGVfc3RhcnRzdG9wX3QgY2Ryb21fZG9fbmV3cGNfY29udChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCByZXF1ZXN0ICpycSA9IEhXR1JPVVAoZHJpdmUpLT5ycTsKCglpZiAoIXJxLT50aW1lb3V0KQoJCXJxLT50aW1lb3V0ID0gQVRBUElfV0FJVF9QQzsKCglyZXR1cm4gY2Ryb21fdHJhbnNmZXJfcGFja2V0X2NvbW1hbmQoZHJpdmUsIHJxLCBjZHJvbV9uZXdwY19pbnRyKTsKfQoKc3RhdGljIGlkZV9zdGFydHN0b3BfdCBjZHJvbV9kb19ibG9ja19wYyhpZGVfZHJpdmVfdCAqZHJpdmUsIHN0cnVjdCByZXF1ZXN0ICpycSkKewoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJcnEtPmZsYWdzIHw9IFJFUV9RVUlFVDsKCglpbmZvLT5kbWEgPSAwOwoKCS8qCgkgKiBzZyByZXF1ZXN0CgkgKi8KCWlmIChycS0+YmlvKSB7CgkJaW50IG1hc2sgPSBkcml2ZS0+cXVldWUtPmRtYV9hbGlnbm1lbnQ7CgkJdW5zaWduZWQgbG9uZyBhZGRyID0gKHVuc2lnbmVkIGxvbmcpIHBhZ2VfYWRkcmVzcyhiaW9fcGFnZShycS0+YmlvKSk7CgoJCWluZm8tPmRtYSA9IGRyaXZlLT51c2luZ19kbWE7CgoJCS8qCgkJICogY2hlY2sgaWYgZG1hIGlzIHNhZmUKCQkgKgoJCSAqIE5PVEUhIFRoZSAibGVuIiBhbmQgImFkZHIiIGNoZWNrcyBzaG91bGQgcG9zc2libHkgaGF2ZQoJCSAqIHNlcGFyYXRlIG1hc2tzLgoJCSAqLwoJCWlmICgocnEtPmRhdGFfbGVuICYgMTUpIHx8IChhZGRyICYgbWFzaykpCgkJCWluZm8tPmRtYSA9IDA7Cgl9CgoJLyogU3RhcnQgc2VuZGluZyB0aGUgY29tbWFuZCB0byB0aGUgZHJpdmUuICovCglyZXR1cm4gY2Ryb21fc3RhcnRfcGFja2V0X2NvbW1hbmQoZHJpdmUsIHJxLT5kYXRhX2xlbiwgY2Ryb21fZG9fbmV3cGNfY29udCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIGNkcm9tIGRyaXZlciByZXF1ZXN0IHJvdXRpbmUuCiAqLwpzdGF0aWMgaWRlX3N0YXJ0c3RvcF90CmlkZV9kb19yd19jZHJvbSAoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdCAqcnEsIHNlY3Rvcl90IGJsb2NrKQp7CglpZGVfc3RhcnRzdG9wX3QgYWN0aW9uOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CgoJaWYgKGJsa19mc19yZXF1ZXN0KHJxKSkgewoJCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zZWVraW5nKSB7CgkJCXVuc2lnbmVkIGxvbmcgZWxhcHNlZCA9IGppZmZpZXMgLSBpbmZvLT5zdGFydF9zZWVrOwoJCQlpbnQgc3RhdCA9IEhXSUYoZHJpdmUpLT5JTkIoSURFX1NUQVRVU19SRUcpOwoKCQkJaWYgKChzdGF0ICYgU0VFS19TVEFUKSAhPSBTRUVLX1NUQVQpIHsKCQkJCWlmIChlbGFwc2VkIDwgSURFQ0RfU0VFS19USU1FT1VUKSB7CgkJCQkJaWRlX3N0YWxsX3F1ZXVlKGRyaXZlLCBJREVDRF9TRUVLX1RJTUVSKTsKCQkJCQlyZXR1cm4gaWRlX3N0b3BwZWQ7CgkJCQl9CgkJCQlwcmludGsgKEtFUk5fRVJSICIlczogRFNDIHRpbWVvdXRcbiIsIGRyaXZlLT5uYW1lKTsKCQkJfQoJCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zZWVraW5nID0gMDsKCQl9CgkJaWYgKChycV9kYXRhX2RpcihycSkgPT0gUkVBRCkgJiYgSURFX0xBUkdFX1NFRUsoaW5mby0+bGFzdF9ibG9jaywgYmxvY2ssIElERUNEX1NFRUtfVEhSRVNIT0xEKSAmJiBkcml2ZS0+ZHNjX292ZXJsYXApIHsKCQkJYWN0aW9uID0gY2Ryb21fc3RhcnRfc2Vlayhkcml2ZSwgYmxvY2spOwoJCX0gZWxzZSB7CgkJCWlmIChycV9kYXRhX2RpcihycSkgPT0gUkVBRCkKCQkJCWFjdGlvbiA9IGNkcm9tX3N0YXJ0X3JlYWQoZHJpdmUsIGJsb2NrKTsKCQkJZWxzZQoJCQkJYWN0aW9uID0gY2Ryb21fc3RhcnRfd3JpdGUoZHJpdmUsIHJxKTsKCQl9CgkJaW5mby0+bGFzdF9ibG9jayA9IGJsb2NrOwoJCXJldHVybiBhY3Rpb247Cgl9IGVsc2UgaWYgKHJxLT5mbGFncyAmIChSRVFfUEMgfCBSRVFfU0VOU0UpKSB7CgkJcmV0dXJuIGNkcm9tX2RvX3BhY2tldF9jb21tYW5kKGRyaXZlKTsKCX0gZWxzZSBpZiAocnEtPmZsYWdzICYgUkVRX0JMT0NLX1BDKSB7CgkJcmV0dXJuIGNkcm9tX2RvX2Jsb2NrX3BjKGRyaXZlLCBycSk7Cgl9IGVsc2UgaWYgKHJxLT5mbGFncyAmIFJFUV9TUEVDSUFMKSB7CgkJLyoKCQkgKiByaWdodCBub3cgdGhpcyBjYW4gb25seSBiZSBhIHJlc2V0Li4uCgkJICovCgkJY2Ryb21fZW5kX3JlcXVlc3QoZHJpdmUsIDEpOwoJCXJldHVybiBpZGVfc3RvcHBlZDsKCX0KCglibGtfZHVtcF9ycV9mbGFncyhycSwgImlkZS1jZCBiYWQgZmxhZ3MiKTsKCWNkcm9tX2VuZF9yZXF1ZXN0KGRyaXZlLCAwKTsKCXJldHVybiBpZGVfc3RvcHBlZDsKfQoKCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJb2N0bCBoYW5kbGluZy4KICoKICogUm91dGluZXMgd2hpY2ggcXVldWUgcGFja2V0IGNvbW1hbmRzIHRha2UgYXMgYSBmaW5hbCBhcmd1bWVudCBhIHBvaW50ZXIKICogdG8gYSByZXF1ZXN0X3NlbnNlIHN0cnVjdC4gIElmIGV4ZWN1dGlvbiBvZiB0aGUgY29tbWFuZCByZXN1bHRzCiAqIGluIGFuIGVycm9yIHdpdGggYSBDSEVDSyBDT05ESVRJT04gc3RhdHVzLCB0aGlzIHN0cnVjdHVyZSB3aWxsIGJlIGZpbGxlZAogKiB3aXRoIHRoZSByZXN1bHRzIG9mIHRoZSBzdWJzZXF1ZW50IHJlcXVlc3Qgc2Vuc2UgY29tbWFuZC4gIFRoZSBwb2ludGVyCiAqIGNhbiBhbHNvIGJlIE5VTEwsIGluIHdoaWNoIGNhc2Ugbm8gc2Vuc2UgaW5mb3JtYXRpb24gaXMgcmV0dXJuZWQuCiAqLwoKI2lmICEgU1RBTkRBUkRfQVRBUEkKc3RhdGljIGlubGluZQppbnQgYmluMmJjZCAoaW50IHgpCnsKCXJldHVybiAoeCUxMCkgfCAoKHgvMTApIDw8IDQpOwp9CgoKc3RhdGljIGlubGluZQppbnQgYmNkMmJpbiAoaW50IHgpCnsKCXJldHVybiAoeCA+PiA0KSAqIDEwICsgKHggJiAweDBmKTsKfQoKc3RhdGljCnZvaWQgbXNmX2Zyb21fYmNkIChzdHJ1Y3QgYXRhcGlfbXNmICptc2YpCnsKCW1zZi0+bWludXRlID0gYmNkMmJpbiAobXNmLT5taW51dGUpOwoJbXNmLT5zZWNvbmQgPSBiY2QyYmluIChtc2YtPnNlY29uZCk7Cgltc2YtPmZyYW1lICA9IGJjZDJiaW4gKG1zZi0+ZnJhbWUpOwp9CgojZW5kaWYgLyogbm90IFNUQU5EQVJEX0FUQVBJICovCgoKc3RhdGljIGlubGluZQp2b2lkIGxiYV90b19tc2YgKGludCBsYmEsIGJ5dGUgKm0sIGJ5dGUgKnMsIGJ5dGUgKmYpCnsKCWxiYSArPSBDRF9NU0ZfT0ZGU0VUOwoJbGJhICY9IDB4ZmZmZmZmOyAgLyogbmVnYXRpdmUgbGJhcyB1c2Ugb25seSAyNCBiaXRzICovCgkqbSA9IGxiYSAvIChDRF9TRUNTICogQ0RfRlJBTUVTKTsKCWxiYSAlPSAoQ0RfU0VDUyAqIENEX0ZSQU1FUyk7CgkqcyA9IGxiYSAvIENEX0ZSQU1FUzsKCSpmID0gbGJhICUgQ0RfRlJBTUVTOwp9CgoKc3RhdGljIGlubGluZQppbnQgbXNmX3RvX2xiYSAoYnl0ZSBtLCBieXRlIHMsIGJ5dGUgZikKewoJcmV0dXJuICgoKG0gKiBDRF9TRUNTKSArIHMpICogQ0RfRlJBTUVTICsgZikgLSBDRF9NU0ZfT0ZGU0VUOwp9CgpzdGF0aWMgaW50IGNkcm9tX2NoZWNrX3N0YXR1cyhpZGVfZHJpdmVfdCAqZHJpdmUsIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJc3RydWN0IHJlcXVlc3QgcmVxOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSA9ICZpbmZvLT5kZXZpbmZvOwoKCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgoJcmVxLnNlbnNlID0gc2Vuc2U7CglyZXEuY21kWzBdID0gR1BDTURfVEVTVF9VTklUX1JFQURZOwoJcmVxLmZsYWdzIHw9IFJFUV9RVUlFVDsKCiNpZiAhIFNUQU5EQVJEX0FUQVBJCiAgICAgICAgLyogdGhlIFNhbnlvIDMgQ0QgY2hhbmdlciB1c2VzIGJ5dGUgNyBvZiBURVNUX1VOSVRfUkVBRFkgdG8gCiAgICAgICAgICAgc3dpdGNoIENEcyBpbnN0ZWFkIG9mIHN1cHBvcnRpbmcgdGhlIExPQURfVU5MT0FEIG9wY29kZSAgICovCgoJcmVxLmNtZFs3XSA9IGNkaS0+c2FueW9fc2xvdCAlIDM7CiNlbmRpZiAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCglyZXR1cm4gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwp9CgoKLyogTG9jayB0aGUgZG9vciBpZiBMT0NLRkxBRyBpcyBub256ZXJvOyB1bmxvY2sgaXQgb3RoZXJ3aXNlLiAqLwpzdGF0aWMgaW50CmNkcm9tX2xvY2tkb29yKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IGxvY2tmbGFnLCBzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqc2Vuc2UpCnsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIG15X3NlbnNlOwoJc3RydWN0IHJlcXVlc3QgcmVxOwoJaW50IHN0YXQ7CgoJaWYgKHNlbnNlID09IE5VTEwpCgkJc2Vuc2UgPSAmbXlfc2Vuc2U7CgoJLyogSWYgdGhlIGRyaXZlIGNhbm5vdCBsb2NrIHRoZSBkb29yLCBqdXN0IHByZXRlbmQuICovCglpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZG9vcmxvY2spIHsKCQlzdGF0ID0gMDsKCX0gZWxzZSB7CgkJY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGRyaXZlLCAmcmVxKTsKCQlyZXEuc2Vuc2UgPSBzZW5zZTsKCQlyZXEuY21kWzBdID0gR1BDTURfUFJFVkVOVF9BTExPV19NRURJVU1fUkVNT1ZBTDsKCQlyZXEuY21kWzRdID0gbG9ja2ZsYWcgPyAxIDogMDsKCQlzdGF0ID0gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwoJfQoKCS8qIElmIHdlIGdvdCBhbiBpbGxlZ2FsIGZpZWxkIGVycm9yLCB0aGUgZHJpdmUKCSAgIHByb2JhYmx5IGNhbm5vdCBsb2NrIHRoZSBkb29yLiAqLwoJaWYgKHN0YXQgIT0gMCAmJgoJICAgIHNlbnNlLT5zZW5zZV9rZXkgPT0gSUxMRUdBTF9SRVFVRVNUICYmCgkgICAgKHNlbnNlLT5hc2MgPT0gMHgyNCB8fCBzZW5zZS0+YXNjID09IDB4MjApKSB7CgkJcHJpbnRrIChLRVJOX0VSUiAiJXM6IGRvb3IgbG9ja2luZyBub3Qgc3VwcG9ydGVkXG4iLAoJCQlkcml2ZS0+bmFtZSk7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZG9vcmxvY2sgPSAxOwoJCXN0YXQgPSAwOwoJfQoJCgkvKiBubyBtZWRpdW0sIHRoYXQncyBhbHJpZ2h0LiAqLwoJaWYgKHN0YXQgIT0gMCAmJiBzZW5zZS0+c2Vuc2Vfa2V5ID09IE5PVF9SRUFEWSAmJiBzZW5zZS0+YXNjID09IDB4M2EpCgkJc3RhdCA9IDA7CgoJaWYgKHN0YXQgPT0gMCkKCQlDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmRvb3JfbG9ja2VkID0gbG9ja2ZsYWc7CgoJcmV0dXJuIHN0YXQ7Cn0KCgovKiBFamVjdCB0aGUgZGlzayBpZiBFSkVDVEZMQUcgaXMgMC4KICAgSWYgRUpFQ1RGTEFHIGlzIDEsIHRyeSB0byByZWxvYWQgdGhlIGRpc2suICovCnN0YXRpYyBpbnQgY2Ryb21fZWplY3QoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgZWplY3RmbGFnLAoJCSAgICAgICBzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqc2Vuc2UpCnsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCWNoYXIgbG9laiA9IDB4MDI7CgoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2VqZWN0ICYmICFlamVjdGZsYWcpCgkJcmV0dXJuIC1FRFJJVkVfQ0FOVF9ET19USElTOwoJCgkvKiByZWxvYWQgZmFpbHMgb24gc29tZSBkcml2ZXMsIGlmIHRoZSB0cmF5IGlzIGxvY2tlZCAqLwoJaWYgKENEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+ZG9vcl9sb2NrZWQgJiYgZWplY3RmbGFnKQoJCXJldHVybiAwOwoKCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgoJLyogb25seSB0ZWxsIGRyaXZlIHRvIGNsb3NlIHRyYXkgaWYgb3BlbiwgaWYgaXQgY2FuIGRvIHRoYXQgKi8KCWlmIChlamVjdGZsYWcgJiYgIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNsb3NlX3RyYXkpCgkJbG9laiA9IDA7CgoJcmVxLnNlbnNlID0gc2Vuc2U7CglyZXEuY21kWzBdID0gR1BDTURfU1RBUlRfU1RPUF9VTklUOwoJcmVxLmNtZFs0XSA9IGxvZWogfCAoZWplY3RmbGFnICE9IDApOwoJcmV0dXJuIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKfQoKc3RhdGljIGludCBjZHJvbV9yZWFkX2NhcGFjaXR5KGlkZV9kcml2ZV90ICpkcml2ZSwgdW5zaWduZWQgbG9uZyAqY2FwYWNpdHksCgkJCSAgICAgICB1bnNpZ25lZCBsb25nICpzZWN0b3JzX3Blcl9mcmFtZSwKCQkJICAgICAgIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJc3RydWN0IHsKCQlfX3UzMiBsYmE7CgkJX191MzIgYmxvY2tsZW47Cgl9IGNhcGJ1ZjsKCglpbnQgc3RhdDsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsICZyZXEpOwoKCXJlcS5zZW5zZSA9IHNlbnNlOwoJcmVxLmNtZFswXSA9IEdQQ01EX1JFQURfQ0RWRF9DQVBBQ0lUWTsKCXJlcS5kYXRhID0gKGNoYXIgKikmY2FwYnVmOwoJcmVxLmRhdGFfbGVuID0gc2l6ZW9mKGNhcGJ1Zik7CgoJc3RhdCA9IGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKCWlmIChzdGF0ID09IDApIHsKCQkqY2FwYWNpdHkgPSAxICsgYmUzMl90b19jcHUoY2FwYnVmLmxiYSk7CgkJKnNlY3RvcnNfcGVyX2ZyYW1lID0KCQkJYmUzMl90b19jcHUoY2FwYnVmLmJsb2NrbGVuKSA+PiBTRUNUT1JfQklUUzsKCX0KCglyZXR1cm4gc3RhdDsKfQoKc3RhdGljIGludCBjZHJvbV9yZWFkX3RvY2VudHJ5KGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IHRyYWNrbm8sIGludCBtc2ZfZmxhZywKCQkJCWludCBmb3JtYXQsIGNoYXIgKmJ1ZiwgaW50IGJ1ZmxlbiwKCQkJCXN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJc3RydWN0IHJlcXVlc3QgcmVxOwoKCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgoJcmVxLnNlbnNlID0gc2Vuc2U7CglyZXEuZGF0YSA9ICBidWY7CglyZXEuZGF0YV9sZW4gPSBidWZsZW47CglyZXEuZmxhZ3MgfD0gUkVRX1FVSUVUOwoJcmVxLmNtZFswXSA9IEdQQ01EX1JFQURfVE9DX1BNQV9BVElQOwoJcmVxLmNtZFs2XSA9IHRyYWNrbm87CglyZXEuY21kWzddID0gKGJ1ZmxlbiA+PiA4KTsKCXJlcS5jbWRbOF0gPSAoYnVmbGVuICYgMHhmZik7CglyZXEuY21kWzldID0gKGZvcm1hdCA8PCA2KTsKCglpZiAobXNmX2ZsYWcpCgkJcmVxLmNtZFsxXSA9IDI7CgoJcmV0dXJuIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKfQoKCi8qIFRyeSB0byByZWFkIHRoZSBlbnRpcmUgVE9DIGZvciB0aGUgZGlzayBpbnRvIG91ciBpbnRlcm5hbCBidWZmZXIuICovCnN0YXRpYyBpbnQgY2Ryb21fcmVhZF90b2MoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgcmVxdWVzdF9zZW5zZSAqc2Vuc2UpCnsKCWludCBzdGF0LCBudHJhY2tzLCBpOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSA9ICZpbmZvLT5kZXZpbmZvOwoJc3RydWN0IGF0YXBpX3RvYyAqdG9jID0gaW5mby0+dG9jOwoJc3RydWN0IHsKCQlzdHJ1Y3QgYXRhcGlfdG9jX2hlYWRlciBoZHI7CgkJc3RydWN0IGF0YXBpX3RvY19lbnRyeSAgZW50OwoJfSBtc190bXA7Cglsb25nIGxhc3Rfd3JpdHRlbjsKCXVuc2lnbmVkIGxvbmcgc2VjdG9yc19wZXJfZnJhbWUgPSBTRUNUT1JTX1BFUl9GUkFNRTsKCglpZiAodG9jID09IE5VTEwpIHsKCQkvKiBUcnkgdG8gYWxsb2NhdGUgc3BhY2UuICovCgkJdG9jID0ga21hbGxvYyhzaXplb2Yoc3RydWN0IGF0YXBpX3RvYyksIEdGUF9LRVJORUwpOwoJCWlmICh0b2MgPT0gTlVMTCkgewoJCQlwcmludGsgKEtFUk5fRVJSICIlczogTm8gY2Ryb20gVE9DIGJ1ZmZlciFcbiIsIGRyaXZlLT5uYW1lKTsKCQkJcmV0dXJuIC1FTk9NRU07CgkJfQoJCWluZm8tPnRvYyA9IHRvYzsKCX0KCgkvKiBDaGVjayB0byBzZWUgaWYgdGhlIGV4aXN0aW5nIGRhdGEgaXMgc3RpbGwgdmFsaWQuCgkgICBJZiBpdCBpcywganVzdCByZXR1cm4uICovCgkodm9pZCkgY2Ryb21fY2hlY2tfc3RhdHVzKGRyaXZlLCBzZW5zZSk7CgoJaWYgKENEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkKQoJCXJldHVybiAwOwoKCS8qIFRyeSB0byBnZXQgdGhlIHRvdGFsIGNkcm9tIGNhcGFjaXR5IGFuZCBzZWN0b3Igc2l6ZS4gKi8KCXN0YXQgPSBjZHJvbV9yZWFkX2NhcGFjaXR5KGRyaXZlLCAmdG9jLT5jYXBhY2l0eSwgJnNlY3RvcnNfcGVyX2ZyYW1lLAoJCQkJICAgc2Vuc2UpOwoJaWYgKHN0YXQpCgkJdG9jLT5jYXBhY2l0eSA9IDB4MWZmZmZmOwoKCXNldF9jYXBhY2l0eShpbmZvLT5kaXNrLCB0b2MtPmNhcGFjaXR5ICogc2VjdG9yc19wZXJfZnJhbWUpOwoJYmxrX3F1ZXVlX2hhcmRzZWN0X3NpemUoZHJpdmUtPnF1ZXVlLAoJCQkJc2VjdG9yc19wZXJfZnJhbWUgPDwgU0VDVE9SX0JJVFMpOwoKCS8qIEZpcnN0IHJlYWQganVzdCB0aGUgaGVhZGVyLCBzbyB3ZSBrbm93IGhvdyBsb25nIHRoZSBUT0MgaXMuICovCglzdGF0ID0gY2Ryb21fcmVhZF90b2NlbnRyeShkcml2ZSwgMCwgMSwgMCwgKGNoYXIgKikgJnRvYy0+aGRyLAoJCQkJICAgIHNpemVvZihzdHJ1Y3QgYXRhcGlfdG9jX2hlYWRlciksIHNlbnNlKTsKCWlmIChzdGF0KQoJCXJldHVybiBzdGF0OwoKI2lmICEgU1RBTkRBUkRfQVRBUEkKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50b2N0cmFja3NfYXNfYmNkKSB7CgkJdG9jLT5oZHIuZmlyc3RfdHJhY2sgPSBiY2QyYmluKHRvYy0+aGRyLmZpcnN0X3RyYWNrKTsKCQl0b2MtPmhkci5sYXN0X3RyYWNrICA9IGJjZDJiaW4odG9jLT5oZHIubGFzdF90cmFjayk7Cgl9CiNlbmRpZiAgLyogbm90IFNUQU5EQVJEX0FUQVBJICovCgoJbnRyYWNrcyA9IHRvYy0+aGRyLmxhc3RfdHJhY2sgLSB0b2MtPmhkci5maXJzdF90cmFjayArIDE7CglpZiAobnRyYWNrcyA8PSAwKQoJCXJldHVybiAtRUlPOwoJaWYgKG50cmFja3MgPiBNQVhfVFJBQ0tTKQoJCW50cmFja3MgPSBNQVhfVFJBQ0tTOwoKCS8qIE5vdyByZWFkIHRoZSB3aG9sZSBzY2htZWVyLiAqLwoJc3RhdCA9IGNkcm9tX3JlYWRfdG9jZW50cnkoZHJpdmUsIHRvYy0+aGRyLmZpcnN0X3RyYWNrLCAxLCAwLAoJCQkJICAoY2hhciAqKSZ0b2MtPmhkciwKCQkJCSAgIHNpemVvZihzdHJ1Y3QgYXRhcGlfdG9jX2hlYWRlcikgKwoJCQkJICAgKG50cmFja3MgKyAxKSAqCgkJCQkgICBzaXplb2Yoc3RydWN0IGF0YXBpX3RvY19lbnRyeSksIHNlbnNlKTsKCglpZiAoc3RhdCAmJiB0b2MtPmhkci5maXJzdF90cmFjayA+IDEpIHsKCQkvKiBDZHMgd2l0aCBDREkgdHJhY2tzIG9ubHkgZG9uJ3QgaGF2ZSBhbnkgVE9DIGVudHJpZXMsCgkJICAgZGVzcGl0ZSBvZiB0aGlzIHRoZSByZXR1cm5lZCB2YWx1ZXMgYXJlCgkJICAgZmlyc3RfdHJhY2sgPT0gbGFzdF90cmFjayA9IG51bWJlciBvZiBDREkgdHJhY2tzICsgMSwKCQkgICBzbyB0aGF0IHRoaXMgY2FzZSBpcyBpbmRpc3Rpbmd1aXNoYWJsZSBmcm9tIHRoZSBzYW1lCgkJICAgbGF5b3V0IHBsdXMgYW4gYWRkaXRpb25hbCBhdWRpbyB0cmFjay4KCQkgICBJZiB3ZSBnZXQgYW4gZXJyb3IgZm9yIHRoZSByZWd1bGFyIGNhc2UsIHdlIGFzc3VtZQoJCSAgIGEgQ0RJIHdpdGhvdXQgYWRkaXRpb25hbCBhdWRpbyB0cmFja3MuIEluIHRoaXMgY2FzZQoJCSAgIHRoZSByZWFkYWJsZSBUT0MgaXMgZW1wdHkgKENESSB0cmFja3MgYXJlIG5vdCBpbmNsdWRlZCkKCQkgICBhbmQgb25seSBob2xkcyB0aGUgTGVhZG91dCBlbnRyeS4gSGVpa28gRWnfZmVsZHQgKi8KCQludHJhY2tzID0gMDsKCQlzdGF0ID0gY2Ryb21fcmVhZF90b2NlbnRyeShkcml2ZSwgQ0RST01fTEVBRE9VVCwgMSwgMCwKCQkJCQkgICAoY2hhciAqKSZ0b2MtPmhkciwKCQkJCQkgICBzaXplb2Yoc3RydWN0IGF0YXBpX3RvY19oZWFkZXIpICsKCQkJCQkgICAobnRyYWNrcyArIDEpICoKCQkJCQkgICBzaXplb2Yoc3RydWN0IGF0YXBpX3RvY19lbnRyeSksCgkJCQkJICAgc2Vuc2UpOwoJCWlmIChzdGF0KSB7CgkJCXJldHVybiBzdGF0OwoJCX0KI2lmICEgU1RBTkRBUkRfQVRBUEkKCQlpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jdHJhY2tzX2FzX2JjZCkgewoJCQl0b2MtPmhkci5maXJzdF90cmFjayA9IGJpbjJiY2QoQ0RST01fTEVBRE9VVCk7CgkJCXRvYy0+aGRyLmxhc3RfdHJhY2sgPSBiaW4yYmNkKENEUk9NX0xFQURPVVQpOwoJCX0gZWxzZQojZW5kaWYgIC8qIG5vdCBTVEFOREFSRF9BVEFQSSAqLwoJCXsKCQkJdG9jLT5oZHIuZmlyc3RfdHJhY2sgPSBDRFJPTV9MRUFET1VUOwoJCQl0b2MtPmhkci5sYXN0X3RyYWNrID0gQ0RST01fTEVBRE9VVDsKCQl9Cgl9CgoJaWYgKHN0YXQpCgkJcmV0dXJuIHN0YXQ7CgoJdG9jLT5oZHIudG9jX2xlbmd0aCA9IG50b2hzICh0b2MtPmhkci50b2NfbGVuZ3RoKTsKCiNpZiAhIFNUQU5EQVJEX0FUQVBJCglpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jdHJhY2tzX2FzX2JjZCkgewoJCXRvYy0+aGRyLmZpcnN0X3RyYWNrID0gYmNkMmJpbih0b2MtPmhkci5maXJzdF90cmFjayk7CgkJdG9jLT5oZHIubGFzdF90cmFjayAgPSBiY2QyYmluKHRvYy0+aGRyLmxhc3RfdHJhY2spOwoJfQojZW5kaWYgIC8qIG5vdCBTVEFOREFSRF9BVEFQSSAqLwoKCWZvciAoaT0wOyBpPD1udHJhY2tzOyBpKyspIHsKI2lmICEgU1RBTkRBUkRfQVRBUEkKCQlpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jYWRkcl9hc19iY2QpIHsKCQkJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY3RyYWNrc19hc19iY2QpCgkJCQl0b2MtPmVudFtpXS50cmFjayA9IGJjZDJiaW4odG9jLT5lbnRbaV0udHJhY2spOwoJCQltc2ZfZnJvbV9iY2QoJnRvYy0+ZW50W2ldLmFkZHIubXNmKTsKCQl9CiNlbmRpZiAgLyogbm90IFNUQU5EQVJEX0FUQVBJICovCgkJdG9jLT5lbnRbaV0uYWRkci5sYmEgPSBtc2ZfdG9fbGJhICh0b2MtPmVudFtpXS5hZGRyLm1zZi5taW51dGUsCgkJCQkJCSAgIHRvYy0+ZW50W2ldLmFkZHIubXNmLnNlY29uZCwKCQkJCQkJICAgdG9jLT5lbnRbaV0uYWRkci5tc2YuZnJhbWUpOwoJfQoKCS8qIFJlYWQgdGhlIG11bHRpc2Vzc2lvbiBpbmZvcm1hdGlvbi4gKi8KCWlmICh0b2MtPmhkci5maXJzdF90cmFjayAhPSBDRFJPTV9MRUFET1VUKSB7CgkJLyogUmVhZCB0aGUgbXVsdGlzZXNzaW9uIGluZm9ybWF0aW9uLiAqLwoJCXN0YXQgPSBjZHJvbV9yZWFkX3RvY2VudHJ5KGRyaXZlLCAwLCAwLCAxLCAoY2hhciAqKSZtc190bXAsCgkJCQkJICAgc2l6ZW9mKG1zX3RtcCksIHNlbnNlKTsKCQlpZiAoc3RhdCkKCQkJcmV0dXJuIHN0YXQ7CgoJCXRvYy0+bGFzdF9zZXNzaW9uX2xiYSA9IGJlMzJfdG9fY3B1KG1zX3RtcC5lbnQuYWRkci5sYmEpOwoJfSBlbHNlIHsKCQltc190bXAuaGRyLmZpcnN0X3RyYWNrID0gbXNfdG1wLmhkci5sYXN0X3RyYWNrID0gQ0RST01fTEVBRE9VVDsKCQl0b2MtPmxhc3Rfc2Vzc2lvbl9sYmEgPSBtc2ZfdG9fbGJhKDAsIDIsIDApOyAvKiAwbSAycyAwZiAqLwoJfQoKI2lmICEgU1RBTkRBUkRfQVRBUEkKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50b2NhZGRyX2FzX2JjZCkgewoJCS8qIFJlLXJlYWQgbXVsdGlzZXNzaW9uIGluZm9ybWF0aW9uIHVzaW5nIE1TRiBmb3JtYXQgKi8KCQlzdGF0ID0gY2Ryb21fcmVhZF90b2NlbnRyeShkcml2ZSwgMCwgMSwgMSwgKGNoYXIgKikmbXNfdG1wLAoJCQkJCSAgIHNpemVvZihtc190bXApLCBzZW5zZSk7CgkJaWYgKHN0YXQpCgkJCXJldHVybiBzdGF0OwoKCQltc2ZfZnJvbV9iY2QgKCZtc190bXAuZW50LmFkZHIubXNmKTsKCQl0b2MtPmxhc3Rfc2Vzc2lvbl9sYmEgPSBtc2ZfdG9fbGJhKG1zX3RtcC5lbnQuYWRkci5tc2YubWludXRlLAoJCQkJCSAgCSAgIG1zX3RtcC5lbnQuYWRkci5tc2Yuc2Vjb25kLAoJCQkJCQkgICBtc190bXAuZW50LmFkZHIubXNmLmZyYW1lKTsKCX0KI2VuZGlmICAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCgl0b2MtPnhhX2ZsYWcgPSAobXNfdG1wLmhkci5maXJzdF90cmFjayAhPSBtc190bXAuaGRyLmxhc3RfdHJhY2spOwoKCS8qIE5vdyB0cnkgdG8gZ2V0IHRoZSB0b3RhbCBjZHJvbSBjYXBhY2l0eS4gKi8KCXN0YXQgPSBjZHJvbV9nZXRfbGFzdF93cml0dGVuKGNkaSwgJmxhc3Rfd3JpdHRlbik7CglpZiAoIXN0YXQgJiYgKGxhc3Rfd3JpdHRlbiA+IHRvYy0+Y2FwYWNpdHkpKSB7CgkJdG9jLT5jYXBhY2l0eSA9IGxhc3Rfd3JpdHRlbjsKCQlzZXRfY2FwYWNpdHkoaW5mby0+ZGlzaywgdG9jLT5jYXBhY2l0eSAqIHNlY3RvcnNfcGVyX2ZyYW1lKTsKCX0KCgkvKiBSZW1lbWJlciB0aGF0IHdlJ3ZlIHJlYWQgdGhpcyBzdHVmZi4gKi8KCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkID0gMTsKCglyZXR1cm4gMDsKfQoKCnN0YXRpYyBpbnQgY2Ryb21fcmVhZF9zdWJjaGFubmVsKGlkZV9kcml2ZV90ICpkcml2ZSwgaW50IGZvcm1hdCwgY2hhciAqYnVmLAoJCQkJIGludCBidWZsZW4sIHN0cnVjdCByZXF1ZXN0X3NlbnNlICpzZW5zZSkKewoJc3RydWN0IHJlcXVlc3QgcmVxOwoKCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgoJcmVxLnNlbnNlID0gc2Vuc2U7CglyZXEuZGF0YSA9IGJ1ZjsKCXJlcS5kYXRhX2xlbiA9IGJ1ZmxlbjsKCXJlcS5jbWRbMF0gPSBHUENNRF9SRUFEX1NVQkNIQU5ORUw7CglyZXEuY21kWzFdID0gMjsgICAgIC8qIE1TRiBhZGRyZXNzaW5nICovCglyZXEuY21kWzJdID0gMHg0MDsgIC8qIHJlcXVlc3Qgc3ViUSBkYXRhICovCglyZXEuY21kWzNdID0gZm9ybWF0OwoJcmVxLmNtZFs3XSA9IChidWZsZW4gPj4gOCk7CglyZXEuY21kWzhdID0gKGJ1ZmxlbiAmIDB4ZmYpOwoJcmV0dXJuIGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKfQoKLyogQVRBUEkgY2Ryb20gZHJpdmVzIGFyZSBmcmVlIHRvIHNlbGVjdCB0aGUgc3BlZWQgeW91IHJlcXVlc3Qgb3IgYW55IHNsb3dlcgogICByYXRlIDotKCBSZXF1ZXN0aW5nIHRvbyBmYXN0IGEgc3BlZWQgd2lsbCBfbm90XyBwcm9kdWNlIGFuIGVycm9yLiAqLwpzdGF0aWMgaW50IGNkcm9tX3NlbGVjdF9zcGVlZChpZGVfZHJpdmVfdCAqZHJpdmUsIGludCBzcGVlZCwKCQkJICAgICAgc3RydWN0IHJlcXVlc3Rfc2Vuc2UgKnNlbnNlKQp7CglzdHJ1Y3QgcmVxdWVzdCByZXE7CgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsICZyZXEpOwoKCXJlcS5zZW5zZSA9IHNlbnNlOwoJaWYgKHNwZWVkID09IDApCgkJc3BlZWQgPSAweGZmZmY7IC8qIHNldCB0byBtYXggKi8KCWVsc2UKCQlzcGVlZCAqPSAxNzc7ICAgLyogTnggdG8ga2J5dGVzL3MgKi8KCglyZXEuY21kWzBdID0gR1BDTURfU0VUX1NQRUVEOwoJLyogUmVhZCBEcml2ZSBzcGVlZCBpbiBrYnl0ZXMvc2Vjb25kIE1TQiAqLwoJcmVxLmNtZFsyXSA9IChzcGVlZCA+PiA4KSAmIDB4ZmY7CQoJLyogUmVhZCBEcml2ZSBzcGVlZCBpbiBrYnl0ZXMvc2Vjb25kIExTQiAqLwoJcmVxLmNtZFszXSA9IHNwZWVkICYgMHhmZjsKCWlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yIHx8CgkgICAgQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2RfcncgfHwKCSAgICBDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcikgewoJCS8qIFdyaXRlIERyaXZlIHNwZWVkIGluIGtieXRlcy9zZWNvbmQgTVNCICovCgkJcmVxLmNtZFs0XSA9IChzcGVlZCA+PiA4KSAmIDB4ZmY7CgkJLyogV3JpdGUgRHJpdmUgc3BlZWQgaW4ga2J5dGVzL3NlY29uZCBMU0IgKi8KCQlyZXEuY21kWzVdID0gc3BlZWQgJiAweGZmOwogICAgICAgfQoKCXJldHVybiBjZHJvbV9xdWV1ZV9wYWNrZXRfY29tbWFuZChkcml2ZSwgJnJlcSk7Cn0KCnN0YXRpYyBpbnQgY2Ryb21fcGxheV9hdWRpbyhpZGVfZHJpdmVfdCAqZHJpdmUsIGludCBsYmFfc3RhcnQsIGludCBsYmFfZW5kKQp7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCgljZHJvbV9wcmVwYXJlX3JlcXVlc3QoZHJpdmUsICZyZXEpOwoKCXJlcS5zZW5zZSA9ICZzZW5zZTsKCXJlcS5jbWRbMF0gPSBHUENNRF9QTEFZX0FVRElPX01TRjsKCWxiYV90b19tc2YobGJhX3N0YXJ0LCAmcmVxLmNtZFszXSwgJnJlcS5jbWRbNF0sICZyZXEuY21kWzVdKTsKCWxiYV90b19tc2YobGJhX2VuZC0xLCAmcmVxLmNtZFs2XSwgJnJlcS5jbWRbN10sICZyZXEuY21kWzhdKTsKCglyZXR1cm4gY2Ryb21fcXVldWVfcGFja2V0X2NvbW1hbmQoZHJpdmUsICZyZXEpOwp9CgpzdGF0aWMgaW50IGNkcm9tX2dldF90b2NfZW50cnkoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgdHJhY2ssCgkJCQlzdHJ1Y3QgYXRhcGlfdG9jX2VudHJ5ICoqZW50KQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBhdGFwaV90b2MgKnRvYyA9IGluZm8tPnRvYzsKCWludCBudHJhY2tzOwoKCS8qCgkgKiBkb24ndCBzZXJ2ZSBjYWNoZWQgZGF0YSwgaWYgdGhlIHRvYyBpc24ndCB2YWxpZAoJICovCglpZiAoIUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkKQoJCXJldHVybiAtRUlOVkFMOwoKCS8qIENoZWNrIHZhbGlkaXR5IG9mIHJlcXVlc3RlZCB0cmFjayBudW1iZXIuICovCgludHJhY2tzID0gdG9jLT5oZHIubGFzdF90cmFjayAtIHRvYy0+aGRyLmZpcnN0X3RyYWNrICsgMTsKCWlmICh0b2MtPmhkci5maXJzdF90cmFjayA9PSBDRFJPTV9MRUFET1VUKSBudHJhY2tzID0gMDsKCWlmICh0cmFjayA9PSBDRFJPTV9MRUFET1VUKQoJCSplbnQgPSAmdG9jLT5lbnRbbnRyYWNrc107CgllbHNlIGlmICh0cmFjayA8IHRvYy0+aGRyLmZpcnN0X3RyYWNrIHx8CgkJIHRyYWNrID4gdG9jLT5oZHIubGFzdF90cmFjaykKCQlyZXR1cm4gLUVJTlZBTDsKCWVsc2UKCQkqZW50ID0gJnRvYy0+ZW50W3RyYWNrIC0gdG9jLT5oZHIuZmlyc3RfdHJhY2tdOwoKCXJldHVybiAwOwp9CgovKiB0aGUgZ2VuZXJpYyBwYWNrZXQgaW50ZXJmYWNlIHRvIGNkcm9tLmMgKi8Kc3RhdGljIGludCBpZGVfY2Ryb21fcGFja2V0KHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLAoJCQkgICAgc3RydWN0IHBhY2tldF9jb21tYW5kICpjZ2MpCnsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoKCWlmIChjZ2MtPnRpbWVvdXQgPD0gMCkKCQljZ2MtPnRpbWVvdXQgPSBBVEFQSV9XQUlUX1BDOwoKCS8qIGhlcmUgd2UgcXVldWUgdGhlIGNvbW1hbmRzIGZyb20gdGhlIHVuaWZvcm0gQ0QtUk9NCgkgICBsYXllci4gdGhlIHBhY2tldCBtdXN0IGJlIGNvbXBsZXRlLCBhcyB3ZSBkbyBub3QKCSAgIHRvdWNoIGl0IGF0IGFsbC4gKi8KCWNkcm9tX3ByZXBhcmVfcmVxdWVzdChkcml2ZSwgJnJlcSk7CgltZW1jcHkocmVxLmNtZCwgY2djLT5jbWQsIENEUk9NX1BBQ0tFVF9TSVpFKTsKCWlmIChjZ2MtPnNlbnNlKQoJCW1lbXNldChjZ2MtPnNlbnNlLCAwLCBzaXplb2Yoc3RydWN0IHJlcXVlc3Rfc2Vuc2UpKTsKCXJlcS5kYXRhID0gY2djLT5idWZmZXI7CglyZXEuZGF0YV9sZW4gPSBjZ2MtPmJ1ZmxlbjsKCXJlcS50aW1lb3V0ID0gY2djLT50aW1lb3V0OwoKCWlmIChjZ2MtPnF1aWV0KQoJCXJlcS5mbGFncyB8PSBSRVFfUVVJRVQ7CgoJcmVxLnNlbnNlID0gY2djLT5zZW5zZTsKCWNnYy0+c3RhdCA9IGNkcm9tX3F1ZXVlX3BhY2tldF9jb21tYW5kKGRyaXZlLCAmcmVxKTsKCWlmICghY2djLT5zdGF0KQoJCWNnYy0+YnVmbGVuIC09IHJlcS5kYXRhX2xlbjsKCXJldHVybiBjZ2MtPnN0YXQ7Cn0KCnN0YXRpYwppbnQgaWRlX2Nkcm9tX2Rldl9pb2N0bCAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksCgkJCSB1bnNpZ25lZCBpbnQgY21kLCB1bnNpZ25lZCBsb25nIGFyZykKewoJc3RydWN0IHBhY2tldF9jb21tYW5kIGNnYzsKCWNoYXIgYnVmZmVyWzE2XTsKCWludCBzdGF0OwoKCWluaXRfY2Ryb21fY29tbWFuZCgmY2djLCBidWZmZXIsIHNpemVvZihidWZmZXIpLCBDR0NfREFUQV9VTktOT1dOKTsKCgkvKiBUaGVzZSB3aWxsIGJlIG1vdmVkIGludG8gdGhlIFVuaWZvcm0gbGF5ZXIgc2hvcnRseS4uLiAqLwoJc3dpdGNoIChjbWQpIHsKIAljYXNlIENEUk9NU0VUU1BJTkRPV046IHsKIAkJY2hhciBzcGluZG93bjsKIAogCQlpZiAoY29weV9mcm9tX3VzZXIoJnNwaW5kb3duLCAodm9pZCBfX3VzZXIgKikgYXJnLCBzaXplb2YoY2hhcikpKQoJCQlyZXR1cm4gLUVGQVVMVDsKIAogICAgICAgICAgICAgICAgaWYgKChzdGF0ID0gY2Ryb21fbW9kZV9zZW5zZShjZGksICZjZ2MsIEdQTU9ERV9DRFJPTV9QQUdFLCAwKSkpCgkJCXJldHVybiBzdGF0OwoKIAkJYnVmZmVyWzExXSA9IChidWZmZXJbMTFdICYgMHhmMCkgfCAoc3BpbmRvd24gJiAweDBmKTsKCiAJCXJldHVybiBjZHJvbV9tb2RlX3NlbGVjdChjZGksICZjZ2MpOwogCX0gCiAKIAljYXNlIENEUk9NR0VUU1BJTkRPV046IHsKIAkJY2hhciBzcGluZG93bjsKIAogICAgICAgICAgICAgICAgaWYgKChzdGF0ID0gY2Ryb21fbW9kZV9zZW5zZShjZGksICZjZ2MsIEdQTU9ERV9DRFJPTV9QQUdFLCAwKSkpCgkJCXJldHVybiBzdGF0OwogCiAJCXNwaW5kb3duID0gYnVmZmVyWzExXSAmIDB4MGY7CiAKCQlpZiAoY29weV90b191c2VyKCh2b2lkIF9fdXNlciAqKSBhcmcsICZzcGluZG93biwgc2l6ZW9mIChjaGFyKSkpCgkJCXJldHVybiAtRUZBVUxUOwogCiAJCXJldHVybiAwOwogCX0KICAKCWRlZmF1bHQ6CgkJcmV0dXJuIC1FSU5WQUw7Cgl9Cgp9CgpzdGF0aWMKaW50IGlkZV9jZHJvbV9hdWRpb19pb2N0bCAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksCgkJCSAgIHVuc2lnbmVkIGludCBjbWQsIHZvaWQgKmFyZykKCQkJICAgCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBkcml2ZS0+ZHJpdmVyX2RhdGE7CglpbnQgc3RhdDsKCglzd2l0Y2ggKGNtZCkgewoJLyoKCSAqIGVtdWxhdGUgUExBWV9BVURJT19USSBjb21tYW5kIHdpdGggUExBWV9BVURJT18xMCwgc2luY2UKCSAqIGF0YXBpIGRvZXNuJ3Qgc3VwcG9ydCBpdAoJICovCgljYXNlIENEUk9NUExBWVRSS0lORDogewoJCXVuc2lnbmVkIGxvbmcgbGJhX3N0YXJ0LCBsYmFfZW5kOwoJCXN0cnVjdCBjZHJvbV90aSAqdGkgPSBhcmc7CgkJc3RydWN0IGF0YXBpX3RvY19lbnRyeSAqZmlyc3RfdG9jLCAqbGFzdF90b2M7CgoJCXN0YXQgPSBjZHJvbV9nZXRfdG9jX2VudHJ5KGRyaXZlLCB0aS0+Y2R0aV90cmswLCAmZmlyc3RfdG9jKTsKCQlpZiAoc3RhdCkKCQkJcmV0dXJuIHN0YXQ7CgoJCXN0YXQgPSBjZHJvbV9nZXRfdG9jX2VudHJ5KGRyaXZlLCB0aS0+Y2R0aV90cmsxLCAmbGFzdF90b2MpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCgkJaWYgKHRpLT5jZHRpX3RyazEgIT0gQ0RST01fTEVBRE9VVCkKCQkJKytsYXN0X3RvYzsKCQlsYmFfc3RhcnQgPSBmaXJzdF90b2MtPmFkZHIubGJhOwoJCWxiYV9lbmQgICA9IGxhc3RfdG9jLT5hZGRyLmxiYTsKCgkJaWYgKGxiYV9lbmQgPD0gbGJhX3N0YXJ0KQoJCQlyZXR1cm4gLUVJTlZBTDsKCgkJcmV0dXJuIGNkcm9tX3BsYXlfYXVkaW8oZHJpdmUsIGxiYV9zdGFydCwgbGJhX2VuZCk7Cgl9CgoJY2FzZSBDRFJPTVJFQURUT0NIRFI6IHsKCQlzdHJ1Y3QgY2Ryb21fdG9jaGRyICp0b2NoZHIgPSBhcmc7CgkJc3RydWN0IGF0YXBpX3RvYyAqdG9jOwoKCQkvKiBNYWtlIHN1cmUgb3VyIHNhdmVkIFRPQyBpcyB2YWxpZC4gKi8KCQlzdGF0ID0gY2Ryb21fcmVhZF90b2MoZHJpdmUsIE5VTEwpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCgkJdG9jID0gaW5mby0+dG9jOwoJCXRvY2hkci0+Y2R0aF90cmswID0gdG9jLT5oZHIuZmlyc3RfdHJhY2s7CgkJdG9jaGRyLT5jZHRoX3RyazEgPSB0b2MtPmhkci5sYXN0X3RyYWNrOwoKCQlyZXR1cm4gMDsKCX0KCgljYXNlIENEUk9NUkVBRFRPQ0VOVFJZOiB7CgkJc3RydWN0IGNkcm9tX3RvY2VudHJ5ICp0b2NlbnRyeSA9IGFyZzsKCQlzdHJ1Y3QgYXRhcGlfdG9jX2VudHJ5ICp0b2NlOwoKCQlzdGF0ID0gY2Ryb21fZ2V0X3RvY19lbnRyeShkcml2ZSwgdG9jZW50cnktPmNkdGVfdHJhY2ssICZ0b2NlKTsKCQlpZiAoc3RhdCkKCQkJcmV0dXJuIHN0YXQ7CgoJCXRvY2VudHJ5LT5jZHRlX2N0cmwgPSB0b2NlLT5jb250cm9sOwoJCXRvY2VudHJ5LT5jZHRlX2FkciAgPSB0b2NlLT5hZHI7CgkJaWYgKHRvY2VudHJ5LT5jZHRlX2Zvcm1hdCA9PSBDRFJPTV9NU0YpIHsKCQkJbGJhX3RvX21zZiAodG9jZS0+YWRkci5sYmEsCgkJCQkgICAmdG9jZW50cnktPmNkdGVfYWRkci5tc2YubWludXRlLAoJCQkJICAgJnRvY2VudHJ5LT5jZHRlX2FkZHIubXNmLnNlY29uZCwKCQkJCSAgICZ0b2NlbnRyeS0+Y2R0ZV9hZGRyLm1zZi5mcmFtZSk7CgkJfSBlbHNlCgkJCXRvY2VudHJ5LT5jZHRlX2FkZHIubGJhID0gdG9jZS0+YWRkci5sYmE7CgoJCXJldHVybiAwOwoJfQoKCWRlZmF1bHQ6CgkJcmV0dXJuIC1FSU5WQUw7Cgl9Cn0KCnN0YXRpYwppbnQgaWRlX2Nkcm9tX3Jlc2V0IChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSkKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCXN0cnVjdCByZXF1ZXN0IHJlcTsKCWludCByZXQ7CgoJY2Ryb21fcHJlcGFyZV9yZXF1ZXN0KGRyaXZlLCAmcmVxKTsKCXJlcS5mbGFncyA9IFJFUV9TUEVDSUFMIHwgUkVRX1FVSUVUOwoJcmV0ID0gaWRlX2RvX2RyaXZlX2NtZChkcml2ZSwgJnJlcSwgaWRlX3dhaXQpOwoKCS8qCgkgKiBBIHJlc2V0IHdpbGwgdW5sb2NrIHRoZSBkb29yLiBJZiBpdCB3YXMgcHJldmlvdXNseSBsb2NrZWQsCgkgKiBsb2NrIGl0IGFnYWluLgoJICovCglpZiAoQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT5kb29yX2xvY2tlZCkKCQkodm9pZCkgY2Ryb21fbG9ja2Rvb3IoZHJpdmUsIDEsICZzZW5zZSk7CgoJcmV0dXJuIHJldDsKfQoKCnN0YXRpYwppbnQgaWRlX2Nkcm9tX3RyYXlfbW92ZSAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksIGludCBwb3NpdGlvbikKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCglpZiAocG9zaXRpb24pIHsKCQlpbnQgc3RhdCA9IGNkcm9tX2xvY2tkb29yKGRyaXZlLCAwLCAmc2Vuc2UpOwoJCWlmIChzdGF0KQoJCQlyZXR1cm4gc3RhdDsKCX0KCglyZXR1cm4gY2Ryb21fZWplY3QoZHJpdmUsICFwb3NpdGlvbiwgJnNlbnNlKTsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fbG9ja19kb29yIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwgaW50IGxvY2spCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGNkaS0+aGFuZGxlOwoJcmV0dXJuIGNkcm9tX2xvY2tkb29yKGRyaXZlLCBsb2NrLCBOVUxMKTsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fZ2V0X2NhcGFiaWxpdGllcyhpZGVfZHJpdmVfdCAqZHJpdmUsIHN0cnVjdCBhdGFwaV9jYXBhYmlsaXRpZXNfcGFnZSAqY2FwKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpID0gJmluZm8tPmRldmluZm87CglzdHJ1Y3QgcGFja2V0X2NvbW1hbmQgY2djOwoJaW50IHN0YXQsIGF0dGVtcHRzID0gMywgc2l6ZSA9IHNpemVvZigqY2FwKTsKCgkvKgoJICogQUNFUjUwIChhbmQgb3RoZXJzPykgcmVxdWlyZSB0aGUgZnVsbCBzcGVjIGxlbmd0aCBtb2RlIHNlbnNlCgkgKiBwYWdlIGNhcGFiaWxpdGllcyBzaXplLCBidXQgb2xkZXIgZHJpdmVzIGJyZWFrLgoJICovCglpZiAoISghc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJBVEFQSSBDRCBST00gRFJJVkUgNTBYIE1BWCIpIHx8CgkgICAgIXN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiV1BJIENEUy0zMlgiKSkpCgkJc2l6ZSAtPSBzaXplb2YoY2FwLT5wYWQpOwoKCWluaXRfY2Ryb21fY29tbWFuZCgmY2djLCBjYXAsIHNpemUsIENHQ19EQVRBX1VOS05PV04pOwoJZG8geyAvKiB3ZSBzZWVtIHRvIGdldCBzdGF0PTB4MDEsZXJyPTB4MDAgdGhlIGZpcnN0IHRpbWUgKD8/KSAqLwoJCXN0YXQgPSBjZHJvbV9tb2RlX3NlbnNlKGNkaSwgJmNnYywgR1BNT0RFX0NBUEFCSUxJVElFU19QQUdFLCAwKTsKCQlpZiAoIXN0YXQpCgkJCWJyZWFrOwoJfSB3aGlsZSAoLS1hdHRlbXB0cyk7CglyZXR1cm4gc3RhdDsKfQoKc3RhdGljCnZvaWQgaWRlX2Nkcm9tX3VwZGF0ZV9zcGVlZCAoaWRlX2RyaXZlX3QgKmRyaXZlLCBzdHJ1Y3QgYXRhcGlfY2FwYWJpbGl0aWVzX3BhZ2UgKmNhcCkKewoJLyogVGhlIEFDRVIvQU9wZW4gMjRYIGNkcm9tIGhhcyB0aGUgc3BlZWQgZmllbGRzIGJ5dGUtc3dhcHBlZCAqLwoJaWYgKCFkcml2ZS0+aWQtPm1vZGVsWzBdICYmCgkgICAgIXN0cm5jbXAoZHJpdmUtPmlkLT5md19yZXYsICIyNDFOIiwgNCkpIHsKCQlDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmN1cnJlbnRfc3BlZWQgID0KCQkJKCgodW5zaWduZWQgaW50KWNhcC0+Y3Vyc3BlZWQpICsgKDE3Ni8yKSkgLyAxNzY7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bWF4X3NwZWVkID0KCQkJKCgodW5zaWduZWQgaW50KWNhcC0+bWF4c3BlZWQpICsgKDE3Ni8yKSkgLyAxNzY7Cgl9IGVsc2UgewoJCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+Y3VycmVudF9zcGVlZCAgPQoJCQkobnRvaHMoY2FwLT5jdXJzcGVlZCkgKyAoMTc2LzIpKSAvIDE3NjsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5tYXhfc3BlZWQgPQoJCQkobnRvaHMoY2FwLT5tYXhzcGVlZCkgKyAoMTc2LzIpKSAvIDE3NjsKCX0KfQoKc3RhdGljCmludCBpZGVfY2Ryb21fc2VsZWN0X3NwZWVkIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwgaW50IHNwZWVkKQp7CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBjZGktPmhhbmRsZTsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIHNlbnNlOwoJc3RydWN0IGF0YXBpX2NhcGFiaWxpdGllc19wYWdlIGNhcDsKCWludCBzdGF0OwoKCWlmICgoc3RhdCA9IGNkcm9tX3NlbGVjdF9zcGVlZChkcml2ZSwgc3BlZWQsICZzZW5zZSkpIDwgMCkKCQlyZXR1cm4gc3RhdDsKCglpZiAoIWlkZV9jZHJvbV9nZXRfY2FwYWJpbGl0aWVzKGRyaXZlLCAmY2FwKSkgewoJCWlkZV9jZHJvbV91cGRhdGVfc3BlZWQoZHJpdmUsICZjYXApOwoJCWNkaS0+c3BlZWQgPSBDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmN1cnJlbnRfc3BlZWQ7Cgl9CiAgICAgICAgcmV0dXJuIDA7Cn0KCi8qCiAqIGFkZCBsb2dpYyB0byB0cnkgR0VUX0VWRU5UIGNvbW1hbmQgZmlyc3QgdG8gY2hlY2sgZm9yIG1lZGlhIGFuZCB0cmF5CiAqIHN0YXR1cy4gdGhpcyBzaG91bGQgYmUgc3VwcG9ydGVkIGJ5IG5ld2VyIGNkLXIvdyBhbmQgYWxsIERWRCBldGMKICogZHJpdmVzCiAqLwpzdGF0aWMKaW50IGlkZV9jZHJvbV9kcml2ZV9zdGF0dXMgKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLCBpbnQgc2xvdF9ucikKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CglzdHJ1Y3QgbWVkaWFfZXZlbnRfZGVzYyBtZWQ7CglzdHJ1Y3QgcmVxdWVzdF9zZW5zZSBzZW5zZTsKCWludCBzdGF0OwoKCWlmIChzbG90X25yICE9IENEU0xfQ1VSUkVOVCkKCQlyZXR1cm4gLUVJTlZBTDsKCglzdGF0ID0gY2Ryb21fY2hlY2tfc3RhdHVzKGRyaXZlLCAmc2Vuc2UpOwoJaWYgKCFzdGF0IHx8IHNlbnNlLnNlbnNlX2tleSA9PSBVTklUX0FUVEVOVElPTikKCQlyZXR1cm4gQ0RTX0RJU0NfT0s7CgoJaWYgKCFjZHJvbV9nZXRfbWVkaWFfZXZlbnQoY2RpLCAmbWVkKSkgewoJCWlmIChtZWQubWVkaWFfcHJlc2VudCkKCQkJcmV0dXJuIENEU19ESVNDX09LOwoJCWVsc2UgaWYgKG1lZC5kb29yX29wZW4pCgkJCXJldHVybiBDRFNfVFJBWV9PUEVOOwoJCWVsc2UKCQkJcmV0dXJuIENEU19OT19ESVNDOwoJfQoKCWlmIChzZW5zZS5zZW5zZV9rZXkgPT0gTk9UX1JFQURZICYmIHNlbnNlLmFzYyA9PSAweDA0ICYmIHNlbnNlLmFzY3EgPT0gMHgwNCkKCQlyZXR1cm4gQ0RTX0RJU0NfT0s7CgoJLyoKCSAqIElmIG5vdCB1c2luZyBNdCBGdWppIGV4dGVuZGVkIG1lZGlhIHRyYXkgcmVwb3J0cywKCSAqIGp1c3QgcmV0dXJuIFRSQVlfT1BFTiBzaW5jZSBBVEFQSSBkb2Vzbid0IHByb3ZpZGUKCSAqIGFueSBvdGhlciB3YXkgdG8gZGV0ZWN0IHRoaXMuLi4KCSAqLwoJaWYgKHNlbnNlLnNlbnNlX2tleSA9PSBOT1RfUkVBRFkpIHsKCQlpZiAoc2Vuc2UuYXNjID09IDB4M2EpIHsKCQkJaWYgKHNlbnNlLmFzY3EgPT0gMSkKCQkJCXJldHVybiBDRFNfTk9fRElTQzsKCQkJZWxzZSBpZiAoc2Vuc2UuYXNjcSA9PSAwIHx8IHNlbnNlLmFzY3EgPT0gMikKCQkJCXJldHVybiBDRFNfVFJBWV9PUEVOOwoJCX0KCX0KCglyZXR1cm4gQ0RTX0RSSVZFX05PVF9SRUFEWTsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fZ2V0X2xhc3Rfc2Vzc2lvbiAoc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGksCgkJCQlzdHJ1Y3QgY2Ryb21fbXVsdGlzZXNzaW9uICptc19pbmZvKQp7CglzdHJ1Y3QgYXRhcGlfdG9jICp0b2M7CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBjZGktPmhhbmRsZTsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IHJlcXVlc3Rfc2Vuc2Ugc2Vuc2U7CglpbnQgcmV0OwoKCWlmICghQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT50b2NfdmFsaWQgfHwgaW5mby0+dG9jID09IE5VTEwpCgkJaWYgKChyZXQgPSBjZHJvbV9yZWFkX3RvYyhkcml2ZSwgJnNlbnNlKSkpCgkJCXJldHVybiByZXQ7CgoJdG9jID0gaW5mby0+dG9jOwoJbXNfaW5mby0+YWRkci5sYmEgPSB0b2MtPmxhc3Rfc2Vzc2lvbl9sYmE7Cgltc19pbmZvLT54YV9mbGFnID0gdG9jLT54YV9mbGFnOwoKCXJldHVybiAwOwp9CgpzdGF0aWMKaW50IGlkZV9jZHJvbV9nZXRfbWNuIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwKCQkgICAgICAgc3RydWN0IGNkcm9tX21jbiAqbWNuX2luZm8pCnsKCWludCBzdGF0OwoJY2hhciBtY25idWZbMjRdOwoJaWRlX2RyaXZlX3QgKmRyaXZlID0gY2RpLT5oYW5kbGU7CgovKiBnZXQgTUNOICovCglpZiAoKHN0YXQgPSBjZHJvbV9yZWFkX3N1YmNoYW5uZWwoZHJpdmUsIDIsIG1jbmJ1Ziwgc2l6ZW9mIChtY25idWYpLCBOVUxMKSkpCgkJcmV0dXJuIHN0YXQ7CgoJbWVtY3B5IChtY25faW5mby0+bWVkaXVtX2NhdGFsb2dfbnVtYmVyLCBtY25idWYrOSwKCQlzaXplb2YgKG1jbl9pbmZvLT5tZWRpdW1fY2F0YWxvZ19udW1iZXIpLTEpOwoJbWNuX2luZm8tPm1lZGl1bV9jYXRhbG9nX251bWJlcltzaXplb2YgKG1jbl9pbmZvLT5tZWRpdW1fY2F0YWxvZ19udW1iZXIpLTFdCgkJPSAnXDAnOwoKCXJldHVybiAwOwp9CgoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIE90aGVyIGRyaXZlciByZXF1ZXN0cyAob3BlbiwgY2xvc2UsIGNoZWNrIG1lZGlhIGNoYW5nZSkuCiAqLwoKc3RhdGljCmludCBpZGVfY2Ryb21fY2hlY2tfbWVkaWFfY2hhbmdlX3JlYWwgKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpLAoJCQkJICAgICAgIGludCBzbG90X25yKQp7CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBjZGktPmhhbmRsZTsKCWludCByZXR2YWw7CgkKCWlmIChzbG90X25yID09IENEU0xfQ1VSUkVOVCkgewoJCSh2b2lkKSBjZHJvbV9jaGVja19zdGF0dXMoZHJpdmUsIE5VTEwpOwoJCXJldHZhbCA9IENEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+bWVkaWFfY2hhbmdlZDsKCQlDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPm1lZGlhX2NoYW5nZWQgPSAwOwoJCXJldHVybiByZXR2YWw7Cgl9IGVsc2UgewoJCXJldHVybiAtRUlOVkFMOwoJfQp9CgoKc3RhdGljCmludCBpZGVfY2Ryb21fb3Blbl9yZWFsIChzdHJ1Y3QgY2Ryb21fZGV2aWNlX2luZm8gKmNkaSwgaW50IHB1cnBvc2UpCnsKCXJldHVybiAwOwp9CgovKgogKiBDbG9zZSBkb3duIHRoZSBkZXZpY2UuICBJbnZhbGlkYXRlIGFsbCBjYWNoZWQgYmxvY2tzLgogKi8KCnN0YXRpYwp2b2lkIGlkZV9jZHJvbV9yZWxlYXNlX3JlYWwgKHN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqY2RpKQp7CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBjZGktPmhhbmRsZTsKCglpZiAoIWNkaS0+dXNlX2NvdW50KQoJCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+dG9jX3ZhbGlkID0gMDsKfQoKCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBEZXZpY2UgaW5pdGlhbGl6YXRpb24uCiAqLwpzdGF0aWMgc3RydWN0IGNkcm9tX2RldmljZV9vcHMgaWRlX2Nkcm9tX2RvcHMgPSB7Cgkub3BlbgkJCT0gaWRlX2Nkcm9tX29wZW5fcmVhbCwKCS5yZWxlYXNlCQk9IGlkZV9jZHJvbV9yZWxlYXNlX3JlYWwsCgkuZHJpdmVfc3RhdHVzCQk9IGlkZV9jZHJvbV9kcml2ZV9zdGF0dXMsCgkubWVkaWFfY2hhbmdlZAkJPSBpZGVfY2Ryb21fY2hlY2tfbWVkaWFfY2hhbmdlX3JlYWwsCgkudHJheV9tb3ZlCQk9IGlkZV9jZHJvbV90cmF5X21vdmUsCgkubG9ja19kb29yCQk9IGlkZV9jZHJvbV9sb2NrX2Rvb3IsCgkuc2VsZWN0X3NwZWVkCQk9IGlkZV9jZHJvbV9zZWxlY3Rfc3BlZWQsCgkuZ2V0X2xhc3Rfc2Vzc2lvbgk9IGlkZV9jZHJvbV9nZXRfbGFzdF9zZXNzaW9uLAoJLmdldF9tY24JCT0gaWRlX2Nkcm9tX2dldF9tY24sCgkucmVzZXQJCQk9IGlkZV9jZHJvbV9yZXNldCwKCS5hdWRpb19pb2N0bAkJPSBpZGVfY2Ryb21fYXVkaW9faW9jdGwsCgkuZGV2X2lvY3RsCQk9IGlkZV9jZHJvbV9kZXZfaW9jdGwsCgkuY2FwYWJpbGl0eQkJPSBDRENfQ0xPU0VfVFJBWSB8IENEQ19PUEVOX1RSQVkgfCBDRENfTE9DSyB8CgkJCQlDRENfU0VMRUNUX1NQRUVEIHwgQ0RDX1NFTEVDVF9ESVNDIHwKCQkJCUNEQ19NVUxUSV9TRVNTSU9OIHwgQ0RDX01DTiB8CgkJCQlDRENfTUVESUFfQ0hBTkdFRCB8IENEQ19QTEFZX0FVRElPIHwgQ0RDX1JFU0VUIHwKCQkJCUNEQ19JT0NUTFMgfCBDRENfRFJJVkVfU1RBVFVTIHwgQ0RDX0NEX1IgfAoJCQkJQ0RDX0NEX1JXIHwgQ0RDX0RWRCB8IENEQ19EVkRfUnwgQ0RDX0RWRF9SQU0gfAoJCQkJQ0RDX0dFTkVSSUNfUEFDS0VUIHwgQ0RDX01PX0RSSVZFIHwgQ0RDX01SVyB8CgkJCQlDRENfTVJXX1cgfCBDRENfUkFNLAoJLmdlbmVyaWNfcGFja2V0CQk9IGlkZV9jZHJvbV9wYWNrZXQsCn07CgpzdGF0aWMgaW50IGlkZV9jZHJvbV9yZWdpc3RlciAoaWRlX2RyaXZlX3QgKmRyaXZlLCBpbnQgbnNsb3RzKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqZGV2aW5mbyA9ICZpbmZvLT5kZXZpbmZvOwoKCWRldmluZm8tPm9wcyA9ICZpZGVfY2Ryb21fZG9wczsKCWRldmluZm8tPm1hc2sgPSAwOwoJZGV2aW5mby0+c3BlZWQgPSBDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPmN1cnJlbnRfc3BlZWQ7CglkZXZpbmZvLT5jYXBhY2l0eSA9IG5zbG90czsKCWRldmluZm8tPmhhbmRsZSA9IGRyaXZlOwoJc3RyY3B5KGRldmluZm8tPm5hbWUsIGRyaXZlLT5uYW1lKTsKCQoJLyogc2V0IGNhcGFiaWxpdHkgbWFzayB0byBtYXRjaCB0aGUgcHJvYmUuICovCglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3IpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfQ0RfUjsKCWlmICghQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2RfcncpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfQ0RfUlc7CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZCkKCQlkZXZpbmZvLT5tYXNrIHw9IENEQ19EVkQ7CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yKQoJCWRldmluZm8tPm1hc2sgfD0gQ0RDX0RWRF9SOwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcmFtKQoJCWRldmluZm8tPm1hc2sgfD0gQ0RDX0RWRF9SQU07CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmlzX2NoYW5nZXIpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfU0VMRUNUX0RJU0M7CglpZiAoIUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmF1ZGlvX3BsYXkpCgkJZGV2aW5mby0+bWFzayB8PSBDRENfUExBWV9BVURJTzsKCWlmICghQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2xvc2VfdHJheSkKCQlkZXZpbmZvLT5tYXNrIHw9IENEQ19DTE9TRV9UUkFZOwoJaWYgKCFDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5tb19kcml2ZSkKCQlkZXZpbmZvLT5tYXNrIHw9IENEQ19NT19EUklWRTsKCglkZXZpbmZvLT5kaXNrID0gaW5mby0+ZGlzazsKCXJldHVybiByZWdpc3Rlcl9jZHJvbShkZXZpbmZvKTsKfQoKc3RhdGljCmludCBpZGVfY2Ryb21fcHJvYmVfY2FwYWJpbGl0aWVzIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGkgPSAmaW5mby0+ZGV2aW5mbzsKCXN0cnVjdCBhdGFwaV9jYXBhYmlsaXRpZXNfcGFnZSBjYXA7CglpbnQgbnNsb3RzID0gMTsKCglpZiAoZHJpdmUtPm1lZGlhID09IGlkZV9vcHRpY2FsKSB7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bW9fZHJpdmUgPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnJhbSA9IDE7CgkJcHJpbnRrKEtFUk5fRVJSICIlczogQVRBUEkgbWFnbmV0by1vcHRpY2FsIGRyaXZlXG4iLCBkcml2ZS0+bmFtZSk7CgkJcmV0dXJuIG5zbG90czsKCX0KCglpZiAoQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bmVjMjYwIHx8CgkgICAgIXN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCJTVElOR1JBWSA4NDIyIElERSA4WCBDRC1ST00gNy0yNy05NSIpKSB7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZWplY3QgPSAwOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmF1ZGlvX3BsYXkgPSAxOwoJCXJldHVybiBuc2xvdHM7Cgl9CgoJLyoKCSAqIHdlIGhhdmUgdG8gY2hlYXQgYSBsaXR0bGUgaGVyZS4gdGhlIHBhY2tldCB3aWxsIGV2ZW50dWFsbHkKCSAqIGJlIHF1ZXVlZCB3aXRoIGlkZV9jZHJvbV9wYWNrZXQoKSwgd2hpY2ggZXh0cmFjdHMgdGhlCgkgKiBkcml2ZSBmcm9tIGNkaS0+aGFuZGxlLiBTaW5jZSB0aGlzIGRldmljZSBoYXNuJ3QgYmVlbgoJICogcmVnaXN0ZXJlZCB3aXRoIHRoZSBVbmlmb3JtIGxheWVyIHlldCwgaXQgY2FuJ3QgZG8gdGhpcy4KCSAqIFNhbWUgZ29lcyBmb3IgY2RpLT5vcHMuCgkgKi8KCWNkaS0+aGFuZGxlID0gZHJpdmU7CgljZGktPm9wcyA9ICZpZGVfY2Ryb21fZG9wczsKCglpZiAoaWRlX2Nkcm9tX2dldF9jYXBhYmlsaXRpZXMoZHJpdmUsICZjYXApKQoJCXJldHVybiAwOwoKCWlmIChjYXAubG9jayA9PSAwKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2Rvb3Jsb2NrID0gMTsKCWlmIChjYXAuZWplY3QpCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bm9fZWplY3QgPSAwOwoJaWYgKGNhcC5jZF9yX3dyaXRlKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3IgPSAxOwoJaWYgKGNhcC5jZF9yd193cml0ZSkgewoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3J3ID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5yYW0gPSAxOwoJfQoJaWYgKGNhcC50ZXN0X3dyaXRlKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRlc3Rfd3JpdGUgPSAxOwoJaWYgKGNhcC5kdmRfcmFtX3JlYWQgfHwgY2FwLmR2ZF9yX3JlYWQgfHwgY2FwLmR2ZF9yb20pCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+ZHZkID0gMTsKCWlmIChjYXAuZHZkX3JhbV93cml0ZSkgewoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yYW0gPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnJhbSA9IDE7Cgl9CglpZiAoY2FwLmR2ZF9yX3dyaXRlKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yID0gMTsKCWlmIChjYXAuYXVkaW9fcGxheSkKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5hdWRpb19wbGF5ID0gMTsKCWlmIChjYXAubWVjaHR5cGUgPT0gbWVjaHR5cGVfY2FkZHkgfHwgY2FwLm1lY2h0eXBlID09IG1lY2h0eXBlX3BvcHVwKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNsb3NlX3RyYXkgPSAwOwoKCS8qIFNvbWUgZHJpdmVzIHVzZWQgYnkgQXBwbGUgZG9uJ3QgYWR2ZXJ0aXNlIGF1ZGlvIHBsYXkKCSAqIGJ1dCB0aGV5IGRvIHN1cHBvcnQgcmVhZGluZyBUT0MgJiBhdWRpbyBkYXRhcwoJICovCglpZiAoc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJNQVRTSElUQURWRC1ST00gU1ItODE4NyIpID09IDAgfHwKCSAgICBzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIk1BVFNISVRBRFZELVJPTSBTUi04MTg2IikgPT0gMCB8fAoJICAgIHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiTUFUU0hJVEFEVkQtUk9NIFNSLTgxNzYiKSA9PSAwIHx8CgkgICAgc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJNQVRTSElUQURWRC1ST00gU1ItODE3NCIpID09IDApCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+YXVkaW9fcGxheSA9IDE7CgojaWYgISBTVEFOREFSRF9BVEFQSQoJaWYgKGNkaS0+c2FueW9fc2xvdCA+IDApIHsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5pc19jaGFuZ2VyID0gMTsKCQluc2xvdHMgPSAzOwoJfQoKCWVsc2UKI2VuZGlmIC8qIG5vdCBTVEFOREFSRF9BVEFQSSAqLwoJaWYgKGNhcC5tZWNodHlwZSA9PSBtZWNodHlwZV9pbmRpdmlkdWFsX2NoYW5nZXIgfHwKCSAgICBjYXAubWVjaHR5cGUgPT0gbWVjaHR5cGVfY2FydHJpZGdlX2NoYW5nZXIpIHsKCQlpZiAoKG5zbG90cyA9IGNkcm9tX251bWJlcl9vZl9zbG90cyhjZGkpKSA+IDEpIHsKCQkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+aXNfY2hhbmdlciA9IDE7CgkJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnN1cHBfZGlzY19wcmVzZW50ID0gMTsKCQl9Cgl9CgoJaWRlX2Nkcm9tX3VwZGF0ZV9zcGVlZChkcml2ZSwgJmNhcCk7CgkvKiBkb24ndCBwcmludCBzcGVlZCBpZiB0aGUgZHJpdmUgcmVwb3J0ZWQgMC4KCSAqLwoJcHJpbnRrKEtFUk5fSU5GTyAiJXM6IEFUQVBJIiwgZHJpdmUtPm5hbWUpOwoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm1heF9zcGVlZCkKCQlwcmludGsoIiAlZFgiLCBDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5tYXhfc3BlZWQpOwoJcHJpbnRrKCIgJXMiLCBDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmQgPyAiRFZELVJPTSIgOiAiQ0QtUk9NIik7CgoJaWYgKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yfENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yYW0pCiAgICAgICAgCXByaW50aygiIERWRCVzJXMiLCAKICAgICAgICAJKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yKT8gIi1SIiA6ICIiLCAKICAgICAgICAJKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZF9yYW0pPyAiLVJBTSIgOiAiIik7CgogICAgICAgIGlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yfENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3J3KSAKICAgICAgICAJcHJpbnRrKCIgQ0QlcyVzIiwgCiAgICAgICAgCShDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9yKT8gIi1SIiA6ICIiLCAKICAgICAgICAJKENEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmNkX3J3KT8gIi9SVyIgOiAiIik7CgogICAgICAgIGlmIChDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5pc19jaGFuZ2VyKSAKICAgICAgICAJcHJpbnRrKCIgY2hhbmdlciB3LyVkIHNsb3RzIiwgbnNsb3RzKTsKICAgICAgICBlbHNlIAkKICAgICAgICAJcHJpbnRrKCIgZHJpdmUiKTsKCglwcmludGsoIiwgJWRrQiBDYWNoZSIsIGJlMTZfdG9fY3B1KGNhcC5idWZmZXJfc2l6ZSkpOwoKCWlmIChkcml2ZS0+dXNpbmdfZG1hKQoJCWlkZV9kbWFfdmVyYm9zZShkcml2ZSk7CgoJcHJpbnRrKCJcbiIpOwoKCXJldHVybiBuc2xvdHM7Cn0KCnN0YXRpYyB2b2lkIGlkZV9jZHJvbV9hZGRfc2V0dGluZ3MoaWRlX2RyaXZlX3QgKmRyaXZlKQp7CglpZGVfYWRkX3NldHRpbmcoZHJpdmUsCSJkc2Nfb3ZlcmxhcCIsCQlTRVRUSU5HX1JXLCAtMSwgLTEsIFRZUEVfQllURSwgMCwgMSwgMSwJMSwgJmRyaXZlLT5kc2Nfb3ZlcmxhcCwgTlVMTCk7Cn0KCi8qCiAqIHN0YW5kYXJkIHByZXBfcnFfZm4gdGhhdCBidWlsZHMgMTAgYnl0ZSBjbWRzCiAqLwpzdGF0aWMgaW50IGlkZV9jZHJvbV9wcmVwX2ZzKHJlcXVlc3RfcXVldWVfdCAqcSwgc3RydWN0IHJlcXVlc3QgKnJxKQp7CglpbnQgaGFyZF9zZWN0ID0gcXVldWVfaGFyZHNlY3Rfc2l6ZShxKTsKCWxvbmcgYmxvY2sgPSAobG9uZylycS0+aGFyZF9zZWN0b3IgLyAoaGFyZF9zZWN0ID4+IDkpOwoJdW5zaWduZWQgbG9uZyBibG9ja3MgPSBycS0+aGFyZF9ucl9zZWN0b3JzIC8gKGhhcmRfc2VjdCA+PiA5KTsKCgltZW1zZXQocnEtPmNtZCwgMCwgc2l6ZW9mKHJxLT5jbWQpKTsKCglpZiAocnFfZGF0YV9kaXIocnEpID09IFJFQUQpCgkJcnEtPmNtZFswXSA9IEdQQ01EX1JFQURfMTA7CgllbHNlCgkJcnEtPmNtZFswXSA9IEdQQ01EX1dSSVRFXzEwOwoKCS8qCgkgKiBmaWxsIGluIGxiYQoJICovCglycS0+Y21kWzJdID0gKGJsb2NrID4+IDI0KSAmIDB4ZmY7CglycS0+Y21kWzNdID0gKGJsb2NrID4+IDE2KSAmIDB4ZmY7CglycS0+Y21kWzRdID0gKGJsb2NrID4+ICA4KSAmIDB4ZmY7CglycS0+Y21kWzVdID0gYmxvY2sgJiAweGZmOwoKCS8qCgkgKiBhbmQgdHJhbnNmZXIgbGVuZ3RoCgkgKi8KCXJxLT5jbWRbN10gPSAoYmxvY2tzID4+IDgpICYgMHhmZjsKCXJxLT5jbWRbOF0gPSBibG9ja3MgJiAweGZmOwoJcnEtPmNtZF9sZW4gPSAxMDsKCXJldHVybiBCTEtQUkVQX09LOwp9CgovKgogKiBNb3N0IG9mIHRoZSBTQ1NJIGNvbW1hbmRzIGFyZSBzdXBwb3J0ZWQgZGlyZWN0bHkgYnkgQVRBUEkgZGV2aWNlcy4KICogVGhpcyB0cmFuc2Zvcm0gaGFuZGxlcyB0aGUgZmV3IGV4Y2VwdGlvbnMuCiAqLwpzdGF0aWMgaW50IGlkZV9jZHJvbV9wcmVwX3BjKHN0cnVjdCByZXF1ZXN0ICpycSkKewoJdTggKmMgPSBycS0+Y21kOwoKCS8qCgkgKiBUcmFuc2Zvcm0gNi1ieXRlIHJlYWQvd3JpdGUgY29tbWFuZHMgdG8gdGhlIDEwLWJ5dGUgdmVyc2lvbgoJICovCglpZiAoY1swXSA9PSBSRUFEXzYgfHwgY1swXSA9PSBXUklURV82KSB7CgkJY1s4XSA9IGNbNF07CgkJY1s1XSA9IGNbM107CgkJY1s0XSA9IGNbMl07CgkJY1szXSA9IGNbMV0gJiAweDFmOwoJCWNbMl0gPSAwOwoJCWNbMV0gJj0gMHhlMDsKCQljWzBdICs9IChSRUFEXzEwIC0gUkVBRF82KTsKCQlycS0+Y21kX2xlbiA9IDEwOwoJCXJldHVybiBCTEtQUkVQX09LOwoJfQoKCS8qCgkgKiBpdCdzIHNpbGx5IHRvIHByZXRlbmQgd2UgdW5kZXJzdGFuZCA2LWJ5dGUgc2Vuc2UgY29tbWFuZHMsIGp1c3QKCSAqIHJlamVjdCB3aXRoIElMTEVHQUxfUkVRVUVTVCBhbmQgdGhlIGNhbGxlciBzaG91bGQgdGFrZSB0aGUKCSAqIGFwcHJvcHJpYXRlIGFjdGlvbgoJICovCglpZiAoY1swXSA9PSBNT0RFX1NFTlNFIHx8IGNbMF0gPT0gTU9ERV9TRUxFQ1QpIHsKCQlycS0+ZXJyb3JzID0gSUxMRUdBTF9SRVFVRVNUOwoJCXJldHVybiBCTEtQUkVQX0tJTEw7Cgl9CgkKCXJldHVybiBCTEtQUkVQX09LOwp9CgpzdGF0aWMgaW50IGlkZV9jZHJvbV9wcmVwX2ZuKHJlcXVlc3RfcXVldWVfdCAqcSwgc3RydWN0IHJlcXVlc3QgKnJxKQp7CglpZiAocnEtPmZsYWdzICYgUkVRX0NNRCkKCQlyZXR1cm4gaWRlX2Nkcm9tX3ByZXBfZnMocSwgcnEpOwoJZWxzZSBpZiAocnEtPmZsYWdzICYgUkVRX0JMT0NLX1BDKQoJCXJldHVybiBpZGVfY2Ryb21fcHJlcF9wYyhycSk7CgoJcmV0dXJuIDA7Cn0KCnN0YXRpYwppbnQgaWRlX2Nkcm9tX3NldHVwIChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gZHJpdmUtPmRyaXZlcl9kYXRhOwoJc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpjZGkgPSAmaW5mby0+ZGV2aW5mbzsKCWludCBuc2xvdHM7CgoJYmxrX3F1ZXVlX3ByZXBfcnEoZHJpdmUtPnF1ZXVlLCBpZGVfY2Ryb21fcHJlcF9mbik7CglibGtfcXVldWVfZG1hX2FsaWdubWVudChkcml2ZS0+cXVldWUsIDMxKTsKCWRyaXZlLT5xdWV1ZS0+dW5wbHVnX2RlbGF5ID0gKDEgKiBIWikgLyAxMDAwOwoJaWYgKCFkcml2ZS0+cXVldWUtPnVucGx1Z19kZWxheSkKCQlkcml2ZS0+cXVldWUtPnVucGx1Z19kZWxheSA9IDE7CgoJZHJpdmUtPnNwZWNpYWwuYWxsCT0gMDsKCglDRFJPTV9TVEFURV9GTEFHUyhkcml2ZSktPm1lZGlhX2NoYW5nZWQgPSAxOwoJQ0RST01fU1RBVEVfRkxBR1MoZHJpdmUpLT50b2NfdmFsaWQgICAgID0gMDsKCUNEUk9NX1NUQVRFX0ZMQUdTKGRyaXZlKS0+ZG9vcl9sb2NrZWQgICA9IDA7CgojaWYgTk9fRE9PUl9MT0NLSU5HCglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5ub19kb29ybG9jayA9IDE7CiNlbHNlCglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5ub19kb29ybG9jayA9IDA7CiNlbmRpZgoKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmRycV9pbnRlcnJ1cHQgPSAoKGRyaXZlLT5pZC0+Y29uZmlnICYgMHgwMDYwKSA9PSAweDIwKTsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmlzX2NoYW5nZXIgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2RfciA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5jZF9ydyA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50ZXN0X3dyaXRlID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmR2ZCA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfciA9IDA7CglDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5kdmRfcmFtID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPm5vX2VqZWN0ID0gMTsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnN1cHBfZGlzY19wcmVzZW50ID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmF1ZGlvX3BsYXkgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+Y2xvc2VfdHJheSA9IDE7CgkKCS8qIGxpbWl0IHRyYW5zZmVyIHNpemUgcGVyIGludGVycnVwdC4gKi8KCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmxpbWl0X25mcmFtZXMgPSAwOwoJLyogYSB0ZXN0YW1lbnQgdG8gdGhlIG5pY2UgcXVhbGl0eSBvZiBTYW1zdW5nIGRyaXZlcy4uLiAqLwoJaWYgKCFzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIlNBTVNVTkcgQ0QtUk9NIFNDUi0yNDMwIikpCgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bGltaXRfbmZyYW1lcyA9IDE7CgllbHNlIGlmICghc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJTQU1TVU5HIENELVJPTSBTQ1ItMjQzMiIpKQoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPmxpbWl0X25mcmFtZXMgPSAxOwoJLyogdGhlIDMyMzEgbW9kZWwgZG9lcyBub3Qgc3VwcG9ydCB0aGUgU0VUX0NEX1NQRUVEIGNvbW1hbmQgKi8KCWVsc2UgaWYgKCFzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIlNBTVNVTkcgQ0QtUk9NIFNDUi0zMjMxIikpCgkJY2RpLT5tYXNrIHw9IENEQ19TRUxFQ1RfU1BFRUQ7CgojaWYgISBTVEFOREFSRF9BVEFQSQoJLyogYnkgZGVmYXVsdCBTYW55byAzIENEIGNoYW5nZXIgc3VwcG9ydCBpcyB0dXJuZWQgb2ZmIGFuZAogICAgICAgICAgIEFUQVBJIFJldiAyLjIrIHN0YW5kYXJkIHN1cHBvcnQgZm9yIENEIGNoYW5nZXJzIGlzIHVzZWQgKi8KCWNkaS0+c2FueW9fc2xvdCA9IDA7CgoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+bmVjMjYwID0gMDsKCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY3RyYWNrc19hc19iY2QgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+dG9jYWRkcl9hc19iY2QgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+cGxheW1zZl9hc19iY2QgPSAwOwoJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+c3ViY2hhbl9hc19iY2QgPSAwOwoKCWlmIChzdHJjbXAgKGRyaXZlLT5pZC0+bW9kZWwsICJWMDAzUzBEUyIpID09IDAgJiYKCSAgICBkcml2ZS0+aWQtPmZ3X3Jldls0XSA9PSAnMScgJiYKCSAgICBkcml2ZS0+aWQtPmZ3X3Jldls2XSA8PSAnMicpIHsKCQkvKiBWZXJ0b3MgMzAwLgoJCSAgIFNvbWUgdmVyc2lvbnMgb2YgdGhpcyBkcml2ZSBsaWtlIHRvIHRhbGsgQkNELiAqLwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY3RyYWNrc19hc19iY2QgPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY2FkZHJfYXNfYmNkID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5wbGF5bXNmX2FzX2JjZCA9IDE7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+c3ViY2hhbl9hc19iY2QgPSAxOwoJfQoKCWVsc2UgaWYgKHN0cmNtcCAoZHJpdmUtPmlkLT5tb2RlbCwgIlYwMDZFMERTIikgPT0gMCAmJgoJICAgIGRyaXZlLT5pZC0+ZndfcmV2WzRdID09ICcxJyAmJgoJICAgIGRyaXZlLT5pZC0+ZndfcmV2WzZdIDw9ICcyJykgewoJCS8qIFZlcnRvcyA2MDAgRVNELiAqLwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnRvY3RyYWNrc19hc19iY2QgPSAxOwoJfQoJZWxzZSBpZiAoc3RyY21wKGRyaXZlLT5pZC0+bW9kZWwsICJORUMgQ0QtUk9NIERSSVZFOjI2MCIpID09IDAgJiYKCQkgc3RybmNtcChkcml2ZS0+aWQtPmZ3X3JldiwgIjEuMDEiLCA0KSA9PSAwKSB7IC8qIEZJWE1FICovCgkJLyogT2xkIE5FQzI2MCAobm90IFIpLgoJCSAgIFRoaXMgZHJpdmUgd2FzIHJlbGVhc2VkIGJlZm9yZSB0aGUgMS4yIHZlcnNpb24KCQkgICBvZiB0aGUgc3BlYy4gKi8KCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT50b2NhZGRyX2FzX2JjZCA9IDE7CgkJQ0RST01fQ09ORklHX0ZMQUdTKGRyaXZlKS0+cGxheW1zZl9hc19iY2QgPSAxOwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnN1YmNoYW5fYXNfYmNkID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5uZWMyNjAgICAgICAgICA9IDE7Cgl9CgllbHNlIGlmIChzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIldFQVJORVMgQ0RELTEyMCIpID09IDAgJiYKCQkgc3RybmNtcChkcml2ZS0+aWQtPmZ3X3JldiwgIkExLjEiLCA0KSA9PSAwKSB7IC8qIEZJWE1FICovCgkJLyogV2Vhcm5lcyAqLwoJCUNEUk9NX0NPTkZJR19GTEFHUyhkcml2ZSktPnBsYXltc2ZfYXNfYmNkID0gMTsKCQlDRFJPTV9DT05GSUdfRkxBR1MoZHJpdmUpLT5zdWJjaGFuX2FzX2JjZCA9IDE7Cgl9CiAgICAgICAgLyogU2FueW8gMyBDRCBjaGFuZ2VyIHVzZXMgYSBub24tc3RhbmRhcmQgY29tbWFuZAogICAgICAgICAgIGZvciBDRCBjaGFuZ2luZyAqLwogICAgICAgIGVsc2UgaWYgKChzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIkNELVJPTSBDRFItQzMgRyIpID09IDApIHx8CiAgICAgICAgICAgICAgICAgKHN0cmNtcChkcml2ZS0+aWQtPm1vZGVsLCAiQ0QtUk9NIENEUi1DM0ciKSA9PSAwKSB8fAogICAgICAgICAgICAgICAgIChzdHJjbXAoZHJpdmUtPmlkLT5tb2RlbCwgIkNELVJPTSBDRFJfQzM2IikgPT0gMCkpIHsKICAgICAgICAgICAgICAgICAvKiB1c2VzIENEIGluIHNsb3QgMCB3aGVuIHZhbHVlIGlzIHNldCB0byAzICovCiAgICAgICAgICAgICAgICAgY2RpLT5zYW55b19zbG90ID0gMzsKICAgICAgICB9CiNlbmRpZiAvKiBub3QgU1RBTkRBUkRfQVRBUEkgKi8KCglpbmZvLT50b2MJCT0gTlVMTDsKCWluZm8tPmJ1ZmZlcgkJPSBOVUxMOwoJaW5mby0+c2VjdG9yX2J1ZmZlcmVkCT0gMDsKCWluZm8tPm5zZWN0b3JzX2J1ZmZlcmVkCT0gMDsKCWluZm8tPmNoYW5nZXJfaW5mbyAgICAgID0gTlVMTDsKCWluZm8tPmxhc3RfYmxvY2sJPSAwOwoJaW5mby0+c3RhcnRfc2Vlawk9IDA7CgoJbnNsb3RzID0gaWRlX2Nkcm9tX3Byb2JlX2NhcGFiaWxpdGllcyAoZHJpdmUpOwoKCS8qCgkgKiBzZXQgY29ycmVjdCBibG9jayBzaXplCgkgKi8KCWJsa19xdWV1ZV9oYXJkc2VjdF9zaXplKGRyaXZlLT5xdWV1ZSwgQ0RfRlJBTUVTSVpFKTsKCglpZiAoZHJpdmUtPmF1dG90dW5lID09IElERV9UVU5FX0RFRkFVTFQgfHwKCSAgICBkcml2ZS0+YXV0b3R1bmUgPT0gSURFX1RVTkVfQVVUTykKCQlkcml2ZS0+ZHNjX292ZXJsYXAgPSAoZHJpdmUtPm5leHQgIT0gZHJpdmUpOwojaWYgMAoJZHJpdmUtPmRzY19vdmVybGFwID0gKEhXSUYoZHJpdmUpLT5ub19kc2MpID8gMCA6IDE7CglpZiAoSFdJRihkcml2ZSktPm5vX2RzYykgewoJCXByaW50ayhLRVJOX0lORk8gImlkZS1jZDogJXM6IGRpc2FibGluZyBEU0Mgb3ZlcmxhcFxuIiwKCQkJZHJpdmUtPm5hbWUpOwoJCWRyaXZlLT5kc2Nfb3ZlcmxhcCA9IDA7Cgl9CiNlbmRpZgoKCWlmIChpZGVfY2Ryb21fcmVnaXN0ZXIoZHJpdmUsIG5zbG90cykpIHsKCQlwcmludGsgKEtFUk5fRVJSICIlczogaWRlX2Nkcm9tX3NldHVwIGZhaWxlZCB0byByZWdpc3RlciBkZXZpY2Ugd2l0aCB0aGUgY2Ryb20gZHJpdmVyLlxuIiwgZHJpdmUtPm5hbWUpOwoJCWluZm8tPmRldmluZm8uaGFuZGxlID0gTlVMTDsKCQlyZXR1cm4gMTsKCX0KCWlkZV9jZHJvbV9hZGRfc2V0dGluZ3MoZHJpdmUpOwoJcmV0dXJuIDA7Cn0KCiNpZmRlZiBDT05GSUdfUFJPQ19GUwpzdGF0aWMKc2VjdG9yX3QgaWRlX2Nkcm9tX2NhcGFjaXR5IChpZGVfZHJpdmVfdCAqZHJpdmUpCnsKCXVuc2lnbmVkIGxvbmcgY2FwYWNpdHksIHNlY3RvcnNfcGVyX2ZyYW1lOwoKCWlmIChjZHJvbV9yZWFkX2NhcGFjaXR5KGRyaXZlLCAmY2FwYWNpdHksICZzZWN0b3JzX3Blcl9mcmFtZSwgTlVMTCkpCgkJcmV0dXJuIDA7CgoJcmV0dXJuIGNhcGFjaXR5ICogc2VjdG9yc19wZXJfZnJhbWU7Cn0KI2VuZGlmCgpzdGF0aWMgaW50IGlkZV9jZF9yZW1vdmUoc3RydWN0IGRldmljZSAqZGV2KQp7CglpZGVfZHJpdmVfdCAqZHJpdmUgPSB0b19pZGVfZGV2aWNlKGRldik7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGRyaXZlLT5kcml2ZXJfZGF0YTsKCglpZGVfdW5yZWdpc3Rlcl9zdWJkcml2ZXIoZHJpdmUsIGluZm8tPmRyaXZlcik7CgoJZGVsX2dlbmRpc2soaW5mby0+ZGlzayk7CgoJaWRlX2NkX3B1dChpbmZvKTsKCglyZXR1cm4gMDsKfQoKc3RhdGljIHZvaWQgaWRlX2NkX3JlbGVhc2Uoc3RydWN0IGtyZWYgKmtyZWYpCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gdG9faWRlX2NkKGtyZWYpOwoJc3RydWN0IGNkcm9tX2RldmljZV9pbmZvICpkZXZpbmZvID0gJmluZm8tPmRldmluZm87CglpZGVfZHJpdmVfdCAqZHJpdmUgPSBpbmZvLT5kcml2ZTsKCXN0cnVjdCBnZW5kaXNrICpnID0gaW5mby0+ZGlzazsKCglrZnJlZShpbmZvLT5idWZmZXIpOwoJa2ZyZWUoaW5mby0+dG9jKTsKCWtmcmVlKGluZm8tPmNoYW5nZXJfaW5mbyk7CglpZiAoZGV2aW5mby0+aGFuZGxlID09IGRyaXZlICYmIHVucmVnaXN0ZXJfY2Ryb20oZGV2aW5mbykpCgkJcHJpbnRrKEtFUk5fRVJSICIlczogJXMgZmFpbGVkIHRvIHVucmVnaXN0ZXIgZGV2aWNlIGZyb20gdGhlIGNkcm9tICIKCQkJCSJkcml2ZXIuXG4iLCBfX0ZVTkNUSU9OX18sIGRyaXZlLT5uYW1lKTsKCWRyaXZlLT5kc2Nfb3ZlcmxhcCA9IDA7Cglkcml2ZS0+ZHJpdmVyX2RhdGEgPSBOVUxMOwoJYmxrX3F1ZXVlX3ByZXBfcnEoZHJpdmUtPnF1ZXVlLCBOVUxMKTsKCWctPnByaXZhdGVfZGF0YSA9IE5VTEw7CglwdXRfZGlzayhnKTsKCWtmcmVlKGluZm8pOwp9CgpzdGF0aWMgaW50IGlkZV9jZF9wcm9iZShzdHJ1Y3QgZGV2aWNlICopOwoKI2lmZGVmIENPTkZJR19QUk9DX0ZTCnN0YXRpYyBpbnQgcHJvY19pZGVjZF9yZWFkX2NhcGFjaXR5CgkoY2hhciAqcGFnZSwgY2hhciAqKnN0YXJ0LCBvZmZfdCBvZmYsIGludCBjb3VudCwgaW50ICplb2YsIHZvaWQgKmRhdGEpCnsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGRhdGE7CglpbnQgbGVuOwoKCWxlbiA9IHNwcmludGYocGFnZSwiJWxsdVxuIiwgKGxvbmcgbG9uZylpZGVfY2Ryb21fY2FwYWNpdHkoZHJpdmUpKTsKCVBST0NfSURFX1JFQURfUkVUVVJOKHBhZ2Usc3RhcnQsb2ZmLGNvdW50LGVvZixsZW4pOwp9CgpzdGF0aWMgaWRlX3Byb2NfZW50cnlfdCBpZGVjZF9wcm9jW10gPSB7Cgl7ICJjYXBhY2l0eSIsIFNfSUZSRUd8U19JUlVHTywgcHJvY19pZGVjZF9yZWFkX2NhcGFjaXR5LCBOVUxMIH0sCgl7IE5VTEwsIDAsIE5VTEwsIE5VTEwgfQp9OwojZWxzZQojIGRlZmluZSBpZGVjZF9wcm9jCU5VTEwKI2VuZGlmCgpzdGF0aWMgaWRlX2RyaXZlcl90IGlkZV9jZHJvbV9kcml2ZXIgPSB7CgkuZ2VuX2RyaXZlciA9IHsKCQkub3duZXIJCT0gVEhJU19NT0RVTEUsCgkJLm5hbWUJCT0gImlkZS1jZHJvbSIsCgkJLmJ1cwkJPSAmaWRlX2J1c190eXBlLAoJCS5wcm9iZQkJPSBpZGVfY2RfcHJvYmUsCgkJLnJlbW92ZQkJPSBpZGVfY2RfcmVtb3ZlLAoJfSwKCS52ZXJzaW9uCQk9IElERUNEX1ZFUlNJT04sCgkubWVkaWEJCQk9IGlkZV9jZHJvbSwKCS5zdXBwb3J0c19kc2Nfb3ZlcmxhcAk9IDEsCgkuZG9fcmVxdWVzdAkJPSBpZGVfZG9fcndfY2Ryb20sCgkuZW5kX3JlcXVlc3QJCT0gaWRlX2VuZF9yZXF1ZXN0LAoJLmVycm9yCQkJPSBfX2lkZV9lcnJvciwKCS5hYm9ydAkJCT0gX19pZGVfYWJvcnQsCgkucHJvYwkJCT0gaWRlY2RfcHJvYywKfTsKCnN0YXRpYyBpbnQgaWRlY2Rfb3BlbihzdHJ1Y3QgaW5vZGUgKiBpbm9kZSwgc3RydWN0IGZpbGUgKiBmaWxlKQp7CglzdHJ1Y3QgZ2VuZGlzayAqZGlzayA9IGlub2RlLT5pX2JkZXYtPmJkX2Rpc2s7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbzsKCWlkZV9kcml2ZV90ICpkcml2ZTsKCWludCByYyA9IC1FTk9NRU07CgoJaWYgKCEoaW5mbyA9IGlkZV9jZF9nZXQoZGlzaykpKQoJCXJldHVybiAtRU5YSU87CgoJZHJpdmUgPSBpbmZvLT5kcml2ZTsKCglkcml2ZS0+dXNhZ2UrKzsKCglpZiAoIWluZm8tPmJ1ZmZlcikKCQlpbmZvLT5idWZmZXIgPSBrbWFsbG9jKFNFQ1RPUl9CVUZGRVJfU0laRSwKCQkJCQlHRlBfS0VSTkVMfF9fR0ZQX1JFUEVBVCk7CiAgICAgICAgaWYgKCFpbmZvLT5idWZmZXIgfHwgKHJjID0gY2Ryb21fb3BlbigmaW5mby0+ZGV2aW5mbywgaW5vZGUsIGZpbGUpKSkKCQlkcml2ZS0+dXNhZ2UtLTsKCglpZiAocmMgPCAwKQoJCWlkZV9jZF9wdXQoaW5mbyk7CgoJcmV0dXJuIHJjOwp9CgpzdGF0aWMgaW50IGlkZWNkX3JlbGVhc2Uoc3RydWN0IGlub2RlICogaW5vZGUsIHN0cnVjdCBmaWxlICogZmlsZSkKewoJc3RydWN0IGdlbmRpc2sgKmRpc2sgPSBpbm9kZS0+aV9iZGV2LT5iZF9kaXNrOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm8gPSBpZGVfY2RfZyhkaXNrKTsKCWlkZV9kcml2ZV90ICpkcml2ZSA9IGluZm8tPmRyaXZlOwoKCWNkcm9tX3JlbGVhc2UgKCZpbmZvLT5kZXZpbmZvLCBmaWxlKTsKCWRyaXZlLT51c2FnZS0tOwoKCWlkZV9jZF9wdXQoaW5mbyk7CgoJcmV0dXJuIDA7Cn0KCnN0YXRpYyBpbnQgaWRlY2RfaW9jdGwgKHN0cnVjdCBpbm9kZSAqaW5vZGUsIHN0cnVjdCBmaWxlICpmaWxlLAoJCQl1bnNpZ25lZCBpbnQgY21kLCB1bnNpZ25lZCBsb25nIGFyZykKewoJc3RydWN0IGJsb2NrX2RldmljZSAqYmRldiA9IGlub2RlLT5pX2JkZXY7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGlkZV9jZF9nKGJkZXYtPmJkX2Rpc2spOwoJaW50IGVycjsKCgllcnIgID0gZ2VuZXJpY19pZGVfaW9jdGwoaW5mby0+ZHJpdmUsIGZpbGUsIGJkZXYsIGNtZCwgYXJnKTsKCWlmIChlcnIgPT0gLUVJTlZBTCkKCQllcnIgPSBjZHJvbV9pb2N0bChmaWxlLCAmaW5mby0+ZGV2aW5mbywgaW5vZGUsIGNtZCwgYXJnKTsKCglyZXR1cm4gZXJyOwp9CgpzdGF0aWMgaW50IGlkZWNkX21lZGlhX2NoYW5nZWQoc3RydWN0IGdlbmRpc2sgKmRpc2spCnsKCXN0cnVjdCBjZHJvbV9pbmZvICppbmZvID0gaWRlX2NkX2coZGlzayk7CglyZXR1cm4gY2Ryb21fbWVkaWFfY2hhbmdlZCgmaW5mby0+ZGV2aW5mbyk7Cn0KCnN0YXRpYyBpbnQgaWRlY2RfcmV2YWxpZGF0ZV9kaXNrKHN0cnVjdCBnZW5kaXNrICpkaXNrKQp7CglzdHJ1Y3QgY2Ryb21faW5mbyAqaW5mbyA9IGlkZV9jZF9nKGRpc2spOwoJc3RydWN0IHJlcXVlc3Rfc2Vuc2Ugc2Vuc2U7CgljZHJvbV9yZWFkX3RvYyhpbmZvLT5kcml2ZSwgJnNlbnNlKTsKCXJldHVybiAgMDsKfQoKc3RhdGljIHN0cnVjdCBibG9ja19kZXZpY2Vfb3BlcmF0aW9ucyBpZGVjZF9vcHMgPSB7Cgkub3duZXIJCT0gVEhJU19NT0RVTEUsCgkub3BlbgkJPSBpZGVjZF9vcGVuLAoJLnJlbGVhc2UJPSBpZGVjZF9yZWxlYXNlLAoJLmlvY3RsCQk9IGlkZWNkX2lvY3RsLAoJLm1lZGlhX2NoYW5nZWQJPSBpZGVjZF9tZWRpYV9jaGFuZ2VkLAoJLnJldmFsaWRhdGVfZGlzaz0gaWRlY2RfcmV2YWxpZGF0ZV9kaXNrCn07CgovKiBvcHRpb25zICovCnN0YXRpYyBjaGFyICppZ25vcmUgPSBOVUxMOwoKbW9kdWxlX3BhcmFtKGlnbm9yZSwgY2hhcnAsIDA0MDApOwpNT0RVTEVfREVTQ1JJUFRJT04oIkFUQVBJIENELVJPTSBEcml2ZXIiKTsKCnN0YXRpYyBpbnQgaWRlX2NkX3Byb2JlKHN0cnVjdCBkZXZpY2UgKmRldikKewoJaWRlX2RyaXZlX3QgKmRyaXZlID0gdG9faWRlX2RldmljZShkZXYpOwoJc3RydWN0IGNkcm9tX2luZm8gKmluZm87CglzdHJ1Y3QgZ2VuZGlzayAqZzsKCXN0cnVjdCByZXF1ZXN0X3NlbnNlIHNlbnNlOwoKCWlmICghc3Ryc3RyKCJpZGUtY2Ryb20iLCBkcml2ZS0+ZHJpdmVyX3JlcSkpCgkJZ290byBmYWlsZWQ7CglpZiAoIWRyaXZlLT5wcmVzZW50KQoJCWdvdG8gZmFpbGVkOwoJaWYgKGRyaXZlLT5tZWRpYSAhPSBpZGVfY2Ryb20gJiYgZHJpdmUtPm1lZGlhICE9IGlkZV9vcHRpY2FsKQoJCWdvdG8gZmFpbGVkOwoJLyogc2tpcCBkcml2ZXMgdGhhdCB3ZSB3ZXJlIHRvbGQgdG8gaWdub3JlICovCglpZiAoaWdub3JlICE9IE5VTEwpIHsKCQlpZiAoc3Ryc3RyKGlnbm9yZSwgZHJpdmUtPm5hbWUpKSB7CgkJCXByaW50ayhLRVJOX0lORk8gImlkZS1jZDogaWdub3JpbmcgZHJpdmUgJXNcbiIsIGRyaXZlLT5uYW1lKTsKCQkJZ290byBmYWlsZWQ7CgkJfQoJfQoJaWYgKGRyaXZlLT5zY3NpKSB7CgkJcHJpbnRrKEtFUk5fSU5GTyAiaWRlLWNkOiBwYXNzaW5nIGRyaXZlICVzIHRvIGlkZS1zY3NpIGVtdWxhdGlvbi5cbiIsIGRyaXZlLT5uYW1lKTsKCQlnb3RvIGZhaWxlZDsKCX0KCWluZm8gPSBremFsbG9jKHNpemVvZihzdHJ1Y3QgY2Ryb21faW5mbyksIEdGUF9LRVJORUwpOwoJaWYgKGluZm8gPT0gTlVMTCkgewoJCXByaW50ayhLRVJOX0VSUiAiJXM6IENhbid0IGFsbG9jYXRlIGEgY2Ryb20gc3RydWN0dXJlXG4iLCBkcml2ZS0+bmFtZSk7CgkJZ290byBmYWlsZWQ7Cgl9CgoJZyA9IGFsbG9jX2Rpc2soMSA8PCBQQVJUTl9CSVRTKTsKCWlmICghZykKCQlnb3RvIG91dF9mcmVlX2NkOwoKCWlkZV9pbml0X2Rpc2soZywgZHJpdmUpOwoKCWlkZV9yZWdpc3Rlcl9zdWJkcml2ZXIoZHJpdmUsICZpZGVfY2Ryb21fZHJpdmVyKTsKCglrcmVmX2luaXQoJmluZm8tPmtyZWYpOwoKCWluZm8tPmRyaXZlID0gZHJpdmU7CglpbmZvLT5kcml2ZXIgPSAmaWRlX2Nkcm9tX2RyaXZlcjsKCWluZm8tPmRpc2sgPSBnOwoKCWctPnByaXZhdGVfZGF0YSA9ICZpbmZvLT5kcml2ZXI7CgoJZHJpdmUtPmRyaXZlcl9kYXRhID0gaW5mbzsKCglnLT5taW5vcnMgPSAxOwoJc25wcmludGYoZy0+ZGV2ZnNfbmFtZSwgc2l6ZW9mKGctPmRldmZzX25hbWUpLAoJCQkiJXMvY2QiLCBkcml2ZS0+ZGV2ZnNfbmFtZSk7CglnLT5kcml2ZXJmc19kZXYgPSAmZHJpdmUtPmdlbmRldjsKCWctPmZsYWdzID0gR0VOSERfRkxfQ0QgfCBHRU5IRF9GTF9SRU1PVkFCTEU7CglpZiAoaWRlX2Nkcm9tX3NldHVwKGRyaXZlKSkgewoJCXN0cnVjdCBjZHJvbV9kZXZpY2VfaW5mbyAqZGV2aW5mbyA9ICZpbmZvLT5kZXZpbmZvOwoJCWlkZV91bnJlZ2lzdGVyX3N1YmRyaXZlcihkcml2ZSwgJmlkZV9jZHJvbV9kcml2ZXIpOwoJCWtmcmVlKGluZm8tPmJ1ZmZlcik7CgkJa2ZyZWUoaW5mby0+dG9jKTsKCQlrZnJlZShpbmZvLT5jaGFuZ2VyX2luZm8pOwoJCWlmIChkZXZpbmZvLT5oYW5kbGUgPT0gZHJpdmUgJiYgdW5yZWdpc3Rlcl9jZHJvbShkZXZpbmZvKSkKCQkJcHJpbnRrIChLRVJOX0VSUiAiJXM6IGlkZV9jZHJvbV9jbGVhbnVwIGZhaWxlZCB0byB1bnJlZ2lzdGVyIGRldmljZSBmcm9tIHRoZSBjZHJvbSBkcml2ZXIuXG4iLCBkcml2ZS0+bmFtZSk7CgkJa2ZyZWUoaW5mbyk7CgkJZHJpdmUtPmRyaXZlcl9kYXRhID0gTlVMTDsKCQlnb3RvIGZhaWxlZDsKCX0KCgljZHJvbV9yZWFkX3RvYyhkcml2ZSwgJnNlbnNlKTsKCWctPmZvcHMgPSAmaWRlY2Rfb3BzOwoJZy0+ZmxhZ3MgfD0gR0VOSERfRkxfUkVNT1ZBQkxFOwoJYWRkX2Rpc2soZyk7CglyZXR1cm4gMDsKCm91dF9mcmVlX2NkOgoJa2ZyZWUoaW5mbyk7CmZhaWxlZDoKCXJldHVybiAtRU5PREVWOwp9CgpzdGF0aWMgdm9pZCBfX2V4aXQgaWRlX2Nkcm9tX2V4aXQodm9pZCkKewoJZHJpdmVyX3VucmVnaXN0ZXIoJmlkZV9jZHJvbV9kcml2ZXIuZ2VuX2RyaXZlcik7Cn0KCnN0YXRpYyBpbnQgX19pbml0IGlkZV9jZHJvbV9pbml0KHZvaWQpCnsKCXJldHVybiBkcml2ZXJfcmVnaXN0ZXIoJmlkZV9jZHJvbV9kcml2ZXIuZ2VuX2RyaXZlcik7Cn0KCk1PRFVMRV9BTElBUygiaWRlOiptLWNkcm9tKiIpOwptb2R1bGVfaW5pdChpZGVfY2Ryb21faW5pdCk7Cm1vZHVsZV9leGl0KGlkZV9jZHJvbV9leGl0KTsKTU9EVUxFX0xJQ0VOU0UoIkdQTCIpOwo=