Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
TokenTracker
Latest 1 from a total of 1 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
0x6101e060 | 19785311 | 163 days ago | IN | 0 ETH | 0.01825861 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Name:
ECOx
Compiler Version
v0.8.17+commit.8df45f5f
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
/* -*- c-basic-offset: 4 -*- */ // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./IECO.sol"; import "./TotalSupplySnapshots.sol"; /** * @title An ERC20 token interface for ECOx * */ contract ECOx is TotalSupplySnapshots { ////////////////////////////////////////////// //////////////////// VARS //////////////////// ////////////////////////////////////////////// /** * @dev Mapping storing contracts able to rebase the token */ mapping(address => bool) public snapshotters; ////////////////////////////////////////////// /////////////////// ERRORS /////////////////// ////////////////////////////////////////////// /** * error for when transfer returns false * used by contracts that import this contract */ error TransferFailed(); /** * error for when an address tries to snapshot without permission */ error OnlySnapshotters(); ////////////////////////////////////////////// /////////////////// EVENTS /////////////////// ////////////////////////////////////////////// /** * emits when the snapshotters permissions are changed * @param actor denotes the new address whose permissions are being updated * @param newPermission denotes the new ability of the actor address (true for can snapshot, false for cannot) */ event UpdatedSnapshotters(address actor, bool newPermission); ////////////////////////////////////////////// ////////////////// MODIFIERS ///////////////// ////////////////////////////////////////////// /** * @dev Modifier for checking if the sender is a snapshotter */ modifier onlySnapshotterRole() { if (!snapshotters[msg.sender]) { revert OnlySnapshotters(); } _; } constructor( Policy _policy, address _pauser ) TotalSupplySnapshots(_policy, "ECOx", "ECOx", _pauser) {} function initialize( address _self ) public virtual override onlyConstruction { super.initialize(_self); pauser = ERC20Pausable(_self).pauser(); } function snapshot() public onlySnapshotterRole { _snapshot(); } /** * @dev change the rebasing permissions for an address * only callable by tokenRoleAdmin * @param _key the address to change permissions for * @param _value the new permission. true = can snapshot, false = cannot snapshot */ function updateSnapshotters(address _key, bool _value) public onlyPolicy { snapshotters[_key] = _value; emit UpdatedSnapshotters(_key, _value); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol) pragma solidity ^0.8.0; import "../utils/Context.sol"; /** * @dev Contract module which allows children to implement an emergency stop * mechanism that can be triggered by an authorized account. * * This module is used through inheritance. It will make available the * modifiers `whenNotPaused` and `whenPaused`, which can be applied to * the functions of your contract. Note that they will not be pausable by * simply including this module, only once the modifiers are put in place. */ abstract contract Pausable is Context { /** * @dev Emitted when the pause is triggered by `account`. */ event Paused(address account); /** * @dev Emitted when the pause is lifted by `account`. */ event Unpaused(address account); bool private _paused; /** * @dev Initializes the contract in unpaused state. */ constructor() { _paused = false; } /** * @dev Modifier to make a function callable only when the contract is not paused. * * Requirements: * * - The contract must not be paused. */ modifier whenNotPaused() { _requireNotPaused(); _; } /** * @dev Modifier to make a function callable only when the contract is paused. * * Requirements: * * - The contract must be paused. */ modifier whenPaused() { _requirePaused(); _; } /** * @dev Returns true if the contract is paused, and false otherwise. */ function paused() public view virtual returns (bool) { return _paused; } /** * @dev Throws if the contract is paused. */ function _requireNotPaused() internal view virtual { require(!paused(), "Pausable: paused"); } /** * @dev Throws if the contract is not paused. */ function _requirePaused() internal view virtual { require(paused(), "Pausable: not paused"); } /** * @dev Triggers stopped state. * * Requirements: * * - The contract must not be paused. */ function _pause() internal virtual whenNotPaused { _paused = true; emit Paused(_msgSender()); } /** * @dev Returns to normal state. * * Requirements: * * - The contract must be paused. */ function _unpause() internal virtual whenPaused { _paused = false; emit Unpaused(_msgSender()); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-IERC20Permit.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. */ interface IERC20Permit { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol) pragma solidity ^0.8.0; import "../IERC20.sol"; /** * @dev Interface for the optional metadata functions from the ERC20 standard. * * _Available since v4.1._ */ interface IERC20Metadata is IERC20 { /** * @dev Returns the name of the token. */ function name() external view returns (string memory); /** * @dev Returns the symbol of the token. */ function symbol() external view returns (string memory); /** * @dev Returns the decimals places of the token. */ function decimals() external view returns (uint8); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `from` to `to` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 amount ) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Counters.sol) pragma solidity ^0.8.0; /** * @title Counters * @author Matt Condon (@shrugs) * @dev Provides counters that can only be incremented, decremented or reset. 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;` */ library Counters { 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 { unchecked { counter._value += 1; } } function decrement(Counter storage counter) internal { uint256 value = counter._value; require(value > 0, "Counter: decrement overflow"); unchecked { counter._value = value - 1; } } function reset(Counter storage counter) internal { counter._value = 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/ECDSA.sol) pragma solidity ^0.8.0; import "../Strings.sol"; /** * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations. * * These functions can be used to verify that a message was signed by the holder * of the private keys of a given address. */ library ECDSA { enum RecoverError { NoError, InvalidSignature, InvalidSignatureLength, InvalidSignatureS, InvalidSignatureV // Deprecated in v4.8 } function _throwError(RecoverError error) private pure { if (error == RecoverError.NoError) { return; // no error: do nothing } else if (error == RecoverError.InvalidSignature) { revert("ECDSA: invalid signature"); } else if (error == RecoverError.InvalidSignatureLength) { revert("ECDSA: invalid signature length"); } else if (error == RecoverError.InvalidSignatureS) { revert("ECDSA: invalid signature 's' value"); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature` or error string. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. * * Documentation for signature generation: * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js] * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers] * * _Available since v4.3._ */ function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) { if (signature.length == 65) { bytes32 r; bytes32 s; uint8 v; // ecrecover takes the signature parameters, and the only way to get them // currently is to use assembly. /// @solidity memory-safe-assembly assembly { r := mload(add(signature, 0x20)) s := mload(add(signature, 0x40)) v := byte(0, mload(add(signature, 0x60))) } return tryRecover(hash, v, r, s); } else { return (address(0), RecoverError.InvalidSignatureLength); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature`. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. */ function recover(bytes32 hash, bytes memory signature) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, signature); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately. * * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures] * * _Available since v4.3._ */ function tryRecover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address, RecoverError) { bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff); uint8 v = uint8((uint256(vs) >> 255) + 27); return tryRecover(hash, v, r, s); } /** * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately. * * _Available since v4.2._ */ function recover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, r, vs); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `v`, * `r` and `s` signature fields separately. * * _Available since v4.3._ */ function tryRecover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address, RecoverError) { // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most // signatures from current libraries generate a unique signature with an s-value in the lower half order. // // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept // these malleable signatures as well. if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { return (address(0), RecoverError.InvalidSignatureS); } // If the signature is valid (and not malleable), return the signer address address signer = ecrecover(hash, v, r, s); if (signer == address(0)) { return (address(0), RecoverError.InvalidSignature); } return (signer, RecoverError.NoError); } /** * @dev Overload of {ECDSA-recover} that receives the `v`, * `r` and `s` signature fields separately. */ function recover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, v, r, s); _throwError(error); return recovered; } /** * @dev Returns an Ethereum Signed Message, created from a `hash`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) { // 32 is the length in bytes of hash, // enforced by the type signature above return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash)); } /** * @dev Returns an Ethereum Signed Message, created from `s`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s)); } /** * @dev Returns an Ethereum Signed Typed Data, created from a * `domainSeparator` and a `structHash`. This produces hash corresponding * to the one signed with the * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`] * JSON-RPC method as part of EIP-712. * * See {recover}. */ function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/EIP712.sol) pragma solidity ^0.8.0; import "./ECDSA.sol"; /** * @dev https://eips.ethereum.org/EIPS/eip-712[EIP 712] is a standard for hashing and signing of typed structured data. * * The encoding specified in the EIP is very generic, and such a generic implementation in Solidity is not feasible, * thus this contract does not implement the encoding itself. Protocols need to implement the type-specific encoding * they need in their contracts using a combination of `abi.encode` and `keccak256`. * * This contract implements the EIP 712 domain separator ({_domainSeparatorV4}) that is used as part of the encoding * scheme, and the final step of the encoding to obtain the message digest that is then signed via ECDSA * ({_hashTypedDataV4}). * * The implementation of the domain separator was designed to be as efficient as possible while still properly updating * the chain id to protect against replay attacks on an eventual fork of the chain. * * NOTE: This contract implements the version of the encoding known as "v4", as implemented by the JSON RPC method * https://docs.metamask.io/guide/signing-data.html[`eth_signTypedDataV4` in MetaMask]. * * _Available since v3.4._ */ abstract contract EIP712 { /* solhint-disable var-name-mixedcase */ // Cache the domain separator as an immutable value, but also store the chain id that it corresponds to, in order to // invalidate the cached domain separator if the chain id changes. bytes32 private immutable _CACHED_DOMAIN_SEPARATOR; uint256 private immutable _CACHED_CHAIN_ID; address private immutable _CACHED_THIS; bytes32 private immutable _HASHED_NAME; bytes32 private immutable _HASHED_VERSION; bytes32 private immutable _TYPE_HASH; /* solhint-enable var-name-mixedcase */ /** * @dev Initializes the domain separator and parameter caches. * * The meaning of `name` and `version` is specified in * https://eips.ethereum.org/EIPS/eip-712#definition-of-domainseparator[EIP 712]: * * - `name`: the user readable name of the signing domain, i.e. the name of the DApp or the protocol. * - `version`: the current major version of the signing domain. * * NOTE: These parameters cannot be changed except through a xref:learn::upgrading-smart-contracts.adoc[smart * contract upgrade]. */ constructor(string memory name, string memory version) { bytes32 hashedName = keccak256(bytes(name)); bytes32 hashedVersion = keccak256(bytes(version)); bytes32 typeHash = keccak256( "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" ); _HASHED_NAME = hashedName; _HASHED_VERSION = hashedVersion; _CACHED_CHAIN_ID = block.chainid; _CACHED_DOMAIN_SEPARATOR = _buildDomainSeparator(typeHash, hashedName, hashedVersion); _CACHED_THIS = address(this); _TYPE_HASH = typeHash; } /** * @dev Returns the domain separator for the current chain. */ function _domainSeparatorV4() internal view returns (bytes32) { if (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) { return _CACHED_DOMAIN_SEPARATOR; } else { return _buildDomainSeparator(_TYPE_HASH, _HASHED_NAME, _HASHED_VERSION); } } function _buildDomainSeparator( bytes32 typeHash, bytes32 nameHash, bytes32 versionHash ) private view returns (bytes32) { return keccak256(abi.encode(typeHash, nameHash, versionHash, block.chainid, address(this))); } /** * @dev Given an already https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct[hashed struct], this * function returns the hash of the fully encoded EIP712 message for this domain. * * This hash can be used together with {ECDSA-recover} to obtain the signer of a message. For example: * * ```solidity * bytes32 digest = _hashTypedDataV4(keccak256(abi.encode( * keccak256("Mail(address to,string contents)"), * mailTo, * keccak256(bytes(mailContents)) * ))); * address signer = ECDSA.recover(digest, signature); * ``` */ function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) { return ECDSA.toTypedDataHash(_domainSeparatorV4(), structHash); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @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 Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv( uint256 x, uint256 y, uint256 denominator ) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv( uint256 x, uint256 y, uint256 denominator, Rounding rounding ) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10**64) { value /= 10**64; result += 64; } if (value >= 10**32) { value /= 10**32; result += 32; } if (value >= 10**16) { value /= 10**16; result += 16; } if (value >= 10**8) { value /= 10**8; result += 8; } if (value >= 10**4) { value /= 10**4; result += 4; } if (value >= 10**2) { value /= 10**2; result += 2; } if (value >= 10**1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/Strings.sol) pragma solidity ^0.8.0; import "./math/Math.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant _SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../utils/StringPacker.sol"; import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; import "./ERC20Permit.sol"; /** * Implementation of the {IERC20} interface. * * This comment taken from the openzeppelin source contract. * * This implementation is agnostic to the way tokens are created. This means * that a supply mechanism has to be added in a derived contract using {_mint}. * For a generic mechanism see {ERC20PresetMinterPauser}. * * TIP: For a detailed writeup see open zeppelin guide * [How to implement supply mechanisms](https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226). * * We have followed general OpenZeppelin Contracts guidelines: functions revert * instead returning `false` on failure. This behavior is nonetheless * conventional and does not conflict with the expectations of ERC20 * applications. * * Additionally, an {Approval} event is emitted on calls to {transferFrom}. * This allows applications to reconstruct the allowance for all accounts just * by listening to said events. Other implementations of the EIP may not emit * these events, as it isn't required by the specification. * * Finally, the non-standard {decreaseAllowance} and {increaseAllowance} * functions have been added to mitigate the well-known issues around setting * allowances. See {IERC20-approve}. */ // internal _name and _symbol are stored immutable as bytes32 and unpacked via StringPacker contract ERC20 is ERC20Permit { mapping(address => uint256) internal _balances; mapping(address => mapping(address => uint256)) private _allowances; uint256 internal _totalSupply; bytes32 internal immutable _name; bytes32 internal immutable _symbol; /** * Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval( address indexed owner, address indexed spender, uint256 value ); /** * Sets the values for {name} and {symbol}. * * The default value of {decimals} is 18. To select a different value for * {decimals} you should overload it. * * All two of these values are immutable: they can only be set once during * construction. */ constructor(string memory name_, string memory symbol_) ERC20Permit(name_) { _name = StringPacker.pack(name_); _symbol = StringPacker.pack(symbol_); } /** * Returns the name of the token. */ function name() public view virtual returns (string memory) { return StringPacker.unpack(_name); } /** * Returns the symbol of the token, usually a shorter version of the * name. */ function symbol() public view virtual returns (string memory) { return StringPacker.unpack(_symbol); } /** * Returns the number of decimals used to get its user representation. * For example, if `decimals` equals `2`, a balance of `505` tokens should * be displayed to a user as `5.05` (`505 / 10 ** 2`). * * Tokens usually opt for a value of 18, imitating the relationship between * Ether and Wei. This is the value {ERC20} uses, unless this function is * overridden; * * NOTE: This information is only used for _display_ purposes: it in * no way affects any of the arithmetic of the contract, including * {IERC20-balanceOf} and {IERC20-transfer}. */ function decimals() public view virtual returns (uint8) { return 18; } /** * See {IERC20-totalSupply}. */ function totalSupply() public view virtual returns (uint256) { return _totalSupply; } /** * See {IERC20-balanceOf}. */ function balanceOf(address account) public view virtual returns (uint256) { return _balances[account]; } /** * See {IERC20-transfer}. * * Requirements: * * - `recipient` cannot be the zero address. * - the caller must have a balance of at least `amount`. */ function transfer( address recipient, uint256 amount ) public virtual returns (bool) { _transfer(msg.sender, recipient, amount); return true; } /** * See {IERC20-allowance}. */ function allowance( address owner, address spender ) public view virtual returns (uint256) { return _allowances[owner][spender]; } /** * See {IERC20-approve}. * * Requirements: * * - `spender` cannot be the zero address. */ function approve( address spender, uint256 amount ) public virtual returns (bool) { _approve(msg.sender, spender, amount); return true; } /** * See {IERC20-transferFrom}. * * Emits an {Approval} event indicating the updated allowance. This is not * required by the EIP. See the note at the beginning of {ERC20}. * * Requirements: * * - `sender` and `recipient` cannot be the zero address. * - `sender` must have a balance of at least `amount`. * - the caller must have allowance for ``sender``'s tokens of at least * `amount`. */ function transferFrom( address sender, address recipient, uint256 amount ) public virtual returns (bool) { _transfer(sender, recipient, amount); uint256 currentAllowance = _allowances[sender][msg.sender]; require( currentAllowance >= amount, "ERC20: transfer amount exceeds allowance" ); unchecked { _approve(sender, msg.sender, currentAllowance - amount); } return true; } /** * Atomically increases the allowance granted to `spender` by the caller. * * This is an alternative to {approve} that can be used as a mitigation for * problems described in {IERC20-approve}. * * Emits an {Approval} event indicating the updated allowance. * * Requirements: * * - `spender` cannot be the zero address. */ function increaseAllowance( address spender, uint256 addedValue ) public virtual returns (bool) { _approve( msg.sender, spender, _allowances[msg.sender][spender] + addedValue ); return true; } /** * Atomically decreases the allowance granted to `spender` by the caller. * * This is an alternative to {approve} that can be used as a mitigation for * problems described in {IERC20-approve}. * * Emits an {Approval} event indicating the updated allowance. * * Requirements: * * - `spender` cannot be the zero address. * - `spender` must have allowance for the caller of at least * `subtractedValue`. */ function decreaseAllowance( address spender, uint256 subtractedValue ) public virtual returns (bool) { uint256 currentAllowance = _allowances[msg.sender][spender]; require( currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero" ); unchecked { _approve(msg.sender, spender, currentAllowance - subtractedValue); } return true; } /** * Moves `amount` of tokens from `sender` to `recipient`. * * This internal function is equivalent to {transfer}, and can be used to * e.g. implement automatic token fees, slashing mechanisms, etc. * * Emits a {Transfer} event. * * Requirements: * * - `sender` cannot be the zero address. * - `recipient` cannot be the zero address. * - `sender` must have a balance of at least `amount`. */ function _transfer( address sender, address recipient, uint256 originalAmount ) internal virtual { require(recipient != address(0), "ERC20: transfer to the zero address"); uint256 amount = _beforeTokenTransfer( sender, recipient, originalAmount ); uint256 senderBalance = _balances[sender]; require( senderBalance >= amount, "ERC20: transfer amount exceeds balance" ); unchecked { _balances[sender] = senderBalance - amount; } _balances[recipient] += amount; emit Transfer(sender, recipient, originalAmount); _afterTokenTransfer(sender, recipient, amount); } /** Creates `amount` tokens and assigns them to `account`, increasing * the total supply. * * Emits a {Transfer} event with `from` set to the zero address. * * Requirements: * * - `account` cannot be the zero address. */ function _mint( address account, uint256 originalAmount ) internal virtual returns (uint256) { require(account != address(0), "ERC20: mint to the zero address"); uint256 amount = _beforeTokenTransfer( address(0), account, originalAmount ); _totalSupply += amount; _balances[account] += amount; emit Transfer(address(0), account, originalAmount); _afterTokenTransfer(address(0), account, amount); return amount; } /** * Destroys `amount` tokens from `account`, reducing the * total supply. * * Emits a {Transfer} event with `to` set to the zero address. * * Requirements: * * - `account` cannot be the zero address. * - `account` must have at least `amount` tokens. */ function _burn( address account, uint256 originalAmount ) internal virtual returns (uint256) { uint256 amount = _beforeTokenTransfer( account, address(0), originalAmount ); uint256 accountBalance = _balances[account]; require(accountBalance >= amount, "ERC20: burn amount exceeds balance"); unchecked { _balances[account] = accountBalance - amount; } _totalSupply -= amount; emit Transfer(account, address(0), originalAmount); _afterTokenTransfer(account, address(0), amount); return amount; } /** * Sets `amount` as the allowance of `spender` over the `owner` s tokens. * * This internal function is equivalent to `approve`, and can be used to * e.g. set automatic allowances for certain subsystems, etc. * * Emits an {Approval} event. * * Requirements: * * - `owner` cannot be the zero address. * - `spender` cannot be the zero address. */ function _approve( address owner, address spender, uint256 amount ) internal virtual override { require(spender != address(0), "ERC20: approve to the zero address"); _allowances[owner][spender] = amount; emit Approval(owner, spender, amount); } /** * Hook that is called before any transfer of tokens. This includes * minting and burning. * * Calling conditions: * * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens * will be transferred to `to`. * - when `from` is zero, `amount` tokens will be minted for `to`. * - when `to` is zero, `amount` of ``from``'s tokens will be burned. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address, // from address, // to uint256 amount ) internal virtual returns (uint256) { return amount; } /** * Hook that is called after any transfer of tokens. This includes * minting and burning. * * Calling conditions: * * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens * has been transferred to `to`. * - when `from` is zero, `amount` tokens have been minted for `to`. * - when `to` is zero, `amount` of ``from``'s tokens have been burned. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _afterTokenTransfer( address from, address to, uint256 amount ) internal virtual {} }
/* -*- c-basic-offset: 4 -*- */ // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../policy/PolicedUpgradeable.sol"; import "./ERC20Pausable.sol"; /** * @title An ERC20 token interface for ECOx * */ contract ERC20MintAndBurn is ERC20Pausable, PolicedUpgradeable { // storage gap covers all of ECO contract's old delegation functionality (no ECOx storage to be gapped) uint256[10] private __gapMintAndBurn; // additional useable gap for future upgradeability uint256[50] private __gapMintAndBurn2; ////////////////////////////////////////////// //////////////////// VARS //////////////////// ////////////////////////////////////////////// /** * Mapping storing contracts able to mint tokens */ mapping(address => bool) public minters; /** * Mapping storing contracts able to burn tokens */ mapping(address => bool) public burners; ////////////////////////////////////////////// /////////////////// ERRORS /////////////////// ////////////////////////////////////////////// /** * error for when an address tries to mint tokens without permission */ error OnlyMinters(); /** * error for when an address tries to burn tokens without permission */ error OnlyBurners(); ////////////////////////////////////////////// /////////////////// EVENTS /////////////////// ////////////////////////////////////////////// /** * emits when the minters permissions are changed * @param actor denotes the new address whose permissions are being updated * @param newPermission denotes the new ability of the actor address (true for can mint, false for cannot) */ event UpdatedMinters(address actor, bool newPermission); /** * emits when the burners permissions are changed * @param actor denotes the new address whose permissions are being updated * @param newPermission denotes the new ability of the actor address (true for can burn, false for cannot) */ event UpdatedBurners(address actor, bool newPermission); ////////////////////////////////////////////// ////////////////// MODIFIERS ///////////////// ////////////////////////////////////////////// /** * Modifier for checking if the sender is a minter */ modifier onlyMinterRole() { if (!minters[msg.sender]) { revert OnlyMinters(); } _; } /** * Modifier for checking if the sender is allowed to burn * both burners and the message sender can burn * @param _from the address burning tokens */ modifier onlyBurnerRoleOrSelf(address _from) { if (_from != msg.sender && !burners[msg.sender]) { revert OnlyBurners(); } _; } constructor( Policy policy, string memory name, string memory ticker, address pauser ) Policed(policy) ERC20Pausable(name, ticker, address(policy), pauser) {} /** * change the minting permissions for an address * only callable by tokenRoleAdmin * @param _key the address to change permissions for * @param _value the new permission. true = can mint, false = cannot mint */ function updateMinters(address _key, bool _value) public onlyPolicy { minters[_key] = _value; emit UpdatedMinters(_key, _value); } /** * change the burning permissions for an address * only callable by tokenRoleAdmin * @param _key the address to change permissions for * @param _value the new permission. true = can burn, false = cannot burn */ function updateBurners(address _key, bool _value) public onlyPolicy { burners[_key] = _value; emit UpdatedBurners(_key, _value); } /** * mints tokens to a given address * @param _to the address receiving tokens * @param _value the amount of tokens being minted */ function mint(address _to, uint256 _value) external onlyMinterRole { _mint(_to, _value); } /** * burns tokens to a given address * @param _from the address whose tokens are being burned * @param _value the amount of tokens being burned */ function burn( address _from, uint256 _value ) external onlyBurnerRoleOrSelf(_from) { _burn(_from, _value); } // protecting future upgradeability uint256[50] private __gapMintAndBurn3; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/security/Pausable.sol"; import "./ERC20.sol"; /** * Implementation of the {IERC20} interface with pausability * When paused by the pauser admin, transfers revert. */ contract ERC20Pausable is ERC20, Pausable { address public immutable roleAdmin; // initially no-one should have the pauser role // it can be granted and revoked by the admin policy address public pauser; /** * @notice event indicating the pauser was updated * @param pauser The new pauser */ event PauserAssignment(address indexed pauser); constructor( string memory name, string memory symbol, address _roleAdmin, address _initialPauser ) ERC20(name, symbol) { require( address(_roleAdmin) != address(0), "Unrecoverable: do not set the _roleAdmin as the zero address" ); roleAdmin = _roleAdmin; pauser = _initialPauser; emit PauserAssignment(_initialPauser); } modifier onlyAdmin() { require(msg.sender == roleAdmin, "ERC20Pausable: not admin"); _; } modifier onlyPauser() { require(msg.sender == pauser, "ERC20Pausable: not pauser"); _; } /** * Hook that is called before any transfer of tokens. This includes * minting and burning. * * If the token is not paused, it will pass through the amount */ function _beforeTokenTransfer( address from, address to, uint256 amount ) internal virtual override whenNotPaused returns (uint256) { return amount; } /** * @notice pauses transfers of this token * * Security Notes * * - only callable by the pause * - reverts if already paused */ function pause() external onlyPauser { _pause(); } /** * @notice unpauses transfers of this token * only callable by the pauser * * Security Notes * * - only callable by the pause * - reverts if already unpaused */ function unpause() external onlyPauser { _unpause(); } /** * @notice set the given address as the pauser * @param _pauser The address that can pause this token * only the roleAdmin can call this function */ function setPauser(address _pauser) public onlyAdmin { require(_pauser != pauser, "ERC20Pausable: must change pauser"); pauser = _pauser; emit PauserAssignment(_pauser); } }
// SPDX-License-Identifier: MIT // Heavily inspired by: // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-ERC20Permit.sol) pragma solidity ^0.8.0; import "@openzeppelin/contracts/token/ERC20/extensions/draft-IERC20Permit.sol"; import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import "@openzeppelin/contracts/utils/Counters.sol"; import "@openzeppelin/contracts/utils/cryptography/ECDSA.sol"; import "@openzeppelin/contracts/utils/cryptography/EIP712.sol"; /** * Implementation of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * [EIP-2612](https://eips.ethereum.org/EIPS/eip-2612). * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on `{IERC20-approve}`, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. * * _Available since v3.4._ */ abstract contract ERC20Permit is IERC20Permit, EIP712 { using Counters for Counters.Counter; mapping(address => Counters.Counter) private _nonces; // solhint-disable-next-line var-name-mixedcase bytes32 private immutable _PERMIT_TYPEHASH = keccak256( "Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)" ); /** * Initializes the {EIP712} domain separator using the `name` parameter, and setting `version` to `"2"`. * * version number 1 was used already in a previous implementation */ constructor(string memory name) EIP712(name, "2") { //empty block in order to pass parameters to the parent EIP712 constructor } /** * See {IERC20Permit-permit}. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) public virtual override { require(block.timestamp <= deadline, "ERC20Permit: expired deadline"); bytes32 structHash = keccak256( abi.encode( _PERMIT_TYPEHASH, owner, spender, value, _useNonce(owner), deadline ) ); bytes32 hash = _hashTypedDataV4(structHash); address signer = ECDSA.recover(hash, v, r, s); require(signer == owner, "ERC20Permit: invalid signature"); _approve(owner, spender, value); } /** * See {IERC20Permit-nonces}. */ function nonces( address owner ) public view virtual override returns (uint256) { return _nonces[owner].current(); } /** * See {IERC20Permit-DOMAIN_SEPARATOR}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view override returns (bytes32) { return _domainSeparatorV4(); } /** * "Consume a nonce": return the current value and increment. * * _Available since v4.1._ */ function _useNonce( address owner ) internal virtual returns (uint256 current) { Counters.Counter storage nonce = _nonces[owner]; current = nonce.current(); nonce.increment(); } /** * Sets `amount` as the allowance of `spender` over the `owner` s tokens. * * This internal function is equivalent to `approve`, and can be used to * e.g. set automatic allowances for certain subsystems, etc. * * Emits an {Approval} event. * * Requirements: * * - `owner` cannot be the zero address. * - `spender` cannot be the zero address. */ function _approve( address owner, address spender, uint256 amount ) internal virtual; }
/* -*- c-basic-offset: 4 -*- */ // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; interface IECO is IERC20 { /** * Only available to minters * @param to the address to mint to * @param amount the amount to mint */ function mint(address to, uint256 amount) external; /** * Only available to token holders for their own tokens and burners * @param from the address to burn from * @param amount the amount to burn */ function burn(address from, uint256 amount) external; /** * Returns the votes for the current snapshot * @param account the address whose vote balance to check * @return balance the balance of the account at the time of the Snapshot */ function voteBalanceSnapshot( address account ) external view returns (uint256 balance); /** * Returns the inflation multiplier value for the current snapshot * @return multiplier inflation multipler value */ function inflationMultiplierSnapshot() external view returns (uint256 multiplier); /** * Returns the total supply for the current snapshot * @return total total supply of the current snapshot */ function totalSupplySnapshot() external view returns (uint256 total); /** * Enables voting with your ECO balance, but will transaction cost */ function enableVoting() external; /** * Allows others to delegate voting power to you * Disallows you from delegating your voting power to others */ function enableDelegationTo() external; /** * Disallows others from delegating to you * Does not change your ability to delegate to others */ function disableDelegationTo() external; /** * Allows others to delegate to you * Disallows you from delegating to others */ function reenableDelegating() external; /** * Returns true if the address has no amount of their balance delegated, otherwise false * @param account the address whose delegation status is being checked */ function isOwnDelegate(address account) external returns (bool); /** * Fetches the primary address `account` is currently delegating to. Defaults to the account address itself if none specified. * The primary delegate is the one that is delegated any new funds the address recieves * @param account the address whose primary delegate is being fetched */ function getPrimaryDelegate( address account ) external view returns (address); /** * Delegates all votes from the sender to `delegatee` * This function assumes that you do not have partial delegations * It will revert with "ERC20Delegated: must have an undelegated amount available to cover delegation" if you do * @param delegatee the address being delegated to */ function delegate(address delegatee) external; /** * Delegates all votes from the sender to `delegatee` * This function assumes that you do not have partial delegations * It will revert with "ERC20Delegated: must have an undelegated amount available to cover delegation" if you do * @param delegator the address delegating votes * @param delegatee the address being delegated to * @param deadline the time at which the signature expires * @param v signature value * @param r signature value * @param s signature value */ function delegateBySig( address delegator, address delegatee, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * Delegate an `amount` of votes from the sender to `delegatee` * @param delegatee the address being delegated to * @param amount the amount of tokens being allocated */ function delegateAmount(address delegatee, uint256 amount) external; /** * Undelegate all votes from the sender's primary delegate */ function undelegate() external; /** * Allows a primary delegated individual to revoke delegations of unwanted delegators * Useful for allowing yourself to call reenableDelegating after calling disableDelegationTo * @param delegator the address whose delegation is being revoked */ function revokeDelegation(address delegator) external; /** * Undelegate a specific amount of votes from the `delegatee` back to the sender * @param delegatee the address being undelegated to * @param amount the amount of tokens being undelegated */ function undelegateAmountFromAddress( address delegatee, uint256 amount ) external; /** * See {IERC20-balanceOf}. * @param account the address whose vote balance is being checked */ function voteBalanceOf(address account) external view returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./ERC20MintAndBurn.sol"; /** * @dev Basic snapshotting just for total supply. * * This extension maintains a snapshot the total supply which updates on mint or burn after a new snapshot is taken. */ abstract contract TotalSupplySnapshots is ERC20MintAndBurn { // structure for saving snapshotted values struct Snapshot { uint32 snapshotBlock; uint224 value; } // the reference snapshotBlock that the update function checks against uint32 public currentSnapshotBlock; // the snapshot to track the token total supply Snapshot internal _totalSupplySnapshot; /** * @dev Emitted by {_snapshot} when a new snapshot is created. * * @param block the new value of currentSnapshotBlock */ event NewSnapshotBlock(uint256 block); constructor( Policy _policy, string memory _name, string memory _symbol, address _initialPauser ) ERC20MintAndBurn(_policy, _name, _symbol, _initialPauser) { // snapshot on creation to make it clear that everyone's balances should be updated _snapshot(); } function initialize( address _self ) public virtual override onlyConstruction { super.initialize(_self); // snapshot on initialization to make it clear that everyone's balances should be updated after upgrade _snapshot(); } /** * @dev Retrieve the `totalSupply` for the snapshot */ function totalSupplySnapshot() public view virtual returns (uint256) { if ( currentSnapshotBlock != block.number && _totalSupplySnapshot.snapshotBlock < currentSnapshotBlock ) { return _totalSupply; } else { return _totalSupplySnapshot.value; } } /** * Update total supply snapshots before the values are modified. This is implemented * in the _beforeTokenTransfer hook, which is executed for _mint, _burn, and _transfer operations. */ function _beforeTokenTransfer( address from, address to, uint256 amount ) internal virtual override returns (uint256) { if (from == address(0)) { // mint _updateTotalSupplySnapshot(); } else if (to == address(0)) { // burn _updateTotalSupplySnapshot(); } return super._beforeTokenTransfer(from, to, amount); } /** * @dev Creates a new snapshot and returns its snapshot id. * * Emits a {NewSnapshotBlock} event that contains the same id. */ function _snapshot() internal virtual { // the math will error if the snapshot overflows currentSnapshotBlock = uint32(block.number); emit NewSnapshotBlock(block.number); } function _updateTotalSupplySnapshot() private { uint32 _currentSnapshotBlock = currentSnapshotBlock; // take no action during the snapshot block, only after it if (_currentSnapshotBlock == block.number) { return; } if (_totalSupplySnapshot.snapshotBlock < _currentSnapshotBlock) { uint256 currentValue = _totalSupply; require( currentValue <= type(uint224).max, "VoteSnapshots: new snapshot cannot be casted safely" ); _totalSupplySnapshot.snapshotBlock = _currentSnapshotBlock; _totalSupplySnapshot.value = uint224(currentValue); } } // protecting future upgradeability uint256[50] private __gapTotalSupplySnapshots; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./Policy.sol"; /** * @title Policed Contracts * @notice A policed contract is any contract managed by a policy. */ abstract contract Policed { /** The address of the root policy instance overseeing this instance. */ Policy public immutable policy; /** If the policy address is set to zero, the contract is unrecoverably ungovernable */ error NonZeroPolicyAddr(); /** * If this address is set to zero the contract is an unusable state * @param contractName the name of the contract that was given as the zero address */ error NonZeroContractAddr(string contractName); /** For if a non-policy address tries to access policy role gated functionality */ error PolicyOnlyFunction(); /** Restrict method access to the root policy instance only. */ modifier onlyPolicy() { if (msg.sender != address(policy)) { revert PolicyOnlyFunction(); } _; } /** constructor * @param _policy the address of the owning policy contract */ constructor(Policy _policy) { // _setPolicy(_policy); if (address(_policy) == address(0)) { revert NonZeroPolicyAddr(); } policy = _policy; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./Policed.sol"; /** @title Policed Proxy Contracts * * A PolicedUpgradeable contract is any proxied contract managed by a policy. */ abstract contract PolicedUpgradeable is Policed, ForwardTarget { // storage gap covers PolicedUtils.expectedInterfaceSet // relevant for ECOxStaking uint256 private __gapPolicedUpgradeable; function setImplementation(address _impl) public onlyPolicy { _setImplementation(_impl); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../proxy/ForwardTarget.sol"; /** * @title The policy contract that oversees other contracts * @notice Policy contracts provide a mechanism for building pluggable (after deploy) * governance systems for other contracts. */ contract Policy is ForwardTarget { uint256 private __gapPolicy; // to cover setters mapping /** * the contract allowed enact proposals */ address public governor; /** * error for when an address tries submit proposal actions without permission */ error OnlyGovernor(); /** * error for when an address tries to call a pseudo-internal function */ error OnlySelf(); /** * for when a part of enacting a proposal reverts without a readable error * @param proposal the proposal address that got reverted during enaction */ error FailedProposal(address proposal); /** * emits when the governor permissions are changed * @param oldGovernor denotes the old address whose permissions are being removed * @param newGovernor denotes the new address whose permissions are being added */ event UpdatedGovernor(address oldGovernor, address newGovernor); /** * emits when enaction happens to keep record of enaction * @param proposal the proposal address that got successfully enacted * @param governor the contract which was the source of the proposal, source for looking up the calldata */ event EnactedGovernanceProposal(address proposal, address governor); /** * Modifier for checking if the sender is a governor */ modifier onlyGovernorRole() { if (msg.sender != governor) { revert OnlyGovernor(); } _; } /** * Modifier for faux internal calls * needed for function to be called only during delegate call */ modifier onlySelf() { if (msg.sender != address(this)) { revert OnlySelf(); } _; } constructor(address _governor) { governor = _governor; } /** * initializes the governor */ function initialize( address _self ) public virtual override onlyConstruction { super.initialize(_self); governor = Policy(_self).governor(); } /** * pass the governance permissions to another address * @param _newGovernor the address to make the new governor */ function updateGovernor(address _newGovernor) public onlySelf { emit UpdatedGovernor(governor, _newGovernor); governor = _newGovernor; } function enact(address proposal) external virtual onlyGovernorRole { // solhint-disable-next-line avoid-low-level-calls (bool _success, bytes memory returndata) = proposal.delegatecall( abi.encodeWithSignature("enacted(address)", proposal) ); if (!_success) { if (returndata.length == 0) revert FailedProposal(proposal); assembly { revert(add(32, returndata), mload(returndata)) } } emit EnactedGovernanceProposal(proposal, msg.sender); } // function required for backwards compatability of the contract interface to allow for the old governance execution call to complete function removeSelf(bytes32) external {} }
/* -*- c-basic-offset: 4 -*- */ // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /* solhint-disable no-inline-assembly */ /** * @title Target for ForwardProxy and EcoInitializable */ abstract contract ForwardTarget { // Must match definition in ForwardProxy // keccak256("com.eco.ForwardProxy.target") uint256 private constant IMPLEMENTATION_SLOT = 0xf86c915dad5894faca0dfa067c58fdf4307406d255ed0a65db394f82b77f53d4; modifier onlyConstruction() { require( implementation() == address(0), "Can only be called during initialization" ); _; } constructor() { _setImplementation(address(this)); } /** Storage initialization of cloned contract * * This is used to initialize the storage of the forwarded contract, and * should (typically) copy or repeat any work that would normally be * done in the constructor of the proxied contract. * * Implementations of ForwardTarget should override this function, * and chain to super.initialize(_self). * * @param _self The address of the original contract instance (the one being * forwarded to). */ function initialize(address _self) public virtual onlyConstruction { address _implAddress = address(ForwardTarget(_self).implementation()); require( _implAddress != address(0), "initialization failure: nothing to implement" ); _setImplementation(_implAddress); } /** Get the address of the proxy target contract. */ function implementation() public view returns (address _impl) { assembly { _impl := sload(IMPLEMENTATION_SLOT) } } /** Set new implementation */ function _setImplementation(address _impl) internal { require(implementation() != _impl, "Implementation already matching"); assembly { sstore(IMPLEMENTATION_SLOT, _impl) } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; // might not need this library StringPacker { // takes a string of 31 or less characters and converts it to bytes32 function pack( string memory unpacked ) internal pure returns (bytes32 packed) { // do not use this function in a lossy way, it will not work // only strings with 31 or less characters are stored in memory packed with their length value require(bytes(unpacked).length < 32); // shift the memory pointer to pack the length of the string into the high byte // by assigning this to the return value, the type of bytes32 means that, when returning, // the pointer continues to read into the string data assembly { packed := mload(add(unpacked, 31)) } } // takes a bytes32 packed in the format above and unpacks it into a string function unpack( bytes32 packed ) internal pure returns (string memory unpacked) { // get the high byte which stores the length of the string when unpacked uint256 len = uint256(packed >> 248); // ensure that the length of the unpacked string doesn't read beyond the input value require(len < 32); // initialize the return value with the length unpacked = string(new bytes(len)); // shift the pointer so that the length will be at the bottom of the word to match string encoding // then store the packed value assembly { // Potentially writes into unallocated memory as the length in the packed form will trail off the end // This is fine as there are no other relevant memory values to overwrite mstore(add(unpacked, 31), packed) } } }
{ "optimizer": { "enabled": true, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"contract Policy","name":"_policy","type":"address"},{"internalType":"address","name":"_pauser","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"string","name":"contractName","type":"string"}],"name":"NonZeroContractAddr","type":"error"},{"inputs":[],"name":"NonZeroPolicyAddr","type":"error"},{"inputs":[],"name":"OnlyBurners","type":"error"},{"inputs":[],"name":"OnlyMinters","type":"error"},{"inputs":[],"name":"OnlySnapshotters","type":"error"},{"inputs":[],"name":"PolicyOnlyFunction","type":"error"},{"inputs":[],"name":"TransferFailed","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"block","type":"uint256"}],"name":"NewSnapshotBlock","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pauser","type":"address"}],"name":"PauserAssignment","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"actor","type":"address"},{"indexed":false,"internalType":"bool","name":"newPermission","type":"bool"}],"name":"UpdatedBurners","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"actor","type":"address"},{"indexed":false,"internalType":"bool","name":"newPermission","type":"bool"}],"name":"UpdatedMinters","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"actor","type":"address"},{"indexed":false,"internalType":"bool","name":"newPermission","type":"bool"}],"name":"UpdatedSnapshotters","type":"event"},{"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"uint256","name":"_value","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"burners","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentSnapshotBlock","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"subtractedValue","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"implementation","outputs":[{"internalType":"address","name":"_impl","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"addedValue","type":"uint256"}],"name":"increaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_self","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_value","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"minters","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pauser","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"permit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"policy","outputs":[{"internalType":"contract Policy","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"roleAdmin","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_impl","type":"address"}],"name":"setImplementation","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_pauser","type":"address"}],"name":"setPauser","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"snapshot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"snapshotters","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupplySnapshot","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"sender","type":"address"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_key","type":"address"},{"internalType":"bool","name":"_value","type":"bool"}],"name":"updateBurners","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_key","type":"address"},{"internalType":"bool","name":"_value","type":"bool"}],"name":"updateMinters","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_key","type":"address"},{"internalType":"bool","name":"_value","type":"bool"}],"name":"updateSnapshotters","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
6101e06040527f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9610140523480156200003757600080fd5b50604051620023eb380380620023eb8339810160408190526200005a9162000400565b60408051808201825260048082526308a869ef60e31b60208084018281528551808701875293845283820192909252845180860190955260018552601960f91b908501528251902060e08190527fad7c5bef027816a800da1736444fb58a807ef4c9603b7848673f7e3a68eb14a56101008190524660a0528694869286928692869286928692869286928592879286928692849283927f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f620001618184846040805160208101859052908101839052606081018290524660808201523060a082015260009060c0016040516020818303038152906040528051906020012090509392505050565b6080523060c05261012052506200018a93508592505050620002f2602090811b62000f2317901c565b6101608181525050620001a881620002f260201b62000f231760201c565b6101805250506004805460ff191690556001600160a01b0382166200023a5760405162461bcd60e51b815260206004820152603c60248201527f556e7265636f76657261626c653a20646f206e6f742073657420746865205f7260448201527f6f6c6541646d696e20617320746865207a65726f20616464726573730000000060648201526084015b60405180910390fd5b6001600160a01b038281166101a05260048054610100600160a81b0319166101009284169283021790556040517fc1c43aa8035756476ab2ca6cfd05434a7fd574b0f3b39c7174828b1fb82acd3e90600090a25050506001600160a01b0382169050620002ba57604051636cf3f82f60e11b815260040160405180910390fd5b6001600160a01b03166101c052620002d2306200030b565b50505050620002e66200039860201b60201c565b5050505050506200043f565b600060208251106200030357600080fd5b50601f015190565b6001600160a01b0381166200032d600080516020620023cb8339815191525490565b6001600160a01b031603620003855760405162461bcd60e51b815260206004820152601f60248201527f496d706c656d656e746174696f6e20616c7265616479206d61746368696e6700604482015260640162000231565b600080516020620023cb83398151915255565b6076805463ffffffff19164363ffffffff8116919091179091556040519081527fa0b956e5b675053f9154bc6b2cf4652191e5914c27ce93a0ad4afa4196166b119060200160405180910390a1565b6001600160a01b0381168114620003fd57600080fd5b50565b600080604083850312156200041457600080fd5b82516200042181620003e7565b60208401519092506200043481620003e7565b809150509250929050565b60805160a05160c05160e05161010051610120516101405161016051610180516101a0516101c051611ee2620004e960003960008181610253015281816108600152818161090d01528181610be00152610ed90152600081816103cb015261063f01526000610a7b0152600061054501526000610da001526000611269015260006112b801526000611293015260006111ec01526000611216015260006112400152611ee26000f3fe608060405234801561001057600080fd5b50600436106102115760003560e01c806378efe53911610125578063a457c2d7116100ad578063d505accf1161007c578063d505accf14610499578063d784d426146104ac578063dd62ed3e146104bf578063e26ddaf3146104f8578063f46eccc41461051b57600080fd5b8063a457c2d71461044d578063a9059cbb14610460578063a9570f1314610473578063c4d66de81461048657600080fd5b806395d89b41116100f457806395d89b41146103ed5780639711715a146103f55780639dc29fac146103fd5780639fd0506d14610410578063a0f3a90e1461042857600080fd5b806378efe539146103a35780637ecebe00146103ab5780638456cb59146103be57806393867fb5146103c657600080fd5b80633644e515116101a85780635c60da1b116101775780635c60da1b146103345780635c975abb146103495780636782674f146103545780636bbb7abc1461036757806370a082311461037a57600080fd5b80633644e515146102fe57806339509351146103065780633f4ba83a1461031957806340c10f191461032157600080fd5b806318160ddd116101e457806318160ddd146102b557806323b872dd146102c75780632d88af4a146102da578063313ce567146102ef57600080fd5b806303d41e0e146102165780630505c8c91461024e57806306fdde031461028d578063095ea7b3146102a2575b600080fd5b610239610224366004611c04565b60436020526000908152604090205460ff1681565b60405190151581526020015b60405180910390f35b6102757f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b039091168152602001610245565b61029561053e565b6040516102459190611c21565b6102396102b0366004611c6f565b61056e565b6003545b604051908152602001610245565b6102396102d5366004611c9b565b610585565b6102ed6102e8366004611c04565b610634565b005b60405160128152602001610245565b6102b961076b565b610239610314366004611c6f565b610775565b6102ed6107b1565b6102ed61032f366004611c6f565b610816565b600080516020611e8d83398151915254610275565b60045460ff16610239565b6102ed610362366004611cdc565b610855565b6102ed610375366004611cdc565b610902565b6102b9610388366004611c04565b6001600160a01b031660009081526001602052604090205490565b6102b96109a7565b6102b96103b9366004611c04565b6109f3565b6102ed610a11565b6102757f000000000000000000000000000000000000000000000000000000000000000081565b610295610a74565b6102ed610a9f565b6102ed61040b366004611c6f565b610ad7565b6004546102759061010090046001600160a01b031681565b6076546104389063ffffffff1681565b60405163ffffffff9091168152602001610245565b61023961045b366004611c6f565b610b2f565b61023961046e366004611c6f565b610bc8565b6102ed610481366004611cdc565b610bd5565b6102ed610494366004611c04565b610c7a565b6102ed6104a7366004611d1a565b610d4c565b6102ed6104ba366004611c04565b610ece565b6102b96104cd366004611d91565b6001600160a01b03918216600090815260026020908152604080832093909416825291909152205490565b610239610506366004611c04565b60aa6020526000908152604090205460ff1681565b610239610529366004611c04565b60426020526000908152604090205460ff1681565b60606105697f0000000000000000000000000000000000000000000000000000000000000000610f3b565b905090565b600061057b338484610fa1565b5060015b92915050565b6000610592848484611063565b6001600160a01b03841660009081526002602090815260408083203384529091529020548281101561061c5760405162461bcd60e51b815260206004820152602860248201527f45524332303a207472616e7366657220616d6f756e74206578636565647320616044820152676c6c6f77616e636560c01b60648201526084015b60405180910390fd5b6106298533858403610fa1565b506001949350505050565b336001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016146106ac5760405162461bcd60e51b815260206004820152601860248201527f45524332305061757361626c653a206e6f742061646d696e00000000000000006044820152606401610613565b6004546001600160a01b036101009091048116908216036107195760405162461bcd60e51b815260206004820152602160248201527f45524332305061757361626c653a206d757374206368616e67652070617573656044820152603960f91b6064820152608401610613565b60048054610100600160a81b0319166101006001600160a01b038416908102919091179091556040517fc1c43aa8035756476ab2ca6cfd05434a7fd574b0f3b39c7174828b1fb82acd3e90600090a250565b60006105696111df565b3360008181526002602090815260408083206001600160a01b0387168452909152812054909161057b9185906107ac908690611dd5565b610fa1565b60045461010090046001600160a01b0316331461080c5760405162461bcd60e51b815260206004820152601960248201527822a92199182830bab9b0b136329d103737ba103830bab9b2b960391b6044820152606401610613565b610814611306565b565b3360009081526042602052604090205460ff16610846576040516338048a8360e01b815260040160405180910390fd5b6108508282611359565b505050565b336001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000161461089e5760405163d8de412960e01b815260040160405180910390fd5b6001600160a01b038216600081815260436020908152604091829020805460ff19168515159081179091558251938452908301527f1be68a6f2d48d1ce3c595f00e67a6a08d8af834f11b5d4e72d5b0be6e95add2f91015b60405180910390a15050565b336001600160a01b037f0000000000000000000000000000000000000000000000000000000000000000161461094b5760405163d8de412960e01b815260040160405180910390fd5b6001600160a01b038216600081815260aa6020908152604091829020805460ff19168515159081179091558251938452908301527f28945214f2349c127afdca7d5f61cef0d2ffa4c5744b0cfc5049e67889aa3a9a91016108f6565b60765460009063ffffffff1643148015906109cf575060765460775463ffffffff9182169116105b156109db575060035490565b5060775464010000000090046001600160e01b031690565b6001600160a01b03811660009081526020819052604081205461057f565b60045461010090046001600160a01b03163314610a6c5760405162461bcd60e51b815260206004820152601960248201527822a92199182830bab9b0b136329d103737ba103830bab9b2b960391b6044820152606401610613565b61081461144e565b60606105697f0000000000000000000000000000000000000000000000000000000000000000610f3b565b33600090815260aa602052604090205460ff16610acf57604051632dbc5b3360e01b815260040160405180910390fd5b61081461148b565b816001600160a01b0381163314801590610b0157503360009081526043602052604090205460ff16155b15610b1f5760405163356454bb60e21b815260040160405180910390fd5b610b2983836114d5565b50505050565b3360009081526002602090815260408083206001600160a01b038616845290915281205482811015610bb15760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b6064820152608401610613565b610bbe3385858403610fa1565b5060019392505050565b600061057b338484611063565b336001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001614610c1e5760405163d8de412960e01b815260040160405180910390fd5b6001600160a01b038216600081815260426020908152604091829020805460ff19168515159081179091558251938452908301527fb12309378ed0478f557372c22a6a79f84d7e8d817ede419b9bb75b05a9f6a81291016108f6565b6000610c92600080516020611e8d8339815191525490565b6001600160a01b031614610cb85760405162461bcd60e51b815260040161061390611de8565b610cc1816115d8565b806001600160a01b0316639fd0506d6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610cff573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d239190611e30565b600460016101000a8154816001600160a01b0302191690836001600160a01b0316021790555050565b83421115610d9c5760405162461bcd60e51b815260206004820152601d60248201527f45524332305065726d69743a206578706972656420646561646c696e650000006044820152606401610613565b60007f0000000000000000000000000000000000000000000000000000000000000000888888610de98c6001600160a01b0316600090815260208190526040902080546001810190915590565b6040805160208101969096526001600160a01b0394851690860152929091166060840152608083015260a082015260c0810186905260e0016040516020818303038152906040528051906020012090506000610e4482611627565b90506000610e5482878787611675565b9050896001600160a01b0316816001600160a01b031614610eb75760405162461bcd60e51b815260206004820152601e60248201527f45524332305065726d69743a20696e76616c6964207369676e617475726500006044820152606401610613565b610ec28a8a8a610fa1565b50505050505050505050565b336001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001614610f175760405163d8de412960e01b815260040160405180910390fd5b610f208161169d565b50565b60006020825110610f3357600080fd5b50601f015190565b606060f882901c60208110610f4f57600080fd5b8067ffffffffffffffff811115610f6857610f68611e4d565b6040519080825280601f01601f191660200182016040528015610f92576020820181803683370190505b50915082601f83015250919050565b6001600160a01b0382166110025760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610613565b6001600160a01b0383811660008181526002602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b6001600160a01b0382166110c55760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610613565b60006110d2848484611725565b6001600160a01b0385166000908152600160205260409020549091508181101561114d5760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610613565b6001600160a01b03808616600090815260016020526040808220858503905591861681529081208054849290611184908490611dd5565b92505081905550836001600160a01b0316856001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040516111d091815260200190565b60405180910390a35050505050565b6000306001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001614801561123857507f000000000000000000000000000000000000000000000000000000000000000046145b1561126257507f000000000000000000000000000000000000000000000000000000000000000090565b50604080517f00000000000000000000000000000000000000000000000000000000000000006020808301919091527f0000000000000000000000000000000000000000000000000000000000000000828401527f000000000000000000000000000000000000000000000000000000000000000060608301524660808301523060a0808401919091528351808403909101815260c0909201909252805191012090565b61130e61176b565b6004805460ff191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa335b6040516001600160a01b0390911681526020015b60405180910390a1565b60006001600160a01b0383166113b15760405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606401610613565b60006113bf60008585611725565b905080600360008282546113d39190611dd5565b90915550506001600160a01b03841660009081526001602052604081208054839290611400908490611dd5565b90915550506040518381526001600160a01b038516906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9060200160405180910390a35b9392505050565b6114566117b4565b6004805460ff191660011790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a25861133b3390565b6076805463ffffffff19164363ffffffff8116919091179091556040519081527fa0b956e5b675053f9154bc6b2cf4652191e5914c27ce93a0ad4afa4196166b119060200161134f565b6000806114e484600085611725565b6001600160a01b0385166000908152600160205260409020549091508181101561155b5760405162461bcd60e51b815260206004820152602260248201527f45524332303a206275726e20616d6f756e7420657863656564732062616c616e604482015261636560f01b6064820152608401610613565b6001600160a01b038516600090815260016020526040812083830390556003805484929061158a908490611e63565b90915550506040518481526000906001600160a01b038716907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9060200160405180910390a3509392505050565b60006115f0600080516020611e8d8339815191525490565b6001600160a01b0316146116165760405162461bcd60e51b815260040161061390611de8565b61161f816117fa565b610f2061148b565b600061057f6116346111df565b8360405161190160f01b6020820152602281018390526042810182905260009060620160405160208183030381529060405280519060200120905092915050565b600080600061168687878787611916565b91509150611693816119da565b5095945050505050565b806001600160a01b03166116bd600080516020611e8d8339815191525490565b6001600160a01b0316036117135760405162461bcd60e51b815260206004820152601f60248201527f496d706c656d656e746174696f6e20616c7265616479206d61746368696e67006044820152606401610613565b600080516020611e8d83398151915255565b60006001600160a01b0384166117425761173d611b24565b611758565b6001600160a01b03831661175857611758611b24565b611763848484611bde565b949350505050565b60045460ff166108145760405162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b6044820152606401610613565b60045460ff16156108145760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b6044820152606401610613565b6000611812600080516020611e8d8339815191525490565b6001600160a01b0316146118385760405162461bcd60e51b815260040161061390611de8565b6000816001600160a01b0316635c60da1b6040518163ffffffff1660e01b8152600401602060405180830381865afa158015611878573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061189c9190611e30565b90506001600160a01b0381166119095760405162461bcd60e51b815260206004820152602c60248201527f696e697469616c697a6174696f6e206661696c7572653a206e6f7468696e672060448201526b1d1bc81a5b5c1b195b595b9d60a21b6064820152608401610613565b6119128161169d565b5050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a083111561194d57506000905060036119d1565b6040805160008082526020820180845289905260ff881692820192909252606081018690526080810185905260019060a0016020604051602081039080840390855afa1580156119a1573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b0381166119ca576000600192509250506119d1565b9150600090505b94509492505050565b60008160048111156119ee576119ee611e76565b036119f65750565b6001816004811115611a0a57611a0a611e76565b03611a575760405162461bcd60e51b815260206004820152601860248201527f45434453413a20696e76616c6964207369676e617475726500000000000000006044820152606401610613565b6002816004811115611a6b57611a6b611e76565b03611ab85760405162461bcd60e51b815260206004820152601f60248201527f45434453413a20696e76616c6964207369676e6174757265206c656e677468006044820152606401610613565b6003816004811115611acc57611acc611e76565b03610f205760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202773272076616c604482015261756560f01b6064820152608401610613565b60765463ffffffff16438103611b375750565b60775463ffffffff80831691161015610f20576003546001600160e01b03811115611bc05760405162461bcd60e51b815260206004820152603360248201527f566f7465536e617073686f74733a206e657720736e617073686f742063616e6e6044820152726f742062652063617374656420736166656c7960681b6064820152608401610613565b6001600160e01b03166401000000000263ffffffff82161760775550565b6000611be86117b4565b5092915050565b6001600160a01b0381168114610f2057600080fd5b600060208284031215611c1657600080fd5b813561144781611bef565b600060208083528351808285015260005b81811015611c4e57858101830151858201604001528201611c32565b506000604082860101526040601f19601f8301168501019250505092915050565b60008060408385031215611c8257600080fd5b8235611c8d81611bef565b946020939093013593505050565b600080600060608486031215611cb057600080fd5b8335611cbb81611bef565b92506020840135611ccb81611bef565b929592945050506040919091013590565b60008060408385031215611cef57600080fd5b8235611cfa81611bef565b915060208301358015158114611d0f57600080fd5b809150509250929050565b600080600080600080600060e0888a031215611d3557600080fd5b8735611d4081611bef565b96506020880135611d5081611bef565b95506040880135945060608801359350608088013560ff81168114611d7457600080fd5b9699959850939692959460a0840135945060c09093013592915050565b60008060408385031215611da457600080fd5b8235611daf81611bef565b91506020830135611d0f81611bef565b634e487b7160e01b600052601160045260246000fd5b8082018082111561057f5761057f611dbf565b60208082526028908201527f43616e206f6e6c792062652063616c6c656420647572696e6720696e697469616040820152673634bd30ba34b7b760c11b606082015260800190565b600060208284031215611e4257600080fd5b815161144781611bef565b634e487b7160e01b600052604160045260246000fd5b8181038181111561057f5761057f611dbf565b634e487b7160e01b600052602160045260246000fdfef86c915dad5894faca0dfa067c58fdf4307406d255ed0a65db394f82b77f53d4a26469706673582212202713f905dcee46f8fbfc11eafd35c52912dbfc80d1312ef4ccba68ca9a39e4bc64736f6c63430008110033f86c915dad5894faca0dfa067c58fdf4307406d255ed0a65db394f82b77f53d40000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e680000000000000000000000000000000000000000000000000000000000000000
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106102115760003560e01c806378efe53911610125578063a457c2d7116100ad578063d505accf1161007c578063d505accf14610499578063d784d426146104ac578063dd62ed3e146104bf578063e26ddaf3146104f8578063f46eccc41461051b57600080fd5b8063a457c2d71461044d578063a9059cbb14610460578063a9570f1314610473578063c4d66de81461048657600080fd5b806395d89b41116100f457806395d89b41146103ed5780639711715a146103f55780639dc29fac146103fd5780639fd0506d14610410578063a0f3a90e1461042857600080fd5b806378efe539146103a35780637ecebe00146103ab5780638456cb59146103be57806393867fb5146103c657600080fd5b80633644e515116101a85780635c60da1b116101775780635c60da1b146103345780635c975abb146103495780636782674f146103545780636bbb7abc1461036757806370a082311461037a57600080fd5b80633644e515146102fe57806339509351146103065780633f4ba83a1461031957806340c10f191461032157600080fd5b806318160ddd116101e457806318160ddd146102b557806323b872dd146102c75780632d88af4a146102da578063313ce567146102ef57600080fd5b806303d41e0e146102165780630505c8c91461024e57806306fdde031461028d578063095ea7b3146102a2575b600080fd5b610239610224366004611c04565b60436020526000908152604090205460ff1681565b60405190151581526020015b60405180910390f35b6102757f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e6881565b6040516001600160a01b039091168152602001610245565b61029561053e565b6040516102459190611c21565b6102396102b0366004611c6f565b61056e565b6003545b604051908152602001610245565b6102396102d5366004611c9b565b610585565b6102ed6102e8366004611c04565b610634565b005b60405160128152602001610245565b6102b961076b565b610239610314366004611c6f565b610775565b6102ed6107b1565b6102ed61032f366004611c6f565b610816565b600080516020611e8d83398151915254610275565b60045460ff16610239565b6102ed610362366004611cdc565b610855565b6102ed610375366004611cdc565b610902565b6102b9610388366004611c04565b6001600160a01b031660009081526001602052604090205490565b6102b96109a7565b6102b96103b9366004611c04565b6109f3565b6102ed610a11565b6102757f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e6881565b610295610a74565b6102ed610a9f565b6102ed61040b366004611c6f565b610ad7565b6004546102759061010090046001600160a01b031681565b6076546104389063ffffffff1681565b60405163ffffffff9091168152602001610245565b61023961045b366004611c6f565b610b2f565b61023961046e366004611c6f565b610bc8565b6102ed610481366004611cdc565b610bd5565b6102ed610494366004611c04565b610c7a565b6102ed6104a7366004611d1a565b610d4c565b6102ed6104ba366004611c04565b610ece565b6102b96104cd366004611d91565b6001600160a01b03918216600090815260026020908152604080832093909416825291909152205490565b610239610506366004611c04565b60aa6020526000908152604090205460ff1681565b610239610529366004611c04565b60426020526000908152604090205460ff1681565b60606105697f0445434f78000000000000000000000000000000000000000000000000000000610f3b565b905090565b600061057b338484610fa1565b5060015b92915050565b6000610592848484611063565b6001600160a01b03841660009081526002602090815260408083203384529091529020548281101561061c5760405162461bcd60e51b815260206004820152602860248201527f45524332303a207472616e7366657220616d6f756e74206578636565647320616044820152676c6c6f77616e636560c01b60648201526084015b60405180910390fd5b6106298533858403610fa1565b506001949350505050565b336001600160a01b037f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e6816146106ac5760405162461bcd60e51b815260206004820152601860248201527f45524332305061757361626c653a206e6f742061646d696e00000000000000006044820152606401610613565b6004546001600160a01b036101009091048116908216036107195760405162461bcd60e51b815260206004820152602160248201527f45524332305061757361626c653a206d757374206368616e67652070617573656044820152603960f91b6064820152608401610613565b60048054610100600160a81b0319166101006001600160a01b038416908102919091179091556040517fc1c43aa8035756476ab2ca6cfd05434a7fd574b0f3b39c7174828b1fb82acd3e90600090a250565b60006105696111df565b3360008181526002602090815260408083206001600160a01b0387168452909152812054909161057b9185906107ac908690611dd5565b610fa1565b60045461010090046001600160a01b0316331461080c5760405162461bcd60e51b815260206004820152601960248201527822a92199182830bab9b0b136329d103737ba103830bab9b2b960391b6044820152606401610613565b610814611306565b565b3360009081526042602052604090205460ff16610846576040516338048a8360e01b815260040160405180910390fd5b6108508282611359565b505050565b336001600160a01b037f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e68161461089e5760405163d8de412960e01b815260040160405180910390fd5b6001600160a01b038216600081815260436020908152604091829020805460ff19168515159081179091558251938452908301527f1be68a6f2d48d1ce3c595f00e67a6a08d8af834f11b5d4e72d5b0be6e95add2f91015b60405180910390a15050565b336001600160a01b037f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e68161461094b5760405163d8de412960e01b815260040160405180910390fd5b6001600160a01b038216600081815260aa6020908152604091829020805460ff19168515159081179091558251938452908301527f28945214f2349c127afdca7d5f61cef0d2ffa4c5744b0cfc5049e67889aa3a9a91016108f6565b60765460009063ffffffff1643148015906109cf575060765460775463ffffffff9182169116105b156109db575060035490565b5060775464010000000090046001600160e01b031690565b6001600160a01b03811660009081526020819052604081205461057f565b60045461010090046001600160a01b03163314610a6c5760405162461bcd60e51b815260206004820152601960248201527822a92199182830bab9b0b136329d103737ba103830bab9b2b960391b6044820152606401610613565b61081461144e565b60606105697f0445434f78000000000000000000000000000000000000000000000000000000610f3b565b33600090815260aa602052604090205460ff16610acf57604051632dbc5b3360e01b815260040160405180910390fd5b61081461148b565b816001600160a01b0381163314801590610b0157503360009081526043602052604090205460ff16155b15610b1f5760405163356454bb60e21b815260040160405180910390fd5b610b2983836114d5565b50505050565b3360009081526002602090815260408083206001600160a01b038616845290915281205482811015610bb15760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b6064820152608401610613565b610bbe3385858403610fa1565b5060019392505050565b600061057b338484611063565b336001600160a01b037f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e681614610c1e5760405163d8de412960e01b815260040160405180910390fd5b6001600160a01b038216600081815260426020908152604091829020805460ff19168515159081179091558251938452908301527fb12309378ed0478f557372c22a6a79f84d7e8d817ede419b9bb75b05a9f6a81291016108f6565b6000610c92600080516020611e8d8339815191525490565b6001600160a01b031614610cb85760405162461bcd60e51b815260040161061390611de8565b610cc1816115d8565b806001600160a01b0316639fd0506d6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610cff573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d239190611e30565b600460016101000a8154816001600160a01b0302191690836001600160a01b0316021790555050565b83421115610d9c5760405162461bcd60e51b815260206004820152601d60248201527f45524332305065726d69743a206578706972656420646561646c696e650000006044820152606401610613565b60007f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9888888610de98c6001600160a01b0316600090815260208190526040902080546001810190915590565b6040805160208101969096526001600160a01b0394851690860152929091166060840152608083015260a082015260c0810186905260e0016040516020818303038152906040528051906020012090506000610e4482611627565b90506000610e5482878787611675565b9050896001600160a01b0316816001600160a01b031614610eb75760405162461bcd60e51b815260206004820152601e60248201527f45524332305065726d69743a20696e76616c6964207369676e617475726500006044820152606401610613565b610ec28a8a8a610fa1565b50505050505050505050565b336001600160a01b037f0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e681614610f175760405163d8de412960e01b815260040160405180910390fd5b610f208161169d565b50565b60006020825110610f3357600080fd5b50601f015190565b606060f882901c60208110610f4f57600080fd5b8067ffffffffffffffff811115610f6857610f68611e4d565b6040519080825280601f01601f191660200182016040528015610f92576020820181803683370190505b50915082601f83015250919050565b6001600160a01b0382166110025760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610613565b6001600160a01b0383811660008181526002602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b6001600160a01b0382166110c55760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610613565b60006110d2848484611725565b6001600160a01b0385166000908152600160205260409020549091508181101561114d5760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610613565b6001600160a01b03808616600090815260016020526040808220858503905591861681529081208054849290611184908490611dd5565b92505081905550836001600160a01b0316856001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040516111d091815260200190565b60405180910390a35050505050565b6000306001600160a01b037f0000000000000000000000007b06ab9e43ef35e58079cc0cf388625a1977b1fa1614801561123857507f000000000000000000000000000000000000000000000000000000000000000146145b1561126257507f2f9b4de20f3a8c4876655db162be65bc23ff05aeec5dc501544d19b5a0af867490565b50604080517f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f6020808301919091527fe10ab6c94f1da69921a0ca1c1b96b4fc339699153931c9bfd565e91f44c19b0b828401527fad7c5bef027816a800da1736444fb58a807ef4c9603b7848673f7e3a68eb14a560608301524660808301523060a0808401919091528351808403909101815260c0909201909252805191012090565b61130e61176b565b6004805460ff191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa335b6040516001600160a01b0390911681526020015b60405180910390a1565b60006001600160a01b0383166113b15760405162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f2061646472657373006044820152606401610613565b60006113bf60008585611725565b905080600360008282546113d39190611dd5565b90915550506001600160a01b03841660009081526001602052604081208054839290611400908490611dd5565b90915550506040518381526001600160a01b038516906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9060200160405180910390a35b9392505050565b6114566117b4565b6004805460ff191660011790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a25861133b3390565b6076805463ffffffff19164363ffffffff8116919091179091556040519081527fa0b956e5b675053f9154bc6b2cf4652191e5914c27ce93a0ad4afa4196166b119060200161134f565b6000806114e484600085611725565b6001600160a01b0385166000908152600160205260409020549091508181101561155b5760405162461bcd60e51b815260206004820152602260248201527f45524332303a206275726e20616d6f756e7420657863656564732062616c616e604482015261636560f01b6064820152608401610613565b6001600160a01b038516600090815260016020526040812083830390556003805484929061158a908490611e63565b90915550506040518481526000906001600160a01b038716907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9060200160405180910390a3509392505050565b60006115f0600080516020611e8d8339815191525490565b6001600160a01b0316146116165760405162461bcd60e51b815260040161061390611de8565b61161f816117fa565b610f2061148b565b600061057f6116346111df565b8360405161190160f01b6020820152602281018390526042810182905260009060620160405160208183030381529060405280519060200120905092915050565b600080600061168687878787611916565b91509150611693816119da565b5095945050505050565b806001600160a01b03166116bd600080516020611e8d8339815191525490565b6001600160a01b0316036117135760405162461bcd60e51b815260206004820152601f60248201527f496d706c656d656e746174696f6e20616c7265616479206d61746368696e67006044820152606401610613565b600080516020611e8d83398151915255565b60006001600160a01b0384166117425761173d611b24565b611758565b6001600160a01b03831661175857611758611b24565b611763848484611bde565b949350505050565b60045460ff166108145760405162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b6044820152606401610613565b60045460ff16156108145760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b6044820152606401610613565b6000611812600080516020611e8d8339815191525490565b6001600160a01b0316146118385760405162461bcd60e51b815260040161061390611de8565b6000816001600160a01b0316635c60da1b6040518163ffffffff1660e01b8152600401602060405180830381865afa158015611878573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061189c9190611e30565b90506001600160a01b0381166119095760405162461bcd60e51b815260206004820152602c60248201527f696e697469616c697a6174696f6e206661696c7572653a206e6f7468696e672060448201526b1d1bc81a5b5c1b195b595b9d60a21b6064820152608401610613565b6119128161169d565b5050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a083111561194d57506000905060036119d1565b6040805160008082526020820180845289905260ff881692820192909252606081018690526080810185905260019060a0016020604051602081039080840390855afa1580156119a1573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b0381166119ca576000600192509250506119d1565b9150600090505b94509492505050565b60008160048111156119ee576119ee611e76565b036119f65750565b6001816004811115611a0a57611a0a611e76565b03611a575760405162461bcd60e51b815260206004820152601860248201527f45434453413a20696e76616c6964207369676e617475726500000000000000006044820152606401610613565b6002816004811115611a6b57611a6b611e76565b03611ab85760405162461bcd60e51b815260206004820152601f60248201527f45434453413a20696e76616c6964207369676e6174757265206c656e677468006044820152606401610613565b6003816004811115611acc57611acc611e76565b03610f205760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202773272076616c604482015261756560f01b6064820152608401610613565b60765463ffffffff16438103611b375750565b60775463ffffffff80831691161015610f20576003546001600160e01b03811115611bc05760405162461bcd60e51b815260206004820152603360248201527f566f7465536e617073686f74733a206e657720736e617073686f742063616e6e6044820152726f742062652063617374656420736166656c7960681b6064820152608401610613565b6001600160e01b03166401000000000263ffffffff82161760775550565b6000611be86117b4565b5092915050565b6001600160a01b0381168114610f2057600080fd5b600060208284031215611c1657600080fd5b813561144781611bef565b600060208083528351808285015260005b81811015611c4e57858101830151858201604001528201611c32565b506000604082860101526040601f19601f8301168501019250505092915050565b60008060408385031215611c8257600080fd5b8235611c8d81611bef565b946020939093013593505050565b600080600060608486031215611cb057600080fd5b8335611cbb81611bef565b92506020840135611ccb81611bef565b929592945050506040919091013590565b60008060408385031215611cef57600080fd5b8235611cfa81611bef565b915060208301358015158114611d0f57600080fd5b809150509250929050565b600080600080600080600060e0888a031215611d3557600080fd5b8735611d4081611bef565b96506020880135611d5081611bef565b95506040880135945060608801359350608088013560ff81168114611d7457600080fd5b9699959850939692959460a0840135945060c09093013592915050565b60008060408385031215611da457600080fd5b8235611daf81611bef565b91506020830135611d0f81611bef565b634e487b7160e01b600052601160045260246000fd5b8082018082111561057f5761057f611dbf565b60208082526028908201527f43616e206f6e6c792062652063616c6c656420647572696e6720696e697469616040820152673634bd30ba34b7b760c11b606082015260800190565b600060208284031215611e4257600080fd5b815161144781611bef565b634e487b7160e01b600052604160045260246000fd5b8181038181111561057f5761057f611dbf565b634e487b7160e01b600052602160045260246000fdfef86c915dad5894faca0dfa067c58fdf4307406d255ed0a65db394f82b77f53d4a26469706673582212202713f905dcee46f8fbfc11eafd35c52912dbfc80d1312ef4ccba68ca9a39e4bc64736f6c63430008110033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e680000000000000000000000000000000000000000000000000000000000000000
-----Decoded View---------------
Arg [0] : _policy (address): 0x8c02D4cc62F79AcEB652321a9f8988c0f6E71E68
Arg [1] : _pauser (address): 0x0000000000000000000000000000000000000000
-----Encoded View---------------
2 Constructor Arguments found :
Arg [0] : 0000000000000000000000008c02d4cc62f79aceb652321a9f8988c0f6e71e68
Arg [1] : 0000000000000000000000000000000000000000000000000000000000000000
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 27 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.