Forkchoice Ethereum Mainnet

Address Contract Partially Verified

Address 0x779832DD1d56DC9712e3059Ee9b77a4b2704388e
Balance 0 ETH
Nonce 1
Code Size 20085 bytes
Indexed Transactions 0 (1 on-chain, 1.6% indexed)
External Etherscan · Sourcify

Contract Bytecode

20085 bytes
Copy Bytecode
0x60806040526004361061036b5760003560e01c80637d90dcba116101c65780639dc29fac116100f7578063d84d2a4711610095578063f2fde38b1161006f578063f2fde38b14610948578063fbd9574d14610968578063fc3b72b114610996578063fe056342146109ab5761036b565b8063d84d2a47146108f3578063dd62ed3e14610908578063eebc5081146109285761036b565b8063c4d2b1b3116100d1578063c4d2b1b314610873578063cd4fa66d146108a0578063cfb65bb9146108c0578063d65a5021146108d35761036b565b80639dc29fac14610813578063a9059cbb14610833578063b9fe1a8f146108535761036b565b80638da5cb5b1161016457806395d89b411161013e57806395d89b41146107b457806396c7871b146107c9578063995363d3146107de5780639b3a54d1146107f35761036b565b80638da5cb5b146107775780638f6ede1f1461078c5780638fb807c51461079f5761036b565b8063829b38f4116101a0578063829b38f41461070d5780638325a1c01461072d5780638423acd614610742578063894ca308146107625761036b565b80637d90dcba146106b85780637ff9b596146106d857806381a6b250146106ed5761036b565b8063284e2f56116102a0578063612ef80b1161023e578063736ee3d311610218578063736ee3d31461064b5780637866c6c114610660578063797bf3851461068e5780637b7933b4146106a35761036b565b8063612ef80b146105f657806370a082311461060b5780637288b3441461062b5761036b565b8063330691ac1161027a578063330691ac1461058a57806340c10f191461059f57806344a4a003146105bf5780634780eac1146105d45761036b565b8063284e2f56146105245780632ecae90a14610546578063313ce567146105685761036b565b80631c5d1da51161030d57806320f6d07c116102e757806320f6d07c1461049b57806323b872dd146104b057806324d25f4a146104d05780632515aacd146104f05761036b565b80631c5d1da51461045e5780631d0806ae146104715780631f68f20a146104865761036b565b806309ec6b6b1161034957806309ec6b6b146103ff5780630c4925fd14610414578063124168981461042957806318160ddd146104495761036b565b806306b3efd61461037a57806306fdde03146103b0578063095ea7b3146103d2575b34801561037757600080fd5b50005b34801561038657600080fd5b5061039a610395366004613d7f565b6109cb565b6040516103a791906149f8565b60405180910390f35b3480156103bc57600080fd5b506103c5610a0d565b6040516103a79190614ab9565b3480156103de57600080fd5b506103f26103ed366004613e8a565b610a98565b6040516103a791906149ea565b34801561040b57600080fd5b5061039a610b03565b34801561042057600080fd5b5061039a610b16565b34801561043557600080fd5b5061039a610444366004613ed8565b610b1c565b34801561045557600080fd5b5061039a610b40565b61039a61046c36600461419e565b610b46565b34801561047d57600080fd5b5061039a610d59565b34801561049257600080fd5b5061039a610d5f565b3480156104a757600080fd5b5061039a610d65565b3480156104bc57600080fd5b506103f26104cb366004613df5565b610d6b565b3480156104dc57600080fd5b5061039a6104eb366004614020565b610f99565b3480156104fc57600080fd5b5061051061050b366004613ed8565b610fde565b6040516103a7989796959493929190614a42565b34801561053057600080fd5b5061054461053f366004613e42565b61102a565b005b34801561055257600080fd5b5061055b611162565b6040516103a791906149d9565b34801561057457600080fd5b5061057d6111ba565b6040516103a79190614cf1565b34801561059657600080fd5b5061039a6111c3565b3480156105ab57600080fd5b5061039a6105ba366004613e8a565b6111c9565b3480156105cb57600080fd5b5061039a61120d565b3480156105e057600080fd5b506105e9611247565b6040516103a791906148d1565b34801561060257600080fd5b5061039a611256565b34801561061757600080fd5b5061039a610626366004613d7f565b611281565b34801561063757600080fd5b5061039a610646366004613fbe565b61129c565b34801561065757600080fd5b506105e96112ea565b34801561066c57600080fd5b5061068061067b366004613ed8565b6112fe565b6040516103a7929190614996565b34801561069a57600080fd5b506105e9611333565b3480156106af57600080fd5b5061039a611342565b3480156106c457600080fd5b5061039a6106d3366004613f8e565b611348565b3480156106e457600080fd5b5061039a611354565b3480156106f957600080fd5b5061039a610708366004613e8a565b611383565b34801561071957600080fd5b5061039a610728366004613ed8565b6114f0565b34801561073957600080fd5b5061039a6115b3565b34801561074e57600080fd5b5061039a61075d366004614020565b6115c0565b34801561076e57600080fd5b506105e9611825565b34801561078357600080fd5b506105e9611834565b61039a61079a366004613d7f565b611843565b3480156107ab57600080fd5b5061039a6118ac565b3480156107c057600080fd5b506103c56118cd565b3480156107d557600080fd5b506105e9611928565b3480156107ea57600080fd5b506105e9611937565b3480156107ff57600080fd5b5061039a61080e366004613ed8565b61194b565b34801561081f57600080fd5b5061039a61082e366004613e8a565b611969565b34801561083f57600080fd5b506103f261084e366004613e8a565b6119db565b34801561085f57600080fd5b5061039a61086e366004613ed8565b611b72565b34801561087f57600080fd5b5061089361088e366004613ed8565b611b7f565b6040516103a79190614c6a565b3480156108ac57600080fd5b506103f26108bb366004613ef6565b611bfe565b61039a6108ce3660046140d6565b611dfc565b3480156108df57600080fd5b5061039a6108ee366004613ed8565b6120e7565b3480156108ff57600080fd5b5061039a6120f8565b34801561091457600080fd5b5061039a610923366004613dbb565b6120fe565b34801561093457600080fd5b5061039a610943366004613d7f565b612129565b34801561095457600080fd5b50610544610963366004613d7f565b612144565b34801561097457600080fd5b50610988610983366004613d7f565b612167565b6040516103a7929190614c79565b3480156109a257600080fd5b5061039a612183565b3480156109b757600080fd5b5061039a6109c6366004613ed8565b612190565b6000610a05670de0b6b3a76400006109f96109e4611354565b6109ed86611281565b9063ffffffff6121a216565b9063ffffffff6121c716565b90505b919050565b6002805460408051602060018416156101000260001901909316849004601f81018490048402820184019092528181529291830182828015610a905780601f10610a6557610100808354040283529160200191610a90565b820191906000526020600020905b815481529060010190602001808311610a7357829003601f168201915b505050505081565b336000818152601a602090815260408083206001600160a01b038716808552925280832085905551919290917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92590610af19086906149f8565b60405180910390a35060015b92915050565b6000610b106104446118ac565b90505b90565b60135481565b6015546000908015610b3a57610b32818461129c565b915050610a08565b50919050565b601b5490565b6001546000906001600160a01b03163214610b7f57604051600160e51b62461bcd028152600401610b7690614bba565b60405180910390fd5b6001600160a01b03831615801590610ba557506008546001600160a01b03848116911614155b610bc457604051600160e51b62461bcd028152600401610b7690614aea565b8a6001600160a01b038681169085161415610c6e57600654600854604051600160e51b6232ccd50281526001600160a01b03928316926306599aa092610c1492899290911690869060040161496e565b60606040518083038186803b158015610c2c57600080fd5b505afa158015610c40573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250610c649190810190613fdd565b9250610c9e915050565b6008546001600160a01b03878116911614610c9e57604051600160e51b62461bcd028152600401610b7690614b7a565b610d498b60405180608001604052808a6001600160a01b03166001600160a01b03168152602001886001600160a01b03166001600160a01b03168152602001876001600160a01b03166001600160a01b031681526020018a6001600160a01b03166001600160a01b03168152506040518060e0016040528060008152602001858152602001600081526020018e81526020018d81526020018c815260200160008152506001876121da565b9c9b505050505050505050505050565b60185481565b600b5481565b60155481565b6001600160a01b0383166000818152601a6020908152604080832033845282528083205493835260199091528120549091908311801590610dac5750808311155b8015610dc057506001600160a01b03841615155b610ddf57604051600160e51b62461bcd028152600401610b7690614b4a565b6001600160a01b038516600090815260196020526040902054610e08908463ffffffff61235d16565b6001600160a01b038087166000908152601960205260408082209390935590861681522054610e3d908463ffffffff61236f16565b6001600160a01b038516600090815260196020526040902055600019811015610e9557610e70818463ffffffff61235d16565b6001600160a01b0386166000908152601a602090815260408083203384529091529020555b6000610e9f611354565b6001600160a01b03871660009081526019602052604090205490915015610ee0576001600160a01b0386166000908152600960205260409020819055610efa565b6001600160a01b0386166000908152600960205260408120555b6001600160a01b03851660009081526019602052604090205415610f38576001600160a01b0385166000908152600960205260409020819055610f52565b6001600160a01b0385166000908152600960205260408120555b846001600160a01b0316866001600160a01b0316600080516020614e1c83398151915286604051610f8391906149f8565b60405180910390a36001925050505b9392505050565b60008382604051602001610fae9291906148ab565b6040516020818303038152906040528051906020012060001c9350610fd58585858561237c565b95945050505050565b600f60205260009081526040902080546001820154600283015460038401546004850154600586015460068701546007909701549596949593949293919290916001600160a01b031688565b6001546001600160a01b031633146110b5577f7ad06df6a0af6bd602d90db766e0d5f253b45187c3717a0f9026ea8b10ff0d4b547f34b31cff1dbd8374124bd4505521fc29cab0f9554a5386ba7d784a4e611c7e3154336001600160a01b0383161480156110a95750806001600160a01b0316846001600160a01b0316145b6110b257600080fd5b50505b601c80546001600160a01b038481166001600160a01b031983161790925560405191169060009030906110e990859061489f565b6000604051808303816000865af19150503d8060008114611126576040519150601f19603f3d011682016040523d82523d6000602084013e61112b565b606091505b50506040519091503d90816000823e82611143578181fd5b601c80546001600160a01b0319166001600160a01b0386161790558181f35b606060108054806020026020016040519081016040528092919081815260200182805480156111b057602002820191906000526020600020905b81548152602001906001019080831161119c575b5050505050905090565b60045460ff1681565b600c5481565b60006001600054146111f057604051600160e51b62461bcd028152600401610b7690614c1a565b60026000556111ff838361259d565b90505b600160005592915050565b601554600090801561123b576112336112246118ac565b6109f96016546109ed856126cf565b915050610b13565b611233612707565b5090565b6007546001600160a01b031681565b6000806112616118ac565b90506015548111156112435760155461123390829063ffffffff61235d16565b6001600160a01b031660009081526019602052604090205490565b600082158015906112ad5750828210155b15610afd576112e3701d6329f1c35ca4bfabb9f56100000000006109f9600d546109ed6112da888861272c565b6109ed896126cf565b9050610afd565b600a5461010090046001600160a01b031681565b6011818154811061130b57fe5b6000918252602090912060029091020180546001909101546001600160a01b03909116915082565b6008546001600160a01b031681565b60165481565b6000610f92838361275e565b600080426017541461136c576113686127d9565b9150505b61137d611378826128a3565b612903565b91505090565b60006001600054146113aa57604051600160e51b62461bcd028152600401610b7690614c1a565b60026000556007546008546001600160a01b039081169116146113e257604051600160e51b62461bcd028152600401610b7690614b1a565b6113eb82612932565b90508015611202576008546040805180820190915260018152600160fa1b600d026020820152733b5bdccdfa2a0a1911984f203c19628eeb6036e091611440916001600160a01b039091169083908590612aa9565b604051600160e41b630bfcf63b0281526001600160a01b0382169063bfcf63b0906114719087908690600401614996565b602060405180830381600087803b15801561148b57600080fd5b505af115801561149f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506114c39190810190613f70565b82146114e457604051600160e51b62461bcd028152600401610b7690614aca565b50600160005592915050565b60006114fa613a8a565b506000828152600e60209081526040808320548352600f8252918290208251610100810184528154815260018201549281019290925260028101549282018390526003810154606083015260048101546080830152600581015460a0830152600681015460c0830152600701546001600160a01b031660e082015290611584576000915050610a08565b610f926115a368056bc75e2d6310000083608001518460400151612b0c565b6109f983604001516109ed611256565b6000610b1060008061275e565b6000841561181d5783826040516020016115db9291906148ab565b6040516020818303038152906040528051906020012060001c93506115fe613a8a565b506000848152600e60209081526040808320548352600f82528083208151610100810183528154815260018201549381019390935260028101549183018290526003810154606084015260048101546080840152600581015460a0840152600681015460c0840152600701546001600160a01b031660e08301529091906116949068056bc75e2d6310000063ffffffff61236f16565b90506116bd69021e19e0c9bab24000006109f96116b088612b4d565b8a9063ffffffff6121a216565b600854604051600160e01b6370a082310281529198506001600160a01b0316906370a08231906116f19030906004016148d1565b60206040518083038186803b15801561170957600080fd5b505afa15801561171d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506117419190810190613f70565b871161181a5760045460085461181191600a916001600160a01b0361010090920482169163bc6cb1d9918116908916611785576007546001600160a01b0316611787565b885b60065460405163ffffffff851660e01b81526117b59392916001600160a01b0316908f908a90600401614922565b60206040518083038186803b1580156117cd57600080fd5b505afa1580156117e1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506118059190810190613f70565b9063ffffffff61236f16565b9250505061181d565b50505b949350505050565b6005546001600160a01b031681565b6001546001600160a01b031681565b600060016000541461186a57604051600160e51b62461bcd028152600401610b7690614c1a565b60026000556007546008546001600160a01b039081169116146118a257604051600160e51b62461bcd028152600401610b7690614bea565b611202823461259d565b60008042601754146118c4576118c06127d9565b9150505b61137d816128a3565b6003805460408051602060026001851615610100026000190190941693909304601f81018490048402820184019092528181529291830182828015610a905780601f10610a6557610100808354040283529160200191610a90565b6006546001600160a01b031681565b60045461010090046001600160a01b031681565b6010818154811061195857fe5b600091825260209091200154905081565b600060016000541461199057604051600160e51b62461bcd028152600401610b7690614c1a565b600260005561199e82612932565b90508015611202576008546040805180820190915260018152600160f81b6035026020820152611202916001600160a01b03169085908490612aa9565b336000908152601960205260408120548211801590611a0257506001600160a01b03831615155b611a2157604051600160e51b62461bcd028152600401610b7690614b8a565b33600090815260196020526040902054611a41908363ffffffff61235d16565b33600090815260196020526040808220929092556001600160a01b03851681522054611a73908363ffffffff61236f16565b6001600160a01b038416600090815260196020526040812091909155611a97611354565b3360009081526019602052604090205490915015611ac657336000908152600960205260409020819055611ad7565b336000908152600960205260408120555b6001600160a01b03841660009081526019602052604090205415611b15576001600160a01b0384166000908152600960205260409020819055611b2f565b6001600160a01b0384166000908152600960205260408120555b836001600160a01b0316336001600160a01b0316600080516020614e1c83398151915285604051611b6091906149f8565b60405180910390a35060019392505050565b6000610a0582600061275e565b611b87613a8a565b506000908152600f6020908152604091829020825161010081018452815481526001820154928101929092526002810154928201929092526003820154606082015260048201546080820152600582015460a0820152600682015460c08201526007909101546001600160a01b031660e082015290565b60048054600654604051600160e11b6338f5892f0281526000936001600160a01b036101009094048416936371eb125e93611c3c93911691016148d1565b60206040518083038186803b158015611c5457600080fd5b505afa158015611c68573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250611c8c9190810190613d9d565b6001600160a01b0316336001600160a01b031614611cbf57604051600160e51b62461bcd028152600401610b7690614bfa565b611cc7612b99565b611ccf613a8a565b50610120860180516000908152600f602090815260409182902082516101008101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a0840152600681015460c0840152600701546001600160a01b031660e0830152915190911415611def578360155411611d63576000611d76565b601554611d76908563ffffffff61235d16565b60155585516101208801516040516001600160a01b03909216917f85dfc0033a3e5b3b9b3151bd779c1f9b855d66b83ff5bb79283b68d82e8e5b7390611dc1908990899089906149b1565b60405180910390a383611dd8576001915050610fd5565b611de260006128a3565b6016555060019050610fd5565b5060009695505050505050565b6001546000906001600160a01b03163214611e2c57604051600160e51b62461bcd028152600401610b7690614bba565b34158015611e4257506001600160a01b03831615155b8015611e4d57508515155b80611e8a57503415801590611e8057506001600160a01b0383161580611e8057506007546001600160a01b038481169116145b8015611e8a575085155b611ea957604051600160e51b62461bcd028152600401610b7690614c2a565b3415611ec1576007543496506001600160a01b031692505b6040518990611ed6908a9086906020016148ab565b60408051601f1981840301815291815281516020928301206000818152600e909352912054909950915081611f2057604051600160e51b62461bcd028152600401610b7690614b2a565b611f28612b99565b611f30613adb565b611f38613a8a565b506000838152600f60209081526040918290208251610100810184528154815260018201549281019290925260028101549282019290925260038201546060820152600482015460808201819052600583015460a0830152600683015460c08301526007909201546001600160a01b031660e0820152901583611fef57611fc18a8d8d8a61237c565b935083611fe357604051600160e51b62461bcd028152600401610b7690614b5a565b60c08301849052611ff7565b60c083018490525b61200c8461200560006128a3565b8d84612c3c565b60408681019290925291855280516080810182526001600160a01b038c811682528a8116602080840191909152600083850152908c166060830152825160e081018452875181529081018490529296506120af928892810187600260200201518152602001600081526020018e8152602001600081526020018760066007811061209257fe5b602002015181525060405180602001604052806000815250612c8d565b60c0840181905284146120d757604051600160e51b62461bcd028152600401610b7690614c3a565b5050505098975050505050505050565b6000610a05610444836118056118ac565b600d5481565b6001600160a01b039182166000908152601a6020908152604080832093909416825291909152205490565b6001600160a01b031660009081526009602052604090205490565b6001546001600160a01b0316331461215b57600080fd5b61216481612ef4565b50565b6012602052600090815260409020805460019091015460ff1682565b6000610b106015546126cf565b600e6020526000908152604090205481565b6000826121b157506000610afd565b50818102818382816121bf57fe5b0414610afd57fe5b60008183816121d257fe5b049392505050565b602083015160009061220157604051600160e51b62461bcd028152600401610b7690614c5a565b506000858152600e60205260409020548061223157604051600160e51b62461bcd028152600401610b7690614c0a565b612239612b99565b612241613a8a565b506000818152600f60209081526040918290208251610100810184528154815260018201549281019290925260028101549282019290925260038201546060820152600482015460808201819052600583015460a0830152600683015460c08301526007909201546001600160a01b031660e0820152901584156122e4576122d183876001602002015183612f63565b87526020870181905260c0870152612300565b60208601516122fd906122f760006128a3565b8361307f565b86525b60408701516001600160a01b031661231a57600060a08701525b600061232884898988612c8d565b6020880151909150811461235157604051600160e51b62461bcd028152600401610b7690614ada565b50505095945050505050565b60008282111561236957fe5b50900390565b81810182811015610afd57fe5b6000841561181d5761238c613a8a565b506000848152600e60209081526040808320548352600f82528083208151610100810183528154815260018201549381019390935260028101549183018290526003810154606084015260048101546080840152600581015460a0840152600681015460c0840152600701546001600160a01b031660e08301529091906124229068056bc75e2d6310000063ffffffff61236f16565b6004546008549192506001600160a01b0361010090910481169163f3d75a9c9190811690871661245d576007546001600160a01b031661245f565b865b60065460405163ffffffff851660e01b815261248d9392916001600160a01b0316908d908890600401614922565b60206040518083038186803b1580156124a557600080fd5b505afa1580156124b9573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506124dd9190810190613f70565b92506125056124eb86612b4d565b6109f98569021e19e0c9bab240000063ffffffff6121a216565b600854604051600160e01b6370a082310281529194506001600160a01b0316906370a08231906125399030906004016148d1565b60206040518083038186803b15801561255157600080fd5b505afa158015612565573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506125899190810190613f70565b83111561181a575060009695505050505050565b6000816125bf57604051600160e51b62461bcd028152600401610b7690614baa565b6125c7612b99565b60006125d661137860006128a3565b90506125f4816109f985670de0b6b3a764000063ffffffff6121a216565b915034612638576008546040805180820190915260028152600160f31b610627026020820152612633916001600160a01b031690339030908790613257565b6126a2565b600760009054906101000a90046001600160a01b03166001600160a01b031663d0e30db0846040518263ffffffff1660e01b81526004016000604051808303818588803b15801561268857600080fd5b505af115801561269c573d6000803e3d6000fd5b50505050505b6126ae84838584613285565b6001600160a01b039093166000908152600960205260409020929092555090565b60008115610a085760006126e16127d9565b509050610b3261016d6109ed856109f98568056bc75e2d6310000063ffffffff6121a216565b7f3d82e958c891799f357c1316ae5543412952ae5c423336f8929ed7458039c9955490565b6000821580159061273c57508115155b15610afd576112e3826109f98568056bc75e2d6310000063ffffffff6121a216565b60008083156127c657426017541461277c576127786127d9565b9150505b600854604051600160e01b6370a082310281526000916127b69184916001600160a01b0316906370a08231906117b59030906004016148d1565b9050808511156127c4578094505b505b61181d846127d3836128a3565b8561307f565b60048054600654600854604051600160e31b63015216af028152600094859461010090046001600160a01b0390811694630a90b578946128239430949284169390911691016148fa565b60806040518083038186803b15801561283b57600080fd5b505afa15801561284f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506128739190810190614081565b600d54919550935061289d925068056bc75e2d6310000091506109f990849063ffffffff6121a216565b90509091565b6000601b54600014610a0857601354806128f357601554600854604051600160e01b6370a082310281526128f092916001600160a01b0316906370a08231906117b59030906004016148d1565b90505b610b32818463ffffffff61236f16565b601b546000908061291657601854610f92565b610f92816109f985670de0b6b3a764000063ffffffff6121a216565b60008161295457604051600160e51b62461bcd028152600401610b7690614bca565b61295d33611281565b8211156129705761296d33611281565b91505b612978612b99565b600061298761137860006128a3565b905060006129a7670de0b6b3a76400006109f9868563ffffffff6121a216565b600854604051600160e01b6370a082310281529192506000916001600160a01b03909116906370a08231906129e09030906004016148d1565b60206040518083038186803b1580156129f857600080fd5b505afa158015612a0c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250612a309190810190613f70565b905081935080841115612a5857604051600160e51b62461bcd028152600401610b7690614b3a565b612a6433868686613386565b3360009081526019602052604090205415612a9057336000908152600960205260409020839055612aa1565b336000908152600960205260408120555b505050919050565b604051612b06908590600160e01b63a9059cbb0290612ace9087908790602401614996565b60408051601f198184030181529190526020810180516001600160e01b03166001600160e01b031990931692909217909152836134e1565b50505050565b600082612b225768056bc75e2d6310000061181d565b61181d68056bc75e2d63100000611805846109f9876109ed6301e13380838c8863ffffffff6121a216565b6000610a0569021e19e0c9bab24000006118056204cfe06109f9866109ed600b5461180568056bc75e2d631000006109f96804563918244f400000600c546121a290919063ffffffff16565b4260175414612c3a5760048054600654600854604051600160e01b63327ab6390281526001600160a01b0361010090940484169463327ab63994612be2948116931691016148df565b602060405180830381600087803b158015612bfc57600080fd5b505af1158015612c10573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250612c349190810190613f70565b50426017555b565b6000806000612c4d878787876135ad565b9250612c6c9050612c64888463ffffffff61236f16565b8787876135ad565b9093509150612c81878363ffffffff61236f16565b90509450945094915050565b6000612c976135ec565b600854604051600160e01b6370a082310281526001600160a01b03909116906370a0823190612cca9030906004016148d1565b60206040518083038186803b158015612ce257600080fd5b505afa158015612cf6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250612d1a9190810190613f70565b602084015111801590612d36575083516001600160a01b031615155b612d5557604051600160e51b62461bcd028152600401610b7690614b6a565b60608401516001600160a01b0316612d785783516001600160a01b031660608501525b612d82848461366f565b60208301516060840151612d959161236f565b606084015260003415612db05750303134811115612db05750345b60048054604051600160e01b63b1eac3ad0281526101009091046001600160a01b03169163b1eac3ad918491612dee918b918b918b918b9101614a06565b6020604051808303818588803b158015612e0757600080fd5b505af1158015612e1b573d6000803e3d6000fd5b50505050506040513d601f19601f82011682018060405250612e409190810190613f70565b60208501819052612e6657604051600160e51b62461bcd028152600401610b7690614b9a565b6020840151601554612e779161236f565b601555612e8460006128a3565b60165584516020858101518651918801516040808a015190516001600160a01b03958616957f86e15dd78cd784ab7788bcf5b96b9395e86030e048e5faedcfe752c700f6157e95612edf959490939092909182161590614c94565b60405180910390a25050506020015192915050565b6001600160a01b038116612f0757600080fd5b6001546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600180546001600160a01b0319166001600160a01b0392909216919091179055565b600080612f6e613a8a565b506000858152600f60209081526040918290208251610100810184528154815260018201549281019290925260028101549282018390526003810154606083015260048101546080830152600581015460a0830152600681015460c0830152600701546001600160a01b031660e082015290612fff57604051600160e51b62461bcd028152600401610b7690614c4a565b604081015161303390613025906109f98868056bc75e2d6310000063ffffffff6121a216565b61302d6118ac565b8661307f565b915061307481604001516109f96130538585608001518660400151612b0c565b6109f989701d6329f1c35ca4bfabb9f561000000000063ffffffff6121a216565b925050935093915050565b6000806130a061309a8660155461236f90919063ffffffff16565b8561272c565b90506000806000808615613117576804563918244f4000008510156130cc576804563918244f40000094505b50507f185a40c6b6d3f849f72c71ea950323d21149c27a9d90f7dc5e5ea2d332edcf7f547f9ff54bc0049f5eab56ca7cd14591be3f7ed6355b856d01e3770305c74a004ea254613164565b6802b5e3af16b188000085101561315b57613130612707565b91507f2b4858b1bc9e2d14afab03340ce5f6c81b703c86a0c570653ae586534e095fb1549050613164565b5050600b54600c545b6804e1003b28d92800008511156131f55761318e856804e1003b28d928000063ffffffff61235d16565b9450678ac7230489e800008511156131ac57678ac7230489e8000094505b6131c660646109f9605a6109ed858763ffffffff61236f16565b92506131ee83611805678ac7230489e800006109f96116b068056bc75e2d631000008961235d565b955061324b565b6132168261180568056bc75e2d631000006109f9898663ffffffff6121a216565b9550909250829061322d818363ffffffff61236f16565b92508386101561323f5783955061324b565b8286111561324b578295505b50505050509392505050565b60405161327e908690600160e01b6323b872dd0290612ace9088908890889060240161496e565b5050505050565b6001600160a01b0384166132ae57604051600160e51b62461bcd028152600401610b7690614afa565b601b546132c1908463ffffffff61236f16565b601b556001600160a01b0384166000908152601960205260409020546132ed908463ffffffff61236f16565b6001600160a01b038516600081815260196020526040908190209290925590517fb4c03061fb5b7fed76389d5af8f2e0ddb09f8c70d1333abbb62582835e10accb9061333e90869086908690614cd6565b60405180910390a2836001600160a01b031660006001600160a01b0316600080516020614e1c8339815191528560405161337891906149f8565b60405180910390a350505050565b6001600160a01b0384166000908152601960205260409020548311156133c157604051600160e51b62461bcd028152600401610b7690614b0a565b6001600160a01b0384166000908152601960205260409020546133ea908463ffffffff61235d16565b6001600160a01b0385166000908152601960205260409020819055600a10613452576001600160a01b03841660009081526019602052604090205461343690849063ffffffff61236f16565b6001600160a01b03851660009081526019602052604081205592505b601b54613465908463ffffffff61235d16565b601b556040516001600160a01b038516907f743033787f4738ff4d6a7225ce2bd0977ee5f86b91a902a58f5e4d0b297b4644906134a790869086908690614cd6565b60405180910390a260006001600160a01b0316846001600160a01b0316600080516020614e1c8339815191528560405161337891906149f8565b60006060846001600160a01b0316846040516134fd919061489f565b6000604051808303816000865af19150503d806000811461353a576040519150601f19603f3d011682016040523d82523d6000602084013e61353f565b606091505b509150915081839061356757604051600160e51b62461bcd028152600401610b769190614ab9565b5080511561327e57808060200190516135839190810190613eba565b83906135a557604051600160e51b62461bcd028152600401610b769190614ab9565b505050505050565b6000806135bb86868561307f565b91506135e16b0a3098c68eb9427db80000006109f9866109ed8a8763ffffffff6121a216565b905094509492505050565b600080356001600160e01b0319167fd46a704bc285dbd6ff5ad3863506260b1df02812f4f857c8cc852317a6ac64f260405160200161362c929190614879565b604051602081830303815290604052805190602001209050600081549050801561366b57604051600160e51b62461bcd028152600401610b7690614bba565b5050565b60208083015160408401516060808601519385015190850151608086015160a087015160c08801519596949560006001600160a01b038816613825576007546008546001600160a01b039081169116141561379257600854604080516020810190915260008152733b5bdccdfa2a0a1911984f203c19628eeb6036e091613705916001600160a01b039091169083908690612aa9565b604051600160e41b630bfcf63b0281526001600160a01b0382169063bfcf63b090613736908b908790600401614996565b602060405180830381600087803b15801561375057600080fd5b505af1158015613764573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052506137889190810190613f70565b83149150506137bf565b6008546040805160208101909152600081526137bb916001600160a01b03169089908590612aa9565b5060015b8080156137cb57508186115b1561380057600854600554604080516020810190915260008152613800926001600160a01b03908116921690858a0390612aa9565b8061382057604051600160e51b62461bcd028152600401610b7690614bda565b613860565b6008546005546040805180820190915260028152600160f11b61191b026020820152613860926001600160a01b039081169216908990612aa9565b83156139ce576007546001600160a01b038a8116911614801561388257503415155b801561388d57503484145b1561393857600760009054906101000a90046001600160a01b03166001600160a01b031663d0e30db0856040518263ffffffff1660e01b81526004016000604051808303818588803b1580156138e257600080fd5b505af11580156138f6573d6000803e3d6000fd5b50506005546040805180820190915260028152600160f01b61323702602082015261393394508d93506001600160a01b0390911691508790612aa9565b6139ce565b6008546001600160a01b038a8116911614156139655761395e858563ffffffff61236f16565b94506139ce565b876001600160a01b0316896001600160a01b031614156139965761398f838563ffffffff61236f16565b92506139ce565b6005546040805180820190915260028152600160f01b6132370260208201526139ce918b9133916001600160a01b0316908890613257565b8415613a3f576008546001600160a01b0389811691161415613a01576139fa838663ffffffff61236f16565b9250613a3f565b6008546005546040805180820190915260028152600160f01b613331026020820152613a3f926001600160a01b039081169233929116908990613257565b8215613a7d576005546040805180820190915260028152600160f11b611999026020820152613a7d918a9133916001600160a01b0316908790613257565b5050505050505050505050565b6040518061010001604052806000801916815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160006001600160a01b031681525090565b6040518060e001604052806007906020820280388339509192915050565b8035610afd81614df5565b8051610afd81614df5565b8035610afd81614e09565b8051610afd81614e09565b8035610afd81614e12565b600082601f830112613b4157600080fd5b8135613b54613b4f82614d26565b614cff565b91508082526020830160208301858383011115613b7057600080fd5b613b7b838284614d9c565b50505092915050565b60006101408284031215613b9757600080fd5b613ba2610140614cff565b90506000613bb08484613af9565b8252506020613bc184848301613af9565b6020830152506040613bd584828501613af9565b6040830152506060613be984828501613af9565b6060830152506080613bfd84828501613b25565b60808301525060a0613c1184828501613b25565b60a08301525060c0613c2584828501613b25565b60c08301525060e0613c3984828501613b25565b60e083015250610100613c4e84828501613b25565b61010083015250610120613c6484828501613b25565b6101208301525092915050565b60006101608284031215613c8457600080fd5b613c8f610160614cff565b90506000613c9d8484613af9565b8252506020613cae84848301613af9565b6020830152506040613cc284828501613af9565b6040830152506060613cd684828501613b25565b6060830152506080613cea84828501613b25565b60808301525060a0613cfe84828501613b25565b60a08301525060c0613d1284828501613b25565b60c08301525060e0613d2684828501613b25565b60e083015250610100613d3b84828501613b25565b61010083015250610120613d5184828501613b0f565b61012083015250610140613d6784828501613b25565b6101408301525092915050565b8051610afd81614e12565b600060208284031215613d9157600080fd5b600061181d8484613af9565b600060208284031215613daf57600080fd5b600061181d8484613b04565b60008060408385031215613dce57600080fd5b6000613dda8585613af9565b9250506020613deb85828601613af9565b9150509250929050565b600080600060608486031215613e0a57600080fd5b6000613e168686613af9565b9350506020613e2786828701613af9565b9250506040613e3886828701613b25565b9150509250925092565b60008060408385031215613e5557600080fd5b6000613e618585613af9565b925050602083013567ffffffffffffffff811115613e7e57600080fd5b613deb85828601613b30565b60008060408385031215613e9d57600080fd5b6000613ea98585613af9565b9250506020613deb85828601613b25565b600060208284031215613ecc57600080fd5b600061181d8484613b1a565b600060208284031215613eea57600080fd5b600061181d8484613b25565b60008060008060006103008688031215613f0f57600080fd5b6000613f1b8888613b84565b955050610140613f2d88828901613c71565b9450506102a0613f3f88828901613af9565b9350506102c0613f5188828901613b25565b9250506102e0613f6388828901613b0f565b9150509295509295909350565b600060208284031215613f8257600080fd5b600061181d8484613d74565b60008060408385031215613fa157600080fd5b6000613fad8585613b25565b9250506020613deb85828601613b0f565b60008060408385031215613fd157600080fd5b6000613ea98585613b25565b600080600060608486031215613ff257600080fd5b6000613ffe8686613d74565b935050602061400f86828701613d74565b9250506040613e3886828701613d74565b6000806000806080858703121561403657600080fd5b60006140428787613b25565b945050602061405387828801613b25565b935050604061406487828801613b25565b925050606061407587828801613af9565b91505092959194509250565b6000806000806080858703121561409757600080fd5b60006140a38787613d74565b94505060206140b487828801613d74565b93505060406140c587828801613d74565b925050606061407587828801613d74565b600080600080600080600080610100898b0312156140f357600080fd5b60006140ff8b8b613b25565b98505060206141108b828c01613b25565b97505060406141218b828c01613b25565b96505060606141328b828c01613b25565b95505060806141438b828c01613af9565b94505060a06141548b828c01613af9565b93505060c06141658b828c01613af9565b92505060e089013567ffffffffffffffff81111561418257600080fd5b61418e8b828c01613b30565b9150509295985092959890939650565b6000806000806000806000806000806101408b8d0312156141be57600080fd5b60006141ca8d8d613b25565b9a505060206141db8d828e01613b25565b99505060406141ec8d828e01613b25565b98505060606141fd8d828e01613b25565b975050608061420e8d828e01613b25565b96505060a061421f8d828e01613af9565b95505060c06142308d828e01613af9565b94505060e06142418d828e01613af9565b9350506101006142538d828e01613af9565b9250506101208b013567ffffffffffffffff81111561427157600080fd5b61427d8d828e01613b30565b9150509295989b9194979a5092959850565b600061429b83836142af565b505060200190565b600061429b83836143c9565b6142b881614d6d565b82525050565b6142b86142ca82614d6d565b614dd4565b6142d881614d54565b6142e28184610a08565b92506142ed82610b13565b8060005b838110156135a5578151614305878261428f565b965061431083614d4e565b9250506001016142f1565b61432481614d5a565b61432e8184610a08565b925061433982610b13565b8060005b838110156135a557815161435187826142a3565b965061435c83614d4e565b92505060010161433d565b600061437282614d60565b61437c8185614d64565b935061438783614d4e565b8060005b838110156143b557815161439f88826142a3565b97506143aa83614d4e565b92505060010161438b565b509495945050505050565b6142b881614d78565b6142b881610b13565b6142b86143de82614d7d565b610b13565b60006143ee82614d60565b6143f88185614d64565b9350614408818560208601614da8565b61441181614de5565b9093019392505050565b600061442682614d60565b6144308185610a08565b9350614440818560208601614da8565b9290920192915050565b6000614457600183614d64565b600160fa1b600d02815260200192915050565b6000614477600283614d64565b600160f01b61323302815260200192915050565b6000614498600283614d64565b600160f41b61031302815260200192915050565b60006144b9600283614d64565b600160f01b61313502815260200192915050565b60006144da600283614d64565b600160f11b61189b02815260200192915050565b60006144fb600183614d64565b600160f81b603302815260200192915050565b600061451b600183614d64565b600160f81b603702815260200192915050565b600061453b600283614d64565b600160f01b61333702815260200192915050565b600061455c600283614d64565b600160f21b610c4d02815260200192915050565b600061457d600283614d64565b600160f01b61333502815260200192915050565b600061459e600283614d64565b600160f21b610c8d02815260200192915050565b60006145bf600283614d64565b600160f01b61313102815260200192915050565b60006145e0600283614d64565b600160f01b61313302815260200192915050565b6000614601600283614d64565b600160f01b61323502815260200192915050565b6000614622600283614d64565b600160f01b61313702815260200192915050565b6000614643600c83614d64565b7f756e617574686f72697a65640000000000000000000000000000000000000000815260200192915050565b600061467c600283614d64565b600160f01b61313902815260200192915050565b600061469d600283614d64565b600160f11b61191b02815260200192915050565b60006146be600183614d64565b600160f91b601902815260200192915050565b60006146de600183614d64565b600160f81b603102815260200192915050565b60006146fe600283614d64565b600160f11b61191902815260200192915050565b600061471f600c83614d64565b7f6e6f6e5265656e7472616e740000000000000000000000000000000000000000815260200192915050565b6000614758600183614d64565b600160f91b601b02815260200192915050565b6000614778600183614d64565b600160fb1b600702815260200192915050565b6000614798600283614d64565b600160f01b61333302815260200192915050565b60006147b9600283614d64565b600160f01b61323102815260200192915050565b80516101008301906147df84826143c9565b5060208201516147f260208501826143c9565b50604082015161480560408501826143c9565b50606082015161481860608501826143c9565b50608082015161482b60808501826143c9565b5060a082015161483e60a08501826143c9565b5060c082015161485160c08501826143c9565b5060e0820151612b0660e08501826142af565b6142b86143de82610b13565b6142b881614d96565b600061488582856143d2565b6004820191506148958284614864565b5060200192915050565b6000610f92828461441b565b60006148b78285614864565b6020820191506148c782846142be565b5060140192915050565b60208101610afd82846142af565b604081016148ed82856142af565b610f9260208301846142af565b6060810161490882866142af565b61491560208301856142af565b61181d60408301846142af565b60a0810161493082886142af565b61493d60208301876142af565b61494a60408301866142af565b61495760608301856143c9565b61496460808301846143c9565b9695505050505050565b6060810161497c82866142af565b61498960208301856142af565b61181d60408301846143c9565b604081016149a482856142af565b610f9260208301846143c9565b606081016149bf82866142af565b6149cc60208301856143c9565b61181d60408301846143c0565b60208082528101610f928184614367565b60208101610afd82846143c0565b60208101610afd82846143c9565b6101a08101614a1582876143c9565b614a2260208301866142cf565b614a2f60a083018561431b565b81810361018083015261496481846143e3565b6101008101614a51828b6143c9565b614a5e602083018a6143c9565b614a6b60408301896143c9565b614a7860608301886143c9565b614a8560808301876143c9565b614a9260a08301866143c9565b614a9f60c08301856143c9565b614aac60e08301846142af565b9998505050505050505050565b60208082528101610f9281846143e3565b60208082528101610a058161444a565b60208082528101610a058161446a565b60208082528101610a058161448b565b60208082528101610a05816144ac565b60208082528101610a05816144cd565b60208082528101610a05816144ee565b60208082528101610a058161450e565b60208082528101610a058161452e565b60208082528101610a058161454f565b60208082528101610a0581614570565b60208082528101610a0581614591565b60208082528101610a05816145b2565b60208082528101610a05816145d3565b60208082528101610a05816145f4565b60208082528101610a0581614615565b60208082528101610a0581614636565b60208082528101610a058161466f565b60208082528101610a0581614690565b60208082528101610a05816146b1565b60208082528101610a05816146d1565b60208082528101610a05816146f1565b60208082528101610a0581614712565b60208082528101610a058161474b565b60208082528101610a058161476b565b60208082528101610a058161478b565b60208082528101610a05816147ac565b6101008101610afd82846147cd565b60408101614c8782856143c9565b610f9260208301846143c0565b60a08101614ca282886143c9565b614caf60208301876143c9565b614cbc60408301866142af565b614cc960608301856142af565b61496460808301846143c0565b60608101614ce482866143c9565b61498960208301856143c9565b60208101610afd8284614870565b60405181810167ffffffffffffffff81118282101715614d1e57600080fd5b604052919050565b600067ffffffffffffffff821115614d3d57600080fd5b506020601f91909101601f19160190565b60200190565b50600490565b50600790565b5190565b90815260200190565b6000610a0582614d8a565b151590565b6001600160e01b03191690565b6001600160a01b031690565b60ff1690565b82818337506000910152565b60005b83811015614dc3578181015183820152602001614dab565b83811115612b065750506000910152565b6000610a05826000610a0582614def565b601f01601f191690565b60601b90565b614dfe81614d6d565b811461216457600080fd5b614dfe81614d78565b614dfe81610b1356feddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa265627a7a72305820de1eacc7c8331fe8cdf0c979ae16fb9fceba5da8fae74469c84dd040bfa3c3eb6c6578706572696d656e74616cf50037

