Contract name:
YodedexFactory
Optimization enabled
true
Compiler version
v0.5.16+commit.9c3226ce
Optimization runs
999999
EVM Version
default
Verified at
2022-08-20T19:11:34.029682Z
Constructor Arguments
a265627a7a7231582049bc4a7953ebb0a539de5f6ef3311f91d85729247332bc6a48bec016441804ce64736f6c63430005100032454950373132446f6d61696e28737472696e67206e616d652c737472696e672076657273696f6e2c75696e7432353620636861696e49642c6164647265737320766572696679696e67436f6e747261637429a265627a7a723158205bdf8943c0c184cd8ce54c184e5f349195ce97c7894911ed84364b642d33617664736f6c634300051000320000000000000000000000002a2e3486204c9eeeab2bef8faa3356bc19e9db6f
Arg [0] (address ) : 0x7953ebb0a539de5f6ef3311f91d85729247332bc
Contract source code
Copy Source Code
// File contracts/uniswapv2/interfaces/IYodedexFactory.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexFactory {
event PairCreated(
address indexed token0,
address indexed token1,
address pair,
uint256
);
function feeTo() external view returns (address);
function feeToSetter() external view returns (address);
function getPair(address tokenA, address tokenB)
external
view
returns (address pair);
function allPairs(uint256) external view returns (address pair);
function allPairsLength() external view returns (uint256);
function createPair(address tokenA, address tokenB)
external
returns (address pair);
function setFeeTo(address) external;
function setFeeToSetter(address) external;
}
// File contracts/uniswapv2/interfaces/IYodedexPair.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexPair {
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
event Transfer(address indexed from, address indexed to, uint256 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 (uint256);
function balanceOf(address owner) external view returns (uint256);
function allowance(address owner, address spender)
external
view
returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transfer(address to, uint256 value) external returns (bool);
function transferFrom(
address from,
address to,
uint256 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 (uint256);
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
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);
function MINIMUM_LIQUIDITY() external pure returns (uint256);
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 (uint256);
function price1CumulativeLast() external view returns (uint256);
function kLast() external view returns (uint256);
function mint(address to) external returns (uint256 liquidity);
function burn(address to)
external
returns (uint256 amount0, uint256 amount1);
function swap(
uint256 amount0Out,
uint256 amount1Out,
address to,
bytes calldata data
) external;
function skim(address to) external;
function sync() external;
function initialize(address, address) external;
}
// File contracts/uniswapv2/interfaces/IYodedexERC20.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexERC20 {
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
event Transfer(address indexed from, address indexed to, uint256 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 (uint256);
function balanceOf(address owner) external view returns (uint256);
function allowance(address owner, address spender)
external
view
returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transfer(address to, uint256 value) external returns (bool);
function transferFrom(
address from,
address to,
uint256 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 (uint256);
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
}
// File contracts/uniswapv2/libraries/SafeMath.sol
pragma solidity =0.5.16;
// a library for performing overflow-safe math, courtesy of DappHub (https://github.com/dapphub/ds-math)
library SafeMath {
function add(uint x, uint y) internal pure returns (uint z) {
require((z = x + y) >= x, 'ds-math-add-overflow');
}
function sub(uint x, uint y) internal pure returns (uint z) {
require((z = x - y) <= x, 'ds-math-sub-underflow');
}
function mul(uint x, uint y) internal pure returns (uint z) {
require(y == 0 || (z = x * y) / y == x, 'ds-math-mul-overflow');
}
}
// File contracts/uniswapv2/YodedexERC20.sol
pragma solidity =0.5.16;
contract YodedexERC20 is IYodedexERC20 {
using SafeMath for uint256;
string public constant name = "Yodedex LP";
string public constant symbol = "YLP";
uint8 public constant decimals = 18;
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
mapping(address => mapping(address => uint256)) public allowance;
bytes32 public DOMAIN_SEPARATOR;
// keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
bytes32 public constant PERMIT_TYPEHASH =
0x6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9;
mapping(address => uint256) public nonces;
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
event Transfer(address indexed from, address indexed to, uint256 value);
constructor() public {
uint256 chainId;
assembly {
chainId := chainid
}
DOMAIN_SEPARATOR = keccak256(
abi.encode(
keccak256(
"EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
),
keccak256(bytes(name)),
keccak256(bytes("1")),
chainId,
address(this)
)
);
}
function _mint(address to, uint256 value) internal {
totalSupply = totalSupply.add(value);
balanceOf[to] = balanceOf[to].add(value);
emit Transfer(address(0), to, value);
}
function _burn(address from, uint256 value) internal {
balanceOf[from] = balanceOf[from].sub(value);
totalSupply = totalSupply.sub(value);
emit Transfer(from, address(0), value);
}
function _approve(
address owner,
address spender,
uint256 value
) private {
allowance[owner][spender] = value;
emit Approval(owner, spender, value);
}
function _transfer(
address from,
address to,
uint256 value
) private {
balanceOf[from] = balanceOf[from].sub(value);
balanceOf[to] = balanceOf[to].add(value);
emit Transfer(from, to, value);
}
function approve(address spender, uint256 value) external returns (bool) {
_approve(msg.sender, spender, value);
return true;
}
function transfer(address to, uint256 value) external returns (bool) {
_transfer(msg.sender, to, value);
return true;
}
function transferFrom(
address from,
address to,
uint256 value
) external returns (bool) {
if (allowance[from][msg.sender] != uint256(-1)) {
allowance[from][msg.sender] = allowance[from][msg.sender].sub(
value
);
}
_transfer(from, to, value);
return true;
}
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external {
require(deadline >= block.timestamp, "Yodedex: EXPIRED");
bytes32 digest = keccak256(
abi.encodePacked(
"\x19\x01",
DOMAIN_SEPARATOR,
keccak256(
abi.encode(
PERMIT_TYPEHASH,
owner,
spender,
value,
nonces[owner]++,
deadline
)
)
)
);
address recoveredAddress = ecrecover(digest, v, r, s);
require(
recoveredAddress != address(0) && recoveredAddress == owner,
"Yodedex: INVALID_SIGNATURE"
);
_approve(owner, spender, value);
}
}
// File contracts/uniswapv2/libraries/Math.sol
pragma solidity =0.5.16;
// a library for performing various math operations
library Math {
function min(uint x, uint y) internal pure returns (uint z) {
z = x < y ? x : y;
}
// babylonian method (https://en.wikipedia.org/wiki/Methods_of_computing_square_roots#Babylonian_method)
function sqrt(uint y) internal pure returns (uint z) {
if (y > 3) {
z = y;
uint x = y / 2 + 1;
while (x < z) {
z = x;
x = (y / x + x) / 2;
}
} else if (y != 0) {
z = 1;
}
}
}
// File contracts/uniswapv2/libraries/UQ112x112.sol
pragma solidity =0.5.16;
// a library for handling binary fixed point numbers (https://en.wikipedia.org/wiki/Q_(number_format))
// range: [0, 2**112 - 1]
// resolution: 1 / 2**112
library UQ112x112 {
uint224 constant Q112 = 2**112;
// encode a uint112 as a UQ112x112
function encode(uint112 y) internal pure returns (uint224 z) {
z = uint224(y) * Q112; // never overflows
}
// divide a UQ112x112 by a uint112, returning a UQ112x112
function uqdiv(uint224 x, uint112 y) internal pure returns (uint224 z) {
z = x / uint224(y);
}
}
// File contracts/uniswapv2/interfaces/IERC20.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IERC20 {
event Approval(address indexed owner, address indexed spender, uint256 value);
event Transfer(address indexed from, address indexed to, uint256 value);
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
function totalSupply() external view returns (uint256);
function balanceOf(address owner) external view returns (uint256);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transfer(address to, uint256 value) external returns (bool);
function transferFrom(address from, address to, uint256 value) external returns (bool);
}
// File contracts/uniswapv2/interfaces/IYodedexCallee.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexCallee {
function YodedexCall(
address sender,
uint amount0,
uint amount1,
bytes calldata data
) external;
}
// File contracts/uniswapv2/YodedexPair.sol
pragma solidity =0.5.16;
contract YodedexPair is IYodedexPair, YodedexERC20 {
using SafeMath for uint256;
using UQ112x112 for uint224;
uint256 public constant MINIMUM_LIQUIDITY = 10**3;
bytes4 private constant SELECTOR =
bytes4(keccak256(bytes("transfer(address,uint256)")));
address public factory;
address public token0;
address public token1;
uint112 private reserve0; // uses single storage slot, accessible via getReserves
uint112 private reserve1; // uses single storage slot, accessible via getReserves
uint32 private blockTimestampLast; // uses single storage slot, accessible via getReserves
uint256 public price0CumulativeLast;
uint256 public price1CumulativeLast;
uint256 public kLast; // reserve0 * reserve1, as of immediately after the most recent liquidity event
uint32 public swapFee = 50; // uses 0.5% default
uint40 public devFee = 5151515152; // uses 0.33% default from swap fee
uint256 private unlocked = 1;
modifier lock() {
require(unlocked == 1, "Yodedex: LOCKED");
unlocked = 0;
_;
unlocked = 1;
}
function getReserves()
public
view
returns (
uint112 _reserve0,
uint112 _reserve1,
uint32 _blockTimestampLast
)
{
_reserve0 = reserve0;
_reserve1 = reserve1;
_blockTimestampLast = blockTimestampLast;
}
function _safeTransfer(
address token,
address to,
uint256 value
) private {
(bool success, bytes memory data) = token.call(
abi.encodeWithSelector(SELECTOR, to, value)
);
require(
success && (data.length == 0 || abi.decode(data, (bool))),
"Yodedex: TRANSFER_FAILED"
);
}
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);
constructor() public {
factory = msg.sender;
}
// called once by the factory at time of deployment
function initialize(address _token0, address _token1) external {
require(msg.sender == factory, "Yodedex: FORBIDDEN"); // sufficient check
token0 = _token0;
token1 = _token1;
}
function setSwapFee(uint32 _swapFee) external {
require(_swapFee > 0, "Yodedex: lower then 0");
require(msg.sender == factory, "Yodedex: FORBIDDEN");
require(_swapFee <= 1000, "Yodedex: FORBIDDEN_FEE");
swapFee = _swapFee;
}
function setDevFee(uint32 _devFee) external {
require(_devFee > 0, "Yodedex: lower then 0");
require(msg.sender == factory, "Yodedex: FORBIDDEN");
devFee = _devFee;
}
// update reserves and, on the first call per block, price accumulators
function _update(
uint256 balance0,
uint256 balance1,
uint112 _reserve0,
uint112 _reserve1
) private {
require(
balance0 <= uint112(-1) && balance1 <= uint112(-1),
"Yodedex: OVERFLOW"
);
uint32 blockTimestamp = uint32(block.timestamp % 2**32);
uint32 timeElapsed = blockTimestamp - blockTimestampLast; // overflow is desired
if (timeElapsed > 0 && _reserve0 != 0 && _reserve1 != 0) {
// * never overflows, and + overflow is desired
price0CumulativeLast +=
uint256(UQ112x112.encode(_reserve1).uqdiv(_reserve0)) *
timeElapsed;
price1CumulativeLast +=
uint256(UQ112x112.encode(_reserve0).uqdiv(_reserve1)) *
timeElapsed;
}
reserve0 = uint112(balance0);
reserve1 = uint112(balance1);
blockTimestampLast = blockTimestamp;
emit Sync(reserve0, reserve1);
}
// if fee is on, mint liquidity equivalent to 1/6th of the growth in sqrt(k)
function _mintFee(uint112 _reserve0, uint112 _reserve1)
private
returns (bool feeOn)
{
address feeTo = IYodedexFactory(factory).feeTo();
feeOn = feeTo != address(0);
uint256 _kLast = kLast; // gas savings
if (feeOn) {
if (_kLast != 0) {
uint256 rootK = Math.sqrt(uint256(_reserve0).mul(_reserve1));
uint256 rootKLast = Math.sqrt(_kLast);
if (rootK > rootKLast) {
uint256 numerator = totalSupply.mul(rootK.sub(rootKLast));
uint256 denominator = (rootK.mul(devFee) / 10000000000).add(
rootKLast
);
uint256 liquidity = numerator / denominator;
if (liquidity > 0) _mint(feeTo, liquidity);
}
}
} else if (_kLast != 0) {
kLast = 0;
}
}
// this low-level function should be called from a contract which performs important safety checks
function mint(address to) external lock returns (uint256 liquidity) {
(uint112 _reserve0, uint112 _reserve1, ) = getReserves(); // gas savings
uint256 balance0 = IERC20(token0).balanceOf(address(this));
uint256 balance1 = IERC20(token1).balanceOf(address(this));
uint256 amount0 = balance0.sub(_reserve0);
uint256 amount1 = balance1.sub(_reserve1);
bool feeOn = _mintFee(_reserve0, _reserve1);
uint256 _totalSupply = totalSupply; // gas savings, must be defined here since totalSupply can update in _mintFee
if (_totalSupply == 0) {
liquidity = Math.sqrt(amount0.mul(amount1)).sub(MINIMUM_LIQUIDITY);
_mint(address(0), MINIMUM_LIQUIDITY); // permanently lock the first MINIMUM_LIQUIDITY tokens
} else {
liquidity = Math.min(
amount0.mul(_totalSupply) / _reserve0,
amount1.mul(_totalSupply) / _reserve1
);
}
require(liquidity > 0, "Yodedex: INSUFFICIENT_LIQUIDITY_MINTED");
_mint(to, liquidity);
_update(balance0, balance1, _reserve0, _reserve1);
if (feeOn) kLast = uint256(reserve0).mul(reserve1); // reserve0 and reserve1 are up-to-date
emit Mint(msg.sender, amount0, amount1);
}
// this low-level function should be called from a contract which performs important safety checks
function burn(address to)
external
lock
returns (uint256 amount0, uint256 amount1)
{
(uint112 _reserve0, uint112 _reserve1, ) = getReserves(); // gas savings
address _token0 = token0; // gas savings
address _token1 = token1; // gas savings
uint256 balance0 = IERC20(_token0).balanceOf(address(this));
uint256 balance1 = IERC20(_token1).balanceOf(address(this));
uint256 liquidity = balanceOf[address(this)];
bool feeOn = _mintFee(_reserve0, _reserve1);
uint256 _totalSupply = totalSupply; // gas savings, must be defined here since totalSupply can update in _mintFee
amount0 = liquidity.mul(balance0) / _totalSupply; // using balances ensures pro-rata distribution
amount1 = liquidity.mul(balance1) / _totalSupply; // using balances ensures pro-rata distribution
require(
amount0 > 0 && amount1 > 0,
"Yodedex: INSUFFICIENT_LIQUIDITY_BURNED"
);
_burn(address(this), liquidity);
_safeTransfer(_token0, to, amount0);
_safeTransfer(_token1, to, amount1);
balance0 = IERC20(_token0).balanceOf(address(this));
balance1 = IERC20(_token1).balanceOf(address(this));
_update(balance0, balance1, _reserve0, _reserve1);
if (feeOn) kLast = uint256(reserve0).mul(reserve1); // reserve0 and reserve1 are up-to-date
emit Burn(msg.sender, amount0, amount1, to);
}
// this low-level function should be called from a contract which performs important safety checks
function swap(
uint256 amount0Out,
uint256 amount1Out,
address to,
bytes calldata data
) external lock {
require(
amount0Out > 0 || amount1Out > 0,
"Yodedex: INSUFFICIENT_OUTPUT_AMOUNT"
);
(uint112 _reserve0, uint112 _reserve1, ) = getReserves(); // gas savings
require(
amount0Out < _reserve0 && amount1Out < _reserve1,
"Yodedex: INSUFFICIENT_LIQUIDITY"
);
uint256 balance0;
uint256 balance1;
{
// scope for _token{0,1}, avoids stack too deep errors
address _token0 = token0;
address _token1 = token1;
require(to != _token0 && to != _token1, "Yodedex: INVALID_TO");
if (amount0Out > 0) _safeTransfer(_token0, to, amount0Out); // optimistically transfer tokens
if (amount1Out > 0) _safeTransfer(_token1, to, amount1Out); // optimistically transfer tokens
if (data.length > 0)
IYodedexCallee(to).YodedexCall(
msg.sender,
amount0Out,
amount1Out,
data
);
balance0 = IERC20(_token0).balanceOf(address(this));
balance1 = IERC20(_token1).balanceOf(address(this));
}
uint256 amount0In = balance0 > _reserve0 - amount0Out
? balance0 - (_reserve0 - amount0Out)
: 0;
uint256 amount1In = balance1 > _reserve1 - amount1Out
? balance1 - (_reserve1 - amount1Out)
: 0;
require(
amount0In > 0 || amount1In > 0,
"Yodedex: INSUFFICIENT_INPUT_AMOUNT"
);
{
uint _swapFee = swapFee;
// scope for reserve{0,1}Adjusted, avoids stack too deep errors
uint balance0Adjusted = balance0.mul(10000).sub(
amount0In.mul(_swapFee)
);
uint balance1Adjusted = balance1.mul(10000).sub(
amount1In.mul(_swapFee)
);
require(
balance0Adjusted.mul(balance1Adjusted) >=
uint256(_reserve0).mul(_reserve1).mul(10000**2),
"Yodedex: K"
);
}
_update(balance0, balance1, _reserve0, _reserve1);
emit Swap(msg.sender, amount0In, amount1In, amount0Out, amount1Out, to);
}
// force balances to match reserves
function skim(address to) external lock {
address _token0 = token0; // gas savings
address _token1 = token1; // gas savings
_safeTransfer(
_token0,
to,
IERC20(_token0).balanceOf(address(this)).sub(reserve0)
);
_safeTransfer(
_token1,
to,
IERC20(_token1).balanceOf(address(this)).sub(reserve1)
);
}
// force reserves to match balances
function sync() external lock {
_update(
IERC20(token0).balanceOf(address(this)),
IERC20(token1).balanceOf(address(this)),
reserve0,
reserve1
);
}
}
// File contracts/uniswapv2/YodedexFactory.sol
pragma solidity =0.5.16;
contract YodedexFactory is IYodedexFactory {
bytes32 public constant INIT_CODE_PAIR_HASH =
keccak256(abi.encodePacked(type(YodedexPair).creationCode));
address public feeTo;
address public feeToSetter;
mapping(address => mapping(address => address)) public getPair;
address[] public allPairs;
event PairCreated(
address indexed token0,
address indexed token1,
address pair,
uint256
);
constructor(address _feeToSetter) public {
feeToSetter = _feeToSetter;
}
function allPairsLength() external view returns (uint256) {
return allPairs.length;
}
function createPair(address tokenA, address tokenB)
external
returns (address pair)
{
require(tokenA != tokenB, "Yodedex: IDENTICAL_ADDRESSES");
(address token0, address token1) = tokenA < tokenB
? (tokenA, tokenB)
: (tokenB, tokenA);
require(token0 != address(0), "Yodedex: ZERO_ADDRESS");
require(getPair[token0][token1] == address(0), "Yodedex: PAIR_EXISTS"); // single check is sufficient
bytes memory bytecode = type(YodedexPair).creationCode;
bytes32 salt = keccak256(abi.encodePacked(token0, token1));
assembly {
pair := create2(0, add(bytecode, 32), mload(bytecode), salt)
}
IYodedexPair(pair).initialize(token0, token1);
getPair[token0][token1] = pair;
getPair[token1][token0] = pair; // populate mapping in the reverse direction
allPairs.push(pair);
emit PairCreated(token0, token1, pair, allPairs.length);
}
function setFeeTo(address _feeTo) external {
require(msg.sender == feeToSetter, "Yodedex: FORBIDDEN");
feeTo = _feeTo;
}
function setFeeToSetter(address _feeToSetter) external {
require(msg.sender == feeToSetter, "Yodedex: FORBIDDEN");
feeToSetter = _feeToSetter;
}
function setDevFee(address _pair, uint8 _devFee) external {
require(msg.sender == feeToSetter, "Yodedex: FORBIDDEN");
require(_devFee > 0, "Yodedex: FORBIDDEN_FEE");
YodedexPair(_pair).setDevFee(_devFee);
}
function setSwapFee(address _pair, uint32 _swapFee) external {
require(msg.sender == feeToSetter, "Yodedex: FORBIDDEN");
YodedexPair(_pair).setSwapFee(_swapFee);
}
}
./contracts/uniswapv2/interfaces/IYodedexFactory.sol
Copy Source Code
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexFactory {
event PairCreated(
address indexed token0,
address indexed token1,
address pair,
uint256
);
function feeTo() external view returns (address);
function feeToSetter() external view returns (address);
function getPair(address tokenA, address tokenB)
external
view
returns (address pair);
function allPairs(uint256) external view returns (address pair);
function allPairsLength() external view returns (uint256);
function createPair(address tokenA, address tokenB)
external
returns (address pair);
function setFeeTo(address) external;
function setFeeToSetter(address) external;
}
./contracts/uniswapv2/YodedexERC20.sol
Copy Source Code
pragma solidity =0.5.16;
import "./interfaces/IYodedexERC20.sol";
import "./libraries/SafeMath.sol";
contract YodedexERC20 is IYodedexERC20 {
using SafeMath for uint256;
string public constant name = "Yodedex LP";
string public constant symbol = "YLP";
uint8 public constant decimals = 18;
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
mapping(address => mapping(address => uint256)) public allowance;
bytes32 public DOMAIN_SEPARATOR;
// keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
bytes32 public constant PERMIT_TYPEHASH =
0x6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9;
mapping(address => uint256) public nonces;
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
event Transfer(address indexed from, address indexed to, uint256 value);
constructor() public {
uint256 chainId;
assembly {
chainId := chainid
}
DOMAIN_SEPARATOR = keccak256(
abi.encode(
keccak256(
"EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
),
keccak256(bytes(name)),
keccak256(bytes("1")),
chainId,
address(this)
)
);
}
function _mint(address to, uint256 value) internal {
totalSupply = totalSupply.add(value);
balanceOf[to] = balanceOf[to].add(value);
emit Transfer(address(0), to, value);
}
function _burn(address from, uint256 value) internal {
balanceOf[from] = balanceOf[from].sub(value);
totalSupply = totalSupply.sub(value);
emit Transfer(from, address(0), value);
}
function _approve(
address owner,
address spender,
uint256 value
) private {
allowance[owner][spender] = value;
emit Approval(owner, spender, value);
}
function _transfer(
address from,
address to,
uint256 value
) private {
balanceOf[from] = balanceOf[from].sub(value);
balanceOf[to] = balanceOf[to].add(value);
emit Transfer(from, to, value);
}
function approve(address spender, uint256 value) external returns (bool) {
_approve(msg.sender, spender, value);
return true;
}
function transfer(address to, uint256 value) external returns (bool) {
_transfer(msg.sender, to, value);
return true;
}
function transferFrom(
address from,
address to,
uint256 value
) external returns (bool) {
if (allowance[from][msg.sender] != uint256(-1)) {
allowance[from][msg.sender] = allowance[from][msg.sender].sub(
value
);
}
_transfer(from, to, value);
return true;
}
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external {
require(deadline >= block.timestamp, "Yodedex: EXPIRED");
bytes32 digest = keccak256(
abi.encodePacked(
"\x19\x01",
DOMAIN_SEPARATOR,
keccak256(
abi.encode(
PERMIT_TYPEHASH,
owner,
spender,
value,
nonces[owner]++,
deadline
)
)
)
);
address recoveredAddress = ecrecover(digest, v, r, s);
require(
recoveredAddress != address(0) && recoveredAddress == owner,
"Yodedex: INVALID_SIGNATURE"
);
_approve(owner, spender, value);
}
}
./contracts/uniswapv2/YodedexPair.sol
Copy Source Code
pragma solidity =0.5.16;
import "./interfaces/IYodedexPair.sol";
import "./YodedexERC20.sol";
import "./libraries/Math.sol";
import "./libraries/UQ112x112.sol";
import "./interfaces/IERC20.sol";
import "./interfaces/IYodedexFactory.sol";
import "./interfaces/IYodedexCallee.sol";
contract YodedexPair is IYodedexPair, YodedexERC20 {
using SafeMath for uint256;
using UQ112x112 for uint224;
uint256 public constant MINIMUM_LIQUIDITY = 10**3;
bytes4 private constant SELECTOR =
bytes4(keccak256(bytes("transfer(address,uint256)")));
address public factory;
address public token0;
address public token1;
uint112 private reserve0; // uses single storage slot, accessible via getReserves
uint112 private reserve1; // uses single storage slot, accessible via getReserves
uint32 private blockTimestampLast; // uses single storage slot, accessible via getReserves
uint256 public price0CumulativeLast;
uint256 public price1CumulativeLast;
uint256 public kLast; // reserve0 * reserve1, as of immediately after the most recent liquidity event
uint32 public swapFee = 50; // uses 0.5% default
uint40 public devFee = 5151515152; // uses 0.33% default from swap fee
uint256 private unlocked = 1;
modifier lock() {
require(unlocked == 1, "Yodedex: LOCKED");
unlocked = 0;
_;
unlocked = 1;
}
function getReserves()
public
view
returns (
uint112 _reserve0,
uint112 _reserve1,
uint32 _blockTimestampLast
)
{
_reserve0 = reserve0;
_reserve1 = reserve1;
_blockTimestampLast = blockTimestampLast;
}
function _safeTransfer(
address token,
address to,
uint256 value
) private {
(bool success, bytes memory data) = token.call(
abi.encodeWithSelector(SELECTOR, to, value)
);
require(
success && (data.length == 0 || abi.decode(data, (bool))),
"Yodedex: TRANSFER_FAILED"
);
}
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);
constructor() public {
factory = msg.sender;
}
// called once by the factory at time of deployment
function initialize(address _token0, address _token1) external {
require(msg.sender == factory, "Yodedex: FORBIDDEN"); // sufficient check
token0 = _token0;
token1 = _token1;
}
function setSwapFee(uint32 _swapFee) external {
require(_swapFee > 0, "Yodedex: lower then 0");
require(msg.sender == factory, "Yodedex: FORBIDDEN");
require(_swapFee <= 1000, "Yodedex: FORBIDDEN_FEE");
swapFee = _swapFee;
}
function setDevFee(uint32 _devFee) external {
require(_devFee > 0, "Yodedex: lower then 0");
require(msg.sender == factory, "Yodedex: FORBIDDEN");
devFee = _devFee;
}
// update reserves and, on the first call per block, price accumulators
function _update(
uint256 balance0,
uint256 balance1,
uint112 _reserve0,
uint112 _reserve1
) private {
require(
balance0 <= uint112(-1) && balance1 <= uint112(-1),
"Yodedex: OVERFLOW"
);
uint32 blockTimestamp = uint32(block.timestamp % 2**32);
uint32 timeElapsed = blockTimestamp - blockTimestampLast; // overflow is desired
if (timeElapsed > 0 && _reserve0 != 0 && _reserve1 != 0) {
// * never overflows, and + overflow is desired
price0CumulativeLast +=
uint256(UQ112x112.encode(_reserve1).uqdiv(_reserve0)) *
timeElapsed;
price1CumulativeLast +=
uint256(UQ112x112.encode(_reserve0).uqdiv(_reserve1)) *
timeElapsed;
}
reserve0 = uint112(balance0);
reserve1 = uint112(balance1);
blockTimestampLast = blockTimestamp;
emit Sync(reserve0, reserve1);
}
// if fee is on, mint liquidity equivalent to 1/6th of the growth in sqrt(k)
function _mintFee(uint112 _reserve0, uint112 _reserve1)
private
returns (bool feeOn)
{
address feeTo = IYodedexFactory(factory).feeTo();
feeOn = feeTo != address(0);
uint256 _kLast = kLast; // gas savings
if (feeOn) {
if (_kLast != 0) {
uint256 rootK = Math.sqrt(uint256(_reserve0).mul(_reserve1));
uint256 rootKLast = Math.sqrt(_kLast);
if (rootK > rootKLast) {
uint256 numerator = totalSupply.mul(rootK.sub(rootKLast));
uint256 denominator = (rootK.mul(devFee) / 10000000000).add(
rootKLast
);
uint256 liquidity = numerator / denominator;
if (liquidity > 0) _mint(feeTo, liquidity);
}
}
} else if (_kLast != 0) {
kLast = 0;
}
}
// this low-level function should be called from a contract which performs important safety checks
function mint(address to) external lock returns (uint256 liquidity) {
(uint112 _reserve0, uint112 _reserve1, ) = getReserves(); // gas savings
uint256 balance0 = IERC20(token0).balanceOf(address(this));
uint256 balance1 = IERC20(token1).balanceOf(address(this));
uint256 amount0 = balance0.sub(_reserve0);
uint256 amount1 = balance1.sub(_reserve1);
bool feeOn = _mintFee(_reserve0, _reserve1);
uint256 _totalSupply = totalSupply; // gas savings, must be defined here since totalSupply can update in _mintFee
if (_totalSupply == 0) {
liquidity = Math.sqrt(amount0.mul(amount1)).sub(MINIMUM_LIQUIDITY);
_mint(address(0), MINIMUM_LIQUIDITY); // permanently lock the first MINIMUM_LIQUIDITY tokens
} else {
liquidity = Math.min(
amount0.mul(_totalSupply) / _reserve0,
amount1.mul(_totalSupply) / _reserve1
);
}
require(liquidity > 0, "Yodedex: INSUFFICIENT_LIQUIDITY_MINTED");
_mint(to, liquidity);
_update(balance0, balance1, _reserve0, _reserve1);
if (feeOn) kLast = uint256(reserve0).mul(reserve1); // reserve0 and reserve1 are up-to-date
emit Mint(msg.sender, amount0, amount1);
}
// this low-level function should be called from a contract which performs important safety checks
function burn(address to)
external
lock
returns (uint256 amount0, uint256 amount1)
{
(uint112 _reserve0, uint112 _reserve1, ) = getReserves(); // gas savings
address _token0 = token0; // gas savings
address _token1 = token1; // gas savings
uint256 balance0 = IERC20(_token0).balanceOf(address(this));
uint256 balance1 = IERC20(_token1).balanceOf(address(this));
uint256 liquidity = balanceOf[address(this)];
bool feeOn = _mintFee(_reserve0, _reserve1);
uint256 _totalSupply = totalSupply; // gas savings, must be defined here since totalSupply can update in _mintFee
amount0 = liquidity.mul(balance0) / _totalSupply; // using balances ensures pro-rata distribution
amount1 = liquidity.mul(balance1) / _totalSupply; // using balances ensures pro-rata distribution
require(
amount0 > 0 && amount1 > 0,
"Yodedex: INSUFFICIENT_LIQUIDITY_BURNED"
);
_burn(address(this), liquidity);
_safeTransfer(_token0, to, amount0);
_safeTransfer(_token1, to, amount1);
balance0 = IERC20(_token0).balanceOf(address(this));
balance1 = IERC20(_token1).balanceOf(address(this));
_update(balance0, balance1, _reserve0, _reserve1);
if (feeOn) kLast = uint256(reserve0).mul(reserve1); // reserve0 and reserve1 are up-to-date
emit Burn(msg.sender, amount0, amount1, to);
}
// this low-level function should be called from a contract which performs important safety checks
function swap(
uint256 amount0Out,
uint256 amount1Out,
address to,
bytes calldata data
) external lock {
require(
amount0Out > 0 || amount1Out > 0,
"Yodedex: INSUFFICIENT_OUTPUT_AMOUNT"
);
(uint112 _reserve0, uint112 _reserve1, ) = getReserves(); // gas savings
require(
amount0Out < _reserve0 && amount1Out < _reserve1,
"Yodedex: INSUFFICIENT_LIQUIDITY"
);
uint256 balance0;
uint256 balance1;
{
// scope for _token{0,1}, avoids stack too deep errors
address _token0 = token0;
address _token1 = token1;
require(to != _token0 && to != _token1, "Yodedex: INVALID_TO");
if (amount0Out > 0) _safeTransfer(_token0, to, amount0Out); // optimistically transfer tokens
if (amount1Out > 0) _safeTransfer(_token1, to, amount1Out); // optimistically transfer tokens
if (data.length > 0)
IYodedexCallee(to).YodedexCall(
msg.sender,
amount0Out,
amount1Out,
data
);
balance0 = IERC20(_token0).balanceOf(address(this));
balance1 = IERC20(_token1).balanceOf(address(this));
}
uint256 amount0In = balance0 > _reserve0 - amount0Out
? balance0 - (_reserve0 - amount0Out)
: 0;
uint256 amount1In = balance1 > _reserve1 - amount1Out
? balance1 - (_reserve1 - amount1Out)
: 0;
require(
amount0In > 0 || amount1In > 0,
"Yodedex: INSUFFICIENT_INPUT_AMOUNT"
);
{
uint _swapFee = swapFee;
// scope for reserve{0,1}Adjusted, avoids stack too deep errors
uint balance0Adjusted = balance0.mul(10000).sub(
amount0In.mul(_swapFee)
);
uint balance1Adjusted = balance1.mul(10000).sub(
amount1In.mul(_swapFee)
);
require(
balance0Adjusted.mul(balance1Adjusted) >=
uint256(_reserve0).mul(_reserve1).mul(10000**2),
"Yodedex: K"
);
}
_update(balance0, balance1, _reserve0, _reserve1);
emit Swap(msg.sender, amount0In, amount1In, amount0Out, amount1Out, to);
}
// force balances to match reserves
function skim(address to) external lock {
address _token0 = token0; // gas savings
address _token1 = token1; // gas savings
_safeTransfer(
_token0,
to,
IERC20(_token0).balanceOf(address(this)).sub(reserve0)
);
_safeTransfer(
_token1,
to,
IERC20(_token1).balanceOf(address(this)).sub(reserve1)
);
}
// force reserves to match balances
function sync() external lock {
_update(
IERC20(token0).balanceOf(address(this)),
IERC20(token1).balanceOf(address(this)),
reserve0,
reserve1
);
}
}
./contracts/uniswapv2/interfaces/IERC20.sol
Copy Source Code
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IERC20 {
event Approval(address indexed owner, address indexed spender, uint256 value);
event Transfer(address indexed from, address indexed to, uint256 value);
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
function totalSupply() external view returns (uint256);
function balanceOf(address owner) external view returns (uint256);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transfer(address to, uint256 value) external returns (bool);
function transferFrom(address from, address to, uint256 value) external returns (bool);
}
./contracts/uniswapv2/interfaces/IYodedexCallee.sol
Copy Source Code
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexCallee {
function YodedexCall(
address sender,
uint amount0,
uint amount1,
bytes calldata data
) external;
}
./contracts/uniswapv2/interfaces/IYodedexERC20.sol
Copy Source Code
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexERC20 {
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
event Transfer(address indexed from, address indexed to, uint256 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 (uint256);
function balanceOf(address owner) external view returns (uint256);
function allowance(address owner, address spender)
external
view
returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transfer(address to, uint256 value) external returns (bool);
function transferFrom(
address from,
address to,
uint256 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 (uint256);
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
}
./contracts/uniswapv2/interfaces/IYodedexPair.sol
Copy Source Code
// SPDX-License-Identifier: MIT
pragma solidity >=0.5.0;
interface IYodedexPair {
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
event Transfer(address indexed from, address indexed to, uint256 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 (uint256);
function balanceOf(address owner) external view returns (uint256);
function allowance(address owner, address spender)
external
view
returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transfer(address to, uint256 value) external returns (bool);
function transferFrom(
address from,
address to,
uint256 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 (uint256);
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
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);
function MINIMUM_LIQUIDITY() external pure returns (uint256);
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 (uint256);
function price1CumulativeLast() external view returns (uint256);
function kLast() external view returns (uint256);
function mint(address to) external returns (uint256 liquidity);
function burn(address to)
external
returns (uint256 amount0, uint256 amount1);
function swap(
uint256 amount0Out,
uint256 amount1Out,
address to,
bytes calldata data
) external;
function skim(address to) external;
function sync() external;
function initialize(address, address) external;
}
./contracts/uniswapv2/libraries/Math.sol
Copy Source Code
pragma solidity =0.5.16;
// a library for performing various math operations
library Math {
function min(uint x, uint y) internal pure returns (uint z) {
z = x < y ? x : y;
}
// babylonian method (https://en.wikipedia.org/wiki/Methods_of_computing_square_roots#Babylonian_method)
function sqrt(uint y) internal pure returns (uint z) {
if (y > 3) {
z = y;
uint x = y / 2 + 1;
while (x < z) {
z = x;
x = (y / x + x) / 2;
}
} else if (y != 0) {
z = 1;
}
}
}
./contracts/uniswapv2/libraries/SafeMath.sol
Copy Source Code
pragma solidity =0.5.16;
// a library for performing overflow-safe math, courtesy of DappHub (https://github.com/dapphub/ds-math)
library SafeMath {
function add(uint x, uint y) internal pure returns (uint z) {
require((z = x + y) >= x, 'ds-math-add-overflow');
}
function sub(uint x, uint y) internal pure returns (uint z) {
require((z = x - y) <= x, 'ds-math-sub-underflow');
}
function mul(uint x, uint y) internal pure returns (uint z) {
require(y == 0 || (z = x * y) / y == x, 'ds-math-mul-overflow');
}
}
./contracts/uniswapv2/libraries/UQ112x112.sol
Copy Source Code
pragma solidity =0.5.16;
// a library for handling binary fixed point numbers (https://en.wikipedia.org/wiki/Q_(number_format))
// range: [0, 2**112 - 1]
// resolution: 1 / 2**112
library UQ112x112 {
uint224 constant Q112 = 2**112;
// encode a uint112 as a UQ112x112
function encode(uint112 y) internal pure returns (uint224 z) {
z = uint224(y) * Q112; // never overflows
}
// divide a UQ112x112 by a uint112, returning a UQ112x112
function uqdiv(uint224 x, uint112 y) internal pure returns (uint224 z) {
z = x / uint224(y);
}
}
Contract ABI
Copy ABI
[{"type":"constructor","stateMutability":"nonpayable","payable":false,"inputs":[{"type":"address","name":"_feeToSetter","internalType":"address"}]},{"type":"event","name":"PairCreated","inputs":[{"type":"address","name":"token0","internalType":"address","indexed":true},{"type":"address","name":"token1","internalType":"address","indexed":true},{"type":"address","name":"pair","internalType":"address","indexed":false},{"type":"uint256","name":"","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"function","stateMutability":"view","payable":false,"outputs":[{"type":"bytes32","name":"","internalType":"bytes32"}],"name":"INIT_CODE_PAIR_HASH","inputs":[],"constant":true},{"type":"function","stateMutability":"view","payable":false,"outputs":[{"type":"address","name":"","internalType":"address"}],"name":"allPairs","inputs":[{"type":"uint256","name":"","internalType":"uint256"}],"constant":true},{"type":"function","stateMutability":"view","payable":false,"outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"allPairsLength","inputs":[],"constant":true},{"type":"function","stateMutability":"nonpayable","payable":false,"outputs":[{"type":"address","name":"pair","internalType":"address"}],"name":"createPair","inputs":[{"type":"address","name":"tokenA","internalType":"address"},{"type":"address","name":"tokenB","internalType":"address"}],"constant":false},{"type":"function","stateMutability":"view","payable":false,"outputs":[{"type":"address","name":"","internalType":"address"}],"name":"feeTo","inputs":[],"constant":true},{"type":"function","stateMutability":"view","payable":false,"outputs":[{"type":"address","name":"","internalType":"address"}],"name":"feeToSetter","inputs":[],"constant":true},{"type":"function","stateMutability":"view","payable":false,"outputs":[{"type":"address","name":"","internalType":"address"}],"name":"getPair","inputs":[{"type":"address","name":"","internalType":"address"},{"type":"address","name":"","internalType":"address"}],"constant":true},{"type":"function","stateMutability":"nonpayable","payable":false,"outputs":[],"name":"setDevFee","inputs":[{"type":"address","name":"_pair","internalType":"address"},{"type":"uint8","name":"_devFee","internalType":"uint8"}],"constant":false},{"type":"function","stateMutability":"nonpayable","payable":false,"outputs":[],"name":"setFeeTo","inputs":[{"type":"address","name":"_feeTo","internalType":"address"}],"constant":false},{"type":"function","stateMutability":"nonpayable","payable":false,"outputs":[],"name":"setFeeToSetter","inputs":[{"type":"address","name":"_feeToSetter","internalType":"address"}],"constant":false},{"type":"function","stateMutability":"nonpayable","payable":false,"outputs":[],"name":"setSwapFee","inputs":[{"type":"address","name":"_pair","internalType":"address"},{"type":"uint32","name":"_swapFee","internalType":"uint32"}],"constant":false}]
Contract Creation Code
Copy Contract Creation Code
0x608060405234801561001057600080fd5b50604051613e70380380613e708339818101604052602081101561003357600080fd5b5051600180546001600160a01b0319166001600160a01b03909216919091179055613e0d806100636000396000f3fe608060405234801561001057600080fd5b50600436106100c95760003560e01c80639e68ceb811610081578063e6a439051161005b578063e6a43905146101f5578063ef0bc99314610230578063f46901ed1461026c576100c9565b80639e68ceb814610146578063a2e74af614610187578063c9c65396146101ba576100c9565b80631e3dd18b116100b25780631e3dd18b14610107578063574f2ba3146101245780635855a25a1461013e576100c9565b8063017e7e58146100ce578063094b7415146100ff575b600080fd5b6100d661029f565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b6100d66102bb565b6100d66004803603602081101561011d57600080fd5b50356102d7565b61012c61030b565b60408051918252519081900360200190f35b61012c610311565b6101856004803603604081101561015c57600080fd5b50803573ffffffffffffffffffffffffffffffffffffffff16906020013563ffffffff166103c2565b005b6101856004803603602081101561019d57600080fd5b503573ffffffffffffffffffffffffffffffffffffffff166104d7565b6100d6600480360360408110156101d057600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160200135166105a4565b6100d66004803603604081101561020b57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160200135166109f7565b6101856004803603604081101561024657600080fd5b50803573ffffffffffffffffffffffffffffffffffffffff16906020013560ff16610a2a565b6101856004803603602081101561028257600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16610b92565b60005473ffffffffffffffffffffffffffffffffffffffff1681565b60015473ffffffffffffffffffffffffffffffffffffffff1681565b600381815481106102e457fe5b60009182526020909120015473ffffffffffffffffffffffffffffffffffffffff16905081565b60035490565b60405161032060208201610c5f565b6020820181038252601f19601f820116604052506040516020018082805190602001908083835b6020831061038457805182527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe09092019160209182019101610347565b6001836020036101000a0380198251168184511680821785525050505050509050019150506040516020818303038152906040528051906020012081565b60015473ffffffffffffffffffffffffffffffffffffffff16331461044857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b604080517fd6d788c300000000000000000000000000000000000000000000000000000000815263ffffffff83166004820152905173ffffffffffffffffffffffffffffffffffffffff84169163d6d788c391602480830192600092919082900301818387803b1580156104bb57600080fd5b505af11580156104cf573d6000803e3d6000fd5b505050505050565b60015473ffffffffffffffffffffffffffffffffffffffff16331461055d57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b600180547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b60008173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16141561064157604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f596f64656465783a204944454e544943414c5f41444452455353455300000000604482015290519081900360640190fd5b6000808373ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff161061067e578385610681565b84845b909250905073ffffffffffffffffffffffffffffffffffffffff821661070857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f596f64656465783a205a45524f5f414444524553530000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff8281166000908152600260209081526040808320858516845290915290205416156107a957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f596f64656465783a20504149525f455849535453000000000000000000000000604482015290519081900360640190fd5b6060604051806020016107bb90610c5f565b6020820181038252601f19601f82011660405250905060008383604051602001808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1660601b81526014018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1660601b815260140192505050604051602081830303815290604052805190602001209050808251602084016000f5604080517f485cc95500000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8781166004830152868116602483015291519297509087169163485cc9559160448082019260009290919082900301818387803b1580156108e857600080fd5b505af11580156108fc573d6000803e3d6000fd5b5050505073ffffffffffffffffffffffffffffffffffffffff84811660008181526002602081815260408084208987168086529083528185208054978d167fffffffffffffffffffffffff000000000000000000000000000000000000000098891681179091559383528185208686528352818520805488168517905560038054600181018255958190527fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b90950180549097168417909655925483519283529082015281517f0d3648bd0f6ba80134a33ba9275ac585d9d315f0ad8355cddefde31afa28d0e9929181900390910190a35050505092915050565b600260209081526000928352604080842090915290825290205473ffffffffffffffffffffffffffffffffffffffff1681565b60015473ffffffffffffffffffffffffffffffffffffffff163314610ab057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b60008160ff1611610b2257604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f596f64656465783a20464f5242494444454e5f46454500000000000000000000604482015290519081900360640190fd5b604080517f03c150b000000000000000000000000000000000000000000000000000000000815260ff83166004820152905173ffffffffffffffffffffffffffffffffffffffff8416916303c150b091602480830192600092919082900301818387803b1580156104bb57600080fd5b60015473ffffffffffffffffffffffffffffffffffffffff163314610c1857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b600080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b61316c80610c6d8339019056fe6080604052600c8054603263ffffffff199091161764ffffffffff60201b19166801330de210000000001790556001600d5534801561003d57600080fd5b50604051469080605261311a8239604080519182900360520182208282018252600a8352690596f6465646578204c560b41b6020938401528151808301835260018152603160f81b908401528151808401919091527f46666cbeb35ad337d8b9dbb7a8dfd3542a0c1bdcf9d827f8061455c243449d46818301527fc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc6606082015260808101949094523060a0808601919091528151808603909101815260c09094019052825192019190912060035550600580546001600160a01b03191633179055612fed8061012d6000396000f3fe608060405234801561001057600080fd5b50600436106101e55760003560e01c80636827e7641161010f578063ba9a7a56116100a2578063d505accf11610071578063d505accf14610674578063d6d788c3146106d2578063dd62ed3e146106f5578063fff6cae914610730576101e5565b8063ba9a7a5614610629578063bc25cf7714610631578063c45a015514610664578063d21220a71461066c576101e5565b80637ecebe00116100de5780637ecebe001461056957806389afcb441461059c57806395d89b41146105e8578063a9059cbb146105f0576101e5565b80636827e764146104d95780636a627842146104fb57806370a082311461052e5780637464fc3d14610561576101e5565b806323b872dd11610187578063485cc95511610156578063485cc9551461046d57806354cf2aeb146104a85780635909c0d5146104c95780635a3d5493146104d1576101e5565b806323b872dd146103fc57806330adf81f1461043f578063313ce567146104475780633644e51514610465576101e5565b80630902f1ac116101c35780630902f1ac14610325578063095ea7b3146103645780630dfe1681146103b157806318160ddd146103e2576101e5565b8063022c0d9f146101ea57806303c150b01461028557806306fdde03146102a8575b600080fd5b6102836004803603608081101561020057600080fd5b81359160208101359173ffffffffffffffffffffffffffffffffffffffff604083013516919081019060808101606082013564010000000081111561024457600080fd5b82018360208201111561025657600080fd5b8035906020019184600183028401116401000000008311171561027857600080fd5b509092509050610738565b005b6102836004803603602081101561029b57600080fd5b503563ffffffff16610e2e565b6102b0610f67565b6040805160208082528351818301528351919283929083019185019080838360005b838110156102ea5781810151838201526020016102d2565b50505050905090810190601f1680156103175780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b61032d610fa0565b604080516dffffffffffffffffffffffffffff948516815292909316602083015263ffffffff168183015290519081900360600190f35b61039d6004803603604081101561037a57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff8135169060200135610ff5565b604080519115158252519081900360200190f35b6103b961100c565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b6103ea611028565b60408051918252519081900360200190f35b61039d6004803603606081101561041257600080fd5b5073ffffffffffffffffffffffffffffffffffffffff81358116916020810135909116906040013561102e565b6103ea61110d565b61044f611131565b6040805160ff9092168252519081900360200190f35b6103ea611136565b6102836004803603604081101561048357600080fd5b5073ffffffffffffffffffffffffffffffffffffffff8135811691602001351661113c565b6104b0611215565b6040805163ffffffff9092168252519081900360200190f35b6103ea611221565b6103ea611227565b6104e161122d565b6040805164ffffffffff9092168252519081900360200190f35b6103ea6004803603602081101561051157600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611242565b6103ea6004803603602081101561054457600080fd5b503573ffffffffffffffffffffffffffffffffffffffff166115fc565b6103ea61160e565b6103ea6004803603602081101561057f57600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611614565b6105cf600480360360208110156105b257600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611626565b6040805192835260208301919091528051918290030190f35b6102b0611ac3565b61039d6004803603604081101561060657600080fd5b5073ffffffffffffffffffffffffffffffffffffffff8135169060200135611afc565b6103ea611b09565b6102836004803603602081101561064757600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611b0f565b6103b9611d05565b6103b9611d21565b610283600480360360e081101561068a57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160208101359091169060408101359060608101359060ff6080820135169060a08101359060c00135611d3d565b610283600480360360208110156106e857600080fd5b503563ffffffff16612009565b6103ea6004803603604081101561070b57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160200135166121b2565b6102836121cf565b600d546001146107a957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d55841515806107bc5750600084115b610811576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526023815260200180612f4e6023913960400191505060405180910390fd5b60008061081c610fa0565b5091509150816dffffffffffffffffffffffffffff168710801561084f5750806dffffffffffffffffffffffffffff1686105b6108ba57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f596f64656465783a20494e53554646494349454e545f4c495155494449545900604482015290519081900360640190fd5b600654600754600091829173ffffffffffffffffffffffffffffffffffffffff91821691908116908916821480159061091f57508073ffffffffffffffffffffffffffffffffffffffff168973ffffffffffffffffffffffffffffffffffffffff1614155b61098a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601360248201527f596f64656465783a20494e56414c49445f544f00000000000000000000000000604482015290519081900360640190fd5b8a1561099b5761099b828a8d6123b5565b89156109ac576109ac818a8c6123b5565b8615610a8e578873ffffffffffffffffffffffffffffffffffffffff1663230b25d9338d8d8c8c6040518663ffffffff1660e01b8152600401808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001858152602001848152602001806020018281038252848482818152602001925080828437600081840152601f19601f8201169050808301925050509650505050505050600060405180830381600087803b158015610a7557600080fd5b505af1158015610a89573d6000803e3d6000fd5b505050505b604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff8416916370a08231916024808301926020929190829003018186803b158015610afa57600080fd5b505afa158015610b0e573d6000803e3d6000fd5b505050506040513d6020811015610b2457600080fd5b5051604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905191955073ffffffffffffffffffffffffffffffffffffffff8316916370a0823191602480820192602092909190829003018186803b158015610b9657600080fd5b505afa158015610baa573d6000803e3d6000fd5b505050506040513d6020811015610bc057600080fd5b5051925060009150506dffffffffffffffffffffffffffff85168a90038311610bea576000610c00565b89856dffffffffffffffffffffffffffff160383035b9050600089856dffffffffffffffffffffffffffff16038311610c24576000610c3a565b89856dffffffffffffffffffffffffffff160383035b90506000821180610c4b5750600081115b610ca0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526022815260200180612f716022913960400191505060405180910390fd5b600c5463ffffffff90811690600090610cdf90610cc190869085906125c216565b610cd38861271063ffffffff6125c216565b9063ffffffff61264816565b90506000610cf6610cc1858563ffffffff6125c216565b9050610d2f6305f5e100610d236dffffffffffffffffffffffffffff8c8116908c1663ffffffff6125c216565b9063ffffffff6125c216565b610d3f838363ffffffff6125c216565b1015610dac57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f596f64656465783a204b00000000000000000000000000000000000000000000604482015290519081900360640190fd5b505050610dbb848488886126ba565b60408051838152602081018390528082018d9052606081018c9052905173ffffffffffffffffffffffffffffffffffffffff8b169133917fd78ad95fa46c994b6551d0da85fc275fe613ce37657fb8d5e3d130840159d8229181900360800190a350506001600d55505050505050505050565b60008163ffffffff1611610ea357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f596f64656465783a206c6f776572207468656e20300000000000000000000000604482015290519081900360640190fd5b60055473ffffffffffffffffffffffffffffffffffffffff163314610f2957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b600c80547fffffffffffffffffffffffffffffffffffffffffffffff0000000000ffffffff1663ffffffff9290921664010000000002919091179055565b6040518060400160405280600a81526020017f596f6465646578204c500000000000000000000000000000000000000000000081525081565b6008546dffffffffffffffffffffffffffff808216926e0100000000000000000000000000008304909116917c0100000000000000000000000000000000000000000000000000000000900463ffffffff1690565b6000611002338484612976565b5060015b92915050565b60065473ffffffffffffffffffffffffffffffffffffffff1681565b60005481565b73ffffffffffffffffffffffffffffffffffffffff831660009081526002602090815260408083203384529091528120547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff146110f85773ffffffffffffffffffffffffffffffffffffffff841660009081526002602090815260408083203384529091529020546110c6908363ffffffff61264816565b73ffffffffffffffffffffffffffffffffffffffff851660009081526002602090815260408083203384529091529020555b6111038484846129e5565b5060019392505050565b7f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c981565b601281565b60035481565b60055473ffffffffffffffffffffffffffffffffffffffff1633146111c257604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b6006805473ffffffffffffffffffffffffffffffffffffffff9384167fffffffffffffffffffffffff00000000000000000000000000000000000000009182161790915560078054929093169116179055565b600c5463ffffffff1681565b60095481565b600a5481565b600c54640100000000900464ffffffffff1681565b6000600d546001146112b557604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d819055806112c5610fa0565b50600654604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905193955091935060009273ffffffffffffffffffffffffffffffffffffffff909116916370a08231916024808301926020929190829003018186803b15801561133f57600080fd5b505afa158015611353573d6000803e3d6000fd5b505050506040513d602081101561136957600080fd5b5051600754604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905192935060009273ffffffffffffffffffffffffffffffffffffffff909216916370a0823191602480820192602092909190829003018186803b1580156113e257600080fd5b505afa1580156113f6573d6000803e3d6000fd5b505050506040513d602081101561140c57600080fd5b505190506000611432836dffffffffffffffffffffffffffff871663ffffffff61264816565b90506000611456836dffffffffffffffffffffffffffff871663ffffffff61264816565b905060006114648787612ac6565b600054909150806114a15761148d6103e8610cd3611488878763ffffffff6125c216565b612c71565b985061149c60006103e8612cc3565b6114fe565b6114fb6dffffffffffffffffffffffffffff89166114c5868463ffffffff6125c216565b816114cc57fe5b046dffffffffffffffffffffffffffff89166114ee868563ffffffff6125c216565b816114f557fe5b04612d73565b98505b60008911611557576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526026815260200180612f286026913960400191505060405180910390fd5b6115618a8a612cc3565b61156d86868a8a6126ba565b81156115af576008546115ab906dffffffffffffffffffffffffffff808216916e01000000000000000000000000000090041663ffffffff6125c216565b600b555b6040805185815260208101859052815133927f4c209b5fc8ad50758f13e2e1088ba56a560dff690a1c6fef26394f4c03821c4f928290030190a250506001600d5550949695505050505050565b60016020526000908152604090205481565b600b5481565b60046020526000908152604090205481565b600080600d5460011461169a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d819055806116aa610fa0565b50600654600754604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905194965092945073ffffffffffffffffffffffffffffffffffffffff9182169391169160009184916370a08231916024808301926020929190829003018186803b15801561172c57600080fd5b505afa158015611740573d6000803e3d6000fd5b505050506040513d602081101561175657600080fd5b5051604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905191925060009173ffffffffffffffffffffffffffffffffffffffff8516916370a08231916024808301926020929190829003018186803b1580156117ca57600080fd5b505afa1580156117de573d6000803e3d6000fd5b505050506040513d60208110156117f457600080fd5b5051306000908152600160205260408120549192506118138888612ac6565b6000549091508061182a848763ffffffff6125c216565b8161183157fe5b049a5080611845848663ffffffff6125c216565b8161184c57fe5b04995060008b11801561185f575060008a115b6118b4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526026815260200180612f936026913960400191505060405180910390fd5b6118be3084612d8b565b6118c9878d8d6123b5565b6118d4868d8c6123b5565b604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff8916916370a08231916024808301926020929190829003018186803b15801561194057600080fd5b505afa158015611954573d6000803e3d6000fd5b505050506040513d602081101561196a57600080fd5b5051604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905191965073ffffffffffffffffffffffffffffffffffffffff8816916370a0823191602480820192602092909190829003018186803b1580156119dc57600080fd5b505afa1580156119f0573d6000803e3d6000fd5b505050506040513d6020811015611a0657600080fd5b50519350611a1685858b8b6126ba565b8115611a5857600854611a54906dffffffffffffffffffffffffffff808216916e01000000000000000000000000000090041663ffffffff6125c216565b600b555b604080518c8152602081018c9052815173ffffffffffffffffffffffffffffffffffffffff8f169233927fdccd412f0b1252819cb1fd330b93224ca42612892bb3f4f789976e6d81936496929081900390910190a35050505050505050506001600d81905550915091565b6040518060400160405280600381526020017f594c50000000000000000000000000000000000000000000000000000000000081525081565b60006110023384846129e5565b6103e881565b600d54600114611b8057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d55600654600754600854604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff9485169490931692611c5c9285928792611c57926dffffffffffffffffffffffffffff169185916370a0823191602480820192602092909190829003018186803b158015611c1f57600080fd5b505afa158015611c33573d6000803e3d6000fd5b505050506040513d6020811015611c4957600080fd5b50519063ffffffff61264816565b6123b5565b600854604080517f70a082310000000000000000000000000000000000000000000000000000000081523060048201529051611cfb9284928792611c57926e01000000000000000000000000000090046dffffffffffffffffffffffffffff169173ffffffffffffffffffffffffffffffffffffffff8616916370a0823191602480820192602092909190829003018186803b158015611c1f57600080fd5b50506001600d5550565b60055473ffffffffffffffffffffffffffffffffffffffff1681565b60075473ffffffffffffffffffffffffffffffffffffffff1681565b42841015611dac57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601060248201527f596f64656465783a204558504952454400000000000000000000000000000000604482015290519081900360640190fd5b60035473ffffffffffffffffffffffffffffffffffffffff80891660008181526004602090815260408083208054600180820190925582517f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c98186015280840196909652958d166060860152608085018c905260a085019590955260c08085018b90528151808603909101815260e0850182528051908301207f19010000000000000000000000000000000000000000000000000000000000006101008601526101028501969096526101228085019690965280518085039096018652610142840180825286519683019690962095839052610162840180825286905260ff89166101828501526101a284018890526101c28401879052519193926101e2808201937fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe081019281900390910190855afa158015611f0d573d6000803e3d6000fd5b50506040517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0015191505073ffffffffffffffffffffffffffffffffffffffff811615801590611f8857508873ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16145b611ff357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601a60248201527f596f64656465783a20494e56414c49445f5349474e4154555245000000000000604482015290519081900360640190fd5b611ffe898989612976565b505050505050505050565b60008163ffffffff161161207e57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f596f64656465783a206c6f776572207468656e20300000000000000000000000604482015290519081900360640190fd5b60055473ffffffffffffffffffffffffffffffffffffffff16331461210457604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b6103e88163ffffffff16111561217b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f596f64656465783a20464f5242494444454e5f46454500000000000000000000604482015290519081900360640190fd5b600c80547fffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000001663ffffffff92909216919091179055565b600260209081526000928352604080842090915290825290205481565b600d5460011461224057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d55600654604080517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015290516123ae9273ffffffffffffffffffffffffffffffffffffffff16916370a08231916024808301926020929190829003018186803b1580156122b757600080fd5b505afa1580156122cb573d6000803e3d6000fd5b505050506040513d60208110156122e157600080fd5b5051600754604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff909216916370a0823191602480820192602092909190829003018186803b15801561235457600080fd5b505afa158015612368573d6000803e3d6000fd5b505050506040513d602081101561237e57600080fd5b50516008546dffffffffffffffffffffffffffff808216916e0100000000000000000000000000009004166126ba565b6001600d55565b604080518082018252601981527f7472616e7366657228616464726573732c75696e743235362900000000000000602091820152815173ffffffffffffffffffffffffffffffffffffffff85811660248301526044808301869052845180840390910181526064909201845291810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fa9059cbb000000000000000000000000000000000000000000000000000000001781529251815160009460609489169392918291908083835b602083106124bb57805182527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0909201916020918201910161247e565b6001836020036101000a0380198251168184511680821785525050505050509050019150506000604051808303816000865af19150503d806000811461251d576040519150601f19603f3d011682016040523d82523d6000602084013e612522565b606091505b5091509150818015612550575080511580612550575080806020019051602081101561254d57600080fd5b50515b6125bb57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601860248201527f596f64656465783a205452414e534645525f4641494c45440000000000000000604482015290519081900360640190fd5b5050505050565b60008115806125dd575050808202828282816125da57fe5b04145b61100657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f64732d6d6174682d6d756c2d6f766572666c6f77000000000000000000000000604482015290519081900360640190fd5b8082038281111561100657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f64732d6d6174682d7375622d756e646572666c6f770000000000000000000000604482015290519081900360640190fd5b6dffffffffffffffffffffffffffff84118015906126e657506dffffffffffffffffffffffffffff8311155b61275157604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601160248201527f596f64656465783a204f564552464c4f57000000000000000000000000000000604482015290519081900360640190fd5b60085463ffffffff428116917c0100000000000000000000000000000000000000000000000000000000900481168203908116158015906127a157506dffffffffffffffffffffffffffff841615155b80156127bc57506dffffffffffffffffffffffffffff831615155b1561286c578063ffffffff166127ff856127d586612e50565b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff169063ffffffff612e7416565b600980547bffffffffffffffffffffffffffffffffffffffffffffffffffffffff929092169290920201905563ffffffff811661283f846127d587612e50565b600a80547bffffffffffffffffffffffffffffffffffffffffffffffffffffffff92909216929092020190555b600880547fffffffffffffffffffffffffffffffffffff0000000000000000000000000000166dffffffffffffffffffffffffffff888116919091177fffffffff0000000000000000000000000000ffffffffffffffffffffffffffff166e0100000000000000000000000000008883168102919091177bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167c010000000000000000000000000000000000000000000000000000000063ffffffff871602179283905560408051848416815291909304909116602082015281517f1c411e9a96e071241c2f21f7726b17ae89e3cab4c78be50e062b03a9fffbbad1929181900390910190a1505050505050565b73ffffffffffffffffffffffffffffffffffffffff808416600081815260026020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b73ffffffffffffffffffffffffffffffffffffffff8316600090815260016020526040902054612a1b908263ffffffff61264816565b73ffffffffffffffffffffffffffffffffffffffff8085166000908152600160205260408082209390935590841681522054612a5d908263ffffffff612eb516565b73ffffffffffffffffffffffffffffffffffffffff80841660008181526001602090815260409182902094909455805185815290519193928716927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef92918290030190a3505050565b600080600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663017e7e586040518163ffffffff1660e01b815260040160206040518083038186803b158015612b3157600080fd5b505afa158015612b45573d6000803e3d6000fd5b505050506040513d6020811015612b5b57600080fd5b5051600b5473ffffffffffffffffffffffffffffffffffffffff8216158015945091925090612c5d578015612c58576000612bb26114886dffffffffffffffffffffffffffff88811690881663ffffffff6125c216565b90506000612bbf83612c71565b905080821115612c55576000612bed612bde848463ffffffff61264816565b6000549063ffffffff6125c216565b600c54909150600090612c319084906402540be40090612c1d908890640100000000900464ffffffffff166125c2565b81612c2457fe5b049063ffffffff612eb516565b90506000818381612c3e57fe5b0490508015612c5157612c518782612cc3565b5050505b50505b612c69565b8015612c69576000600b555b505092915050565b60006003821115612cb4575080600160028204015b81811015612cae57809150600281828581612c9d57fe5b040181612ca657fe5b049050612c86565b50612cbe565b8115612cbe575060015b919050565b600054612cd6908263ffffffff612eb516565b600090815573ffffffffffffffffffffffffffffffffffffffff8316815260016020526040902054612d0e908263ffffffff612eb516565b73ffffffffffffffffffffffffffffffffffffffff831660008181526001602090815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b6000818310612d825781612d84565b825b9392505050565b73ffffffffffffffffffffffffffffffffffffffff8216600090815260016020526040902054612dc1908263ffffffff61264816565b73ffffffffffffffffffffffffffffffffffffffff831660009081526001602052604081209190915554612dfb908263ffffffff61264816565b600090815560408051838152905173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef919081900360200190a35050565b6dffffffffffffffffffffffffffff166e0100000000000000000000000000000290565b60006dffffffffffffffffffffffffffff82167bffffffffffffffffffffffffffffffffffffffffffffffffffffffff841681612ead57fe5b049392505050565b8082018281101561100657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f64732d6d6174682d6164642d6f766572666c6f77000000000000000000000000604482015290519081900360640190fdfe596f64656465783a20494e53554646494349454e545f4c49515549444954595f4d494e544544596f64656465783a20494e53554646494349454e545f4f55545055545f414d4f554e54596f64656465783a20494e53554646494349454e545f494e5055545f414d4f554e54596f64656465783a20494e53554646494349454e545f4c49515549444954595f4255524e4544a265627a7a7231582049bc4a7953ebb0a539de5f6ef3311f91d85729247332bc6a48bec016441804ce64736f6c63430005100032454950373132446f6d61696e28737472696e67206e616d652c737472696e672076657273696f6e2c75696e7432353620636861696e49642c6164647265737320766572696679696e67436f6e747261637429a265627a7a723158205bdf8943c0c184cd8ce54c184e5f349195ce97c7894911ed84364b642d33617664736f6c634300051000320000000000000000000000002a2e3486204c9eeeab2bef8faa3356bc19e9db6f
Deployed ByteCode
Copy Deployed ByteCode
0x608060405234801561001057600080fd5b50600436106100c95760003560e01c80639e68ceb811610081578063e6a439051161005b578063e6a43905146101f5578063ef0bc99314610230578063f46901ed1461026c576100c9565b80639e68ceb814610146578063a2e74af614610187578063c9c65396146101ba576100c9565b80631e3dd18b116100b25780631e3dd18b14610107578063574f2ba3146101245780635855a25a1461013e576100c9565b8063017e7e58146100ce578063094b7415146100ff575b600080fd5b6100d661029f565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b6100d66102bb565b6100d66004803603602081101561011d57600080fd5b50356102d7565b61012c61030b565b60408051918252519081900360200190f35b61012c610311565b6101856004803603604081101561015c57600080fd5b50803573ffffffffffffffffffffffffffffffffffffffff16906020013563ffffffff166103c2565b005b6101856004803603602081101561019d57600080fd5b503573ffffffffffffffffffffffffffffffffffffffff166104d7565b6100d6600480360360408110156101d057600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160200135166105a4565b6100d66004803603604081101561020b57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160200135166109f7565b6101856004803603604081101561024657600080fd5b50803573ffffffffffffffffffffffffffffffffffffffff16906020013560ff16610a2a565b6101856004803603602081101561028257600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16610b92565b60005473ffffffffffffffffffffffffffffffffffffffff1681565b60015473ffffffffffffffffffffffffffffffffffffffff1681565b600381815481106102e457fe5b60009182526020909120015473ffffffffffffffffffffffffffffffffffffffff16905081565b60035490565b60405161032060208201610c5f565b6020820181038252601f19601f820116604052506040516020018082805190602001908083835b6020831061038457805182527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe09092019160209182019101610347565b6001836020036101000a0380198251168184511680821785525050505050509050019150506040516020818303038152906040528051906020012081565b60015473ffffffffffffffffffffffffffffffffffffffff16331461044857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b604080517fd6d788c300000000000000000000000000000000000000000000000000000000815263ffffffff83166004820152905173ffffffffffffffffffffffffffffffffffffffff84169163d6d788c391602480830192600092919082900301818387803b1580156104bb57600080fd5b505af11580156104cf573d6000803e3d6000fd5b505050505050565b60015473ffffffffffffffffffffffffffffffffffffffff16331461055d57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b600180547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b60008173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff16141561064157604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f596f64656465783a204944454e544943414c5f41444452455353455300000000604482015290519081900360640190fd5b6000808373ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff161061067e578385610681565b84845b909250905073ffffffffffffffffffffffffffffffffffffffff821661070857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f596f64656465783a205a45524f5f414444524553530000000000000000000000604482015290519081900360640190fd5b73ffffffffffffffffffffffffffffffffffffffff8281166000908152600260209081526040808320858516845290915290205416156107a957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f596f64656465783a20504149525f455849535453000000000000000000000000604482015290519081900360640190fd5b6060604051806020016107bb90610c5f565b6020820181038252601f19601f82011660405250905060008383604051602001808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1660601b81526014018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1660601b815260140192505050604051602081830303815290604052805190602001209050808251602084016000f5604080517f485cc95500000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8781166004830152868116602483015291519297509087169163485cc9559160448082019260009290919082900301818387803b1580156108e857600080fd5b505af11580156108fc573d6000803e3d6000fd5b5050505073ffffffffffffffffffffffffffffffffffffffff84811660008181526002602081815260408084208987168086529083528185208054978d167fffffffffffffffffffffffff000000000000000000000000000000000000000098891681179091559383528185208686528352818520805488168517905560038054600181018255958190527fc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b90950180549097168417909655925483519283529082015281517f0d3648bd0f6ba80134a33ba9275ac585d9d315f0ad8355cddefde31afa28d0e9929181900390910190a35050505092915050565b600260209081526000928352604080842090915290825290205473ffffffffffffffffffffffffffffffffffffffff1681565b60015473ffffffffffffffffffffffffffffffffffffffff163314610ab057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b60008160ff1611610b2257604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f596f64656465783a20464f5242494444454e5f46454500000000000000000000604482015290519081900360640190fd5b604080517f03c150b000000000000000000000000000000000000000000000000000000000815260ff83166004820152905173ffffffffffffffffffffffffffffffffffffffff8416916303c150b091602480830192600092919082900301818387803b1580156104bb57600080fd5b60015473ffffffffffffffffffffffffffffffffffffffff163314610c1857604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b600080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b61316c80610c6d8339019056fe6080604052600c8054603263ffffffff199091161764ffffffffff60201b19166801330de210000000001790556001600d5534801561003d57600080fd5b50604051469080605261311a8239604080519182900360520182208282018252600a8352690596f6465646578204c560b41b6020938401528151808301835260018152603160f81b908401528151808401919091527f46666cbeb35ad337d8b9dbb7a8dfd3542a0c1bdcf9d827f8061455c243449d46818301527fc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc6606082015260808101949094523060a0808601919091528151808603909101815260c09094019052825192019190912060035550600580546001600160a01b03191633179055612fed8061012d6000396000f3fe608060405234801561001057600080fd5b50600436106101e55760003560e01c80636827e7641161010f578063ba9a7a56116100a2578063d505accf11610071578063d505accf14610674578063d6d788c3146106d2578063dd62ed3e146106f5578063fff6cae914610730576101e5565b8063ba9a7a5614610629578063bc25cf7714610631578063c45a015514610664578063d21220a71461066c576101e5565b80637ecebe00116100de5780637ecebe001461056957806389afcb441461059c57806395d89b41146105e8578063a9059cbb146105f0576101e5565b80636827e764146104d95780636a627842146104fb57806370a082311461052e5780637464fc3d14610561576101e5565b806323b872dd11610187578063485cc95511610156578063485cc9551461046d57806354cf2aeb146104a85780635909c0d5146104c95780635a3d5493146104d1576101e5565b806323b872dd146103fc57806330adf81f1461043f578063313ce567146104475780633644e51514610465576101e5565b80630902f1ac116101c35780630902f1ac14610325578063095ea7b3146103645780630dfe1681146103b157806318160ddd146103e2576101e5565b8063022c0d9f146101ea57806303c150b01461028557806306fdde03146102a8575b600080fd5b6102836004803603608081101561020057600080fd5b81359160208101359173ffffffffffffffffffffffffffffffffffffffff604083013516919081019060808101606082013564010000000081111561024457600080fd5b82018360208201111561025657600080fd5b8035906020019184600183028401116401000000008311171561027857600080fd5b509092509050610738565b005b6102836004803603602081101561029b57600080fd5b503563ffffffff16610e2e565b6102b0610f67565b6040805160208082528351818301528351919283929083019185019080838360005b838110156102ea5781810151838201526020016102d2565b50505050905090810190601f1680156103175780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b61032d610fa0565b604080516dffffffffffffffffffffffffffff948516815292909316602083015263ffffffff168183015290519081900360600190f35b61039d6004803603604081101561037a57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff8135169060200135610ff5565b604080519115158252519081900360200190f35b6103b961100c565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b6103ea611028565b60408051918252519081900360200190f35b61039d6004803603606081101561041257600080fd5b5073ffffffffffffffffffffffffffffffffffffffff81358116916020810135909116906040013561102e565b6103ea61110d565b61044f611131565b6040805160ff9092168252519081900360200190f35b6103ea611136565b6102836004803603604081101561048357600080fd5b5073ffffffffffffffffffffffffffffffffffffffff8135811691602001351661113c565b6104b0611215565b6040805163ffffffff9092168252519081900360200190f35b6103ea611221565b6103ea611227565b6104e161122d565b6040805164ffffffffff9092168252519081900360200190f35b6103ea6004803603602081101561051157600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611242565b6103ea6004803603602081101561054457600080fd5b503573ffffffffffffffffffffffffffffffffffffffff166115fc565b6103ea61160e565b6103ea6004803603602081101561057f57600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611614565b6105cf600480360360208110156105b257600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611626565b6040805192835260208301919091528051918290030190f35b6102b0611ac3565b61039d6004803603604081101561060657600080fd5b5073ffffffffffffffffffffffffffffffffffffffff8135169060200135611afc565b6103ea611b09565b6102836004803603602081101561064757600080fd5b503573ffffffffffffffffffffffffffffffffffffffff16611b0f565b6103b9611d05565b6103b9611d21565b610283600480360360e081101561068a57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160208101359091169060408101359060608101359060ff6080820135169060a08101359060c00135611d3d565b610283600480360360208110156106e857600080fd5b503563ffffffff16612009565b6103ea6004803603604081101561070b57600080fd5b5073ffffffffffffffffffffffffffffffffffffffff813581169160200135166121b2565b6102836121cf565b600d546001146107a957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d55841515806107bc5750600084115b610811576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526023815260200180612f4e6023913960400191505060405180910390fd5b60008061081c610fa0565b5091509150816dffffffffffffffffffffffffffff168710801561084f5750806dffffffffffffffffffffffffffff1686105b6108ba57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f596f64656465783a20494e53554646494349454e545f4c495155494449545900604482015290519081900360640190fd5b600654600754600091829173ffffffffffffffffffffffffffffffffffffffff91821691908116908916821480159061091f57508073ffffffffffffffffffffffffffffffffffffffff168973ffffffffffffffffffffffffffffffffffffffff1614155b61098a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601360248201527f596f64656465783a20494e56414c49445f544f00000000000000000000000000604482015290519081900360640190fd5b8a1561099b5761099b828a8d6123b5565b89156109ac576109ac818a8c6123b5565b8615610a8e578873ffffffffffffffffffffffffffffffffffffffff1663230b25d9338d8d8c8c6040518663ffffffff1660e01b8152600401808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001858152602001848152602001806020018281038252848482818152602001925080828437600081840152601f19601f8201169050808301925050509650505050505050600060405180830381600087803b158015610a7557600080fd5b505af1158015610a89573d6000803e3d6000fd5b505050505b604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff8416916370a08231916024808301926020929190829003018186803b158015610afa57600080fd5b505afa158015610b0e573d6000803e3d6000fd5b505050506040513d6020811015610b2457600080fd5b5051604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905191955073ffffffffffffffffffffffffffffffffffffffff8316916370a0823191602480820192602092909190829003018186803b158015610b9657600080fd5b505afa158015610baa573d6000803e3d6000fd5b505050506040513d6020811015610bc057600080fd5b5051925060009150506dffffffffffffffffffffffffffff85168a90038311610bea576000610c00565b89856dffffffffffffffffffffffffffff160383035b9050600089856dffffffffffffffffffffffffffff16038311610c24576000610c3a565b89856dffffffffffffffffffffffffffff160383035b90506000821180610c4b5750600081115b610ca0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526022815260200180612f716022913960400191505060405180910390fd5b600c5463ffffffff90811690600090610cdf90610cc190869085906125c216565b610cd38861271063ffffffff6125c216565b9063ffffffff61264816565b90506000610cf6610cc1858563ffffffff6125c216565b9050610d2f6305f5e100610d236dffffffffffffffffffffffffffff8c8116908c1663ffffffff6125c216565b9063ffffffff6125c216565b610d3f838363ffffffff6125c216565b1015610dac57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600a60248201527f596f64656465783a204b00000000000000000000000000000000000000000000604482015290519081900360640190fd5b505050610dbb848488886126ba565b60408051838152602081018390528082018d9052606081018c9052905173ffffffffffffffffffffffffffffffffffffffff8b169133917fd78ad95fa46c994b6551d0da85fc275fe613ce37657fb8d5e3d130840159d8229181900360800190a350506001600d55505050505050505050565b60008163ffffffff1611610ea357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f596f64656465783a206c6f776572207468656e20300000000000000000000000604482015290519081900360640190fd5b60055473ffffffffffffffffffffffffffffffffffffffff163314610f2957604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b600c80547fffffffffffffffffffffffffffffffffffffffffffffff0000000000ffffffff1663ffffffff9290921664010000000002919091179055565b6040518060400160405280600a81526020017f596f6465646578204c500000000000000000000000000000000000000000000081525081565b6008546dffffffffffffffffffffffffffff808216926e0100000000000000000000000000008304909116917c0100000000000000000000000000000000000000000000000000000000900463ffffffff1690565b6000611002338484612976565b5060015b92915050565b60065473ffffffffffffffffffffffffffffffffffffffff1681565b60005481565b73ffffffffffffffffffffffffffffffffffffffff831660009081526002602090815260408083203384529091528120547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff146110f85773ffffffffffffffffffffffffffffffffffffffff841660009081526002602090815260408083203384529091529020546110c6908363ffffffff61264816565b73ffffffffffffffffffffffffffffffffffffffff851660009081526002602090815260408083203384529091529020555b6111038484846129e5565b5060019392505050565b7f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c981565b601281565b60035481565b60055473ffffffffffffffffffffffffffffffffffffffff1633146111c257604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b6006805473ffffffffffffffffffffffffffffffffffffffff9384167fffffffffffffffffffffffff00000000000000000000000000000000000000009182161790915560078054929093169116179055565b600c5463ffffffff1681565b60095481565b600a5481565b600c54640100000000900464ffffffffff1681565b6000600d546001146112b557604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d819055806112c5610fa0565b50600654604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905193955091935060009273ffffffffffffffffffffffffffffffffffffffff909116916370a08231916024808301926020929190829003018186803b15801561133f57600080fd5b505afa158015611353573d6000803e3d6000fd5b505050506040513d602081101561136957600080fd5b5051600754604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905192935060009273ffffffffffffffffffffffffffffffffffffffff909216916370a0823191602480820192602092909190829003018186803b1580156113e257600080fd5b505afa1580156113f6573d6000803e3d6000fd5b505050506040513d602081101561140c57600080fd5b505190506000611432836dffffffffffffffffffffffffffff871663ffffffff61264816565b90506000611456836dffffffffffffffffffffffffffff871663ffffffff61264816565b905060006114648787612ac6565b600054909150806114a15761148d6103e8610cd3611488878763ffffffff6125c216565b612c71565b985061149c60006103e8612cc3565b6114fe565b6114fb6dffffffffffffffffffffffffffff89166114c5868463ffffffff6125c216565b816114cc57fe5b046dffffffffffffffffffffffffffff89166114ee868563ffffffff6125c216565b816114f557fe5b04612d73565b98505b60008911611557576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526026815260200180612f286026913960400191505060405180910390fd5b6115618a8a612cc3565b61156d86868a8a6126ba565b81156115af576008546115ab906dffffffffffffffffffffffffffff808216916e01000000000000000000000000000090041663ffffffff6125c216565b600b555b6040805185815260208101859052815133927f4c209b5fc8ad50758f13e2e1088ba56a560dff690a1c6fef26394f4c03821c4f928290030190a250506001600d5550949695505050505050565b60016020526000908152604090205481565b600b5481565b60046020526000908152604090205481565b600080600d5460011461169a57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d819055806116aa610fa0565b50600654600754604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905194965092945073ffffffffffffffffffffffffffffffffffffffff9182169391169160009184916370a08231916024808301926020929190829003018186803b15801561172c57600080fd5b505afa158015611740573d6000803e3d6000fd5b505050506040513d602081101561175657600080fd5b5051604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905191925060009173ffffffffffffffffffffffffffffffffffffffff8516916370a08231916024808301926020929190829003018186803b1580156117ca57600080fd5b505afa1580156117de573d6000803e3d6000fd5b505050506040513d60208110156117f457600080fd5b5051306000908152600160205260408120549192506118138888612ac6565b6000549091508061182a848763ffffffff6125c216565b8161183157fe5b049a5080611845848663ffffffff6125c216565b8161184c57fe5b04995060008b11801561185f575060008a115b6118b4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401808060200182810382526026815260200180612f936026913960400191505060405180910390fd5b6118be3084612d8b565b6118c9878d8d6123b5565b6118d4868d8c6123b5565b604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff8916916370a08231916024808301926020929190829003018186803b15801561194057600080fd5b505afa158015611954573d6000803e3d6000fd5b505050506040513d602081101561196a57600080fd5b5051604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905191965073ffffffffffffffffffffffffffffffffffffffff8816916370a0823191602480820192602092909190829003018186803b1580156119dc57600080fd5b505afa1580156119f0573d6000803e3d6000fd5b505050506040513d6020811015611a0657600080fd5b50519350611a1685858b8b6126ba565b8115611a5857600854611a54906dffffffffffffffffffffffffffff808216916e01000000000000000000000000000090041663ffffffff6125c216565b600b555b604080518c8152602081018c9052815173ffffffffffffffffffffffffffffffffffffffff8f169233927fdccd412f0b1252819cb1fd330b93224ca42612892bb3f4f789976e6d81936496929081900390910190a35050505050505050506001600d81905550915091565b6040518060400160405280600381526020017f594c50000000000000000000000000000000000000000000000000000000000081525081565b60006110023384846129e5565b6103e881565b600d54600114611b8057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d55600654600754600854604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff9485169490931692611c5c9285928792611c57926dffffffffffffffffffffffffffff169185916370a0823191602480820192602092909190829003018186803b158015611c1f57600080fd5b505afa158015611c33573d6000803e3d6000fd5b505050506040513d6020811015611c4957600080fd5b50519063ffffffff61264816565b6123b5565b600854604080517f70a082310000000000000000000000000000000000000000000000000000000081523060048201529051611cfb9284928792611c57926e01000000000000000000000000000090046dffffffffffffffffffffffffffff169173ffffffffffffffffffffffffffffffffffffffff8616916370a0823191602480820192602092909190829003018186803b158015611c1f57600080fd5b50506001600d5550565b60055473ffffffffffffffffffffffffffffffffffffffff1681565b60075473ffffffffffffffffffffffffffffffffffffffff1681565b42841015611dac57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601060248201527f596f64656465783a204558504952454400000000000000000000000000000000604482015290519081900360640190fd5b60035473ffffffffffffffffffffffffffffffffffffffff80891660008181526004602090815260408083208054600180820190925582517f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c98186015280840196909652958d166060860152608085018c905260a085019590955260c08085018b90528151808603909101815260e0850182528051908301207f19010000000000000000000000000000000000000000000000000000000000006101008601526101028501969096526101228085019690965280518085039096018652610142840180825286519683019690962095839052610162840180825286905260ff89166101828501526101a284018890526101c28401879052519193926101e2808201937fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe081019281900390910190855afa158015611f0d573d6000803e3d6000fd5b50506040517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0015191505073ffffffffffffffffffffffffffffffffffffffff811615801590611f8857508873ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16145b611ff357604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601a60248201527f596f64656465783a20494e56414c49445f5349474e4154555245000000000000604482015290519081900360640190fd5b611ffe898989612976565b505050505050505050565b60008163ffffffff161161207e57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f596f64656465783a206c6f776572207468656e20300000000000000000000000604482015290519081900360640190fd5b60055473ffffffffffffffffffffffffffffffffffffffff16331461210457604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f596f64656465783a20464f5242494444454e0000000000000000000000000000604482015290519081900360640190fd5b6103e88163ffffffff16111561217b57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f596f64656465783a20464f5242494444454e5f46454500000000000000000000604482015290519081900360640190fd5b600c80547fffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000001663ffffffff92909216919091179055565b600260209081526000928352604080842090915290825290205481565b600d5460011461224057604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f596f64656465783a204c4f434b45440000000000000000000000000000000000604482015290519081900360640190fd5b6000600d55600654604080517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015290516123ae9273ffffffffffffffffffffffffffffffffffffffff16916370a08231916024808301926020929190829003018186803b1580156122b757600080fd5b505afa1580156122cb573d6000803e3d6000fd5b505050506040513d60208110156122e157600080fd5b5051600754604080517f70a08231000000000000000000000000000000000000000000000000000000008152306004820152905173ffffffffffffffffffffffffffffffffffffffff909216916370a0823191602480820192602092909190829003018186803b15801561235457600080fd5b505afa158015612368573d6000803e3d6000fd5b505050506040513d602081101561237e57600080fd5b50516008546dffffffffffffffffffffffffffff808216916e0100000000000000000000000000009004166126ba565b6001600d55565b604080518082018252601981527f7472616e7366657228616464726573732c75696e743235362900000000000000602091820152815173ffffffffffffffffffffffffffffffffffffffff85811660248301526044808301869052845180840390910181526064909201845291810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fa9059cbb000000000000000000000000000000000000000000000000000000001781529251815160009460609489169392918291908083835b602083106124bb57805182527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0909201916020918201910161247e565b6001836020036101000a0380198251168184511680821785525050505050509050019150506000604051808303816000865af19150503d806000811461251d576040519150601f19603f3d011682016040523d82523d6000602084013e612522565b606091505b5091509150818015612550575080511580612550575080806020019051602081101561254d57600080fd5b50515b6125bb57604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601860248201527f596f64656465783a205452414e534645525f4641494c45440000000000000000604482015290519081900360640190fd5b5050505050565b60008115806125dd575050808202828282816125da57fe5b04145b61100657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f64732d6d6174682d6d756c2d6f766572666c6f77000000000000000000000000604482015290519081900360640190fd5b8082038281111561100657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f64732d6d6174682d7375622d756e646572666c6f770000000000000000000000604482015290519081900360640190fd5b6dffffffffffffffffffffffffffff84118015906126e657506dffffffffffffffffffffffffffff8311155b61275157604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601160248201527f596f64656465783a204f564552464c4f57000000000000000000000000000000604482015290519081900360640190fd5b60085463ffffffff428116917c0100000000000000000000000000000000000000000000000000000000900481168203908116158015906127a157506dffffffffffffffffffffffffffff841615155b80156127bc57506dffffffffffffffffffffffffffff831615155b1561286c578063ffffffff166127ff856127d586612e50565b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff169063ffffffff612e7416565b600980547bffffffffffffffffffffffffffffffffffffffffffffffffffffffff929092169290920201905563ffffffff811661283f846127d587612e50565b600a80547bffffffffffffffffffffffffffffffffffffffffffffffffffffffff92909216929092020190555b600880547fffffffffffffffffffffffffffffffffffff0000000000000000000000000000166dffffffffffffffffffffffffffff888116919091177fffffffff0000000000000000000000000000ffffffffffffffffffffffffffff166e0100000000000000000000000000008883168102919091177bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167c010000000000000000000000000000000000000000000000000000000063ffffffff871602179283905560408051848416815291909304909116602082015281517f1c411e9a96e071241c2f21f7726b17ae89e3cab4c78be50e062b03a9fffbbad1929181900390910190a1505050505050565b73ffffffffffffffffffffffffffffffffffffffff808416600081815260026020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b73ffffffffffffffffffffffffffffffffffffffff8316600090815260016020526040902054612a1b908263ffffffff61264816565b73ffffffffffffffffffffffffffffffffffffffff8085166000908152600160205260408082209390935590841681522054612a5d908263ffffffff612eb516565b73ffffffffffffffffffffffffffffffffffffffff80841660008181526001602090815260409182902094909455805185815290519193928716927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef92918290030190a3505050565b600080600560009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663017e7e586040518163ffffffff1660e01b815260040160206040518083038186803b158015612b3157600080fd5b505afa158015612b45573d6000803e3d6000fd5b505050506040513d6020811015612b5b57600080fd5b5051600b5473ffffffffffffffffffffffffffffffffffffffff8216158015945091925090612c5d578015612c58576000612bb26114886dffffffffffffffffffffffffffff88811690881663ffffffff6125c216565b90506000612bbf83612c71565b905080821115612c55576000612bed612bde848463ffffffff61264816565b6000549063ffffffff6125c216565b600c54909150600090612c319084906402540be40090612c1d908890640100000000900464ffffffffff166125c2565b81612c2457fe5b049063ffffffff612eb516565b90506000818381612c3e57fe5b0490508015612c5157612c518782612cc3565b5050505b50505b612c69565b8015612c69576000600b555b505092915050565b60006003821115612cb4575080600160028204015b81811015612cae57809150600281828581612c9d57fe5b040181612ca657fe5b049050612c86565b50612cbe565b8115612cbe575060015b919050565b600054612cd6908263ffffffff612eb516565b600090815573ffffffffffffffffffffffffffffffffffffffff8316815260016020526040902054612d0e908263ffffffff612eb516565b73ffffffffffffffffffffffffffffffffffffffff831660008181526001602090815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b6000818310612d825781612d84565b825b9392505050565b73ffffffffffffffffffffffffffffffffffffffff8216600090815260016020526040902054612dc1908263ffffffff61264816565b73ffffffffffffffffffffffffffffffffffffffff831660009081526001602052604081209190915554612dfb908263ffffffff61264816565b600090815560408051838152905173ffffffffffffffffffffffffffffffffffffffff8516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef919081900360200190a35050565b6dffffffffffffffffffffffffffff166e0100000000000000000000000000000290565b60006dffffffffffffffffffffffffffff82167bffffffffffffffffffffffffffffffffffffffffffffffffffffffff841681612ead57fe5b049392505050565b8082018281101561100657604080517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f64732d6d6174682d6164642d6f766572666c6f77000000000000000000000000604482015290519081900360640190fdfe596f64656465783a20494e53554646494349454e545f4c49515549444954595f4d494e544544596f64656465783a20494e53554646494349454e545f4f55545055545f414d4f554e54596f64656465783a20494e53554646494349454e545f494e5055545f414d4f554e54596f64656465783a20494e53554646494349454e545f4c49515549444954595f4255524e4544a265627a7a7231582049bc4a7953ebb0a539de5f6ef3311f91d85729247332bc6a48bec016441804ce64736f6c63430005100032454950373132446f6d61696e28737472696e67206e616d652c737472696e672076657273696f6e2c75696e7432353620636861696e49642c6164647265737320766572696679696e67436f6e747261637429a265627a7a723158205bdf8943c0c184cd8ce54c184e5f349195ce97c7894911ed84364b642d33617664736f6c63430005100032