Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 7,010 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Start Exit With ... | 20453360 | 165 days ago | IN | 0 ETH | 0.00030733 | ||||
Start Exit With ... | 20453299 | 165 days ago | IN | 0 ETH | 0.00030733 | ||||
Start Exit With ... | 20453097 | 165 days ago | IN | 0 ETH | 0.00030733 | ||||
Start Exit With ... | 20452939 | 165 days ago | IN | 0 ETH | 0.00042987 | ||||
Start Exit With ... | 16861619 | 669 days ago | IN | 0 ETH | 0.00568835 | ||||
Start Exit With ... | 16849858 | 671 days ago | IN | 0 ETH | 0.00168505 | ||||
Start Exit With ... | 16698680 | 692 days ago | IN | 0 ETH | 0.00224059 | ||||
Start Exit With ... | 16290250 | 749 days ago | IN | 0 ETH | 0.00135277 | ||||
Start Exit With ... | 16268028 | 752 days ago | IN | 0 ETH | 0.00072547 | ||||
Start Exit With ... | 16047282 | 783 days ago | IN | 0 ETH | 0.00077296 | ||||
Start Exit With ... | 16046976 | 783 days ago | IN | 0 ETH | 0.00086574 | ||||
Start Exit With ... | 16046960 | 783 days ago | IN | 0 ETH | 0.00081476 | ||||
Start Exit With ... | 16040232 | 784 days ago | IN | 0 ETH | 0.00082507 | ||||
Start Exit With ... | 16039125 | 784 days ago | IN | 0 ETH | 0.00091226 | ||||
Start Exit With ... | 16038684 | 784 days ago | IN | 0 ETH | 0.00095626 | ||||
Start Exit With ... | 16038452 | 784 days ago | IN | 0 ETH | 0.00080601 | ||||
Start Exit With ... | 16036517 | 785 days ago | IN | 0 ETH | 0.00093541 | ||||
Start Exit With ... | 16036232 | 785 days ago | IN | 0 ETH | 0.0008673 | ||||
Start Exit With ... | 15870997 | 808 days ago | IN | 0 ETH | 0.00724686 | ||||
Start Exit With ... | 15664694 | 837 days ago | IN | 0 ETH | 0.00050351 | ||||
Start Exit With ... | 15427887 | 872 days ago | IN | 0 ETH | 0.00026471 | ||||
Start Exit With ... | 15424891 | 873 days ago | IN | 0 ETH | 0.00027157 | ||||
Start Exit With ... | 15375121 | 881 days ago | IN | 0 ETH | 0.00270537 | ||||
Start Exit With ... | 15335924 | 887 days ago | IN | 0 ETH | 0.00455471 | ||||
Start Exit With ... | 15236105 | 902 days ago | IN | 0 ETH | 0.0066426 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Name:
ERC20PredicateBurnOnly
Compiler Version
v0.5.17+commit.d19bba13
Contract Source Code (Solidity)
/** *Submitted for verification at Etherscan.io on 2021-04-21 */ // File: openzeppelin-solidity/contracts/math/SafeMath.sol pragma solidity ^0.5.2; /** * @title SafeMath * @dev Unsigned math operations with safety checks that revert on error */ library SafeMath { /** * @dev Multiplies two unsigned integers, reverts on overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { // Gas optimization: this is cheaper than requiring '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; } uint256 c = a * b; require(c / a == b); return c; } /** * @dev Integer division of two unsigned integers truncating the quotient, reverts on division by zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { // Solidity only automatically asserts when dividing by 0 require(b > 0); uint256 c = a / b; // assert(a == b * c + a % b); // There is no case in which this doesn't hold return c; } /** * @dev Subtracts two unsigned integers, reverts on overflow (i.e. if subtrahend is greater than minuend). */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { require(b <= a); uint256 c = a - b; return c; } /** * @dev Adds two unsigned integers, reverts on overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; require(c >= a); return c; } /** * @dev Divides two unsigned integers and returns the remainder (unsigned integer modulo), * reverts when dividing by zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { require(b != 0); return a % b; } } // File: contracts/common/lib/BytesLib.sol pragma solidity ^0.5.2; library BytesLib { function concat(bytes memory _preBytes, bytes memory _postBytes) internal pure returns (bytes memory) { bytes memory tempBytes; assembly { // Get a location of some free memory and store it in tempBytes as // Solidity does for memory variables. tempBytes := mload(0x40) // Store the length of the first bytes array at the beginning of // the memory for tempBytes. let length := mload(_preBytes) mstore(tempBytes, length) // Maintain a memory counter for the current write location in the // temp bytes array by adding the 32 bytes for the array length to // the starting location. let mc := add(tempBytes, 0x20) // Stop copying when the memory counter reaches the length of the // first bytes array. let end := add(mc, length) for { // Initialize a copy counter to the start of the _preBytes data, // 32 bytes into its memory. let cc := add(_preBytes, 0x20) } lt(mc, end) { // Increase both counters by 32 bytes each iteration. mc := add(mc, 0x20) cc := add(cc, 0x20) } { // Write the _preBytes data into the tempBytes memory 32 bytes // at a time. mstore(mc, mload(cc)) } // Add the length of _postBytes to the current length of tempBytes // and store it as the new length in the first 32 bytes of the // tempBytes memory. length := mload(_postBytes) mstore(tempBytes, add(length, mload(tempBytes))) // Move the memory counter back from a multiple of 0x20 to the // actual end of the _preBytes data. mc := end // Stop copying when the memory counter reaches the new combined // length of the arrays. end := add(mc, length) for { let cc := add(_postBytes, 0x20) } lt(mc, end) { mc := add(mc, 0x20) cc := add(cc, 0x20) } { mstore(mc, mload(cc)) } // Update the free-memory pointer by padding our last write location // to 32 bytes: add 31 bytes to the end of tempBytes to move to the // next 32 byte block, then round down to the nearest multiple of // 32. If the sum of the length of the two arrays is zero then add // one before rounding down to leave a blank 32 bytes (the length block with 0). mstore( 0x40, and( add(add(end, iszero(add(length, mload(_preBytes)))), 31), not(31) // Round down to the nearest 32 bytes. ) ) } return tempBytes; } function slice(bytes memory _bytes, uint256 _start, uint256 _length) internal pure returns (bytes memory) { require(_bytes.length >= (_start + _length)); bytes memory tempBytes; assembly { switch iszero(_length) case 0 { // Get a location of some free memory and store it in tempBytes as // Solidity does for memory variables. tempBytes := mload(0x40) // The first word of the slice result is potentially a partial // word read from the original array. To read it, we calculate // the length of that partial word and start copying that many // bytes into the array. The first word we copy will start with // data we don't care about, but the last `lengthmod` bytes will // land at the beginning of the contents of the new array. When // we're done copying, we overwrite the full first word with // the actual length of the slice. let lengthmod := and(_length, 31) // The multiplication in the next line is necessary // because when slicing multiples of 32 bytes (lengthmod == 0) // the following copy loop was copying the origin's length // and then ending prematurely not copying everything it should. let mc := add( add(tempBytes, lengthmod), mul(0x20, iszero(lengthmod)) ) let end := add(mc, _length) for { // The multiplication in the next line has the same exact purpose // as the one above. let cc := add( add( add(_bytes, lengthmod), mul(0x20, iszero(lengthmod)) ), _start ) } lt(mc, end) { mc := add(mc, 0x20) cc := add(cc, 0x20) } { mstore(mc, mload(cc)) } mstore(tempBytes, _length) //update free-memory pointer //allocating the array padded to 32 bytes like the compiler does now mstore(0x40, and(add(mc, 31), not(31))) } //if we want a zero-length slice let's just return a zero-length array default { tempBytes := mload(0x40) mstore(0x40, add(tempBytes, 0x20)) } } return tempBytes; } // Pad a bytes array to 32 bytes function leftPad(bytes memory _bytes) internal pure returns (bytes memory) { // may underflow if bytes.length < 32. Hence using SafeMath.sub bytes memory newBytes = new bytes(SafeMath.sub(32, _bytes.length)); return concat(newBytes, _bytes); } function toBytes32(bytes memory b) internal pure returns (bytes32) { require(b.length >= 32, "Bytes array should atleast be 32 bytes"); bytes32 out; for (uint256 i = 0; i < 32; i++) { out |= bytes32(b[i] & 0xFF) >> (i * 8); } return out; } function toBytes4(bytes memory b) internal pure returns (bytes4 result) { assembly { result := mload(add(b, 32)) } } function fromBytes32(bytes32 x) internal pure returns (bytes memory) { bytes memory b = new bytes(32); for (uint256 i = 0; i < 32; i++) { b[i] = bytes1(uint8(uint256(x) / (2**(8 * (31 - i))))); } return b; } function fromUint(uint256 _num) internal pure returns (bytes memory _ret) { _ret = new bytes(32); assembly { mstore(add(_ret, 32), _num) } } function toUint(bytes memory _bytes, uint256 _start) internal pure returns (uint256) { require(_bytes.length >= (_start + 32)); uint256 tempUint; assembly { tempUint := mload(add(add(_bytes, 0x20), _start)) } return tempUint; } function toAddress(bytes memory _bytes, uint256 _start) internal pure returns (address) { require(_bytes.length >= (_start + 20)); address tempAddress; assembly { tempAddress := div( mload(add(add(_bytes, 0x20), _start)), 0x1000000000000000000000000 ) } return tempAddress; } } // File: contracts/common/lib/Common.sol pragma solidity ^0.5.2; library Common { function getV(bytes memory v, uint16 chainId) public pure returns (uint8) { if (chainId > 0) { return uint8( BytesLib.toUint(BytesLib.leftPad(v), 0) - (chainId * 2) - 8 ); } else { return uint8(BytesLib.toUint(BytesLib.leftPad(v), 0)); } } //assemble the given address bytecode. If bytecode exists then the _addr is a contract. function isContract(address _addr) public view returns (bool) { uint256 length; assembly { //retrieve the size of the code on target address, this needs assembly length := extcodesize(_addr) } return (length > 0); } // convert bytes to uint8 function toUint8(bytes memory _arg) public pure returns (uint8) { return uint8(_arg[0]); } function toUint16(bytes memory _arg) public pure returns (uint16) { return (uint16(uint8(_arg[0])) << 8) | uint16(uint8(_arg[1])); } } // File: openzeppelin-solidity/contracts/math/Math.sol pragma solidity ^0.5.2; /** * @title Math * @dev Assorted math operations */ library Math { /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a >= b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Calculates the average of two numbers. Since these are integers, * averages of an even and odd number cannot be represented, and will be * rounded down. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow, so we distribute return (a / 2) + (b / 2) + ((a % 2 + b % 2) / 2); } } // File: contracts/common/lib/RLPEncode.sol // Library for RLP encoding a list of bytes arrays. // Modeled after ethereumjs/rlp (https://github.com/ethereumjs/rlp) // [Very] modified version of Sam Mayo's library. pragma solidity ^0.5.2; library RLPEncode { // Encode an item (bytes memory) function encodeItem(bytes memory self) internal pure returns (bytes memory) { bytes memory encoded; if (self.length == 1 && uint8(self[0] & 0xFF) < 0x80) { encoded = new bytes(1); encoded = self; } else { encoded = BytesLib.concat(encodeLength(self.length, 128), self); } return encoded; } // Encode a list of items function encodeList(bytes[] memory self) internal pure returns (bytes memory) { bytes memory encoded; for (uint256 i = 0; i < self.length; i++) { encoded = BytesLib.concat(encoded, encodeItem(self[i])); } return BytesLib.concat(encodeLength(encoded.length, 192), encoded); } // Hack to encode nested lists. If you have a list as an item passed here, included // pass = true in that index. E.g. // [item, list, item] --> pass = [false, true, false] // function encodeListWithPasses(bytes[] memory self, bool[] pass) internal pure returns (bytes memory) { // bytes memory encoded; // for (uint i=0; i < self.length; i++) { // if (pass[i] == true) { // encoded = BytesLib.concat(encoded, self[i]); // } else { // encoded = BytesLib.concat(encoded, encodeItem(self[i])); // } // } // return BytesLib.concat(encodeLength(encoded.length, 192), encoded); // } // Generate the prefix for an item or the entire list based on RLP spec function encodeLength(uint256 L, uint256 offset) internal pure returns (bytes memory) { if (L < 56) { bytes memory prefix = new bytes(1); prefix[0] = bytes1(uint8(L + offset)); return prefix; } else { // lenLen is the length of the hex representation of the data length uint256 lenLen; uint256 i = 0x1; while (L / i != 0) { lenLen++; i *= 0x100; } bytes memory prefix0 = getLengthBytes(offset + 55 + lenLen); bytes memory prefix1 = getLengthBytes(L); return BytesLib.concat(prefix0, prefix1); } } function getLengthBytes(uint256 x) internal pure returns (bytes memory b) { // Figure out if we need 1 or two bytes to express the length. // 1 byte gets us to max 255 // 2 bytes gets us to max 65535 (no payloads will be larger than this) uint256 nBytes = 1; if (x > 255) { nBytes = 2; } b = new bytes(nBytes); // Encode the length and return it for (uint256 i = 0; i < nBytes; i++) { b[i] = bytes1(uint8(x / (2**(8 * (nBytes - 1 - i))))); } } } // File: solidity-rlp/contracts/RLPReader.sol // SPDX-License-Identifier: Apache-2.0 /* * @author Hamdi Allam [email protected] * Please reach out with any questions or concerns */ pragma solidity >=0.5.0 <0.7.0; library RLPReader { uint8 constant STRING_SHORT_START = 0x80; uint8 constant STRING_LONG_START = 0xb8; uint8 constant LIST_SHORT_START = 0xc0; uint8 constant LIST_LONG_START = 0xf8; uint8 constant WORD_SIZE = 32; struct RLPItem { uint len; uint memPtr; } struct Iterator { RLPItem item; // Item that's being iterated over. uint nextPtr; // Position of the next item in the list. } /* * @dev Returns the next element in the iteration. Reverts if it has not next element. * @param self The iterator. * @return The next element in the iteration. */ function next(Iterator memory self) internal pure returns (RLPItem memory) { require(hasNext(self)); uint ptr = self.nextPtr; uint itemLength = _itemLength(ptr); self.nextPtr = ptr + itemLength; return RLPItem(itemLength, ptr); } /* * @dev Returns true if the iteration has more elements. * @param self The iterator. * @return true if the iteration has more elements. */ function hasNext(Iterator memory self) internal pure returns (bool) { RLPItem memory item = self.item; return self.nextPtr < item.memPtr + item.len; } /* * @param item RLP encoded bytes */ function toRlpItem(bytes memory item) internal pure returns (RLPItem memory) { uint memPtr; assembly { memPtr := add(item, 0x20) } return RLPItem(item.length, memPtr); } /* * @dev Create an iterator. Reverts if item is not a list. * @param self The RLP item. * @return An 'Iterator' over the item. */ function iterator(RLPItem memory self) internal pure returns (Iterator memory) { require(isList(self)); uint ptr = self.memPtr + _payloadOffset(self.memPtr); return Iterator(self, ptr); } /* * @param the RLP item. */ function rlpLen(RLPItem memory item) internal pure returns (uint) { return item.len; } /* * @param the RLP item. * @return (memPtr, len) pair: location of the item's payload in memory. */ function payloadLocation(RLPItem memory item) internal pure returns (uint, uint) { uint offset = _payloadOffset(item.memPtr); uint memPtr = item.memPtr + offset; uint len = item.len - offset; // data length return (memPtr, len); } /* * @param the RLP item. */ function payloadLen(RLPItem memory item) internal pure returns (uint) { (, uint len) = payloadLocation(item); return len; } /* * @param the RLP item containing the encoded list. */ function toList(RLPItem memory item) internal pure returns (RLPItem[] memory) { require(isList(item)); uint items = numItems(item); RLPItem[] memory result = new RLPItem[](items); uint memPtr = item.memPtr + _payloadOffset(item.memPtr); uint dataLen; for (uint i = 0; i < items; i++) { dataLen = _itemLength(memPtr); result[i] = RLPItem(dataLen, memPtr); memPtr = memPtr + dataLen; } return result; } // @return indicator whether encoded payload is a list. negate this function call for isData. function isList(RLPItem memory item) internal pure returns (bool) { if (item.len == 0) return false; uint8 byte0; uint memPtr = item.memPtr; assembly { byte0 := byte(0, mload(memPtr)) } if (byte0 < LIST_SHORT_START) return false; return true; } /* * @dev A cheaper version of keccak256(toRlpBytes(item)) that avoids copying memory. * @return keccak256 hash of RLP encoded bytes. */ function rlpBytesKeccak256(RLPItem memory item) internal pure returns (bytes32) { uint256 ptr = item.memPtr; uint256 len = item.len; bytes32 result; assembly { result := keccak256(ptr, len) } return result; } /* * @dev A cheaper version of keccak256(toBytes(item)) that avoids copying memory. * @return keccak256 hash of the item payload. */ function payloadKeccak256(RLPItem memory item) internal pure returns (bytes32) { (uint memPtr, uint len) = payloadLocation(item); bytes32 result; assembly { result := keccak256(memPtr, len) } return result; } /** RLPItem conversions into data types **/ // @returns raw rlp encoding in bytes function toRlpBytes(RLPItem memory item) internal pure returns (bytes memory) { bytes memory result = new bytes(item.len); if (result.length == 0) return result; uint ptr; assembly { ptr := add(0x20, result) } copy(item.memPtr, ptr, item.len); return result; } // any non-zero byte except "0x80" is considered true function toBoolean(RLPItem memory item) internal pure returns (bool) { require(item.len == 1); uint result; uint memPtr = item.memPtr; assembly { result := byte(0, mload(memPtr)) } // SEE Github Issue #5. // Summary: Most commonly used RLP libraries (i.e Geth) will encode // "0" as "0x80" instead of as "0". We handle this edge case explicitly // here. if (result == 0 || result == STRING_SHORT_START) { return false; } else { return true; } } function toAddress(RLPItem memory item) internal pure returns (address) { // 1 byte for the length prefix require(item.len == 21); return address(toUint(item)); } function toUint(RLPItem memory item) internal pure returns (uint) { require(item.len > 0 && item.len <= 33); (uint memPtr, uint len) = payloadLocation(item); uint result; assembly { result := mload(memPtr) // shfit to the correct location if neccesary if lt(len, 32) { result := div(result, exp(256, sub(32, len))) } } return result; } // enforces 32 byte length function toUintStrict(RLPItem memory item) internal pure returns (uint) { // one byte prefix require(item.len == 33); uint result; uint memPtr = item.memPtr + 1; assembly { result := mload(memPtr) } return result; } function toBytes(RLPItem memory item) internal pure returns (bytes memory) { require(item.len > 0); (uint memPtr, uint len) = payloadLocation(item); bytes memory result = new bytes(len); uint destPtr; assembly { destPtr := add(0x20, result) } copy(memPtr, destPtr, len); return result; } /* * Private Helpers */ // @return number of payload items inside an encoded list. function numItems(RLPItem memory item) private pure returns (uint) { if (item.len == 0) return 0; uint count = 0; uint currPtr = item.memPtr + _payloadOffset(item.memPtr); uint endPtr = item.memPtr + item.len; while (currPtr < endPtr) { currPtr = currPtr + _itemLength(currPtr); // skip over an item count++; } return count; } // @return entire rlp item byte length function _itemLength(uint memPtr) private pure returns (uint) { uint itemLen; uint byte0; assembly { byte0 := byte(0, mload(memPtr)) } if (byte0 < STRING_SHORT_START) itemLen = 1; else if (byte0 < STRING_LONG_START) itemLen = byte0 - STRING_SHORT_START + 1; else if (byte0 < LIST_SHORT_START) { assembly { let byteLen := sub(byte0, 0xb7) // # of bytes the actual length is memPtr := add(memPtr, 1) // skip over the first byte /* 32 byte word size */ let dataLen := div(mload(memPtr), exp(256, sub(32, byteLen))) // right shifting to get the len itemLen := add(dataLen, add(byteLen, 1)) } } else if (byte0 < LIST_LONG_START) { itemLen = byte0 - LIST_SHORT_START + 1; } else { assembly { let byteLen := sub(byte0, 0xf7) memPtr := add(memPtr, 1) let dataLen := div(mload(memPtr), exp(256, sub(32, byteLen))) // right shifting to the correct length itemLen := add(dataLen, add(byteLen, 1)) } } return itemLen; } // @return number of bytes until the data function _payloadOffset(uint memPtr) private pure returns (uint) { uint byte0; assembly { byte0 := byte(0, mload(memPtr)) } if (byte0 < STRING_SHORT_START) return 0; else if (byte0 < STRING_LONG_START || (byte0 >= LIST_SHORT_START && byte0 < LIST_LONG_START)) return 1; else if (byte0 < LIST_SHORT_START) // being explicit return byte0 - (STRING_LONG_START - 1) + 1; else return byte0 - (LIST_LONG_START - 1) + 1; } /* * @param src Pointer to source * @param dest Pointer to destination * @param len Amount of memory to copy from the source */ function copy(uint src, uint dest, uint len) private pure { if (len == 0) return; // copy as many word sizes as possible for (; len >= WORD_SIZE; len -= WORD_SIZE) { assembly { mstore(dest, mload(src)) } src += WORD_SIZE; dest += WORD_SIZE; } // left over bytes. Mask is used to remove unwanted bytes from the word uint mask = 256 ** (WORD_SIZE - len) - 1; assembly { let srcpart := and(mload(src), not(mask)) // zero out src let destpart := and(mload(dest), mask) // retrieve the bytes mstore(dest, or(destpart, srcpart)) } } } // File: contracts/root/withdrawManager/IWithdrawManager.sol pragma solidity ^0.5.2; contract IWithdrawManager { function createExitQueue(address token) external; function verifyInclusion( bytes calldata data, uint8 offset, bool verifyTxInclusion ) external view returns (uint256 age); function addExitToQueue( address exitor, address childToken, address rootToken, uint256 exitAmountOrTokenId, bytes32 txHash, bool isRegularExit, uint256 priority ) external; function addInput( uint256 exitId, uint256 age, address utxoOwner, address token ) external; function challengeExit( uint256 exitId, uint256 inputId, bytes calldata challengeData, address adjudicatorPredicate ) external; } // File: contracts/root/depositManager/IDepositManager.sol pragma solidity ^0.5.2; interface IDepositManager { function depositEther() external payable; function transferAssets( address _token, address _user, uint256 _amountOrNFTId ) external; function depositERC20(address _token, uint256 _amount) external; function depositERC721(address _token, uint256 _tokenId) external; } // File: openzeppelin-solidity/contracts/ownership/Ownable.sol pragma solidity ^0.5.2; /** * @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 private _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 () internal { _owner = msg.sender; emit OwnershipTransferred(address(0), _owner); } /** * @return the address of the owner. */ function owner() public view returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(isOwner()); _; } /** * @return true if `msg.sender` is the owner of the contract. */ function isOwner() public view returns (bool) { return msg.sender == _owner; } /** * @dev Allows the current owner to relinquish control of the contract. * It will not be possible to call the functions with the `onlyOwner` * modifier anymore. * @notice Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public onlyOwner { emit OwnershipTransferred(_owner, address(0)); _owner = address(0); } /** * @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; } } // File: contracts/common/misc/ProxyStorage.sol pragma solidity ^0.5.2; contract ProxyStorage is Ownable { address internal proxyTo; } // File: contracts/common/governance/IGovernance.sol pragma solidity ^0.5.2; interface IGovernance { function update(address target, bytes calldata data) external; } // File: contracts/common/governance/Governable.sol pragma solidity ^0.5.2; contract Governable { IGovernance public governance; constructor(address _governance) public { governance = IGovernance(_governance); } modifier onlyGovernance() { _assertGovernance(); _; } function _assertGovernance() private view { require( msg.sender == address(governance), "Only governance contract is authorized" ); } } // File: contracts/common/Registry.sol pragma solidity ^0.5.2; contract Registry is Governable { // @todo hardcode constants bytes32 private constant WETH_TOKEN = keccak256("wethToken"); bytes32 private constant DEPOSIT_MANAGER = keccak256("depositManager"); bytes32 private constant STAKE_MANAGER = keccak256("stakeManager"); bytes32 private constant VALIDATOR_SHARE = keccak256("validatorShare"); bytes32 private constant WITHDRAW_MANAGER = keccak256("withdrawManager"); bytes32 private constant CHILD_CHAIN = keccak256("childChain"); bytes32 private constant STATE_SENDER = keccak256("stateSender"); bytes32 private constant SLASHING_MANAGER = keccak256("slashingManager"); address public erc20Predicate; address public erc721Predicate; mapping(bytes32 => address) public contractMap; mapping(address => address) public rootToChildToken; mapping(address => address) public childToRootToken; mapping(address => bool) public proofValidatorContracts; mapping(address => bool) public isERC721; enum Type {Invalid, ERC20, ERC721, Custom} struct Predicate { Type _type; } mapping(address => Predicate) public predicates; event TokenMapped(address indexed rootToken, address indexed childToken); event ProofValidatorAdded(address indexed validator, address indexed from); event ProofValidatorRemoved(address indexed validator, address indexed from); event PredicateAdded(address indexed predicate, address indexed from); event PredicateRemoved(address indexed predicate, address indexed from); event ContractMapUpdated(bytes32 indexed key, address indexed previousContract, address indexed newContract); constructor(address _governance) public Governable(_governance) {} function updateContractMap(bytes32 _key, address _address) external onlyGovernance { emit ContractMapUpdated(_key, contractMap[_key], _address); contractMap[_key] = _address; } /** * @dev Map root token to child token * @param _rootToken Token address on the root chain * @param _childToken Token address on the child chain * @param _isERC721 Is the token being mapped ERC721 */ function mapToken( address _rootToken, address _childToken, bool _isERC721 ) external onlyGovernance { require(_rootToken != address(0x0) && _childToken != address(0x0), "INVALID_TOKEN_ADDRESS"); rootToChildToken[_rootToken] = _childToken; childToRootToken[_childToken] = _rootToken; isERC721[_rootToken] = _isERC721; IWithdrawManager(contractMap[WITHDRAW_MANAGER]).createExitQueue(_rootToken); emit TokenMapped(_rootToken, _childToken); } function addErc20Predicate(address predicate) public onlyGovernance { require(predicate != address(0x0), "Can not add null address as predicate"); erc20Predicate = predicate; addPredicate(predicate, Type.ERC20); } function addErc721Predicate(address predicate) public onlyGovernance { erc721Predicate = predicate; addPredicate(predicate, Type.ERC721); } function addPredicate(address predicate, Type _type) public onlyGovernance { require(predicates[predicate]._type == Type.Invalid, "Predicate already added"); predicates[predicate]._type = _type; emit PredicateAdded(predicate, msg.sender); } function removePredicate(address predicate) public onlyGovernance { require(predicates[predicate]._type != Type.Invalid, "Predicate does not exist"); delete predicates[predicate]; emit PredicateRemoved(predicate, msg.sender); } function getValidatorShareAddress() public view returns (address) { return contractMap[VALIDATOR_SHARE]; } function getWethTokenAddress() public view returns (address) { return contractMap[WETH_TOKEN]; } function getDepositManagerAddress() public view returns (address) { return contractMap[DEPOSIT_MANAGER]; } function getStakeManagerAddress() public view returns (address) { return contractMap[STAKE_MANAGER]; } function getSlashingManagerAddress() public view returns (address) { return contractMap[SLASHING_MANAGER]; } function getWithdrawManagerAddress() public view returns (address) { return contractMap[WITHDRAW_MANAGER]; } function getChildChainAndStateSender() public view returns (address, address) { return (contractMap[CHILD_CHAIN], contractMap[STATE_SENDER]); } function isTokenMapped(address _token) public view returns (bool) { return rootToChildToken[_token] != address(0x0); } function isTokenMappedAndIsErc721(address _token) public view returns (bool) { require(isTokenMapped(_token), "TOKEN_NOT_MAPPED"); return isERC721[_token]; } function isTokenMappedAndGetPredicate(address _token) public view returns (address) { if (isTokenMappedAndIsErc721(_token)) { return erc721Predicate; } return erc20Predicate; } function isChildTokenErc721(address childToken) public view returns (bool) { address rootToken = childToRootToken[childToken]; require(rootToken != address(0x0), "Child token is not mapped"); return isERC721[rootToken]; } } // File: contracts/common/mixin/ChainIdMixin.sol pragma solidity ^0.5.2; contract ChainIdMixin { bytes constant public networkId = hex"89"; uint256 constant public CHAINID = 137; } // File: contracts/root/RootChainStorage.sol pragma solidity ^0.5.2; contract RootChainHeader { event NewHeaderBlock( address indexed proposer, uint256 indexed headerBlockId, uint256 indexed reward, uint256 start, uint256 end, bytes32 root ); // housekeeping event event ResetHeaderBlock(address indexed proposer, uint256 indexed headerBlockId); struct HeaderBlock { bytes32 root; uint256 start; uint256 end; uint256 createdAt; address proposer; } } contract RootChainStorage is ProxyStorage, RootChainHeader, ChainIdMixin { bytes32 public heimdallId; uint8 public constant VOTE_TYPE = 2; uint16 internal constant MAX_DEPOSITS = 10000; uint256 public _nextHeaderBlock = MAX_DEPOSITS; uint256 internal _blockDepositId = 1; mapping(uint256 => HeaderBlock) public headerBlocks; Registry internal registry; } // File: contracts/staking/stakeManager/IStakeManager.sol pragma solidity 0.5.17; contract IStakeManager { // validator replacement function startAuction( uint256 validatorId, uint256 amount, bool acceptDelegation, bytes calldata signerPubkey ) external; function confirmAuctionBid(uint256 validatorId, uint256 heimdallFee) external; function transferFunds( uint256 validatorId, uint256 amount, address delegator ) external returns (bool); function delegationDeposit( uint256 validatorId, uint256 amount, address delegator ) external returns (bool); function unstake(uint256 validatorId) external; function totalStakedFor(address addr) external view returns (uint256); function stakeFor( address user, uint256 amount, uint256 heimdallFee, bool acceptDelegation, bytes memory signerPubkey ) public; function checkSignatures( uint256 blockInterval, bytes32 voteHash, bytes32 stateRoot, address proposer, uint[3][] calldata sigs ) external returns (uint256); function updateValidatorState(uint256 validatorId, int256 amount) public; function ownerOf(uint256 tokenId) public view returns (address); function slash(bytes calldata slashingInfoList) external returns (uint256); function validatorStake(uint256 validatorId) public view returns (uint256); function epoch() public view returns (uint256); function getRegistry() public view returns (address); function withdrawalDelay() public view returns (uint256); function delegatedAmount(uint256 validatorId) public view returns(uint256); function decreaseValidatorDelegatedAmount(uint256 validatorId, uint256 amount) public; function withdrawDelegatorsReward(uint256 validatorId) public returns(uint256); function delegatorsReward(uint256 validatorId) public view returns(uint256); function dethroneAndStake( address auctionUser, uint256 heimdallFee, uint256 validatorId, uint256 auctionAmount, bool acceptDelegation, bytes calldata signerPubkey ) external; } // File: contracts/root/IRootChain.sol pragma solidity ^0.5.2; interface IRootChain { function slash() external; function submitHeaderBlock(bytes calldata data, bytes calldata sigs) external; function submitCheckpoint(bytes calldata data, uint[3][] calldata sigs) external; function getLastChildBlock() external view returns (uint256); function currentHeaderBlock() external view returns (uint256); } // File: contracts/root/RootChain.sol pragma solidity ^0.5.2; contract RootChain is RootChainStorage, IRootChain { using SafeMath for uint256; using RLPReader for bytes; using RLPReader for RLPReader.RLPItem; modifier onlyDepositManager() { require(msg.sender == registry.getDepositManagerAddress(), "UNAUTHORIZED_DEPOSIT_MANAGER_ONLY"); _; } function submitHeaderBlock(bytes calldata data, bytes calldata sigs) external { revert(); } function submitCheckpoint(bytes calldata data, uint[3][] calldata sigs) external { (address proposer, uint256 start, uint256 end, bytes32 rootHash, bytes32 accountHash, uint256 _borChainID) = abi .decode(data, (address, uint256, uint256, bytes32, bytes32, uint256)); require(CHAINID == _borChainID, "Invalid bor chain id"); require(_buildHeaderBlock(proposer, start, end, rootHash), "INCORRECT_HEADER_DATA"); // check if it is better to keep it in local storage instead IStakeManager stakeManager = IStakeManager(registry.getStakeManagerAddress()); uint256 _reward = stakeManager.checkSignatures( end.sub(start).add(1), /** prefix 01 to data 01 represents positive vote on data and 00 is negative vote malicious validator can try to send 2/3 on negative vote so 01 is appended */ keccak256(abi.encodePacked(bytes(hex"01"), data)), accountHash, proposer, sigs ); require(_reward != 0, "Invalid checkpoint"); emit NewHeaderBlock(proposer, _nextHeaderBlock, _reward, start, end, rootHash); _nextHeaderBlock = _nextHeaderBlock.add(MAX_DEPOSITS); _blockDepositId = 1; } function updateDepositId(uint256 numDeposits) external onlyDepositManager returns (uint256 depositId) { depositId = currentHeaderBlock().add(_blockDepositId); // deposit ids will be (_blockDepositId, _blockDepositId + 1, .... _blockDepositId + numDeposits - 1) _blockDepositId = _blockDepositId.add(numDeposits); require( // Since _blockDepositId is initialized to 1; only (MAX_DEPOSITS - 1) deposits per header block are allowed _blockDepositId <= MAX_DEPOSITS, "TOO_MANY_DEPOSITS" ); } function getLastChildBlock() external view returns (uint256) { return headerBlocks[currentHeaderBlock()].end; } function slash() external { //TODO: future implementation } function currentHeaderBlock() public view returns (uint256) { return _nextHeaderBlock.sub(MAX_DEPOSITS); } function _buildHeaderBlock( address proposer, uint256 start, uint256 end, bytes32 rootHash ) private returns (bool) { uint256 nextChildBlock; /* The ID of the 1st header block is MAX_DEPOSITS. if _nextHeaderBlock == MAX_DEPOSITS, then the first header block is yet to be submitted, hence nextChildBlock = 0 */ if (_nextHeaderBlock > MAX_DEPOSITS) { nextChildBlock = headerBlocks[currentHeaderBlock()].end + 1; } if (nextChildBlock != start) { return false; } HeaderBlock memory headerBlock = HeaderBlock({ root: rootHash, start: nextChildBlock, end: end, createdAt: now, proposer: proposer }); headerBlocks[_nextHeaderBlock] = headerBlock; return true; } // Housekeeping function. @todo remove later function setNextHeaderBlock(uint256 _value) public onlyOwner { require(_value % MAX_DEPOSITS == 0, "Invalid value"); for (uint256 i = _value; i < _nextHeaderBlock; i += MAX_DEPOSITS) { delete headerBlocks[i]; } _nextHeaderBlock = _value; _blockDepositId = 1; emit ResetHeaderBlock(msg.sender, _nextHeaderBlock); } // Housekeeping function. @todo remove later function setHeimdallId(string memory _heimdallId) public onlyOwner { heimdallId = keccak256(abi.encodePacked(_heimdallId)); } } // File: openzeppelin-solidity/contracts/introspection/IERC165.sol pragma solidity ^0.5.2; /** * @title IERC165 * @dev https://eips.ethereum.org/EIPS/eip-165 */ interface IERC165 { /** * @notice Query if a contract implements an interface * @param interfaceId The interface identifier, as specified in ERC-165 * @dev Interface identification is specified in ERC-165. This function * uses less than 30,000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); } // File: openzeppelin-solidity/contracts/token/ERC721/IERC721.sol pragma solidity ^0.5.2; /** * @title ERC721 Non-Fungible Token Standard basic interface * @dev see https://eips.ethereum.org/EIPS/eip-721 */ contract IERC721 is IERC165 { event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); event ApprovalForAll(address indexed owner, address indexed operator, bool approved); function balanceOf(address owner) public view returns (uint256 balance); function ownerOf(uint256 tokenId) public view returns (address owner); function approve(address to, uint256 tokenId) public; function getApproved(uint256 tokenId) public view returns (address operator); function setApprovalForAll(address operator, bool _approved) public; function isApprovedForAll(address owner, address operator) public view returns (bool); function transferFrom(address from, address to, uint256 tokenId) public; function safeTransferFrom(address from, address to, uint256 tokenId) public; function safeTransferFrom(address from, address to, uint256 tokenId, bytes memory data) public; } // File: openzeppelin-solidity/contracts/token/ERC721/IERC721Receiver.sol pragma solidity ^0.5.2; /** * @title ERC721 token receiver interface * @dev Interface for any contract that wants to support safeTransfers * from ERC721 asset contracts. */ contract IERC721Receiver { /** * @notice Handle the receipt of an NFT * @dev The ERC721 smart contract calls this function on the recipient * after a `safeTransfer`. This function MUST return the function selector, * otherwise the caller will revert the transaction. The selector to be * returned can be obtained as `this.onERC721Received.selector`. This * function MAY throw to revert and reject the transfer. * Note: the ERC721 contract address is always the message sender. * @param operator The address which called `safeTransferFrom` function * @param from The address which previously owned the token * @param tokenId The NFT identifier which is being transferred * @param data Additional data with no specified format * @return bytes4 `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))` */ function onERC721Received(address operator, address from, uint256 tokenId, bytes memory data) public returns (bytes4); } // File: openzeppelin-solidity/contracts/utils/Address.sol pragma solidity ^0.5.2; /** * Utility library of inline functions on addresses */ library Address { /** * Returns whether the target address is a contract * @dev This function will return false if invoked during the constructor of a contract, * as the code is not actually created until after the constructor finishes. * @param account address of the account to check * @return whether the target address is a contract */ function isContract(address account) internal view returns (bool) { uint256 size; // XXX Currently there is no better way to check if there is a contract in an address // than to check the size of the code at that address. // See https://ethereum.stackexchange.com/a/14016/36603 // for more details about how this works. // TODO Check this again before the Serenity release, because all addresses will be // contracts then. // solhint-disable-next-line no-inline-assembly assembly { size := extcodesize(account) } return size > 0; } } // File: openzeppelin-solidity/contracts/drafts/Counters.sol pragma solidity ^0.5.2; /** * @title Counters * @author Matt Condon (@shrugs) * @dev Provides counters that can only be incremented or decremented by one. This can be used e.g. to track the number * of elements in a mapping, issuing ERC721 ids, or counting request ids * * Include with `using Counters for Counters.Counter;` * Since it is not possible to overflow a 256 bit integer with increments of one, `increment` can skip the SafeMath * overflow check, thereby saving gas. This does assume however correct usage, in that the underlying `_value` is never * directly accessed. */ library Counters { using SafeMath for uint256; struct Counter { // This variable should never be directly accessed by users of the library: interactions must be restricted to // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add // this feature: see https://github.com/ethereum/solidity/issues/4637 uint256 _value; // default: 0 } function current(Counter storage counter) internal view returns (uint256) { return counter._value; } function increment(Counter storage counter) internal { counter._value += 1; } function decrement(Counter storage counter) internal { counter._value = counter._value.sub(1); } } // File: openzeppelin-solidity/contracts/introspection/ERC165.sol pragma solidity ^0.5.2; /** * @title ERC165 * @author Matt Condon (@shrugs) * @dev Implements ERC165 using a lookup table. */ contract ERC165 is IERC165 { bytes4 private constant _INTERFACE_ID_ERC165 = 0x01ffc9a7; /* * 0x01ffc9a7 === * bytes4(keccak256('supportsInterface(bytes4)')) */ /** * @dev a mapping of interface id to whether or not it's supported */ mapping(bytes4 => bool) private _supportedInterfaces; /** * @dev A contract implementing SupportsInterfaceWithLookup * implement ERC165 itself */ constructor () internal { _registerInterface(_INTERFACE_ID_ERC165); } /** * @dev implement supportsInterface(bytes4) using a lookup table */ function supportsInterface(bytes4 interfaceId) external view returns (bool) { return _supportedInterfaces[interfaceId]; } /** * @dev internal method for registering an interface */ function _registerInterface(bytes4 interfaceId) internal { require(interfaceId != 0xffffffff); _supportedInterfaces[interfaceId] = true; } } // File: openzeppelin-solidity/contracts/token/ERC721/ERC721.sol pragma solidity ^0.5.2; /** * @title ERC721 Non-Fungible Token Standard basic implementation * @dev see https://eips.ethereum.org/EIPS/eip-721 */ contract ERC721 is ERC165, IERC721 { using SafeMath for uint256; using Address for address; using Counters for Counters.Counter; // Equals to `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))` // which can be also obtained as `IERC721Receiver(0).onERC721Received.selector` bytes4 private constant _ERC721_RECEIVED = 0x150b7a02; // Mapping from token ID to owner mapping (uint256 => address) private _tokenOwner; // Mapping from token ID to approved address mapping (uint256 => address) private _tokenApprovals; // Mapping from owner to number of owned token mapping (address => Counters.Counter) private _ownedTokensCount; // Mapping from owner to operator approvals mapping (address => mapping (address => bool)) private _operatorApprovals; bytes4 private constant _INTERFACE_ID_ERC721 = 0x80ac58cd; /* * 0x80ac58cd === * bytes4(keccak256('balanceOf(address)')) ^ * bytes4(keccak256('ownerOf(uint256)')) ^ * bytes4(keccak256('approve(address,uint256)')) ^ * bytes4(keccak256('getApproved(uint256)')) ^ * bytes4(keccak256('setApprovalForAll(address,bool)')) ^ * bytes4(keccak256('isApprovedForAll(address,address)')) ^ * bytes4(keccak256('transferFrom(address,address,uint256)')) ^ * bytes4(keccak256('safeTransferFrom(address,address,uint256)')) ^ * bytes4(keccak256('safeTransferFrom(address,address,uint256,bytes)')) */ constructor () public { // register the supported interfaces to conform to ERC721 via ERC165 _registerInterface(_INTERFACE_ID_ERC721); } /** * @dev Gets the balance of the specified address * @param owner address to query the balance of * @return uint256 representing the amount owned by the passed address */ function balanceOf(address owner) public view returns (uint256) { require(owner != address(0)); return _ownedTokensCount[owner].current(); } /** * @dev Gets the owner of the specified token ID * @param tokenId uint256 ID of the token to query the owner of * @return address currently marked as the owner of the given token ID */ function ownerOf(uint256 tokenId) public view returns (address) { address owner = _tokenOwner[tokenId]; require(owner != address(0)); return owner; } /** * @dev Approves another address to transfer the given token ID * The zero address indicates there is no approved address. * There can only be one approved address per token at a given time. * Can only be called by the token owner or an approved operator. * @param to address to be approved for the given token ID * @param tokenId uint256 ID of the token to be approved */ function approve(address to, uint256 tokenId) public { address owner = ownerOf(tokenId); require(to != owner); require(msg.sender == owner || isApprovedForAll(owner, msg.sender)); _tokenApprovals[tokenId] = to; emit Approval(owner, to, tokenId); } /** * @dev Gets the approved address for a token ID, or zero if no address set * Reverts if the token ID does not exist. * @param tokenId uint256 ID of the token to query the approval of * @return address currently approved for the given token ID */ function getApproved(uint256 tokenId) public view returns (address) { require(_exists(tokenId)); return _tokenApprovals[tokenId]; } /** * @dev Sets or unsets the approval of a given operator * An operator is allowed to transfer all tokens of the sender on their behalf * @param to operator address to set the approval * @param approved representing the status of the approval to be set */ function setApprovalForAll(address to, bool approved) public { require(to != msg.sender); _operatorApprovals[msg.sender][to] = approved; emit ApprovalForAll(msg.sender, to, approved); } /** * @dev Tells whether an operator is approved by a given owner * @param owner owner address which you want to query the approval of * @param operator operator address which you want to query the approval of * @return bool whether the given operator is approved by the given owner */ function isApprovedForAll(address owner, address operator) public view returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev Transfers the ownership of a given token ID to another address * Usage of this method is discouraged, use `safeTransferFrom` whenever possible * Requires the msg.sender to be the owner, approved, or operator * @param from current owner of the token * @param to address to receive the ownership of the given token ID * @param tokenId uint256 ID of the token to be transferred */ function transferFrom(address from, address to, uint256 tokenId) public { require(_isApprovedOrOwner(msg.sender, tokenId)); _transferFrom(from, to, tokenId); } /** * @dev Safely transfers the ownership of a given token ID to another address * If the target address is a contract, it must implement `onERC721Received`, * which is called upon a safe transfer, and return the magic value * `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`; otherwise, * the transfer is reverted. * Requires the msg.sender to be the owner, approved, or operator * @param from current owner of the token * @param to address to receive the ownership of the given token ID * @param tokenId uint256 ID of the token to be transferred */ function safeTransferFrom(address from, address to, uint256 tokenId) public { safeTransferFrom(from, to, tokenId, ""); } /** * @dev Safely transfers the ownership of a given token ID to another address * If the target address is a contract, it must implement `onERC721Received`, * which is called upon a safe transfer, and return the magic value * `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`; otherwise, * the transfer is reverted. * Requires the msg.sender to be the owner, approved, or operator * @param from current owner of the token * @param to address to receive the ownership of the given token ID * @param tokenId uint256 ID of the token to be transferred * @param _data bytes data to send along with a safe transfer check */ function safeTransferFrom(address from, address to, uint256 tokenId, bytes memory _data) public { transferFrom(from, to, tokenId); require(_checkOnERC721Received(from, to, tokenId, _data)); } /** * @dev Returns whether the specified token exists * @param tokenId uint256 ID of the token to query the existence of * @return bool whether the token exists */ function _exists(uint256 tokenId) internal view returns (bool) { address owner = _tokenOwner[tokenId]; return owner != address(0); } /** * @dev Returns whether the given spender can transfer a given token ID * @param spender address of the spender to query * @param tokenId uint256 ID of the token to be transferred * @return bool whether the msg.sender is approved for the given token ID, * is an operator of the owner, or is the owner of the token */ function _isApprovedOrOwner(address spender, uint256 tokenId) internal view returns (bool) { address owner = ownerOf(tokenId); return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender)); } /** * @dev Internal function to mint a new token * Reverts if the given token ID already exists * @param to The address that will own the minted token * @param tokenId uint256 ID of the token to be minted */ function _mint(address to, uint256 tokenId) internal { require(to != address(0)); require(!_exists(tokenId)); _tokenOwner[tokenId] = to; _ownedTokensCount[to].increment(); emit Transfer(address(0), to, tokenId); } /** * @dev Internal function to burn a specific token * Reverts if the token does not exist * Deprecated, use _burn(uint256) instead. * @param owner owner of the token to burn * @param tokenId uint256 ID of the token being burned */ function _burn(address owner, uint256 tokenId) internal { require(ownerOf(tokenId) == owner); _clearApproval(tokenId); _ownedTokensCount[owner].decrement(); _tokenOwner[tokenId] = address(0); emit Transfer(owner, address(0), tokenId); } /** * @dev Internal function to burn a specific token * Reverts if the token does not exist * @param tokenId uint256 ID of the token being burned */ function _burn(uint256 tokenId) internal { _burn(ownerOf(tokenId), tokenId); } /** * @dev Internal function to transfer ownership of a given token ID to another address. * As opposed to transferFrom, this imposes no restrictions on msg.sender. * @param from current owner of the token * @param to address to receive the ownership of the given token ID * @param tokenId uint256 ID of the token to be transferred */ function _transferFrom(address from, address to, uint256 tokenId) internal { require(ownerOf(tokenId) == from); require(to != address(0)); _clearApproval(tokenId); _ownedTokensCount[from].decrement(); _ownedTokensCount[to].increment(); _tokenOwner[tokenId] = to; emit Transfer(from, to, tokenId); } /** * @dev Internal function to invoke `onERC721Received` on a target address * The call is not executed if the target address is not a contract * @param from address representing the previous owner of the given token ID * @param to target address that will receive the tokens * @param tokenId uint256 ID of the token to be transferred * @param _data bytes optional data to send along with the call * @return bool whether the call correctly returned the expected magic value */ function _checkOnERC721Received(address from, address to, uint256 tokenId, bytes memory _data) internal returns (bool) { if (!to.isContract()) { return true; } bytes4 retval = IERC721Receiver(to).onERC721Received(msg.sender, from, tokenId, _data); return (retval == _ERC721_RECEIVED); } /** * @dev Private function to clear current approval of a given token ID * @param tokenId uint256 ID of the token to be transferred */ function _clearApproval(uint256 tokenId) private { if (_tokenApprovals[tokenId] != address(0)) { _tokenApprovals[tokenId] = address(0); } } } // File: contracts/root/withdrawManager/ExitNFT.sol pragma solidity ^0.5.2; contract ExitNFT is ERC721 { Registry internal registry; modifier onlyWithdrawManager() { require( msg.sender == registry.getWithdrawManagerAddress(), "UNAUTHORIZED_WITHDRAW_MANAGER_ONLY" ); _; } constructor(address _registry) public { registry = Registry(_registry); } function mint(address _owner, uint256 _tokenId) external onlyWithdrawManager { _mint(_owner, _tokenId); } function burn(uint256 _tokenId) external onlyWithdrawManager { _burn(_tokenId); } function exists(uint256 tokenId) public view returns (bool) { return _exists(tokenId); } } // File: contracts/root/withdrawManager/WithdrawManagerStorage.sol pragma solidity ^0.5.2; contract ExitsDataStructure { struct Input { address utxoOwner; address predicate; address token; } struct PlasmaExit { uint256 receiptAmountOrNFTId; bytes32 txHash; address owner; address token; bool isRegularExit; address predicate; // Mapping from age of input to Input mapping(uint256 => Input) inputs; } } contract WithdrawManagerHeader is ExitsDataStructure { event Withdraw(uint256 indexed exitId, address indexed user, address indexed token, uint256 amount); event ExitStarted( address indexed exitor, uint256 indexed exitId, address indexed token, uint256 amount, bool isRegularExit ); event ExitUpdated(uint256 indexed exitId, uint256 indexed age, address signer); event ExitPeriodUpdate(uint256 indexed oldExitPeriod, uint256 indexed newExitPeriod); event ExitCancelled(uint256 indexed exitId); } contract WithdrawManagerStorage is ProxyStorage, WithdrawManagerHeader { // 0.5 week = 7 * 86400 / 2 = 302400 uint256 public HALF_EXIT_PERIOD = 302400; // Bonded exits collaterized at 0.1 ETH uint256 internal constant BOND_AMOUNT = 10**17; Registry internal registry; RootChain internal rootChain; mapping(uint128 => bool) isKnownExit; mapping(uint256 => PlasmaExit) public exits; // mapping with token => (owner => exitId) keccak(token+owner) keccak(token+owner+tokenId) mapping(bytes32 => uint256) public ownerExits; mapping(address => address) public exitsQueues; ExitNFT public exitNft; // ERC721, ERC20 and Weth transfers require 155000, 100000, 52000 gas respectively // Processing each exit in a while loop iteration requires ~52000 gas (@todo check if this changed) // uint32 constant internal ITERATION_GAS = 52000; // So putting an upper limit of 155000 + 52000 + leeway uint32 public ON_FINALIZE_GAS_LIMIT = 300000; uint256 public exitWindow; } // File: contracts/root/predicates/IPredicate.sol pragma solidity ^0.5.2; interface IPredicate { /** * @notice Verify the deprecation of a state update * @param exit ABI encoded PlasmaExit data * @param inputUtxo ABI encoded Input UTXO data * @param challengeData RLP encoded data of the challenge reference tx that encodes the following fields * headerNumber Header block number of which the reference tx was a part of * blockProof Proof that the block header (in the child chain) is a leaf in the submitted merkle root * blockNumber Block number of which the reference tx is a part of * blockTime Reference tx block time * blocktxRoot Transactions root of block * blockReceiptsRoot Receipts root of block * receipt Receipt of the reference transaction * receiptProof Merkle proof of the reference receipt * branchMask Merkle proof branchMask for the receipt * logIndex Log Index to read from the receipt * tx Challenge transaction * txProof Merkle proof of the challenge tx * @return Whether or not the state is deprecated */ function verifyDeprecation( bytes calldata exit, bytes calldata inputUtxo, bytes calldata challengeData ) external returns (bool); function interpretStateUpdate(bytes calldata state) external view returns (bytes memory); function onFinalizeExit(bytes calldata data) external; } contract PredicateUtils is ExitsDataStructure, ChainIdMixin { using RLPReader for RLPReader.RLPItem; // Bonded exits collaterized at 0.1 ETH uint256 private constant BOND_AMOUNT = 10**17; IWithdrawManager internal withdrawManager; IDepositManager internal depositManager; modifier onlyWithdrawManager() { require( msg.sender == address(withdrawManager), "ONLY_WITHDRAW_MANAGER" ); _; } modifier isBondProvided() { require(msg.value == BOND_AMOUNT, "Invalid Bond amount"); _; } function onFinalizeExit(bytes calldata data) external onlyWithdrawManager { (, address token, address exitor, uint256 tokenId) = decodeExitForProcessExit( data ); depositManager.transferAssets(token, exitor, tokenId); } function sendBond() internal { address(uint160(address(withdrawManager))).transfer(BOND_AMOUNT); } function getAddressFromTx(RLPReader.RLPItem[] memory txList) internal pure returns (address signer, bytes32 txHash) { bytes[] memory rawTx = new bytes[](9); for (uint8 i = 0; i <= 5; i++) { rawTx[i] = txList[i].toBytes(); } rawTx[6] = networkId; rawTx[7] = hex""; // [7] and [8] have something to do with v, r, s values rawTx[8] = hex""; txHash = keccak256(RLPEncode.encodeList(rawTx)); signer = ecrecover( txHash, Common.getV(txList[6].toBytes(), Common.toUint16(networkId)), bytes32(txList[7].toUint()), bytes32(txList[8].toUint()) ); } function decodeExit(bytes memory data) internal pure returns (PlasmaExit memory) { (address owner, address token, uint256 amountOrTokenId, bytes32 txHash, bool isRegularExit) = abi .decode(data, (address, address, uint256, bytes32, bool)); return PlasmaExit( amountOrTokenId, txHash, owner, token, isRegularExit, address(0) /* predicate value is not required */ ); } function decodeExitForProcessExit(bytes memory data) internal pure returns (uint256 exitId, address token, address exitor, uint256 tokenId) { (exitId, token, exitor, tokenId) = abi.decode( data, (uint256, address, address, uint256) ); } function decodeInputUtxo(bytes memory data) internal pure returns (uint256 age, address signer, address predicate, address token) { (age, signer, predicate, token) = abi.decode( data, (uint256, address, address, address) ); } } contract IErcPredicate is IPredicate, PredicateUtils { enum ExitType {Invalid, OutgoingTransfer, IncomingTransfer, Burnt} struct ExitTxData { uint256 amountOrToken; bytes32 txHash; address childToken; address signer; ExitType exitType; } struct ReferenceTxData { uint256 closingBalance; uint256 age; address childToken; address rootToken; } uint256 internal constant MAX_LOGS = 10; constructor(address _withdrawManager, address _depositManager) public { withdrawManager = IWithdrawManager(_withdrawManager); depositManager = IDepositManager(_depositManager); } } // File: contracts/root/predicates/ERC20PredicateBurnOnly.sol pragma solidity ^0.5.2; contract ERC20PredicateBurnOnly is IErcPredicate { using RLPReader for bytes; using RLPReader for RLPReader.RLPItem; using SafeMath for uint256; // keccak256('Withdraw(address,address,uint256,uint256,uint256)') bytes32 constant WITHDRAW_EVENT_SIG = 0xebff2602b3f468259e1e99f613fed6691f3a6526effe6ef3e768ba7ae7a36c4f; Registry registry; constructor( address _withdrawManager, address _depositManager, address _registry ) public IErcPredicate(_withdrawManager, _depositManager) { registry = Registry(_registry); } function startExitWithBurntTokens(bytes calldata data) external { RLPReader.RLPItem[] memory referenceTxData = data.toRlpItem().toList(); bytes memory receipt = referenceTxData[6].toBytes(); RLPReader.RLPItem[] memory inputItems = receipt.toRlpItem().toList(); uint256 logIndex = referenceTxData[9].toUint(); require(logIndex < MAX_LOGS, "Supporting a max of 10 logs"); uint256 age = withdrawManager.verifyInclusion( data, 0, /* offset */ false /* verifyTxInclusion */ ); inputItems = inputItems[3].toList()[logIndex].toList(); // select log based on given logIndex // "address" (contract address that emitted the log) field in the receipt address childToken = RLPReader.toAddress(inputItems[0]); bytes memory logData = inputItems[2].toBytes(); inputItems = inputItems[1].toList(); // topics // now, inputItems[i] refers to i-th (0-based) topic in the topics array // event Withdraw(address indexed token, address indexed from, uint256 amountOrTokenId, uint256 input1, uint256 output1) require( bytes32(inputItems[0].toUint()) == WITHDRAW_EVENT_SIG, "Not a withdraw event signature" ); address rootToken = address(RLPReader.toUint(inputItems[1])); require( msg.sender == address(inputItems[2].toUint()), // from "Withdrawer and burn exit tx do not match" ); uint256 exitAmount = BytesLib.toUint(logData, 0); // amountOrTokenId withdrawManager.addExitToQueue( msg.sender, childToken, rootToken, exitAmount, bytes32(0x0), true, /* isRegularExit */ age << 1 ); } function verifyDeprecation( bytes calldata exit, bytes calldata inputUtxo, bytes calldata challengeData ) external returns (bool) {} function interpretStateUpdate(bytes calldata state) external view returns (bytes memory) {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_withdrawManager","type":"address"},{"internalType":"address","name":"_depositManager","type":"address"},{"internalType":"address","name":"_registry","type":"address"}],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"constant":true,"inputs":[],"name":"CHAINID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"bytes","name":"state","type":"bytes"}],"name":"interpretStateUpdate","outputs":[{"internalType":"bytes","name":"","type":"bytes"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"networkId","outputs":[{"internalType":"bytes","name":"","type":"bytes"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"internalType":"bytes","name":"data","type":"bytes"}],"name":"onFinalizeExit","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"bytes","name":"data","type":"bytes"}],"name":"startExitWithBurntTokens","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"bytes","name":"exit","type":"bytes"},{"internalType":"bytes","name":"inputUtxo","type":"bytes"},{"internalType":"bytes","name":"challengeData","type":"bytes"}],"name":"verifyDeprecation","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
608060405234801561001057600080fd5b50604051610db6380380610db68339818101604052606081101561003357600080fd5b5080516020820151604090920151600080546001600160a01b039384166001600160a01b0319918216179091556001805494841694821694909417909355600280549290911691909216179055610d278061008f6000396000f3fe608060405234801561001057600080fd5b50600436106100625760003560e01c80637bd94e03146100675780637c5264b4146100d757806382e3464c146101455780639025e64c14610228578063cc79f97b14610230578063ec58410c1461024a575b600080fd5b6100d56004803603602081101561007d57600080fd5b810190602081018135600160201b81111561009757600080fd5b8201836020820111156100a957600080fd5b803590602001918460018302840111600160201b831117156100ca57600080fd5b50909250905061036c565b005b6100d5600480360360208110156100ed57600080fd5b810190602081018135600160201b81111561010757600080fd5b82018360208201111561011957600080fd5b803590602001918460018302840111600160201b8311171561013a57600080fd5b50909250905061048a565b6101b36004803603602081101561015b57600080fd5b810190602081018135600160201b81111561017557600080fd5b82018360208201111561018757600080fd5b803590602001918460018302840111600160201b831117156101a857600080fd5b509092509050610853565b6040805160208082528351818301528351919283929083019185019080838360005b838110156101ed5781810151838201526020016101d5565b50505050905090810190601f16801561021a5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6101b361085b565b610238610878565b60408051918252519081900360200190f35b6103586004803603606081101561026057600080fd5b810190602081018135600160201b81111561027a57600080fd5b82018360208201111561028c57600080fd5b803590602001918460018302840111600160201b831117156102ad57600080fd5b919390929091602081019035600160201b8111156102ca57600080fd5b8201836020820111156102dc57600080fd5b803590602001918460018302840111600160201b831117156102fd57600080fd5b919390929091602081019035600160201b81111561031a57600080fd5b82018360208201111561032c57600080fd5b803590602001918460018302840111600160201b8311171561034d57600080fd5b50909250905061087d565b604080519115158252519081900360200190f35b6000546001600160a01b031633146103c3576040805162461bcd60e51b815260206004820152601560248201527427a7262cafaba4aa24222920abafa6a0a720a3a2a960591b604482015290519081900360640190fd5b600080600061040785858080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061088992505050565b600154604080516349f4cc1760e01b81526001600160a01b03808716600483015280861660248301526044820185905291519599509397509195501692506349f4cc1791606480830192600092919082900301818387803b15801561046b57600080fd5b505af115801561047f573d6000803e3d6000fd5b505050505050505050565b60606104d36104ce84848080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152506108c692505050565b6108ec565b905060606104f4826006815181106104e757fe5b60200260200101516109bd565b905060606105046104ce836108c6565b905060006105258460098151811061051857fe5b6020026020010151610a23565b9050600a811061057c576040805162461bcd60e51b815260206004820152601b60248201527f537570706f7274696e672061206d6178206f66203130206c6f67730000000000604482015290519081900360640190fd5b6000805460405163ad1d806960e01b81526024810183905260448101839052606060048201908152606482018990526001600160a01b039092169163ad1d8069918a918a91869182918190608401868680828437600081840152601f19601f8201169050808301925050509550505050505060206040518083038186803b15801561060657600080fd5b505afa15801561061a573d6000803e3d6000fd5b505050506040513d602081101561063057600080fd5b5051835190915061066490610659908590600390811061064c57fe5b60200260200101516108ec565b838151811061064c57fe5b925060006106858460008151811061067857fe5b6020026020010151610a71565b90506060610699856002815181106104e757fe5b90506106ab8560018151811061064c57fe5b94507febff2602b3f468259e1e99f613fed6691f3a6526effe6ef3e768ba7ae7a36c4f60001b6106e18660008151811061051857fe5b14610733576040805162461bcd60e51b815260206004820152601e60248201527f4e6f742061207769746864726177206576656e74207369676e61747572650000604482015290519081900360640190fd5b60006107458660018151811061051857fe5b90506107578660028151811061051857fe5b6001600160a01b0316336001600160a01b0316146107a65760405162461bcd60e51b8152600401808060200182810382526028815260200180610ccb6028913960400191505060405180910390fd5b60006107b3836000610a91565b600080546040805163d931a86960e01b81523360048201526001600160a01b03898116602483015287811660448301526064820186905260848201859052600160a483018190528b901b60c4830152915194955091169263d931a8699260e48084019391929182900301818387803b15801561082e57600080fd5b505af1158015610842573d6000803e3d6000fd5b505050505050505050505050505050565b606092915050565b604051806040016040528060018152602001608960f81b81525081565b608981565b60009695505050505050565b6000806000808480602001905160808110156108a457600080fd5b5080516020820151604083015160609093015191989097509195509350915050565b6108ce610cb0565b5060408051808201909152815181526020828101908201525b919050565b60606108f782610aad565b61090057600080fd5b600061090b83610ae7565b905060608160405190808252806020026020018201604052801561094957816020015b610936610cb0565b81526020019060019003908161092e5790505b509050600061095b8560200151610b43565b60208601510190506000805b848110156109b25761097883610ba6565b915060405180604001604052808381526020018481525084828151811061099b57fe5b602090810291909101015291810191600101610967565b509195945050505050565b80516060906109cb57600080fd5b6000806109d784610c3f565b915091506060816040519080825280601f01601f191660200182016040528015610a08576020820181803883390190505b50905060208101610a1a848285610c65565b50949350505050565b805160009015801590610a3857508151602110155b610a4157600080fd5b600080610a4d84610c3f565b815191935091506020821015610a695760208290036101000a90045b949350505050565b8051600090601514610a8257600080fd5b610a8b82610a23565b92915050565b60008160200183511015610aa457600080fd5b50016020015190565b8051600090610abe575060006108e7565b6020820151805160001a9060c0821015610add576000925050506108e7565b5060019392505050565b8051600090610af8575060006108e7565b60008090506000610b0c8460200151610b43565b602085015185519181019250015b80821015610b3a57610b2b82610ba6565b60019093019290910190610b1a565b50909392505050565b8051600090811a6080811015610b5d5760009150506108e7565b60b8811080610b78575060c08110801590610b78575060f881105b15610b875760019150506108e7565b60c0811015610b9b5760b5190190506108e7565b60f5190190506108e7565b80516000908190811a6080811015610bc15760019150610c38565b60b8811015610bd657607e1981019150610c38565b60c0811015610c035760b78103600185019450806020036101000a85510460018201810193505050610c38565b60f8811015610c185760be1981019150610c38565b60f78103600185019450806020036101000a855104600182018101935050505b5092915050565b6000806000610c518460200151610b43565b602085015194519481019594039392505050565b80610c6f57610cab565b5b60208110610c8f578251825260209283019290910190601f1901610c70565b8251825160208390036101000a60001901801990921691161782525b505050565b60405180604001604052806000815260200160008152509056fe5769746864726177657220616e64206275726e206578697420747820646f206e6f74206d61746368a265627a7a72315820642f27a7dd308a6120ebc0d9fe1908bc2a25fa88c9dba93a98b247e08b5f4b9d64736f6c634300051100320000000000000000000000002a88696e0ffa76baa1338f2c74497cc013495922000000000000000000000000401f6c983ea34274ec46f84d70b31c151321188b00000000000000000000000033a02e6cc863d393d6bf231b697b82f6e499ca71
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106100625760003560e01c80637bd94e03146100675780637c5264b4146100d757806382e3464c146101455780639025e64c14610228578063cc79f97b14610230578063ec58410c1461024a575b600080fd5b6100d56004803603602081101561007d57600080fd5b810190602081018135600160201b81111561009757600080fd5b8201836020820111156100a957600080fd5b803590602001918460018302840111600160201b831117156100ca57600080fd5b50909250905061036c565b005b6100d5600480360360208110156100ed57600080fd5b810190602081018135600160201b81111561010757600080fd5b82018360208201111561011957600080fd5b803590602001918460018302840111600160201b8311171561013a57600080fd5b50909250905061048a565b6101b36004803603602081101561015b57600080fd5b810190602081018135600160201b81111561017557600080fd5b82018360208201111561018757600080fd5b803590602001918460018302840111600160201b831117156101a857600080fd5b509092509050610853565b6040805160208082528351818301528351919283929083019185019080838360005b838110156101ed5781810151838201526020016101d5565b50505050905090810190601f16801561021a5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6101b361085b565b610238610878565b60408051918252519081900360200190f35b6103586004803603606081101561026057600080fd5b810190602081018135600160201b81111561027a57600080fd5b82018360208201111561028c57600080fd5b803590602001918460018302840111600160201b831117156102ad57600080fd5b919390929091602081019035600160201b8111156102ca57600080fd5b8201836020820111156102dc57600080fd5b803590602001918460018302840111600160201b831117156102fd57600080fd5b919390929091602081019035600160201b81111561031a57600080fd5b82018360208201111561032c57600080fd5b803590602001918460018302840111600160201b8311171561034d57600080fd5b50909250905061087d565b604080519115158252519081900360200190f35b6000546001600160a01b031633146103c3576040805162461bcd60e51b815260206004820152601560248201527427a7262cafaba4aa24222920abafa6a0a720a3a2a960591b604482015290519081900360640190fd5b600080600061040785858080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061088992505050565b600154604080516349f4cc1760e01b81526001600160a01b03808716600483015280861660248301526044820185905291519599509397509195501692506349f4cc1791606480830192600092919082900301818387803b15801561046b57600080fd5b505af115801561047f573d6000803e3d6000fd5b505050505050505050565b60606104d36104ce84848080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152506108c692505050565b6108ec565b905060606104f4826006815181106104e757fe5b60200260200101516109bd565b905060606105046104ce836108c6565b905060006105258460098151811061051857fe5b6020026020010151610a23565b9050600a811061057c576040805162461bcd60e51b815260206004820152601b60248201527f537570706f7274696e672061206d6178206f66203130206c6f67730000000000604482015290519081900360640190fd5b6000805460405163ad1d806960e01b81526024810183905260448101839052606060048201908152606482018990526001600160a01b039092169163ad1d8069918a918a91869182918190608401868680828437600081840152601f19601f8201169050808301925050509550505050505060206040518083038186803b15801561060657600080fd5b505afa15801561061a573d6000803e3d6000fd5b505050506040513d602081101561063057600080fd5b5051835190915061066490610659908590600390811061064c57fe5b60200260200101516108ec565b838151811061064c57fe5b925060006106858460008151811061067857fe5b6020026020010151610a71565b90506060610699856002815181106104e757fe5b90506106ab8560018151811061064c57fe5b94507febff2602b3f468259e1e99f613fed6691f3a6526effe6ef3e768ba7ae7a36c4f60001b6106e18660008151811061051857fe5b14610733576040805162461bcd60e51b815260206004820152601e60248201527f4e6f742061207769746864726177206576656e74207369676e61747572650000604482015290519081900360640190fd5b60006107458660018151811061051857fe5b90506107578660028151811061051857fe5b6001600160a01b0316336001600160a01b0316146107a65760405162461bcd60e51b8152600401808060200182810382526028815260200180610ccb6028913960400191505060405180910390fd5b60006107b3836000610a91565b600080546040805163d931a86960e01b81523360048201526001600160a01b03898116602483015287811660448301526064820186905260848201859052600160a483018190528b901b60c4830152915194955091169263d931a8699260e48084019391929182900301818387803b15801561082e57600080fd5b505af1158015610842573d6000803e3d6000fd5b505050505050505050505050505050565b606092915050565b604051806040016040528060018152602001608960f81b81525081565b608981565b60009695505050505050565b6000806000808480602001905160808110156108a457600080fd5b5080516020820151604083015160609093015191989097509195509350915050565b6108ce610cb0565b5060408051808201909152815181526020828101908201525b919050565b60606108f782610aad565b61090057600080fd5b600061090b83610ae7565b905060608160405190808252806020026020018201604052801561094957816020015b610936610cb0565b81526020019060019003908161092e5790505b509050600061095b8560200151610b43565b60208601510190506000805b848110156109b25761097883610ba6565b915060405180604001604052808381526020018481525084828151811061099b57fe5b602090810291909101015291810191600101610967565b509195945050505050565b80516060906109cb57600080fd5b6000806109d784610c3f565b915091506060816040519080825280601f01601f191660200182016040528015610a08576020820181803883390190505b50905060208101610a1a848285610c65565b50949350505050565b805160009015801590610a3857508151602110155b610a4157600080fd5b600080610a4d84610c3f565b815191935091506020821015610a695760208290036101000a90045b949350505050565b8051600090601514610a8257600080fd5b610a8b82610a23565b92915050565b60008160200183511015610aa457600080fd5b50016020015190565b8051600090610abe575060006108e7565b6020820151805160001a9060c0821015610add576000925050506108e7565b5060019392505050565b8051600090610af8575060006108e7565b60008090506000610b0c8460200151610b43565b602085015185519181019250015b80821015610b3a57610b2b82610ba6565b60019093019290910190610b1a565b50909392505050565b8051600090811a6080811015610b5d5760009150506108e7565b60b8811080610b78575060c08110801590610b78575060f881105b15610b875760019150506108e7565b60c0811015610b9b5760b5190190506108e7565b60f5190190506108e7565b80516000908190811a6080811015610bc15760019150610c38565b60b8811015610bd657607e1981019150610c38565b60c0811015610c035760b78103600185019450806020036101000a85510460018201810193505050610c38565b60f8811015610c185760be1981019150610c38565b60f78103600185019450806020036101000a855104600182018101935050505b5092915050565b6000806000610c518460200151610b43565b602085015194519481019594039392505050565b80610c6f57610cab565b5b60208110610c8f578251825260209283019290910190601f1901610c70565b8251825160208390036101000a60001901801990921691161782525b505050565b60405180604001604052806000815260200160008152509056fe5769746864726177657220616e64206275726e206578697420747820646f206e6f74206d61746368a265627a7a72315820642f27a7dd308a6120ebc0d9fe1908bc2a25fa88c9dba93a98b247e08b5f4b9d64736f6c63430005110032
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
0000000000000000000000002a88696e0ffa76baa1338f2c74497cc013495922000000000000000000000000401f6c983ea34274ec46f84d70b31c151321188b00000000000000000000000033a02e6cc863d393d6bf231b697b82f6e499ca71
-----Decoded View---------------
Arg [0] : _withdrawManager (address): 0x2A88696e0fFA76bAA1338F2C74497cC013495922
Arg [1] : _depositManager (address): 0x401F6c983eA34274ec46f84D70b31C151321188b
Arg [2] : _registry (address): 0x33a02E6cC863D393d6Bf231B697b82F6e499cA71
-----Encoded View---------------
3 Constructor Arguments found :
Arg [0] : 0000000000000000000000002a88696e0ffa76baa1338f2c74497cc013495922
Arg [1] : 000000000000000000000000401f6c983ea34274ec46f84d70b31c151321188b
Arg [2] : 00000000000000000000000033a02e6cc863d393d6bf231b697b82f6e499ca71
Deployed Bytecode Sourcemap
71419:2753:0:-;;;;8:9:-1;5:2;;;30:1;27;20:12;5:2;71419:2753:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;68274:264;;;;;;13:2:-1;8:3;5:11;2:2;;;29:1;26;19:12;2:2;68274:264:0;;;;;;;;-1:-1:-1;;;5:28;;2:2;;;46:1;43;36:12;2:2;68274:264:0;;35:9:-1;28:4;12:14;8:25;5:40;2:2;;;58:1;55;48:12;2:2;68274:264:0;;;;;;100:9:-1;95:1;81:12;77:20;67:8;63:35;60:50;-1:-1;;;25:12;22:29;11:107;8:2;;;131:1;128;121:12;8:2;-1:-1;68274:264:0;;-1:-1:-1;68274:264:0;-1:-1:-1;68274:264:0;:::i;:::-;;72025:1846;;;;;;13:2:-1;8:3;5:11;2:2;;;29:1;26;19:12;2:2;72025:1846:0;;;;;;;;-1:-1:-1;;;5:28;;2:2;;;46:1;43;36:12;2:2;72025:1846:0;;35:9:-1;28:4;12:14;8:25;5:40;2:2;;;58:1;55;48:12;2:2;72025:1846:0;;;;;;100:9:-1;95:1;81:12;77:20;67:8;63:35;60:50;-1:-1;;;25:12;22:29;11:107;8:2;;;131:1;128;121:12;8:2;-1:-1;72025:1846:0;;-1:-1:-1;72025:1846:0;-1:-1:-1;72025:1846:0;:::i;74051:118::-;;;;;;13:2:-1;8:3;5:11;2:2;;;29:1;26;19:12;2:2;74051:118:0;;;;;;;;-1:-1:-1;;;5:28;;2:2;;;46:1;43;36:12;2:2;74051:118:0;;35:9:-1;28:4;12:14;8:25;5:40;2:2;;;58:1;55;48:12;2:2;74051:118:0;;;;;;100:9:-1;95:1;81:12;77:20;67:8;63:35;60:50;-1:-1;;;25:12;22:29;11:107;8:2;;;131:1;128;121:12;8:2;-1:-1;74051:118:0;;-1:-1:-1;74051:118:0;-1:-1:-1;74051:118:0;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;8:100:-1;33:3;30:1;27:10;8:100;;;90:11;;;84:18;71:11;;;64:39;52:2;45:10;8:100;;;12:14;74051:118:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;36404:41;;;:::i;36450:37::-;;;:::i;:::-;;;;;;;;;;;;;;;;73879:164;;;;;;13:2:-1;8:3;5:11;2:2;;;29:1;26;19:12;2:2;73879:164:0;;;;;;;;-1:-1:-1;;;5:28;;2:2;;;46:1;43;36:12;2:2;73879:164:0;;35:9:-1;28:4;12:14;8:25;5:40;2:2;;;58:1;55;48:12;2:2;73879:164:0;;;;;;100:9:-1;95:1;81:12;77:20;67:8;63:35;60:50;-1:-1;;;25:12;22:29;11:107;8:2;;;131:1;128;121:12;8:2;73879:164:0;;;;;;;;;;;-1:-1:-1;;;5:28;;2:2;;;46:1;43;36:12;2:2;73879:164:0;;35:9:-1;28:4;12:14;8:25;5:40;2:2;;;58:1;55;48:12;2:2;73879:164:0;;;;;;100:9:-1;95:1;81:12;77:20;67:8;63:35;60:50;-1:-1;;;25:12;22:29;11:107;8:2;;;131:1;128;121:12;8:2;73879:164:0;;;;;;;;;;;-1:-1:-1;;;5:28;;2:2;;;46:1;43;36:12;2:2;73879:164:0;;35:9:-1;28:4;12:14;8:25;5:40;2:2;;;58:1;55;48:12;2:2;73879:164:0;;;;;;100:9:-1;95:1;81:12;77:20;67:8;63:35;60:50;-1:-1;;;25:12;22:29;11:107;8:2;;;131:1;128;121:12;8:2;-1:-1;73879:164:0;;-1:-1:-1;73879:164:0;-1:-1:-1;73879:164:0;:::i;:::-;;;;;;;;;;;;;;;;;;68274:264;68060:15;;-1:-1:-1;;;;;68060:15:0;68038:10;:38;68016:109;;;;;-1:-1:-1;;;68016:109:0;;;;;;;;;;;;-1:-1:-1;;;68016:109:0;;;;;;;;;;;;;;;68362:13;68377:14;68393:15;68412:54;68451:4;;68412:54;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;30:3:-1;22:6;14;1:33;99:1;81:16;;74:27;;;;-1:-1;68412:24:0;;-1:-1:-1;;;68412:54:0:i;:::-;68477:14;;:53;;;-1:-1:-1;;;68477:53:0;;-1:-1:-1;;;;;68477:53:0;;;;;;;;;;;;;;;;;;;;;;68359:107;;-1:-1:-1;68359:107:0;;-1:-1:-1;68359:107:0;;-1:-1:-1;68477:14:0;;-1:-1:-1;68477:29:0;;:53;;;;;:14;;:53;;;;;;;:14;;:53;;;5:2:-1;;;;30:1;27;20:12;5:2;68477:53:0;;;;8:9:-1;5:2;;;45:16;42:1;39;24:38;77:16;74:1;67:27;5:2;68477:53:0;;;;68136:1;;;68274:264;;:::o;72025:1846::-;72100:42;72145:25;:16;:4;;:14;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;30:3:-1;22:6;14;1:33;99:1;81:16;;74:27;;;;-1:-1;72145:14:0;;-1:-1:-1;;;72145:16:0:i;:::-;:23;:25::i;:::-;72100:70;;72181:20;72204:28;:15;72220:1;72204:18;;;;;;;;;;;;;;:26;:28::i;:::-;72181:51;;72243:37;72283:28;:19;:7;:17;:19::i;:28::-;72243:68;;72322:16;72341:27;:15;72357:1;72341:18;;;;;;;;;;;;;;:25;:27::i;:::-;72322:46;;71102:2;72387:8;:19;72379:59;;;;;-1:-1:-1;;;72379:59:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;72449:11;72463:15;;:134;;-1:-1:-1;;;72463:134:0;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;72463:15:0;;;;:31;;72509:4;;;;72449:11;;;;72463:134;;;;72509:4;;;;72463:134;1:33:-1;99:1;93:3;85:6;81:16;74:27;137:4;133:9;126:4;121:3;117:14;113:30;106:37;;169:3;161:6;157:16;147:26;;72463:134:0;;;;;;;;;;;;;;;;;;;;8:9:-1;5:2;;;30:1;27;20:12;5:2;72463:134:0;;;;8:9:-1;5:2;;;45:16;42:1;39;24:38;77:16;74:1;67:27;5:2;72463:134:0;;;;;;;13:2:-1;8:3;5:11;2:2;;;29:1;26;19:12;2:2;-1:-1;72463:134:0;72621:13;;72463:134;;-1:-1:-1;72621:41:0;;:22;;:10;;72632:1;;72621:13;;;;;;;;;;;;:20;:22::i;:::-;72644:8;72621:32;;;;;;;:41;72608:54;;72796:18;72817:34;72837:10;72848:1;72837:13;;;;;;;;;;;;;;72817:19;:34::i;:::-;72796:55;;72862:20;72885:23;:10;72896:1;72885:13;;;;;;;:23;72862:46;;72932:22;:10;72943:1;72932:13;;;;;;;:22;72919:35;;71695:66;73244:18;;73217:22;:10;73228:1;73217:13;;;;;;;:22;73209:53;73187:133;;;;;-1:-1:-1;;;73187:133:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;73331:17;73359:31;73376:10;73387:1;73376:13;;;;;;;73359:31;73331:60;;73446:22;:10;73457:1;73446:13;;;;;;;:22;-1:-1:-1;;;;;73424:45:0;:10;-1:-1:-1;;;;;73424:45:0;;73402:143;;;;-1:-1:-1;;;73402:143:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;73556:18;73577:27;73593:7;73602:1;73577:15;:27::i;:::-;73634:15;;;:229;;;-1:-1:-1;;;73634:229:0;;73679:10;73634:229;;;;-1:-1:-1;;;;;73634:229:0;;;;;;;;;;;;;;;;;;;;;;;;;;:15;:229;;;;;;73844:8;;;73634:229;;;;;;73556:48;;-1:-1:-1;73634:15:0;;;:30;;:229;;;;;:15;;:229;;;;;;:15;;:229;;;5:2:-1;;;;30:1;27;20:12;5:2;73634:229:0;;;;8:9:-1;5:2;;;45:16;42:1;39;24:38;77:16;74:1;67:27;5:2;73634:229:0;;;;72025:1846;;;;;;;;;;;:::o;74051:118::-;74153:12;74051:118;;;;:::o;36404:41::-;;;;;;;;;;;;;;-1:-1:-1;;;36404:41:0;;;;:::o;36450:37::-;36484:3;36450:37;:::o;73879:164::-;74035:4;73879:164;;;;;;;;:::o;69967:317::-;70070:14;70086:13;70101:14;70117:15;70210:4;70185:91;;;;;13:3:-1;8;5:12;2:2;;;30:1;27;20:12;2:2;-1:-1;70185:91:0;;;;;;;;;;;;;;;;;;;-1:-1:-1;70185:91:0;;-1:-1:-1;70185:91:0;-1:-1:-1;69967:317:0;-1:-1:-1;;69967:317:0:o;17012:225::-;17073:14;;:::i;:::-;-1:-1:-1;17201:28:0;;;;;;;;;17209:11;;17201:28;;17166:4;17156:15;;;17201:28;;;;17012:225;;;;:::o;18455:523::-;18515:16;18552:12;18559:4;18552:6;:12::i;:::-;18544:21;;;;;;18578:10;18591:14;18600:4;18591:8;:14::i;:::-;18578:27;;18616:23;18656:5;18642:20;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;18616:46;;18675:11;18703:27;18718:4;:11;;;18703:14;:27::i;:::-;18689:11;;;;:41;;-1:-1:-1;18741:12:0;;18764:181;18785:5;18781:1;:9;18764:181;;;18822:19;18834:6;18822:11;:19::i;:::-;18812:29;;18868:24;;;;;;;;18876:7;18868:24;;;;18885:6;18868:24;;;18856:6;18863:1;18856:9;;;;;;;;;;;;;;;;;:36;18917:16;;;;18792:3;;18764:181;;;-1:-1:-1;18964:6:0;;18455:523;-1:-1:-1;;;;;18455:523:0:o;22461:383::-;22555:8;;22522:12;;22547:21;;;;;;22582:11;22595:8;22607:21;22623:4;22607:15;:21::i;:::-;22581:47;;;;22639:19;22671:3;22661:14;;;;;;;;;;;;;;;;;;;;;;;;;21:6:-1;;104:10;22661:14:0;87:34:-1;135:17;;-1:-1;22661:14:0;-1:-1:-1;22639:36:0;-1:-1:-1;22750:4:0;22746:17;;22786:26;22791:6;22746:17;22808:3;22786:4;:26::i;:::-;-1:-1:-1;22830:6:0;22461:383;-1:-1:-1;;;;22461:383:0:o;21642:471::-;21727:8;;21702:4;;21727:12;;;;:30;;-1:-1:-1;21743:8:0;;21755:2;-1:-1:-1;21743:14:0;21727:30;21719:39;;;;;;21772:11;21785:8;21797:21;21813:4;21797:15;:21::i;:::-;21887:13;;21771:47;;-1:-1:-1;21771:47:0;-1:-1:-1;21986:2:0;21978:11;;21975:2;;;22044;22040:12;;;22035:3;22031:22;22019:35;;21975:2;22099:6;21642:471;-1:-1:-1;;;;21642:471:0:o;21438:196::-;21570:8;;21501:7;;21582:2;21570:14;21562:23;;;;;;21613:12;21620:4;21613:6;:12::i;:::-;21598:28;21438:196;-1:-1:-1;;21438:196:0:o;9403:322::-;9506:7;9557:6;9566:2;9557:11;9539:6;:13;:30;;9531:39;;;;;;-1:-1:-1;9650:30:0;9666:4;9650:30;9644:37;;9403:322::o;19085:342::-;19166:8;;19145:4;;19162:31;;-1:-1:-1;19188:5:0;19181:12;;19162:31;19242:11;;;;19305:13;;19206:11;19297:22;;15803:4;19346:24;;19342:55;;;19392:5;19385:12;;;;;;19342:55;-1:-1:-1;19415:4:0;;19085:342;-1:-1:-1;;;19085:342:0:o;22957:422::-;23039:8;;23018:4;;23035:27;;-1:-1:-1;23061:1:0;23054:8;;23035:27;23075:10;23088:1;23075:14;;23100:12;23129:27;23144:4;:11;;;23129:14;:27::i;:::-;23115:11;;;;23195:8;;23115:41;;;;-1:-1:-1;23181:22:0;23214:133;23231:6;23221:7;:16;23214:133;;;23273:20;23285:7;23273:11;:20::i;:::-;23328:7;;;;;23263:30;;;;23214:133;;;-1:-1:-1;23366:5:0;;22957:422;-1:-1:-1;;;22957:422:0:o;24813:552::-;24951:13;;24872:4;;24943:22;;15709:4;24992:26;;24988:369;;;25041:1;25034:8;;;;;24988:369;15756:4;25062:25;;;:83;;-1:-1:-1;15803:4:0;25092:25;;;;;:52;;-1:-1:-1;15850:4:0;25121:23;;25092:52;25058:299;;;25167:1;25160:8;;;;;25058:299;15803:4;25188:24;;25184:173;;;-1:-1:-1;;25253:35:0;;-1:-1:-1;25246:42:0;;25184:173;-1:-1:-1;;25324:33:0;;-1:-1:-1;25317:40:0;;23431:1327;23589:13;;23487:4;;;;23581:22;;15709:4;23630:26;;23626:1098;;;23681:1;23671:11;;23626:1098;;;15756:4;23712:25;;23708:1016;;;-1:-1:-1;;23762:30:0;;;-1:-1:-1;23708:1016:0;;;15803:4;23814:24;;23810:914;;;23909:4;23902:5;23898:16;23989:1;23981:6;23977:14;23967:24;;24147:7;24143:2;24139:16;24134:3;24130:26;24121:6;24115:13;24111:46;24245:1;24236:7;24232:15;24223:7;24219:29;24208:40;;23864:399;;;;;15850:4;24295:23;;24291:433;;;-1:-1:-1;;24345:28:0;;;-1:-1:-1;24291:433:0;;;24472:4;24465:5;24461:16;24517:1;24509:6;24505:14;24495:24;;24590:7;24586:2;24582:16;24577:3;24573:26;24564:6;24558:13;24554:46;24695:1;24686:7;24682:15;24673:7;24669:29;24658:40;;24427:286;;;-1:-1:-1;24743:7:0;23431:1327;-1:-1:-1;;23431:1327:0:o;17906:271::-;17975:4;17981;17998:11;18012:27;18027:4;:11;;;18012:14;:27::i;:::-;18064:11;;;;18106:8;;18064:20;;;;18106:17;;;17906:271;-1:-1:-1;;;17906:271:0:o;25526:717::-;25599:8;25595:21;;25609:7;;25595:21;25676:201;15888:2;25683:16;;25676:201;;25775:10;;25762:24;;15888:2;25817:16;;;;25848:17;;;;-1:-1:-1;;25701:16:0;25676:201;;;26064:10;;26136:11;;15888:2;25990:15;;;25982:3;:24;-1:-1:-1;;25982:28:0;26076:9;;26060:26;;;26132:22;;26203:21;26190:35;;26030:206;;;;:::o;71419:2753::-;;;;;;;;;;;;;;;;;;;:::o
Swarm Source
bzzr://642f27a7dd308a6120ebc0d9fe1908bc2a25fa88c9dba93a98b247e08b5f4b9d
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.