Verified Source Code Partial Match

Compiler: v0.5.8+commit.23d335f2 EVM: petersburg Optimization: Yes (200 runs)
LoanTokenLogicV4.sol 1833 lines
/**
 * Copyright 2017-2020, bZeroX, LLC. All Rights Reserved.
 * Licensed under the Apache License, Version 2.0.
 */

pragma solidity 0.5.8;
pragma experimental ABIEncoderV2;


/**
 * @title ERC20Basic
 * @dev Simpler version of ERC20 interface
 * See https://github.com/ethereum/EIPs/issues/179
 */
contract ERC20Basic {
  function totalSupply() public view returns (uint256);
  function balanceOf(address _who) public view returns (uint256);
  function transfer(address _to, uint256 _value) public returns (bool);
  event Transfer(address indexed from, address indexed to, uint256 value);
}

/**
 * @title ERC20 interface
 * @dev see https://github.com/ethereum/EIPs/issues/20
 */
contract ERC20 is ERC20Basic {
  function allowance(address _owner, address _spender)
    public view returns (uint256);

  function transferFrom(address _from, address _to, uint256 _value)
    public returns (bool);

  function approve(address _spender, uint256 _value) public returns (bool);
  event Approval(
    address indexed owner,
    address indexed spender,
    uint256 value
  );
}

