- Contract name:
- MixedRouteQuoterV1
- Optimization enabled
- true
- Compiler version
- v0.7.6+commit.7338295f
- Optimization runs
- 10
- Verified at
- 2023-10-15T17:43:07.963318Z
Constructor Arguments
0000000000000000000000007342530a1a1b39df96755504e236525395cc84a50000000000000000000000009c3afddea87a726891a44c037242393d524cacfe0000000000000000000000008e5dff1c121f661971d02950698f8c5efc3dfa780000000000000000000000000000000000000000000000000000000000000000000000000000000000000000730253eee681a77435a68c7f0c3f23abed458858
Arg [0] (address) : 0x7342530a1a1b39df96755504e236525395cc84a5
Arg [1] (address) : 0x9c3afddea87a726891a44c037242393d524cacfe
Arg [2] (address) : 0x8e5dff1c121f661971d02950698f8c5efc3dfa78
Arg [3] (address) : 0x0000000000000000000000000000000000000000
Arg [4] (address) : 0x730253eee681a77435a68c7f0c3f23abed458858
contracts/lens/MixedRouteQuoterV1.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity =0.7.6;pragma abicoder v2;import '@donaswap/v3-periphery/contracts/base/PeripheryImmutableState.sol';import '@donaswap/v3-core/contracts/libraries/SafeCast.sol';import '@donaswap/v3-core/contracts/libraries/TickMath.sol';import '@donaswap/v3-core/contracts/libraries/TickBitmap.sol';import '@donaswap/v3-core/contracts/interfaces/IDonaswapV3Pool.sol';import '@donaswap/v3-core/contracts/interfaces/callback/IDonaswapV3SwapCallback.sol';import '@donaswap/v3-periphery/contracts/libraries/Path.sol';import '@uniswap/v2-core/contracts/interfaces/IUniswapV2Pair.sol';import '../base/ImmutableState.sol';import '../interfaces/IMixedRouteQuoterV1.sol';import '../interfaces/IStableSwap.sol';import '../libraries/PoolTicksCounter.sol';import '../libraries/SmartRouterHelper.sol';/// @title Provides on chain quotes for V3, V2, Stable and MixedRoute exact input swaps/// @notice Allows getting the expected amount out for a given swap without executing the swap/// @notice Does not support exact output swaps since using the contract balance between exactOut swaps is not supported/// @dev These functions are not gas efficient and should _not_ be called on chain. Instead, optimistically execute/// the swap and check the amounts in the callback.contract MixedRouteQuoterV1 is IMixedRouteQuoterV1, IDonaswapV3SwapCallback, PeripheryImmutableState {using Path for bytes;using SafeCast for uint256;using PoolTicksCounter for IDonaswapV3Pool;address public immutable factoryV2;address public immutable factoryStable;/**/// @dev Value to bit mask with path fee to determine if V2 or V3 route// max V3 fee: 000011110100001001000000 (24 bits)// mask: 1 << 23 = 100000000000000000000000 = decimal value 8388608uint24 private constant flagBitmask = 8388608;/// @dev Transient storage variable used to check a safety condition in exact output swaps.uint256 private amountOutCached;*/
@donaswap/v3-periphery/contracts/base/PeripheryImmutableState.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity =0.7.6;import '../interfaces/IPeripheryImmutableState.sol';/// @title Immutable state/// @notice Immutable state used by periphery contractsabstract contract PeripheryImmutableState is IPeripheryImmutableState {/// @inheritdoc IPeripheryImmutableStateaddress public immutable override deployer;/// @inheritdoc IPeripheryImmutableStateaddress public immutable override factory;/// @inheritdoc IPeripheryImmutableStateaddress public immutable override WETH9;constructor(address _deployer, address _factory, address _WETH9) {deployer = _deployer;factory = _factory;WETH9 = _WETH9;}}
@donaswap/v3-periphery/contracts/interfaces/IPeripheryImmutableState.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Immutable state/// @notice Functions that return immutable state of the routerinterface IPeripheryImmutableState {/// @return Returns the address of the Donaswap V3 deployerfunction deployer() external view returns (address);/// @return Returns the address of the Donaswap V3 factoryfunction factory() external view returns (address);/// @return Returns the address of WETH9function WETH9() external view returns (address);}
@donaswap/v3-core/contracts/interfaces/IDonaswapV3Pool.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;import './pool/IDonaswapV3PoolImmutables.sol';import './pool/IDonaswapV3PoolState.sol';import './pool/IDonaswapV3PoolDerivedState.sol';import './pool/IDonaswapV3PoolActions.sol';import './pool/IDonaswapV3PoolOwnerActions.sol';import './pool/IDonaswapV3PoolEvents.sol';/// @title The interface for a Donaswap V3 Pool/// @notice A Donaswap pool facilitates swapping and automated market making between any two assets that strictly conform/// to the ERC20 specification/// @dev The pool interface is broken up into many smaller piecesinterface IDonaswapV3Pool isIDonaswapV3PoolImmutables,IDonaswapV3PoolState,IDonaswapV3PoolDerivedState,IDonaswapV3PoolActions,IDonaswapV3PoolOwnerActions,IDonaswapV3PoolEvents{}
@donaswap/v3-core/contracts/interfaces/callback/IDonaswapV3SwapCallback.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Callback for IDonaswapV3PoolActions#swap/// @notice Any contract that calls IDonaswapV3PoolActions#swap must implement this interfaceinterface IDonaswapV3SwapCallback {/// @notice Called to `msg.sender` after executing a swap via IDonaswapV3Pool#swap./// @dev In the implementation you must pay the pool tokens owed for the swap./// The caller of this method must be checked to be a DonaswapV3Pool deployed by the canonical DonaswapV3Factory./// amount0Delta and amount1Delta can both be 0 if no tokens were swapped./// @param amount0Delta The amount of token0 that was sent (negative) or must be received (positive) by the pool by/// the end of the swap. If positive, the callback must send that amount of token0 to the pool./// @param amount1Delta The amount of token1 that was sent (negative) or must be received (positive) by the pool by/// the end of the swap. If positive, the callback must send that amount of token1 to the pool./// @param data Any data passed through by the caller via the IDonaswapV3PoolActions#swap callfunction donaswapV3SwapCallback(int256 amount0Delta,int256 amount1Delta,bytes calldata data) external;}
@donaswap/v3-core/contracts/interfaces/pool/IDonaswapV3PoolActions.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Permissionless pool actions/// @notice Contains pool methods that can be called by anyoneinterface IDonaswapV3PoolActions {/// @notice Sets the initial price for the pool/// @dev Price is represented as a sqrt(amountToken1/amountToken0) Q64.96 value/// @param sqrtPriceX96 the initial sqrt price of the pool as a Q64.96function initialize(uint160 sqrtPriceX96) external;/// @notice Adds liquidity for the given recipient/tickLower/tickUpper position/// @dev The caller of this method receives a callback in the form of IDonaswapV3MintCallback#donaswapV3MintCallback/// in which they must pay any token0 or token1 owed for the liquidity. The amount of token0/token1 due depends/// on tickLower, tickUpper, the amount of liquidity, and the current price./// @param recipient The address for which the liquidity will be created/// @param tickLower The lower tick of the position in which to add liquidity/// @param tickUpper The upper tick of the position in which to add liquidity/// @param amount The amount of liquidity to mint/// @param data Any data that should be passed through to the callback/// @return amount0 The amount of token0 that was paid to mint the given amount of liquidity. Matches the value in the callback/// @return amount1 The amount of token1 that was paid to mint the given amount of liquidity. Matches the value in the callbackfunction mint(address recipient,int24 tickLower,int24 tickUpper,uint128 amount,bytes calldata data) external returns (uint256 amount0, uint256 amount1);/// @notice Collects tokens owed to a position/// @dev Does not recompute fees earned, which must be done either via mint or burn of any amount of liquidity./// Collect must be called by the position owner. To withdraw only token0 or only token1, amount0Requested or/// amount1Requested may be set to zero. To withdraw all tokens owed, caller may pass any value greater than the/// actual tokens owed, e.g. type(uint128).max. Tokens owed may be from accumulated swap fees or burned liquidity./// @param recipient The address which should receive the fees collected/// @param tickLower The lower tick of the position for which to collect fees/// @param tickUpper The upper tick of the position for which to collect fees/// @param amount0Requested How much token0 should be withdrawn from the fees owed/// @param amount1Requested How much token1 should be withdrawn from the fees owed/// @return amount0 The amount of fees collected in token0
@donaswap/v3-core/contracts/interfaces/pool/IDonaswapV3PoolDerivedState.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Pool state that is not stored/// @notice Contains view functions to provide information about the pool that is computed rather than stored on the/// blockchain. The functions here may have variable gas costs.interface IDonaswapV3PoolDerivedState {/// @notice Returns the cumulative tick and liquidity as of each timestamp `secondsAgo` from the current block timestamp/// @dev To get a time weighted average tick or liquidity-in-range, you must call this with two values, one representing/// the beginning of the period and another for the end of the period. E.g., to get the last hour time-weighted average tick,/// you must call it with secondsAgos = [3600, 0]./// @dev The time weighted average tick represents the geometric time weighted average price of the pool, in/// log base sqrt(1.0001) of token1 / token0. The TickMath library can be used to go from a tick value to a ratio./// @param secondsAgos From how long ago each cumulative tick and liquidity value should be returned/// @return tickCumulatives Cumulative tick values as of each `secondsAgos` from the current block timestamp/// @return secondsPerLiquidityCumulativeX128s Cumulative seconds per liquidity-in-range value as of each `secondsAgos` from the current block/// timestampfunction observe(uint32[] calldata secondsAgos)externalviewreturns (int56[] memory tickCumulatives, uint160[] memory secondsPerLiquidityCumulativeX128s);/// @notice Returns a snapshot of the tick cumulative, seconds per liquidity and seconds inside a tick range/// @dev Snapshots must only be compared to other snapshots, taken over a period for which a position existed./// I.e., snapshots cannot be compared if a position is not held for the entire period between when the first/// snapshot is taken and the second snapshot is taken./// @param tickLower The lower tick of the range/// @param tickUpper The upper tick of the range/// @return tickCumulativeInside The snapshot of the tick accumulator for the range/// @return secondsPerLiquidityInsideX128 The snapshot of seconds per liquidity for the range/// @return secondsInside The snapshot of seconds per liquidity for the rangefunction snapshotCumulativesInside(int24 tickLower, int24 tickUpper)externalviewreturns (int56 tickCumulativeInside,uint160 secondsPerLiquidityInsideX128,uint32 secondsInside);}
@donaswap/v3-core/contracts/interfaces/pool/IDonaswapV3PoolEvents.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Events emitted by a pool/// @notice Contains all events emitted by the poolinterface IDonaswapV3PoolEvents {/// @notice Emitted exactly once by a pool when #initialize is first called on the pool/// @dev Mint/Burn/Swap cannot be emitted by the pool before Initialize/// @param sqrtPriceX96 The initial sqrt price of the pool, as a Q64.96/// @param tick The initial tick of the pool, i.e. log base 1.0001 of the starting price of the poolevent Initialize(uint160 sqrtPriceX96, int24 tick);/// @notice Emitted when liquidity is minted for a given position/// @param sender The address that minted the liquidity/// @param owner The owner of the position and recipient of any minted liquidity/// @param tickLower The lower tick of the position/// @param tickUpper The upper tick of the position/// @param amount The amount of liquidity minted to the position range/// @param amount0 How much token0 was required for the minted liquidity/// @param amount1 How much token1 was required for the minted liquidityevent Mint(address sender,address indexed owner,int24 indexed tickLower,int24 indexed tickUpper,uint128 amount,uint256 amount0,uint256 amount1);/// @notice Emitted when fees are collected by the owner of a position/// @dev Collect events may be emitted with zero amount0 and amount1 when the caller chooses not to collect fees/// @param owner The owner of the position for which fees are collected/// @param tickLower The lower tick of the position/// @param tickUpper The upper tick of the position/// @param amount0 The amount of token0 fees collected/// @param amount1 The amount of token1 fees collectedevent Collect(address indexed owner,address recipient,int24 indexed tickLower,
@donaswap/v3-core/contracts/interfaces/pool/IDonaswapV3PoolImmutables.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Pool state that never changes/// @notice These parameters are fixed for a pool forever, i.e., the methods will always return the same valuesinterface IDonaswapV3PoolImmutables {/// @notice The contract that deployed the pool, which must adhere to the IDonaswapV3Factory interface/// @return The contract addressfunction factory() external view returns (address);/// @notice The first of the two tokens of the pool, sorted by address/// @return The token contract addressfunction token0() external view returns (address);/// @notice The second of the two tokens of the pool, sorted by address/// @return The token contract addressfunction token1() external view returns (address);/// @notice The pool's fee in hundredths of a bip, i.e. 1e-6/// @return The feefunction fee() external view returns (uint24);/// @notice The pool tick spacing/// @dev Ticks can only be used at multiples of this value, minimum of 1 and always positive/// e.g.: a tickSpacing of 3 means ticks can be initialized every 3rd tick, i.e., ..., -6, -3, 0, 3, 6, .../// This value is an int24 to avoid casting even though it is always positive./// @return The tick spacingfunction tickSpacing() external view returns (int24);/// @notice The maximum amount of position liquidity that can use any tick in the range/// @dev This parameter is enforced per tick to prevent liquidity from overflowing a uint128 at any point, and/// also prevents out-of-range liquidity from being used to prevent adding in-range liquidity to a pool/// @return The max amount of liquidity per tickfunction maxLiquidityPerTick() external view returns (uint128);}
@donaswap/v3-core/contracts/interfaces/pool/IDonaswapV3PoolOwnerActions.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Permissioned pool actions/// @notice Contains pool methods that may only be called by the factory ownerinterface IDonaswapV3PoolOwnerActions {/// @notice Set the denominator of the protocol's % share of the fees/// @param feeProtocol0 new protocol fee for token0 of the pool/// @param feeProtocol1 new protocol fee for token1 of the poolfunction setFeeProtocol(uint32 feeProtocol0, uint32 feeProtocol1) external;/// @notice Collect the protocol fee accrued to the pool/// @param recipient The address to which collected protocol fees should be sent/// @param amount0Requested The maximum amount of token0 to send, can be 0 to collect fees in only token1/// @param amount1Requested The maximum amount of token1 to send, can be 0 to collect fees in only token0/// @return amount0 The protocol fee collected in token0/// @return amount1 The protocol fee collected in token1function collectProtocol(address recipient,uint128 amount0Requested,uint128 amount1Requested) external returns (uint128 amount0, uint128 amount1);/// @notice Set the LM pool to enable liquidity miningfunction setLmPool(address lmPool) external;}
@donaswap/v3-core/contracts/interfaces/pool/IDonaswapV3PoolState.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Pool state that can change/// @notice These methods compose the pool's state, and can change with any frequency including multiple times/// per transactioninterface IDonaswapV3PoolState {/// @notice The 0th storage slot in the pool stores many values, and is exposed as a single method to save gas/// when accessed externally./// @return sqrtPriceX96 The current price of the pool as a sqrt(token1/token0) Q64.96 value/// tick The current tick of the pool, i.e. according to the last tick transition that was run./// This value may not always be equal to SqrtTickMath.getTickAtSqrtRatio(sqrtPriceX96) if the price is on a tick/// boundary./// observationIndex The index of the last oracle observation that was written,/// observationCardinality The current maximum number of observations stored in the pool,/// observationCardinalityNext The next maximum number of observations, to be updated when the observation./// feeProtocol The protocol fee for both tokens of the pool./// Encoded as two 4 bit values, where the protocol fee of token1 is shifted 4 bits and the protocol fee of token0/// is the lower 4 bits. Used as the denominator of a fraction of the swap fee, e.g. 4 means 1/4th of the swap fee./// unlocked Whether the pool is currently locked to reentrancyfunction slot0()externalviewreturns (uint160 sqrtPriceX96,int24 tick,uint16 observationIndex,uint16 observationCardinality,uint16 observationCardinalityNext,uint32 feeProtocol,bool unlocked);/// @notice The fee growth as a Q128.128 fees of token0 collected per unit of liquidity for the entire life of the pool/// @dev This value can overflow the uint256function feeGrowthGlobal0X128() external view returns (uint256);/// @notice The fee growth as a Q128.128 fees of token1 collected per unit of liquidity for the entire life of the pool/// @dev This value can overflow the uint256function feeGrowthGlobal1X128() external view returns (uint256);
@donaswap/v3-core/contracts/libraries/BitMath.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title BitMath/// @dev This library provides functionality for computing bit properties of an unsigned integerlibrary BitMath {/// @notice Returns the index of the most significant bit of the number,/// where the least significant bit is at index 0 and the most significant bit is at index 255/// @dev The function satisfies the property:/// x >= 2**mostSignificantBit(x) and x < 2**(mostSignificantBit(x)+1)/// @param x the value for which to compute the most significant bit, must be greater than 0/// @return r the index of the most significant bitfunction mostSignificantBit(uint256 x) internal pure returns (uint8 r) {require(x > 0);if (x >= 0x100000000000000000000000000000000) {x >>= 128;r += 128;}if (x >= 0x10000000000000000) {x >>= 64;r += 64;}if (x >= 0x100000000) {x >>= 32;r += 32;}if (x >= 0x10000) {x >>= 16;r += 16;}if (x >= 0x100) {x >>= 8;r += 8;}if (x >= 0x10) {x >>= 4;r += 4;}if (x >= 0x4) {x >>= 2;
@donaswap/v3-core/contracts/libraries/LowGasSafeMath.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.7.0;/// @title Optimized overflow and underflow safe math operations/// @notice Contains methods for doing math operations that revert on overflow or underflow for minimal gas costlibrary LowGasSafeMath {/// @notice Returns x + y, reverts if sum overflows uint256/// @param x The augend/// @param y The addend/// @return z The sum of x and yfunction add(uint256 x, uint256 y) internal pure returns (uint256 z) {require((z = x + y) >= x);}/// @notice Returns x - y, reverts if underflows/// @param x The minuend/// @param y The subtrahend/// @return z The difference of x and yfunction sub(uint256 x, uint256 y) internal pure returns (uint256 z) {require((z = x - y) <= x);}/// @notice Returns x * y, reverts if overflows/// @param x The multiplicand/// @param y The multiplier/// @return z The product of x and yfunction mul(uint256 x, uint256 y) internal pure returns (uint256 z) {require(x == 0 || (z = x * y) / x == y);}/// @notice Returns x + y, reverts if overflows or underflows/// @param x The augend/// @param y The addend/// @return z The sum of x and yfunction add(int256 x, int256 y) internal pure returns (int256 z) {require((z = x + y) >= x == (y >= 0));}/// @notice Returns x - y, reverts if overflows or underflows/// @param x The minuend/// @param y The subtrahend
@donaswap/v3-core/contracts/libraries/SafeCast.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Safe casting methods/// @notice Contains methods for safely casting between typeslibrary SafeCast {/// @notice Cast a uint256 to a uint160, revert on overflow/// @param y The uint256 to be downcasted/// @return z The downcasted integer, now type uint160function toUint160(uint256 y) internal pure returns (uint160 z) {require((z = uint160(y)) == y);}/// @notice Cast a int256 to a int128, revert on overflow or underflow/// @param y The int256 to be downcasted/// @return z The downcasted integer, now type int128function toInt128(int256 y) internal pure returns (int128 z) {require((z = int128(y)) == y);}/// @notice Cast a uint256 to a int256, revert on overflow/// @param y The uint256 to be casted/// @return z The casted integer, now type int256function toInt256(uint256 y) internal pure returns (int256 z) {require(y < 2**255);z = int256(y);}}
@donaswap/v3-core/contracts/libraries/TickBitmap.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;import './BitMath.sol';/// @title Packed tick initialized state library/// @notice Stores a packed mapping of tick index to its initialized state/// @dev The mapping uses int16 for keys since ticks are represented as int24 and there are 256 (2^8) values per word.library TickBitmap {/// @notice Computes the position in the mapping where the initialized bit for a tick lives/// @param tick The tick for which to compute the position/// @return wordPos The key in the mapping containing the word in which the bit is stored/// @return bitPos The bit position in the word where the flag is storedfunction position(int24 tick) private pure returns (int16 wordPos, uint8 bitPos) {wordPos = int16(tick >> 8);bitPos = uint8(tick % 256);}/// @notice Flips the initialized state for a given tick from false to true, or vice versa/// @param self The mapping in which to flip the tick/// @param tick The tick to flip/// @param tickSpacing The spacing between usable ticksfunction flipTick(mapping(int16 => uint256) storage self,int24 tick,int24 tickSpacing) internal {require(tick % tickSpacing == 0); // ensure that the tick is spaced(int16 wordPos, uint8 bitPos) = position(tick / tickSpacing);uint256 mask = 1 << bitPos;self[wordPos] ^= mask;}/// @notice Returns the next initialized tick contained in the same word (or adjacent word) as the tick that is either/// to the left (less than or equal to) or right (greater than) of the given tick/// @param self The mapping in which to compute the next initialized tick/// @param tick The starting tick/// @param tickSpacing The spacing between usable ticks/// @param lte Whether to search for the next initialized tick to the left (less than or equal to the starting tick)/// @return next The next initialized or uninitialized tick up to 256 ticks away from the current tick/// @return initialized Whether the next tick is initialized, as the function only searches within up to 256 ticks
@donaswap/v3-core/contracts/libraries/TickMath.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0 <0.8.0;/// @title Math library for computing sqrt prices from ticks and vice versa/// @notice Computes sqrt price for ticks of size 1.0001, i.e. sqrt(1.0001^tick) as fixed point Q64.96 numbers. Supports/// prices between 2**-128 and 2**128library TickMath {/// @dev The minimum tick that may be passed to #getSqrtRatioAtTick computed from log base 1.0001 of 2**-128int24 internal constant MIN_TICK = -887272;/// @dev The maximum tick that may be passed to #getSqrtRatioAtTick computed from log base 1.0001 of 2**128int24 internal constant MAX_TICK = -MIN_TICK;/// @dev The minimum value that can be returned from #getSqrtRatioAtTick. Equivalent to getSqrtRatioAtTick(MIN_TICK)uint160 internal constant MIN_SQRT_RATIO = 4295128739;/// @dev The maximum value that can be returned from #getSqrtRatioAtTick. Equivalent to getSqrtRatioAtTick(MAX_TICK)uint160 internal constant MAX_SQRT_RATIO = 1461446703485210103287273052203988822378723970342;/// @notice Calculates sqrt(1.0001^tick) * 2^96/// @dev Throws if |tick| > max tick/// @param tick The input tick for the above formula/// @return sqrtPriceX96 A Fixed point Q64.96 number representing the sqrt of the ratio of the two assets (token1/token0)/// at the given tickfunction getSqrtRatioAtTick(int24 tick) internal pure returns (uint160 sqrtPriceX96) {uint256 absTick = tick < 0 ? uint256(-int256(tick)) : uint256(int256(tick));require(absTick <= uint256(MAX_TICK), 'T');uint256 ratio = absTick & 0x1 != 0 ? 0xfffcb933bd6fad37aa2d162d1a594001 : 0x100000000000000000000000000000000;if (absTick & 0x2 != 0) ratio = (ratio * 0xfff97272373d413259a46990580e213a) >> 128;if (absTick & 0x4 != 0) ratio = (ratio * 0xfff2e50f5f656932ef12357cf3c7fdcc) >> 128;if (absTick & 0x8 != 0) ratio = (ratio * 0xffe5caca7e10e4e61c3624eaa0941cd0) >> 128;if (absTick & 0x10 != 0) ratio = (ratio * 0xffcb9843d60f6159c9db58835c926644) >> 128;if (absTick & 0x20 != 0) ratio = (ratio * 0xff973b41fa98c081472e6896dfb254c0) >> 128;if (absTick & 0x40 != 0) ratio = (ratio * 0xff2ea16466c96a3843ec78b326b52861) >> 128;if (absTick & 0x80 != 0) ratio = (ratio * 0xfe5dee046a99a2a811c461f1969c3053) >> 128;if (absTick & 0x100 != 0) ratio = (ratio * 0xfcbe86c7900a88aedcffc83b479aa3a4) >> 128;if (absTick & 0x200 != 0) ratio = (ratio * 0xf987a7253ac413176f2b074cf7815e54) >> 128;if (absTick & 0x400 != 0) ratio = (ratio * 0xf3392b0822b70005940c7a398e4b70f3) >> 128;if (absTick & 0x800 != 0) ratio = (ratio * 0xe7159475a2c29b7443b29c7fa6e889d9) >> 128;if (absTick & 0x1000 != 0) ratio = (ratio * 0xd097f3bdfd2022b8845ad8f792aa5825) >> 128;if (absTick & 0x2000 != 0) ratio = (ratio * 0xa9f746462d870fdf8a65dc1f90e061e5) >> 128;if (absTick & 0x4000 != 0) ratio = (ratio * 0x70d869a156d2a1b890bb3df62baf32f7) >> 128;
@donaswap/v3-periphery/contracts/libraries/BytesLib.sol
// SPDX-License-Identifier: GPL-2.0-or-later/** @title Solidity Bytes Arrays Utils* @author Gonçalo Sá <goncalo.sa@consensys.net>** @dev Bytes tightly packed arrays utility library for ethereum contracts written in Solidity.* The library lets you concatenate, slice and type cast bytes arrays both in memory and storage.*/pragma solidity >=0.5.0 <0.8.0;library BytesLib {function slice(bytes memory _bytes,uint256 _start,uint256 _length) internal pure returns (bytes memory) {require(_length + 31 >= _length, 'slice_overflow');require(_start + _length >= _start, 'slice_overflow');require(_bytes.length >= _start + _length, 'slice_outOfBounds');bytes memory tempBytes;assembly {switch iszero(_length)case 0 {// Get a location of some free memory and store it in tempBytes as// Solidity does for memory variables.tempBytes := mload(0x40)// The first word of the slice result is potentially a partial// word read from the original array. To read it, we calculate// the length of that partial word and start copying that many// bytes into the array. The first word we copy will start with// data we don't care about, but the last `lengthmod` bytes will// land at the beginning of the contents of the new array. When// we're done copying, we overwrite the full first word with// the actual length of the slice.let lengthmod := and(_length, 31)// The multiplication in the next line is necessary// because when slicing multiples of 32 bytes (lengthmod == 0)
@donaswap/v3-periphery/contracts/libraries/Path.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.6.0;import './BytesLib.sol';/// @title Functions for manipulating path data for multihop swapslibrary Path {using BytesLib for bytes;/// @dev The length of the bytes encoded addressuint256 private constant ADDR_SIZE = 20;/// @dev The length of the bytes encoded feeuint256 private constant FEE_SIZE = 3;/// @dev The offset of a single token address and pool feeuint256 private constant NEXT_OFFSET = ADDR_SIZE + FEE_SIZE;/// @dev The offset of an encoded pool keyuint256 private constant POP_OFFSET = NEXT_OFFSET + ADDR_SIZE;/// @dev The minimum length of an encoding that contains 2 or more poolsuint256 private constant MULTIPLE_POOLS_MIN_LENGTH = POP_OFFSET + NEXT_OFFSET;/// @notice Returns true iff the path contains two or more pools/// @param path The encoded swap path/// @return True if path contains two or more pools, otherwise falsefunction hasMultiplePools(bytes memory path) internal pure returns (bool) {return path.length >= MULTIPLE_POOLS_MIN_LENGTH;}/// @notice Returns the number of pools in the path/// @param path The encoded swap path/// @return The number of pools in the pathfunction numPools(bytes memory path) internal pure returns (uint256) {// Ignore the first token address. From then on every fee and token offset indicates a pool.return ((path.length - ADDR_SIZE) / NEXT_OFFSET);}/// @notice Decodes the first pool in path/// @param path The bytes encoded swap path/// @return tokenA The first token of the given pool/// @return tokenB The second token of the given pool/// @return fee The fee level of the pool
@uniswap/v2-core/contracts/interfaces/IUniswapV2Pair.sol
pragma solidity >=0.5.0;interface IUniswapV2Pair {event Approval(address indexed owner, address indexed spender, uint value);event Transfer(address indexed from, address indexed to, uint value);function name() external pure returns (string memory);function symbol() external pure returns (string memory);function decimals() external pure returns (uint8);function totalSupply() external view returns (uint);function balanceOf(address owner) external view returns (uint);function allowance(address owner, address spender) external view returns (uint);function approve(address spender, uint value) external returns (bool);function transfer(address to, uint value) external returns (bool);function transferFrom(address from, address to, uint value) external returns (bool);function DOMAIN_SEPARATOR() external view returns (bytes32);function PERMIT_TYPEHASH() external pure returns (bytes32);function nonces(address owner) external view returns (uint);function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external;event Mint(address indexed sender, uint amount0, uint amount1);event Burn(address indexed sender, uint amount0, uint amount1, address indexed to);event Swap(address indexed sender,uint amount0In,uint amount1In,uint amount0Out,uint amount1Out,address indexed to);event Sync(uint112 reserve0, uint112 reserve1);function MINIMUM_LIQUIDITY() external pure returns (uint);function factory() external view returns (address);function token0() external view returns (address);function token1() external view returns (address);function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);function price0CumulativeLast() external view returns (uint);
contracts/base/ImmutableState.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity =0.7.6;import '../interfaces/IImmutableState.sol';/// @title Immutable state/// @notice Immutable state used by the swap routerabstract contract ImmutableState is IImmutableState {/// @inheritdoc IImmutableStateaddress public immutable override factoryV2;/// @inheritdoc IImmutableStateaddress public immutable override positionManager;constructor(address _factoryV2, address _positionManager) {factoryV2 = _factoryV2;positionManager = _positionManager;}}
contracts/interfaces/IImmutableState.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.5.0;/// @title Immutable state/// @notice Functions that return immutable state of the routerinterface IImmutableState {/// @return Returns the address of the Donaswap V2 factoryfunction factoryV2() external view returns (address);/// @return Returns the address of Donaswap V3 NFT position managerfunction positionManager() external view returns (address);}
contracts/interfaces/IMixedRouteQuoterV1.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.7.5;pragma abicoder v2;/// @title MixedRouteQuoterV1 Interface/// @notice Supports quoting the calculated amounts for exact input swaps. Is specialized for routes containing a mix of Stable, V2 and V3 liquidity./// @notice For each pool also tells you the number of initialized ticks crossed and the sqrt price of the pool after the swap./// @dev These functions are not marked view because they rely on calling non-view functions and reverting/// to compute the result. They are also not gas efficient and should not be called on-chain.interface IMixedRouteQuoterV1 {/// @notice Returns the amount out received for a given exact input swap without executing the swap/// @param path The path of the swap, i.e. each token pair and the pool fee/// @param flag 0 for V3, 1 for V2, 2 for 2pool, 3 for 3pool/// @param amountIn The amount of the first token to swap/// @return amountOut The amount of the last token that would be received/// @return v3SqrtPriceX96AfterList List of the sqrt price after the swap for each v3 pool in the path, 0 for v2 or stable pools/// @return v3InitializedTicksCrossedList List of the initialized ticks that the swap crossed for each v3 pool in the path, 0 for v2 or stable pools/// @return v3SwapGasEstimate The estimate of the gas that the v3 swaps in the path consumefunction quoteExactInput(bytes memory path, uint256[] memory flag, uint256 amountIn)externalreturns (uint256 amountOut,uint160[] memory v3SqrtPriceX96AfterList,uint32[] memory v3InitializedTicksCrossedList,uint256 v3SwapGasEstimate);struct QuoteExactInputSingleV3Params {address tokenIn;address tokenOut;uint256 amountIn;uint24 fee;uint160 sqrtPriceLimitX96;}struct QuoteExactInputSingleV2Params {address tokenIn;address tokenOut;uint256 amountIn;}
contracts/interfaces/IStableSwap.sol
// SPDX-License-Identifier: MITpragma solidity =0.7.6;pragma abicoder v2;interface IStableSwap {// solium-disable-next-line mixedcasefunction get_dy(uint256 i,uint256 j,uint256 dx) external view returns (uint256 dy);// solium-disable-next-line mixedcasefunction exchange(uint256 i,uint256 j,uint256 dx,uint256 minDy) external payable;// solium-disable-next-line mixedcasefunction coins(uint256 i) external view returns (address);// solium-disable-next-line mixedcasefunction balances(uint256 i) external view returns (uint256);// solium-disable-next-line mixedcasefunction A() external view returns (uint256);// solium-disable-next-line mixedcasefunction fee() external view returns (uint256);}
contracts/interfaces/IStableSwapFactory.sol
// SPDX-License-Identifier: MITpragma solidity =0.7.6;pragma abicoder v2;interface IStableSwapFactory {struct StableSwapPairInfo {address swapContract;address token0;address token1;address LPContract;}struct StableSwapThreePoolPairInfo {address swapContract;address token0;address token1;address token2;address LPContract;}// solium-disable-next-line mixedcasefunction pairLength() external view returns (uint256);function getPairInfo(address _tokenA, address _tokenB)externalviewreturns (StableSwapPairInfo memory info);function getThreePoolPairInfo(address _tokenA, address _tokenB)externalviewreturns (StableSwapThreePoolPairInfo memory info);}
contracts/interfaces/IStableSwapInfo.sol
// SPDX-License-Identifier: MITpragma solidity =0.7.6;pragma abicoder v2;interface IStableSwapInfo {function get_dx(address _swap,uint256 i,uint256 j,uint256 dy,uint256 max_dx) external view returns (uint256);}
contracts/libraries/PoolTicksCounter.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity >=0.6.0;import '@donaswap/v3-core/contracts/interfaces/IDonaswapV3Pool.sol';library PoolTicksCounter {/// @dev This function counts the number of initialized ticks that would incur a gas cost between tickBefore and tickAfter./// When tickBefore and/or tickAfter themselves are initialized, the logic over whether we should count them depends on the/// direction of the swap. If we are swapping upwards (tickAfter > tickBefore) we don't want to count tickBefore but we do/// want to count tickAfter. The opposite is true if we are swapping downwards.function countInitializedTicksCrossed(IDonaswapV3Pool self,int24 tickBefore,int24 tickAfter) internal view returns (uint32 initializedTicksCrossed) {int16 wordPosLower;int16 wordPosHigher;uint8 bitPosLower;uint8 bitPosHigher;bool tickBeforeInitialized;bool tickAfterInitialized;{// Get the key and offset in the tick bitmap of the active tick before and after the swap.int16 wordPos = int16((tickBefore / self.tickSpacing()) >> 8);uint8 bitPos = uint8((tickBefore / self.tickSpacing()) % 256);int16 wordPosAfter = int16((tickAfter / self.tickSpacing()) >> 8);uint8 bitPosAfter = uint8((tickAfter / self.tickSpacing()) % 256);// In the case where tickAfter is initialized, we only want to count it if we are swapping downwards.// If the initializable tick after the swap is initialized, our original tickAfter is a// multiple of tick spacing, and we are swapping downwards we know that tickAfter is initialized// and we shouldn't count it.tickAfterInitialized =((self.tickBitmap(wordPosAfter) & (1 << bitPosAfter)) > 0) &&((tickAfter % self.tickSpacing()) == 0) &&(tickBefore > tickAfter);// In the case where tickBefore is initialized, we only want to count it if we are swapping upwards.// Use the same logic as above to decide whether we should count tickBefore or not.
contracts/libraries/SmartRouterHelper.sol
// SPDX-License-Identifier: GPL-2.0-or-laterpragma solidity =0.7.6;pragma abicoder v2;import '../interfaces/IStableSwapFactory.sol';import '../interfaces/IStableSwapInfo.sol';import '@uniswap/v2-core/contracts/interfaces/IUniswapV2Pair.sol';import '@donaswap/v3-core/contracts/libraries/LowGasSafeMath.sol';import '@donaswap/v3-core/contracts/interfaces/IDonaswapV3Pool.sol';library SmartRouterHelper {using LowGasSafeMath for uint256;/************************************************** Stable **************************************************/// get the pool info in stable swapfunction getStableInfo(address stableSwapFactory,address input,address output,uint256 flag) public view returns (uint256 i, uint256 j, address swapContract) {if (flag == 2) {IStableSwapFactory.StableSwapPairInfo memory info = IStableSwapFactory(stableSwapFactory).getPairInfo(input, output);i = input == info.token0 ? 0 : 1;j = (i == 0) ? 1 : 0;swapContract = info.swapContract;} else if (flag == 3) {IStableSwapFactory.StableSwapThreePoolPairInfo memory info = IStableSwapFactory(stableSwapFactory).getThreePoolPairInfo(input, output);if (input == info.token0) i = 0;else if (input == info.token1) i = 1;else if (input == info.token2) i = 2;if (output == info.token0) j = 0;else if (output == info.token1) j = 1;else if (output == info.token2) j = 2;swapContract = info.swapContract;}
Contract ABI
[{"type":"constructor","stateMutability":"nonpayable","inputs":[{"type":"address","name":"_deployer","internalType":"address"},{"type":"address","name":"_factory","internalType":"address"},{"type":"address","name":"_factoryV2","internalType":"address"},{"type":"address","name":"_factoryStable","internalType":"address"},{"type":"address","name":"_WETH9","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"WETH9","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"deployer","inputs":[]},{"type":"function","stateMutability":"view","outputs":[],"name":"donaswapV3SwapCallback","inputs":[{"type":"int256","name":"amount0Delta","internalType":"int256"},{"type":"int256","name":"amount1Delta","internalType":"int256"},{"type":"bytes","name":"path","internalType":"bytes"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"factory","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"factoryStable","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"factoryV2","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[{"type":"uint256","name":"amountOut","internalType":"uint256"},{"type":"uint160[]","name":"v3SqrtPriceX96AfterList","internalType":"uint160[]"},{"type":"uint32[]","name":"v3InitializedTicksCrossedList","internalType":"uint32[]"},{"type":"uint256","name":"v3SwapGasEstimate","internalType":"uint256"}],"name":"quoteExactInput","inputs":[{"type":"bytes","name":"path","internalType":"bytes"},{"type":"uint256[]","name":"flag","internalType":"uint256[]"},{"type":"uint256","name":"amountIn","internalType":"uint256"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"amountOut","internalType":"uint256"}],"name":"quoteExactInputSingleStable","inputs":[{"type":"tuple","name":"params","internalType":"struct IMixedRouteQuoterV1.QuoteExactInputSingleStableParams","components":[{"type":"address","name":"tokenIn","internalType":"address"},{"type":"address","name":"tokenOut","internalType":"address"},{"type":"uint256","name":"amountIn","internalType":"uint256"},{"type":"uint256","name":"flag","internalType":"uint256"}]}]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"amountOut","internalType":"uint256"}],"name":"quoteExactInputSingleV2","inputs":[{"type":"tuple","name":"params","internalType":"struct IMixedRouteQuoterV1.QuoteExactInputSingleV2Params","components":[{"type":"address","name":"tokenIn","internalType":"address"},{"type":"address","name":"tokenOut","internalType":"address"},{"type":"uint256","name":"amountIn","internalType":"uint256"}]}]},{"type":"function","stateMutability":"nonpayable","outputs":[{"type":"uint256","name":"amountOut","internalType":"uint256"},{"type":"uint160","name":"sqrtPriceX96After","internalType":"uint160"},{"type":"uint32","name":"initializedTicksCrossed","internalType":"uint32"},{"type":"uint256","name":"gasEstimate","internalType":"uint256"}],"name":"quoteExactInputSingleV3","inputs":[{"type":"tuple","name":"params","internalType":"struct IMixedRouteQuoterV1.QuoteExactInputSingleV3Params","components":[{"type":"address","name":"tokenIn","internalType":"address"},{"type":"address","name":"tokenOut","internalType":"address"},{"type":"uint256","name":"amountIn","internalType":"uint256"},{"type":"uint24","name":"fee","internalType":"uint24"},{"type":"uint160","name":"sqrtPriceLimitX96","internalType":"uint160"}]}]}]
Contract Creation Code
0x6101206040523480156200001257600080fd5b5060405162001f6b38038062001f6b83398101604081905262000035916200008a565b6001600160601b0319606095861b811660805293851b841660a052841b831660c05290831b821660e05290911b1661010052620000f9565b80516001600160a01b03811681146200008557600080fd5b919050565b600080600080600060a08688031215620000a2578081fd5b620000ad866200006d565b9450620000bd602087016200006d565b9350620000cd604087016200006d565b9250620000dd606087016200006d565b9150620000ed608087016200006d565b90509295509295909350565b60805160601c60a05160601c60c05160601c60e05160601c6101005160601c611e0a620001616000398061041f52806104855250806103fb52806105cd525080610161525080610ba4525080610729528061082052806109c45280610bc85250611e0a6000f3fe608060405234801561001057600080fd5b506004361061008e5760003560e01c80634aa4a4fc14610093578063558b66fb146100b157806368e0d4e1146100d45780637966a066146100dc57806388cf8466146100e45780638a49928614610104578063ad8d98d314610117578063b174a0a71461012c578063c45a01551461014f578063d5f3948814610157575b600080fd5b61009b61015f565b6040516100a89190611b1d565b60405180910390f35b6100c46100bf366004611671565b610183565b6040516100a89493929190611c6d565b61009b6103f9565b61009b61041d565b6100f76100f2366004611838565b610441565b6040516100a89190611c64565b6100f76101123660046118a5565b610590565b61012a610125366004611782565b6106e1565b005b61013f61013a366004611908565b61096e565b6040516100a89493929190611d08565b61009b610ba2565b61009b610bc6565b7f000000000000000000000000000000000000000000000000000000000000000081565b6000606080600061019387610bea565b6001600160401b03811180156101a857600080fd5b506040519080825280602002602001820160405280156101d2578160200160208202803683370190505b5092506101de87610bea565b6001600160401b03811180156101f357600080fd5b5060405190808252806020026020018201604052801561021d578160200160208202803683370190505b50915060005b60008060006102318b610bfb565b92509250925089848151811061024357fe5b60200260200101516001141561028f576102886040518060600160405280856001600160a01b03168152602001846001600160a01b031681526020018b815250610590565b98506103b8565b89848151811061029b57fe5b602002602001015160001415610367576000806000806103016040518060a00160405280896001600160a01b03168152602001886001600160a01b031681526020018f81526020018762ffffff16815260200160006001600160a01b031681525061096e565b9350935093509350828b898151811061031657fe5b60200260200101906001600160a01b031690816001600160a01b031681525050818a898151811061034357fe5b63ffffffff90921660209283029190910190910152929b50505094909401936103b8565b6103b56040518060800160405280856001600160a01b03168152602001846001600160a01b031681526020018b81526020018c87815181106103a557fe5b6020026020010151815250610441565b98505b6001909301926103c78b610c2c565b156103dc576103d58b610c34565b9a506103e8565b889750505050506103f0565b505050610223565b93509350935093565b7f000000000000000000000000000000000000000000000000000000000000000081565b7f000000000000000000000000000000000000000000000000000000000000000081565b80516020820151606083015160405163b735aecd60e01b8152600093849384938493733b07ea3556ce5bf50e1fa41bae2aea8dbfe922029363b735aecd936104ad937f000000000000000000000000000000000000000000000000000000000000000093600401611b82565b60606040518083038186803b1580156104c557600080fd5b505af41580156104d9573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104fd9190611a83565b925092509250806001600160a01b031663556d6e9f848488604001516040518463ffffffff1660e01b815260040161053793929190611d32565b60206040518083038186803b15801561054f57600080fd5b505afa158015610563573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105879190611a2e565b95945050505050565b80516020820151604051633274946160e01b815260009283928392733b07ea3556ce5bf50e1fa41bae2aea8dbfe92202926332749461926105f7927f0000000000000000000000000000000000000000000000000000000000000000929190600401611b31565b604080518083038186803b15801561060e57600080fd5b505af4158015610622573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610646919061175f565b91509150733b07ea3556ce5bf50e1fa41bae2aea8dbfe9220263054d50d4856040015184846040518463ffffffff1660e01b815260040161068993929190611d32565b60206040518083038186803b1580156106a157600080fd5b505af41580156106b5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106d99190611a2e565b949350505050565b60008313806106f05750600082135b6106f957600080fd5b600080600061070784610bfb565b925092509250733b07ea3556ce5bf50e1fa41bae2aea8dbfe92202638bdb19257f00000000000000000000000000000000000000000000000000000000000000008585856040518563ffffffff1660e01b815260040161076a9493929190611b54565b60206040518083038186803b15801561078257600080fd5b505af4158015610796573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906107ba919061173c565b50600080600088136107e457846001600160a01b0316846001600160a01b031610886000036107fe565b836001600160a01b0316856001600160a01b031610876000035b915091506000733b07ea3556ce5bf50e1fa41bae2aea8dbfe92202634e6c8ed87f00000000000000000000000000000000000000000000000000000000000000008888886040518563ffffffff1660e01b81526004016108619493929190611b54565b60206040518083038186803b15801561087957600080fd5b505af415801561088d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108b1919061173c565b9050600080826001600160a01b0316633850c7bd6040518163ffffffff1660e01b815260040160e06040518083038186803b1580156108ef57600080fd5b505afa158015610903573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109279190611994565b505050505091509150841561094d57604051848152826020820152816040820152606081fd5b60405162461bcd60e51b815260040161096590611c2f565b60405180910390fd5b6020810151815160608301516040516309cd91db60e31b81526000938493849384936001600160a01b0380851690841610938593733b07ea3556ce5bf50e1fa41bae2aea8dbfe9220293634e6c8ed8936109ef937f000000000000000000000000000000000000000000000000000000000000000093929091600401611b54565b60206040518083038186803b158015610a0757600080fd5b505af4158015610a1b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a3f919061173c565b905060005a9050816001600160a01b031663128acb083085610a648c60400151610c51565b60808d01516001600160a01b031615610a81578c60800151610aa7565b87610aa05773fffd8963efd1fc6a506488495d951d5263988d25610aa7565b6401000276a45b8d600001518e606001518f60200151604051602001610ac893929190611ae7565b6040516020818303038152906040526040518663ffffffff1660e01b8152600401610af7959493929190611bac565b6040805180830381600087803b158015610b1057600080fd5b505af1925050508015610b40575060408051601f3d908101601f19168201909252610b3d9181019061175f565b60015b610b95573d808015610b6e576040519150601f19603f3d011682016040523d82523d6000602084013e610b73565b606091505b505a82039450610b84818487610c67565b975097509750975050505050610b9b565b50505050505b9193509193565b7f000000000000000000000000000000000000000000000000000000000000000081565b7f000000000000000000000000000000000000000000000000000000000000000081565b80516017601319909101045b919050565b60008080610c098482610d21565b9250610c16846014610dd1565b9050610c23846017610d21565b91509193909250565b516042111590565b8051606090610c4b90839060179060161901610e78565b92915050565b6000600160ff1b8210610c6357600080fd5b5090565b600080600080600080876001600160a01b0316633850c7bd6040518163ffffffff1660e01b815260040160e06040518083038186803b158015610ca957600080fd5b505afa158015610cbd573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610ce19190611994565b50939650610cf694508d9350610fc992505050565b91975095509050610d116001600160a01b038916838361104d565b9350869250505093509350935093565b600081826014011015610d70576040805162461bcd60e51b8152602060048201526012602482015271746f416464726573735f6f766572666c6f7760701b604482015290519081900360640190fd5b8160140183511015610dc1576040805162461bcd60e51b8152602060048201526015602482015274746f416464726573735f6f75744f66426f756e647360581b604482015290519081900360640190fd5b500160200151600160601b900490565b600081826003011015610e1f576040805162461bcd60e51b8152602060048201526011602482015270746f55696e7432345f6f766572666c6f7760781b604482015290519081900360640190fd5b8160030183511015610e6f576040805162461bcd60e51b8152602060048201526014602482015273746f55696e7432345f6f75744f66426f756e647360601b604482015290519081900360640190fd5b50016003015190565b60608182601f011015610ec3576040805162461bcd60e51b815260206004820152600e60248201526d736c6963655f6f766572666c6f7760901b604482015290519081900360640190fd5b828284011015610f0b576040805162461bcd60e51b815260206004820152600e60248201526d736c6963655f6f766572666c6f7760901b604482015290519081900360640190fd5b81830184511015610f57576040805162461bcd60e51b8152602060048201526011602482015270736c6963655f6f75744f66426f756e647360781b604482015290519081900360640190fd5b606082158015610f765760405191506000825260208201604052610fc0565b6040519150601f8416801560200281840101858101878315602002848b0101015b81831015610faf578051835260209283019201610f97565b5050858452601f01601f1916604052505b50949350505050565b6000806000835160601461102c57604484511015610ff95760405162461bcd60e51b815260040161096590611c05565b6004840193508380602001905181019061101391906117cf565b60405162461bcd60e51b81526004016109659190611bf2565b838060200190518101906110409190611a46565b9250925092509193909250565b60008060008060008060008060088b6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561109457600080fd5b505afa1580156110a8573d6000803e3d6000fd5b505050506040513d60208110156110be57600080fd5b5051600290810b908c900b816110d057fe5b0560020b901d905060006101008c6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561111657600080fd5b505afa15801561112a573d6000803e3d6000fd5b505050506040513d602081101561114057600080fd5b5051600290810b908d900b8161115257fe5b0560020b8161115d57fe5b079050600060088d6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561119d57600080fd5b505afa1580156111b1573d6000803e3d6000fd5b505050506040513d60208110156111c757600080fd5b5051600290810b908d900b816111d957fe5b0560020b901d905060006101008e6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561121f57600080fd5b505afa158015611233573d6000803e3d6000fd5b505050506040513d602081101561124957600080fd5b5051600290810b908e900b8161125b57fe5b0560020b8161126657fe5b07905060008160ff166001901b8f6001600160a01b0316635339c296856040518263ffffffff1660e01b8152600401808260010b815260200191505060206040518083038186803b1580156112ba57600080fd5b505afa1580156112ce573d6000803e3d6000fd5b505050506040513d60208110156112e457600080fd5b50511611801561136a57508d6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561132857600080fd5b505afa15801561133c573d6000803e3d6000fd5b505050506040513d602081101561135257600080fd5b5051600290810b908d900b8161136457fe5b0760020b155b801561137b57508b60020b8d60020b135b945060008360ff166001901b8f6001600160a01b0316635339c296876040518263ffffffff1660e01b8152600401808260010b815260200191505060206040518083038186803b1580156113ce57600080fd5b505afa1580156113e2573d6000803e3d6000fd5b505050506040513d60208110156113f857600080fd5b50511611801561147e57508d6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561143c57600080fd5b505afa158015611450573d6000803e3d6000fd5b505050506040513d602081101561146657600080fd5b5051600290810b908e900b8161147857fe5b0760020b155b801561148f57508b60020b8d60020b125b95508160010b8460010b12806114bb57508160010b8460010b1480156114bb57508060ff168360ff1611155b156114d1578399508297508198508096506114de565b8199508097508398508296505b505060001960ff87161b9150505b8560010b8760010b136115ae578560010b8760010b14156115135760001960ff858103161c165b6000818c6001600160a01b0316635339c2968a6040518263ffffffff1660e01b8152600401808260010b815260200191505060206040518083038186803b15801561155d57600080fd5b505afa158015611571573d6000803e3d6000fd5b505050506040513d602081101561158757600080fd5b5051169050611595816115d6565b61ffff16989098019750506001909501946000196114ec565b81156115bb576001880397505b82156115c8576001880397505b505050505050509392505050565b6000805b8215610c4b576000198301909216916001016115da565b600082601f830112611601578081fd5b813561161461160f82611d6b565b611d48565b818152846020838601011115611628578283fd5b816020850160208301379081016020019190915292915050565b8051600281900b8114610bf657600080fd5b8035610bf681611dbc565b805161ffff81168114610bf657600080fd5b600080600060608486031215611685578283fd5b83356001600160401b038082111561169b578485fd5b6116a7878388016115f1565b94506020915081860135818111156116bd578485fd5b8601601f810188136116cd578485fd5b8035828111156116d957fe5b83810292506116e9848401611d48565b8181528481019083860185850187018c1015611703578889fd5b8895505b83861015611725578035835260019590950194918601918601611707565b50979a979950505050604095909501359450505050565b60006020828403121561174d578081fd5b815161175881611dbc565b9392505050565b60008060408385031215611771578182fd5b505080516020909101519092909150565b600080600060608486031215611796578081fd5b833592506020840135915060408401356001600160401b038111156117b9578182fd5b6117c5868287016115f1565b9150509250925092565b6000602082840312156117e0578081fd5b81516001600160401b038111156117f5578182fd5b8201601f81018413611805578182fd5b805161181361160f82611d6b565b818152856020838501011115611827578384fd5b610587826020830160208601611d8c565b600060808284031215611849578081fd5b604051608081016001600160401b038111828210171561186557fe5b604052823561187381611dbc565b8152602083013561188381611dbc565b6020820152604083810135908201526060928301359281019290925250919050565b6000606082840312156118b6578081fd5b604051606081016001600160401b03811182821017156118d257fe5b60405282356118e081611dbc565b815260208301356118f081611dbc565b60208201526040928301359281019290925250919050565b600060a08284031215611919578081fd5b60405160a081016001600160401b038111828210171561193557fe5b604052823561194381611dbc565b8152602083013561195381611dbc565b602082015260408381013590820152606083013562ffffff81168114611977578283fd5b606082015261198860808401611654565b60808201529392505050565b600080600080600080600060e0888a0312156119ae578485fd5b87516119b981611dbc565b96506119c760208901611642565b95506119d56040890161165f565b94506119e36060890161165f565b93506119f16080890161165f565b925060a088015163ffffffff81168114611a09578283fd5b60c08901519092508015158114611a1e578182fd5b8091505092959891949750929550565b600060208284031215611a3f578081fd5b5051919050565b600080600060608486031215611a5a578081fd5b835192506020840151611a6c81611dbc565b9150611a7a60408501611642565b90509250925092565b600080600060608486031215611a97578081fd5b83519250602084015191506040840151611ab081611dbc565b809150509250925092565b60008151808452611ad3816020860160208601611d8c565b601f01601f19169290920160200192915050565b606093841b6001600160601b0319908116825260e89390931b6001600160e81b0319166014820152921b166017820152602b0190565b6001600160a01b0391909116815260200190565b6001600160a01b0393841681529183166020830152909116604082015260600190565b6001600160a01b03948516815292841660208401529216604082015262ffffff909116606082015260800190565b6001600160a01b039485168152928416602084015292166040820152606081019190915260800190565b6001600160a01b0386811682528515156020830152604082018590528316606082015260a060808201819052600090611be790830184611abb565b979650505050505050565b6000602082526117586020830184611abb565b60208082526010908201526f2ab732bc3832b1ba32b21032b93937b960811b604082015260600190565b6020808252818101527f4578616374206f75747075742071756f7465206e6f7420737570706f72746564604082015260600190565b90815260200190565b600060808201868352602060808185015281875180845260a0860191508289019350845b81811015611cb65784516001600160a01b031683529383019391830191600101611c91565b505084810360408601528651808252908201925081870190845b81811015611cf257825163ffffffff1685529383019391830191600101611cd0565b5050505060609290920192909252949350505050565b9384526001600160a01b0392909216602084015263ffffffff166040830152606082015260800190565b9283526020830191909152604082015260600190565b6040518181016001600160401b0381118282101715611d6357fe5b604052919050565b60006001600160401b03821115611d7e57fe5b50601f01601f191660200190565b60005b83811015611da7578181015183820152602001611d8f565b83811115611db6576000848401525b50505050565b6001600160a01b0381168114611dd157600080fd5b5056fea2646970667358221220be8f07032f0e5210561cab2ecea7f28165a5b95e4959e880ff2fbb8d5a85fa8464736f6c634300070600330000000000000000000000007342530a1a1b39df96755504e236525395cc84a50000000000000000000000009c3afddea87a726891a44c037242393d524cacfe0000000000000000000000008e5dff1c121f661971d02950698f8c5efc3dfa780000000000000000000000000000000000000000000000000000000000000000000000000000000000000000730253eee681a77435a68c7f0c3f23abed458858
Deployed ByteCode
0x608060405234801561001057600080fd5b506004361061008e5760003560e01c80634aa4a4fc14610093578063558b66fb146100b157806368e0d4e1146100d45780637966a066146100dc57806388cf8466146100e45780638a49928614610104578063ad8d98d314610117578063b174a0a71461012c578063c45a01551461014f578063d5f3948814610157575b600080fd5b61009b61015f565b6040516100a89190611b1d565b60405180910390f35b6100c46100bf366004611671565b610183565b6040516100a89493929190611c6d565b61009b6103f9565b61009b61041d565b6100f76100f2366004611838565b610441565b6040516100a89190611c64565b6100f76101123660046118a5565b610590565b61012a610125366004611782565b6106e1565b005b61013f61013a366004611908565b61096e565b6040516100a89493929190611d08565b61009b610ba2565b61009b610bc6565b7f000000000000000000000000730253eee681a77435a68c7f0c3f23abed45885881565b6000606080600061019387610bea565b6001600160401b03811180156101a857600080fd5b506040519080825280602002602001820160405280156101d2578160200160208202803683370190505b5092506101de87610bea565b6001600160401b03811180156101f357600080fd5b5060405190808252806020026020018201604052801561021d578160200160208202803683370190505b50915060005b60008060006102318b610bfb565b92509250925089848151811061024357fe5b60200260200101516001141561028f576102886040518060600160405280856001600160a01b03168152602001846001600160a01b031681526020018b815250610590565b98506103b8565b89848151811061029b57fe5b602002602001015160001415610367576000806000806103016040518060a00160405280896001600160a01b03168152602001886001600160a01b031681526020018f81526020018762ffffff16815260200160006001600160a01b031681525061096e565b9350935093509350828b898151811061031657fe5b60200260200101906001600160a01b031690816001600160a01b031681525050818a898151811061034357fe5b63ffffffff90921660209283029190910190910152929b50505094909401936103b8565b6103b56040518060800160405280856001600160a01b03168152602001846001600160a01b031681526020018b81526020018c87815181106103a557fe5b6020026020010151815250610441565b98505b6001909301926103c78b610c2c565b156103dc576103d58b610c34565b9a506103e8565b889750505050506103f0565b505050610223565b93509350935093565b7f0000000000000000000000008e5dff1c121f661971d02950698f8c5efc3dfa7881565b7f000000000000000000000000000000000000000000000000000000000000000081565b80516020820151606083015160405163b735aecd60e01b8152600093849384938493733b07ea3556ce5bf50e1fa41bae2aea8dbfe922029363b735aecd936104ad937f000000000000000000000000000000000000000000000000000000000000000093600401611b82565b60606040518083038186803b1580156104c557600080fd5b505af41580156104d9573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104fd9190611a83565b925092509250806001600160a01b031663556d6e9f848488604001516040518463ffffffff1660e01b815260040161053793929190611d32565b60206040518083038186803b15801561054f57600080fd5b505afa158015610563573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105879190611a2e565b95945050505050565b80516020820151604051633274946160e01b815260009283928392733b07ea3556ce5bf50e1fa41bae2aea8dbfe92202926332749461926105f7927f0000000000000000000000008e5dff1c121f661971d02950698f8c5efc3dfa78929190600401611b31565b604080518083038186803b15801561060e57600080fd5b505af4158015610622573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610646919061175f565b91509150733b07ea3556ce5bf50e1fa41bae2aea8dbfe9220263054d50d4856040015184846040518463ffffffff1660e01b815260040161068993929190611d32565b60206040518083038186803b1580156106a157600080fd5b505af41580156106b5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106d99190611a2e565b949350505050565b60008313806106f05750600082135b6106f957600080fd5b600080600061070784610bfb565b925092509250733b07ea3556ce5bf50e1fa41bae2aea8dbfe92202638bdb19257f0000000000000000000000007342530a1a1b39df96755504e236525395cc84a58585856040518563ffffffff1660e01b815260040161076a9493929190611b54565b60206040518083038186803b15801561078257600080fd5b505af4158015610796573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906107ba919061173c565b50600080600088136107e457846001600160a01b0316846001600160a01b031610886000036107fe565b836001600160a01b0316856001600160a01b031610876000035b915091506000733b07ea3556ce5bf50e1fa41bae2aea8dbfe92202634e6c8ed87f0000000000000000000000007342530a1a1b39df96755504e236525395cc84a58888886040518563ffffffff1660e01b81526004016108619493929190611b54565b60206040518083038186803b15801561087957600080fd5b505af415801561088d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108b1919061173c565b9050600080826001600160a01b0316633850c7bd6040518163ffffffff1660e01b815260040160e06040518083038186803b1580156108ef57600080fd5b505afa158015610903573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109279190611994565b505050505091509150841561094d57604051848152826020820152816040820152606081fd5b60405162461bcd60e51b815260040161096590611c2f565b60405180910390fd5b6020810151815160608301516040516309cd91db60e31b81526000938493849384936001600160a01b0380851690841610938593733b07ea3556ce5bf50e1fa41bae2aea8dbfe9220293634e6c8ed8936109ef937f0000000000000000000000007342530a1a1b39df96755504e236525395cc84a593929091600401611b54565b60206040518083038186803b158015610a0757600080fd5b505af4158015610a1b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a3f919061173c565b905060005a9050816001600160a01b031663128acb083085610a648c60400151610c51565b60808d01516001600160a01b031615610a81578c60800151610aa7565b87610aa05773fffd8963efd1fc6a506488495d951d5263988d25610aa7565b6401000276a45b8d600001518e606001518f60200151604051602001610ac893929190611ae7565b6040516020818303038152906040526040518663ffffffff1660e01b8152600401610af7959493929190611bac565b6040805180830381600087803b158015610b1057600080fd5b505af1925050508015610b40575060408051601f3d908101601f19168201909252610b3d9181019061175f565b60015b610b95573d808015610b6e576040519150601f19603f3d011682016040523d82523d6000602084013e610b73565b606091505b505a82039450610b84818487610c67565b975097509750975050505050610b9b565b50505050505b9193509193565b7f0000000000000000000000009c3afddea87a726891a44c037242393d524cacfe81565b7f0000000000000000000000007342530a1a1b39df96755504e236525395cc84a581565b80516017601319909101045b919050565b60008080610c098482610d21565b9250610c16846014610dd1565b9050610c23846017610d21565b91509193909250565b516042111590565b8051606090610c4b90839060179060161901610e78565b92915050565b6000600160ff1b8210610c6357600080fd5b5090565b600080600080600080876001600160a01b0316633850c7bd6040518163ffffffff1660e01b815260040160e06040518083038186803b158015610ca957600080fd5b505afa158015610cbd573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610ce19190611994565b50939650610cf694508d9350610fc992505050565b91975095509050610d116001600160a01b038916838361104d565b9350869250505093509350935093565b600081826014011015610d70576040805162461bcd60e51b8152602060048201526012602482015271746f416464726573735f6f766572666c6f7760701b604482015290519081900360640190fd5b8160140183511015610dc1576040805162461bcd60e51b8152602060048201526015602482015274746f416464726573735f6f75744f66426f756e647360581b604482015290519081900360640190fd5b500160200151600160601b900490565b600081826003011015610e1f576040805162461bcd60e51b8152602060048201526011602482015270746f55696e7432345f6f766572666c6f7760781b604482015290519081900360640190fd5b8160030183511015610e6f576040805162461bcd60e51b8152602060048201526014602482015273746f55696e7432345f6f75744f66426f756e647360601b604482015290519081900360640190fd5b50016003015190565b60608182601f011015610ec3576040805162461bcd60e51b815260206004820152600e60248201526d736c6963655f6f766572666c6f7760901b604482015290519081900360640190fd5b828284011015610f0b576040805162461bcd60e51b815260206004820152600e60248201526d736c6963655f6f766572666c6f7760901b604482015290519081900360640190fd5b81830184511015610f57576040805162461bcd60e51b8152602060048201526011602482015270736c6963655f6f75744f66426f756e647360781b604482015290519081900360640190fd5b606082158015610f765760405191506000825260208201604052610fc0565b6040519150601f8416801560200281840101858101878315602002848b0101015b81831015610faf578051835260209283019201610f97565b5050858452601f01601f1916604052505b50949350505050565b6000806000835160601461102c57604484511015610ff95760405162461bcd60e51b815260040161096590611c05565b6004840193508380602001905181019061101391906117cf565b60405162461bcd60e51b81526004016109659190611bf2565b838060200190518101906110409190611a46565b9250925092509193909250565b60008060008060008060008060088b6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561109457600080fd5b505afa1580156110a8573d6000803e3d6000fd5b505050506040513d60208110156110be57600080fd5b5051600290810b908c900b816110d057fe5b0560020b901d905060006101008c6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561111657600080fd5b505afa15801561112a573d6000803e3d6000fd5b505050506040513d602081101561114057600080fd5b5051600290810b908d900b8161115257fe5b0560020b8161115d57fe5b079050600060088d6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561119d57600080fd5b505afa1580156111b1573d6000803e3d6000fd5b505050506040513d60208110156111c757600080fd5b5051600290810b908d900b816111d957fe5b0560020b901d905060006101008e6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561121f57600080fd5b505afa158015611233573d6000803e3d6000fd5b505050506040513d602081101561124957600080fd5b5051600290810b908e900b8161125b57fe5b0560020b8161126657fe5b07905060008160ff166001901b8f6001600160a01b0316635339c296856040518263ffffffff1660e01b8152600401808260010b815260200191505060206040518083038186803b1580156112ba57600080fd5b505afa1580156112ce573d6000803e3d6000fd5b505050506040513d60208110156112e457600080fd5b50511611801561136a57508d6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561132857600080fd5b505afa15801561133c573d6000803e3d6000fd5b505050506040513d602081101561135257600080fd5b5051600290810b908d900b8161136457fe5b0760020b155b801561137b57508b60020b8d60020b135b945060008360ff166001901b8f6001600160a01b0316635339c296876040518263ffffffff1660e01b8152600401808260010b815260200191505060206040518083038186803b1580156113ce57600080fd5b505afa1580156113e2573d6000803e3d6000fd5b505050506040513d60208110156113f857600080fd5b50511611801561147e57508d6001600160a01b031663d0c93a7c6040518163ffffffff1660e01b815260040160206040518083038186803b15801561143c57600080fd5b505afa158015611450573d6000803e3d6000fd5b505050506040513d602081101561146657600080fd5b5051600290810b908e900b8161147857fe5b0760020b155b801561148f57508b60020b8d60020b125b95508160010b8460010b12806114bb57508160010b8460010b1480156114bb57508060ff168360ff1611155b156114d1578399508297508198508096506114de565b8199508097508398508296505b505060001960ff87161b9150505b8560010b8760010b136115ae578560010b8760010b14156115135760001960ff858103161c165b6000818c6001600160a01b0316635339c2968a6040518263ffffffff1660e01b8152600401808260010b815260200191505060206040518083038186803b15801561155d57600080fd5b505afa158015611571573d6000803e3d6000fd5b505050506040513d602081101561158757600080fd5b5051169050611595816115d6565b61ffff16989098019750506001909501946000196114ec565b81156115bb576001880397505b82156115c8576001880397505b505050505050509392505050565b6000805b8215610c4b576000198301909216916001016115da565b600082601f830112611601578081fd5b813561161461160f82611d6b565b611d48565b818152846020838601011115611628578283fd5b816020850160208301379081016020019190915292915050565b8051600281900b8114610bf657600080fd5b8035610bf681611dbc565b805161ffff81168114610bf657600080fd5b600080600060608486031215611685578283fd5b83356001600160401b038082111561169b578485fd5b6116a7878388016115f1565b94506020915081860135818111156116bd578485fd5b8601601f810188136116cd578485fd5b8035828111156116d957fe5b83810292506116e9848401611d48565b8181528481019083860185850187018c1015611703578889fd5b8895505b83861015611725578035835260019590950194918601918601611707565b50979a979950505050604095909501359450505050565b60006020828403121561174d578081fd5b815161175881611dbc565b9392505050565b60008060408385031215611771578182fd5b505080516020909101519092909150565b600080600060608486031215611796578081fd5b833592506020840135915060408401356001600160401b038111156117b9578182fd5b6117c5868287016115f1565b9150509250925092565b6000602082840312156117e0578081fd5b81516001600160401b038111156117f5578182fd5b8201601f81018413611805578182fd5b805161181361160f82611d6b565b818152856020838501011115611827578384fd5b610587826020830160208601611d8c565b600060808284031215611849578081fd5b604051608081016001600160401b038111828210171561186557fe5b604052823561187381611dbc565b8152602083013561188381611dbc565b6020820152604083810135908201526060928301359281019290925250919050565b6000606082840312156118b6578081fd5b604051606081016001600160401b03811182821017156118d257fe5b60405282356118e081611dbc565b815260208301356118f081611dbc565b60208201526040928301359281019290925250919050565b600060a08284031215611919578081fd5b60405160a081016001600160401b038111828210171561193557fe5b604052823561194381611dbc565b8152602083013561195381611dbc565b602082015260408381013590820152606083013562ffffff81168114611977578283fd5b606082015261198860808401611654565b60808201529392505050565b600080600080600080600060e0888a0312156119ae578485fd5b87516119b981611dbc565b96506119c760208901611642565b95506119d56040890161165f565b94506119e36060890161165f565b93506119f16080890161165f565b925060a088015163ffffffff81168114611a09578283fd5b60c08901519092508015158114611a1e578182fd5b8091505092959891949750929550565b600060208284031215611a3f578081fd5b5051919050565b600080600060608486031215611a5a578081fd5b835192506020840151611a6c81611dbc565b9150611a7a60408501611642565b90509250925092565b600080600060608486031215611a97578081fd5b83519250602084015191506040840151611ab081611dbc565b809150509250925092565b60008151808452611ad3816020860160208601611d8c565b601f01601f19169290920160200192915050565b606093841b6001600160601b0319908116825260e89390931b6001600160e81b0319166014820152921b166017820152602b0190565b6001600160a01b0391909116815260200190565b6001600160a01b0393841681529183166020830152909116604082015260600190565b6001600160a01b03948516815292841660208401529216604082015262ffffff909116606082015260800190565b6001600160a01b039485168152928416602084015292166040820152606081019190915260800190565b6001600160a01b0386811682528515156020830152604082018590528316606082015260a060808201819052600090611be790830184611abb565b979650505050505050565b6000602082526117586020830184611abb565b60208082526010908201526f2ab732bc3832b1ba32b21032b93937b960811b604082015260600190565b6020808252818101527f4578616374206f75747075742071756f7465206e6f7420737570706f72746564604082015260600190565b90815260200190565b600060808201868352602060808185015281875180845260a0860191508289019350845b81811015611cb65784516001600160a01b031683529383019391830191600101611c91565b505084810360408601528651808252908201925081870190845b81811015611cf257825163ffffffff1685529383019391830191600101611cd0565b5050505060609290920192909252949350505050565b9384526001600160a01b0392909216602084015263ffffffff166040830152606082015260800190565b9283526020830191909152604082015260600190565b6040518181016001600160401b0381118282101715611d6357fe5b604052919050565b60006001600160401b03821115611d7e57fe5b50601f01601f191660200190565b60005b83811015611da7578181015183820152602001611d8f565b83811115611db6576000848401525b50505050565b6001600160a01b0381168114611dd157600080fd5b5056fea2646970667358221220be8f07032f0e5210561cab2ecea7f28165a5b95e4959e880ff2fbb8d5a85fa8464736f6c63430007060033