ETH Price: $2,034.70 (-1.95%)

Transaction Decoder

Block:
14580913 at Apr-14-2022 02:41:24 AM +UTC
Transaction Fee:
0.003233857646693824 ETH $6.58
Gas Used:
77,552 Gas / 41.699216612 Gwei

Emitted Events:

Account State Difference:

  Address   Before After State Difference Code
0x4E15361F...873717870
0x4F1BF140...D58cb6Da8
0.059078991188998273 Eth
Nonce: 26
0.055845133542304449 Eth
Nonce: 27
0.003233857646693824
(Miner: 0xb7e...707)
29.217269325563779745 Eth29.217385653563779745 Eth0.000116328

Execution Trace

AnyswapV3Router.anySwapOutUnderlying( token=0xB01371072fDcB9B4433b855e16A682B461F94AB3, to=0x4F1BF1406B726653d3c9f167D06a2D6D58cb6Da8, amount=424742825599421424039, toChainID=250 )
  • AnyswapV5ERC20.STATICCALL( )
  • FantomToken.transferFrom( _from=0x4F1BF1406B726653d3c9f167D06a2D6D58cb6Da8, _to=0xB01371072fDcB9B4433b855e16A682B461F94AB3, _amount=424742825599421424039 ) => ( success=True )
  • AnyswapV5ERC20.depositVault( amount=424742825599421424039, to=0x4F1BF1406B726653d3c9f167D06a2D6D58cb6Da8 ) => ( 424742825599421424039 )
  • AnyswapV5ERC20.burn( from=0x4F1BF1406B726653d3c9f167D06a2D6D58cb6Da8, amount=424742825599421424039 ) => ( True )
    File 1 of 3: AnyswapV3Router
    /**
     *Submitted for verification at BscScan.com on 2021-10-28
    */
    
    /**
     *Submitted for verification at arbiscan.io on 2021-09-22
    */
    
    // SPDX-License-Identifier: GPL-3.0-or-later
    
    pragma solidity >=0.8.2;
    
    interface ISushiswapV2Pair {
        event Approval(address indexed owner, address indexed spender, uint value);
        event Transfer(address indexed from, address indexed to, uint value);
    
        function name() external pure returns (string memory);
        function symbol() external pure returns (string memory);
        function decimals() external pure returns (uint8);
        function totalSupply() external view returns (uint);
        function balanceOf(address owner) external view returns (uint);
        function allowance(address owner, address spender) external view returns (uint);
    
        function approve(address spender, uint value) external returns (bool);
        function transfer(address to, uint value) external returns (bool);
        function transferFrom(address from, address to, uint value) external returns (bool);
    
        function DOMAIN_SEPARATOR() external view returns (bytes32);
        function PERMIT_TYPEHASH() external pure returns (bytes32);
        function nonces(address owner) external view returns (uint);
    
        function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external;
    
        event Mint(address indexed sender, uint amount0, uint amount1);
        event Burn(address indexed sender, uint amount0, uint amount1, address indexed to);
        event Swap(
            address indexed sender,
            uint amount0In,
            uint amount1In,
            uint amount0Out,
            uint amount1Out,
            address indexed to
        );
        event Sync(uint112 reserve0, uint112 reserve1);
    
        function MINIMUM_LIQUIDITY() external pure returns (uint);
        function factory() external view returns (address);
        function token0() external view returns (address);
        function token1() external view returns (address);
        function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);
        function price0CumulativeLast() external view returns (uint);
        function price1CumulativeLast() external view returns (uint);
        function kLast() external view returns (uint);
    
        function mint(address to) external returns (uint liquidity);
        function burn(address to) external returns (uint amount0, uint amount1);
        function swap(uint amount0Out, uint amount1Out, address to, bytes calldata data) external;
        function skim(address to) external;
        function sync() external;
    
        function initialize(address, address) external;
    }
    
    // a library for performing overflow-safe math, courtesy of DappHub (https://github.com/dapphub/ds-math)
    
    library SafeMathSushiswap {
        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');
        }
    }
    
    library SushiswapV2Library {
        using SafeMathSushiswap for uint;
    
        // returns sorted token addresses, used to handle return values from pairs sorted in this order
        function sortTokens(address tokenA, address tokenB) internal pure returns (address token0, address token1) {
            require(tokenA != tokenB, 'SushiswapV2Library: IDENTICAL_ADDRESSES');
            (token0, token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA);
            require(token0 != address(0), 'SushiswapV2Library: ZERO_ADDRESS');
        }
    
        // calculates the CREATE2 address for a pair without making any external calls
        function pairFor(address factory, address tokenA, address tokenB) internal pure returns (address pair) {
            (address token0, address token1) = sortTokens(tokenA, tokenB);
            pair = address(uint160(uint256(keccak256(abi.encodePacked(
                    hex'ff',
                    factory,
                    keccak256(abi.encodePacked(token0, token1)),
                    hex'e18a34eb0e04b04f7a0ac29a6e80748dca96319b42c54d679cb821dca90c6303' // init code hash
                )))));
        }
    
        // fetches and sorts the reserves for a pair
        function getReserves(address factory, address tokenA, address tokenB) internal view returns (uint reserveA, uint reserveB) {
            (address token0,) = sortTokens(tokenA, tokenB);
            (uint reserve0, uint reserve1,) = ISushiswapV2Pair(pairFor(factory, tokenA, tokenB)).getReserves();
            (reserveA, reserveB) = tokenA == token0 ? (reserve0, reserve1) : (reserve1, reserve0);
        }
    
        // given some amount of an asset and pair reserves, returns an equivalent amount of the other asset
        function quote(uint amountA, uint reserveA, uint reserveB) internal pure returns (uint amountB) {
            require(amountA > 0, 'SushiswapV2Library: INSUFFICIENT_AMOUNT');
            require(reserveA > 0 && reserveB > 0, 'SushiswapV2Library: INSUFFICIENT_LIQUIDITY');
            amountB = amountA.mul(reserveB) / reserveA;
        }
    
        // given an input amount of an asset and pair reserves, returns the maximum output amount of the other asset
        function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) internal pure returns (uint amountOut) {
            require(amountIn > 0, 'SushiswapV2Library: INSUFFICIENT_INPUT_AMOUNT');
            require(reserveIn > 0 && reserveOut > 0, 'SushiswapV2Library: INSUFFICIENT_LIQUIDITY');
            uint amountInWithFee = amountIn.mul(997);
            uint numerator = amountInWithFee.mul(reserveOut);
            uint denominator = reserveIn.mul(1000).add(amountInWithFee);
            amountOut = numerator / denominator;
        }
    
        // given an output amount of an asset and pair reserves, returns a required input amount of the other asset
        function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) internal pure returns (uint amountIn) {
            require(amountOut > 0, 'SushiswapV2Library: INSUFFICIENT_OUTPUT_AMOUNT');
            require(reserveIn > 0 && reserveOut > 0, 'SushiswapV2Library: INSUFFICIENT_LIQUIDITY');
            uint numerator = reserveIn.mul(amountOut).mul(1000);
            uint denominator = reserveOut.sub(amountOut).mul(997);
            amountIn = (numerator / denominator).add(1);
        }
    
        // performs chained getAmountOut calculations on any number of pairs
        function getAmountsOut(address factory, uint amountIn, address[] memory path) internal view returns (uint[] memory amounts) {
            require(path.length >= 2, 'SushiswapV2Library: INVALID_PATH');
            amounts = new uint[](path.length);
            amounts[0] = amountIn;
            for (uint i; i < path.length - 1; i++) {
                (uint reserveIn, uint reserveOut) = getReserves(factory, path[i], path[i + 1]);
                amounts[i + 1] = getAmountOut(amounts[i], reserveIn, reserveOut);
            }
        }
    
        // performs chained getAmountIn calculations on any number of pairs
        function getAmountsIn(address factory, uint amountOut, address[] memory path) internal view returns (uint[] memory amounts) {
            require(path.length >= 2, 'SushiswapV2Library: INVALID_PATH');
            amounts = new uint[](path.length);
            amounts[amounts.length - 1] = amountOut;
            for (uint i = path.length - 1; i > 0; i--) {
                (uint reserveIn, uint reserveOut) = getReserves(factory, path[i - 1], path[i]);
                amounts[i - 1] = getAmountIn(amounts[i], reserveIn, reserveOut);
            }
        }
    }
    
    // helper methods for interacting with ERC20 tokens and sending NATIVE that do not consistently return true/false
    library TransferHelper {
        function safeApprove(address token, address to, uint value) internal {
            // bytes4(keccak256(bytes('approve(address,uint256)')));
            (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x095ea7b3, to, value));
            require(success && (data.length == 0 || abi.decode(data, (bool))), 'TransferHelper: APPROVE_FAILED');
        }
    
        function safeTransfer(address token, address to, uint value) internal {
            // bytes4(keccak256(bytes('transfer(address,uint256)')));
            (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0xa9059cbb, to, value));
            require(success && (data.length == 0 || abi.decode(data, (bool))), 'TransferHelper: TRANSFER_FAILED');
        }
    
        function safeTransferFrom(address token, address from, address to, uint value) internal {
            // bytes4(keccak256(bytes('transferFrom(address,address,uint256)')));
            (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x23b872dd, from, to, value));
            require(success && (data.length == 0 || abi.decode(data, (bool))), 'TransferHelper: TRANSFER_FROM_FAILED');
        }
    
        function safeTransferNative(address to, uint value) internal {
            (bool success,) = to.call{value:value}(new bytes(0));
            require(success, 'TransferHelper: NATIVE_TRANSFER_FAILED');
        }
    }
    
    interface ISushiswapV2Factory {
        event PairCreated(address indexed token0, address indexed token1, address pair, uint);
    
        function feeTo() external view returns (address);
        function feeToSetter() external view returns (address);
        function migrator() external view returns (address);
    
        function getPair(address tokenA, address tokenB) external view returns (address pair);
        function allPairs(uint) external view returns (address pair);
        function allPairsLength() external view returns (uint);
    
        function createPair(address tokenA, address tokenB) external returns (address pair);
    
        function setFeeTo(address) external;
        function setFeeToSetter(address) external;
        function setMigrator(address) external;
    }
    
    interface IwNATIVE {
        function deposit() external payable;
        function transfer(address to, uint value) external returns (bool);
        function withdraw(uint) external;
    }
    
    interface AnyswapV1ERC20 {
        function mint(address to, uint256 amount) external returns (bool);
        function burn(address from, uint256 amount) external returns (bool);
        function changeVault(address newVault) external returns (bool);
        function depositVault(uint amount, address to) external returns (uint);
        function withdrawVault(address from, uint amount, address to) external returns (uint);
        function underlying() external view returns (address);
        function deposit(uint amount, address to) external returns (uint);
        function withdraw(uint amount, address to) external returns (uint);
    }
    
    /**
     * @dev Interface of the ERC20 standard as defined in the EIP.
     */
    interface IERC20 {
        function totalSupply() external view returns (uint256);
        function balanceOf(address account) external view returns (uint256);
        function transfer(address recipient, uint256 amount) external returns (bool);
        function allowance(address owner, address spender) external view returns (uint256);
        function approve(address spender, uint256 amount) external returns (bool);
        function permit(address target, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external;
        function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
        function transferWithPermit(address target, address to, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external returns (bool);
    
        event Transfer(address indexed from, address indexed to, uint256 value);
        event Approval(address indexed owner, address indexed spender, uint256 value);
    }
    
    library Address {
        function isContract(address account) internal view returns (bool) {
            bytes32 codehash;
            bytes32 accountHash = 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470;
            // solhint-disable-next-line no-inline-assembly
            assembly { codehash := extcodehash(account) }
            return (codehash != 0x0 && codehash != accountHash);
        }
    }
    
    library SafeERC20 {
        using Address for address;
    
        function safeTransfer(IERC20 token, address to, uint value) internal {
            callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
        }
    
        function safeTransferFrom(IERC20 token, address from, address to, uint value) internal {
            callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
        }
    
        function safeApprove(IERC20 token, address spender, uint value) internal {
            require((value == 0) || (token.allowance(address(this), spender) == 0),
                "SafeERC20: approve from non-zero to non-zero allowance"
            );
            callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
        }
        function callOptionalReturn(IERC20 token, bytes memory data) private {
            require(address(token).isContract(), "SafeERC20: call to non-contract");
    
            // solhint-disable-next-line avoid-low-level-calls
            (bool success, bytes memory returndata) = address(token).call(data);
            require(success, "SafeERC20: low-level call failed");
    
            if (returndata.length > 0) { // Return data is optional
                // solhint-disable-next-line max-line-length
                require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
            }
        }
    }
    
    contract AnyswapV3Router {
        using SafeERC20 for IERC20;
        using SafeMathSushiswap for uint;
    
        address public immutable factory;
        address public immutable wNATIVE;
    
        modifier ensure(uint deadline) {
            require(deadline >= block.timestamp, 'AnyswapV3Router: EXPIRED');
            _;
        }
    
        constructor(address _factory, address _wNATIVE, address _mpc) {
            _newMPC = _mpc;
            _newMPCEffectiveTime = block.timestamp;
            factory = _factory;
            wNATIVE = _wNATIVE;
        }
    
        receive() external payable {
            assert(msg.sender == wNATIVE); // only accept Native via fallback from the wNative contract
        }
    
        address private _oldMPC;
        address private _newMPC;
        uint256 private _newMPCEffectiveTime;
    
    
        event LogChangeMPC(address indexed oldMPC, address indexed newMPC, uint indexed effectiveTime, uint chainID);
        event LogChangeRouter(address indexed oldRouter, address indexed newRouter, uint chainID);
        event LogAnySwapIn(bytes32 indexed txhash, address indexed token, address indexed to, uint amount, uint fromChainID, uint toChainID);
        event LogAnySwapOut(address indexed token, address indexed from, address indexed to, uint amount, uint fromChainID, uint toChainID);
        event LogAnySwapTradeTokensForTokens(address[] path, address indexed from, address indexed to, uint amountIn, uint amountOutMin, uint fromChainID, uint toChainID);
        event LogAnySwapTradeTokensForNative(address[] path, address indexed from, address indexed to, uint amountIn, uint amountOutMin, uint fromChainID, uint toChainID);
    
        modifier onlyMPC() {
            require(msg.sender == mpc(), "AnyswapV3Router: FORBIDDEN");
            _;
        }
    
        function mpc() public view returns (address) {
            if (block.timestamp >= _newMPCEffectiveTime) {
                return _newMPC;
            }
            return _oldMPC;
        }
    
        function cID() public view returns (uint id) {
            assembly {id := chainid()}
        }
    
        function changeMPC(address newMPC) public onlyMPC returns (bool) {
            require(newMPC != address(0), "AnyswapV3Router: address(0x0)");
            _oldMPC = mpc();
            _newMPC = newMPC;
            _newMPCEffectiveTime = block.timestamp + 2*24*3600;
            emit LogChangeMPC(_oldMPC, _newMPC, _newMPCEffectiveTime, cID());
            return true;
        }
    
        function changeVault(address token, address newVault) public onlyMPC returns (bool) {
            require(newVault != address(0), "AnyswapV3Router: address(0x0)");
            return AnyswapV1ERC20(token).changeVault(newVault);
        }
    
        function _anySwapOut(address from, address token, address to, uint amount, uint toChainID) internal {
            AnyswapV1ERC20(token).burn(from, amount);
            emit LogAnySwapOut(token, from, to, amount, cID(), toChainID);
        }
    
        // Swaps `amount` `token` from this chain to `toChainID` chain with recipient `to`
        function anySwapOut(address token, address to, uint amount, uint toChainID) external {
            _anySwapOut(msg.sender, token, to, amount, toChainID);
        }
    
        // Swaps `amount` `token` from this chain to `toChainID` chain with recipient `to` by minting with `underlying`
        function anySwapOutUnderlying(address token, address to, uint amount, uint toChainID) external {
            IERC20(AnyswapV1ERC20(token).underlying()).safeTransferFrom(msg.sender, token, amount);
            AnyswapV1ERC20(token).depositVault(amount, msg.sender);
            _anySwapOut(msg.sender, token, to, amount, toChainID);
        }
    
        function anySwapOutNative(address token, address to, uint toChainID) external payable {
            require(AnyswapV1ERC20(token).underlying() == wNATIVE, "AnyswapV3Router: underlying is not wNATIVE");
            IwNATIVE(wNATIVE).deposit{value: msg.value}();
            assert(IwNATIVE(wNATIVE).transfer(token, msg.value));
            AnyswapV1ERC20(token).depositVault(msg.value, msg.sender);
            _anySwapOut(msg.sender, token, to, msg.value, toChainID);
        }
    
        function anySwapOutUnderlyingWithPermit(
            address from,
            address token,
            address to,
            uint amount,
            uint deadline,
            uint8 v,
            bytes32 r,
            bytes32 s,
            uint toChainID
        ) external {
            address _underlying = AnyswapV1ERC20(token).underlying();
            IERC20(_underlying).permit(from, address(this), amount, deadline, v, r, s);
            IERC20(_underlying).safeTransferFrom(from, token, amount);
            AnyswapV1ERC20(token).depositVault(amount, from);
            _anySwapOut(from, token, to, amount, toChainID);
        }
    
        function anySwapOutUnderlyingWithTransferPermit(
            address from,
            address token,
            address to,
            uint amount,
            uint deadline,
            uint8 v,
            bytes32 r,
            bytes32 s,
            uint toChainID
        ) external {
            IERC20(AnyswapV1ERC20(token).underlying()).transferWithPermit(from, token, amount, deadline, v, r, s);
            AnyswapV1ERC20(token).depositVault(amount, from);
            _anySwapOut(from, token, to, amount, toChainID);
        }
    
        function anySwapOut(address[] calldata tokens, address[] calldata to, uint[] calldata amounts, uint[] calldata toChainIDs) external {
            for (uint i = 0; i < tokens.length; i++) {
                _anySwapOut(msg.sender, tokens[i], to[i], amounts[i], toChainIDs[i]);
            }
        }
    
        // swaps `amount` `token` in `fromChainID` to `to` on this chainID
        function _anySwapIn(bytes32 txs, address token, address to, uint amount, uint fromChainID) internal {
            AnyswapV1ERC20(token).mint(to, amount);
            emit LogAnySwapIn(txs, token, to, amount, fromChainID, cID());
        }
    
        // swaps `amount` `token` in `fromChainID` to `to` on this chainID
        // triggered by `anySwapOut`
        function anySwapIn(bytes32 txs, address token, address to, uint amount, uint fromChainID) external onlyMPC {
            _anySwapIn(txs, token, to, amount, fromChainID);
        }
    
        // swaps `amount` `token` in `fromChainID` to `to` on this chainID with `to` receiving `underlying`
        function anySwapInUnderlying(bytes32 txs, address token, address to, uint amount, uint fromChainID) external onlyMPC {
            _anySwapIn(txs, token, to, amount, fromChainID);
            AnyswapV1ERC20(token).withdrawVault(to, amount, to);
        }
    
        // swaps `amount` `token` in `fromChainID` to `to` on this chainID with `to` receiving `underlying` if possible
        function anySwapInAuto(bytes32 txs, address token, address to, uint amount, uint fromChainID) external onlyMPC {
            _anySwapIn(txs, token, to, amount, fromChainID);
            AnyswapV1ERC20 _anyToken = AnyswapV1ERC20(token);
            address _underlying = _anyToken.underlying();
            if (_underlying != address(0) && IERC20(_underlying).balanceOf(token) >= amount) {
                if (_underlying == wNATIVE) {
                    _anyToken.withdrawVault(to, amount, address(this));
                    IwNATIVE(wNATIVE).withdraw(amount);
                    TransferHelper.safeTransferNative(to, amount);
                } else {
                    _anyToken.withdrawVault(to, amount, to);
                }
            }
        }
    
        function depositNative(address token, address to) external payable returns (uint) {
            require(AnyswapV1ERC20(token).underlying() == wNATIVE, "AnyswapV3Router: underlying is not wNATIVE");
            IwNATIVE(wNATIVE).deposit{value: msg.value}();
            assert(IwNATIVE(wNATIVE).transfer(token, msg.value));
            AnyswapV1ERC20(token).depositVault(msg.value, to);
            return msg.value;
        }
    
        function withdrawNative(address token, uint amount, address to) external returns (uint) {
            require(AnyswapV1ERC20(token).underlying() == wNATIVE, "AnyswapV3Router: underlying is not wNATIVE");
            AnyswapV1ERC20(token).withdrawVault(msg.sender, amount, address(this));
            IwNATIVE(wNATIVE).withdraw(amount);
            TransferHelper.safeTransferNative(to, amount);
            return amount;
        }
    
        // extracts mpc fee from bridge fees
        function anySwapFeeTo(address token, uint amount) external onlyMPC {
            address _mpc = mpc();
            AnyswapV1ERC20(token).mint(_mpc, amount);
            AnyswapV1ERC20(token).withdrawVault(_mpc, amount, _mpc);
        }
    
        function anySwapIn(bytes32[] calldata txs, address[] calldata tokens, address[] calldata to, uint256[] calldata amounts, uint[] calldata fromChainIDs) external onlyMPC {
            for (uint i = 0; i < tokens.length; i++) {
                _anySwapIn(txs[i], tokens[i], to[i], amounts[i], fromChainIDs[i]);
            }
        }
    
        // **** SWAP ****
        // requires the initial amount to have already been sent to the first pair
        function _swap(uint[] memory amounts, address[] memory path, address _to) internal virtual {
            for (uint i; i < path.length - 1; i++) {
                (address input, address output) = (path[i], path[i + 1]);
                (address token0,) = SushiswapV2Library.sortTokens(input, output);
                uint amountOut = amounts[i + 1];
                (uint amount0Out, uint amount1Out) = input == token0 ? (uint(0), amountOut) : (amountOut, uint(0));
                address to = i < path.length - 2 ? SushiswapV2Library.pairFor(factory, output, path[i + 2]) : _to;
                ISushiswapV2Pair(SushiswapV2Library.pairFor(factory, input, output)).swap(
                    amount0Out, amount1Out, to, new bytes(0)
                );
            }
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForTokens(
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint toChainID
        ) external virtual ensure(deadline) {
            AnyswapV1ERC20(path[0]).burn(msg.sender, amountIn);
            emit LogAnySwapTradeTokensForTokens(path, msg.sender, to, amountIn, amountOutMin, cID(), toChainID);
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForTokensUnderlying(
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint toChainID
        ) external virtual ensure(deadline) {
            IERC20(AnyswapV1ERC20(path[0]).underlying()).safeTransferFrom(msg.sender, path[0], amountIn);
            AnyswapV1ERC20(path[0]).depositVault(amountIn, msg.sender);
            AnyswapV1ERC20(path[0]).burn(msg.sender, amountIn);
            emit LogAnySwapTradeTokensForTokens(path, msg.sender, to, amountIn, amountOutMin, cID(), toChainID);
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForTokensUnderlyingWithPermit(
            address from,
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint8 v,
            bytes32 r,
            bytes32 s,
            uint toChainID
        ) external virtual ensure(deadline) {
            address _underlying = AnyswapV1ERC20(path[0]).underlying();
            IERC20(_underlying).permit(from, address(this), amountIn, deadline, v, r, s);
            IERC20(_underlying).safeTransferFrom(from, path[0], amountIn);
            AnyswapV1ERC20(path[0]).depositVault(amountIn, from);
            AnyswapV1ERC20(path[0]).burn(from, amountIn);
            {
            address[] memory _path = path;
            address _from = from;
            address _to = to;
            uint _amountIn = amountIn;
            uint _amountOutMin = amountOutMin;
            uint _cID = cID();
            uint _toChainID = toChainID;
            emit LogAnySwapTradeTokensForTokens(_path, _from, _to, _amountIn, _amountOutMin, _cID, _toChainID);
            }
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForTokensUnderlyingWithTransferPermit(
            address from,
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint8 v,
            bytes32 r,
            bytes32 s,
            uint toChainID
        ) external virtual ensure(deadline) {
            IERC20(AnyswapV1ERC20(path[0]).underlying()).transferWithPermit(from, path[0], amountIn, deadline, v, r, s);
            AnyswapV1ERC20(path[0]).depositVault(amountIn, from);
            AnyswapV1ERC20(path[0]).burn(from, amountIn);
            emit LogAnySwapTradeTokensForTokens(path, from, to, amountIn, amountOutMin, cID(), toChainID);
        }
    
        // Swaps `amounts[path.length-1]` `path[path.length-1]` to `to` on this chain
        // Triggered by `anySwapOutExactTokensForTokens`
        function anySwapInExactTokensForTokens(
            bytes32 txs,
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint fromChainID
        ) external onlyMPC virtual ensure(deadline) returns (uint[] memory amounts) {
            amounts = SushiswapV2Library.getAmountsOut(factory, amountIn, path);
            require(amounts[amounts.length - 1] >= amountOutMin, 'SushiswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT');
            _anySwapIn(txs, path[0], SushiswapV2Library.pairFor(factory, path[0], path[1]), amounts[0], fromChainID);
            _swap(amounts, path, to);
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForNative(
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint toChainID
        ) external virtual ensure(deadline) {
            AnyswapV1ERC20(path[0]).burn(msg.sender, amountIn);
            emit LogAnySwapTradeTokensForNative(path, msg.sender, to, amountIn, amountOutMin, cID(), toChainID);
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForNativeUnderlying(
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint toChainID
        ) external virtual ensure(deadline) {
            IERC20(AnyswapV1ERC20(path[0]).underlying()).safeTransferFrom(msg.sender, path[0], amountIn);
            AnyswapV1ERC20(path[0]).depositVault(amountIn, msg.sender);
            AnyswapV1ERC20(path[0]).burn(msg.sender, amountIn);
            emit LogAnySwapTradeTokensForNative(path, msg.sender, to, amountIn, amountOutMin, cID(), toChainID);
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForNativeUnderlyingWithPermit(
            address from,
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint8 v,
            bytes32 r,
            bytes32 s,
            uint toChainID
        ) external virtual ensure(deadline) {
            address _underlying = AnyswapV1ERC20(path[0]).underlying();
            IERC20(_underlying).permit(from, address(this), amountIn, deadline, v, r, s);
            IERC20(_underlying).safeTransferFrom(from, path[0], amountIn);
            AnyswapV1ERC20(path[0]).depositVault(amountIn, from);
            AnyswapV1ERC20(path[0]).burn(from, amountIn);
            {
            address[] memory _path = path;
            address _from = from;
            address _to = to;
            uint _amountIn = amountIn;
            uint _amountOutMin = amountOutMin;
            uint _cID = cID();
            uint _toChainID = toChainID;
            emit LogAnySwapTradeTokensForNative(_path, _from, _to, _amountIn, _amountOutMin, _cID, _toChainID);
            }
        }
    
        // sets up a cross-chain trade from this chain to `toChainID` for `path` trades to `to`
        function anySwapOutExactTokensForNativeUnderlyingWithTransferPermit(
            address from,
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint8 v,
            bytes32 r,
            bytes32 s,
            uint toChainID
        ) external virtual ensure(deadline) {
            IERC20(AnyswapV1ERC20(path[0]).underlying()).transferWithPermit(from, path[0], amountIn, deadline, v, r, s);
            AnyswapV1ERC20(path[0]).depositVault(amountIn, from);
            AnyswapV1ERC20(path[0]).burn(from, amountIn);
            emit LogAnySwapTradeTokensForNative(path, from, to, amountIn, amountOutMin, cID(), toChainID);
        }
    
        // Swaps `amounts[path.length-1]` `path[path.length-1]` to `to` on this chain
        // Triggered by `anySwapOutExactTokensForNative`
        function anySwapInExactTokensForNative(
            bytes32 txs,
            uint amountIn,
            uint amountOutMin,
            address[] calldata path,
            address to,
            uint deadline,
            uint fromChainID
        ) external onlyMPC virtual ensure(deadline) returns (uint[] memory amounts) {
            require(path[path.length - 1] == wNATIVE, 'AnyswapV3Router: INVALID_PATH');
            amounts = SushiswapV2Library.getAmountsOut(factory, amountIn, path);
            require(amounts[amounts.length - 1] >= amountOutMin, 'AnyswapV3Router: INSUFFICIENT_OUTPUT_AMOUNT');
            _anySwapIn(txs, path[0],  SushiswapV2Library.pairFor(factory, path[0], path[1]), amounts[0], fromChainID);
            _swap(amounts, path, address(this));
            IwNATIVE(wNATIVE).withdraw(amounts[amounts.length - 1]);
            TransferHelper.safeTransferNative(to, amounts[amounts.length - 1]);
        }
    
        // **** LIBRARY FUNCTIONS ****
        function quote(uint amountA, uint reserveA, uint reserveB) public pure virtual returns (uint amountB) {
            return SushiswapV2Library.quote(amountA, reserveA, reserveB);
        }
    
        function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut)
            public
            pure
            virtual
            returns (uint amountOut)
        {
            return SushiswapV2Library.getAmountOut(amountIn, reserveIn, reserveOut);
        }
    
        function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut)
            public
            pure
            virtual
            returns (uint amountIn)
        {
            return SushiswapV2Library.getAmountIn(amountOut, reserveIn, reserveOut);
        }
    
        function getAmountsOut(uint amountIn, address[] memory path)
            public
            view
            virtual
            returns (uint[] memory amounts)
        {
            return SushiswapV2Library.getAmountsOut(factory, amountIn, path);
        }
    
        function getAmountsIn(uint amountOut, address[] memory path)
            public
            view
            virtual
            returns (uint[] memory amounts)
        {
            return SushiswapV2Library.getAmountsIn(factory, amountOut, path);
        }
    }

    File 2 of 3: AnyswapV5ERC20
    /**
     *Submitted for verification at BscScan.com on 2021-06-15
    */
    
    /**
     *Submitted for verification at BscScan.com on 2021-06-11
    */
    
    /**
     *Submitted for verification at polygonscan.com on 2021-06-11
    */
    
    /**
     *Submitted for verification at Etherscan.io on 2021-06-08
    */
    
    /**
     *Submitted for verification at Etherscan.io on 2021-06-07
    */
    
    // SPDX-License-Identifier: GPL-3.0-or-later
    
    pragma solidity 0.8.2;
    
    /**
     * @dev Interface of the ERC20 standard as defined in the EIP.
     */
    interface IERC20 {
        function totalSupply() external view returns (uint256);
        function decimals() external view returns (uint8);
        function balanceOf(address account) external view returns (uint256);
        function transfer(address recipient, uint256 amount) external returns (bool);
        function allowance(address owner, address spender) external view returns (uint256);
        function approve(address spender, uint256 amount) external returns (bool);
        function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
        function permit(address target, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external;
        function transferWithPermit(address target, address to, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external returns (bool);
        event Transfer(address indexed from, address indexed to, uint256 value);
        event Approval(address indexed owner, address indexed spender, uint256 value);
    }
    
    /**
     * @dev Interface of the ERC2612 standard as defined in the EIP.
     *
     * Adds the {permit} method, which can be used to change one's
     * {IERC20-allowance} without having to send a transaction, by signing a
     * message. This allows users to spend tokens without having to hold Ether.
     *
     * See https://eips.ethereum.org/EIPS/eip-2612.
     */
    interface IERC2612 {
    
        /**
         * @dev Returns the current ERC2612 nonce for `owner`. This value must be
         * included whenever a signature is generated for {permit}.
         *
         * Every successful call to {permit} increases ``owner``'s nonce by one. This
         * prevents a signature from being used multiple times.
         */
        function nonces(address owner) external view returns (uint256);
    }
    
    /// @dev Wrapped ERC-20 v10 (AnyswapV3ERC20) is an ERC-20 ERC-20 wrapper. You can `deposit` ERC-20 and obtain an AnyswapV3ERC20 balance which can then be operated as an ERC-20 token. You can
    /// `withdraw` ERC-20 from AnyswapV3ERC20, which will then burn AnyswapV3ERC20 token in your wallet. The amount of AnyswapV3ERC20 token in any wallet is always identical to the
    /// balance of ERC-20 deposited minus the ERC-20 withdrawn with that specific wallet.
    interface IAnyswapV3ERC20 is IERC20, IERC2612 {
    
        /// @dev Sets `value` as allowance of `spender` account over caller account's AnyswapV3ERC20 token,
        /// after which a call is executed to an ERC677-compliant contract with the `data` parameter.
        /// Emits {Approval} event.
        /// Returns boolean value indicating whether operation succeeded.
        /// For more information on approveAndCall format, see https://github.com/ethereum/EIPs/issues/677.
        function approveAndCall(address spender, uint256 value, bytes calldata data) external returns (bool);
    
        /// @dev Moves `value` AnyswapV3ERC20 token from caller's account to account (`to`),
        /// after which a call is executed to an ERC677-compliant contract with the `data` parameter.
        /// A transfer to `address(0)` triggers an ERC-20 withdraw matching the sent AnyswapV3ERC20 token in favor of caller.
        /// Emits {Transfer} event.
        /// Returns boolean value indicating whether operation succeeded.
        /// Requirements:
        ///   - caller account must have at least `value` AnyswapV3ERC20 token.
        /// For more information on transferAndCall format, see https://github.com/ethereum/EIPs/issues/677.
        function transferAndCall(address to, uint value, bytes calldata data) external returns (bool);
    }
    
    interface ITransferReceiver {
        function onTokenTransfer(address, uint, bytes calldata) external returns (bool);
    }
    
    interface IApprovalReceiver {
        function onTokenApproval(address, uint, bytes calldata) external returns (bool);
    }
    
    library Address {
        function isContract(address account) internal view returns (bool) {
            bytes32 codehash;
            bytes32 accountHash = 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470;
            // solhint-disable-next-line no-inline-assembly
            assembly { codehash := extcodehash(account) }
            return (codehash != 0x0 && codehash != accountHash);
        }
    }
    
    library SafeERC20 {
        using Address for address;
    
        function safeTransfer(IERC20 token, address to, uint value) internal {
            callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
        }
    
        function safeTransferFrom(IERC20 token, address from, address to, uint value) internal {
            callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
        }
    
        function safeApprove(IERC20 token, address spender, uint value) internal {
            require((value == 0) || (token.allowance(address(this), spender) == 0),
                "SafeERC20: approve from non-zero to non-zero allowance"
            );
            callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
        }
        function callOptionalReturn(IERC20 token, bytes memory data) private {
            require(address(token).isContract(), "SafeERC20: call to non-contract");
    
            // solhint-disable-next-line avoid-low-level-calls
            (bool success, bytes memory returndata) = address(token).call(data);
            require(success, "SafeERC20: low-level call failed");
    
            if (returndata.length > 0) { // Return data is optional
                // solhint-disable-next-line max-line-length
                require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
            }
        }
    }
    
    contract AnyswapV5ERC20 is IAnyswapV3ERC20 {
        using SafeERC20 for IERC20;
        string public name;
        string public symbol;
        uint8  public immutable override decimals;
    
        address public immutable underlying;
    
        bytes32 public constant PERMIT_TYPEHASH = keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
        bytes32 public constant TRANSFER_TYPEHASH = keccak256("Transfer(address owner,address to,uint256 value,uint256 nonce,uint256 deadline)");
        bytes32 public immutable DOMAIN_SEPARATOR;
    
        /// @dev Records amount of AnyswapV3ERC20 token owned by account.
        mapping (address => uint256) public override balanceOf;
        uint256 private _totalSupply;
    
        // init flag for setting immediate vault, needed for CREATE2 support
        bool private _init;
    
        // flag to enable/disable swapout vs vault.burn so multiple events are triggered
        bool private _vaultOnly;
    
        // configurable delay for timelock functions
        uint public delay = 2*24*3600;
    
    
        // set of minters, can be this bridge or other bridges
        mapping(address => bool) public isMinter;
        address[] public minters;
    
        // primary controller of the token contract
        address public vault;
    
        address public pendingMinter;
        uint public delayMinter;
    
        address public pendingVault;
        uint public delayVault;
    
        uint public pendingDelay;
        uint public delayDelay;
    
    
        modifier onlyAuth() {
            require(isMinter[msg.sender], "AnyswapV4ERC20: FORBIDDEN");
            _;
        }
    
        modifier onlyVault() {
            require(msg.sender == mpc(), "AnyswapV3ERC20: FORBIDDEN");
            _;
        }
    
        function owner() public view returns (address) {
            return mpc();
        }
    
        function mpc() public view returns (address) {
            if (block.timestamp >= delayVault) {
                return pendingVault;
            }
            return vault;
        }
    
        function setVaultOnly(bool enabled) external onlyVault {
            _vaultOnly = enabled;
        }
    
        function initVault(address _vault) external onlyVault {
            require(_init);
            vault = _vault;
            pendingVault = _vault;
            isMinter[_vault] = true;
            minters.push(_vault);
            delayVault = block.timestamp;
            _init = false;
        }
    
        function setMinter(address _auth) external onlyVault {
            pendingMinter = _auth;
            delayMinter = block.timestamp + delay;
        }
    
        function setVault(address _vault) external onlyVault {
            pendingVault = _vault;
            delayVault = block.timestamp + delay;
        }
    
        function applyVault() external onlyVault {
            require(block.timestamp >= delayVault);
            vault = pendingVault;
        }
    
        function applyMinter() external onlyVault {
            require(block.timestamp >= delayMinter);
            isMinter[pendingMinter] = true;
            minters.push(pendingMinter);
        }
    
        // No time delay revoke minter emergency function
        function revokeMinter(address _auth) external onlyVault {
            isMinter[_auth] = false;
        }
    
        function getAllMinters() external view returns (address[] memory) {
            return minters;
        }
    
    
        function changeVault(address newVault) external onlyVault returns (bool) {
            require(newVault != address(0), "AnyswapV3ERC20: address(0x0)");
            pendingVault = newVault;
            delayVault = block.timestamp + delay;
            emit LogChangeVault(vault, pendingVault, delayVault);
            return true;
        }
    
        function changeMPCOwner(address newVault) public onlyVault returns (bool) {
            require(newVault != address(0), "AnyswapV3ERC20: address(0x0)");
            pendingVault = newVault;
            delayVault = block.timestamp + delay;
            emit LogChangeMPCOwner(vault, pendingVault, delayVault);
            return true;
        }
    
        function mint(address to, uint256 amount) external onlyAuth returns (bool) {
            _mint(to, amount);
            return true;
        }
    
        function burn(address from, uint256 amount) external onlyAuth returns (bool) {
            require(from != address(0), "AnyswapV3ERC20: address(0x0)");
            _burn(from, amount);
            return true;
        }
    
        function Swapin(bytes32 txhash, address account, uint256 amount) public onlyAuth returns (bool) {
            _mint(account, amount);
            emit LogSwapin(txhash, account, amount);
            return true;
        }
    
        function Swapout(uint256 amount, address bindaddr) public returns (bool) {
            require(!_vaultOnly, "AnyswapV4ERC20: onlyAuth");
            require(bindaddr != address(0), "AnyswapV3ERC20: address(0x0)");
            _burn(msg.sender, amount);
            emit LogSwapout(msg.sender, bindaddr, amount);
            return true;
        }
    
        /// @dev Records current ERC2612 nonce for account. This value must be included whenever signature is generated for {permit}.
        /// Every successful call to {permit} increases account's nonce by one. This prevents signature from being used multiple times.
        mapping (address => uint256) public override nonces;
    
        /// @dev Records number of AnyswapV3ERC20 token that account (second) will be allowed to spend on behalf of another account (first) through {transferFrom}.
        mapping (address => mapping (address => uint256)) public override allowance;
    
        event LogChangeVault(address indexed oldVault, address indexed newVault, uint indexed effectiveTime);
        event LogChangeMPCOwner(address indexed oldOwner, address indexed newOwner, uint indexed effectiveHeight);
        event LogSwapin(bytes32 indexed txhash, address indexed account, uint amount);
        event LogSwapout(address indexed account, address indexed bindaddr, uint amount);
        event LogAddAuth(address indexed auth, uint timestamp);
    
        constructor(string memory _name, string memory _symbol, uint8 _decimals, address _underlying, address _vault) {
            name = _name;
            symbol = _symbol;
            decimals = _decimals;
            underlying = _underlying;
            if (_underlying != address(0x0)) {
                require(_decimals == IERC20(_underlying).decimals());
            }
    
            // Use init to allow for CREATE2 accross all chains
            _init = true;
    
            // Disable/Enable swapout for v1 tokens vs mint/burn for v3 tokens
            _vaultOnly = false;
    
            vault = _vault;
            pendingVault = _vault;
            delayVault = block.timestamp;
    
            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)));
        }
    
        /// @dev Returns the total supply of AnyswapV3ERC20 token as the ETH held in this contract.
        function totalSupply() external view override returns (uint256) {
            return _totalSupply;
        }
    
        function depositWithPermit(address target, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s, address to) external returns (uint) {
            IERC20(underlying).permit(target, address(this), value, deadline, v, r, s);
            IERC20(underlying).safeTransferFrom(target, address(this), value);
            return _deposit(value, to);
        }
    
        function depositWithTransferPermit(address target, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s, address to) external returns (uint) {
            IERC20(underlying).transferWithPermit(target, address(this), value, deadline, v, r, s);
            return _deposit(value, to);
        }
    
        function deposit() external returns (uint) {
            uint _amount = IERC20(underlying).balanceOf(msg.sender);
            IERC20(underlying).safeTransferFrom(msg.sender, address(this), _amount);
            return _deposit(_amount, msg.sender);
        }
    
        function deposit(uint amount) external returns (uint) {
            IERC20(underlying).safeTransferFrom(msg.sender, address(this), amount);
            return _deposit(amount, msg.sender);
        }
    
        function deposit(uint amount, address to) external returns (uint) {
            IERC20(underlying).safeTransferFrom(msg.sender, address(this), amount);
            return _deposit(amount, to);
        }
    
        function depositVault(uint amount, address to) external onlyVault returns (uint) {
            return _deposit(amount, to);
        }
    
        function _deposit(uint amount, address to) internal returns (uint) {
            require(underlying != address(0x0) && underlying != address(this));
            _mint(to, amount);
            return amount;
        }
    
        function withdraw() external returns (uint) {
            return _withdraw(msg.sender, balanceOf[msg.sender], msg.sender);
        }
    
        function withdraw(uint amount) external returns (uint) {
            return _withdraw(msg.sender, amount, msg.sender);
        }
    
        function withdraw(uint amount, address to) external returns (uint) {
            return _withdraw(msg.sender, amount, to);
        }
    
        function withdrawVault(address from, uint amount, address to) external onlyVault returns (uint) {
            return _withdraw(from, amount, to);
        }
    
        function _withdraw(address from, uint amount, address to) internal returns (uint) {
            _burn(from, amount);
            IERC20(underlying).safeTransfer(to, amount);
            return amount;
        }
    
        /** @dev Creates `amount` tokens and assigns them to `account`, increasing
         * the total supply.
         *
         * Emits a {Transfer} event with `from` set to the zero address.
         *
         * Requirements
         *
         * - `to` cannot be the zero address.
         */
        function _mint(address account, uint256 amount) internal {
            require(account != address(0), "ERC20: mint to the zero address");
    
            _totalSupply += amount;
            balanceOf[account] += amount;
            emit Transfer(address(0), account, amount);
        }
    
        /**
         * @dev Destroys `amount` tokens from `account`, reducing the
         * total supply.
         *
         * Emits a {Transfer} event with `to` set to the zero address.
         *
         * Requirements
         *
         * - `account` cannot be the zero address.
         * - `account` must have at least `amount` tokens.
         */
        function _burn(address account, uint256 amount) internal {
            require(account != address(0), "ERC20: burn from the zero address");
    
            balanceOf[account] -= amount;
            _totalSupply -= amount;
            emit Transfer(account, address(0), amount);
        }
    
        /// @dev Sets `value` as allowance of `spender` account over caller account's AnyswapV3ERC20 token.
        /// Emits {Approval} event.
        /// Returns boolean value indicating whether operation succeeded.
        function approve(address spender, uint256 value) external override returns (bool) {
            // _approve(msg.sender, spender, value);
            allowance[msg.sender][spender] = value;
            emit Approval(msg.sender, spender, value);
    
            return true;
        }
    
        /// @dev Sets `value` as allowance of `spender` account over caller account's AnyswapV3ERC20 token,
        /// after which a call is executed to an ERC677-compliant contract with the `data` parameter.
        /// Emits {Approval} event.
        /// Returns boolean value indicating whether operation succeeded.
        /// For more information on approveAndCall format, see https://github.com/ethereum/EIPs/issues/677.
        function approveAndCall(address spender, uint256 value, bytes calldata data) external override returns (bool) {
            // _approve(msg.sender, spender, value);
            allowance[msg.sender][spender] = value;
            emit Approval(msg.sender, spender, value);
    
            return IApprovalReceiver(spender).onTokenApproval(msg.sender, value, data);
        }
    
        /// @dev Sets `value` as allowance of `spender` account over `owner` account's AnyswapV3ERC20 token, given `owner` account's signed approval.
        /// Emits {Approval} event.
        /// Requirements:
        ///   - `deadline` must be timestamp in future.
        ///   - `v`, `r` and `s` must be valid `secp256k1` signature from `owner` account over EIP712-formatted function arguments.
        ///   - the signature must use `owner` account's current nonce (see {nonces}).
        ///   - the signer cannot be zero address and must be `owner` account.
        /// For more information on signature format, see https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP section].
        /// AnyswapV3ERC20 token implementation adapted from https://github.com/albertocuestacanada/ERC20Permit/blob/master/contracts/ERC20Permit.sol.
        function permit(address target, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external override {
            require(block.timestamp <= deadline, "AnyswapV3ERC20: Expired permit");
    
            bytes32 hashStruct = keccak256(
                abi.encode(
                    PERMIT_TYPEHASH,
                    target,
                    spender,
                    value,
                    nonces[target]++,
                    deadline));
    
            require(verifyEIP712(target, hashStruct, v, r, s) || verifyPersonalSign(target, hashStruct, v, r, s));
    
            // _approve(owner, spender, value);
            allowance[target][spender] = value;
            emit Approval(target, spender, value);
        }
    
        function transferWithPermit(address target, address to, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external override returns (bool) {
            require(block.timestamp <= deadline, "AnyswapV3ERC20: Expired permit");
    
            bytes32 hashStruct = keccak256(
                abi.encode(
                    TRANSFER_TYPEHASH,
                    target,
                    to,
                    value,
                    nonces[target]++,
                    deadline));
    
            require(verifyEIP712(target, hashStruct, v, r, s) || verifyPersonalSign(target, hashStruct, v, r, s));
    
            require(to != address(0) || to != address(this));
    
            uint256 balance = balanceOf[target];
            require(balance >= value, "AnyswapV3ERC20: transfer amount exceeds balance");
    
            balanceOf[target] = balance - value;
            balanceOf[to] += value;
            emit Transfer(target, to, value);
    
            return true;
        }
    
        function verifyEIP712(address target, bytes32 hashStruct, uint8 v, bytes32 r, bytes32 s) internal view returns (bool) {
            bytes32 hash = keccak256(
                abi.encodePacked(
                    "\x19\x01",
                    DOMAIN_SEPARATOR,
                    hashStruct));
            address signer = ecrecover(hash, v, r, s);
            return (signer != address(0) && signer == target);
        }
    
        function verifyPersonalSign(address target, bytes32 hashStruct, uint8 v, bytes32 r, bytes32 s) internal view returns (bool) {
            bytes32 hash = prefixed(hashStruct);
            address signer = ecrecover(hash, v, r, s);
            return (signer != address(0) && signer == target);
        }
    
        // Builds a prefixed hash to mimic the behavior of eth_sign.
        function prefixed(bytes32 hash) internal view returns (bytes32) {
            return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", DOMAIN_SEPARATOR, hash));
        }
    
        /// @dev Moves `value` AnyswapV3ERC20 token from caller's account to account (`to`).
        /// A transfer to `address(0)` triggers an ETH withdraw matching the sent AnyswapV3ERC20 token in favor of caller.
        /// Emits {Transfer} event.
        /// Returns boolean value indicating whether operation succeeded.
        /// Requirements:
        ///   - caller account must have at least `value` AnyswapV3ERC20 token.
        function transfer(address to, uint256 value) external override returns (bool) {
            require(to != address(0) || to != address(this));
            uint256 balance = balanceOf[msg.sender];
            require(balance >= value, "AnyswapV3ERC20: transfer amount exceeds balance");
    
            balanceOf[msg.sender] = balance - value;
            balanceOf[to] += value;
            emit Transfer(msg.sender, to, value);
    
            return true;
        }
    
        /// @dev Moves `value` AnyswapV3ERC20 token from account (`from`) to account (`to`) using allowance mechanism.
        /// `value` is then deducted from caller account's allowance, unless set to `type(uint256).max`.
        /// A transfer to `address(0)` triggers an ETH withdraw matching the sent AnyswapV3ERC20 token in favor of caller.
        /// Emits {Approval} event to reflect reduced allowance `value` for caller account to spend from account (`from`),
        /// unless allowance is set to `type(uint256).max`
        /// Emits {Transfer} event.
        /// Returns boolean value indicating whether operation succeeded.
        /// Requirements:
        ///   - `from` account must have at least `value` balance of AnyswapV3ERC20 token.
        ///   - `from` account must have approved caller to spend at least `value` of AnyswapV3ERC20 token, unless `from` and caller are the same account.
        function transferFrom(address from, address to, uint256 value) external override returns (bool) {
            require(to != address(0) || to != address(this));
            if (from != msg.sender) {
                // _decreaseAllowance(from, msg.sender, value);
                uint256 allowed = allowance[from][msg.sender];
                if (allowed != type(uint256).max) {
                    require(allowed >= value, "AnyswapV3ERC20: request exceeds allowance");
                    uint256 reduced = allowed - value;
                    allowance[from][msg.sender] = reduced;
                    emit Approval(from, msg.sender, reduced);
                }
            }
    
            uint256 balance = balanceOf[from];
            require(balance >= value, "AnyswapV3ERC20: transfer amount exceeds balance");
    
            balanceOf[from] = balance - value;
            balanceOf[to] += value;
            emit Transfer(from, to, value);
    
            return true;
        }
    
        /// @dev Moves `value` AnyswapV3ERC20 token from caller's account to account (`to`),
        /// after which a call is executed to an ERC677-compliant contract with the `data` parameter.
        /// A transfer to `address(0)` triggers an ETH withdraw matching the sent AnyswapV3ERC20 token in favor of caller.
        /// Emits {Transfer} event.
        /// Returns boolean value indicating whether operation succeeded.
        /// Requirements:
        ///   - caller account must have at least `value` AnyswapV3ERC20 token.
        /// For more information on transferAndCall format, see https://github.com/ethereum/EIPs/issues/677.
        function transferAndCall(address to, uint value, bytes calldata data) external override returns (bool) {
            require(to != address(0) || to != address(this));
    
            uint256 balance = balanceOf[msg.sender];
            require(balance >= value, "AnyswapV3ERC20: transfer amount exceeds balance");
    
            balanceOf[msg.sender] = balance - value;
            balanceOf[to] += value;
            emit Transfer(msg.sender, to, value);
    
            return ITransferReceiver(to).onTokenTransfer(msg.sender, value, data);
        }
    }

    File 3 of 3: FantomToken
    pragma solidity ^0.4.23;
    
    // ----------------------------------------------------------------------------
    //
    // Fantom Foundation FTM token public sale contract
    //
    // For details, please visit: http://fantom.foundation
    //
    //
    // written by Alex Kampa - ak@sikoba.com
    //
    // ----------------------------------------------------------------------------
    
    
    // ----------------------------------------------------------------------------
    //
    // SafeMath
    //
    // ----------------------------------------------------------------------------
    
    library SafeMath {
    
        function add(uint a, uint b) internal pure returns (uint c) {
            c = a + b;
            require(c >= a);
        }
    
        function sub(uint a, uint b) internal pure returns (uint c) {
            require(b <= a);
            c = a - b;
        }
    
        function mul(uint a, uint b) internal pure returns (uint c) {
            c = a * b;
            require(a == 0 || c / a == b);
        }
    
    }
    
    // ----------------------------------------------------------------------------
    //
    // Owned
    //
    // ----------------------------------------------------------------------------
    
    contract Owned {
    
        address public owner;
        address public newOwner;
    
        mapping(address => bool) public isAdmin;
    
        event OwnershipTransferProposed(address indexed _from, address indexed _to);
        event OwnershipTransferred(address indexed _from, address indexed _to);
        event AdminChange(address indexed _admin, bool _status);
    
        modifier onlyOwner {require(msg.sender == owner); _;}
        modifier onlyAdmin {require(isAdmin[msg.sender]); _;}
    
        constructor() public {
            owner = msg.sender;
            isAdmin[owner] = true;
        }
    
        function transferOwnership(address _newOwner) public onlyOwner {
            require(_newOwner != address(0x0));
            emit OwnershipTransferProposed(owner, _newOwner);
            newOwner = _newOwner;
        }
    
        function acceptOwnership() public {
            require(msg.sender == newOwner);
            emit OwnershipTransferred(owner, newOwner);
            owner = newOwner;
        }
    
        function addAdmin(address _a) public onlyOwner {
            require(isAdmin[_a] == false);
            isAdmin[_a] = true;
            emit AdminChange(_a, true);
        }
    
        function removeAdmin(address _a) public onlyOwner {
            require(isAdmin[_a] == true);
            isAdmin[_a] = false;
            emit AdminChange(_a, false);
        }
    
    }
    
    
    // ----------------------------------------------------------------------------
    //
    // Wallet
    //
    // ----------------------------------------------------------------------------
    
    contract Wallet is Owned {
    
        address public wallet;
    
        event WalletUpdated(address newWallet);
    
        constructor() public {
            wallet = owner;
        }
    
        function setWallet(address _wallet) public onlyOwner {
            require(_wallet != address(0x0));
            wallet = _wallet;
            emit WalletUpdated(_wallet);
        }
    
    }
    
    
    // ----------------------------------------------------------------------------
    //
    // ERC20Interface
    //
    // ----------------------------------------------------------------------------
    
    contract ERC20Interface {
    
        event Transfer(address indexed _from, address indexed _to, uint _value);
        event Approval(address indexed _owner, address indexed _spender, uint _value);
    
        function totalSupply() public view returns (uint);
        function balanceOf(address _owner) public view returns (uint balance);
        function transfer(address _to, uint _value) public returns (bool success);
        function transferFrom(address _from, address _to, uint _value) public returns (bool success);
        function approve(address _spender, uint _value) public returns (bool success);
        function allowance(address _owner, address _spender) public view returns (uint remaining);
    
    }
    
    
    // ----------------------------------------------------------------------------
    //
    // ERC Token Standard #20
    //
    // ----------------------------------------------------------------------------
    
    contract ERC20Token is ERC20Interface, Owned {
    
        using SafeMath for uint;
    
        uint public tokensIssuedTotal;
        mapping(address => uint) balances;
        mapping(address => mapping (address => uint)) allowed;
    
        function totalSupply() public view returns (uint) {
            return tokensIssuedTotal;
        }
        // Includes BOTH locked AND unlocked tokens
    
        function balanceOf(address _owner) public view returns (uint) {
            return balances[_owner];
        }
    
        function transfer(address _to, uint _amount) public returns (bool) {
            require(_to != 0x0);
            balances[msg.sender] = balances[msg.sender].sub(_amount);
            balances[_to] = balances[_to].add(_amount);
            emit Transfer(msg.sender, _to, _amount);
            return true;
        }
    
        function approve(address _spender, uint _amount) public returns (bool) {
            allowed[msg.sender][_spender] = _amount;
            emit Approval(msg.sender, _spender, _amount);
            return true;
        }
    
        function transferFrom(address _from, address _to, uint _amount) public returns (bool) {
            require(_to != 0x0);
            balances[_from] = balances[_from].sub(_amount);
            allowed[_from][msg.sender] = allowed[_from][msg.sender].sub(_amount);
            balances[_to] = balances[_to].add(_amount);
            emit Transfer(_from, _to, _amount);
            return true;
        }
    
        function allowance(address _owner, address _spender) public view returns (uint) {
            return allowed[_owner][_spender];
        }
    
    }
    
    
    // ----------------------------------------------------------------------------
    //
    // LockSlots
    //
    // ----------------------------------------------------------------------------
    
    contract LockSlots is ERC20Token {
    
        using SafeMath for uint;
    
        uint public constant LOCK_SLOTS = 5;
        mapping(address => uint[LOCK_SLOTS]) public lockTerm;
        mapping(address => uint[LOCK_SLOTS]) public lockAmnt;
        mapping(address => bool) public mayHaveLockedTokens;
    
        event RegisteredLockedTokens(address indexed account, uint indexed idx, uint tokens, uint term);
    
        function registerLockedTokens(address _account, uint _tokens, uint _term) internal returns (uint idx) {
            require(_term > now, "lock term must be in the future");
    
            // find a slot (clean up while doing this)
            // use either the existing slot with the exact same term,
            // of which there can be at most one, or the first empty slot
            idx = 9999;
            uint[LOCK_SLOTS] storage term = lockTerm[_account];
            uint[LOCK_SLOTS] storage amnt = lockAmnt[_account];
            for (uint i; i < LOCK_SLOTS; i++) {
                if (term[i] < now) {
                    term[i] = 0;
                    amnt[i] = 0;
                    if (idx == 9999) idx = i;
                }
                if (term[i] == _term) idx = i;
            }
    
            // fail if no slot was found
            require(idx != 9999, "registerLockedTokens: no available slot found");
    
            // register locked tokens
            if (term[idx] == 0) term[idx] = _term;
            amnt[idx] = amnt[idx].add(_tokens);
            mayHaveLockedTokens[_account] = true;
            emit RegisteredLockedTokens(_account, idx, _tokens, _term);
        }
    
        // public view functions
    
        function lockedTokens(address _account) public view returns (uint) {
            if (!mayHaveLockedTokens[_account]) return 0;
            return pNumberOfLockedTokens(_account);
        }
    
        function unlockedTokens(address _account) public view returns (uint) {
            return balances[_account].sub(lockedTokens(_account));
        }
    
        function isAvailableLockSlot(address _account, uint _term) public view returns (bool) {
            if (!mayHaveLockedTokens[_account]) return true;
            if (_term < now) return true;
            uint[LOCK_SLOTS] storage term = lockTerm[_account];
            for (uint i; i < LOCK_SLOTS; i++) {
                if (term[i] < now || term[i] == _term) return true;
            }
            return false;
        }
    
        // internal and private functions
    
        function unlockedTokensInternal(address _account) internal returns (uint) {
            // updates mayHaveLockedTokens if necessary
            if (!mayHaveLockedTokens[_account]) return balances[_account];
            uint locked = pNumberOfLockedTokens(_account);
            if (locked == 0) mayHaveLockedTokens[_account] = false;
            return balances[_account].sub(locked);
        }
    
        function pNumberOfLockedTokens(address _account) private view returns (uint locked) {
            uint[LOCK_SLOTS] storage term = lockTerm[_account];
            uint[LOCK_SLOTS] storage amnt = lockAmnt[_account];
            for (uint i; i < LOCK_SLOTS; i++) {
                if (term[i] >= now) locked = locked.add(amnt[i]);
            }
        }
    
    }
    
    
    // ----------------------------------------------------------------------------
    //
    // FantomIcoDates
    //
    // ----------------------------------------------------------------------------
    
    contract FantomIcoDates is Owned {
    
        uint public dateMainStart = 1529053200; // 15-JUN-2018 09:00 GMT + 0
        uint public dateMainEnd   = 1529658000; // 22-JUN-2018 09:00 GMT + 0
    
        uint public constant DATE_LIMIT = 1529658000 + 180 days;
    
        event IcoDateUpdated(uint id, uint unixts);
    
        // check dates
    
        modifier checkDateOrder {
          _ ;
          require ( dateMainStart < dateMainEnd ) ;
          require ( dateMainEnd < DATE_LIMIT ) ;
        }
    
        constructor() public checkDateOrder() {
            require(now < dateMainStart);
        }
    
        // set ico dates
    
        function setDateMainStart(uint _unixts) public onlyOwner checkDateOrder {
            require(now < _unixts && now < dateMainStart);
            dateMainStart = _unixts;
            emit IcoDateUpdated(1, _unixts);
        }
    
        function setDateMainEnd(uint _unixts) public onlyOwner checkDateOrder {
            require(now < _unixts && now < dateMainEnd);
            dateMainEnd = _unixts;
            emit IcoDateUpdated(2, _unixts);
        }
    
        // where are we? Passed first day or not?
    
        function isMainFirstDay() public view returns (bool) {
            if (now > dateMainStart && now <= dateMainStart + 1 days) return true;
            return false;
        }
    
        function isMain() public view returns (bool) {
            if (now > dateMainStart && now < dateMainEnd) return true;
            return false;
        }
    
    }
    
    // ----------------------------------------------------------------------------
    //
    // Fantom public token sale
    //
    // ----------------------------------------------------------------------------
    
    contract FantomToken is ERC20Token, Wallet, LockSlots, FantomIcoDates {
    
        // Utility variable
    
        uint constant E18 = 10**18;
    
        // Basic token data
    
        string public constant name = "Fantom Token";
        string public constant symbol = "FTM";
        uint8 public constant decimals = 18;
    
        // Token number of possible tokens in existance
    
        uint public constant MAX_TOTAL_TOKEN_SUPPLY = 3175000000 * E18;
    
    
        // crowdsale parameters
        // Opening ETH Rate: USD$463.28
        // Therefore, 1 ETH = 11582 FTM
    
    
        uint public tokensPerEth = 11582;
    
        // USD$2,000,000/463.28 = 4317.043668 ether
        // 4317.043668 ether/2551 addresses = 1.692294656 ether per address for the first 24 hours
    
        uint public constant MINIMUM_CONTRIBUTION = 0.2 ether;
        uint public constant MAXIMUM_FIRST_DAY_CONTRIBUTION = 1.692294656 ether;
    
        uint public constant TOKEN_MAIN_CAP = 50000000 * E18;
    
        bool public tokensTradeable;
    
        // whitelisting
    
        mapping(address => bool) public whitelist;
        uint public numberWhitelisted;
    
        // track main sale
    
        uint public tokensMain;
        mapping(address => uint) public balancesMain;
    
        uint public totalEthContributed;
        mapping(address => uint) public ethContributed;
    
        // tracking tokens minted
    
        uint public tokensMinted;
        mapping(address => uint) public balancesMinted;
        mapping(address => mapping(uint => uint)) public balancesMintedByType;
    
        // migration variable
    
        bool public isMigrationPhaseOpen;
    
        // Events ---------------------------------------------
    
        event UpdatedTokensPerEth(uint tokensPerEth);
        event Whitelisted(address indexed account, uint countWhitelisted);
        event TokensMinted(uint indexed mintType, address indexed account, uint tokens, uint term);
        event RegisterContribution(address indexed account, uint tokensIssued, uint ethContributed, uint ethReturned);
        event TokenExchangeRequested(address indexed account, uint tokens);
    
        // Basic Functions ------------------------------------
    
        constructor() public {}
    
        function () public payable {
            buyTokens();
        }
    
        // Information functions
    
    
        function availableToMint() public view returns (uint) {
            return MAX_TOTAL_TOKEN_SUPPLY.sub(TOKEN_MAIN_CAP).sub(tokensMinted);
        }
    
        function firstDayTokenLimit() public view returns (uint) {
            return ethToTokens(MAXIMUM_FIRST_DAY_CONTRIBUTION);
        }
    
        function ethToTokens(uint _eth) public view returns (uint tokens) {
            tokens = _eth.mul(tokensPerEth);
        }
    
        function tokensToEth(uint _tokens) public view returns (uint eth) {
            eth = _tokens / tokensPerEth;
        }
    
        // Admin functions
    
        function addToWhitelist(address _account) public onlyAdmin {
            pWhitelist(_account);
        }
    
        function addToWhitelistMultiple(address[] _addresses) public onlyAdmin {
            for (uint i; i < _addresses.length; i++) {
                pWhitelist(_addresses[i]);
            }
        }
    
        function pWhitelist(address _account) internal {
            if (whitelist[_account]) return;
            whitelist[_account] = true;
            numberWhitelisted = numberWhitelisted.add(1);
            emit Whitelisted(_account, numberWhitelisted);
        }
    
        // Owner functions ------------------------------------
    
        function updateTokensPerEth(uint _tokens_per_eth) public onlyOwner {
            require(now < dateMainStart);
            tokensPerEth = _tokens_per_eth;
            emit UpdatedTokensPerEth(tokensPerEth);
        }
    
        // Only owner can make tokens tradable at any time, or if the date is
        // greater than the end of the mainsale date plus 20 weeks, allow
        // any caller to make tokensTradeable.
    
        function makeTradeable() public {
            require(msg.sender == owner || now > dateMainEnd + 20 weeks);
            tokensTradeable = true;
        }
    
        function openMigrationPhase() public onlyOwner {
            require(now > dateMainEnd);
            isMigrationPhaseOpen = true;
        }
    
        // Token minting --------------------------------------
    
        function mintTokens(uint _mint_type, address _account, uint _tokens) public onlyOwner {
            pMintTokens(_mint_type, _account, _tokens, 0);
        }
    
        function mintTokensMultiple(uint _mint_type, address[] _accounts, uint[] _tokens) public onlyOwner {
            require(_accounts.length == _tokens.length);
            for (uint i; i < _accounts.length; i++) {
                pMintTokens(_mint_type, _accounts[i], _tokens[i], 0);
            }
        }
    
        function mintTokensLocked(uint _mint_type, address _account, uint _tokens, uint _term) public onlyOwner {
            pMintTokens(_mint_type, _account, _tokens, _term);
        }
    
        function mintTokensLockedMultiple(uint _mint_type, address[] _accounts, uint[] _tokens, uint[] _terms) public onlyOwner {
            require(_accounts.length == _tokens.length);
            require(_accounts.length == _terms.length);
            for (uint i; i < _accounts.length; i++) {
                pMintTokens(_mint_type, _accounts[i], _tokens[i], _terms[i]);
            }
        }
    
        function pMintTokens(uint _mint_type, address _account, uint _tokens, uint _term) private {
            require(whitelist[_account]);
            require(_account != 0x0);
            require(_tokens > 0);
            require(_tokens <= availableToMint(), "not enough tokens available to mint");
            require(_term == 0 || _term > now, "either without lock term, or lock term must be in the future");
    
            // register locked tokens (will throw if no slot is found)
            if (_term > 0) registerLockedTokens(_account, _tokens, _term);
    
            // update
            balances[_account] = balances[_account].add(_tokens);
            balancesMinted[_account] = balancesMinted[_account].add(_tokens);
            balancesMintedByType[_account][_mint_type] = balancesMintedByType[_account][_mint_type].add(_tokens);
            tokensMinted = tokensMinted.add(_tokens);
            tokensIssuedTotal = tokensIssuedTotal.add(_tokens);
    
            // log event
            emit Transfer(0x0, _account, _tokens);
            emit TokensMinted(_mint_type, _account, _tokens, _term);
        }
    
        // Main sale ------------------------------------------
    
        function buyTokens() private {
    
            require(isMain());
            require(msg.value >= MINIMUM_CONTRIBUTION);
            require(whitelist[msg.sender]);
    
            uint tokens_available = TOKEN_MAIN_CAP.sub(tokensMain);
    
            // adjust tokens_available on first day, if necessary
            if (isMainFirstDay()) {
                uint tokens_available_first_day = firstDayTokenLimit().sub(balancesMain[msg.sender]);
                if (tokens_available_first_day < tokens_available) {
                    tokens_available = tokens_available_first_day;
                }
            }
    
            require (tokens_available > 0);
    
            uint tokens_requested = ethToTokens(msg.value);
            uint tokens_issued = tokens_requested;
    
            uint eth_contributed = msg.value;
            uint eth_returned;
    
            if (tokens_requested > tokens_available) {
                tokens_issued = tokens_available;
                eth_returned = tokensToEth(tokens_requested.sub(tokens_available));
                eth_contributed = msg.value.sub(eth_returned);
            }
    
            balances[msg.sender] = balances[msg.sender].add(tokens_issued);
            balancesMain[msg.sender] = balancesMain[msg.sender].add(tokens_issued);
            tokensMain = tokensMain.add(tokens_issued);
            tokensIssuedTotal = tokensIssuedTotal.add(tokens_issued);
    
            ethContributed[msg.sender] = ethContributed[msg.sender].add(eth_contributed);
            totalEthContributed = totalEthContributed.add(eth_contributed);
    
            // ether transfers
            if (eth_returned > 0) msg.sender.transfer(eth_returned);
            wallet.transfer(eth_contributed);
    
            // log
            emit Transfer(0x0, msg.sender, tokens_issued);
            emit RegisterContribution(msg.sender, tokens_issued, eth_contributed, eth_returned);
        }
    
        // Token exchange / migration to new platform ---------
    
        function requestTokenExchangeMax() public {
            requestTokenExchange(unlockedTokensInternal(msg.sender));
        }
    
        function requestTokenExchange(uint _tokens) public {
            require(isMigrationPhaseOpen);
            require(_tokens > 0 && _tokens <= unlockedTokensInternal(msg.sender));
            balances[msg.sender] = balances[msg.sender].sub(_tokens);
            tokensIssuedTotal = tokensIssuedTotal.sub(_tokens);
            emit Transfer(msg.sender, 0x0, _tokens);
            emit TokenExchangeRequested(msg.sender, _tokens);
        }
    
        // ERC20 functions -------------------
    
        /* Transfer out any accidentally sent ERC20 tokens */
    
        function transferAnyERC20Token(address _token_address, uint _amount) public onlyOwner returns (bool success) {
            return ERC20Interface(_token_address).transfer(owner, _amount);
        }
    
        /* Override "transfer" */
    
        function transfer(address _to, uint _amount) public returns (bool success) {
            require(tokensTradeable);
            require(_amount <= unlockedTokensInternal(msg.sender));
            return super.transfer(_to, _amount);
        }
    
        /* Override "transferFrom" */
    
        function transferFrom(address _from, address _to, uint _amount) public returns (bool success) {
            require(tokensTradeable);
            require(_amount <= unlockedTokensInternal(_from));
            return super.transferFrom(_from, _to, _amount);
        }
    
        /* Multiple token transfers from one address to save gas */
    
        function transferMultiple(address[] _addresses, uint[] _amounts) external {
            require(_addresses.length <= 100);
            require(_addresses.length == _amounts.length);
    
            // do the transfers
            for (uint j; j < _addresses.length; j++) {
                transfer(_addresses[j], _amounts[j]);
            }
    
        }
    
    }