/**
 * @title EIP20/ERC20 interface
 * @dev see https://github.com/ethereum/EIPs/issues/20
 */
contract EIP20 is ERC20 {
    string public name;
    uint8 public decimals;
    string public symbol;
}

contract WETHInterface is EIP20 {
    function deposit() external payable;
    function withdraw(uint256 wad) external;
}

/**
 * @title SafeMath
 * @dev Math operations with safety checks that throw on error
 */
library SafeMath {

  /**
  * @dev Multiplies two numbers, throws on overflow.
  */
  function mul(uint256 _a, uint256 _b) internal pure returns (uint256 c) {
    // Gas optimization: this is cheaper than asserting 'a' not being zero, but the
    // benefit is lost if 'b' is also tested.
    // See: https://github.com/OpenZeppelin/openzeppelin-solidity/pull/522
    if (_a == 0) {
      return 0;
    }

    c = _a * _b;
    assert(c / _a == _b);
    return c;
  }

  /**
  * @dev Integer division of two numbers, truncating the quotient.
  */
  function div(uint256 _a, uint256 _b) internal pure returns (uint256) {
    // assert(_b > 0); // Solidity automatically throws when dividing by 0
    // uint256 c = _a / _b;
    // assert(_a == _b * c + _a % _b); // There is no case in which this doesn't hold
    return _a / _b;
  }

  /**
  * @dev Integer division of two numbers, rounding up and truncating the quotient
  */
  function divCeil(uint256 _a, uint256 _b) internal pure returns (uint256) {
    if (_a == 0) {
      return 0;
    }

    return ((_a - 1) / _b) + 1;
  }

  /**
  * @dev Subtracts two numbers, throws on overflow (i.e. if subtrahend is greater than minuend).
  */
  function sub(uint256 _a, uint256 _b) internal pure returns (uint256) {
    assert(_b <= _a);
    return _a - _b;
  }

  /**
  * @dev Adds two numbers, throws on overflow.
  */
  function add(uint256 _a, uint256 _b) internal pure returns (uint256 c) {
    c = _a + _b;
    assert(c >= _a);
    return c;
  }
}

