LyoKICogKEMpICBDb3B5cmlnaHQgMjAwMQogKiBTdOR1YmxpIEZhdmVyZ2VzIC0gPHd3dy5zdGF1YmxpLmNvbT4KICogUGllcnJlIEFVQkVSVCAgcC5hdWJlcnRAc3RhdWJsaS5jb20KICogVS1Cb290IHBvcnQgb24gUlBYQ2xhc3NpYyBMRiAoQ0xMRl9CVzMxKSBib2FyZAogKgogKiBSUFhDbGFzc2ljIHVzZXMgQW0yOURMMzIzQiBmbGFzaCBtZW1vcnkgd2l0aCAyIGJhbmtzCiAqCiAqCiAqIChDKSBDb3B5cmlnaHQgMjAwMAogKiBXb2xmZ2FuZyBEZW5rLCBERU5YIFNvZnR3YXJlIEVuZ2luZWVyaW5nLCB3ZEBkZW54LmRlLgogKgogKiBTZWUgZmlsZSBDUkVESVRTIGZvciBsaXN0IG9mIHBlb3BsZSB3aG8gY29udHJpYnV0ZWQgdG8gdGhpcwogKiBwcm9qZWN0LgogKgogKiBUaGlzIHByb2dyYW0gaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yCiAqIG1vZGlmeSBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzCiAqIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAyIG9mCiAqIHRoZSBMaWNlbnNlLCBvciAoYXQgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLgogKgogKiBUaGlzIHByb2dyYW0gaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKICogYnV0IFdJVEhPVVQgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YKICogTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiAgU2VlIHRoZQogKiBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgogKgogKiBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQogKiBhbG9uZyB3aXRoIHRoaXMgcHJvZ3JhbTsgaWYgbm90LCB3cml0ZSB0byB0aGUgRnJlZSBTb2Z0d2FyZQogKiBGb3VuZGF0aW9uLCBJbmMuLCA1OSBUZW1wbGUgUGxhY2UsIFN1aXRlIDMzMCwgQm9zdG9uLAogKiBNQSAwMjExMS0xMzA3IFVTQQogKi8KCgojaW5jbHVkZSA8Y29tbW9uLmg+CiNpbmNsdWRlIDxtcGM4eHguaD4KCmZsYXNoX2luZm9fdAlmbGFzaF9pbmZvW0NGR19NQVhfRkxBU0hfQkFOS1NdOyAvKiBpbmZvIGZvciBGTEFTSCBjaGlwcwkqLwoKLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogKiBGdW5jdGlvbnMKICovCnN0YXRpYyB1bG9uZyBmbGFzaF9nZXRfc2l6ZSAodnVfbG9uZyAqYWRkciwgZmxhc2hfaW5mb190ICppbmZvKTsKc3RhdGljIGludCB3cml0ZV93b3JkIChmbGFzaF9pbmZvX3QgKmluZm8sIHVsb25nIGRlc3QsIHVsb25nIGRhdGEpOwpzdGF0aWMgdm9pZCBmbGFzaF9nZXRfb2Zmc2V0cyAodWxvbmcgYmFzZSwgZmxhc2hfaW5mb190ICppbmZvKTsKCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICovCgp1bnNpZ25lZCBsb25nIGZsYXNoX2luaXQgKHZvaWQpCnsKCXVuc2lnbmVkIGxvbmcgc2l6ZV9iMCA7CglpbnQgaTsKCgkvKiBJbml0OiBubyBGTEFTSGVzIGtub3duICovCglmb3IgKGk9MDsgaTxDRkdfTUFYX0ZMQVNIX0JBTktTOyArK2kpIHsKCQlmbGFzaF9pbmZvW2ldLmZsYXNoX2lkID0gRkxBU0hfVU5LTk9XTjsKCX0KCglzaXplX2IwID0gZmxhc2hfZ2V0X3NpemUoKHZ1X2xvbmcgKilDRkdfRkxBU0hfQkFTRSwgJmZsYXNoX2luZm9bMF0pOwoKCiAgICAgICAgZmxhc2hfZ2V0X29mZnNldHMgKENGR19GTEFTSF9CQVNFLCAmZmxhc2hfaW5mb1swXSk7CgojaWYgQ0ZHX01PTklUT1JfQkFTRSA+PSBDRkdfRkxBU0hfQkFTRQoJLyogbW9uaXRvciBwcm90ZWN0aW9uIE9OIGJ5IGRlZmF1bHQgKi8KCWZsYXNoX3Byb3RlY3QoRkxBR19QUk9URUNUX1NFVCwKCQkgICAgICBDRkdfTU9OSVRPUl9CQVNFLAoJCSAgICAgIENGR19NT05JVE9SX0JBU0UrQ0ZHX01PTklUT1JfTEVOLTEsCgkJICAgICAgJmZsYXNoX2luZm9bMF0pOwojZW5kaWYKCglmbGFzaF9pbmZvWzBdLnNpemUgPSBzaXplX2IwOwoKCXJldHVybiAoc2l6ZV9iMCk7Cn0KCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICovCnN0YXRpYyB2b2lkIGZsYXNoX2dldF9vZmZzZXRzICh1bG9uZyBiYXNlLCBmbGFzaF9pbmZvX3QgKmluZm8pCnsKCWludCBpOwoKCWlmIChpbmZvLT5mbGFzaF9pZCAmIEZMQVNIX0JUWVBFKSB7CgkJLyogc2V0IHNlY3RvciBvZmZzZXRzIGZvciBib3R0b20gYm9vdCBibG9jayB0eXBlCSovCgkJaW5mby0+c3RhcnRbMF0gPSBiYXNlICsgMHgwMDAwMDAwMDsKCQlpbmZvLT5zdGFydFsxXSA9IGJhc2UgKyAweDAwMDA4MDAwOwoJCWluZm8tPnN0YXJ0WzJdID0gYmFzZSArIDB4MDAwMTAwMDA7CgkJaW5mby0+c3RhcnRbM10gPSBiYXNlICsgMHgwMDAxODAwMDsKCQlpbmZvLT5zdGFydFs0XSA9IGJhc2UgKyAweDAwMDIwMDAwOwoJCWluZm8tPnN0YXJ0WzVdID0gYmFzZSArIDB4MDAwMjgwMDA7CgkJaW5mby0+c3RhcnRbNl0gPSBiYXNlICsgMHgwMDAzMDAwMDsKCQlpbmZvLT5zdGFydFs3XSA9IGJhc2UgKyAweDAwMDM4MDAwOwoJCWZvciAoaSA9IDg7IGkgPCBpbmZvLT5zZWN0b3JfY291bnQ7IGkrKykgewoJCQlpbmZvLT5zdGFydFtpXSA9IGJhc2UgKyAoKGktNykgKiAweDAwMDQwMDAwKSA7CgkJfQoJfQp9CgovKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAqLwp2b2lkIGZsYXNoX3ByaW50X2luZm8gIChmbGFzaF9pbmZvX3QgKmluZm8pCnsKCWludCBpOwoKCWlmIChpbmZvLT5mbGFzaF9pZCA9PSBGTEFTSF9VTktOT1dOKSB7CgkJcHJpbnRmICgibWlzc2luZyBvciB1bmtub3duIEZMQVNIIHR5cGVcbiIpOwoJCXJldHVybjsKCX0KCglzd2l0Y2ggKGluZm8tPmZsYXNoX2lkICYgRkxBU0hfVkVORE1BU0spIHsKCWNhc2UgRkxBU0hfTUFOX0FNRDoJcHJpbnRmICgiQU1EICIpOwkJYnJlYWs7CglkZWZhdWx0OgkJcHJpbnRmICgiVW5rbm93biBWZW5kb3IgIik7CWJyZWFrOwoJfQoKCXN3aXRjaCAoaW5mby0+Zmxhc2hfaWQgJiBGTEFTSF9UWVBFTUFTSykgewogICAgICAgIGNhc2UgRkxBU0hfQU1ETDMyM0I6CiAgICAgICAgICAgIHByaW50ZiAoIkFNREwzMjNEQiAoMTYgTWJ5dGVzLCBib3R0b20gYm9vdCBzZWN0KVxuIik7CiAgICAgICAgICAgIGJyZWFrOwoJZGVmYXVsdDoKICAgICAgICAgICAgcHJpbnRmICgiVW5rbm93biBDaGlwIFR5cGVcbiIpOwogICAgICAgICAgICBicmVhazsKCX0KCglwcmludGYgKCIgIFNpemU6ICVsZCBNQiBpbiAlZCBTZWN0b3JzXG4iLAoJCWluZm8tPnNpemUgPj4gMjAsIGluZm8tPnNlY3Rvcl9jb3VudCk7CgoJcHJpbnRmICgiICBTZWN0b3IgU3RhcnQgQWRkcmVzc2VzOiIpOwoJZm9yIChpPTA7IGk8aW5mby0+c2VjdG9yX2NvdW50OyArK2kpIHsKCQlpZiAoKGkgJSA1KSA9PSAwKQoJCQlwcmludGYgKCJcbiAgICIpOwoJCXByaW50ZiAoIiAlMDhsWCVzIiwKCQkJaW5mby0+c3RhcnRbaV0sCgkJCWluZm8tPnByb3RlY3RbaV0gPyAiIChSTykiIDogIiAgICAgIgoJCSk7Cgl9CglwcmludGYgKCJcbiIpOwp9CgovKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAqLwoKCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICovCgovKgogKiBUaGUgZm9sbG93aW5nIGNvZGUgY2Fubm90IGJlIHJ1biBmcm9tIEZMQVNIIQogKi8KCnN0YXRpYyB1bG9uZyBmbGFzaF9nZXRfc2l6ZSAodnVfbG9uZyAqYWRkciwgZmxhc2hfaW5mb190ICppbmZvKQp7CglzaG9ydCBpOwoJdWxvbmcgdmFsdWU7Cgl1bG9uZyBiYXNlID0gKHVsb25nKWFkZHI7CgogICAgICAgIC8qIFJlc2V0IGZsYXNoIGNvbXBvbmVueSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICovCiAgICAgICAgYWRkciBbMF0gPSAweGYwZjBmMGYwOwoKICAgICAgICAvKiBXcml0ZSBhdXRvIHNlbGVjdCBjb21tYW5kOiByZWFkIE1hbnVmYWN0dXJlciBJRCAqLwogICAgICAgIGFkZHJbMHhBQUFdID0gMHhBQUFBQUFBQSA7CglhZGRyWzB4NTU1XSA9IDB4NTU1NTU1NTUgOwoJYWRkclsweEFBQV0gPSAweDkwOTA5MDkwIDsKCgl2YWx1ZSA9IGFkZHJbMF0gOwoKCXN3aXRjaCAodmFsdWUgJiAweDAwRkYwMEZGKSB7CgljYXNlIEFNRF9NQU5VRkFDVDoKCQlpbmZvLT5mbGFzaF9pZCA9IEZMQVNIX01BTl9BTUQ7CgkJYnJlYWs7CglkZWZhdWx0OgoJCWluZm8tPmZsYXNoX2lkID0gRkxBU0hfVU5LTk9XTjsKCQlpbmZvLT5zZWN0b3JfY291bnQgPSAwOwoJCWluZm8tPnNpemUgPSAwOwoJCXJldHVybiAoMCk7CQkJLyogbm8gb3IgdW5rbm93biBmbGFzaAkqLwoJfQoKCXZhbHVlID0gYWRkclsyXSA7CQkvKiBkZXZpY2UgSUQJCSovCgoJc3dpdGNoICh2YWx1ZSAmIDB4MDBGRjAwRkYpIHsKICAgICAgICBjYXNlIChBTURfSURfREwzMjNCICYgMHgwMEZGMDBGRik6CiAgICAgICAgICAgIGluZm8tPmZsYXNoX2lkICs9IEZMQVNIX0FNREwzMjNCOwogICAgICAgICAgICBpbmZvLT5zZWN0b3JfY291bnQgPSA3MTsKICAgICAgICAgICAgaW5mby0+c2l6ZSA9IDB4MDEwMDAwMDA7ICAgICAgICAgICAgLyogMTYgTWIgICAgICAgICAgICAgICAgICAgICAqLwoKICAgICAgICAgICAgYnJlYWs7CglkZWZhdWx0OgoJCWluZm8tPmZsYXNoX2lkID0gRkxBU0hfVU5LTk9XTjsKCQlyZXR1cm4gKDApOwkJCS8qID0+IG5vIG9yIHVua25vd24gZmxhc2ggKi8KCgl9CgkvKiBzZXQgdXAgc2VjdG9yIHN0YXJ0IGFkZHJlc3MgdGFibGUgKi8KICAgICAgICAvKiBzZXQgc2VjdG9yIG9mZnNldHMgZm9yIGJvdHRvbSBib290IGJsb2NrIHR5cGUJKi8KICAgICAgICBpbmZvLT5zdGFydFswXSA9IGJhc2UgKyAweDAwMDAwMDAwOwogICAgICAgIGluZm8tPnN0YXJ0WzFdID0gYmFzZSArIDB4MDAwMDgwMDA7CiAgICAgICAgaW5mby0+c3RhcnRbMl0gPSBiYXNlICsgMHgwMDAxMDAwMDsKICAgICAgICBpbmZvLT5zdGFydFszXSA9IGJhc2UgKyAweDAwMDE4MDAwOwogICAgICAgIGluZm8tPnN0YXJ0WzRdID0gYmFzZSArIDB4MDAwMjAwMDA7CiAgICAgICAgaW5mby0+c3RhcnRbNV0gPSBiYXNlICsgMHgwMDAyODAwMDsKICAgICAgICBpbmZvLT5zdGFydFs2XSA9IGJhc2UgKyAweDAwMDMwMDAwOwogICAgICAgIGluZm8tPnN0YXJ0WzddID0gYmFzZSArIDB4MDAwMzgwMDA7CiAgICAgICAgZm9yIChpID0gODsgaSA8IGluZm8tPnNlY3Rvcl9jb3VudDsgaSsrKSB7CiAgICAgICAgICAgIGluZm8tPnN0YXJ0W2ldID0gYmFzZSArICgoaS03KSAqIDB4MDAwNDAwMDApIDsKICAgICAgICB9CgoJLyogY2hlY2sgZm9yIHByb3RlY3RlZCBzZWN0b3JzICovCiAgICAgICAgZm9yIChpID0gMDsgaSA8IDIzOyBpKyspIHsKICAgICAgICAgICAgLyogcmVhZCBzZWN0b3IgcHJvdGVjdGlvbiBhdCBzZWN0b3IgYWRkcmVzcywgKEE3IC4uIEEwKSA9IDB4MDIgKi8KICAgICAgICAgICAgLyogRDAgPSAxIGlmIHByb3RlY3RlZCAqLwogICAgICAgICAgICBhZGRyID0gKHZvbGF0aWxlIHVuc2lnbmVkIGxvbmcgKikoaW5mby0+c3RhcnRbaV0pOwogICAgICAgICAgICBpbmZvLT5wcm90ZWN0W2ldID0gYWRkcls0XSAmIDEgOwoJfQogICAgICAgIC8qIENoZWNrIGZvciBwcm90ZWN0ZWQgc2VjdG9ycyBpbiB0aGUgMm5kIGJhbmsgICAgICAgICAgICAgICAgICAgICAgICovCiAgICAgICAgYWRkclsweDEwMEFBQV0gPSAweEFBQUFBQUFBIDsKICAgICAgICBhZGRyWzB4MTAwNTU1XSA9IDB4NTU1NTU1NTUgOwogICAgICAgIGFkZHJbMHgxMDBBQUFdID0gMHg5MDkwOTA5MCA7CgogICAgICAgIGZvciAoaSA9IDIzOyBpIDwgaW5mby0+c2VjdG9yX2NvdW50OyBpKyspIHsKICAgICAgICAgICAgLyogcmVhZCBzZWN0b3IgcHJvdGVjdGlvbiBhdCBzZWN0b3IgYWRkcmVzcywgKEE3IC4uIEEwKSA9IDB4MDIgKi8KICAgICAgICAgICAgLyogRDAgPSAxIGlmIHByb3RlY3RlZCAqLwogICAgICAgICAgICBhZGRyID0gKHZvbGF0aWxlIHVuc2lnbmVkIGxvbmcgKikoaW5mby0+c3RhcnRbaV0pOwogICAgICAgICAgICBpbmZvLT5wcm90ZWN0W2ldID0gYWRkcls0XSAmIDEgOwoJfQoKCS8qCgkgKiBQcmV2ZW50IHdyaXRlcyB0byB1bmluaXRpYWxpemVkIEZMQVNILgoJICovCglpZiAoaW5mby0+Zmxhc2hfaWQgIT0gRkxBU0hfVU5LTk9XTikgewoJCWFkZHIgPSAodm9sYXRpbGUgdW5zaWduZWQgbG9uZyAqKWluZm8tPnN0YXJ0WzBdOwoKCQkqYWRkciA9IDB4RjBGMEYwRjA7CS8qIHJlc2V0IGJhbmsgMSAgICAgICAgICAgICAgICAgICAgICAqLwoJCWFkZHIgPSAodm9sYXRpbGUgdW5zaWduZWQgbG9uZyAqKWluZm8tPnN0YXJ0WzIzXTsKCgkJKmFkZHIgPSAweEYwRjBGMEYwOwkvKiByZXNldCBiYW5rIDIgICAgICAgICAgICAgICAgICAgICAgKi8KCgl9CgoJcmV0dXJuIChpbmZvLT5zaXplKTsKfQoKCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICovCgppbnQJZmxhc2hfZXJhc2UgKGZsYXNoX2luZm9fdCAqaW5mbywgaW50IHNfZmlyc3QsIGludCBzX2xhc3QpCnsKCXZ1X2xvbmcgKmFkZHIgPSAodnVfbG9uZyopKGluZm8tPnN0YXJ0WzBdKTsKCWludCBmbGFnLCBwcm90LCBzZWN0LCBsX3NlY3Q7Cgl1bG9uZyBzdGFydCwgbm93LCBsYXN0OwoKCWlmICgoc19maXJzdCA8IDApIHx8IChzX2ZpcnN0ID4gc19sYXN0KSkgewoJCWlmIChpbmZvLT5mbGFzaF9pZCA9PSBGTEFTSF9VTktOT1dOKSB7CgkJCXByaW50ZiAoIi0gbWlzc2luZ1xuIik7CgkJfSBlbHNlIHsKCQkJcHJpbnRmICgiLSBubyBzZWN0b3JzIHRvIGVyYXNlXG4iKTsKCQl9CgkJcmV0dXJuIDE7Cgl9CgoJaWYgKChpbmZvLT5mbGFzaF9pZCA9PSBGTEFTSF9VTktOT1dOKSB8fAoJICAgIChpbmZvLT5mbGFzaF9pZCA+IEZMQVNIX0FNRF9DT01QKSkgewoJCXByaW50ZiAoIkNhbid0IGVyYXNlIHVua25vd24gZmxhc2ggdHlwZSAlMDhseCAtIGFib3J0ZWRcbiIsCgkJCWluZm8tPmZsYXNoX2lkKTsKCQlyZXR1cm4gMTsKCX0KCglwcm90ID0gMDsKCWZvciAoc2VjdD1zX2ZpcnN0OyBzZWN0PD1zX2xhc3Q7ICsrc2VjdCkgewoJCWlmIChpbmZvLT5wcm90ZWN0W3NlY3RdKSB7CgkJCXByb3QrKzsKCQl9Cgl9CgoJaWYgKHByb3QpIHsKCQlwcmludGYgKCItIFdhcm5pbmc6ICVkIHByb3RlY3RlZCBzZWN0b3JzIHdpbGwgbm90IGJlIGVyYXNlZCFcbiIsCgkJCXByb3QpOwoJfSBlbHNlIHsKCQlwcmludGYgKCJcbiIpOwoJfQoKCWxfc2VjdCA9IC0xOwoKCS8qIERpc2FibGUgaW50ZXJydXB0cyB3aGljaCBtaWdodCBjYXVzZSBhIHRpbWVvdXQgaGVyZSAqLwoJZmxhZyA9IGRpc2FibGVfaW50ZXJydXB0cygpOwoKCWFkZHJbMHhBQUFdID0gMHhBQUFBQUFBQTsKCWFkZHJbMHg1NTVdID0gMHg1NTU1NTU1NTsKCWFkZHJbMHhBQUFdID0gMHg4MDgwODA4MDsKCWFkZHJbMHhBQUFdID0gMHhBQUFBQUFBQTsKCWFkZHJbMHg1NTVdID0gMHg1NTU1NTU1NTsKCgkvKiBTdGFydCBlcmFzZSBvbiB1bnByb3RlY3RlZCBzZWN0b3JzICovCglmb3IgKHNlY3QgPSBzX2ZpcnN0OyBzZWN0PD1zX2xhc3Q7IHNlY3QrKykgewoJCWlmIChpbmZvLT5wcm90ZWN0W3NlY3RdID09IDApIHsJLyogbm90IHByb3RlY3RlZCAqLwoJCQlhZGRyID0gKHZ1X2xvbmcgKikoaW5mby0+c3RhcnRbc2VjdF0pIDsKCQkJYWRkclswXSA9IDB4MzAzMDMwMzAgOwoJCQlsX3NlY3QgPSBzZWN0OwoJCX0KCX0KCgkvKiByZS1lbmFibGUgaW50ZXJydXB0cyBpZiBuZWNlc3NhcnkgKi8KCWlmIChmbGFnKQoJCWVuYWJsZV9pbnRlcnJ1cHRzKCk7CgoJLyogd2FpdCBhdCBsZWFzdCA4MHVzIC0gbGV0J3Mgd2FpdCAxIG1zICovCgl1ZGVsYXkgKDEwMDApOwoKCS8qCgkgKiBXZSB3YWl0IGZvciB0aGUgbGFzdCB0cmlnZ2VyZWQgc2VjdG9yCgkgKi8KCWlmIChsX3NlY3QgPCAwKQoJCWdvdG8gRE9ORTsKCglzdGFydCA9IGdldF90aW1lciAoMCk7CglsYXN0ICA9IHN0YXJ0OwoJYWRkciA9ICh2dV9sb25nICopKGluZm8tPnN0YXJ0W2xfc2VjdF0pOwoJd2hpbGUgKChhZGRyWzBdICYgMHg4MDgwODA4MCkgIT0gMHg4MDgwODA4MCkgewoJCWlmICgobm93ID0gZ2V0X3RpbWVyKHN0YXJ0KSkgPiBDRkdfRkxBU0hfRVJBU0VfVE9VVCkgewoJCQlwcmludGYgKCJUaW1lb3V0XG4iKTsKCQkJcmV0dXJuIDE7CgkJfQoJCS8qIHNob3cgdGhhdCB3ZSdyZSB3YWl0aW5nICovCgkJaWYgKChub3cgLSBsYXN0KSA+IDEwMDApIHsJLyogZXZlcnkgc2Vjb25kICovCgkJCXB1dGMgKCcuJyk7CgkJCWxhc3QgPSBub3c7CgkJfQoJfQoKRE9ORToKCS8qIHJlc2V0IHRvIHJlYWQgbW9kZSAqLwoJYWRkciA9ICh2dV9sb25nICopaW5mby0+c3RhcnRbMF07CglhZGRyWzBdID0gMHhGMEYwRjBGMDsJLyogcmVzZXQgYmFuayAqLwoKCXByaW50ZiAoIiBkb25lXG4iKTsKICAgICAgICByZXR1cm4gMDsKfQoKLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogKiBDb3B5IG1lbW9yeSB0byBmbGFzaCwgcmV0dXJuczoKICogMCAtIE9LCiAqIDEgLSB3cml0ZSB0aW1lb3V0CiAqIDIgLSBGbGFzaCBub3QgZXJhc2VkCiAqLwoKaW50IHdyaXRlX2J1ZmYgKGZsYXNoX2luZm9fdCAqaW5mbywgdWNoYXIgKnNyYywgdWxvbmcgYWRkciwgdWxvbmcgY250KQp7Cgl1bG9uZyBjcCwgd3AsIGRhdGE7CglpbnQgaSwgbCwgcmM7CgoJd3AgPSAoYWRkciAmIH4zKTsJLyogZ2V0IGxvd2VyIHdvcmQgYWxpZ25lZCBhZGRyZXNzICovCgoJLyoKCSAqIGhhbmRsZSB1bmFsaWduZWQgc3RhcnQgYnl0ZXMKCSAqLwoJaWYgKChsID0gYWRkciAtIHdwKSAhPSAwKSB7CgkJZGF0YSA9IDA7CgkJZm9yIChpPTAsIGNwPXdwOyBpPGw7ICsraSwgKytjcCkgewoJCQlkYXRhID0gKGRhdGEgPDwgOCkgfCAoKih1Y2hhciAqKWNwKTsKCQl9CgkJZm9yICg7IGk8NCAmJiBjbnQ+MDsgKytpKSB7CgkJCWRhdGEgPSAoZGF0YSA8PCA4KSB8ICpzcmMrKzsKCQkJLS1jbnQ7CgkJCSsrY3A7CgkJfQoJCWZvciAoOyBjbnQ9PTAgJiYgaTw0OyArK2ksICsrY3ApIHsKCQkJZGF0YSA9IChkYXRhIDw8IDgpIHwgKCoodWNoYXIgKiljcCk7CgkJfQoKCQlpZiAoKHJjID0gd3JpdGVfd29yZChpbmZvLCB3cCwgZGF0YSkpICE9IDApIHsKCQkJcmV0dXJuIChyYyk7CgkJfQoJCXdwICs9IDQ7Cgl9CgoJLyoKCSAqIGhhbmRsZSB3b3JkIGFsaWduZWQgcGFydAoJICovCgl3aGlsZSAoY250ID49IDQpIHsKCQlkYXRhID0gMDsKCQlmb3IgKGk9MDsgaTw0OyArK2kpIHsKCQkJZGF0YSA9IChkYXRhIDw8IDgpIHwgKnNyYysrOwoJCX0KCQlpZiAoKHJjID0gd3JpdGVfd29yZChpbmZvLCB3cCwgZGF0YSkpICE9IDApIHsKCQkJcmV0dXJuIChyYyk7CgkJfQoJCXdwICArPSA0OwoJCWNudCAtPSA0OwoJfQoKCWlmIChjbnQgPT0gMCkgewoJCXJldHVybiAoMCk7Cgl9CgoJLyoKCSAqIGhhbmRsZSB1bmFsaWduZWQgdGFpbCBieXRlcwoJICovCglkYXRhID0gMDsKCWZvciAoaT0wLCBjcD13cDsgaTw0ICYmIGNudD4wOyArK2ksICsrY3ApIHsKCQlkYXRhID0gKGRhdGEgPDwgOCkgfCAqc3JjKys7CgkJLS1jbnQ7Cgl9Cglmb3IgKDsgaTw0OyArK2ksICsrY3ApIHsKCQlkYXRhID0gKGRhdGEgPDwgOCkgfCAoKih1Y2hhciAqKWNwKTsKCX0KCglyZXR1cm4gKHdyaXRlX3dvcmQoaW5mbywgd3AsIGRhdGEpKTsKfQoKLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogKiBXcml0ZSBhIHdvcmQgdG8gRmxhc2gsIHJldHVybnM6CiAqIDAgLSBPSwogKiAxIC0gd3JpdGUgdGltZW91dAogKiAyIC0gRmxhc2ggbm90IGVyYXNlZAogKi8Kc3RhdGljIGludCB3cml0ZV93b3JkIChmbGFzaF9pbmZvX3QgKmluZm8sIHVsb25nIGRlc3QsIHVsb25nIGRhdGEpCnsKCXZ1X2xvbmcgKmFkZHIgPSAodnVfbG9uZyAqKShpbmZvLT5zdGFydFswXSk7Cgl1bG9uZyBzdGFydDsKCWludCBmbGFnOwoKCS8qIENoZWNrIGlmIEZsYXNoIGlzIChzdWZmaWNpZW50bHkpIGVyYXNlZCAqLwoJaWYgKCgqKCh2dV9sb25nICopZGVzdCkgJiBkYXRhKSAhPSBkYXRhKSB7CgkJcmV0dXJuICgyKTsKCX0KCS8qIERpc2FibGUgaW50ZXJydXB0cyB3aGljaCBtaWdodCBjYXVzZSBhIHRpbWVvdXQgaGVyZSAqLwoJZmxhZyA9IGRpc2FibGVfaW50ZXJydXB0cygpOwoKCWFkZHJbMHhBQUFdID0gMHhBQUFBQUFBQTsKCWFkZHJbMHg1NTVdID0gMHg1NTU1NTU1NTsKCWFkZHJbMHhBQUFdID0gMHhBMEEwQTBBMDsKCgkqKCh2dV9sb25nICopZGVzdCkgPSBkYXRhOwoKCS8qIHJlLWVuYWJsZSBpbnRlcnJ1cHRzIGlmIG5lY2Vzc2FyeSAqLwoJaWYgKGZsYWcpCgkJZW5hYmxlX2ludGVycnVwdHMoKTsKCgkvKiBkYXRhIHBvbGxpbmcgZm9yIEQ3ICovCglzdGFydCA9IGdldF90aW1lciAoMCk7Cgl3aGlsZSAoKCooKHZ1X2xvbmcgKilkZXN0KSAmIDB4ODA4MDgwODApICE9IChkYXRhICYgMHg4MDgwODA4MCkpIHsKCQlpZiAoZ2V0X3RpbWVyKHN0YXJ0KSA+IENGR19GTEFTSF9XUklURV9UT1VUKSB7CgkJCXJldHVybiAoMSk7CgkJfQoJfQoJcmV0dXJuICgwKTsKfQoKLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogKi8K