Overview
S Balance
S Value
$0.00More Info
Private Name Tags
ContractCreator
Loading...
Loading
Contract Source Code Verified (Exact Match)
Contract Name:
MarbleMinter
Compiler Version
v0.8.28+commit.7893614a
Optimization Enabled:
Yes with 333 runs
Other Settings:
cancun EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; import {Math} from "@openzeppelin/contracts/utils/math/Math.sol"; import {Initializable} from "@openzeppelin/contracts/proxy/utils/Initializable.sol"; import {SafeERC20, IERC20} from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import {Errors} from "contracts/libraries/Errors.sol"; import {MinterStorage} from "contracts/libraries/MinterStorage.sol"; import {IERC20Extended} from "contracts/interfaces/IERC20Extended.sol"; import {IMinter} from "contracts/interfaces/IMinter.sol"; import {IMarbleMinter} from "contracts/interfaces/IMarbleMinter.sol"; import {IAccessHub} from "contracts/interfaces/IAccessHub.sol"; import {IPairFactory} from "contracts/interfaces/IPairFactory.sol"; import {IFeeRecipientFactory} from "contracts/interfaces/IFeeRecipientFactory.sol"; import {IFeeRecipient} from "contracts/interfaces/IFeeRecipient.sol"; import {IPair} from "contracts/interfaces/IPair.sol"; import {IVoter} from "contracts/interfaces/IVoter.sol"; import {IVoteModule} from "contracts/interfaces/IVoteModule.sol"; import {IXShadow} from "contracts/interfaces/IXShadow.sol"; /// @title Marble Zone Minter /// @author /// @notice Aggregates and passes on Shadow emissions and legacy pair fees /// @dev This contract will become the governor of Old Voter contract MarbleMinter is IMarbleMinter, Initializable { using SafeERC20 for IERC20; /// @notice basis invariant 1_000_000 = 100% uint256 public constant DENOM = 1_000_000; /// @notice max deviation of 20% per epoch uint256 public constant MAX_DEVIATION = 200_000; /// @notice initial supply of 3m SHADOW uint256 public constant INITIAL_SUPPLY = 3_000_000 * 1e18; /// @notice max supply of 10m SHADOW uint256 public constant MAX_SUPPLY = 10_000_000 * 1e18; IMinter public constant oldMinter = IMinter(0xc7022F359cD1bDa8aB8a19d1F19d769cbf7F3765); IVoter public constant oldVoter = IVoter(0x3aF1dD7A2755201F8e2D6dCDA1a61d9f54838f4f); IVoteModule public constant voteModule = IVoteModule(0xDCB5A24ec708cc13cee12bFE6799A78a79b666b4); address internal constant shadowGauge = 0xCBcAd939E2bbbe01850A141F204f25DF63b8FC5B; ///////////////// /// Modifiers /// ///////////////// modifier onlyGovernance() { _onlyGovernance(); _; } function _onlyGovernance() internal view { require(msg.sender == MinterStorage.getStorage().accessHub, Errors.NOT_AUTHORIZED(msg.sender)); } modifier onlyVoter() { _onlyVoter(); _; } function _onlyVoter() internal view { require(msg.sender == MinterStorage.getStorage().voter, Errors.NOT_AUTHORIZED(msg.sender)); } modifier checkFeeSplit() { _checkFeeSplit(); _; } function _checkFeeSplit() internal view { /// @dev we need feeSplitWhenNoGauge to redirect fees to this contract IPairFactory legacyFactory = IPairFactory(oldVoter.legacyFactory()); require(legacyFactory.feeSplitWhenNoGauge(), Errors.FEE_SPLIT_WHEN_NO_GAUGE_IS_OFF()); require(legacyFactory.feeSplit() > 0, Errors.FEE_SPLIT_WHEN_NO_GAUGE_IS_OFF()); } //////////////////////////////////// /// Constructor and Initializers /// //////////////////////////////////// constructor() { _disableInitializers(); } /// @dev should be called with upgradeToAndCall so no auth is needed function initialize(address _accessHub, address _operator) external initializer { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); $.accessHub = _accessHub; $.operator = _operator; } /// @inheritdoc IMinter /// @dev _initialMultiplier is be based on 1m = 100% function kickoff( address _shadow, address _voter, uint256, // _initialWeeklyEmissions is unused, it copies from the old minter uint256, // _initialMultiplier is unused, it copies from the old minter address _xShadow ) external onlyGovernance { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); /// @dev ensure the emissions token isn't set yet require(address($.shadow) == address(0), Errors.STARTED()); require(_xShadow != address(0), Errors.INVALID_CONTRACT()); require(_voter != address(0), Errors.INVALID_CONTRACT()); require(_shadow != address(0), Errors.INVALID_CONTRACT()); $.shadow = IERC20Extended(_shadow); $.xShadow = _xShadow; $.voter = _voter; $.accessHub = IVoter(_voter).accessHub(); /// @dev copy weeklyEmissions from old minter $.weeklyEmissions = oldMinter.weeklyEmissions(); /// @dev copy emissionsMultiplier from old minter $.emissionsMultiplier = oldMinter.emissionsMultiplier() * 100; // BASIS OF 10_000 => 1_000_000 /// @dev set the active period to the current $.activePeriod = getPeriod(); /// @dev set the last update as the last period so emissions can be updated once if needed $.lastMultiplierUpdate = $.activePeriod - 1; /// @dev copy first period from old minter $.firstPeriod = oldMinter.firstPeriod(); emit SetVoter(_voter); /// @dev Deposit xShadow into Vote Module (Shadow to be transferred in beforehand) IERC20(_shadow).approve(address(_xShadow), 1000); IXShadow(_xShadow).convertEmissionsToken(1000); IERC20(_xShadow).approve(address(voteModule), 1000); voteModule.deposit(1000); } /// @inheritdoc IMinter function startEmissions() external { /// @dev empty for marble minter, all handled in kickoff } ////////////////////// /// View Functions /// ////////////////////// /// @inheritdoc IMinter function weeklyEmissions() external view returns (uint256) { return MinterStorage.getStorage().weeklyEmissions; } /// @inheritdoc IMinter function emissionsMultiplier() external view returns (uint256) { return MinterStorage.getStorage().emissionsMultiplier; } /// @inheritdoc IMinter function firstPeriod() external view returns (uint256) { return MinterStorage.getStorage().firstPeriod; } /// @inheritdoc IMinter function activePeriod() external view returns (uint256) { return MinterStorage.getStorage().activePeriod; } /// @inheritdoc IMinter function lastMultiplierUpdate() external view returns (uint256) { return MinterStorage.getStorage().lastMultiplierUpdate; } /// @inheritdoc IMinter function operator() external view returns (address) { return MinterStorage.getStorage().operator; } /// @inheritdoc IMinter function accessHub() external view returns (address) { return MinterStorage.getStorage().accessHub; } /// @inheritdoc IMinter function xShadow() external view returns (address) { return MinterStorage.getStorage().xShadow; } /// @inheritdoc IMinter function voter() external view returns (address) { return MinterStorage.getStorage().voter; } /// @inheritdoc IMinter function shadow() external view returns (address) { return address(MinterStorage.getStorage().shadow); } /// @inheritdoc IMinter function calculateWeeklyEmissions() public view returns (uint256) { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); /// @dev fetch proposed emissions uint256 _weeklyEmissions = ($.weeklyEmissions * $.emissionsMultiplier) / DENOM; /// @dev if it's zero if (_weeklyEmissions == 0) return 0; /// @dev if minting goes over the max supply if ($.shadow.totalSupply() + _weeklyEmissions > MAX_SUPPLY) { /// @dev update value to difference _weeklyEmissions = MAX_SUPPLY - $.shadow.totalSupply(); } return _weeklyEmissions; } /// @inheritdoc IMinter function getPeriod() public view returns (uint256 period) { period = block.timestamp / 1 weeks; } /// @inheritdoc IMinter function getEpoch() public view returns (uint256 _epoch) { return getPeriod() - MinterStorage.getStorage().firstPeriod; } /////////////////// /// Period Flip /// /////////////////// /// @inheritdoc IMinter function updatePeriod() external returns (uint256 period) { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); require($.firstPeriod != 0, Errors.EMISSIONS_NOT_STARTED()); /// @dev set period equal to the current activePeriod period = $.activePeriod; /// @dev if >= Thursday 0 UTC if (getPeriod() > period) { /// @dev fetch the current period period = getPeriod(); /// @dev set the active period to the new period $.activePeriod = period; /// @dev calculate the weekly emissions uint256 _weeklyEmissions = calculateWeeklyEmissions(); /// @dev set global value to the above calculated emissions $.weeklyEmissions = _weeklyEmissions; /// @dev update Old Minter oldMinter.updatePeriod(); /// @dev claim emission from Old Voter bool marbled = oldVoter.governor() == address(this); if (marbled) { oldVoter.stuckEmissionsRecovery(shadowGauge, period); } /// @dev vote on shadow gauge if (marbled) { oldVoter.reviveGauge(shadowGauge); } address pool = oldVoter.poolForGauge(shadowGauge); address[] memory pools = new address[](1); uint256[] memory weights = new uint256[](1); pools[0] = pool; weights[0] = 1 ether; oldVoter.vote(address(this), pools, weights); if (marbled) { oldVoter.killGauge(shadowGauge); } if ($.weeklyEmissions > 0 && marbled) { /// @dev approvals for shadow on voter $.shadow.approve($.voter, _weeklyEmissions); /// @dev notify emissions to the voter contract IVoter($.voter).notifyRewardAmount(_weeklyEmissions); } } } ////////////////// /// Governance /// ////////////////// /// @inheritdoc IMarbleMinter function setAccessHub(address _accessHub) external onlyGovernance { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); $.accessHub = _accessHub; } /// @inheritdoc IMarbleMinter function setOperator(address _operator) external onlyGovernance { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); $.operator = _operator; } /// @inheritdoc IMinter function updateEmissionsMultiplier(uint256 _emissionsMultiplier) external onlyGovernance { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); /// @dev ensure that the last time the multiplier was updated was not the same period require($.lastMultiplierUpdate != $.activePeriod, Errors.SAME_PERIOD()); /// @dev set the last update to the current period $.lastMultiplierUpdate = $.activePeriod; /// @dev ensure the multiplier actually is diff require($.emissionsMultiplier != _emissionsMultiplier, Errors.NO_CHANGE()); /// @dev placeholder for deviation uint256 deviation; /// @dev check which way to subtract deviation = $.emissionsMultiplier > _emissionsMultiplier ? ($.emissionsMultiplier - _emissionsMultiplier) : (_emissionsMultiplier - $.emissionsMultiplier); /// @dev require deviation is not above 20% per epoch require(deviation <= MAX_DEVIATION, Errors.TOO_HIGH()); /// @dev set new values $.emissionsMultiplier = _emissionsMultiplier; emit EmissionsMultiplierUpdated(_emissionsMultiplier); } function syncSettings() external onlyGovernance { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); $.weeklyEmissions = oldMinter.weeklyEmissions(); $.emissionsMultiplier = oldMinter.emissionsMultiplier() * 100; // BASIS OF 10_000 => 1_000_000 } /// @inheritdoc IMarbleMinter function rescueTokens(address token, uint256 amount) external onlyGovernance { IERC20(token).safeTransfer(msg.sender, amount); } ///////////////// /// Migration /// ///////////////// /// @inheritdoc IMarbleMinter function migration(address[] calldata gauges) external onlyGovernance checkFeeSplit { uint256 length = gauges.length; for (uint256 i = 0; i < length; i++) { address gauge = gauges[i]; _migrateCreatedGauge(gauge); } } /// @inheritdoc IMarbleMinter /// @dev needed to redirect fees to the new feeRecipients function postCreateLegacyGaugeHook(address pool) external onlyVoter checkFeeSplit { if (oldVoter.governor() != address(this)) return; /// @dev whitelist tokens temporarily, revoke after the gauge is created address token0 = IPair(pool).token0(); address token1 = IPair(pool).token1(); oldVoter.whitelist(token0); oldVoter.whitelist(token1); address gauge = oldVoter.gaugeForPool(pool); if (gauge == address(0)) { gauge = oldVoter.createGauge(pool); } _migrateCreatedGauge(gauge); } function _migrateCreatedGauge(address gauge) internal { /// @dev fetch addresses IPair pool = IPair(oldVoter.poolForGauge(gauge)); address token0 = pool.token0(); address token1 = pool.token1(); /// @dev disable whitelists if needed if (oldVoter.isWhitelisted(token0)) { oldVoter.revokeWhitelist(token0); } if (oldVoter.isWhitelisted(token1)) { oldVoter.revokeWhitelist(token1); } /// @dev revive and kill gauge to set feeRecipient if (!oldVoter.isAlive(gauge)) { oldVoter.reviveGauge(gauge); } oldVoter.killGauge(gauge); } // function postCreateClGaugeHook(address token0, address token1, int24 tickSpacing) external onlyVoter { // /// @dev whitelist tokens temporarily, revoke after the gauge is created // oldVoter.whitelist(token0); // oldVoter.whitelist(token1); // address gauge = oldVoter.createCLGauge(token0, token1, tickSpacing); // _migrateCreatedGauge(gauge); // } /////////////////////////////////// /// Legacy Pair Fee Redirection /// /////////////////////////////////// /// @inheritdoc IMarbleMinter /// @dev no auth needed for this function redirectFees(uint256 start, uint256 batchSize) external { MinterStorage.MinterState storage $ = MinterStorage.getStorage(); IVoter _voter = IVoter($.voter); address treasury = IAccessHub($.accessHub).treasury(); uint256 length = _voter.getPoolsLength(); IFeeRecipientFactory feeRecipientFactory = IFeeRecipientFactory(_voter.feeRecipientFactory()); uint256 end = start + batchSize; if (end > length) { end = length; } for (uint256 i = start; i < end; i++) { address gauge = _voter.getGauge(i); if (_voter.isLegacyGauge(gauge)) { address pair = _voter.poolForGauge(gauge); uint256 amount = IERC20(pair).balanceOf(address(this)); if (amount > 0) { /// @dev if gauge is not alive, send to treasury if (!_voter.isAlive(gauge)) { IERC20(pair).safeTransfer(treasury, amount); continue; } address feeRecipient = feeRecipientFactory.feeRecipientForPair(pair); /// @dev send the pair fees to feeRecipient IERC20(pair).safeTransfer(feeRecipient, amount); /// @dev notify the fees to the FeeDistributor IFeeRecipient(feeRecipient).notifyFees(); } } } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/math/Math.sol) pragma solidity ^0.8.20; import {Panic} from "../Panic.sol"; import {SafeCast} from "./SafeCast.sol"; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { enum Rounding { Floor, // Toward negative infinity Ceil, // Toward positive infinity Trunc, // Toward zero Expand // Away from zero } /** * @dev Returns the addition of two unsigned integers, with an success flag (no overflow). */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an success flag (no overflow). */ function trySub(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an success flag (no overflow). */ function tryMul(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a success flag (no division by zero). */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a success flag (no division by zero). */ function tryMod(uint256 a, uint256 b) internal pure returns (bool success, uint256 result) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Branchless ternary evaluation for `a ? b : c`. Gas costs are constant. * * IMPORTANT: This function may reduce bytecode size and consume less gas when used standalone. * However, the compiler may optimize Solidity ternary operations (i.e. `a ? b : c`) to only compute * one branch when needed, making this function more expensive. */ function ternary(bool condition, uint256 a, uint256 b) internal pure returns (uint256) { unchecked { // branchless ternary works because: // b ^ (a ^ b) == a // b ^ 0 == b return b ^ ((a ^ b) * SafeCast.toUint(condition)); } } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return ternary(a > b, a, b); } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return ternary(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 towards infinity instead * of rounding towards zero. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { if (b == 0) { // Guarantee the same behavior as in a regular Solidity division. Panic.panic(Panic.DIVISION_BY_ZERO); } // The following calculation ensures accurate ceiling division without overflow. // Since a is non-zero, (a - 1) / b will not overflow. // The largest possible result occurs when (a - 1) / b is type(uint256).max, // but the largest value we can obtain is type(uint256).max - 1, which happens // when a = type(uint256).max and b = 1. unchecked { return SafeCast.toUint(a > 0) * ((a - 1) / b + 1); } } /** * @dev Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or * denominator == 0. * * 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²⁵⁶ and mod 2²⁵⁶ - 1, then use // the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2²⁵⁶ + prod0. uint256 prod0 = x * y; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2²⁵⁶. Also prevents denominator == 0. if (denominator <= prod1) { Panic.panic(ternary(denominator == 0, Panic.DIVISION_BY_ZERO, Panic.UNDER_OVERFLOW)); } /////////////////////////////////////////////// // 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. uint256 twos = denominator & (0 - denominator); 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²⁵⁶ / 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²⁵⁶. Now that denominator is an odd number, it has an inverse modulo 2²⁵⁶ such // that denominator * inv ≡ 1 mod 2²⁵⁶. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv ≡ 1 mod 2⁴. 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⁸ inverse *= 2 - denominator * inverse; // inverse mod 2¹⁶ inverse *= 2 - denominator * inverse; // inverse mod 2³² inverse *= 2 - denominator * inverse; // inverse mod 2⁶⁴ inverse *= 2 - denominator * inverse; // inverse mod 2¹²⁸ inverse *= 2 - denominator * inverse; // inverse mod 2²⁵⁶ // 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²⁵⁶. Since the preconditions guarantee that the outcome is // less than 2²⁵⁶, 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; } } /** * @dev 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) { return mulDiv(x, y, denominator) + SafeCast.toUint(unsignedRoundsUp(rounding) && mulmod(x, y, denominator) > 0); } /** * @dev Calculate the modular multiplicative inverse of a number in Z/nZ. * * If n is a prime, then Z/nZ is a field. In that case all elements are inversible, except 0. * If n is not a prime, then Z/nZ is not a field, and some elements might not be inversible. * * If the input value is not inversible, 0 is returned. * * NOTE: If you know for sure that n is (big) a prime, it may be cheaper to use Fermat's little theorem and get the * inverse using `Math.modExp(a, n - 2, n)`. See {invModPrime}. */ function invMod(uint256 a, uint256 n) internal pure returns (uint256) { unchecked { if (n == 0) return 0; // The inverse modulo is calculated using the Extended Euclidean Algorithm (iterative version) // Used to compute integers x and y such that: ax + ny = gcd(a, n). // When the gcd is 1, then the inverse of a modulo n exists and it's x. // ax + ny = 1 // ax = 1 + (-y)n // ax ≡ 1 (mod n) # x is the inverse of a modulo n // If the remainder is 0 the gcd is n right away. uint256 remainder = a % n; uint256 gcd = n; // Therefore the initial coefficients are: // ax + ny = gcd(a, n) = n // 0a + 1n = n int256 x = 0; int256 y = 1; while (remainder != 0) { uint256 quotient = gcd / remainder; (gcd, remainder) = ( // The old remainder is the next gcd to try. remainder, // Compute the next remainder. // Can't overflow given that (a % gcd) * (gcd // (a % gcd)) <= gcd // where gcd is at most n (capped to type(uint256).max) gcd - remainder * quotient ); (x, y) = ( // Increment the coefficient of a. y, // Decrement the coefficient of n. // Can overflow, but the result is casted to uint256 so that the // next value of y is "wrapped around" to a value between 0 and n - 1. x - y * int256(quotient) ); } if (gcd != 1) return 0; // No inverse exists. return ternary(x < 0, n - uint256(-x), uint256(x)); // Wrap the result if it's negative. } } /** * @dev Variant of {invMod}. More efficient, but only works if `p` is known to be a prime greater than `2`. * * From https://en.wikipedia.org/wiki/Fermat%27s_little_theorem[Fermat's little theorem], we know that if p is * prime, then `a**(p-1) ≡ 1 mod p`. As a consequence, we have `a * a**(p-2) ≡ 1 mod p`, which means that * `a**(p-2)` is the modular multiplicative inverse of a in Fp. * * NOTE: this function does NOT check that `p` is a prime greater than `2`. */ function invModPrime(uint256 a, uint256 p) internal view returns (uint256) { unchecked { return Math.modExp(a, p - 2, p); } } /** * @dev Returns the modular exponentiation of the specified base, exponent and modulus (b ** e % m) * * Requirements: * - modulus can't be zero * - underlying staticcall to precompile must succeed * * IMPORTANT: The result is only valid if the underlying call succeeds. When using this function, make * sure the chain you're using it on supports the precompiled contract for modular exponentiation * at address 0x05 as specified in https://eips.ethereum.org/EIPS/eip-198[EIP-198]. Otherwise, * the underlying function will succeed given the lack of a revert, but the result may be incorrectly * interpreted as 0. */ function modExp(uint256 b, uint256 e, uint256 m) internal view returns (uint256) { (bool success, uint256 result) = tryModExp(b, e, m); if (!success) { Panic.panic(Panic.DIVISION_BY_ZERO); } return result; } /** * @dev Returns the modular exponentiation of the specified base, exponent and modulus (b ** e % m). * It includes a success flag indicating if the operation succeeded. Operation will be marked as failed if trying * to operate modulo 0 or if the underlying precompile reverted. * * IMPORTANT: The result is only valid if the success flag is true. When using this function, make sure the chain * you're using it on supports the precompiled contract for modular exponentiation at address 0x05 as specified in * https://eips.ethereum.org/EIPS/eip-198[EIP-198]. Otherwise, the underlying function will succeed given the lack * of a revert, but the result may be incorrectly interpreted as 0. */ function tryModExp(uint256 b, uint256 e, uint256 m) internal view returns (bool success, uint256 result) { if (m == 0) return (false, 0); assembly ("memory-safe") { let ptr := mload(0x40) // | Offset | Content | Content (Hex) | // |-----------|------------|--------------------------------------------------------------------| // | 0x00:0x1f | size of b | 0x0000000000000000000000000000000000000000000000000000000000000020 | // | 0x20:0x3f | size of e | 0x0000000000000000000000000000000000000000000000000000000000000020 | // | 0x40:0x5f | size of m | 0x0000000000000000000000000000000000000000000000000000000000000020 | // | 0x60:0x7f | value of b | 0x<.............................................................b> | // | 0x80:0x9f | value of e | 0x<.............................................................e> | // | 0xa0:0xbf | value of m | 0x<.............................................................m> | mstore(ptr, 0x20) mstore(add(ptr, 0x20), 0x20) mstore(add(ptr, 0x40), 0x20) mstore(add(ptr, 0x60), b) mstore(add(ptr, 0x80), e) mstore(add(ptr, 0xa0), m) // Given the result < m, it's guaranteed to fit in 32 bytes, // so we can use the memory scratch space located at offset 0. success := staticcall(gas(), 0x05, ptr, 0xc0, 0x00, 0x20) result := mload(0x00) } } /** * @dev Variant of {modExp} that supports inputs of arbitrary length. */ function modExp(bytes memory b, bytes memory e, bytes memory m) internal view returns (bytes memory) { (bool success, bytes memory result) = tryModExp(b, e, m); if (!success) { Panic.panic(Panic.DIVISION_BY_ZERO); } return result; } /** * @dev Variant of {tryModExp} that supports inputs of arbitrary length. */ function tryModExp( bytes memory b, bytes memory e, bytes memory m ) internal view returns (bool success, bytes memory result) { if (_zeroBytes(m)) return (false, new bytes(0)); uint256 mLen = m.length; // Encode call args in result and move the free memory pointer result = abi.encodePacked(b.length, e.length, mLen, b, e, m); assembly ("memory-safe") { let dataPtr := add(result, 0x20) // Write result on top of args to avoid allocating extra memory. success := staticcall(gas(), 0x05, dataPtr, mload(result), dataPtr, mLen) // Overwrite the length. // result.length > returndatasize() is guaranteed because returndatasize() == m.length mstore(result, mLen) // Set the memory pointer after the returned data. mstore(0x40, add(dataPtr, mLen)) } } /** * @dev Returns whether the provided byte array is zero. */ function _zeroBytes(bytes memory byteArray) private pure returns (bool) { for (uint256 i = 0; i < byteArray.length; ++i) { if (byteArray[i] != 0) { return false; } } return true; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded * towards zero. * * This method is based on Newton's method for computing square roots; the algorithm is restricted to only * using integer operations. */ function sqrt(uint256 a) internal pure returns (uint256) { unchecked { // Take care of easy edge cases when a == 0 or a == 1 if (a <= 1) { return a; } // In this function, we use Newton's method to get a root of `f(x) := x² - a`. It involves building a // sequence x_n that converges toward sqrt(a). For each iteration x_n, we also define the error between // the current value as `ε_n = | x_n - sqrt(a) |`. // // For our first estimation, we consider `e` the smallest power of 2 which is bigger than the square root // of the target. (i.e. `2**(e-1) ≤ sqrt(a) < 2**e`). We know that `e ≤ 128` because `(2¹²⁸)² = 2²⁵⁶` is // bigger than any uint256. // // By noticing that // `2**(e-1) ≤ sqrt(a) < 2**e → (2**(e-1))² ≤ a < (2**e)² → 2**(2*e-2) ≤ a < 2**(2*e)` // we can deduce that `e - 1` is `log2(a) / 2`. We can thus compute `x_n = 2**(e-1)` using a method similar // to the msb function. uint256 aa = a; uint256 xn = 1; if (aa >= (1 << 128)) { aa >>= 128; xn <<= 64; } if (aa >= (1 << 64)) { aa >>= 64; xn <<= 32; } if (aa >= (1 << 32)) { aa >>= 32; xn <<= 16; } if (aa >= (1 << 16)) { aa >>= 16; xn <<= 8; } if (aa >= (1 << 8)) { aa >>= 8; xn <<= 4; } if (aa >= (1 << 4)) { aa >>= 4; xn <<= 2; } if (aa >= (1 << 2)) { xn <<= 1; } // We now have x_n such that `x_n = 2**(e-1) ≤ sqrt(a) < 2**e = 2 * x_n`. This implies ε_n ≤ 2**(e-1). // // We can refine our estimation by noticing that the middle of that interval minimizes the error. // If we move x_n to equal 2**(e-1) + 2**(e-2), then we reduce the error to ε_n ≤ 2**(e-2). // This is going to be our x_0 (and ε_0) xn = (3 * xn) >> 1; // ε_0 := | x_0 - sqrt(a) | ≤ 2**(e-2) // From here, Newton's method give us: // x_{n+1} = (x_n + a / x_n) / 2 // // One should note that: // x_{n+1}² - a = ((x_n + a / x_n) / 2)² - a // = ((x_n² + a) / (2 * x_n))² - a // = (x_n⁴ + 2 * a * x_n² + a²) / (4 * x_n²) - a // = (x_n⁴ + 2 * a * x_n² + a² - 4 * a * x_n²) / (4 * x_n²) // = (x_n⁴ - 2 * a * x_n² + a²) / (4 * x_n²) // = (x_n² - a)² / (2 * x_n)² // = ((x_n² - a) / (2 * x_n))² // ≥ 0 // Which proves that for all n ≥ 1, sqrt(a) ≤ x_n // // This gives us the proof of quadratic convergence of the sequence: // ε_{n+1} = | x_{n+1} - sqrt(a) | // = | (x_n + a / x_n) / 2 - sqrt(a) | // = | (x_n² + a - 2*x_n*sqrt(a)) / (2 * x_n) | // = | (x_n - sqrt(a))² / (2 * x_n) | // = | ε_n² / (2 * x_n) | // = ε_n² / | (2 * x_n) | // // For the first iteration, we have a special case where x_0 is known: // ε_1 = ε_0² / | (2 * x_0) | // ≤ (2**(e-2))² / (2 * (2**(e-1) + 2**(e-2))) // ≤ 2**(2*e-4) / (3 * 2**(e-1)) // ≤ 2**(e-3) / 3 // ≤ 2**(e-3-log2(3)) // ≤ 2**(e-4.5) // // For the following iterations, we use the fact that, 2**(e-1) ≤ sqrt(a) ≤ x_n: // ε_{n+1} = ε_n² / | (2 * x_n) | // ≤ (2**(e-k))² / (2 * 2**(e-1)) // ≤ 2**(2*e-2*k) / 2**e // ≤ 2**(e-2*k) xn = (xn + a / xn) >> 1; // ε_1 := | x_1 - sqrt(a) | ≤ 2**(e-4.5) -- special case, see above xn = (xn + a / xn) >> 1; // ε_2 := | x_2 - sqrt(a) | ≤ 2**(e-9) -- general case with k = 4.5 xn = (xn + a / xn) >> 1; // ε_3 := | x_3 - sqrt(a) | ≤ 2**(e-18) -- general case with k = 9 xn = (xn + a / xn) >> 1; // ε_4 := | x_4 - sqrt(a) | ≤ 2**(e-36) -- general case with k = 18 xn = (xn + a / xn) >> 1; // ε_5 := | x_5 - sqrt(a) | ≤ 2**(e-72) -- general case with k = 36 xn = (xn + a / xn) >> 1; // ε_6 := | x_6 - sqrt(a) | ≤ 2**(e-144) -- general case with k = 72 // Because e ≤ 128 (as discussed during the first estimation phase), we know have reached a precision // ε_6 ≤ 2**(e-144) < 1. Given we're operating on integers, then we can ensure that xn is now either // sqrt(a) or sqrt(a) + 1. return xn - SafeCast.toUint(xn > a / xn); } } /** * @dev 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 + SafeCast.toUint(unsignedRoundsUp(rounding) && result * result < a); } } /** * @dev Return the log in base 2 of a positive value rounded towards zero. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; uint256 exp; unchecked { exp = 128 * SafeCast.toUint(value > (1 << 128) - 1); value >>= exp; result += exp; exp = 64 * SafeCast.toUint(value > (1 << 64) - 1); value >>= exp; result += exp; exp = 32 * SafeCast.toUint(value > (1 << 32) - 1); value >>= exp; result += exp; exp = 16 * SafeCast.toUint(value > (1 << 16) - 1); value >>= exp; result += exp; exp = 8 * SafeCast.toUint(value > (1 << 8) - 1); value >>= exp; result += exp; exp = 4 * SafeCast.toUint(value > (1 << 4) - 1); value >>= exp; result += exp; exp = 2 * SafeCast.toUint(value > (1 << 2) - 1); value >>= exp; result += exp; result += SafeCast.toUint(value > 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 + SafeCast.toUint(unsignedRoundsUp(rounding) && 1 << result < value); } } /** * @dev Return the log in base 10 of a positive value rounded towards zero. * 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 + SafeCast.toUint(unsignedRoundsUp(rounding) && 10 ** result < value); } } /** * @dev Return the log in base 256 of a positive value rounded towards zero. * 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; uint256 isGt; unchecked { isGt = SafeCast.toUint(value > (1 << 128) - 1); value >>= isGt * 128; result += isGt * 16; isGt = SafeCast.toUint(value > (1 << 64) - 1); value >>= isGt * 64; result += isGt * 8; isGt = SafeCast.toUint(value > (1 << 32) - 1); value >>= isGt * 32; result += isGt * 4; isGt = SafeCast.toUint(value > (1 << 16) - 1); value >>= isGt * 16; result += isGt * 2; result += SafeCast.toUint(value > (1 << 8) - 1); } return result; } /** * @dev Return the log in base 256, 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 + SafeCast.toUint(unsignedRoundsUp(rounding) && 1 << (result << 3) < value); } } /** * @dev Returns whether a provided rounding mode is considered rounding up for unsigned integers. */ function unsignedRoundsUp(Rounding rounding) internal pure returns (bool) { return uint8(rounding) % 2 == 1; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (proxy/utils/Initializable.sol) pragma solidity ^0.8.20; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in * case an upgrade adds a module that needs to be initialized. * * For example: * * [.hljs-theme-light.nopadding] * ```solidity * contract MyToken is ERC20Upgradeable { * function initialize() initializer public { * __ERC20_init("MyToken", "MTK"); * } * } * * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable { * function initializeV2() reinitializer(2) public { * __ERC20Permit_init("MyToken"); * } * } * ``` * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. * * [CAUTION] * ==== * Avoid leaving a contract uninitialized. * * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed: * * [.hljs-theme-light.nopadding] * ``` * /// @custom:oz-upgrades-unsafe-allow constructor * constructor() { * _disableInitializers(); * } * ``` * ==== */ abstract contract Initializable { /** * @dev Storage of the initializable contract. * * It's implemented on a custom ERC-7201 namespace to reduce the risk of storage collisions * when using with upgradeable contracts. * * @custom:storage-location erc7201:openzeppelin.storage.Initializable */ struct InitializableStorage { /** * @dev Indicates that the contract has been initialized. */ uint64 _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool _initializing; } // keccak256(abi.encode(uint256(keccak256("openzeppelin.storage.Initializable")) - 1)) & ~bytes32(uint256(0xff)) bytes32 private constant INITIALIZABLE_STORAGE = 0xf0c57e16840df040f15088dc2f81fe391c3923bec73e23a9662efc9c229c6a00; /** * @dev The contract is already initialized. */ error InvalidInitialization(); /** * @dev The contract is not initializing. */ error NotInitializing(); /** * @dev Triggered when the contract has been initialized or reinitialized. */ event Initialized(uint64 version); /** * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope, * `onlyInitializing` functions can be used to initialize parent contracts. * * Similar to `reinitializer(1)`, except that in the context of a constructor an `initializer` may be invoked any * number of times. This behavior in the constructor can be useful during testing and is not expected to be used in * production. * * Emits an {Initialized} event. */ modifier initializer() { // solhint-disable-next-line var-name-mixedcase InitializableStorage storage $ = _getInitializableStorage(); // Cache values to avoid duplicated sloads bool isTopLevelCall = !$._initializing; uint64 initialized = $._initialized; // Allowed calls: // - initialSetup: the contract is not in the initializing state and no previous version was // initialized // - construction: the contract is initialized at version 1 (no reininitialization) and the // current contract is just being deployed bool initialSetup = initialized == 0 && isTopLevelCall; bool construction = initialized == 1 && address(this).code.length == 0; if (!initialSetup && !construction) { revert InvalidInitialization(); } $._initialized = 1; if (isTopLevelCall) { $._initializing = true; } _; if (isTopLevelCall) { $._initializing = false; emit Initialized(1); } } /** * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be * used to initialize parent contracts. * * A reinitializer may be used after the original initialization step. This is essential to configure modules that * are added through upgrades and that require initialization. * * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer` * cannot be nested. If one is invoked in the context of another, execution will revert. * * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in * a contract, executing them in the right order is up to the developer or operator. * * WARNING: Setting the version to 2**64 - 1 will prevent any future reinitialization. * * Emits an {Initialized} event. */ modifier reinitializer(uint64 version) { // solhint-disable-next-line var-name-mixedcase InitializableStorage storage $ = _getInitializableStorage(); if ($._initializing || $._initialized >= version) { revert InvalidInitialization(); } $._initialized = version; $._initializing = true; _; $._initializing = false; emit Initialized(version); } /** * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the * {initializer} and {reinitializer} modifiers, directly or indirectly. */ modifier onlyInitializing() { _checkInitializing(); _; } /** * @dev Reverts if the contract is not in an initializing state. See {onlyInitializing}. */ function _checkInitializing() internal view virtual { if (!_isInitializing()) { revert NotInitializing(); } } /** * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call. * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized * to any version. It is recommended to use this to lock implementation contracts that are designed to be called * through proxies. * * Emits an {Initialized} event the first time it is successfully executed. */ function _disableInitializers() internal virtual { // solhint-disable-next-line var-name-mixedcase InitializableStorage storage $ = _getInitializableStorage(); if ($._initializing) { revert InvalidInitialization(); } if ($._initialized != type(uint64).max) { $._initialized = type(uint64).max; emit Initialized(type(uint64).max); } } /** * @dev Returns the highest version that has been initialized. See {reinitializer}. */ function _getInitializedVersion() internal view returns (uint64) { return _getInitializableStorage()._initialized; } /** * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}. */ function _isInitializing() internal view returns (bool) { return _getInitializableStorage()._initializing; } /** * @dev Returns a pointer to the storage namespace. */ // solhint-disable-next-line var-name-mixedcase function _getInitializableStorage() private pure returns (InitializableStorage storage $) { assembly { $.slot := INITIALIZABLE_STORAGE } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.20; import {IERC20} from "../IERC20.sol"; import {IERC1363} from "../../../interfaces/IERC1363.sol"; import {Address} from "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC-20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { /** * @dev An operation with an ERC-20 token failed. */ error SafeERC20FailedOperation(address token); /** * @dev Indicates a failed `decreaseAllowance` request. */ error SafeERC20FailedDecreaseAllowance(address spender, uint256 currentAllowance, uint256 requestedDecrease); /** * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeCall(token.transfer, (to, value))); } /** * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful. */ function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeCall(token.transferFrom, (from, to, value))); } /** * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. * * IMPORTANT: If the token implements ERC-7674 (ERC-20 with temporary allowance), and if the "client" * smart contract uses ERC-7674 to set temporary allowances, then the "client" smart contract should avoid using * this function. Performing a {safeIncreaseAllowance} or {safeDecreaseAllowance} operation on a token contract * that has a non-zero temporary allowance (for that particular owner-spender) will result in unexpected behavior. */ function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 oldAllowance = token.allowance(address(this), spender); forceApprove(token, spender, oldAllowance + value); } /** * @dev Decrease the calling contract's allowance toward `spender` by `requestedDecrease`. If `token` returns no * value, non-reverting calls are assumed to be successful. * * IMPORTANT: If the token implements ERC-7674 (ERC-20 with temporary allowance), and if the "client" * smart contract uses ERC-7674 to set temporary allowances, then the "client" smart contract should avoid using * this function. Performing a {safeIncreaseAllowance} or {safeDecreaseAllowance} operation on a token contract * that has a non-zero temporary allowance (for that particular owner-spender) will result in unexpected behavior. */ function safeDecreaseAllowance(IERC20 token, address spender, uint256 requestedDecrease) internal { unchecked { uint256 currentAllowance = token.allowance(address(this), spender); if (currentAllowance < requestedDecrease) { revert SafeERC20FailedDecreaseAllowance(spender, currentAllowance, requestedDecrease); } forceApprove(token, spender, currentAllowance - requestedDecrease); } } /** * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval * to be set to zero before setting it to a non-zero value, such as USDT. * * NOTE: If the token implements ERC-7674, this function will not modify any temporary allowance. This function * only sets the "standard" allowance. Any temporary allowance will remain active, in addition to the value being * set here. */ function forceApprove(IERC20 token, address spender, uint256 value) internal { bytes memory approvalCall = abi.encodeCall(token.approve, (spender, value)); if (!_callOptionalReturnBool(token, approvalCall)) { _callOptionalReturn(token, abi.encodeCall(token.approve, (spender, 0))); _callOptionalReturn(token, approvalCall); } } /** * @dev Performs an {ERC1363} transferAndCall, with a fallback to the simple {ERC20} transfer if the target has no * code. This can be used to implement an {ERC721}-like safe transfer that rely on {ERC1363} checks when * targeting contracts. * * Reverts if the returned value is other than `true`. */ function transferAndCallRelaxed(IERC1363 token, address to, uint256 value, bytes memory data) internal { if (to.code.length == 0) { safeTransfer(token, to, value); } else if (!token.transferAndCall(to, value, data)) { revert SafeERC20FailedOperation(address(token)); } } /** * @dev Performs an {ERC1363} transferFromAndCall, with a fallback to the simple {ERC20} transferFrom if the target * has no code. This can be used to implement an {ERC721}-like safe transfer that rely on {ERC1363} checks when * targeting contracts. * * Reverts if the returned value is other than `true`. */ function transferFromAndCallRelaxed( IERC1363 token, address from, address to, uint256 value, bytes memory data ) internal { if (to.code.length == 0) { safeTransferFrom(token, from, to, value); } else if (!token.transferFromAndCall(from, to, value, data)) { revert SafeERC20FailedOperation(address(token)); } } /** * @dev Performs an {ERC1363} approveAndCall, with a fallback to the simple {ERC20} approve if the target has no * code. This can be used to implement an {ERC721}-like safe transfer that rely on {ERC1363} checks when * targeting contracts. * * NOTE: When the recipient address (`to`) has no code (i.e. is an EOA), this function behaves as {forceApprove}. * Opposedly, when the recipient address (`to`) has code, this function only attempts to call {ERC1363-approveAndCall} * once without retrying, and relies on the returned value to be true. * * Reverts if the returned value is other than `true`. */ function approveAndCallRelaxed(IERC1363 token, address to, uint256 value, bytes memory data) internal { if (to.code.length == 0) { forceApprove(token, to, value); } else if (!token.approveAndCall(to, value, data)) { revert SafeERC20FailedOperation(address(token)); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturnBool} that reverts if call fails to meet the requirements. */ function _callOptionalReturn(IERC20 token, bytes memory data) private { uint256 returnSize; uint256 returnValue; assembly ("memory-safe") { let success := call(gas(), token, 0, add(data, 0x20), mload(data), 0, 0x20) // bubble errors if iszero(success) { let ptr := mload(0x40) returndatacopy(ptr, 0, returndatasize()) revert(ptr, returndatasize()) } returnSize := returndatasize() returnValue := mload(0) } if (returnSize == 0 ? address(token).code.length == 0 : returnValue != 1) { revert SafeERC20FailedOperation(address(token)); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturn} that silently catches all reverts and returns a bool instead. */ function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) { bool success; uint256 returnSize; uint256 returnValue; assembly ("memory-safe") { success := call(gas(), token, 0, add(data, 0x20), mload(data), 0, 0x20) returnSize := returndatasize() returnValue := mload(0) } return success && (returnSize == 0 ? address(token).code.length > 0 : returnValue == 1); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /// @title Central Errors Library /// @notice Contains all custom errors used across the protocol /// @dev Centralized error definitions to prevent redundancy library Errors { /*////////////////////////////////////////////////////////////// VOTER ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when attempting to interact with an already active gauge /// @param gauge The address of the gauge error ACTIVE_GAUGE(address gauge); /// @notice Thrown when attempting to interact with an inactive gauge /// @param gauge The address of the gauge error GAUGE_INACTIVE(address gauge); /// @notice Thrown when attempting to whitelist an already whitelisted token /// @param token The address of the token error ALREADY_WHITELISTED(address token); /// @notice Thrown when caller is not authorized to perform an action /// @param caller The address of the unauthorized caller error NOT_AUTHORIZED(address caller); /// @notice Thrown when token is not whitelisted /// @param token The address of the non-whitelisted token error NOT_WHITELISTED(address token); /// @notice Thrown when both tokens in a pair are not whitelisted error BOTH_NOT_WHITELISTED(); /// @notice Thrown when address is not a valid pool /// @param pool The invalid pool address error NOT_POOL(address pool); /// @notice Thrown when pool is not seeded in PoolUpdater /// @param pool The invalid pool address error NOT_SEEDED(address pool); /// @notice Thrown when contract is not initialized error NOT_INIT(); /// @notice Thrown when array lengths don't match error LENGTH_MISMATCH(); /// @notice Thrown when pool doesn't have an associated gauge /// @param pool The address of the pool error NO_GAUGE(address pool); /// @notice Thrown when rewards are already distributed for a period /// @param gauge The gauge address /// @param period The distribution period error ALREADY_DISTRIBUTED(address gauge, uint256 period); /// @notice Thrown when attempting to vote with zero amount /// @param pool The pool address error ZERO_VOTE(address pool); /// @notice Thrown when ratio exceeds maximum allowed /// @param _xRatio The excessive ratio value error RATIO_TOO_HIGH(uint256 _xRatio); /// @notice Thrown when vote operation fails error VOTE_UNSUCCESSFUL(); /*////////////////////////////////////////////////////////////// GAUGE V3 ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when the pool already has a gauge /// @param pool The address of the pool error GAUGE_EXISTS(address pool); /// @notice Thrown when caller is not the voter /// @param caller The address of the invalid caller error NOT_VOTER(address caller); /// @notice Thrown when amount is not greater than zero /// @param amt The invalid amount error NOT_GT_ZERO(uint256 amt); /// @notice Thrown when attempting to claim future rewards error CANT_CLAIM_FUTURE(); /// @notice Throw when gauge can't determine if using secondsInRange from the pool is safe error NEED_TEAM_TO_UPDATE(); /*////////////////////////////////////////////////////////////// GAUGE ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when amount is zero error ZERO_AMOUNT(); /// @notice Thrown when stake notification fails error CANT_NOTIFY_STAKE(); /// @notice Thrown when reward amount is too high error REWARD_TOO_HIGH(); /// @notice Thrown when amount exceeds remaining balance /// @param amount The requested amount /// @param remaining The remaining balance error NOT_GREATER_THAN_REMAINING(uint256 amount, uint256 remaining); /// @notice Thrown when token operation fails /// @param token The address of the problematic token error TOKEN_ERROR(address token); /// @notice Thrown when an address is not an NfpManager error NOT_NFP_MANAGER(address nfpManager); /*////////////////////////////////////////////////////////////// FEE DISTRIBUTOR ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when period is not finalized /// @param period The unfinalized period error NOT_FINALIZED(uint256 period); /// @notice Thrown when the destination of a redirect is not a feeDistributor /// @param destination Destination of the redirect error NOT_FEE_DISTRIBUTOR(address destination); /// @notice Thrown when the destination of a redirect's pool/pair has completely different tokens error DIFFERENT_DESTINATION_TOKENS(); /*////////////////////////////////////////////////////////////// PAIR ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when ratio is unstable error UNSTABLE_RATIO(); /// @notice Thrown when safe transfer fails error SAFE_TRANSFER_FAILED(); /// @notice Thrown on arithmetic overflow error OVERFLOW(); /// @notice Thrown when skim operation is disabled error SKIM_DISABLED(); /// @notice Thrown when insufficient liquidity is minted error INSUFFICIENT_LIQUIDITY_MINTED(); /// @notice Thrown when insufficient liquidity is burned error INSUFFICIENT_LIQUIDITY_BURNED(); /// @notice Thrown when output amount is insufficient error INSUFFICIENT_OUTPUT_AMOUNT(); /// @notice Thrown when input amount is insufficient error INSUFFICIENT_INPUT_AMOUNT(); /// @notice Generic insufficient liquidity error error INSUFFICIENT_LIQUIDITY(); /// @notice Invalid transfer error error INVALID_TRANSFER(); /// @notice K value error in AMM error K(); /*////////////////////////////////////////////////////////////// PAIR FACTORY ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when fee is too high error FEE_TOO_HIGH(); /// @notice Thrown when fee is zero error ZERO_FEE(); /// @notice Thrown when token assortment is invalid error INVALID_ASSORTMENT(); /// @notice Thrown when address is zero error ZERO_ADDRESS(); /// @notice Thrown when pair already exists error PAIR_EXISTS(); /// @notice Thrown when fee split is invalid error INVALID_FEE_SPLIT(); /*////////////////////////////////////////////////////////////// FEE RECIPIENT FACTORY ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when treasury fee is invalid error INVALID_TREASURY_FEE(); /*////////////////////////////////////////////////////////////// ROUTER ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when deadline has expired error EXPIRED(); /// @notice Thrown when tokens are identical error IDENTICAL(); /// @notice Thrown when amount is insufficient error INSUFFICIENT_AMOUNT(); /// @notice Thrown when path is invalid error INVALID_PATH(); /// @notice Thrown when token B amount is insufficient error INSUFFICIENT_B_AMOUNT(); /// @notice Thrown when token A amount is insufficient error INSUFFICIENT_A_AMOUNT(); /// @notice Thrown when input amount is excessive error EXCESSIVE_INPUT_AMOUNT(); /// @notice Thrown when ETH transfer fails error ETH_TRANSFER_FAILED(); /// @notice Thrown when reserves are invalid error INVALID_RESERVES(); /*////////////////////////////////////////////////////////////// MINTER ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when epoch 0 has already started error STARTED(); /// @notice Thrown when emissions haven't started error EMISSIONS_NOT_STARTED(); /// @notice Thrown when deviation is too high error TOO_HIGH(); /// @notice Thrown when no value change detected error NO_CHANGE(); /// @notice Thrown when updating emissions in same period error SAME_PERIOD(); /// @notice Thrown when contract setup is invalid error INVALID_CONTRACT(); /// @notice Thrown when legacy factory doesn't have feeSplitWhenNoGauge on error FEE_SPLIT_WHEN_NO_GAUGE_IS_OFF(); /*////////////////////////////////////////////////////////////// ACCESS HUB ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when addresses are identical error SAME_ADDRESS(); /// @notice Thrown when caller is not timelock /// @param caller The invalid caller address error NOT_TIMELOCK(address caller); /// @notice Thrown when manual execution fails /// @param reason The failure reason error MANUAL_EXECUTION_FAILURE(bytes reason); /// @notice Thrown when kick operation is forbidden /// @param target The target address error KICK_FORBIDDEN(address target); /*////////////////////////////////////////////////////////////// VOTE MODULE ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when caller is not xShadow error NOT_XSHADOW(); /// @notice Thrown when cooldown period is still active error COOLDOWN_ACTIVE(); /// @notice Thrown when caller is not vote module error NOT_VOTEMODULE(); /// @notice Thrown when caller is unauthorized error UNAUTHORIZED(); /// @notice Thrown when caller is not access hub error NOT_ACCESSHUB(); /// @notice Thrown when address is invalid error INVALID_ADDRESS(); /*////////////////////////////////////////////////////////////// LAUNCHER PLUGIN ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when caller is not authority error NOT_AUTHORITY(); /// @notice Thrown when already an authority error ALREADY_AUTHORITY(); /// @notice Thrown when caller is not operator error NOT_OPERATOR(); /// @notice Thrown when already an operator error ALREADY_OPERATOR(); /// @notice Thrown when pool is not enabled /// @param pool The disabled pool address error NOT_ENABLED(address pool); /// @notice Thrown when fee distributor is missing error NO_FEEDIST(); /// @notice Thrown when already enabled error ENABLED(); /// @notice Thrown when take value is invalid error INVALID_TAKE(); /*////////////////////////////////////////////////////////////// X33 ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when value is zero error ZERO(); /// @notice Thrown when amount is insufficient error NOT_ENOUGH(); /// @notice Thrown when value doesn't conform to scale /// @param value The non-conforming value error NOT_CONFORMED_TO_SCALE(uint256 value); /// @notice Thrown when contract is locked error LOCKED(); /// @notice Thrown when rebase is in progress error REBASE_IN_PROGRESS(); /// @notice Thrown when aggregator reverts /// @param reason The revert reason error AGGREGATOR_REVERTED(bytes reason); /// @notice Thrown when output amount is too low /// @param amount The insufficient amount error AMOUNT_OUT_TOO_LOW(uint256 amount); /// @notice Thrown when aggregator is not whitelisted /// @param aggregator The non-whitelisted aggregator address error AGGREGATOR_NOT_WHITELISTED(address aggregator); /// @notice Thrown when token is forbidden /// @param token The forbidden token address error FORBIDDEN_TOKEN(address token); /*////////////////////////////////////////////////////////////// XSHADOW ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when caller is not minter error NOT_MINTER(); /// @notice Thrown when no vest exists error NO_VEST(); /// @notice Thrown when already exempt error ALREADY_EXEMPT(); /// @notice Thrown when not exempt error NOT_EXEMPT(); /// @notice Thrown when rescue operation is not allowed error CANT_RESCUE(); /// @notice Thrown when array lengths mismatch error ARRAY_LENGTHS(); /// @notice Thrown when vesting periods overlap error VEST_OVERLAP(); /*////////////////////////////////////////////////////////////// V3 FACTORY ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when tokens are identical error IDENTICAL_TOKENS(); /// @notice Thrown when fee is too large error FEE_TOO_LARGE(); /// @notice Address zero error error ADDRESS_ZERO(); /// @notice Fee zero error error F0(); /// @notice Thrown when value is out of bounds /// @param value The out of bounds value error OOB(uint8 value); /*////////////////////////////////////////////////////////////// POOL UPDATER ERRORS //////////////////////////////////////////////////////////////*/ /// @notice Thrown when seeding for a pool fails error TRANSFER_FROM_FOR_SEEDING_FAILED(address token, uint256 amount); /// @notice Thrown when seeding for a pool fails error SEEDING_FAILED(); /// @notice Thrown when updatePools is called too early error TOO_EARLY(); /// @notice Thrown when a callback is called when an update isn't running error NOT_RUNNING(); /// @notice Thrown when updatePools didn't perform any updates error NO_UPDATES(); }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; import {EnumerableSet} from "@openzeppelin/contracts/utils/structs/EnumerableSet.sol"; import {IERC20Extended} from "../interfaces/IERC20Extended.sol"; library MinterStorage { using EnumerableSet for EnumerableSet.AddressSet; /// @dev keccak256(abi.encode(uint256(keccak256("minter.storage")) - 1)) & ~bytes32(uint256(0xff)); bytes32 public constant MINTER_STORAGE_LOCATION = 0xe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836000; /// @custom꞉storage‑location erc7201꞉voter.storage struct MinterState { uint256 weeklyEmissions; /// @notice controls emissions growth or decay uint256 emissionsMultiplier; /// @notice unix timestamp of the first period uint256 firstPeriod; /// @notice currently active unix timestamp of epoch start uint256 activePeriod; /// @notice the last period the emissions multiplier was updated uint256 lastMultiplierUpdate; /// @notice current operator address operator; /// @notice the access control center address accessHub; /// @notice xShadow contract address address xShadow; /// @notice central voter contract address voter; /// @notice the IERC20 version of shadow IERC20Extended shadow; } /// @dev Return state storage struct for reading and writing function getStorage() internal pure returns (MinterState storage $) { assembly { $.slot := MINTER_STORAGE_LOCATION } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import {IERC20Metadata} from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; import {IERC20Permit} from "@openzeppelin/contracts/token/ERC20/extensions/IERC20Permit.sol"; interface IERC20Extended is IERC20, IERC20Metadata, IERC20Permit { function mint(address account, uint256 amount) external; function burn(uint256 amount) external; function transfer(address to, uint256 value) external returns (bool); function transferFrom( address from, address to, uint256 value ) external returns (bool); function burnFrom(address account, uint256 value) external; }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; interface IMinter { event SetVeDist(address _value); event SetVoter(address _value); event Mint(address indexed sender, uint256 weekly); event RebaseUnsuccessful(uint256 _current, uint256 _currentPeriod); event EmissionsMultiplierUpdated(uint256 _emissionsMultiplier); /// @notice decay or inflation scaled to 1_000_000 = 100% /// @return _multiplier the emissions multiplier function emissionsMultiplier() external view returns (uint256 _multiplier); /// @notice unix timestamp of current epoch's start /// @return _activePeriod the active period function activePeriod() external view returns (uint256 _activePeriod); /// @notice update the epoch (period) -- callable once a week at >= Thursday 0 UTC /// @return period the new period function updatePeriod() external returns (uint256 period); /// @notice start emissions for epoch 0 function startEmissions() external; /// @notice updates the decay or inflation scaled to 1_000_000 = 100% /// @param _emissionsMultiplier multiplier for emissions each week function updateEmissionsMultiplier(uint256 _emissionsMultiplier) external; /// @notice calculates the emissions to be sent to the voter /// @return _weeklyEmissions the amount of emissions for the week function calculateWeeklyEmissions() external view returns (uint256 _weeklyEmissions); /// @notice kicks off the initial minting and variable declarations function kickoff( address _shadow, address _voter, uint256 _initialWeeklyEmissions, uint256 _initialMultiplier, address _xShadow ) external; /// @notice returns (block.timestamp / 1 week) for gauge use /// @return period period number function getPeriod() external view returns (uint256 period); /// @notice returns the numerical value of the current epoch /// @return _epoch epoch number function getEpoch() external view returns (uint256 _epoch); /// @notice emissions value function weeklyEmissions() external view returns (uint256); /// @notice unix timestamp of the first period function firstPeriod() external view returns (uint256); /// @notice the last period the emissions multiplier was updated function lastMultiplierUpdate() external view returns (uint256); /// @notice current operator function operator() external view returns (address); /// @notice the access control center function accessHub() external view returns (address); /// @notice xShadow contract address function xShadow() external view returns (address); /// @notice central voter contract function voter() external view returns (address); /// @notice the IERC20 version of shadow function shadow() external view returns (address); }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; import {IMinter} from "./IMinter.sol"; interface IMarbleMinter is IMinter { /// @notice migrate gauges for Marble Zone Migration function migration(address[] calldata gauges) external; /// @notice sets up fee redirection for new legacy gauges function postCreateLegacyGaugeHook(address pool) external; /// @notice redirects legacy pair fees to new feeRecipients function redirectFees(uint256 start, uint256 batchSize) external; /// @notice allows governance to rescue tokens (pair fees without gauges) function rescueTokens(address token, uint256 amount) external; /// @notice allows governance to change Access Hub function setAccessHub(address _accessHub) external; /// @notice allows governance to change Operator function setOperator(address _operator) external; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import {IVoteModule} from "contracts/interfaces/IVoteModule.sol"; import {IVoter} from "contracts/interfaces/IVoter.sol"; interface IAccessHub { /// @dev Struct to hold initialization parameters struct InitParams { address timelock; address treasury; address voter; address minter; address launcherPlugin; address xShadow; address x33; address shadowV3PoolFactory; address poolFactory; address clGaugeFactory; address gaugeFactory; address feeRecipientFactory; address feeDistributorFactory; address feeCollector; address voteModule; } /// @notice protocol timelock address function timelock() external view returns (address timelock); /// @notice protocol treasury address function treasury() external view returns (address treasury); /// @notice vote module function voteModule() external view returns (IVoteModule voteModule); /// @notice voter function voter() external view returns (IVoter voter); /// @notice concentrated (v3) gauge factory function clGaugeFactory() external view returns (address _clGaugeFactory); /// @notice legacy gauge factory address function gaugeFactory() external view returns (address _gaugeFactory); /// @notice the feeDistributor factory address function feeDistributorFactory() external view returns (address _feeDistributorFactory); /// @notice initializing function for setting values in the AccessHub function initialize(InitParams calldata params) external; /// @notice function for reinitializing values in the AccessHub function reinit(InitParams calldata params) external; /// @notice sets the swap fees for multiple pairs function setSwapFees(address[] calldata _pools, uint24[] calldata _swapFees, bool[] calldata _concentrated) external; /// @notice sets the split of fees between LPs and voters function setFeeSplitCL(address[] calldata _pools, uint8[] calldata _feeProtocol) external; /// @notice sets the split of fees between LPs and voters for legacy pools function setFeeSplitLegacy(address[] calldata _pools, uint256[] calldata _feeSplits) external; /** * Voter governance */ /// @notice sets a new governor address in the voter.sol contract function setNewGovernorInVoter(address _newGovernor) external; /// @notice whitelists a token for governance, or removes if boolean is set to false function governanceWhitelist(address[] calldata _token, bool[] calldata _whitelisted) external; /// @notice kills active gauges, removing them from earning further emissions, and claims their fees prior function killGauge(address[] calldata _pairs) external; /// @notice revives inactive/killed gauges function reviveGauge(address[] calldata _pairs) external; /// @notice sets a new NFP Manager function setNfpManager(address _nfpManager) external; /// @notice syncs NFP Managers for gauges function syncNfpManager(address[] calldata gauges) external; /// @notice sets the ratio of xShadow/Shadow awarded globally to LPs function setEmissionsRatioInVoter(uint256 _pct) external; /// @notice allows governance to retrieve emissions in the voter contract that will not be distributed due to the gauge being inactive /// @dev allows per-period retrieval for granularity function retrieveStuckEmissionsToGovernance(address _gauge, uint256 _period) external; /// @notice allows governance to designate a gauge as the "main" one, to prevent governance spam and confusion function setMainGaugeForClPair(address tokenA, address tokenB, address gauge) external; function createGaugeForPool(address _pool) external; function resetVotesOnBehalfOf(address _user) external; /** * xShadow Functions */ /// @notice enables or disables the transfer whitelist in xShadow function transferWhitelistInXShadow(address[] calldata _who, bool[] calldata _whitelisted) external; /// @notice enables or disables the transfer TO whitelists in xShadow function transferToWhitelistInXShadow(address[] calldata _who, bool[] calldata _whitelisted) external; /// @notice enables or disables the governance in xShadow function toggleXShadowGovernance(bool enable) external; /// @notice allows redemption from the operator function operatorRedeemXShadow(uint256 _amount) external; /// @notice migrates the xShadow operator function migrateOperator(address _operator) external; /// @notice rescues any trapped tokens in xShadow function rescueTrappedTokens(address[] calldata _tokens, uint256[] calldata _amounts) external; /// @notice set dust threshold before a rebase can happen function setRebaseThreshold(uint256 _newThreshold) external; /** * X33 Functions */ /// @notice transfers the x33 operator address function transferOperatorInX33(address _newOperator) external; /** * Minter Functions */ /// @notice sets the inflation multiplier /// @param _multiplier the multiplier function setEmissionsMultiplierInMinter(uint256 _multiplier) external; /** * Reward List Functions */ /// @notice function for removing rewards for feeDistributors function removeFeeDistributorRewards(address[] calldata _pools, address[] calldata _rewards) external; /** * LauncherPlugin specific functions */ /// @notice allows migrating the parameters from one pool to the other /// @param _oldPool the current address of the pair /// @param _newPool the new pool's address function migratePoolInLauncherPlugin(address _oldPool, address _newPool) external; /// @notice set launcher configurations for a pool /// @param _pool address of the pool /// @param _take the fee that goes to the designated recipient /// @param _recipient the address that receives the fees function setConfigsInLauncherPlugin(address _pool, uint256 _take, address _recipient) external; /// @notice enables the pool for LauncherConfigs /// @param _pool address of the pool function enablePoolInLauncherPlugin(address _pool) external; /// @notice disables the pool for LauncherConfigs /// @dev clears mappings /// @param _pool address of the pool function disablePoolInLauncherPlugin(address _pool) external; /// @notice sets a new operator address /// @param _newOperator new operator address function setOperatorInLauncherPlugin(address _newOperator) external; /// @notice gives authority to a new contract/address /// @param _newAuthority the suggested new authority function grantAuthorityInLauncherPlugin(address _newAuthority, string calldata _label) external; /// @notice governance ability to label each authority in the system with an arbitrary string function labelAuthorityInLauncherPlugin(address _authority, string calldata _label) external; /// @notice removes authority from a contract/address /// @param _oldAuthority the to-be-removed authority function revokeAuthorityInLauncherPlugin(address _oldAuthority) external; /** * FeeCollector functions */ /// @notice Sets the treasury address to a new value. /// @param newTreasury The new address to set as the treasury. function setTreasuryInFeeCollector(address newTreasury) external; /// @notice Sets the value of treasury fees to a new amount. /// @param _treasuryFees The new amount of treasury fees to be set. function setTreasuryFeesInFeeCollector(uint256 _treasuryFees) external; /** * FeeRecipientFactory functions */ /// @notice set the fee % to be sent to the treasury /// @param _feeToTreasury the fee % to be sent to the treasury function setFeeToTreasuryInFeeRecipientFactory(uint256 _feeToTreasury) external; /// @notice set a new treasury address /// @param _treasury the new address function setTreasuryInFeeRecipientFactory(address _treasury) external; /** * CL Pool Factory functions */ /// @notice enables a tickSpacing with the given initialFee amount /// @dev unlike UniswapV3, we map via the tickSpacing rather than the fee tier /// @dev tickSpacings may never be removed once enabled /// @param tickSpacing The spacing between ticks to be enforced for all pools created /// @param initialFee The initial fee amount, denominated in hundredths of a bip (i.e. 1e-6) function enableTickSpacing(int24 tickSpacing, uint24 initialFee) external; /// @notice sets the feeProtocol (feeSplit) for new CL pools and stored in the factory function setGlobalClFeeProtocol(uint8 _feeProtocolGlobal) external; /// @notice sets the address of the voter in the v3 factory for gauge fee setting function setVoterAddressInFactoryV3(address _voter) external; /// @notice sets the address of the feeCollector in the v3 factory for fee routing function setFeeCollectorInFactoryV3(address _newFeeCollector) external; /** * Legacy Pool Factory functions */ /// @notice sets the treasury address in the legacy factory function setTreasuryInLegacyFactory(address _treasury) external; /// @notice enables or disables if there is a feeSplit when no gauge for legacy pairs function setFeeSplitWhenNoGauge(bool status) external; /// @notice set the default feeSplit in the legacy factory function setLegacyFeeSplitGlobal(uint256 _feeSplit) external; /// @notice set the fee recipient for legacy pairs function setLegacyFeeRecipient(address _pair, address _feeRecipient) external; /// @notice set the default swap fee for legacy pools function setLegacyFeeGlobal(uint256 _fee) external; /// @notice sets whether a pair can have skim() called or not for rebasing purposes function setSkimEnabledLegacy(address _pair, bool _status) external; /** * VoteModule Functions */ /// @notice sets addresses as exempt or removes their exemption function setCooldownExemption(address[] calldata _candidates, bool[] calldata _exempt) external; /// @notice function to change the cooldown in the voteModule function setNewVoteModuleCooldown(uint256 _newCooldown) external; /// @notice allows resetting of inactive votes to prevent dead votes function kickInactive(address[] calldata _nonparticipants) external; /** * Timelock gated functions */ /// @notice timelock gated payload execution in case tokens get stuck or other unexpected behaviors function execute(address _target, bytes calldata _payload) external; /// @notice timelock gated function to change the timelock function setNewTimelock(address _timelock) external; /// @notice function for initializing the voter contract with its dependencies function initializeVoter(IVoter.InitializationParams memory inputs) external; function backupDistribute() external; function backupDistributeBatch(uint256 startIndex, uint256 batchSize) external; }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; interface IPairFactory { event PairCreated(address indexed token0, address indexed token1, address pair, uint256); event SetFee(uint256 indexed fee); event SetPairFee(address indexed pair, uint256 indexed fee); event SetFeeSplit(uint256 indexed _feeSplit); event SetPairFeeSplit(address indexed pair, uint256 indexed _feeSplit); event SkimStatus(address indexed _pair, bool indexed _status); event NewTreasury(address indexed _caller, address indexed _newTreasury); event FeeSplitWhenNoGauge(address indexed _caller, bool indexed _status); event SetFeeRecipient(address indexed pair, address indexed feeRecipient); /// @notice returns the total length of legacy pairs /// @return _length the length function allPairsLength() external view returns (uint256 _length); /// @notice calculates if the address is a legacy pair /// @param pair the address to check /// @return _boolean the bool return function isPair(address pair) external view returns (bool _boolean); /// @notice calculates the pairCodeHash /// @return _hash the pair code hash function pairCodeHash() external view returns (bytes32 _hash); /// @param tokenA address of tokenA /// @param tokenB address of tokenB /// @param stable whether it uses the stable curve /// @return _pair the address of the pair function getPair(address tokenA, address tokenB, bool stable) external view returns (address _pair); /// @notice creates a new legacy pair /// @param tokenA address of tokenA /// @param tokenB address of tokenB /// @param stable whether it uses the stable curve /// @return pair the address of the created pair function createPair(address tokenA, address tokenB, bool stable) external returns (address pair); /// @notice the address of the voter /// @return _voter the address of the voter function voter() external view returns (address _voter); /// @notice returns the address of a pair based on the index /// @param _index the index to check for a pair /// @return _pair the address of the pair at the index function allPairs(uint256 _index) external view returns (address _pair); /// @notice the swap fee of a pair /// @param _pair the address of the pair /// @return _fee the fee function pairFee(address _pair) external view returns (uint256 _fee); /// @notice the split of fees /// @return _split the feeSplit function feeSplit() external view returns (uint256 _split); /// @notice sets the swap fee for a pair /// @param _pair the address of the pair /// @param _fee the fee for the pair function setPairFee(address _pair, uint256 _fee) external; /// @notice set the swap fees of the pair /// @param _fee the fee, scaled to MAX 500_000 = 50% function setFee(uint256 _fee) external; /// @notice the address for the treasury /// @return _treasury address of the treasury function treasury() external view returns (address _treasury); /// @notice sets the pairFees contract /// @param _pair the address of the pair /// @param _pairFees the address of the new Pair Fees function setFeeRecipient(address _pair, address _pairFees) external; /// @notice sets the feeSplit for a pair /// @param _pair the address of the pair /// @param _feeSplit the feeSplit function setPairFeeSplit(address _pair, uint256 _feeSplit) external; /// @notice whether there is feeSplit when there's no gauge /// @return _boolean whether there is a feesplit when no gauge function feeSplitWhenNoGauge() external view returns (bool _boolean); /// @notice whether a pair can be skimmed /// @param _pair the pair address /// @return _boolean whether skim is enabled function skimEnabled(address _pair) external view returns (bool _boolean); /// @notice set whether skim is enabled for a specific pair function setSkimEnabled(address _pair, bool _status) external; /// @notice sets a new treasury address /// @param _treasury the new treasury address function setTreasury(address _treasury) external; /// @notice set whether there should be a feesplit without gauges /// @param status whether enabled or not function setFeeSplitWhenNoGauge(bool status) external; /// @notice sets the feesSplit globally /// @param _feeSplit the fee split function setFeeSplit(uint256 _feeSplit) external; }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; interface IFeeRecipientFactory { /// @notice the pair fees for a specific pair /// @param pair the pair to check /// @return feeRecipient the feeRecipient contract address for the pair function feeRecipientForPair(address pair) external view returns (address feeRecipient); /// @notice the last feeRecipient address created /// @return _feeRecipient the address of the last pair fees contract function lastFeeRecipient() external view returns (address _feeRecipient); /// @notice create the pair fees for a pair /// @param pair the address of the pair /// @return _feeRecipient the address of the newly created feeRecipient function createFeeRecipient(address pair) external returns (address _feeRecipient); /// @notice the fee % going to the treasury /// @return _feeToTreasury the fee % function feeToTreasury() external view returns (uint256 _feeToTreasury); /// @notice get the treasury address /// @return _treasury address of the treasury function treasury() external view returns (address _treasury); /// @notice set the fee % to be sent to the treasury /// @param _feeToTreasury the fee % to be sent to the treasury function setFeeToTreasury(uint256 _feeToTreasury) external; /// @notice set a new treasury address /// @param _treasury the new address function setTreasury(address _treasury) external; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; interface IFeeRecipient { function initialize(address _feeDistributor) external; function notifyFees() external; }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; interface IPair { event Mint(address indexed sender, uint256 amount0, uint256 amount1); event Burn(address indexed sender, uint256 amount0, uint256 amount1, address indexed to); event Swap( address indexed sender, uint256 amount0In, uint256 amount1In, uint256 amount0Out, uint256 amount1Out, address indexed to ); event Sync(uint112 reserve0, uint112 reserve1); /// @notice initialize the pool, called only once programatically function initialize(address _token0, address _token1, bool _stable) external; /// @notice calculate the current reserves of the pool and their last 'seen' timestamp /// @return _reserve0 amount of token0 in reserves /// @return _reserve1 amount of token1 in reserves /// @return _blockTimestampLast the timestamp when the pool was last updated function getReserves() external view returns (uint112 _reserve0, uint112 _reserve1, uint32 _blockTimestampLast); /// @notice mint the pair tokens (LPs) /// @param to where to mint the LP tokens to /// @return liquidity amount of LP tokens to mint function mint(address to) external returns (uint256 liquidity); /// @notice burn the pair tokens (LPs) /// @param to where to send the underlying /// @return amount0 amount of amount0 /// @return amount1 amount of amount1 function burn(address to) external returns (uint256 amount0, uint256 amount1); /// @notice direct swap through the pool function swap(uint256 amount0Out, uint256 amount1Out, address to, bytes calldata data) external; /// @notice force balances to match reserves, can be used to harvest rebases from rebasing tokens or other external factors /// @param to where to send the excess tokens to function skim(address to) external; /// @notice force reserves to match balances, prevents skim excess if skim is enabled function sync() external; /// @notice set the pair fees contract address function setFeeRecipient(address _pairFees) external; /// @notice set the feesplit variable function setFeeSplit(uint256 _feeSplit) external; /// @notice sets the swap fee of the pair /// @dev scaled to a max of 50% (500_000/1_000_000) /// @param _fee the fee function setFee(uint256 _fee) external; /// @notice 'mint' the fees as LP tokens /// @dev this is used for protocol/voter fees function mintFee() external; /// @notice calculates the amount of tokens to receive post swap /// @param amountIn the token amount /// @param tokenIn the address of the token function getAmountOut(uint256 amountIn, address tokenIn) external view returns (uint256 amountOut); /// @notice returns various metadata about the pair function metadata() external view returns ( uint256 _decimals0, uint256 _decimals1, uint256 _reserve0, uint256 _reserve1, bool _stable, address _token0, address _token1 ); /// @notice returns the feeSplit of the pair function feeSplit() external view returns (uint256); /// @notice returns the fee of the pair function fee() external view returns (uint256); /// @notice returns the feeRecipient of the pair function feeRecipient() external view returns (address); /// @notice returns the token0 of the pair function token0() external view returns (address); /// @notice returns the token1 of the pair function token1() external view returns (address); /// @notice returns if pair is stable function stable() external view returns (bool); /// @notice returns kLast function kLast() external view returns (uint256); }
// SPDX-License-Identifier: BUSL-1.1 pragma solidity ^0.8.0; pragma abicoder v2; interface IVoter { event GaugeCreated(address indexed gauge, address creator, address feeDistributor, address indexed pool); event GaugeKilled(address indexed gauge); event GaugeRevived(address indexed gauge); event Voted(address indexed owner, uint256 weight, address indexed pool); event Abstained(address indexed owner, uint256 weight); event Deposit(address indexed lp, address indexed gauge, address indexed owner, uint256 amount); event Withdraw(address indexed lp, address indexed gauge, address indexed owner, uint256 amount); event NotifyReward(address indexed sender, address indexed reward, uint256 amount); event DistributeReward(address indexed sender, address indexed gauge, uint256 amount); event EmissionsRatio(address indexed caller, uint256 oldRatio, uint256 newRatio); event NewGovernor(address indexed sender, address indexed governor); event Whitelisted(address indexed whitelister, address indexed token); event WhitelistRevoked(address indexed forbidder, address indexed token, bool status); event MainTickSpacingChanged(address indexed token0, address indexed token1, int24 indexed newMainTickSpacing); event Poke(address indexed user); event EmissionsRedirected(address indexed sourceGauge, address indexed destinationGauge); struct InitializationParams { address shadow; address legacyFactory; address gauges; address feeDistributorFactory; address minter; address msig; address xShadow; address clFactory; address clGaugeFactory; address nfpManager; address feeRecipientFactory; address voteModule; address launcherPlugin; address poolUpdater; } function initialize(InitializationParams memory inputs) external; /// @notice denominator basis function BASIS() external view returns (uint256); /// @notice ratio of xShadow emissions globally function xRatio() external view returns (uint256); /// @notice xShadow contract address function xShadow() external view returns (address); /// @notice legacy factory address (uni-v2/stableswap) function legacyFactory() external view returns (address); /// @notice concentrated liquidity factory function clFactory() external view returns (address); /// @notice gauge factory for CL function clGaugeFactory() external view returns (address); /// @notice pool updater for CL function poolUpdater() external view returns (address); /// @notice legacy fee recipient factory function feeRecipientFactory() external view returns (address); /// @notice peripheral NFPManager contract function nfpManager() external view returns (address); /// @notice returns the address of the current governor /// @return _governor address of the governor function governor() external view returns (address _governor); /// @notice the address of the vote module /// @return _voteModule the vote module contract address function voteModule() external view returns (address _voteModule); /// @notice address of the central access Hub function accessHub() external view returns (address); /// @notice the address of the shadow launcher plugin to enable third party launchers /// @return _launcherPlugin the address of the plugin function launcherPlugin() external view returns (address _launcherPlugin); /// @notice distributes emissions from the minter to the voter /// @param amount the amount of tokens to notify function notifyRewardAmount(uint256 amount) external; /// @notice distributes the emissions for a specific gauge /// @param _gauge the gauge address function distribute(address _gauge) external; /// @notice returns the address of the gauge factory /// @param _gaugeFactory gauge factory address function gaugeFactory() external view returns (address _gaugeFactory); /// @notice returns the address of the feeDistributor factory /// @return _feeDistributorFactory feeDist factory address function feeDistributorFactory() external view returns (address _feeDistributorFactory); /// @notice returns the address of the minter contract /// @return _minter address of the minter function minter() external view returns (address _minter); /// @notice check if the gauge is active for governance use /// @param _gauge address of the gauge /// @return _trueOrFalse if the gauge is alive function isAlive(address _gauge) external view returns (bool _trueOrFalse); /// @notice allows the token to be paired with other whitelisted assets to participate in governance /// @param _token the address of the token function whitelist(address _token) external; /// @notice effectively disqualifies a token from governance /// @param _token the address of the token function revokeWhitelist(address _token) external; /// @notice returns if the address is a gauge /// @param gauge address of the gauge /// @return _trueOrFalse boolean if the address is a gauge function isGauge(address gauge) external view returns (bool _trueOrFalse); /// @notice disable a gauge from governance /// @param _gauge address of the gauge function killGauge(address _gauge) external; /// @notice re-activate a dead gauge /// @param _gauge address of the gauge function reviveGauge(address _gauge) external; /// @notice re-cast a tokenID's votes /// @param owner address of the owner function poke(address owner) external; /// @notice sets the main destinationGauge of a token pairing /// @param tokenA address of tokenA /// @param tokenB address of tokenB /// @param destinationGauge the main gauge to set to function redirectEmissions(address tokenA, address tokenB, address destinationGauge) external; /// @notice returns if the address is a fee distributor /// @param _feeDistributor address of the feeDist /// @return _trueOrFalse if the address is a fee distributor function isFeeDistributor(address _feeDistributor) external view returns (bool _trueOrFalse); /// @notice returns the address of the emission's token /// @return _shadow emissions token contract address function shadow() external view returns (address _shadow); /// @notice returns the address of the pool's gauge, if any /// @param _pool pool address /// @return _gauge gauge address function gaugeForPool(address _pool) external view returns (address _gauge); /// @notice returns the address of the pool's feeDistributor, if any /// @param _gauge address of the gauge /// @return _feeDistributor address of the pool's feedist function feeDistributorForGauge(address _gauge) external view returns (address _feeDistributor); /// @notice returns the gauge address of a CL pool /// @param tokenA address of token A in the pair /// @param tokenB address of token B in the pair /// @param tickSpacing tickspacing of the pool /// @return gauge address of the gauge function gaugeForClPool(address tokenA, address tokenB, int24 tickSpacing) external view returns (address gauge); /// @notice returns the array of all tickspacings for the tokenA/tokenB combination /// @param tokenA address of token A in the pair /// @param tokenB address of token B in the pair /// @return _ts array of all the tickspacings function tickSpacingsForPair(address tokenA, address tokenB) external view returns (int24[] memory _ts); /// @notice returns the destination of a gauge redirect /// @param gauge address of gauge function gaugeRedirect(address gauge) external view returns (address); /// @notice returns the block.timestamp divided by 1 week in seconds /// @return period the period used for gauges function getPeriod() external view returns (uint256 period); /// @notice cast a vote to direct emissions to gauges and earn incentives /// @param owner address of the owner /// @param _pools the list of pools to vote on /// @param _weights an arbitrary weight per pool which will be normalized to 100% regardless of numerical inputs function vote(address owner, address[] calldata _pools, uint256[] calldata _weights) external; /// @notice reset the vote of an address /// @param owner address of the owner function reset(address owner) external; /// @notice set the governor address /// @param _governor the new governor address function setGovernor(address _governor) external; /// @notice recover stuck emissions /// @param _gauge the gauge address /// @param _period the period function stuckEmissionsRecovery(address _gauge, uint256 _period) external; /// @notice creates a legacy gauge for the pool /// @param _pool pool's address /// @return _gauge address of the new gauge function createGauge(address _pool) external returns (address _gauge); /// @notice create a concentrated liquidity gauge /// @param tokenA the address of tokenA /// @param tokenB the address of tokenB /// @param tickSpacing the tickspacing of the pool /// @return _clGauge address of the new gauge function createCLGauge(address tokenA, address tokenB, int24 tickSpacing) external returns (address _clGauge); /// @notice claim concentrated liquidity gauge rewards for specific NFP token ids /// @param _gauges array of gauges /// @param _tokens two dimensional array for the tokens to claim /// @param _nfpTokenIds two dimensional array for the NFPs function claimClGaugeRewards( address[] calldata _gauges, address[][] calldata _tokens, uint256[][] calldata _nfpTokenIds ) external; /// @notice claim arbitrary rewards from specific feeDists /// @param owner address of the owner /// @param _feeDistributors address of the feeDists /// @param _tokens two dimensional array for the tokens to claim function claimIncentives(address owner, address[] calldata _feeDistributors, address[][] calldata _tokens) external; /// @notice claim arbitrary rewards from specific feeDists and break up legacy pairs /// @param owner address of the owner /// @param _feeDistributors address of the feeDists /// @param _tokens two dimensional array for the tokens to claim function claimLegacyIncentives(address owner, address[] calldata _feeDistributors, address[][] calldata _tokens) external; /// @notice claim arbitrary rewards from specific gauges /// @param _gauges address of the gauges /// @param _tokens two dimensional array for the tokens to claim function claimRewards(address[] calldata _gauges, address[][] calldata _tokens) external; /// @notice claim arbitrary rewards from specific legacy gauges, and exit to shadow /// @param _gauges address of the gauges /// @param _tokens two dimensional array for the tokens to claim function claimLegacyRewardsAndExit(address[] calldata _gauges, address[][] calldata _tokens) external; /// @notice claim arbitrary rewards from specific cl gauges, and exit to shadow /// @param _gauges address of the gauges /// @param _tokens two dimensional array for the tokens to claim /// @param _nfpTokenIds two dimensional array for the nfp to claim function claimClGaugeRewardsAndExit( address[] memory _gauges, address[][] memory _tokens, uint256[][] memory _nfpTokenIds ) external; /// @notice distribute emissions to a gauge for a specific period /// @param _gauge address of the gauge /// @param _period value of the period function distributeForPeriod(address _gauge, uint256 _period) external; /// @notice attempt distribution of emissions to all gauges function distributeAll() external; /// @notice distribute emissions to gauges by index /// @param startIndex start of the loop /// @param endIndex end of the loop function batchDistributeByIndex(uint256 startIndex, uint256 endIndex) external; /// @notice lets governance update lastDistro period for a gauge /// @dev should only be used if distribute() is running out of gas /// @dev gaugePeriodDistributed will stop double claiming /// @param _gauge gauge to update /// @param _period period to update to function updateLastDistro(address _gauge, uint256 _period) external; /// @notice returns the votes cast for a tokenID /// @param owner address of the owner /// @return votes an array of votes casted /// @return weights an array of the weights casted per pool function getVotes(address owner, uint256 period) external view returns (address[] memory votes, uint256[] memory weights); /// @notice returns an array of all the pools /// @return _pools the array of pools function getAllPools() external view returns (address[] memory _pools); /// @notice returns the length of pools function getPoolsLength() external view returns (uint256); /// @notice returns the pool at index function getPool(uint256 index) external view returns (address); /// @notice returns an array of all the gauges /// @return _gauges the array of gauges function getAllGauges() external view returns (address[] memory _gauges); /// @notice returns the length of gauges function getGaugesLength() external view returns (uint256); /// @notice returns the gauge at index function getGauge(uint256 index) external view returns (address); /// @notice returns an array of all the feeDists /// @return _feeDistributors the array of feeDists function getAllFeeDistributors() external view returns (address[] memory _feeDistributors); /// @notice sets the xShadowRatio default function setGlobalRatio(uint256 _xRatio) external; /// @notice whether the token is whitelisted in governance function isWhitelisted(address _token) external view returns (bool _tf); /// @notice function for removing malicious or stuffed tokens function removeFeeDistributorReward(address _feeDist, address _token) external; /// @notice returns the total votes for a pool in a specific period /// @param pool the pool address to check /// @param period the period to check /// @return votes the total votes for the pool in that period function poolTotalVotesPerPeriod(address pool, uint256 period) external view returns (uint256 votes); /// @notice returns the pool address for a given gauge /// @param gauge address of the gauge /// @return pool address of the pool function poolForGauge(address gauge) external view returns (address pool); /// @notice returns the pool address for a given feeDistributor /// @param feeDistributor address of the feeDistributor /// @return pool address of the pool function poolForFeeDistributor(address feeDistributor) external view returns (address pool); /// @notice returns the voting power used by a voter for a period /// @param user address of the user /// @param period the period to check function userVotingPowerPerPeriod(address user, uint256 period) external view returns (uint256 votingPower); /// @notice returns the total votes for a specific period /// @param period the period to check /// @return weight the total votes for that period function totalVotesPerPeriod(uint256 period) external view returns (uint256 weight); /// @notice returns the total rewards allocated for a specific period /// @param period the period to check /// @return amount the total rewards for that period function totalRewardPerPeriod(uint256 period) external view returns (uint256 amount); /// @notice returns the last distribution period for a gauge /// @param _gauge address of the gauge /// @return period the last period distributions occurred function lastDistro(address _gauge) external view returns (uint256 period); /// @notice returns if the gauge is a Cl gauge /// @param gauge the gauge to check function isClGauge(address gauge) external view returns (bool); /// @notice returns if the gauge is a legacy gauge /// @param gauge the gauge to check function isLegacyGauge(address gauge) external view returns (bool); /// @notice sets a new NFP manager function setNfpManager(address _nfpManager) external; }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; interface IVoteModule { /** * Events */ event Deposit(address indexed from, uint256 amount); event Withdraw(address indexed from, uint256 amount); event NotifyReward(address indexed from, uint256 amount); event ClaimRewards(address indexed from, uint256 amount); event ExemptedFromCooldown(address indexed candidate, bool status); event NewDuration(uint256 oldDuration, uint256 newDuration); event NewCooldown(uint256 oldCooldown, uint256 newCooldown); event Delegate(address indexed delegator, address indexed delegatee, bool indexed isAdded); event SetAdmin(address indexed owner, address indexed operator, bool indexed isAdded); /** * Functions */ function delegates(address) external view returns (address); /// @notice mapping for admins for a specific address /// @param owner the owner to check against /// @return operator the address that is designated as an admin/operator function admins(address owner) external view returns (address operator); function accessHub() external view returns (address); /// @notice reward supply for a period function rewardSupply(uint256 period) external view returns (uint256); /// @notice user claimed reward amount for a period /// @dev same mapping order as FeeDistributor so the name is a bit odd function userClaimed(uint256 period, address owner) external view returns (uint256); /// @notice last claimed period for a user function userLastClaimPeriod(address owner) external view returns (uint256); /// @notice returns the current period function getPeriod() external view returns (uint256); /// @notice returns the amount of unclaimed rebase earned by the user function earned(address account) external view returns (uint256 _reward); /// @notice returns the amount of unclaimed rebase earned by the user for a period function periodEarned(uint256 period, address user) external view returns (uint256 amount); /// @notice the time which users can deposit and withdraw function unlockTime() external view returns (uint256 _timestamp); /// @notice claims pending rebase rewards function getReward() external; /// @notice claims pending rebase rewards for a period function getPeriodReward(uint256 period) external; /// @notice allows users to set their own last claimed period in case they haven't claimed in a while /// @param period the new period to start loops from function setUserLastClaimPeriod(uint256 period) external; /// @notice deposits all xShadow in the caller's wallet function depositAll() external; /// @notice deposit a specified amount of xShadow function deposit(uint256 amount) external; /// @notice withdraw all xShadow function withdrawAll() external; /// @notice withdraw a specified amount of xShadow function withdraw(uint256 amount) external; /// @notice check for admin perms /// @param operator the address to check /// @param owner the owner to check against for permissions function isAdminFor(address operator, address owner) external view returns (bool approved); /// @notice check for delegations /// @param delegate the address to check /// @param owner the owner to check against for permissions function isDelegateFor(address delegate, address owner) external view returns (bool approved); /// @notice used by the xShadow contract to notify pending rebases /// @param amount the amount of Shadow to be notified from exit penalties function notifyRewardAmount(uint256 amount) external; /// @notice the address of the xShadow token (staking/voting token) /// @return _xShadow the address function xShadow() external view returns (address _xShadow); /// @notice address of the voter contract /// @return _voter the voter contract address function voter() external view returns (address _voter); /// @notice returns the total voting power (equal to total supply in the VoteModule) /// @return _totalSupply the total voting power function totalSupply() external view returns (uint256 _totalSupply); /// @notice voting power /// @param user the address to check /// @return amount the staked balance function balanceOf(address user) external view returns (uint256 amount); /// @notice delegate voting perms to another address /// @param delegatee who you delegate to /// @dev set address(0) to revoke function delegate(address delegatee) external; /// @notice give admin permissions to a another address /// @param operator the address to give administrative perms to /// @dev set address(0) to revoke function setAdmin(address operator) external; function cooldownExempt(address) external view returns (bool); function setCooldownExemption(address, bool) external; /// @notice lock period after rebase starts accruing function cooldown() external returns (uint256); function setNewCooldown(uint256) external; }
// SPDX-License-Identifier: GPL-2.0-or-later pragma solidity ^0.8.0; import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import {IVoter} from "./IVoter.sol"; import {Pausable} from "@openzeppelin/contracts/utils/Pausable.sol"; interface IXShadow is IERC20 { struct VestPosition { /// @dev amount of xShadow uint256 amount; /// @dev start unix timestamp uint256 start; /// @dev start + MAX_VEST (end timestamp) uint256 maxEnd; /// @dev vest identifier (starting from 0) uint256 vestID; } event CancelVesting(address indexed user, uint256 indexed vestId, uint256 amount); event ExitVesting(address indexed user, uint256 indexed vestId, uint256 amount); event InstantExit(address indexed user, uint256); event NewSlashingPenalty(uint256 penalty); event NewVest(address indexed user, uint256 indexed vestId, uint256 indexed amount); event NewVestingTimes(uint256 min, uint256 max); event Converted(address indexed user, uint256); event Exemption(address indexed candidate, bool status, bool success); event XShadowRedeemed(address indexed user, uint256); event NewOperator(address indexed o, address indexed n); event Rebase(address indexed caller, uint256 amount); event NewRebaseThreshold(uint256 threshold); /// @notice returns info on a user's vests function vestInfo(address user, uint256) external view returns (uint256 amount, uint256 start, uint256 maxEnd, uint256 vestID); /// @notice address of the shadow token function SHADOW() external view returns (IERC20); /// @notice address of the voter function VOTER() external view returns (IVoter); function MINTER() external view returns (address); function ACCESS_HUB() external view returns (address); /// @notice address of the operator function operator() external view returns (address); /// @notice address of the VoteModule function VOTE_MODULE() external view returns (address); /// @notice max slashing amount function SLASHING_PENALTY() external view returns (uint256); /// @notice denominator function BASIS() external view returns (uint256); /// @notice the minimum vesting length function MIN_VEST() external view returns (uint256); /// @notice the maximum vesting length function MAX_VEST() external view returns (uint256); function shadow() external view returns (address); /// @notice the last period rebases were distributed function lastDistributedPeriod() external view returns (uint256); /// @notice amount of pvp rebase penalties accumulated pending to be distributed function pendingRebase() external view returns (uint256); /// @notice dust threshold before a rebase can happen function rebaseThreshold() external view returns (uint256); /// @notice pauses the contract function pause() external; /// @notice unpauses the contract function unpause() external; /** * */ // General use functions /** * */ /// @dev mints xShadows for each shadow. function convertEmissionsToken(uint256 _amount) external; /// @notice function called by the minter to send the rebases once a week function rebase() external; /** * @dev exit instantly with a penalty * @param _amount amount of xShadows to exit */ function exit(uint256 _amount) external returns (uint256 _exitedAmount); /// @dev vesting xShadows --> emissionToken functionality function createVest(uint256 _amount) external; /// @dev handles all situations regarding exiting vests function exitVest(uint256 _vestID) external; /** * */ // Permissioned functions, timelock/operator gated /** * */ /// @dev allows the operator to redeem collected xShadows function operatorRedeem(uint256 _amount) external; /// @dev allows rescue of any non-stake token function rescueTrappedTokens(address[] calldata _tokens, uint256[] calldata _amounts) external; /// @notice migrates the operator to another contract function migrateOperator(address _operator) external; /// @notice set exemption status for an address function setExemption(address[] calldata _exemptee, bool[] calldata _exempt) external; function setExemptionTo(address[] calldata _exemptee, bool[] calldata _exempt) external; /// @notice set dust threshold before a rebase can happen function setRebaseThreshold(uint256 _newThreshold) external; /** * */ // Getter functions /** * */ /// @notice returns the amount of SHADOW within the contract function getBalanceResiding() external view returns (uint256); /// @notice returns the total number of individual vests the user has function usersTotalVests(address _who) external view returns (uint256 _numOfVests); /// @notice whether the address is exempt /// @param _who who to check /// @return _exempt whether it's exempt function isExempt(address _who) external view returns (bool _exempt); /// @notice returns the vest info for a user /// @param _who who to check /// @param _vestID vest ID to check /// @return VestPosition vest info function getVestInfo(address _who, uint256 _vestID) external view returns (VestPosition memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/Panic.sol) pragma solidity ^0.8.20; /** * @dev Helper library for emitting standardized panic codes. * * ```solidity * contract Example { * using Panic for uint256; * * // Use any of the declared internal constants * function foo() { Panic.GENERIC.panic(); } * * // Alternatively * function foo() { Panic.panic(Panic.GENERIC); } * } * ``` * * Follows the list from https://github.com/ethereum/solidity/blob/v0.8.24/libsolutil/ErrorCodes.h[libsolutil]. * * _Available since v5.1._ */ // slither-disable-next-line unused-state library Panic { /// @dev generic / unspecified error uint256 internal constant GENERIC = 0x00; /// @dev used by the assert() builtin uint256 internal constant ASSERT = 0x01; /// @dev arithmetic underflow or overflow uint256 internal constant UNDER_OVERFLOW = 0x11; /// @dev division or modulo by zero uint256 internal constant DIVISION_BY_ZERO = 0x12; /// @dev enum conversion error uint256 internal constant ENUM_CONVERSION_ERROR = 0x21; /// @dev invalid encoding in storage uint256 internal constant STORAGE_ENCODING_ERROR = 0x22; /// @dev empty array pop uint256 internal constant EMPTY_ARRAY_POP = 0x31; /// @dev array out of bounds access uint256 internal constant ARRAY_OUT_OF_BOUNDS = 0x32; /// @dev resource error (too large allocation or too large array) uint256 internal constant RESOURCE_ERROR = 0x41; /// @dev calling invalid internal function uint256 internal constant INVALID_INTERNAL_FUNCTION = 0x51; /// @dev Reverts with a panic code. Recommended to use with /// the internal constants with predefined codes. function panic(uint256 code) internal pure { assembly ("memory-safe") { mstore(0x00, 0x4e487b71) mstore(0x20, code) revert(0x1c, 0x24) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/math/SafeCast.sol) // This file was procedurally generated from scripts/generate/templates/SafeCast.js. pragma solidity ^0.8.20; /** * @dev Wrappers over Solidity's uintXX/intXX/bool casting operators with added overflow * checks. * * Downcasting from uint256/int256 in Solidity does not revert on overflow. This can * easily result in undesired exploitation or bugs, since developers usually * assume that overflows raise errors. `SafeCast` restores this intuition by * reverting the transaction when such an operation overflows. * * Using this library instead of the unchecked operations eliminates an entire * class of bugs, so it's recommended to use it always. */ library SafeCast { /** * @dev Value doesn't fit in an uint of `bits` size. */ error SafeCastOverflowedUintDowncast(uint8 bits, uint256 value); /** * @dev An int value doesn't fit in an uint of `bits` size. */ error SafeCastOverflowedIntToUint(int256 value); /** * @dev Value doesn't fit in an int of `bits` size. */ error SafeCastOverflowedIntDowncast(uint8 bits, int256 value); /** * @dev An uint value doesn't fit in an int of `bits` size. */ error SafeCastOverflowedUintToInt(uint256 value); /** * @dev Returns the downcasted uint248 from uint256, reverting on * overflow (when the input is greater than largest uint248). * * Counterpart to Solidity's `uint248` operator. * * Requirements: * * - input must fit into 248 bits */ function toUint248(uint256 value) internal pure returns (uint248) { if (value > type(uint248).max) { revert SafeCastOverflowedUintDowncast(248, value); } return uint248(value); } /** * @dev Returns the downcasted uint240 from uint256, reverting on * overflow (when the input is greater than largest uint240). * * Counterpart to Solidity's `uint240` operator. * * Requirements: * * - input must fit into 240 bits */ function toUint240(uint256 value) internal pure returns (uint240) { if (value > type(uint240).max) { revert SafeCastOverflowedUintDowncast(240, value); } return uint240(value); } /** * @dev Returns the downcasted uint232 from uint256, reverting on * overflow (when the input is greater than largest uint232). * * Counterpart to Solidity's `uint232` operator. * * Requirements: * * - input must fit into 232 bits */ function toUint232(uint256 value) internal pure returns (uint232) { if (value > type(uint232).max) { revert SafeCastOverflowedUintDowncast(232, value); } return uint232(value); } /** * @dev Returns the downcasted uint224 from uint256, reverting on * overflow (when the input is greater than largest uint224). * * Counterpart to Solidity's `uint224` operator. * * Requirements: * * - input must fit into 224 bits */ function toUint224(uint256 value) internal pure returns (uint224) { if (value > type(uint224).max) { revert SafeCastOverflowedUintDowncast(224, value); } return uint224(value); } /** * @dev Returns the downcasted uint216 from uint256, reverting on * overflow (when the input is greater than largest uint216). * * Counterpart to Solidity's `uint216` operator. * * Requirements: * * - input must fit into 216 bits */ function toUint216(uint256 value) internal pure returns (uint216) { if (value > type(uint216).max) { revert SafeCastOverflowedUintDowncast(216, value); } return uint216(value); } /** * @dev Returns the downcasted uint208 from uint256, reverting on * overflow (when the input is greater than largest uint208). * * Counterpart to Solidity's `uint208` operator. * * Requirements: * * - input must fit into 208 bits */ function toUint208(uint256 value) internal pure returns (uint208) { if (value > type(uint208).max) { revert SafeCastOverflowedUintDowncast(208, value); } return uint208(value); } /** * @dev Returns the downcasted uint200 from uint256, reverting on * overflow (when the input is greater than largest uint200). * * Counterpart to Solidity's `uint200` operator. * * Requirements: * * - input must fit into 200 bits */ function toUint200(uint256 value) internal pure returns (uint200) { if (value > type(uint200).max) { revert SafeCastOverflowedUintDowncast(200, value); } return uint200(value); } /** * @dev Returns the downcasted uint192 from uint256, reverting on * overflow (when the input is greater than largest uint192). * * Counterpart to Solidity's `uint192` operator. * * Requirements: * * - input must fit into 192 bits */ function toUint192(uint256 value) internal pure returns (uint192) { if (value > type(uint192).max) { revert SafeCastOverflowedUintDowncast(192, value); } return uint192(value); } /** * @dev Returns the downcasted uint184 from uint256, reverting on * overflow (when the input is greater than largest uint184). * * Counterpart to Solidity's `uint184` operator. * * Requirements: * * - input must fit into 184 bits */ function toUint184(uint256 value) internal pure returns (uint184) { if (value > type(uint184).max) { revert SafeCastOverflowedUintDowncast(184, value); } return uint184(value); } /** * @dev Returns the downcasted uint176 from uint256, reverting on * overflow (when the input is greater than largest uint176). * * Counterpart to Solidity's `uint176` operator. * * Requirements: * * - input must fit into 176 bits */ function toUint176(uint256 value) internal pure returns (uint176) { if (value > type(uint176).max) { revert SafeCastOverflowedUintDowncast(176, value); } return uint176(value); } /** * @dev Returns the downcasted uint168 from uint256, reverting on * overflow (when the input is greater than largest uint168). * * Counterpart to Solidity's `uint168` operator. * * Requirements: * * - input must fit into 168 bits */ function toUint168(uint256 value) internal pure returns (uint168) { if (value > type(uint168).max) { revert SafeCastOverflowedUintDowncast(168, value); } return uint168(value); } /** * @dev Returns the downcasted uint160 from uint256, reverting on * overflow (when the input is greater than largest uint160). * * Counterpart to Solidity's `uint160` operator. * * Requirements: * * - input must fit into 160 bits */ function toUint160(uint256 value) internal pure returns (uint160) { if (value > type(uint160).max) { revert SafeCastOverflowedUintDowncast(160, value); } return uint160(value); } /** * @dev Returns the downcasted uint152 from uint256, reverting on * overflow (when the input is greater than largest uint152). * * Counterpart to Solidity's `uint152` operator. * * Requirements: * * - input must fit into 152 bits */ function toUint152(uint256 value) internal pure returns (uint152) { if (value > type(uint152).max) { revert SafeCastOverflowedUintDowncast(152, value); } return uint152(value); } /** * @dev Returns the downcasted uint144 from uint256, reverting on * overflow (when the input is greater than largest uint144). * * Counterpart to Solidity's `uint144` operator. * * Requirements: * * - input must fit into 144 bits */ function toUint144(uint256 value) internal pure returns (uint144) { if (value > type(uint144).max) { revert SafeCastOverflowedUintDowncast(144, value); } return uint144(value); } /** * @dev Returns the downcasted uint136 from uint256, reverting on * overflow (when the input is greater than largest uint136). * * Counterpart to Solidity's `uint136` operator. * * Requirements: * * - input must fit into 136 bits */ function toUint136(uint256 value) internal pure returns (uint136) { if (value > type(uint136).max) { revert SafeCastOverflowedUintDowncast(136, value); } return uint136(value); } /** * @dev Returns the downcasted uint128 from uint256, reverting on * overflow (when the input is greater than largest uint128). * * Counterpart to Solidity's `uint128` operator. * * Requirements: * * - input must fit into 128 bits */ function toUint128(uint256 value) internal pure returns (uint128) { if (value > type(uint128).max) { revert SafeCastOverflowedUintDowncast(128, value); } return uint128(value); } /** * @dev Returns the downcasted uint120 from uint256, reverting on * overflow (when the input is greater than largest uint120). * * Counterpart to Solidity's `uint120` operator. * * Requirements: * * - input must fit into 120 bits */ function toUint120(uint256 value) internal pure returns (uint120) { if (value > type(uint120).max) { revert SafeCastOverflowedUintDowncast(120, value); } return uint120(value); } /** * @dev Returns the downcasted uint112 from uint256, reverting on * overflow (when the input is greater than largest uint112). * * Counterpart to Solidity's `uint112` operator. * * Requirements: * * - input must fit into 112 bits */ function toUint112(uint256 value) internal pure returns (uint112) { if (value > type(uint112).max) { revert SafeCastOverflowedUintDowncast(112, value); } return uint112(value); } /** * @dev Returns the downcasted uint104 from uint256, reverting on * overflow (when the input is greater than largest uint104). * * Counterpart to Solidity's `uint104` operator. * * Requirements: * * - input must fit into 104 bits */ function toUint104(uint256 value) internal pure returns (uint104) { if (value > type(uint104).max) { revert SafeCastOverflowedUintDowncast(104, value); } return uint104(value); } /** * @dev Returns the downcasted uint96 from uint256, reverting on * overflow (when the input is greater than largest uint96). * * Counterpart to Solidity's `uint96` operator. * * Requirements: * * - input must fit into 96 bits */ function toUint96(uint256 value) internal pure returns (uint96) { if (value > type(uint96).max) { revert SafeCastOverflowedUintDowncast(96, value); } return uint96(value); } /** * @dev Returns the downcasted uint88 from uint256, reverting on * overflow (when the input is greater than largest uint88). * * Counterpart to Solidity's `uint88` operator. * * Requirements: * * - input must fit into 88 bits */ function toUint88(uint256 value) internal pure returns (uint88) { if (value > type(uint88).max) { revert SafeCastOverflowedUintDowncast(88, value); } return uint88(value); } /** * @dev Returns the downcasted uint80 from uint256, reverting on * overflow (when the input is greater than largest uint80). * * Counterpart to Solidity's `uint80` operator. * * Requirements: * * - input must fit into 80 bits */ function toUint80(uint256 value) internal pure returns (uint80) { if (value > type(uint80).max) { revert SafeCastOverflowedUintDowncast(80, value); } return uint80(value); } /** * @dev Returns the downcasted uint72 from uint256, reverting on * overflow (when the input is greater than largest uint72). * * Counterpart to Solidity's `uint72` operator. * * Requirements: * * - input must fit into 72 bits */ function toUint72(uint256 value) internal pure returns (uint72) { if (value > type(uint72).max) { revert SafeCastOverflowedUintDowncast(72, value); } return uint72(value); } /** * @dev Returns the downcasted uint64 from uint256, reverting on * overflow (when the input is greater than largest uint64). * * Counterpart to Solidity's `uint64` operator. * * Requirements: * * - input must fit into 64 bits */ function toUint64(uint256 value) internal pure returns (uint64) { if (value > type(uint64).max) { revert SafeCastOverflowedUintDowncast(64, value); } return uint64(value); } /** * @dev Returns the downcasted uint56 from uint256, reverting on * overflow (when the input is greater than largest uint56). * * Counterpart to Solidity's `uint56` operator. * * Requirements: * * - input must fit into 56 bits */ function toUint56(uint256 value) internal pure returns (uint56) { if (value > type(uint56).max) { revert SafeCastOverflowedUintDowncast(56, value); } return uint56(value); } /** * @dev Returns the downcasted uint48 from uint256, reverting on * overflow (when the input is greater than largest uint48). * * Counterpart to Solidity's `uint48` operator. * * Requirements: * * - input must fit into 48 bits */ function toUint48(uint256 value) internal pure returns (uint48) { if (value > type(uint48).max) { revert SafeCastOverflowedUintDowncast(48, value); } return uint48(value); } /** * @dev Returns the downcasted uint40 from uint256, reverting on * overflow (when the input is greater than largest uint40). * * Counterpart to Solidity's `uint40` operator. * * Requirements: * * - input must fit into 40 bits */ function toUint40(uint256 value) internal pure returns (uint40) { if (value > type(uint40).max) { revert SafeCastOverflowedUintDowncast(40, value); } return uint40(value); } /** * @dev Returns the downcasted uint32 from uint256, reverting on * overflow (when the input is greater than largest uint32). * * Counterpart to Solidity's `uint32` operator. * * Requirements: * * - input must fit into 32 bits */ function toUint32(uint256 value) internal pure returns (uint32) { if (value > type(uint32).max) { revert SafeCastOverflowedUintDowncast(32, value); } return uint32(value); } /** * @dev Returns the downcasted uint24 from uint256, reverting on * overflow (when the input is greater than largest uint24). * * Counterpart to Solidity's `uint24` operator. * * Requirements: * * - input must fit into 24 bits */ function toUint24(uint256 value) internal pure returns (uint24) { if (value > type(uint24).max) { revert SafeCastOverflowedUintDowncast(24, value); } return uint24(value); } /** * @dev Returns the downcasted uint16 from uint256, reverting on * overflow (when the input is greater than largest uint16). * * Counterpart to Solidity's `uint16` operator. * * Requirements: * * - input must fit into 16 bits */ function toUint16(uint256 value) internal pure returns (uint16) { if (value > type(uint16).max) { revert SafeCastOverflowedUintDowncast(16, value); } return uint16(value); } /** * @dev Returns the downcasted uint8 from uint256, reverting on * overflow (when the input is greater than largest uint8). * * Counterpart to Solidity's `uint8` operator. * * Requirements: * * - input must fit into 8 bits */ function toUint8(uint256 value) internal pure returns (uint8) { if (value > type(uint8).max) { revert SafeCastOverflowedUintDowncast(8, value); } return uint8(value); } /** * @dev Converts a signed int256 into an unsigned uint256. * * Requirements: * * - input must be greater than or equal to 0. */ function toUint256(int256 value) internal pure returns (uint256) { if (value < 0) { revert SafeCastOverflowedIntToUint(value); } return uint256(value); } /** * @dev Returns the downcasted int248 from int256, reverting on * overflow (when the input is less than smallest int248 or * greater than largest int248). * * Counterpart to Solidity's `int248` operator. * * Requirements: * * - input must fit into 248 bits */ function toInt248(int256 value) internal pure returns (int248 downcasted) { downcasted = int248(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(248, value); } } /** * @dev Returns the downcasted int240 from int256, reverting on * overflow (when the input is less than smallest int240 or * greater than largest int240). * * Counterpart to Solidity's `int240` operator. * * Requirements: * * - input must fit into 240 bits */ function toInt240(int256 value) internal pure returns (int240 downcasted) { downcasted = int240(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(240, value); } } /** * @dev Returns the downcasted int232 from int256, reverting on * overflow (when the input is less than smallest int232 or * greater than largest int232). * * Counterpart to Solidity's `int232` operator. * * Requirements: * * - input must fit into 232 bits */ function toInt232(int256 value) internal pure returns (int232 downcasted) { downcasted = int232(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(232, value); } } /** * @dev Returns the downcasted int224 from int256, reverting on * overflow (when the input is less than smallest int224 or * greater than largest int224). * * Counterpart to Solidity's `int224` operator. * * Requirements: * * - input must fit into 224 bits */ function toInt224(int256 value) internal pure returns (int224 downcasted) { downcasted = int224(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(224, value); } } /** * @dev Returns the downcasted int216 from int256, reverting on * overflow (when the input is less than smallest int216 or * greater than largest int216). * * Counterpart to Solidity's `int216` operator. * * Requirements: * * - input must fit into 216 bits */ function toInt216(int256 value) internal pure returns (int216 downcasted) { downcasted = int216(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(216, value); } } /** * @dev Returns the downcasted int208 from int256, reverting on * overflow (when the input is less than smallest int208 or * greater than largest int208). * * Counterpart to Solidity's `int208` operator. * * Requirements: * * - input must fit into 208 bits */ function toInt208(int256 value) internal pure returns (int208 downcasted) { downcasted = int208(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(208, value); } } /** * @dev Returns the downcasted int200 from int256, reverting on * overflow (when the input is less than smallest int200 or * greater than largest int200). * * Counterpart to Solidity's `int200` operator. * * Requirements: * * - input must fit into 200 bits */ function toInt200(int256 value) internal pure returns (int200 downcasted) { downcasted = int200(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(200, value); } } /** * @dev Returns the downcasted int192 from int256, reverting on * overflow (when the input is less than smallest int192 or * greater than largest int192). * * Counterpart to Solidity's `int192` operator. * * Requirements: * * - input must fit into 192 bits */ function toInt192(int256 value) internal pure returns (int192 downcasted) { downcasted = int192(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(192, value); } } /** * @dev Returns the downcasted int184 from int256, reverting on * overflow (when the input is less than smallest int184 or * greater than largest int184). * * Counterpart to Solidity's `int184` operator. * * Requirements: * * - input must fit into 184 bits */ function toInt184(int256 value) internal pure returns (int184 downcasted) { downcasted = int184(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(184, value); } } /** * @dev Returns the downcasted int176 from int256, reverting on * overflow (when the input is less than smallest int176 or * greater than largest int176). * * Counterpart to Solidity's `int176` operator. * * Requirements: * * - input must fit into 176 bits */ function toInt176(int256 value) internal pure returns (int176 downcasted) { downcasted = int176(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(176, value); } } /** * @dev Returns the downcasted int168 from int256, reverting on * overflow (when the input is less than smallest int168 or * greater than largest int168). * * Counterpart to Solidity's `int168` operator. * * Requirements: * * - input must fit into 168 bits */ function toInt168(int256 value) internal pure returns (int168 downcasted) { downcasted = int168(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(168, value); } } /** * @dev Returns the downcasted int160 from int256, reverting on * overflow (when the input is less than smallest int160 or * greater than largest int160). * * Counterpart to Solidity's `int160` operator. * * Requirements: * * - input must fit into 160 bits */ function toInt160(int256 value) internal pure returns (int160 downcasted) { downcasted = int160(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(160, value); } } /** * @dev Returns the downcasted int152 from int256, reverting on * overflow (when the input is less than smallest int152 or * greater than largest int152). * * Counterpart to Solidity's `int152` operator. * * Requirements: * * - input must fit into 152 bits */ function toInt152(int256 value) internal pure returns (int152 downcasted) { downcasted = int152(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(152, value); } } /** * @dev Returns the downcasted int144 from int256, reverting on * overflow (when the input is less than smallest int144 or * greater than largest int144). * * Counterpart to Solidity's `int144` operator. * * Requirements: * * - input must fit into 144 bits */ function toInt144(int256 value) internal pure returns (int144 downcasted) { downcasted = int144(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(144, value); } } /** * @dev Returns the downcasted int136 from int256, reverting on * overflow (when the input is less than smallest int136 or * greater than largest int136). * * Counterpart to Solidity's `int136` operator. * * Requirements: * * - input must fit into 136 bits */ function toInt136(int256 value) internal pure returns (int136 downcasted) { downcasted = int136(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(136, value); } } /** * @dev Returns the downcasted int128 from int256, reverting on * overflow (when the input is less than smallest int128 or * greater than largest int128). * * Counterpart to Solidity's `int128` operator. * * Requirements: * * - input must fit into 128 bits */ function toInt128(int256 value) internal pure returns (int128 downcasted) { downcasted = int128(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(128, value); } } /** * @dev Returns the downcasted int120 from int256, reverting on * overflow (when the input is less than smallest int120 or * greater than largest int120). * * Counterpart to Solidity's `int120` operator. * * Requirements: * * - input must fit into 120 bits */ function toInt120(int256 value) internal pure returns (int120 downcasted) { downcasted = int120(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(120, value); } } /** * @dev Returns the downcasted int112 from int256, reverting on * overflow (when the input is less than smallest int112 or * greater than largest int112). * * Counterpart to Solidity's `int112` operator. * * Requirements: * * - input must fit into 112 bits */ function toInt112(int256 value) internal pure returns (int112 downcasted) { downcasted = int112(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(112, value); } } /** * @dev Returns the downcasted int104 from int256, reverting on * overflow (when the input is less than smallest int104 or * greater than largest int104). * * Counterpart to Solidity's `int104` operator. * * Requirements: * * - input must fit into 104 bits */ function toInt104(int256 value) internal pure returns (int104 downcasted) { downcasted = int104(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(104, value); } } /** * @dev Returns the downcasted int96 from int256, reverting on * overflow (when the input is less than smallest int96 or * greater than largest int96). * * Counterpart to Solidity's `int96` operator. * * Requirements: * * - input must fit into 96 bits */ function toInt96(int256 value) internal pure returns (int96 downcasted) { downcasted = int96(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(96, value); } } /** * @dev Returns the downcasted int88 from int256, reverting on * overflow (when the input is less than smallest int88 or * greater than largest int88). * * Counterpart to Solidity's `int88` operator. * * Requirements: * * - input must fit into 88 bits */ function toInt88(int256 value) internal pure returns (int88 downcasted) { downcasted = int88(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(88, value); } } /** * @dev Returns the downcasted int80 from int256, reverting on * overflow (when the input is less than smallest int80 or * greater than largest int80). * * Counterpart to Solidity's `int80` operator. * * Requirements: * * - input must fit into 80 bits */ function toInt80(int256 value) internal pure returns (int80 downcasted) { downcasted = int80(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(80, value); } } /** * @dev Returns the downcasted int72 from int256, reverting on * overflow (when the input is less than smallest int72 or * greater than largest int72). * * Counterpart to Solidity's `int72` operator. * * Requirements: * * - input must fit into 72 bits */ function toInt72(int256 value) internal pure returns (int72 downcasted) { downcasted = int72(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(72, value); } } /** * @dev Returns the downcasted int64 from int256, reverting on * overflow (when the input is less than smallest int64 or * greater than largest int64). * * Counterpart to Solidity's `int64` operator. * * Requirements: * * - input must fit into 64 bits */ function toInt64(int256 value) internal pure returns (int64 downcasted) { downcasted = int64(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(64, value); } } /** * @dev Returns the downcasted int56 from int256, reverting on * overflow (when the input is less than smallest int56 or * greater than largest int56). * * Counterpart to Solidity's `int56` operator. * * Requirements: * * - input must fit into 56 bits */ function toInt56(int256 value) internal pure returns (int56 downcasted) { downcasted = int56(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(56, value); } } /** * @dev Returns the downcasted int48 from int256, reverting on * overflow (when the input is less than smallest int48 or * greater than largest int48). * * Counterpart to Solidity's `int48` operator. * * Requirements: * * - input must fit into 48 bits */ function toInt48(int256 value) internal pure returns (int48 downcasted) { downcasted = int48(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(48, value); } } /** * @dev Returns the downcasted int40 from int256, reverting on * overflow (when the input is less than smallest int40 or * greater than largest int40). * * Counterpart to Solidity's `int40` operator. * * Requirements: * * - input must fit into 40 bits */ function toInt40(int256 value) internal pure returns (int40 downcasted) { downcasted = int40(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(40, value); } } /** * @dev Returns the downcasted int32 from int256, reverting on * overflow (when the input is less than smallest int32 or * greater than largest int32). * * Counterpart to Solidity's `int32` operator. * * Requirements: * * - input must fit into 32 bits */ function toInt32(int256 value) internal pure returns (int32 downcasted) { downcasted = int32(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(32, value); } } /** * @dev Returns the downcasted int24 from int256, reverting on * overflow (when the input is less than smallest int24 or * greater than largest int24). * * Counterpart to Solidity's `int24` operator. * * Requirements: * * - input must fit into 24 bits */ function toInt24(int256 value) internal pure returns (int24 downcasted) { downcasted = int24(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(24, value); } } /** * @dev Returns the downcasted int16 from int256, reverting on * overflow (when the input is less than smallest int16 or * greater than largest int16). * * Counterpart to Solidity's `int16` operator. * * Requirements: * * - input must fit into 16 bits */ function toInt16(int256 value) internal pure returns (int16 downcasted) { downcasted = int16(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(16, value); } } /** * @dev Returns the downcasted int8 from int256, reverting on * overflow (when the input is less than smallest int8 or * greater than largest int8). * * Counterpart to Solidity's `int8` operator. * * Requirements: * * - input must fit into 8 bits */ function toInt8(int256 value) internal pure returns (int8 downcasted) { downcasted = int8(value); if (downcasted != value) { revert SafeCastOverflowedIntDowncast(8, value); } } /** * @dev Converts an unsigned uint256 into a signed int256. * * Requirements: * * - input must be less than or equal to maxInt256. */ function toInt256(uint256 value) internal pure returns (int256) { // Note: Unsafe cast below is okay because `type(int256).max` is guaranteed to be positive if (value > uint256(type(int256).max)) { revert SafeCastOverflowedUintToInt(value); } return int256(value); } /** * @dev Cast a boolean (false or true) to a uint256 (0 or 1) with no jump. */ function toUint(bool b) internal pure returns (uint256 u) { assembly ("memory-safe") { u := iszero(iszero(b)) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC-20 standard as defined in the ERC. */ 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 value of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the value of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves a `value` amount of 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 value) 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 a `value` amount of tokens 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 value) external returns (bool); /** * @dev Moves a `value` amount of tokens from `from` to `to` using the * allowance mechanism. `value` 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 value) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (interfaces/IERC1363.sol) pragma solidity ^0.8.20; import {IERC20} from "./IERC20.sol"; import {IERC165} from "./IERC165.sol"; /** * @title IERC1363 * @dev Interface of the ERC-1363 standard as defined in the https://eips.ethereum.org/EIPS/eip-1363[ERC-1363]. * * Defines an extension interface for ERC-20 tokens that supports executing code on a recipient contract * after `transfer` or `transferFrom`, or code on a spender contract after `approve`, in a single transaction. */ interface IERC1363 is IERC20, IERC165 { /* * Note: the ERC-165 identifier for this interface is 0xb0202a11. * 0xb0202a11 === * bytes4(keccak256('transferAndCall(address,uint256)')) ^ * bytes4(keccak256('transferAndCall(address,uint256,bytes)')) ^ * bytes4(keccak256('transferFromAndCall(address,address,uint256)')) ^ * bytes4(keccak256('transferFromAndCall(address,address,uint256,bytes)')) ^ * bytes4(keccak256('approveAndCall(address,uint256)')) ^ * bytes4(keccak256('approveAndCall(address,uint256,bytes)')) */ /** * @dev Moves a `value` amount of tokens from the caller's account to `to` * and then calls {IERC1363Receiver-onTransferReceived} on `to`. * @param to The address which you want to transfer to. * @param value The amount of tokens to be transferred. * @return A boolean value indicating whether the operation succeeded unless throwing. */ function transferAndCall(address to, uint256 value) external returns (bool); /** * @dev Moves a `value` amount of tokens from the caller's account to `to` * and then calls {IERC1363Receiver-onTransferReceived} on `to`. * @param to The address which you want to transfer to. * @param value The amount of tokens to be transferred. * @param data Additional data with no specified format, sent in call to `to`. * @return A boolean value indicating whether the operation succeeded unless throwing. */ function transferAndCall(address to, uint256 value, bytes calldata data) external returns (bool); /** * @dev Moves a `value` amount of tokens from `from` to `to` using the allowance mechanism * and then calls {IERC1363Receiver-onTransferReceived} on `to`. * @param from The address which you want to send tokens from. * @param to The address which you want to transfer to. * @param value The amount of tokens to be transferred. * @return A boolean value indicating whether the operation succeeded unless throwing. */ function transferFromAndCall(address from, address to, uint256 value) external returns (bool); /** * @dev Moves a `value` amount of tokens from `from` to `to` using the allowance mechanism * and then calls {IERC1363Receiver-onTransferReceived} on `to`. * @param from The address which you want to send tokens from. * @param to The address which you want to transfer to. * @param value The amount of tokens to be transferred. * @param data Additional data with no specified format, sent in call to `to`. * @return A boolean value indicating whether the operation succeeded unless throwing. */ function transferFromAndCall(address from, address to, uint256 value, bytes calldata data) external returns (bool); /** * @dev Sets a `value` amount of tokens as the allowance of `spender` over the * caller's tokens and then calls {IERC1363Spender-onApprovalReceived} on `spender`. * @param spender The address which will spend the funds. * @param value The amount of tokens to be spent. * @return A boolean value indicating whether the operation succeeded unless throwing. */ function approveAndCall(address spender, uint256 value) external returns (bool); /** * @dev Sets a `value` amount of tokens as the allowance of `spender` over the * caller's tokens and then calls {IERC1363Spender-onApprovalReceived} on `spender`. * @param spender The address which will spend the funds. * @param value The amount of tokens to be spent. * @param data Additional data with no specified format, sent in call to `spender`. * @return A boolean value indicating whether the operation succeeded unless throwing. */ function approveAndCall(address spender, uint256 value, bytes calldata data) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/Address.sol) pragma solidity ^0.8.20; import {Errors} from "./Errors.sol"; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev There's no code at `target` (it is not a contract). */ error AddressEmptyCode(address target); /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.20/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { if (address(this).balance < amount) { revert Errors.InsufficientBalance(address(this).balance, amount); } (bool success, ) = recipient.call{value: amount}(""); if (!success) { revert Errors.FailedCall(); } } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason or custom error, it is bubbled * up by this function (like regular Solidity function calls). However, if * the call reverted with no returned reason, this function reverts with a * {Errors.FailedCall} error. * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { if (address(this).balance < value) { revert Errors.InsufficientBalance(address(this).balance, value); } (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and reverts if the target * was not a contract or bubbling up the revert reason (falling back to {Errors.FailedCall}) in case * of an unsuccessful call. */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata ) internal view returns (bytes memory) { if (!success) { _revert(returndata); } else { // only check if target is a contract if the call was successful and the return data is empty // otherwise we already know that it was a contract if (returndata.length == 0 && target.code.length == 0) { revert AddressEmptyCode(target); } return returndata; } } /** * @dev Tool to verify that a low level call was successful, and reverts if it wasn't, either by bubbling the * revert reason or with a default {Errors.FailedCall} error. */ function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) { if (!success) { _revert(returndata); } else { return returndata; } } /** * @dev Reverts with returndata if present. Otherwise reverts with {Errors.FailedCall}. */ function _revert(bytes memory returndata) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly assembly ("memory-safe") { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert Errors.FailedCall(); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/structs/EnumerableSet.sol) // This file was procedurally generated from scripts/generate/templates/EnumerableSet.js. pragma solidity ^0.8.20; /** * @dev Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ```solidity * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure * unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an * array of EnumerableSet. * ==== */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position is the index of the value in the `values` array plus 1. // Position 0 is used to mean a value is not in the set. mapping(bytes32 value => uint256) _positions; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._positions[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We cache the value's position to prevent multiple reads from the same storage slot uint256 position = set._positions[value]; if (position != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 valueIndex = position - 1; uint256 lastIndex = set._values.length - 1; if (valueIndex != lastIndex) { bytes32 lastValue = set._values[lastIndex]; // Move the lastValue to the index where the value to delete is set._values[valueIndex] = lastValue; // Update the tracked position of the lastValue (that was just moved) set._positions[lastValue] = position; } // Delete the slot where the moved value was stored set._values.pop(); // Delete the tracked position for the deleted slot delete set._positions[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._positions[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { bytes32[] memory store = _values(set._inner); bytes32[] memory result; assembly ("memory-safe") { result := store } return result; } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; assembly ("memory-safe") { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values in the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; assembly ("memory-safe") { result := store } return result; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (token/ERC20/extensions/IERC20Metadata.sol) pragma solidity ^0.8.20; import {IERC20} from "../IERC20.sol"; /** * @dev Interface for the optional metadata functions from the ERC-20 standard. */ 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 v5.1.0) (token/ERC20/extensions/IERC20Permit.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC-20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[ERC-2612]. * * Adds the {permit} method, which can be used to change an account's ERC-20 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. * * ==== Security Considerations * * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be * considered as an intention to spend the allowance in any specific way. The second is that because permits have * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be * generally recommended is: * * ```solidity * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public { * try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {} * doThing(..., value); * } * * function doThing(..., uint256 value) public { * token.safeTransferFrom(msg.sender, address(this), value); * ... * } * ``` * * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also * {SafeERC20-safeTransferFrom}). * * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so * contracts should have entry points that don't rely on permit. */ 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]. * * CAUTION: See Security Considerations above. */ 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 (last updated v5.0.0) (utils/Pausable.sol) pragma solidity ^0.8.20; import {Context} from "../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 { bool private _paused; /** * @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); /** * @dev The operation failed because the contract is paused. */ error EnforcedPause(); /** * @dev The operation failed because the contract is not paused. */ error ExpectedPause(); /** * @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 { if (paused()) { revert EnforcedPause(); } } /** * @dev Throws if the contract is not paused. */ function _requirePaused() internal view virtual { if (!paused()) { revert ExpectedPause(); } } /** * @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 (last updated v5.0.0) (interfaces/IERC20.sol) pragma solidity ^0.8.20; import {IERC20} from "../token/ERC20/IERC20.sol";
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/IERC165.sol) pragma solidity ^0.8.20; import {IERC165} from "../utils/introspection/IERC165.sol";
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/Errors.sol) pragma solidity ^0.8.20; /** * @dev Collection of common custom errors used in multiple contracts * * IMPORTANT: Backwards compatibility is not guaranteed in future versions of the library. * It is recommended to avoid relying on the error API for critical functionality. * * _Available since v5.1._ */ library Errors { /** * @dev The ETH balance of the account is not enough to perform the operation. */ error InsufficientBalance(uint256 balance, uint256 needed); /** * @dev A call to an address target failed. The target may have reverted. */ error FailedCall(); /** * @dev The deployment failed. */ error FailedDeployment(); /** * @dev A necessary precompile is missing. */ error MissingPrecompile(address); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol) pragma solidity ^0.8.20; /** * @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; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.1.0) (utils/introspection/IERC165.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC-165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[ERC]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[ERC section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
{ "remappings": [ "@layerzerolabs/=node_modules/@layerzerolabs/", "@layerzerolabs/lz-evm-protocol-v2/=node_modules/@layerzerolabs/lz-evm-protocol-v2/", "@openzeppelin-contracts-upgradeable/=dependencies/@openzeppelin-contracts-upgradeable-5.1.0/", "@openzeppelin-contracts/contracts/=dependencies/@openzeppelin-contracts-5.1.0/", "@openzeppelin/contracts/=dependencies/@openzeppelin-contracts-5.1.0/", "erc4626-tests/=dependencies/erc4626-property-tests-1.0/", "forge-std/=dependencies/forge-std-1.9.4/src/", "permit2/=lib/permit2/", "@axelar-network/=node_modules/@axelar-network/", "@chainlink/=node_modules/@chainlink/", "@eth-optimism/=node_modules/@eth-optimism/", "@openzeppelin-3.4.2/=node_modules/@openzeppelin-3.4.2/", "@openzeppelin-contracts-5.1.0/=dependencies/@openzeppelin-contracts-5.1.0/", "@openzeppelin-contracts-upgradeable-5.1.0/=dependencies/@openzeppelin-contracts-upgradeable-5.1.0/", "@uniswap/=node_modules/@uniswap/", "base64-sol/=node_modules/base64-sol/", "erc4626-property-tests-1.0/=dependencies/erc4626-property-tests-1.0/", "eth-gas-reporter/=node_modules/eth-gas-reporter/", "forge-std-1.9.4/=dependencies/forge-std-1.9.4/src/", "hardhat-deploy/=node_modules/hardhat-deploy/", "hardhat/=node_modules/hardhat/", "solidity-bytes-utils/=node_modules/solidity-bytes-utils/", "solmate/=node_modules/solmate/" ], "optimizer": { "enabled": true, "runs": 333 }, "metadata": { "useLiteralContent": false, "bytecodeHash": "ipfs", "appendCBOR": true }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "evmVersion": "cancun", "viaIR": false, "libraries": { "contracts/libraries/VoterGovernanceActions.sol": { "VoterGovernanceActions": "0x1Db308F4d48C1d0D7CB7eBb047dD3F9b3B7ab70E" }, "contracts/libraries/VoterRewardClaimers.sol": { "VoterRewardClaimers": "0xF7Df8656F894C8b7dde25cFbCE55e323c84563d8" } } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"EMISSIONS_NOT_STARTED","type":"error"},{"inputs":[],"name":"FEE_SPLIT_WHEN_NO_GAUGE_IS_OFF","type":"error"},{"inputs":[],"name":"INVALID_CONTRACT","type":"error"},{"inputs":[],"name":"InvalidInitialization","type":"error"},{"inputs":[{"internalType":"address","name":"caller","type":"address"}],"name":"NOT_AUTHORIZED","type":"error"},{"inputs":[],"name":"NO_CHANGE","type":"error"},{"inputs":[],"name":"NotInitializing","type":"error"},{"inputs":[],"name":"SAME_PERIOD","type":"error"},{"inputs":[],"name":"STARTED","type":"error"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"SafeERC20FailedOperation","type":"error"},{"inputs":[],"name":"TOO_HIGH","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_emissionsMultiplier","type":"uint256"}],"name":"EmissionsMultiplierUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint64","name":"version","type":"uint64"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"weekly","type":"uint256"}],"name":"Mint","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_current","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_currentPeriod","type":"uint256"}],"name":"RebaseUnsuccessful","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_value","type":"address"}],"name":"SetVeDist","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_value","type":"address"}],"name":"SetVoter","type":"event"},{"inputs":[],"name":"DENOM","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"INITIAL_SUPPLY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_DEVIATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_SUPPLY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"accessHub","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"activePeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"calculateWeeklyEmissions","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"emissionsMultiplier","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"firstPeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getEpoch","outputs":[{"internalType":"uint256","name":"_epoch","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPeriod","outputs":[{"internalType":"uint256","name":"period","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_accessHub","type":"address"},{"internalType":"address","name":"_operator","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_shadow","type":"address"},{"internalType":"address","name":"_voter","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"_xShadow","type":"address"}],"name":"kickoff","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"lastMultiplierUpdate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"gauges","type":"address[]"}],"name":"migration","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"oldMinter","outputs":[{"internalType":"contract IMinter","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"oldVoter","outputs":[{"internalType":"contract IVoter","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"operator","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"pool","type":"address"}],"name":"postCreateLegacyGaugeHook","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"batchSize","type":"uint256"}],"name":"redirectFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"rescueTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_accessHub","type":"address"}],"name":"setAccessHub","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_operator","type":"address"}],"name":"setOperator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"shadow","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"startEmissions","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"syncSettings","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_emissionsMultiplier","type":"uint256"}],"name":"updateEmissionsMultiplier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"updatePeriod","outputs":[{"internalType":"uint256","name":"period","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"voteModule","outputs":[{"internalType":"contract IVoteModule","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"voter","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"weeklyEmissions","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"xShadow","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
6080604052348015600e575f5ffd5b5060156019565b60c9565b7ff0c57e16840df040f15088dc2f81fe391c3923bec73e23a9662efc9c229c6a00805468010000000000000000900460ff161560685760405163f92ee8a960e01b815260040160405180910390fd5b80546001600160401b039081161460c65780546001600160401b0319166001600160401b0390811782556040519081527fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d29060200160405180910390a15b50565b61288b806100d65f395ff3fe608060405234801561000f575f5ffd5b50600436106101e7575f3560e01c8063757991a811610109578063b3ab15fb1161009e578063e3e3fb2e1161006e578063e3e3fb2e1461049f578063e7589b39146104b2578063eb47023f146104cf578063f4c00811146104d7575f5ffd5b8063b3ab15fb1461043e578063b94ef2b814610451578063c4e3a63b14610465578063d47ffef11461048c575f5ffd5b8063ac600a3c116100d9578063ac600a3c146103c9578063ad58ae30146103f9578063b05f233a1461040d578063b380206d14610417575f5ffd5b8063757991a81461038b57806385caf28b146103935780639c3cf4d6146103ae578063a83627de146103c1575f5ffd5b8063485cc9551161017f57806360bc0f081161014f57806360bc0f081461034a5780636503afea1461035d5780636d6652ec1461036557806370ed141814610378575f5ffd5b8063485cc955146102cd578063570ca735146102e057806357376198146103105780635ac0881514610323575f5ffd5b80632ff2e9dc116101ba5780632ff2e9dc1461023557806332cb6b0c146102475780634256f5e71461025957806346c96aac1461029d575f5ffd5b80630a441f7b146101eb57806316343da4146102215780631b02f8451461022b5780631ed241951461022d575b5f5ffd5b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836003545b6040519081526020015b60405180910390f35b61020e620f424081565b005b61020e6104f2565b61020e6a027b46536c66c8e300000081565b61020e6a084595161401484a00000081565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836007546001600160a01b03165b6040516001600160a01b039091168152602001610218565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836008546001600160a01b0316610285565b61022b6102db366004612500565b610505565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836005546001600160a01b0316610285565b61022b61031e366004612537565b610666565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360045461020e565b61022b610358366004612561565b610686565b61020e6106e3565b61022b6103733660046125d2565b610869565b61022b6103863660046125f4565b61089f565b61020e610d60565b61028573dcb5a24ec708cc13cee12bfe6799a78a79b666b481565b61022b6103bc366004612614565b610d96565b61020e6112b5565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836009546001600160a01b0316610285565b5f5160206127f65f395f51905f525461020e565b61020e62030d4081565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360015461020e565b61022b61044c3660046125d2565b61180f565b6102855f5160206128165f395f51905f5281565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360025461020e565b61022b61049a36600461266f565b611858565b61022b6104ad3660046125d2565b6119af565b5f5160206128365f395f51905f52546001600160a01b0316610285565b61022b611cd5565b61028573c7022f359cd1bda8ab8a19d1f19d769cbf7f376581565b5f61050062093a804261269a565b905090565b7ff0c57e16840df040f15088dc2f81fe391c3923bec73e23a9662efc9c229c6a008054600160401b810460ff16159067ffffffffffffffff165f8115801561054a5750825b90505f8267ffffffffffffffff1660011480156105665750303b155b905081158015610574575080155b156105925760405163f92ee8a960e01b815260040160405180910390fd5b845467ffffffffffffffff1916600117855583156105bc57845460ff60401b1916600160401b1785555b5f5160206128365f395f51905f5280546001600160a01b038981166001600160a01b0319928316179092557fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836005805492891692909116919091179055831561065d57845460ff60401b19168555604051600181527fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d29060200160405180910390a15b50505050505050565b61066e611de9565b6106826001600160a01b0383163383611e33565b5050565b61068e611de9565b610696611e8a565b805f5b818110156106dd575f8484838181106106b4576106b46126b9565b90506020020160208101906106c991906125d2565b90506106d481611ff6565b50600101610699565b50505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836001545f5160206127f65f395f51905f5280545f928391620f424091610727916126cd565b610731919061269a565b9050805f03610742575f9250505090565b6a084595161401484a00000081836009015f9054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b8152600401602060405180830381865afa1580156107a1573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906107c591906126e4565b6107cf91906126fb565b111561086357816009015f9054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610827573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061084b91906126e4565b610860906a084595161401484a00000061270e565b90505b92915050565b610871611de9565b5f5160206128365f395f51905f5280546001600160a01b0319166001600160a01b0392909216919091179055565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836008545f5160206128365f395f51905f5254604080516361d027b360e01b815290515f5160206127f65f395f51905f52936001600160a01b03908116935f939116916361d027b3916004808201926020929091908290030181865afa158015610929573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061094d9190612721565b90505f826001600160a01b031663ce883cdb6040518163ffffffff1660e01b8152600401602060405180830381865afa15801561098c573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906109b091906126e4565b90505f836001600160a01b031663d32af6c16040518163ffffffff1660e01b8152600401602060405180830381865afa1580156109ef573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610a139190612721565b90505f610a2087896126fb565b905082811115610a2d5750815b875b81811015610d555760405163afaa701f60e01b8152600481018290525f906001600160a01b0388169063afaa701f90602401602060405180830381865afa158015610a7c573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610aa09190612721565b60405163b616eef360e01b81526001600160a01b0380831660048301529192509088169063b616eef390602401602060405180830381865afa158015610ae8573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610b0c919061273c565b15610d4b5760405163036b50d960e11b81526001600160a01b0382811660048301525f91908916906306d6a1b290602401602060405180830381865afa158015610b58573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610b7c9190612721565b6040516370a0823160e01b81523060048201529091505f906001600160a01b038316906370a0823190602401602060405180830381865afa158015610bc3573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610be791906126e4565b90508015610d4857604051631703e5f960e01b81526001600160a01b0384811660048301528a1690631703e5f990602401602060405180830381865afa158015610c33573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610c57919061273c565b610c7757610c6f6001600160a01b0383168983611e33565b505050610d4d565b6040516308a3a6ad60e11b81526001600160a01b0383811660048301525f91908816906311474d5a90602401602060405180830381865afa158015610cbe573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610ce29190612721565b9050610cf86001600160a01b0384168284611e33565b806001600160a01b0316634c4f2a956040518163ffffffff1660e01b81526004015f604051808303815f87803b158015610d30575f5ffd5b505af1158015610d42573d5f5f3e3d5ffd5b50505050505b50505b505b600101610a2f565b505050505050505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836002545f90610d8c6104f2565b610500919061270e565b610d9e611de9565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836009545f5160206127f65f395f51905f52906001600160a01b031615610df657604051631be35a5960e01b815260040160405180910390fd5b6001600160a01b038216610e1d576040516366c87bfd60e11b815260040160405180910390fd5b6001600160a01b038516610e44576040516366c87bfd60e11b815260040160405180910390fd5b6001600160a01b038616610e6b576040516366c87bfd60e11b815260040160405180910390fd5b6009810180546001600160a01b038089166001600160a01b03199283161790925560078301805485841690831617905560088301805492881692909116821790556040805163e7589b3960e01b8152905163e7589b39916004808201926020929091908290030181865afa158015610ee5573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610f099190612721565b816006015f6101000a8154816001600160a01b0302191690836001600160a01b0316021790555073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663ad58ae306040518163ffffffff1660e01b8152600401602060405180830381865afa158015610f80573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610fa491906126e4565b81556040805163b380206d60e01b8152905173c7022f359cd1bda8ab8a19d1f19d769cbf7f37659163b380206d9160048083019260209291908290030181865afa158015610ff4573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061101891906126e4565b6110239060646126cd565b60018201556110306104f2565b600382018190556110439060019061270e565b816004018190555073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663c4e3a63b6040518163ffffffff1660e01b8152600401602060405180830381865afa15801561109b573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906110bf91906126e4565b60028201556040516001600160a01b03861681527fc6ff127433b785c51da9ae4088ee184c909b1a55b9afd82ae6c64224d3bc15d29060200160405180910390a160405163095ea7b360e01b81526001600160a01b0383811660048301526103e8602483015287169063095ea7b3906044016020604051808303815f875af115801561114d573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611171919061273c565b506040516304ba099d60e21b81526103e860048201526001600160a01b038316906312e82674906024015f604051808303815f87803b1580156111b2575f5ffd5b505af11580156111c4573d5f5f3e3d5ffd5b505060405163095ea7b360e01b815273dcb5a24ec708cc13cee12bfe6799a78a79b666b460048201526103e860248201526001600160a01b038516925063095ea7b391506044016020604051808303815f875af1158015611227573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061124b919061273c565b5060405163b6b55f2560e01b81526103e8600482015273dcb5a24ec708cc13cee12bfe6799a78a79b666b49063b6b55f25906024015f604051808303815f87803b158015611297575f5ffd5b505af11580156112a9573d5f5f3e3d5ffd5b50505050505050505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836002545f905f5160206127f65f395f51905f52908203611307576040516351ff71f960e11b815260040160405180910390fd5b80600301549150816113176104f2565b111561180b576113256104f2565b6003820181905591505f6113376106e3565b905080825f018190555073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663a83627de6040518163ffffffff1660e01b81526004016020604051808303815f875af1158015611392573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906113b691906126e4565b505f306001600160a01b03165f5160206128165f395f51905f526001600160a01b0316630c340a246040518163ffffffff1660e01b8152600401602060405180830381865afa15801561140b573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061142f9190612721565b6001600160a01b031614905080156114b557604051635824780d60e01b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b6004820152602481018590525f5160206128165f395f51905f5290635824780d906044015f604051808303815f87803b15801561149e575f5ffd5b505af11580156114b0573d5f5f3e3d5ffd5b505050505b801561152857604051639f06247b60e01b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b60048201525f5160206128165f395f51905f5290639f06247b906024015f604051808303815f87803b158015611511575f5ffd5b505af1158015611523573d5f5f3e3d5ffd5b505050505b60405163036b50d960e11b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b60048201525f905f5160206128165f395f51905f52906306d6a1b290602401602060405180830381865afa158015611584573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906115a89190612721565b6040805160018082528183019092529192505f91906020808301908036833750506040805160018082528183019092529293505f9291506020808301908036833701905050905082825f81518110611602576116026126b9565b60200260200101906001600160a01b031690816001600160a01b031681525050670de0b6b3a7640000815f8151811061163d5761163d6126b9565b602090810291909101015260405163e4ff5c2f60e01b81525f5160206128165f395f51905f529063e4ff5c2f9061167c9030908690869060040161275b565b5f604051808303815f87803b158015611693575f5ffd5b505af11580156116a5573d5f5f3e3d5ffd5b50505050831561171c5760405163992a793360e01b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b60048201525f5160206128165f395f51905f529063992a7933906024015f604051808303815f87803b158015611705575f5ffd5b505af1158015611717573d5f5f3e3d5ffd5b505050505b8554158015906117295750835b15611805576009860154600887015460405163095ea7b360e01b81526001600160a01b0391821660048201526024810188905291169063095ea7b3906044016020604051808303815f875af1158015611784573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906117a8919061273c565b506008860154604051633c6b16ab60e01b8152600481018790526001600160a01b0390911690633c6b16ab906024015f604051808303815f87803b1580156117ee575f5ffd5b505af1158015611800573d5f5f3e3d5ffd5b505050505b50505050505b5090565b611817611de9565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e858783600580546001600160a01b0319166001600160a01b0392909216919091179055565b611860611de9565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836003547fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836004545f5160206127f65f395f51905f529190036118d25760405163b91f1fed60e01b815260040160405180910390fd5b60038101546004820155600181015482900361191a576040517fff302aa000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5f8282600101541161193a576001820154611935908461270e565b61194a565b82826001015461194a919061270e565b905062030d4081111561197057604051631617020760e21b815260040160405180910390fd5b600182018390556040518381527f846522f798efef551b17983f390974e426a0719438ac1cb4ada7c3519950ef319060200160405180910390a1505050565b6119b7612428565b6119bf611e8a565b306001600160a01b03165f5160206128165f395f51905f526001600160a01b0316630c340a246040518163ffffffff1660e01b8152600401602060405180830381865afa158015611a12573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611a369190612721565b6001600160a01b031603611cd2575f816001600160a01b0316630dfe16816040518163ffffffff1660e01b8152600401602060405180830381865afa158015611a81573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611aa59190612721565b90505f826001600160a01b031663d21220a76040518163ffffffff1660e01b8152600401602060405180830381865afa158015611ae4573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611b089190612721565b604051634d8c928d60e11b81526001600160a01b03841660048201529091505f5160206128165f395f51905f5290639b19251a906024015f604051808303815f87803b158015611b56575f5ffd5b505af1158015611b68573d5f5f3e3d5ffd5b5050604051634d8c928d60e11b81526001600160a01b03841660048201525f5160206128165f395f51905f529250639b19251a91506024015f604051808303815f87803b158015611bb7575f5ffd5b505af1158015611bc9573d5f5f3e3d5ffd5b50506040516302045be960e41b81526001600160a01b03861660048201525f92505f5160206128165f395f51905f529150632045be9090602401602060405180830381865afa158015611c1e573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611c429190612721565b90506001600160a01b038116611cc9576040516352fa180f60e11b81526001600160a01b03851660048201525f5160206128165f395f51905f529063a5f4301e906024016020604051808303815f875af1158015611ca2573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611cc69190612721565b90505b6106dd81611ff6565b50565b611cdd611de9565b5f5f5160206127f65f395f51905f52905073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663ad58ae306040518163ffffffff1660e01b8152600401602060405180830381865afa158015611d3e573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611d6291906126e4565b81556040805163b380206d60e01b8152905173c7022f359cd1bda8ab8a19d1f19d769cbf7f37659163b380206d9160048083019260209291908290030181865afa158015611db2573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611dd691906126e4565b611de19060646126cd565b600190910155565b5f5160206128365f395f51905f525433906001600160a01b03168114611cd257604051632bc10c3360e01b81526001600160a01b0390911660048201526024015b60405180910390fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611e85908490612480565b505050565b5f5f5160206128165f395f51905f526001600160a01b0316632da5347b6040518163ffffffff1660e01b8152600401602060405180830381865afa158015611ed4573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611ef89190612721565b9050806001600160a01b03166383b274f26040518163ffffffff1660e01b8152600401602060405180830381865afa158015611f36573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611f5a919061273c565b611f7757604051631a3052f760e01b815260040160405180910390fd5b5f816001600160a01b0316636373ea696040518163ffffffff1660e01b8152600401602060405180830381865afa158015611fb4573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611fd891906126e4565b11611cd257604051631a3052f760e01b815260040160405180910390fd5b60405163036b50d960e11b81526001600160a01b03821660048201525f905f5160206128165f395f51905f52906306d6a1b290602401602060405180830381865afa158015612047573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061206b9190612721565b90505f816001600160a01b0316630dfe16816040518163ffffffff1660e01b8152600401602060405180830381865afa1580156120aa573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906120ce9190612721565b90505f826001600160a01b031663d21220a76040518163ffffffff1660e01b8152600401602060405180830381865afa15801561210d573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906121319190612721565b604051633af32abf60e01b81526001600160a01b03841660048201529091505f5160206128165f395f51905f5290633af32abf90602401602060405180830381865afa158015612183573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906121a7919061273c565b1561220e57604051639c7f331560e01b81526001600160a01b03831660048201525f5160206128165f395f51905f5290639c7f3315906024015f604051808303815f87803b1580156121f7575f5ffd5b505af1158015612209573d5f5f3e3d5ffd5b505050505b604051633af32abf60e01b81526001600160a01b03821660048201525f5160206128165f395f51905f5290633af32abf90602401602060405180830381865afa15801561225d573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190612281919061273c565b156122e857604051639c7f331560e01b81526001600160a01b03821660048201525f5160206128165f395f51905f5290639c7f3315906024015f604051808303815f87803b1580156122d1575f5ffd5b505af11580156122e3573d5f5f3e3d5ffd5b505050505b604051631703e5f960e01b81526001600160a01b03851660048201525f5160206128165f395f51905f5290631703e5f990602401602060405180830381865afa158015612337573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061235b919061273c565b6123c157604051639f06247b60e01b81526001600160a01b03851660048201525f5160206128165f395f51905f5290639f06247b906024015f604051808303815f87803b1580156123aa575f5ffd5b505af11580156123bc573d5f5f3e3d5ffd5b505050505b60405163992a793360e01b81526001600160a01b03851660048201525f5160206128165f395f51905f529063992a7933906024015f604051808303815f87803b15801561240c575f5ffd5b505af115801561241e573d5f5f3e3d5ffd5b5050505050505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360085433906001600160a01b03168114611cd257604051632bc10c3360e01b81526001600160a01b039091166004820152602401611e2a565b5f5f60205f8451602086015f885af18061249f576040513d5f823e3d81fd5b50505f513d915081156124b65780600114156124c3565b6001600160a01b0384163b155b156106dd57604051635274afe760e01b81526001600160a01b0385166004820152602401611e2a565b6001600160a01b0381168114611cd2575f5ffd5b5f5f60408385031215612511575f5ffd5b823561251c816124ec565b9150602083013561252c816124ec565b809150509250929050565b5f5f60408385031215612548575f5ffd5b8235612553816124ec565b946020939093013593505050565b5f5f60208385031215612572575f5ffd5b823567ffffffffffffffff811115612588575f5ffd5b8301601f81018513612598575f5ffd5b803567ffffffffffffffff8111156125ae575f5ffd5b8560208260051b84010111156125c2575f5ffd5b6020919091019590945092505050565b5f602082840312156125e2575f5ffd5b81356125ed816124ec565b9392505050565b5f5f60408385031215612605575f5ffd5b50508035926020909101359150565b5f5f5f5f5f60a08688031215612628575f5ffd5b8535612633816124ec565b94506020860135612643816124ec565b935060408601359250606086013591506080860135612661816124ec565b809150509295509295909350565b5f6020828403121561267f575f5ffd5b5035919050565b634e487b7160e01b5f52601160045260245ffd5b5f826126b457634e487b7160e01b5f52601260045260245ffd5b500490565b634e487b7160e01b5f52603260045260245ffd5b808202811582820484141761086357610863612686565b5f602082840312156126f4575f5ffd5b5051919050565b8082018082111561086357610863612686565b8181038181111561086357610863612686565b5f60208284031215612731575f5ffd5b81516125ed816124ec565b5f6020828403121561274c575f5ffd5b815180151581146125ed575f5ffd5b5f606082016001600160a01b0386168352606060208401528085518083526080850191506020870192505f5b818110156127ae5783516001600160a01b0316835260209384019390920191600101612787565b50508381036040850152845180825260209182019250908501905f5b818110156127e85782518452602093840193909201916001016127ca565b509197965050505050505056fee6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360000000000000000000000000003af1dd7a2755201f8e2d6dcda1a61d9f54838f4fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836006a26469706673582212202b471f79034e66c930bd2100d5f307b3a07716a8cd8a5e4864aa360dfb88b7d564736f6c634300081c0033
Deployed Bytecode
0x608060405234801561000f575f5ffd5b50600436106101e7575f3560e01c8063757991a811610109578063b3ab15fb1161009e578063e3e3fb2e1161006e578063e3e3fb2e1461049f578063e7589b39146104b2578063eb47023f146104cf578063f4c00811146104d7575f5ffd5b8063b3ab15fb1461043e578063b94ef2b814610451578063c4e3a63b14610465578063d47ffef11461048c575f5ffd5b8063ac600a3c116100d9578063ac600a3c146103c9578063ad58ae30146103f9578063b05f233a1461040d578063b380206d14610417575f5ffd5b8063757991a81461038b57806385caf28b146103935780639c3cf4d6146103ae578063a83627de146103c1575f5ffd5b8063485cc9551161017f57806360bc0f081161014f57806360bc0f081461034a5780636503afea1461035d5780636d6652ec1461036557806370ed141814610378575f5ffd5b8063485cc955146102cd578063570ca735146102e057806357376198146103105780635ac0881514610323575f5ffd5b80632ff2e9dc116101ba5780632ff2e9dc1461023557806332cb6b0c146102475780634256f5e71461025957806346c96aac1461029d575f5ffd5b80630a441f7b146101eb57806316343da4146102215780631b02f8451461022b5780631ed241951461022d575b5f5ffd5b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836003545b6040519081526020015b60405180910390f35b61020e620f424081565b005b61020e6104f2565b61020e6a027b46536c66c8e300000081565b61020e6a084595161401484a00000081565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836007546001600160a01b03165b6040516001600160a01b039091168152602001610218565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836008546001600160a01b0316610285565b61022b6102db366004612500565b610505565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836005546001600160a01b0316610285565b61022b61031e366004612537565b610666565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360045461020e565b61022b610358366004612561565b610686565b61020e6106e3565b61022b6103733660046125d2565b610869565b61022b6103863660046125f4565b61089f565b61020e610d60565b61028573dcb5a24ec708cc13cee12bfe6799a78a79b666b481565b61022b6103bc366004612614565b610d96565b61020e6112b5565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836009546001600160a01b0316610285565b5f5160206127f65f395f51905f525461020e565b61020e62030d4081565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360015461020e565b61022b61044c3660046125d2565b61180f565b6102855f5160206128165f395f51905f5281565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360025461020e565b61022b61049a36600461266f565b611858565b61022b6104ad3660046125d2565b6119af565b5f5160206128365f395f51905f52546001600160a01b0316610285565b61022b611cd5565b61028573c7022f359cd1bda8ab8a19d1f19d769cbf7f376581565b5f61050062093a804261269a565b905090565b7ff0c57e16840df040f15088dc2f81fe391c3923bec73e23a9662efc9c229c6a008054600160401b810460ff16159067ffffffffffffffff165f8115801561054a5750825b90505f8267ffffffffffffffff1660011480156105665750303b155b905081158015610574575080155b156105925760405163f92ee8a960e01b815260040160405180910390fd5b845467ffffffffffffffff1916600117855583156105bc57845460ff60401b1916600160401b1785555b5f5160206128365f395f51905f5280546001600160a01b038981166001600160a01b0319928316179092557fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836005805492891692909116919091179055831561065d57845460ff60401b19168555604051600181527fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d29060200160405180910390a15b50505050505050565b61066e611de9565b6106826001600160a01b0383163383611e33565b5050565b61068e611de9565b610696611e8a565b805f5b818110156106dd575f8484838181106106b4576106b46126b9565b90506020020160208101906106c991906125d2565b90506106d481611ff6565b50600101610699565b50505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836001545f5160206127f65f395f51905f5280545f928391620f424091610727916126cd565b610731919061269a565b9050805f03610742575f9250505090565b6a084595161401484a00000081836009015f9054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b8152600401602060405180830381865afa1580156107a1573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906107c591906126e4565b6107cf91906126fb565b111561086357816009015f9054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610827573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061084b91906126e4565b610860906a084595161401484a00000061270e565b90505b92915050565b610871611de9565b5f5160206128365f395f51905f5280546001600160a01b0319166001600160a01b0392909216919091179055565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836008545f5160206128365f395f51905f5254604080516361d027b360e01b815290515f5160206127f65f395f51905f52936001600160a01b03908116935f939116916361d027b3916004808201926020929091908290030181865afa158015610929573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061094d9190612721565b90505f826001600160a01b031663ce883cdb6040518163ffffffff1660e01b8152600401602060405180830381865afa15801561098c573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906109b091906126e4565b90505f836001600160a01b031663d32af6c16040518163ffffffff1660e01b8152600401602060405180830381865afa1580156109ef573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610a139190612721565b90505f610a2087896126fb565b905082811115610a2d5750815b875b81811015610d555760405163afaa701f60e01b8152600481018290525f906001600160a01b0388169063afaa701f90602401602060405180830381865afa158015610a7c573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610aa09190612721565b60405163b616eef360e01b81526001600160a01b0380831660048301529192509088169063b616eef390602401602060405180830381865afa158015610ae8573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610b0c919061273c565b15610d4b5760405163036b50d960e11b81526001600160a01b0382811660048301525f91908916906306d6a1b290602401602060405180830381865afa158015610b58573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610b7c9190612721565b6040516370a0823160e01b81523060048201529091505f906001600160a01b038316906370a0823190602401602060405180830381865afa158015610bc3573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610be791906126e4565b90508015610d4857604051631703e5f960e01b81526001600160a01b0384811660048301528a1690631703e5f990602401602060405180830381865afa158015610c33573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610c57919061273c565b610c7757610c6f6001600160a01b0383168983611e33565b505050610d4d565b6040516308a3a6ad60e11b81526001600160a01b0383811660048301525f91908816906311474d5a90602401602060405180830381865afa158015610cbe573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610ce29190612721565b9050610cf86001600160a01b0384168284611e33565b806001600160a01b0316634c4f2a956040518163ffffffff1660e01b81526004015f604051808303815f87803b158015610d30575f5ffd5b505af1158015610d42573d5f5f3e3d5ffd5b50505050505b50505b505b600101610a2f565b505050505050505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836002545f90610d8c6104f2565b610500919061270e565b610d9e611de9565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836009545f5160206127f65f395f51905f52906001600160a01b031615610df657604051631be35a5960e01b815260040160405180910390fd5b6001600160a01b038216610e1d576040516366c87bfd60e11b815260040160405180910390fd5b6001600160a01b038516610e44576040516366c87bfd60e11b815260040160405180910390fd5b6001600160a01b038616610e6b576040516366c87bfd60e11b815260040160405180910390fd5b6009810180546001600160a01b038089166001600160a01b03199283161790925560078301805485841690831617905560088301805492881692909116821790556040805163e7589b3960e01b8152905163e7589b39916004808201926020929091908290030181865afa158015610ee5573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610f099190612721565b816006015f6101000a8154816001600160a01b0302191690836001600160a01b0316021790555073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663ad58ae306040518163ffffffff1660e01b8152600401602060405180830381865afa158015610f80573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190610fa491906126e4565b81556040805163b380206d60e01b8152905173c7022f359cd1bda8ab8a19d1f19d769cbf7f37659163b380206d9160048083019260209291908290030181865afa158015610ff4573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061101891906126e4565b6110239060646126cd565b60018201556110306104f2565b600382018190556110439060019061270e565b816004018190555073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663c4e3a63b6040518163ffffffff1660e01b8152600401602060405180830381865afa15801561109b573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906110bf91906126e4565b60028201556040516001600160a01b03861681527fc6ff127433b785c51da9ae4088ee184c909b1a55b9afd82ae6c64224d3bc15d29060200160405180910390a160405163095ea7b360e01b81526001600160a01b0383811660048301526103e8602483015287169063095ea7b3906044016020604051808303815f875af115801561114d573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611171919061273c565b506040516304ba099d60e21b81526103e860048201526001600160a01b038316906312e82674906024015f604051808303815f87803b1580156111b2575f5ffd5b505af11580156111c4573d5f5f3e3d5ffd5b505060405163095ea7b360e01b815273dcb5a24ec708cc13cee12bfe6799a78a79b666b460048201526103e860248201526001600160a01b038516925063095ea7b391506044016020604051808303815f875af1158015611227573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061124b919061273c565b5060405163b6b55f2560e01b81526103e8600482015273dcb5a24ec708cc13cee12bfe6799a78a79b666b49063b6b55f25906024015f604051808303815f87803b158015611297575f5ffd5b505af11580156112a9573d5f5f3e3d5ffd5b50505050505050505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836002545f905f5160206127f65f395f51905f52908203611307576040516351ff71f960e11b815260040160405180910390fd5b80600301549150816113176104f2565b111561180b576113256104f2565b6003820181905591505f6113376106e3565b905080825f018190555073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663a83627de6040518163ffffffff1660e01b81526004016020604051808303815f875af1158015611392573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906113b691906126e4565b505f306001600160a01b03165f5160206128165f395f51905f526001600160a01b0316630c340a246040518163ffffffff1660e01b8152600401602060405180830381865afa15801561140b573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061142f9190612721565b6001600160a01b031614905080156114b557604051635824780d60e01b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b6004820152602481018590525f5160206128165f395f51905f5290635824780d906044015f604051808303815f87803b15801561149e575f5ffd5b505af11580156114b0573d5f5f3e3d5ffd5b505050505b801561152857604051639f06247b60e01b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b60048201525f5160206128165f395f51905f5290639f06247b906024015f604051808303815f87803b158015611511575f5ffd5b505af1158015611523573d5f5f3e3d5ffd5b505050505b60405163036b50d960e11b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b60048201525f905f5160206128165f395f51905f52906306d6a1b290602401602060405180830381865afa158015611584573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906115a89190612721565b6040805160018082528183019092529192505f91906020808301908036833750506040805160018082528183019092529293505f9291506020808301908036833701905050905082825f81518110611602576116026126b9565b60200260200101906001600160a01b031690816001600160a01b031681525050670de0b6b3a7640000815f8151811061163d5761163d6126b9565b602090810291909101015260405163e4ff5c2f60e01b81525f5160206128165f395f51905f529063e4ff5c2f9061167c9030908690869060040161275b565b5f604051808303815f87803b158015611693575f5ffd5b505af11580156116a5573d5f5f3e3d5ffd5b50505050831561171c5760405163992a793360e01b815273cbcad939e2bbbe01850a141f204f25df63b8fc5b60048201525f5160206128165f395f51905f529063992a7933906024015f604051808303815f87803b158015611705575f5ffd5b505af1158015611717573d5f5f3e3d5ffd5b505050505b8554158015906117295750835b15611805576009860154600887015460405163095ea7b360e01b81526001600160a01b0391821660048201526024810188905291169063095ea7b3906044016020604051808303815f875af1158015611784573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906117a8919061273c565b506008860154604051633c6b16ab60e01b8152600481018790526001600160a01b0390911690633c6b16ab906024015f604051808303815f87803b1580156117ee575f5ffd5b505af1158015611800573d5f5f3e3d5ffd5b505050505b50505050505b5090565b611817611de9565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e858783600580546001600160a01b0319166001600160a01b0392909216919091179055565b611860611de9565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836003547fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836004545f5160206127f65f395f51905f529190036118d25760405163b91f1fed60e01b815260040160405180910390fd5b60038101546004820155600181015482900361191a576040517fff302aa000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5f8282600101541161193a576001820154611935908461270e565b61194a565b82826001015461194a919061270e565b905062030d4081111561197057604051631617020760e21b815260040160405180910390fd5b600182018390556040518381527f846522f798efef551b17983f390974e426a0719438ac1cb4ada7c3519950ef319060200160405180910390a1505050565b6119b7612428565b6119bf611e8a565b306001600160a01b03165f5160206128165f395f51905f526001600160a01b0316630c340a246040518163ffffffff1660e01b8152600401602060405180830381865afa158015611a12573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611a369190612721565b6001600160a01b031603611cd2575f816001600160a01b0316630dfe16816040518163ffffffff1660e01b8152600401602060405180830381865afa158015611a81573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611aa59190612721565b90505f826001600160a01b031663d21220a76040518163ffffffff1660e01b8152600401602060405180830381865afa158015611ae4573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611b089190612721565b604051634d8c928d60e11b81526001600160a01b03841660048201529091505f5160206128165f395f51905f5290639b19251a906024015f604051808303815f87803b158015611b56575f5ffd5b505af1158015611b68573d5f5f3e3d5ffd5b5050604051634d8c928d60e11b81526001600160a01b03841660048201525f5160206128165f395f51905f529250639b19251a91506024015f604051808303815f87803b158015611bb7575f5ffd5b505af1158015611bc9573d5f5f3e3d5ffd5b50506040516302045be960e41b81526001600160a01b03861660048201525f92505f5160206128165f395f51905f529150632045be9090602401602060405180830381865afa158015611c1e573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611c429190612721565b90506001600160a01b038116611cc9576040516352fa180f60e11b81526001600160a01b03851660048201525f5160206128165f395f51905f529063a5f4301e906024016020604051808303815f875af1158015611ca2573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611cc69190612721565b90505b6106dd81611ff6565b50565b611cdd611de9565b5f5f5160206127f65f395f51905f52905073c7022f359cd1bda8ab8a19d1f19d769cbf7f37656001600160a01b031663ad58ae306040518163ffffffff1660e01b8152600401602060405180830381865afa158015611d3e573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611d6291906126e4565b81556040805163b380206d60e01b8152905173c7022f359cd1bda8ab8a19d1f19d769cbf7f37659163b380206d9160048083019260209291908290030181865afa158015611db2573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611dd691906126e4565b611de19060646126cd565b600190910155565b5f5160206128365f395f51905f525433906001600160a01b03168114611cd257604051632bc10c3360e01b81526001600160a01b0390911660048201526024015b60405180910390fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611e85908490612480565b505050565b5f5f5160206128165f395f51905f526001600160a01b0316632da5347b6040518163ffffffff1660e01b8152600401602060405180830381865afa158015611ed4573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611ef89190612721565b9050806001600160a01b03166383b274f26040518163ffffffff1660e01b8152600401602060405180830381865afa158015611f36573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611f5a919061273c565b611f7757604051631a3052f760e01b815260040160405180910390fd5b5f816001600160a01b0316636373ea696040518163ffffffff1660e01b8152600401602060405180830381865afa158015611fb4573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190611fd891906126e4565b11611cd257604051631a3052f760e01b815260040160405180910390fd5b60405163036b50d960e11b81526001600160a01b03821660048201525f905f5160206128165f395f51905f52906306d6a1b290602401602060405180830381865afa158015612047573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061206b9190612721565b90505f816001600160a01b0316630dfe16816040518163ffffffff1660e01b8152600401602060405180830381865afa1580156120aa573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906120ce9190612721565b90505f826001600160a01b031663d21220a76040518163ffffffff1660e01b8152600401602060405180830381865afa15801561210d573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906121319190612721565b604051633af32abf60e01b81526001600160a01b03841660048201529091505f5160206128165f395f51905f5290633af32abf90602401602060405180830381865afa158015612183573d5f5f3e3d5ffd5b505050506040513d601f19601f820116820180604052508101906121a7919061273c565b1561220e57604051639c7f331560e01b81526001600160a01b03831660048201525f5160206128165f395f51905f5290639c7f3315906024015f604051808303815f87803b1580156121f7575f5ffd5b505af1158015612209573d5f5f3e3d5ffd5b505050505b604051633af32abf60e01b81526001600160a01b03821660048201525f5160206128165f395f51905f5290633af32abf90602401602060405180830381865afa15801561225d573d5f5f3e3d5ffd5b505050506040513d601f19601f82011682018060405250810190612281919061273c565b156122e857604051639c7f331560e01b81526001600160a01b03821660048201525f5160206128165f395f51905f5290639c7f3315906024015f604051808303815f87803b1580156122d1575f5ffd5b505af11580156122e3573d5f5f3e3d5ffd5b505050505b604051631703e5f960e01b81526001600160a01b03851660048201525f5160206128165f395f51905f5290631703e5f990602401602060405180830381865afa158015612337573d5f5f3e3d5ffd5b505050506040513d601f19601f8201168201806040525081019061235b919061273c565b6123c157604051639f06247b60e01b81526001600160a01b03851660048201525f5160206128165f395f51905f5290639f06247b906024015f604051808303815f87803b1580156123aa575f5ffd5b505af11580156123bc573d5f5f3e3d5ffd5b505050505b60405163992a793360e01b81526001600160a01b03851660048201525f5160206128165f395f51905f529063992a7933906024015f604051808303815f87803b15801561240c575f5ffd5b505af115801561241e573d5f5f3e3d5ffd5b5050505050505050565b7fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360085433906001600160a01b03168114611cd257604051632bc10c3360e01b81526001600160a01b039091166004820152602401611e2a565b5f5f60205f8451602086015f885af18061249f576040513d5f823e3d81fd5b50505f513d915081156124b65780600114156124c3565b6001600160a01b0384163b155b156106dd57604051635274afe760e01b81526001600160a01b0385166004820152602401611e2a565b6001600160a01b0381168114611cd2575f5ffd5b5f5f60408385031215612511575f5ffd5b823561251c816124ec565b9150602083013561252c816124ec565b809150509250929050565b5f5f60408385031215612548575f5ffd5b8235612553816124ec565b946020939093013593505050565b5f5f60208385031215612572575f5ffd5b823567ffffffffffffffff811115612588575f5ffd5b8301601f81018513612598575f5ffd5b803567ffffffffffffffff8111156125ae575f5ffd5b8560208260051b84010111156125c2575f5ffd5b6020919091019590945092505050565b5f602082840312156125e2575f5ffd5b81356125ed816124ec565b9392505050565b5f5f60408385031215612605575f5ffd5b50508035926020909101359150565b5f5f5f5f5f60a08688031215612628575f5ffd5b8535612633816124ec565b94506020860135612643816124ec565b935060408601359250606086013591506080860135612661816124ec565b809150509295509295909350565b5f6020828403121561267f575f5ffd5b5035919050565b634e487b7160e01b5f52601160045260245ffd5b5f826126b457634e487b7160e01b5f52601260045260245ffd5b500490565b634e487b7160e01b5f52603260045260245ffd5b808202811582820484141761086357610863612686565b5f602082840312156126f4575f5ffd5b5051919050565b8082018082111561086357610863612686565b8181038181111561086357610863612686565b5f60208284031215612731575f5ffd5b81516125ed816124ec565b5f6020828403121561274c575f5ffd5b815180151581146125ed575f5ffd5b5f606082016001600160a01b0386168352606060208401528085518083526080850191506020870192505f5b818110156127ae5783516001600160a01b0316835260209384019390920191600101612787565b50508381036040850152845180825260209182019250908501905f5b818110156127e85782518452602093840193909201916001016127ca565b509197965050505050505056fee6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e85878360000000000000000000000000003af1dd7a2755201f8e2d6dcda1a61d9f54838f4fe6b9c2ebce602e282d13679d7c45261a3c7ffe195da42522e8e70e8587836006a26469706673582212202b471f79034e66c930bd2100d5f307b3a07716a8cd8a5e4864aa360dfb88b7d564736f6c634300081c0033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 31 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
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.