/**
 * @title Ownable
 * @dev The Ownable contract has an owner address, and provides basic authorization control
 * functions, this simplifies the implementation of "user permissions".
 */
contract Ownable {
  address public owner;


  event OwnershipTransferred(
    address indexed previousOwner,
    address indexed newOwner
  );


  /**
   * @dev The Ownable constructor sets the original `owner` of the contract to the sender
   * account.
   */
  constructor() public {
    owner = msg.sender;
  }

  /**
   * @dev Throws if called by any account other than the owner.
   */
  modifier onlyOwner() {
    require(msg.sender == owner);
    _;
  }

  /**
   * @dev Allows the current owner to transfer control of the contract to a newOwner.
   * @param _newOwner The address to transfer ownership to.
   */
  function transferOwnership(address _newOwner) public onlyOwner {
    _transferOwnership(_newOwner);
  }

  /**
   * @dev Transfers control of the contract to a newOwner.
   * @param _newOwner The address to transfer ownership to.
   */
  function _transferOwnership(address _newOwner) internal {
    require(_newOwner != address(0));
    emit OwnershipTransferred(owner, _newOwner);
    owner = _newOwner;
  }
}

/**
 * @title Helps contracts guard against reentrancy attacks.
 * @author Remco Bloemen <remco@2π.com>, Eenae <[email protected]>
 * @dev If you mark a function `nonReentrant`, you should also
 * mark it `external`.
 */
contract ReentrancyGuard {

  /// @dev Constant for unlocked guard state - non-zero to prevent extra gas costs.
  /// See: https://github.com/OpenZeppelin/openzeppelin-solidity/issues/1056
  uint256 internal constant REENTRANCY_GUARD_FREE = 1;

  /// @dev Constant for locked guard state
  uint256 internal constant REENTRANCY_GUARD_LOCKED = 2;

  /**
   * @dev We use a single lock for the whole contract.
   */
  uint256 internal reentrancyLock = REENTRANCY_GUARD_FREE;

  /**
   * @dev Prevents a contract from calling itself, directly or indirectly.
   * If you mark a function `nonReentrant`, you should also
   * mark it `external`. Calling one `nonReentrant` function from
   * another is not supported. Instead, you can implement a
   * `private` function doing the actual work, and an `external`
   * wrapper marked as `nonReentrant`.
   */
  modifier nonReentrant() {
    require(reentrancyLock == REENTRANCY_GUARD_FREE, "nonReentrant");
    reentrancyLock = REENTRANCY_GUARD_LOCKED;
    _;
    reentrancyLock = REENTRANCY_GUARD_FREE;
  }

}

contract LoanTokenization is ReentrancyGuard, Ownable {

    uint256 internal constant MAX_UINT = 2**256 - 1;

    string public name;
    string public symbol;
    uint8 public decimals;

    address public bZxContract;
    address public bZxVault;
    address public bZxOracle;
    address public wethContract;

    address public loanTokenAddress;

    // price of token at last user checkpoint
    mapping (address => uint256) internal checkpointPrices_;
}

contract LoanTokenStorage is LoanTokenization {

    struct ListIndex {
        uint256 index;
        bool isSet;
    }

    struct LoanData {
        bytes32 loanOrderHash;
        uint256 leverageAmount;
        uint256 initialMarginAmount;
        uint256 maintenanceMarginAmount;
        uint256 maxDurationUnixTimestampSec;
        uint256 index;
        uint256 marginPremiumAmount;
        address collateralTokenAddress;
    }

    struct TokenReserves {
        address lender;
        uint256 amount;
    }

    event Borrow(
        address indexed borrower,
        uint256 borrowAmount,
        uint256 interestRate,
        address collateralTokenAddress,
        address tradeTokenToFillAddress,
        bool withdrawOnOpen
    );

    event Repay(
        bytes32 indexed loanOrderHash,
        address indexed borrower,
        address closer,
        uint256 amount,
        bool isLiquidation
    );

    event Claim(
        address indexed claimant,
        uint256 tokenAmount,
        uint256 assetAmount,
        uint256 remainingTokenAmount,
        uint256 price
    );

    bool internal isInitialized_ = false;

    address public tokenizedRegistry;

    uint256 public baseRate = 1000000000000000000; // 1.0%
    uint256 public rateMultiplier = 18750000000000000000; // 18.75%

    // slot addition (non-sequential): lowUtilBaseRate = 8000000000000000000; // 8.0%
    // slot addition (non-sequential): lowUtilRateMultiplier = 4750000000000000000; // 4.75%

    // "fee percentage retained by the oracle" = SafeMath.sub(10**20, spreadMultiplier);
    uint256 public spreadMultiplier;

    mapping (uint256 => bytes32) public loanOrderHashes; // mapping of levergeAmount to loanOrderHash
    mapping (bytes32 => LoanData) public loanOrderData; // mapping of loanOrderHash to LoanOrder
    uint256[] public leverageList;

    TokenReserves[] public burntTokenReserveList; // array of TokenReserves
    mapping (address => ListIndex) public burntTokenReserveListIndex; // mapping of lender address to ListIndex objects
    uint256 public burntTokenReserved; // total outstanding burnt token amount
    address internal nextOwedLender_;

    uint256 public totalAssetBorrow; // current amount of loan token amount tied up in loans

    uint256 public checkpointSupply;

    uint256 internal lastSettleTime_;

    uint256 public initialPrice;
}

contract AdvancedTokenStorage is LoanTokenStorage {
    using SafeMath for uint256;

    event Transfer(
        address indexed from,
        address indexed to,
        uint256 value
    );
    event Approval(
        address indexed owner,
        address indexed spender,
        uint256 value
    );
    event Mint(
        address indexed minter,
        uint256 tokenAmount,
        uint256 assetAmount,
        uint256 price
    );
    event Burn(
        address indexed burner,
        uint256 tokenAmount,
        uint256 assetAmount,
        uint256 price
    );

    mapping(address => uint256) internal balances;
    mapping (address => mapping (address => uint256)) internal allowed;
    uint256 internal totalSupply_;

    function totalSupply()
        public
        view
        returns (uint256)
    {
        return totalSupply_;
    }

    function balanceOf(
        address _owner)
        public
        view
        returns (uint256)
    {
        return balances[_owner];
    }

    function allowance(
        address _owner,
        address _spender)
        public
        view
        returns (uint256)
    {
        return allowed[_owner][_spender];
    }
}

contract AdvancedToken is AdvancedTokenStorage {
    using SafeMath for uint256;

    function approve(
        address _spender,
        uint256 _value)
        public
        returns (bool)
    {
        allowed[msg.sender][_spender] = _value;
        emit Approval(msg.sender, _spender, _value);
        return true;
    }

    function _mint(
        address _to,
        uint256 _tokenAmount,
        uint256 _assetAmount,
        uint256 _price)
        internal
    {
        require(_to != address(0), "15");
        totalSupply_ = totalSupply_.add(_tokenAmount);
        balances[_to] = balances[_to].add(_tokenAmount);

        emit Mint(_to, _tokenAmount, _assetAmount, _price);
        emit Transfer(address(0), _to, _tokenAmount);
    }

    function _burn(
        address _who,
        uint256 _tokenAmount,
        uint256 _assetAmount,
        uint256 _price)
        internal
    {
        require(_tokenAmount <= balances[_who], "16");
        // no need to require value <= totalSupply, since that would imply the
        // sender's balance is greater than the totalSupply, which *should* be an assertion failure

        balances[_who] = balances[_who].sub(_tokenAmount);
        if (balances[_who] <= 10) { // we can't leave such small balance quantities
            _tokenAmount = _tokenAmount.add(balances[_who]);
            balances[_who] = 0;
        }

        totalSupply_ = totalSupply_.sub(_tokenAmount);

        emit Burn(_who, _tokenAmount, _assetAmount, _price);
        emit Transfer(_who, address(0), _tokenAmount);
    }
}

contract BZxObjects {

    struct LoanOrder {
        address loanTokenAddress;
        address interestTokenAddress;
        address collateralTokenAddress;
        address oracleAddress;
        uint256 loanTokenAmount;
        uint256 interestAmount;
        uint256 initialMarginAmount;
        uint256 maintenanceMarginAmount;
        uint256 maxDurationUnixTimestampSec;
        bytes32 loanOrderHash;
    }

    struct LoanPosition {
        address trader;
        address collateralTokenAddressFilled;
        address positionTokenAddressFilled;
        uint256 loanTokenAmountFilled;
        uint256 loanTokenAmountUsed;
        uint256 collateralTokenAmountFilled;
        uint256 positionTokenAmountFilled;
        uint256 loanStartUnixTimestampSec;
        uint256 loanEndUnixTimestampSec;
        bool active;
        uint256 positionId;
    }
}

contract OracleNotifierInterface {

    function closeLoanNotifier(
        BZxObjects.LoanOrder memory loanOrder,
        BZxObjects.LoanPosition memory loanPosition,
        address loanCloser,
        uint256 closeAmount,
        bool isLiquidation)
        public
        returns (bool);
}

interface IBZx {
    function takeOrderFromiToken(
        bytes32 loanOrderHash, // existing loan order hash
        address[4] calldata sentAddresses,
            // trader: borrower/trader
            // collateralTokenAddress: collateral token
            // tradeTokenAddress: trade token
            // receiver: receiver of funds (address(0) assumes trader address)
        uint256[7] calldata sentAmounts,
            // newInterestRate: new loan interest rate
            // newLoanAmount: new loan size (principal from lender)
            // interestInitialAmount: interestAmount sent to determine initial loan length (this is included in one of the below)
            // loanTokenSent: loanTokenAmount + interestAmount + any extra
            // collateralTokenSent: collateralAmountRequired + any extra
            // tradeTokenSent: tradeTokenAmount (optional)
            // withdrawalAmount: Actual amount sent to borrower (can't exceed newLoanAmount)
        bytes calldata loanDataBytes)
        external
        payable
        returns (uint256);

    function payInterestForOracle(
        address oracleAddress,
        address interestTokenAddress)
        external
        returns (uint256);

    function getLenderInterestForOracle(
        address lender,
        address oracleAddress,
        address interestTokenAddress)
        external
        view
        returns (
            uint256 interestPaid,
            uint256 interestPaidDate,
            uint256 interestOwedPerDay,
            uint256 interestUnPaid);

    function oracleAddresses(
        address oracleAddress)
        external
        view
        returns (address);

    function getRequiredCollateral(
        address loanTokenAddress,
        address collateralTokenAddress,
        address oracleAddress,
        uint256 newLoanAmount,
        uint256 marginAmount)
        external
        view
        returns (uint256 collateralTokenAmount);

    function getBorrowAmount(
        address loanTokenAddress,
        address collateralTokenAddress,
        address oracleAddress,
        uint256 collateralTokenAmount,
        uint256 marginAmount)
        external
        view
        returns (uint256 borrowAmount);
}

interface IBZxOracle {
    function getTradeData(
        address sourceTokenAddress,
        address destTokenAddress,
        uint256 sourceTokenAmount)
        external
        view
        returns (
            uint256 sourceToDestRate,
            uint256 sourceToDestPrecision,
            uint256 destTokenAmount
        );
}

interface IWethHelper {
    function claimEther(
        address receiver,
        uint256 amount)
        external
        returns (uint256 claimAmount);
}

contract LoanTokenLogicV4 is AdvancedToken, OracleNotifierInterface {
    using SafeMath for uint256;

    address internal target_;

    modifier onlyOracle() {
        require(msg.sender == IBZx(bZxContract).oracleAddresses(bZxOracle), "1");
        _;
    }


    function()
        external
    {}


    /* Public functions */

    function mintWithEther(
        address receiver)
        external
        payable
        nonReentrant
        returns (uint256 mintAmount)
    {
        require(loanTokenAddress == wethContract, "2");
        return _mintToken(
            receiver,
            msg.value
        );
    }

    function mint(
        address receiver,
        uint256 depositAmount)
        external
        nonReentrant
        returns (uint256 mintAmount)
    {
        return _mintToken(
            receiver,
            depositAmount
        );
    }

    function burnToEther(
        address receiver,
        uint256 burnAmount)
        external
        nonReentrant
        returns (uint256 loanAmountPaid)
    {
        require(loanTokenAddress == wethContract, "3");
        loanAmountPaid = _burnToken(
            burnAmount
        );

        if (loanAmountPaid != 0) {
            IWethHelper wethHelper = IWethHelper(0x3b5bDCCDFA2a0a1911984F203C19628EeB6036e0);

            _transfer(loanTokenAddress, address(wethHelper), loanAmountPaid, "4");
            require(loanAmountPaid == wethHelper.claimEther(receiver, loanAmountPaid), "4");
        }
    }

    function burn(
        address receiver,
        uint256 burnAmount)
        external
        nonReentrant
        returns (uint256 loanAmountPaid)
    {
        loanAmountPaid = _burnToken(
            burnAmount
        );

        if (loanAmountPaid != 0) {
            _transfer(loanTokenAddress, receiver, loanAmountPaid, "5");
        }
    }

    function borrowTokenFromDeposit(
        uint256 borrowAmount,
        uint256 leverageAmount,
        uint256 initialLoanDuration,    // duration in seconds
        uint256 collateralTokenSent,    // set to 0 if sending ETH
        address borrower,
        address receiver,
        address collateralTokenAddress, // address(0) means ETH and ETH must be sent with the call
        bytes memory /*loanDataBytes*/) // arbitrary order data
        public
        payable
        returns (bytes32 loanOrderHash)
    {
        require(tx.origin == owner, "unauthorized");

        require(
            ((msg.value == 0 && collateralTokenAddress != address(0) && collateralTokenSent != 0) ||
            (msg.value != 0 && (collateralTokenAddress == address(0) || collateralTokenAddress == wethContract) && collateralTokenSent == 0)),
            "6"
        );

        if (msg.value != 0) {
            collateralTokenAddress = wethContract;
            collateralTokenSent = msg.value;
        }

        uint256 _borrowAmount = borrowAmount;

        leverageAmount = uint256(keccak256(abi.encodePacked(leverageAmount,collateralTokenAddress)));
        loanOrderHash = loanOrderHashes[leverageAmount];
        require(loanOrderHash != 0, "7");

        _settleInterest();

        uint256[7] memory sentAmounts;

        LoanData memory loanOrder = loanOrderData[loanOrderHash];
        bool useFixedInterestModel = loanOrder.maxDurationUnixTimestampSec == 0;

        if (_borrowAmount == 0) {
            _borrowAmount = _getBorrowAmountForDeposit(
                collateralTokenSent,
                leverageAmount,
                initialLoanDuration,
                collateralTokenAddress
            );
            require(_borrowAmount != 0, "35");

            // withdrawalAmount
            sentAmounts[6] = _borrowAmount;
        } else {
            // withdrawalAmount
            sentAmounts[6] = _borrowAmount;
        }

        // interestRate, interestInitialAmount, borrowAmount (newBorrowAmount)
        (sentAmounts[0], sentAmounts[2], _borrowAmount) = _getInterestRateAndAmount(
            _borrowAmount,
            _totalAssetSupply(0), // interest is settled above
            initialLoanDuration,
            useFixedInterestModel
        );

        sentAmounts[6] = _borrowTokenAndUseFinal(
            loanOrderHash,
            [
                borrower,
                collateralTokenAddress,
                address(0), // tradeTokenAddress
                receiver
            ],
            [
                sentAmounts[0],         // interestRate
                _borrowAmount,
                sentAmounts[2],         // interestInitialAmount
                0,                      // loanTokenSent
                collateralTokenSent,
                0,                      // tradeTokenSent
                sentAmounts[6]          // withdrawalAmount
            ],
            ""                          // loanDataBytes
        );
        require(sentAmounts[6] == _borrowAmount, "8");
    }

    // Called to borrow and immediately get into a positions
    // assumption: depositAmount is collateral + interest deposit and will be denominated in deposit token
    // assumption: loan token and interest token are the same
    // returns loanOrderHash for the base protocol loan
    function marginTradeFromDeposit(
        uint256 depositAmount,
        uint256 leverageAmount,
        uint256 loanTokenSent,
        uint256 collateralTokenSent,
        uint256 tradeTokenSent,
        address trader,
        address depositTokenAddress,
        address collateralTokenAddress,
        address tradeTokenAddress,
        bytes memory loanDataBytes)
        public
        payable
        returns (bytes32 loanOrderHash)
    {
        require(tx.origin == owner, "unauthorized");

        require(tradeTokenAddress != address(0) &&
            tradeTokenAddress != loanTokenAddress,
            "10"
        );

        uint256 amount = depositAmount;
        // To calculate borrow amount and interest owed to lender we need deposit amount to be represented as loan token
        if (depositTokenAddress == tradeTokenAddress) {
            (,,amount) = IBZxOracle(bZxOracle).getTradeData(
                tradeTokenAddress,
                loanTokenAddress,
                amount
            );
        } else if (depositTokenAddress != loanTokenAddress) {
            // depositTokenAddress can only be tradeTokenAddress or loanTokenAddress
            revert("11");
        }

        loanOrderHash = _borrowTokenAndUse(
            leverageAmount,
            [
                trader,
                collateralTokenAddress,     // collateralTokenAddress
                tradeTokenAddress,          // tradeTokenAddress
                trader                      // receiver
            ],
            [
                0,                      // interestRate (found later)
                amount,                 // amount of deposit
                0,                      // interestInitialAmount (interest is calculated based on fixed-term loan)
                loanTokenSent,
                collateralTokenSent,
                tradeTokenSent,
                0
            ],
            true,                       // amountIsADeposit
            loanDataBytes
        );
    }

    function transfer(
        address _to,
        uint256 _value)
        public
        returns (bool)
    {
        require(_value <= balances[msg.sender] &&
            _to != address(0),
            "13"
        );

        balances[msg.sender] = balances[msg.sender].sub(_value);
        balances[_to] = balances[_to].add(_value);

        // handle checkpoint update
        uint256 currentPrice = tokenPrice();
        if (balances[msg.sender] != 0) {
            checkpointPrices_[msg.sender] = currentPrice;
        } else {
            checkpointPrices_[msg.sender] = 0;
        }
        if (balances[_to] != 0) {
            checkpointPrices_[_to] = currentPrice;
        } else {
            checkpointPrices_[_to] = 0;
        }

        emit Transfer(msg.sender, _to, _value);
        return true;
    }

    function transferFrom(
        address _from,
        address _to,
        uint256 _value)
        public
        returns (bool)
    {
        uint256 allowanceAmount = allowed[_from][msg.sender];
        require(_value <= balances[_from] &&
            _value <= allowanceAmount &&
            _to != address(0),
            "14"
        );

        balances[_from] = balances[_from].sub(_value);
        balances[_to] = balances[_to].add(_value);
        if (allowanceAmount < MAX_UINT) {
            allowed[_from][msg.sender] = allowanceAmount.sub(_value);
        }

        // handle checkpoint update
        uint256 currentPrice = tokenPrice();
        if (balances[_from] != 0) {
            checkpointPrices_[_from] = currentPrice;
        } else {
            checkpointPrices_[_from] = 0;
        }
        if (balances[_to] != 0) {
            checkpointPrices_[_to] = currentPrice;
        } else {
            checkpointPrices_[_to] = 0;
        }

        emit Transfer(_from, _to, _value);
        return true;
    }


    /* Public View functions */

    function tokenPrice()
        public
        view
        returns (uint256 price)
    {
        uint256 interestUnPaid;
        if (lastSettleTime_ != block.timestamp) {
            (,interestUnPaid) = _getAllInterest();
        }

        return _tokenPrice(_totalAssetSupply(interestUnPaid));
    }

    function checkpointPrice(
        address _user)
        public
        view
        returns (uint256 price)
    {
        return checkpointPrices_[_user];
    }

    function marketLiquidity()
        public
        view
        returns (uint256)
    {
        uint256 totalSupply = totalAssetSupply();
        if (totalSupply > totalAssetBorrow) {
            return totalSupply.sub(totalAssetBorrow);
        }
    }

    function protocolInterestRate()
        public
        view
        returns (uint256)
    {
        return _protocolInterestRate(totalAssetBorrow);
    }

    // the minimum rate the next base protocol borrower will receive for variable-rate loans
    function borrowInterestRate()
        public
        view
        returns (uint256)
    {
        return _nextBorrowInterestRate(
            0,              // borrowAmount
            false           // useFixedInterestModel
        );
    }

    function nextBorrowInterestRate(
        uint256 borrowAmount)
        public
        view
        returns (uint256)
    {
        return _nextBorrowInterestRate(
            borrowAmount,
            false           // useFixedInterestModel
        );
    }

    function nextBorrowInterestRateWithOption(
        uint256 borrowAmount,
        bool useFixedInterestModel)
        public
        view
        returns (uint256)
    {
        return _nextBorrowInterestRate(
            borrowAmount,
            useFixedInterestModel
        );
    }

    // the average interest that borrowers are currently paying for open loans
    function avgBorrowInterestRate()
        public
        view
        returns (uint256)
    {
        uint256 assetBorrow = totalAssetBorrow;
        if (assetBorrow != 0) {
            return _protocolInterestRate(assetBorrow)
                .mul(checkpointSupply)
                .div(totalAssetSupply());
        } else {
            return _getLowUtilBaseRate();
        }
    }

    // interest that lenders are currently receiving when supplying to the pool
    function supplyInterestRate()
        public
        view
        returns (uint256)
    {
        return totalSupplyInterestRate(totalAssetSupply());
    }

    function nextSupplyInterestRate(
        uint256 supplyAmount)
        public
        view
        returns (uint256)
    {
        return totalSupplyInterestRate(totalAssetSupply().add(supplyAmount));
    }

    function totalSupplyInterestRate(
        uint256 assetSupply)
        public
        view
        returns (uint256)
    {
        uint256 assetBorrow = totalAssetBorrow;
        if (assetBorrow != 0) {
            return _supplyInterestRate(
                assetBorrow,
                assetSupply
            );
        }
    }

    function totalAssetSupply()
        public
        view
        returns (uint256)
    {
        uint256 interestUnPaid;
        if (lastSettleTime_ != block.timestamp) {
            (,interestUnPaid) = _getAllInterest();
        }

        return _totalAssetSupply(interestUnPaid);
    }

    function getMaxEscrowAmount(
        uint256 leverageAmount)
        public
        view
        returns (uint256)
    {
        LoanData memory loanData = loanOrderData[loanOrderHashes[leverageAmount]];
        if (loanData.initialMarginAmount == 0)
            return 0;

        return marketLiquidity()
            .mul(loanData.initialMarginAmount)
            .div(_adjustValue(
                10**20, // maximum possible interest (100%)
                loanData.maxDurationUnixTimestampSec,
                loanData.initialMarginAmount));
    }

    function getLeverageList()
        public
        view
        returns (uint256[] memory)
    {
        return leverageList;
    }

    function getLoanData(
        bytes32 loanOrderHash)
        public
        view
        returns (LoanData memory)
    {
        return loanOrderData[loanOrderHash];
    }

    // returns the user's balance of underlying token
    function assetBalanceOf(
        address _owner)
        public
        view
        returns (uint256)
    {
        return balanceOf(_owner)
            .mul(tokenPrice())
            .div(10**18);
    }

    function getDepositAmountForBorrow(
        uint256 borrowAmount,
        uint256 leverageAmount,             // use 2000000000000000000 for 150% initial margin
        uint256 initialLoanDuration,        // duration in seconds
        address collateralTokenAddress)     // address(0) means ETH
        public
        view
        returns (uint256 depositAmount)
    {
        if (borrowAmount != 0) {
            leverageAmount = uint256(keccak256(abi.encodePacked(leverageAmount,collateralTokenAddress)));
            LoanData memory loanOrder = loanOrderData[loanOrderHashes[leverageAmount]];
            uint256 marginAmount = loanOrder.initialMarginAmount
                .add(10**20); // adjust for over-collateralized loan
                //.add(loanOrder.marginPremiumAmount);

            // adjust value since interest is also borrowed
            borrowAmount = borrowAmount
                .mul(_getTargetNextRateMultiplierValue(initialLoanDuration))
                .div(10**22);

            if (borrowAmount <= ERC20(loanTokenAddress).balanceOf(address(this))) {
                return IBZx(bZxContract).getRequiredCollateral(
                    loanTokenAddress,
                    collateralTokenAddress != address(0) ? collateralTokenAddress : wethContract,
                    bZxOracle,
                    borrowAmount,
                    marginAmount
                ).add(10); // add some dust to ensure enough is borrowed later
            }
        }
    }

    function getBorrowAmountForDeposit(
        uint256 depositAmount,
        uint256 leverageAmount,             // use 2000000000000000000 for 150% initial margin
        uint256 initialLoanDuration,        // duration in seconds
        address collateralTokenAddress)     // address(0) means ETH
        public
        view
        returns (uint256 borrowAmount)
    {
        leverageAmount = uint256(keccak256(abi.encodePacked(leverageAmount,collateralTokenAddress)));
        borrowAmount = _getBorrowAmountForDeposit(
            depositAmount,
            leverageAmount,
            initialLoanDuration,
            collateralTokenAddress
        );
    }


    /* Internal functions */

    function _mintToken(
        address receiver,
        uint256 depositAmount)
        internal
        returns (uint256 mintAmount)
    {
        require (depositAmount != 0, "17");

        _settleInterest();

        uint256 currentPrice = _tokenPrice(_totalAssetSupply(0));
        mintAmount = depositAmount.mul(10**18).div(currentPrice);

        if (msg.value == 0) {
            _transferFrom(loanTokenAddress, msg.sender, address(this), depositAmount, "18");
        } else {
            WETHInterface(wethContract).deposit.value(depositAmount)();
        }

        _mint(receiver, mintAmount, depositAmount, currentPrice);

        checkpointPrices_[receiver] = currentPrice;
    }

    function _burnToken(
        uint256 burnAmount)
        internal
        returns (uint256 loanAmountPaid)
    {
        require(burnAmount != 0, "19");

        if (burnAmount > balanceOf(msg.sender)) {
            burnAmount = balanceOf(msg.sender);
        }

        _settleInterest();

        uint256 currentPrice = _tokenPrice(_totalAssetSupply(0));

        uint256 loanAmountOwed = burnAmount.mul(currentPrice).div(10**18);
        uint256 loanAmountAvailableInContract = ERC20(loanTokenAddress).balanceOf(address(this));

        loanAmountPaid = loanAmountOwed;
        require(loanAmountPaid <= loanAmountAvailableInContract, "37");

        _burn(msg.sender, burnAmount, loanAmountPaid, currentPrice);

        if (balances[msg.sender] != 0) {
            checkpointPrices_[msg.sender] = currentPrice;
        } else {
            checkpointPrices_[msg.sender] = 0;
        }
    }

    function _settleInterest()
        internal
    {
        if (lastSettleTime_ != block.timestamp) {
            IBZx(bZxContract).payInterestForOracle(
                bZxOracle, // (leave as original value)
                loanTokenAddress // same as interestTokenAddress
            );

            lastSettleTime_ = block.timestamp;
        }
    }

    function _getBorrowAmountForDeposit(
        uint256 depositAmount,
        uint256 leverageAmount,             // use 2000000000000000000 for 150% initial margin
        uint256 initialLoanDuration,        // duration in seconds
        address collateralTokenAddress)     // address(0) means ETH
        internal
        view
        returns (uint256 borrowAmount)
    {
        if (depositAmount != 0) {
            LoanData memory loanOrder = loanOrderData[loanOrderHashes[leverageAmount]];
            uint256 marginAmount = loanOrder.initialMarginAmount
                .add(10**20); // adjust for over-collateralized loan
                //.add(loanOrder.marginPremiumAmount);

            borrowAmount = IBZx(bZxContract).getBorrowAmount(
                loanTokenAddress,
                collateralTokenAddress != address(0) ? collateralTokenAddress : wethContract,
                bZxOracle,
                depositAmount,
                marginAmount
            );

            // adjust value since interest is also borrowed
            borrowAmount = borrowAmount
                .mul(10**22)
                .div(_getTargetNextRateMultiplierValue(initialLoanDuration));

            if (borrowAmount > ERC20(loanTokenAddress).balanceOf(address(this))) {
                borrowAmount = 0;
            }
        }
    }

    function _getTargetNextRateMultiplierValue(
        uint256 initialLoanDuration)
        internal
        view
        returns (uint256)
    {
        return rateMultiplier
            .mul(80 ether)
            .div(10**20)
            .add(baseRate)
            .mul(initialLoanDuration)
            .div(315360) // 365 * 86400 / 100
            .add(10**22);
    }

    function _getInterestRateAndAmount(
        uint256 borrowAmount,
        uint256 assetSupply,
        uint256 initialLoanDuration,        // duration in seconds
        bool useFixedInterestModel)         // False=variable interest, True=fixed interest
        internal
        view
        returns (uint256 interestRate, uint256 interestInitialAmount, uint256 newBorrowAmount)
    {
        (,interestInitialAmount) = _getInterestRateAndAmount2(
            borrowAmount,
            assetSupply,
            initialLoanDuration,
            useFixedInterestModel
        );

        (interestRate, interestInitialAmount) = _getInterestRateAndAmount2(
            borrowAmount
                .add(interestInitialAmount),
            assetSupply,
            initialLoanDuration,
            useFixedInterestModel
        );

        newBorrowAmount = borrowAmount
            .add(interestInitialAmount);
    }

    function _getInterestRateAndAmount2(
        uint256 borrowAmount,
        uint256 assetSupply,
        uint256 initialLoanDuration,
        bool useFixedInterestModel)
        internal
        view
        returns (uint256 interestRate, uint256 interestInitialAmount)
    {
        interestRate = _nextBorrowInterestRate2(
            borrowAmount,
            assetSupply,
            useFixedInterestModel
        );

        // initial interestInitialAmount
        interestInitialAmount = borrowAmount
            .mul(interestRate)
            .mul(initialLoanDuration)
            .div(31536000 * 10**20); // 365 * 86400 * 10**20
    }

    function _borrowTokenAndUse(
        uint256 leverageAmount,
        address[4] memory sentAddresses,
        uint256[7] memory sentAmounts,
        bool amountIsADeposit,
        bytes memory loanDataBytes)
        internal
        returns (bytes32 loanOrderHash)
    {
        require(sentAmounts[1] != 0, "21"); // amount

        loanOrderHash = loanOrderHashes[leverageAmount];
        require(loanOrderHash != 0, "22");

        _settleInterest();

        LoanData memory loanOrder = loanOrderData[loanOrderHash];
        bool useFixedInterestModel = loanOrder.maxDurationUnixTimestampSec == 0;
        //sentAmounts[7] = loanOrder.marginPremiumAmount;

        if (amountIsADeposit) {
            (sentAmounts[1], sentAmounts[0]) = _getBorrowAmountAndRate( // borrowAmount, interestRate
                loanOrderHash,
                sentAmounts[1], // amount
                useFixedInterestModel
            );

            // update for borrowAmount
            sentAmounts[6] = sentAmounts[1]; // borrowAmount
        } else {
            // amount is borrow amount
            sentAmounts[0] = _nextBorrowInterestRate2( // interestRate
                sentAmounts[1], // amount
                _totalAssetSupply(0),
                useFixedInterestModel
            );
        }

        if (sentAddresses[2] == address(0)) { // tradeTokenAddress
            // tradeTokenSent is ignored if trade token isn't specified
            sentAmounts[5] = 0;
        }

        uint256 borrowAmount = _borrowTokenAndUseFinal(
            loanOrderHash,
            sentAddresses,
            sentAmounts,
            loanDataBytes
        );
        require(borrowAmount == sentAmounts[1], "23");
    }

    // returns borrowAmount
    function _borrowTokenAndUseFinal(
        bytes32 loanOrderHash,
        address[4] memory sentAddresses,
        uint256[7] memory sentAmounts,
        bytes memory loanDataBytes)
        internal
        returns (uint256)
    {
        _checkPause();

        require (sentAmounts[1] <= ERC20(loanTokenAddress).balanceOf(address(this)) && // borrowAmount
            sentAddresses[0] != address(0), // borrower
            "24"
        );

	    if (sentAddresses[3] == address(0)) {
            sentAddresses[3] = sentAddresses[0]; // receiver = borrower
        }

        // handle transfers prior to adding borrowAmount to loanTokenSent
        _verifyTransfers(
            sentAddresses,
            sentAmounts
        );

        // adding the loan token amount from the lender to loanTokenSent
        sentAmounts[3] = sentAmounts[3]
            .add(sentAmounts[1]); // borrowAmount

        uint256 msgValue;
        if (msg.value != 0) {
            msgValue = address(this).balance;
            if (msgValue > msg.value) {
                msgValue = msg.value;
            }
        }
        sentAmounts[1] = IBZx(bZxContract).takeOrderFromiToken.value(msgValue)( // borrowAmount
            loanOrderHash,
            sentAddresses,
            sentAmounts,
            loanDataBytes
        );
        require (sentAmounts[1] != 0, "25");

        // update total borrowed amount outstanding in loans
        totalAssetBorrow = totalAssetBorrow
            .add(sentAmounts[1]); // borrowAmount

        // checkpoint supply since the base protocol borrow stats have changed
        checkpointSupply = _totalAssetSupply(0);

        emit Borrow(
            sentAddresses[0],               // borrower
            sentAmounts[1],                 // borrowAmount
            sentAmounts[0],                 // interestRate
            sentAddresses[1],               // collateralTokenAddress
            sentAddresses[2],               // tradeTokenAddress
            sentAddresses[2] == address(0)  // withdrawOnOpen
        );

        return sentAmounts[1]; // borrowAmount;
    }

    // sentAddresses[0]: borrower
    // sentAddresses[1]: collateralTokenAddress
    // sentAddresses[2]: tradeTokenAddress
    // sentAddresses[3]: receiver
    // sentAmounts[0]: interestRate
    // sentAmounts[1]: borrowAmount
    // sentAmounts[2]: interestInitialAmount
    // sentAmounts[3]: loanTokenSent
    // sentAmounts[4]: collateralTokenSent
    // sentAmounts[5]: tradeTokenSent
    // sentAmounts[6]: withdrawalAmount
    function _verifyTransfers(
        address[4] memory sentAddresses,
        uint256[7] memory sentAmounts)
        internal
    {
        address collateralTokenAddress = sentAddresses[1];
        address tradeTokenAddress = sentAddresses[2];
        address receiver = sentAddresses[3];
        uint256 borrowAmount = sentAmounts[1];
        uint256 loanTokenSent = sentAmounts[3];
        uint256 collateralTokenSent = sentAmounts[4];
        uint256 tradeTokenSent = sentAmounts[5];
        uint256 withdrawalAmount = sentAmounts[6];

        bool success;
        if (tradeTokenAddress == address(0)) { // withdrawOnOpen == true
            if (loanTokenAddress == wethContract) {
                IWethHelper wethHelper = IWethHelper(0x3b5bDCCDFA2a0a1911984F203C19628EeB6036e0);

                _transfer(loanTokenAddress, address(wethHelper), withdrawalAmount, "");
                success = withdrawalAmount == wethHelper.claimEther(receiver, withdrawalAmount);
            } else {
                _transfer(loanTokenAddress, receiver, withdrawalAmount, "");
                success = true;
            }

            if (success && borrowAmount > withdrawalAmount) {
                _transfer(loanTokenAddress, bZxVault, borrowAmount - withdrawalAmount, "");
            }
            require(success, "26");
        } else {
            _transfer(loanTokenAddress, bZxVault, borrowAmount, "26");
        }

        if (collateralTokenSent != 0) {
            if (collateralTokenAddress == wethContract && msg.value != 0 && collateralTokenSent == msg.value) {
                WETHInterface(wethContract).deposit.value(collateralTokenSent)();
                _transfer(collateralTokenAddress, bZxVault, collateralTokenSent, "27");
            } else {
                if (collateralTokenAddress == loanTokenAddress) {
                    loanTokenSent = loanTokenSent.add(collateralTokenSent);
                } else if (collateralTokenAddress == tradeTokenAddress) {
                    tradeTokenSent = tradeTokenSent.add(collateralTokenSent);
                } else {
                    _transferFrom(collateralTokenAddress, msg.sender, bZxVault, collateralTokenSent, "27");
                }
            }
        }

        if (loanTokenSent != 0) {
            if (loanTokenAddress == tradeTokenAddress) {
                tradeTokenSent = tradeTokenSent.add(loanTokenSent);
            } else {
                _transferFrom(loanTokenAddress, msg.sender, bZxVault, loanTokenSent, "31");
            }
        }

        if (tradeTokenSent != 0) {
            _transferFrom(tradeTokenAddress, msg.sender, bZxVault, tradeTokenSent, "32");
        }
    }

    function _transfer(
        address token,
        address to,
        uint256 amount,
        string memory errorMsg)
        internal
    {
        _callOptionalReturn(
            token,
            abi.encodeWithSelector(ERC20(token).transfer.selector, to, amount),
            errorMsg
        );
    }

    function _transferFrom(
        address token,
        address from,
        address to,
        uint256 amount,
        string memory errorMsg)
        internal
    {
        _callOptionalReturn(
            token,
            abi.encodeWithSelector(ERC20(token).transferFrom.selector, from, to, amount),
            errorMsg
        );
    }

    function _callOptionalReturn(
        address token,
        bytes memory data,
        string memory errorMsg)
        internal
    {
        (bool success, bytes memory returndata) = token.call(data);
        require(success, errorMsg);

        if (returndata.length != 0) {
            require(abi.decode(returndata, (bool)), errorMsg);
        }
    }

    /* Internal View functions */

    function _tokenPrice(
        uint256 assetSupply)
        internal
        view
        returns (uint256)
    {
        uint256 totalTokenSupply = totalSupply_;

        return totalTokenSupply != 0 ?
            assetSupply
                .mul(10**18)
                .div(totalTokenSupply) : initialPrice;
    }

    function _protocolInterestRate(
        uint256 assetBorrow)
        internal
        view
        returns (uint256)
    {
        if (assetBorrow != 0) {
            (uint256 interestOwedPerDay,) = _getAllInterest();
            return interestOwedPerDay
                .mul(10**20)
                .div(assetBorrow)
                .mul(365);
        }
    }

    // next supply interest adjustment
    function _supplyInterestRate(
        uint256 assetBorrow,
        uint256 assetSupply)
        public
        view
        returns (uint256)
    {
        if (assetBorrow != 0 && assetSupply >= assetBorrow) {
            return _protocolInterestRate(assetBorrow)
                .mul(_utilizationRate(assetBorrow, assetSupply))
                .mul(spreadMultiplier)
                .div(10**40);
        }
    }

    function _nextBorrowInterestRate(
        uint256 borrowAmount,
        bool useFixedInterestModel)
        internal
        view
        returns (uint256)
    {
        uint256 interestUnPaid;
        if (borrowAmount != 0) {
            if (lastSettleTime_ != block.timestamp) {
                (,interestUnPaid) = _getAllInterest();
            }

            uint256 balance = ERC20(loanTokenAddress).balanceOf(address(this))
                .add(interestUnPaid);
            if (borrowAmount > balance) {
                borrowAmount = balance;
            }
        }

        return _nextBorrowInterestRate2(
            borrowAmount,
            _totalAssetSupply(interestUnPaid),
            useFixedInterestModel
        );
    }

    function _nextBorrowInterestRate2(
        uint256 newBorrowAmount,
        uint256 assetSupply,
        bool useFixedInterestModel)
        internal
        view
        returns (uint256 nextRate)
    {
        uint256 utilRate = _utilizationRate(
            totalAssetBorrow.add(newBorrowAmount),
            assetSupply
        );

        uint256 minRate;
        uint256 maxRate;
        uint256 thisBaseRate;
        uint256 thisRateMultiplier;

        if (useFixedInterestModel) {
            if (utilRate < 80 ether) {
                // target 80% utilization when loan is fixed-rate and utilization is under 80%
                utilRate = 80 ether;
            }

            //keccak256("iToken_FixedInterestBaseRate")
            //keccak256("iToken_FixedInterestRateMultiplier")
            assembly {
                thisBaseRate := sload(0x185a40c6b6d3f849f72c71ea950323d21149c27a9d90f7dc5e5ea2d332edcf7f)
                thisRateMultiplier := sload(0x9ff54bc0049f5eab56ca7cd14591be3f7ed6355b856d01e3770305c74a004ea2)
            }
        } else if (utilRate < 50 ether) {
            thisBaseRate = _getLowUtilBaseRate();

            //keccak256("iToken_LowUtilRateMultiplier")
            assembly {
                thisRateMultiplier := sload(0x2b4858b1bc9e2d14afab03340ce5f6c81b703c86a0c570653ae586534e095fb1)
            }
        } else {
            thisBaseRate = baseRate;
            thisRateMultiplier = rateMultiplier;
        }

        if (utilRate > 90 ether) {
            // scale rate proportionally up to 100%

            utilRate = utilRate.sub(90 ether);
            if (utilRate > 10 ether)
                utilRate = 10 ether;

            maxRate = thisRateMultiplier
                .add(thisBaseRate)
                .mul(90)
                .div(100);

            nextRate = utilRate
                .mul(SafeMath.sub(100 ether, maxRate))
                .div(10 ether)
                .add(maxRate);
        } else {
            nextRate = utilRate
                .mul(thisRateMultiplier)
                .div(10**20)
                .add(thisBaseRate);

            minRate = thisBaseRate;
            maxRate = thisRateMultiplier
                .add(thisBaseRate);

            if (nextRate < minRate)
                nextRate = minRate;
            else if (nextRate > maxRate)
                nextRate = maxRate;
        }
    }

    function _getAllInterest()
        internal
        view
        returns (
            uint256 interestOwedPerDay,
            uin...

// [truncated — 56007 bytes total]

Read Contract

_supplyInterestRate 0x7288b344 → uint256
allowance 0xdd62ed3e → uint256
assetBalanceOf 0x06b3efd6 → uint256
avgBorrowInterestRate 0x44a4a003 → uint256
bZxContract 0x995363d3 → address
bZxOracle 0x96c7871b → address
bZxVault 0x894ca308 → address
balanceOf 0x70a08231 → uint256
baseRate 0x1f68f20a → uint256
borrowInterestRate 0x8325a1c0 → uint256
burntTokenReserveList 0x7866c6c1 → address, uint256
burntTokenReserveListIndex 0xfbd9574d → uint256, bool
burntTokenReserved 0x0c4925fd → uint256
checkpointPrice 0xeebc5081 → uint256
checkpointSupply 0x7b7933b4 → uint256
decimals 0x313ce567 → uint8
getBorrowAmountForDeposit 0x24d25f4a → uint256
getDepositAmountForBorrow 0x8423acd6 → uint256
getLeverageList 0x2ecae90a → uint256[]
getLoanData 0xc4d2b1b3 → tuple
getMaxEscrowAmount 0x829b38f4 → uint256
initialPrice 0x1d0806ae → uint256
leverageList 0x9b3a54d1 → uint256
loanOrderData 0x2515aacd → bytes32, uint256, uint256, uint256, uint256, uint256, uint256, address
loanOrderHashes 0xfe056342 → bytes32
loanTokenAddress 0x797bf385 → address
marketLiquidity 0x612ef80b → uint256
name 0x06fdde03 → string
nextBorrowInterestRate 0xb9fe1a8f → uint256
nextBorrowInterestRateWithOption 0x7d90dcba → uint256
nextSupplyInterestRate 0xd65a5021 → uint256
owner 0x8da5cb5b → address
protocolInterestRate 0xfc3b72b1 → uint256
rateMultiplier 0x330691ac → uint256
spreadMultiplier 0xd84d2a47 → uint256
supplyInterestRate 0x09ec6b6b → uint256
symbol 0x95d89b41 → string
tokenPrice 0x7ff9b596 → uint256
tokenizedRegistry 0x736ee3d3 → address
totalAssetBorrow 0x20f6d07c → uint256
totalAssetSupply 0x8fb807c5 → uint256
totalSupply 0x18160ddd → uint256
totalSupplyInterestRate 0x12416898 → uint256
wethContract 0x4780eac1 → address

Write Contract 12 functions

These functions modify contract state and require a wallet transaction to execute.

approve 0x095ea7b3
address _spender
uint256 _value
returns: bool
borrowTokenFromDeposit 0xcfb65bb9
uint256 borrowAmount
uint256 leverageAmount
uint256 initialLoanDuration
uint256 collateralTokenSent
address borrower
address receiver
address collateralTokenAddress
bytes
returns: bytes32
burn 0x9dc29fac
address receiver
uint256 burnAmount
returns: uint256
burnToEther 0x81a6b250
address receiver
uint256 burnAmount
returns: uint256
closeLoanNotifier 0xfa06df39
tuple loanOrder
tuple loanPosition
address loanCloser
uint256 closeAmount
bool isLiquidation
returns: bool
marginTradeFromDeposit 0x1c5d1da5
uint256 depositAmount
uint256 leverageAmount
uint256 loanTokenSent
uint256 collateralTokenSent
uint256 tradeTokenSent
address trader
address depositTokenAddress
address collateralTokenAddress
address tradeTokenAddress
bytes loanDataBytes
returns: bytes32
mint 0x40c10f19
address receiver
uint256 depositAmount
returns: uint256
mintWithEther 0x8f6ede1f
address receiver
returns: uint256
transfer 0xa9059cbb
address _to
uint256 _value
returns: bool
transferFrom 0x23b872dd
address _from
address _to
uint256 _value
returns: bool
transferOwnership 0xf2fde38b
address _newOwner
updateSettings 0x284e2f56
address settingsTarget
bytes callData

Recent Transactions

This address has 1 on-chain transactions, but only 1.6% of the chain is indexed. Transactions will appear as indexing progresses. View on Etherscan →