ETH Price: $2,521.53 (-0.21%)

Transaction Decoder

Block:
12644109 at Jun-16-2021 07:18:06 AM +UTC
Transaction Fee:
0.002442132 ETH $6.16
Gas Used:
222,012 Gas / 11 Gwei

Emitted Events:

90 Share.Transfer( from=FaaSRewardFund, to=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, value=272550588166652698458 )
91 FaaSPoolLite.RewardPaid( pid=0, account=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, amount=272550588166652698458 )
92 FaaSPoolLite.Transfer( src=[Receiver] FaaSPoolLite, dst=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, amt=34674107316068011492655 )
93 FaaSPoolLite.Withdraw( account=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, amount=34674107316068011492655 )
94 FaaSPoolLite.Transfer( src=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, dst=[Receiver] FaaSPoolLite, amt=34674107316068011492655 )
95 FaaSPoolLite.Transfer( src=[Receiver] FaaSPoolLite, dst=0x0000000000000000000000000000000000000000, amt=34674107316068011492655 )
96 FaaSPoolLite.LOG_EXIT( caller=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, tokenOut=Dollar, tokenAmountOut=1980491712087539053857 )
97 Dollar.Transfer( from=[Receiver] FaaSPoolLite, to=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, value=1980491712087539053857 )
98 FaaSPoolLite.LOG_EXIT( caller=[Sender] 0x81b2ef8e9627e50928f34a84cc28ad607d93de27, tokenOut=FiatTokenProxy, tokenAmountOut=41636233 )
99 FiatTokenProxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x000000000000000000000000cdd2bd61d07b8d42843175dd097a4858a8f764e7, 0x00000000000000000000000081b2ef8e9627e50928f34a84cc28ad607d93de27, 00000000000000000000000000000000000000000000000000000000027b5189 )

Account State Difference:

  Address   Before After State Difference Code
(2Miners: PPLNS)
1,559.003816213931005076 Eth1,559.006258345931005076 Eth0.002442132
0x003e0af2...A9aEd9Fde
0x81b2eF8e...07d93De27
0.01364502 Eth
Nonce: 72
0.011202888 Eth
Nonce: 73
0.002442132
0xA0b86991...E3606eB48
0xCDD2bD61...8A8f764e7
0xE7C9C188...d8ab9c700

Execution Trace

FaaSPoolLite.exitPool( poolAmountIn=34674107316068011492655, minAmountsOut=[0, 0] )
  • FaaSRewardFund.safeTransfer( _token=0xE7C9C188138f7D70945D420d75F8Ca7d8ab9c700, _to=0x81b2eF8e9627e50928f34A84CC28Ad607d93De27, _value=272550588166652698458 )
    • Share.balanceOf( account=0x9E7a4f7e4211c0CE4809cE06B9dDA6b95254BaaC ) => ( 236193817747410257132886 )
    • Share.transfer( recipient=0x81b2eF8e9627e50928f34A84CC28Ad607d93De27, amount=272550588166652698458 ) => ( True )
    • Dollar.transfer( recipient=0x81b2eF8e9627e50928f34A84CC28Ad607d93De27, amount=1980491712087539053857 ) => ( True )
    • FiatTokenProxy.a9059cbb( )
      • FiatTokenV2_1.transfer( to=0x81b2eF8e9627e50928f34A84CC28Ad607d93De27, value=41636233 ) => ( True )
        File 1 of 6: FaaSPoolLite
        // SPDX-License-Identifier: MIT
        
        pragma solidity 0.6.12;
        
        /**
         * @dev Wrappers over Solidity's arithmetic operations with added overflow
         * checks.
         *
         * Arithmetic operations in Solidity wrap on overflow. This can easily result
         * in bugs, because programmers usually assume that an overflow raises an
         * error, which is the standard behavior in high level programming languages.
         * `SafeMath` restores this intuition by reverting the transaction when an
         * operation overflows.
         *
         * Using this library instead of the unchecked operations eliminates an entire
         * class of bugs, so it's recommended to use it always.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `+` operator.
             *
             * Requirements:
             *
             * - Addition cannot overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a, "SafeMath: addition overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                return sub(a, b, "SafeMath: subtraction overflow");
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b <= a, errorMessage);
                uint256 c = a - b;
        
                return c;
            }
        
            /**
             * @dev Returns the multiplication of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `*` operator.
             *
             * Requirements:
             *
             * - Multiplication cannot overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                if (a == 0) {
                    return 0;
                }
        
                uint256 c = a * b;
                require(c / a == b, "SafeMath: multiplication overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return div(a, b, "SafeMath: division by zero");
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                uint256 c = a / b;
                // assert(a == b * c + a % b); // There is no case in which this doesn't hold
        
                return c;
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                return mod(a, b, "SafeMath: modulo by zero");
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts with custom message when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b != 0, errorMessage);
                return a % b;
            }
        }
        
        // This program is free software: you can redistribute it and/or modify
        // it under the terms of the GNU General Public License as published by
        // the Free Software Foundation, either version 3 of the License, or
        // (at your option) any later version.
        // This program is distributed in the hope that it will be useful,
        // but WITHOUT ANY WARRANTY; without even the implied warranty of
        // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        // GNU General Public License for more details.
        // You should have received a copy of the GNU General Public License
        // along with this program.  If not, see <http://www.gnu.org/licenses/>.
        library BConst {
            uint public constant BONE                     = 10**18;
        
            uint public constant MIN_BOUND_TOKENS         = 2;
            uint public constant MAX_BOUND_TOKENS         = 8;
        
            uint public constant DEFAULT_FEE              = BONE * 3 / 1000; // 0.3%
            uint public constant MIN_FEE                  = BONE / 1000; // 0.1%
            uint public constant MAX_FEE                  = BONE / 10; // 10%
        
            uint public constant DEFAULT_COLLECTED_FEE    = BONE * 33 / 10000; // 0.33%
            uint public constant MAX_COLLECTED_FEE        = BONE * 334 / 10000; // 3.34%
        
            uint public constant DEFAULT_EXIT_FEE         = 0;
            uint public constant MAX_EXIT_FEE             = BONE / 1000; // 0.1%
        
            uint public constant MIN_WEIGHT               = BONE;
            uint public constant MAX_WEIGHT               = BONE * 50;
            uint public constant MAX_TOTAL_WEIGHT         = BONE * 50;
            uint public constant MIN_BALANCE              = BONE / 10**12;
        
            uint public constant DEFAULT_INIT_POOL_SUPPLY = BONE * 100;
            uint public constant MIN_INIT_POOL_SUPPLY     = BONE / 1000;
            uint public constant MAX_INIT_POOL_SUPPLY     = BONE * 10**18;
        
            uint public constant MIN_BPOW_BASE            = 1 wei;
            uint public constant MAX_BPOW_BASE            = (2 * BONE) - 1 wei;
            uint public constant BPOW_PRECISION           = BONE / 10**10;
        
            uint public constant MAX_IN_RATIO             = BONE / 2;
            uint public constant MAX_OUT_RATIO            = (BONE / 3) + 1 wei;
        }
        
        // This program is free software: you can redistribute it and/or modify
        // it under the terms of the GNU General Public License as published by
        // the Free Software Foundation, either version 3 of the License, or
        // (at your option) any later version.
        // This program is distributed in the hope that it will be useful,
        // but WITHOUT ANY WARRANTY; without even the implied warranty of
        // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        // GNU General Public License for more details.
        // You should have received a copy of the GNU General Public License
        // along with this program.  If not, see <http://www.gnu.org/licenses/>.
        contract BNum {
        
            function btoi(uint a)
                internal pure 
                returns (uint)
            {
                return a / BConst.BONE;
            }
        
            function bfloor(uint a)
                internal pure
                returns (uint)
            {
                return btoi(a) * BConst.BONE;
            }
        
            function badd(uint a, uint b)
                internal pure
                returns (uint)
            {
                uint c = a + b;
                require(c >= a, "add overflow");
                return c;
            }
        
            function bsub(uint a, uint b)
                internal pure
                returns (uint)
            {
                (uint c, bool flag) = bsubSign(a, b);
                require(!flag, "sub underflow");
                return c;
            }
        
            function bsubSign(uint a, uint b)
                internal pure
                returns (uint, bool)
            {
                if (a >= b) {
                    return (a - b, false);
                } else {
                    return (b - a, true);
                }
            }
        
            function bmul(uint a, uint b)
                internal pure
                returns (uint)
            {
                uint c0 = a * b;
                require(a == 0 || c0 / a == b, "mul overflow");
                uint c1 = c0 + (BConst.BONE / 2);
                require(c1 >= c0, "mul overflow");
                uint c2 = c1 / BConst.BONE;
                return c2;
            }
        
            function bdiv(uint a, uint b)
                internal pure
                returns (uint)
            {
                require(b != 0, "div by 0");
                uint c0 = a * BConst.BONE;
                require(a == 0 || c0 / a == BConst.BONE, "div internal"); // bmul overflow
                uint c1 = c0 + (b / 2);
                require(c1 >= c0, "div internal"); //  badd require
                uint c2 = c1 / b;
                return c2;
            }
        
            // DSMath.wpow
            function bpowi(uint a, uint n)
                internal pure
                returns (uint)
            {
                uint z = n % 2 != 0 ? a : BConst.BONE;
        
                for (n /= 2; n != 0; n /= 2) {
                    a = bmul(a, a);
        
                    if (n % 2 != 0) {
                        z = bmul(z, a);
                    }
                }
                return z;
            }
        
            // Compute b^(e.w) by splitting it into (b^e)*(b^0.w).
            // Use `bpowi` for `b^e` and `bpowK` for k iterations
            // of approximation of b^0.w
            function bpow(uint base, uint exp)
                internal pure
                returns (uint)
            {
                require(base >= BConst.MIN_BPOW_BASE, "base too low");
                require(base <= BConst.MAX_BPOW_BASE, "base too high");
        
                uint whole  = bfloor(exp);   
                uint remain = bsub(exp, whole);
        
                uint wholePow = bpowi(base, btoi(whole));
        
                if (remain == 0) {
                    return wholePow;
                }
        
                uint partialResult = bpowApprox(base, remain, BConst.BPOW_PRECISION);
                return bmul(wholePow, partialResult);
            }
        
            function bpowApprox(uint base, uint exp, uint precision)
                internal pure
                returns (uint)
            {
                // term 0:
                uint a     = exp;
                (uint x, bool xneg)  = bsubSign(base, BConst.BONE);
                uint term = BConst.BONE;
                uint sum   = term;
                bool negative = false;
        
        
                // term(k) = numer / denom 
                //         = (product(a - i - 1, i=1-->k) * x^k) / (k!)
                // each iteration, multiply previous term by (a-(k-1)) * x / k
                // continue until term is less than precision
                for (uint i = 1; term >= precision; i++) {
                    uint bigK = i * BConst.BONE;
                    (uint c, bool cneg) = bsubSign(a, bsub(bigK, BConst.BONE));
                    term = bmul(term, bmul(c, x));
                    term = bdiv(term, bigK);
                    if (term == 0) break;
        
                    if (xneg) negative = !negative;
                    if (cneg) negative = !negative;
                    if (negative) {
                        sum = bsub(sum, term);
                    } else {
                        sum = badd(sum, term);
                    }
                }
        
                return sum;
            }
        
        }
        
        // This program is free software: you can redistribute it and/or modify
        // it under the terms of the GNU General Public License as published by
        // the Free Software Foundation, either version 3 of the License, or
        // (at your option) any later version.
        // This program is distributed in the hope that it will be useful,
        // but WITHOUT ANY WARRANTY; without even the implied warranty of
        // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        // GNU General Public License for more details.
        // You should have received a copy of the GNU General Public License
        // along with this program.  If not, see <http://www.gnu.org/licenses/>.
        // Highly opinionated token implementation
        interface IERC20 {
            event Approval(address indexed src, address indexed dst, uint amt);
            event Transfer(address indexed src, address indexed dst, uint amt);
        
            function totalSupply() external view returns (uint);
            function balanceOf(address whom) external view returns (uint);
            function allowance(address src, address dst) external view returns (uint);
        
            function approve(address dst, uint amt) external returns (bool);
            function transfer(address dst, uint amt) external returns (bool);
            function transferFrom(
                address src, address dst, uint amt
            ) external returns (bool);
        }
        
        contract BTokenBase is BNum {
        
            mapping(address => uint)                   internal _balance;
            mapping(address => mapping(address=>uint)) internal _allowance;
            uint internal _totalSupply;
        
            event Approval(address indexed src, address indexed dst, uint amt);
            event Transfer(address indexed src, address indexed dst, uint amt);
        
            function _mint(uint amt) internal {
                _balance[address(this)] = badd(_balance[address(this)], amt);
                _totalSupply = badd(_totalSupply, amt);
                emit Transfer(address(0), address(this), amt);
            }
        
            function _burn(uint amt) internal {
                require(_balance[address(this)] >= amt, "!bal");
                _balance[address(this)] = bsub(_balance[address(this)], amt);
                _totalSupply = bsub(_totalSupply, amt);
                emit Transfer(address(this), address(0), amt);
            }
        
            function _move(address src, address dst, uint amt) internal {
                require(_balance[src] >= amt, "!bal");
                _balance[src] = bsub(_balance[src], amt);
                _balance[dst] = badd(_balance[dst], amt);
                emit Transfer(src, dst, amt);
            }
        
            function _push(address to, uint amt) internal {
                _move(address(this), to, amt);
            }
        
            function _pull(address from, uint amt) internal {
                _move(from, address(this), amt);
            }
        }
        
        contract BToken is BTokenBase, IERC20 {
            string  private _name     = "Value Liquidity Provider";
            string  private _symbol   = "VLP";
            uint8   private _decimals = 18;
        
            function name() public view returns (string memory) {
                return _name;
            }
        
            function symbol() public view returns (string memory) {
                return _symbol;
            }
        
            function decimals() public view returns(uint8) {
                return _decimals;
            }
        
            function allowance(address src, address dst) external override view returns (uint) {
                return _allowance[src][dst];
            }
        
            function balanceOf(address whom) public override view returns (uint) {
                return _balance[whom];
            }
        
            function totalSupply() public override view returns (uint) {
                return _totalSupply;
            }
        
            function approve(address dst, uint amt) external override returns (bool) {
                _allowance[msg.sender][dst] = amt;
                emit Approval(msg.sender, dst, amt);
                return true;
            }
        
            function increaseApproval(address dst, uint amt) external returns (bool) {
                _allowance[msg.sender][dst] = badd(_allowance[msg.sender][dst], amt);
                emit Approval(msg.sender, dst, _allowance[msg.sender][dst]);
                return true;
            }
        
            function decreaseApproval(address dst, uint amt) external returns (bool) {
                uint oldValue = _allowance[msg.sender][dst];
                if (amt > oldValue) {
                    _allowance[msg.sender][dst] = 0;
                } else {
                    _allowance[msg.sender][dst] = bsub(oldValue, amt);
                }
                emit Approval(msg.sender, dst, _allowance[msg.sender][dst]);
                return true;
            }
        
            function transfer(address dst, uint amt) external override returns (bool) {
                _move(msg.sender, dst, amt);
                return true;
            }
        
            function transferFrom(address src, address dst, uint amt) external override returns (bool) {
                require(msg.sender == src || amt <= _allowance[src][msg.sender], "!spender");
                _move(src, dst, amt);
                if (msg.sender != src && _allowance[src][msg.sender] != uint256(-1)) {
                    _allowance[src][msg.sender] = bsub(_allowance[src][msg.sender], amt);
                    emit Approval(msg.sender, dst, _allowance[src][msg.sender]);
                }
                return true;
            }
        }
        
        // This program is free software: you can redistribute it and/or modify
        // it under the terms of the GNU General Public License as published by
        // the Free Software Foundation, either version 3 of the License, or
        // (at your option) any later version.
        // This program is distributed in the hope that it will be useful,
        // but WITHOUT ANY WARRANTY; without even the implied warranty of
        // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        // GNU General Public License for more details.
        // You should have received a copy of the GNU General Public License
        // along with this program.  If not, see <http://www.gnu.org/licenses/>.
        contract BMathLite is BNum {
            /**********************************************************************************************
            // calcSpotPrice                                                                             //
            // sP = spotPrice                                                                            //
            // bI = tokenBalanceIn                ( bI / wI )         1                                  //
            // bO = tokenBalanceOut         sP =  -----------  *  ----------                             //
            // wI = tokenWeightIn                 ( bO / wO )     ( 1 - sF )                             //
            // wO = tokenWeightOut                                                                       //
            // sF = swapFee (+ collectedFee)                                                             //
            **********************************************************************************************/
            function calcSpotPrice(
                uint tokenBalanceIn,
                uint tokenWeightIn,
                uint tokenBalanceOut,
                uint tokenWeightOut,
                uint swapFee
            )
                public pure
                returns (uint spotPrice)
            {
                uint numer = bdiv(tokenBalanceIn, tokenWeightIn);
                uint denom = bdiv(tokenBalanceOut, tokenWeightOut);
                uint ratio = bdiv(numer, denom);
                uint scale = bdiv(BConst.BONE, bsub(BConst.BONE, swapFee));
                return  (spotPrice = bmul(ratio, scale));
            }
        
            /**********************************************************************************************
            // calcOutGivenIn                                                                            //
            // aO = tokenAmountOut                                                                       //
            // bO = tokenBalanceOut                                                                      //
            // bI = tokenBalanceIn              /      /            bI             \    (wI / wO) \      //
            // aI = tokenAmountIn    aO = bO * |  1 - | --------------------------  | ^            |     //
            // wI = tokenWeightIn               \      \ ( bI + ( aI * ( 1 - sF )) /              /      //
            // wO = tokenWeightOut                                                                       //
            // sF = swapFee (+ collectedFee)                                                             //
            **********************************************************************************************/
            function calcOutGivenIn(
                uint tokenBalanceIn,
                uint tokenWeightIn,
                uint tokenBalanceOut,
                uint tokenWeightOut,
                uint tokenAmountIn,
                uint swapFee
            )
                public pure
                returns (uint tokenAmountOut)
            {
                uint weightRatio = bdiv(tokenWeightIn, tokenWeightOut);
                uint adjustedIn = bsub(BConst.BONE, swapFee);
                adjustedIn = bmul(tokenAmountIn, adjustedIn);
                uint y = bdiv(tokenBalanceIn, badd(tokenBalanceIn, adjustedIn));
                uint foo = bpow(y, weightRatio);
                uint bar = bsub(BConst.BONE, foo);
                tokenAmountOut = bmul(tokenBalanceOut, bar);
                return tokenAmountOut;
            }
        
            /**********************************************************************************************
            // calcInGivenOut                                                                            //
            // aI = tokenAmountIn                                                                        //
            // bO = tokenBalanceOut               /  /     bO      \    (wO / wI)      \                 //
            // bI = tokenBalanceIn          bI * |  | ------------  | ^            - 1  |                //
            // aO = tokenAmountOut    aI =        \  \ ( bO - aO ) /                   /                 //
            // wI = tokenWeightIn           --------------------------------------------                 //
            // wO = tokenWeightOut                          ( 1 - sF )                                   //
            // sF = swapFee (+ collectedFee)                                                             //
            **********************************************************************************************/
            function calcInGivenOut(
                uint tokenBalanceIn,
                uint tokenWeightIn,
                uint tokenBalanceOut,
                uint tokenWeightOut,
                uint tokenAmountOut,
                uint swapFee
            )
                public pure
                returns (uint tokenAmountIn)
            {
                uint weightRatio = bdiv(tokenWeightOut, tokenWeightIn);
                uint diff = bsub(tokenBalanceOut, tokenAmountOut);
                uint y = bdiv(tokenBalanceOut, diff);
                uint foo = bpow(y, weightRatio);
                foo = bsub(foo, BConst.BONE);
                tokenAmountIn = bsub(BConst.BONE, swapFee);
                tokenAmountIn = bdiv(bmul(tokenBalanceIn, foo), tokenAmountIn);
                return tokenAmountIn;
            }
        
            /**********************************************************************************************
            // calcPoolOutGivenSingleIn                                                                  //
            // pAo = poolAmountOut         /                                              \              //
            // tAi = tokenAmountIn        ///      /     //    wI \      \\       \     wI \             //
            // wI = tokenWeightIn        //| tAi *| 1 - || 1 - --  | * sF || + tBi \    --  \            //
            // tW = totalWeight     pAo=||  \      \     \\    tW /      //         | ^ tW   | * pS - pS //
            // tBi = tokenBalanceIn      \\  ------------------------------------- /        /            //
            // pS = poolSupply            \\                    tBi               /        /             //
            // sF = swapFee (+ collectedFee)\                                              /              //
            **********************************************************************************************/
            function calcPoolOutGivenSingleIn(
                uint tokenBalanceIn,
                uint tokenWeightIn,
                uint poolSupply,
                uint totalWeight,
                uint tokenAmountIn,
                uint swapFee
            )
                public pure
                returns (uint poolAmountOut)
            {
                // @dev Charge the trading fee for the proportion of tokenAi
                // which is implicitly traded to the other pool tokens.
                // That proportion is (1- weightTokenIn)
                // tokenAiAfterFee = tAi * (1 - (1-weightTi) * poolFee);
                uint normalizedWeight = bdiv(tokenWeightIn, totalWeight);
                uint zaz = bmul(bsub(BConst.BONE, normalizedWeight), swapFee);
                uint tokenAmountInAfterFee = bmul(tokenAmountIn, bsub(BConst.BONE, zaz));
        
                uint newTokenBalanceIn = badd(tokenBalanceIn, tokenAmountInAfterFee);
                uint tokenInRatio = bdiv(newTokenBalanceIn, tokenBalanceIn);
        
                // uint newPoolSupply = (ratioTi ^ weightTi) * poolSupply;
                uint poolRatio = bpow(tokenInRatio, normalizedWeight);
                uint newPoolSupply = bmul(poolRatio, poolSupply);
                poolAmountOut = bsub(newPoolSupply, poolSupply);
                return poolAmountOut;
            }
        
            /**********************************************************************************************
            // calcSingleOutGivenPoolIn                                                                  //
            // tAo = tokenAmountOut            /      /                                             \\   //
            // bO = tokenBalanceOut           /      // pS - (pAi * (1 - eF)) \     /    1    \      \\  //
            // pAi = poolAmountIn            | bO - || ----------------------- | ^ | --------- | * b0 || //
            // ps = poolSupply                \      \\          pS           /     \(wO / tW)/      //  //
            // wI = tokenWeightIn      tAo =   \      \                                             //   //
            // tW = totalWeight                    /     /      wO \       \                             //
            // sF = swapFee (+ collectedFee)   *  | 1 - |  1 - ---- | * sF  |                            //
            // eF = exitFee                        \     \      tW /       /                             //
            **********************************************************************************************/
            function calcSingleOutGivenPoolIn(
                uint tokenBalanceOut,
                uint tokenWeightOut,
                uint poolSupply,
                uint totalWeight,
                uint poolAmountIn,
                uint swapFee,
                uint exitFee
            )
                public pure
                returns (uint tokenAmountOut)
            {
                uint normalizedWeight = bdiv(tokenWeightOut, totalWeight);
                // charge exit fee on the pool token side
                // pAiAfterExitFee = pAi*(1-exitFee)
                uint poolAmountInAfterExitFee = bmul(poolAmountIn, bsub(BConst.BONE, exitFee));
                uint newPoolSupply = bsub(poolSupply, poolAmountInAfterExitFee);
                uint poolRatio = bdiv(newPoolSupply, poolSupply);
             
                // newBalTo = poolRatio^(1/weightTo) * balTo;
                uint tokenOutRatio = bpow(poolRatio, bdiv(BConst.BONE, normalizedWeight));
                uint newTokenBalanceOut = bmul(tokenOutRatio, tokenBalanceOut);
        
                uint tokenAmountOutBeforeSwapFee = bsub(tokenBalanceOut, newTokenBalanceOut);
        
                // charge swap fee on the output token side 
                //uint tAo = tAoBeforeSwapFee * (1 - (1-weightTo) * swapFee)
                uint zaz = bmul(bsub(BConst.BONE, normalizedWeight), swapFee);
                tokenAmountOut = bmul(tokenAmountOutBeforeSwapFee, bsub(BConst.BONE, zaz));
                return tokenAmountOut;
            }
        
        
        }
        
        interface IBFactory {
            function collectedToken() external view returns (address);
        }
        
        contract BPoolLite is BToken, BMathLite {
            struct Record {
                bool bound;   // is token bound to pool
                uint index;   // private
                uint denorm;  // denormalized weight
                uint balance;
            }
        
            event LOG_SWAP(
                address indexed caller,
                address indexed tokenIn,
                address indexed tokenOut,
                uint256 tokenAmountIn,
                uint256 tokenAmountOut
            );
        
            event LOG_JOIN(
                address indexed caller,
                address indexed tokenIn,
                uint256 tokenAmountIn
            );
        
            event LOG_EXIT(
                address indexed caller,
                address indexed tokenOut,
                uint256 tokenAmountOut
            );
        
            event LOG_COLLECTED_FUND(
                address indexed collectedToken,
                uint256 collectedAmount
            );
        
            event LOG_FINALIZE(
                uint swapFee,
                uint initPoolSupply,
                uint version,
                address[] bindTokens,
                uint[] bindDenorms,
                uint[] balances
            );
        
            modifier _lock_() {
                require(!_mutex, "reentry");
                _mutex = true;
                _;
                _mutex = false;
            }
        
            modifier _viewlock_() {
                require(!_mutex, "reentry");
                _;
            }
        
            bool private _mutex;
        
            uint public version = 2001;
            address public factory;    // BFactory address to push token exitFee to
            address public controller; // has CONTROL role
        
            // `setSwapFee` and `finalize` require CONTROL
            // `finalize` sets `PUBLIC can SWAP`, `PUBLIC can JOIN`
            uint public swapFee;
            uint public collectedFee; // 0.05% | https://yfv.finance/vip-vote/vip_5
            uint public exitFee;
            bool public finalized;
        
            address[] internal _tokens;
            mapping(address => Record) internal _records;
            uint private _totalWeight;
        
            constructor(address _factory) public {
                controller = _factory;
                factory = _factory;
                swapFee = BConst.DEFAULT_FEE;
                collectedFee = BConst.DEFAULT_COLLECTED_FEE;
                exitFee = BConst.DEFAULT_EXIT_FEE;
                finalized = false;
            }
        
            function setCollectedFee(uint _collectedFee) public {
                require(msg.sender == factory, "!fctr");
                require(_collectedFee <= BConst.MAX_COLLECTED_FEE, ">maxCoFee");
                require(bmul(_collectedFee, 2) <= swapFee, ">sFee/2");
                collectedFee = _collectedFee;
            }
        
            function setExitFee(uint _exitFee) public {
                require(!finalized, "fnl");
                require(msg.sender == factory, "!fctr");
                require(_exitFee <= BConst.MAX_EXIT_FEE, ">maxExitFee");
                exitFee = _exitFee;
            }
        
            function isBound(address t)
            external view
            returns (bool)
            {
                return _records[t].bound;
            }
        
            function getNumTokens()
            external view
            returns (uint)
            {
                return _tokens.length;
            }
        
            function getCurrentTokens()
            external view _viewlock_
            returns (address[] memory tokens)
            {
                return _tokens;
            }
        
            function getFinalTokens()
            external view
            _viewlock_
            returns (address[] memory tokens)
            {
                require(finalized, "!fnl");
                return _tokens;
            }
        
            function getDenormalizedWeight(address token)
            external view
            _viewlock_
            returns (uint)
            {
        
                require(_records[token].bound, "!bound");
                return _records[token].denorm;
            }
        
            function getTotalDenormalizedWeight()
            external view
            _viewlock_
            returns (uint)
            {
                return _totalWeight;
            }
        
            function getNormalizedWeight(address token)
            external view
            _viewlock_
            returns (uint)
            {
        
                require(_records[token].bound, "!bound");
                uint denorm = _records[token].denorm;
                return bdiv(denorm, _totalWeight);
            }
        
            function getBalance(address token)
            external view
            _viewlock_
            returns (uint)
            {
        
                require(_records[token].bound, "!bound");
                return _records[token].balance;
            }
        
            function setController(address _controller)
            external
            _lock_
            {
                require(msg.sender == controller, "!cntler");
                controller = _controller;
            }
        
            function finalize(
                uint _swapFee,
                uint _initPoolSupply,
                address[] calldata _bindTokens,
                uint[] calldata _bindDenorms
            ) external _lock_ {
                require(msg.sender == controller, "!cntler");
                require(!finalized, "fnl");
        
                require(_swapFee >= BConst.MIN_FEE, "<minFee");
                require(_swapFee <= BConst.MAX_FEE, ">maxFee");
                swapFee = _swapFee;
                collectedFee = _swapFee / 3;
        
                require(_initPoolSupply >= BConst.MIN_INIT_POOL_SUPPLY, "<minInitPSup");
                require(_initPoolSupply <= BConst.MAX_INIT_POOL_SUPPLY, ">maxInitPSup");
        
                require(_bindTokens.length >= BConst.MIN_BOUND_TOKENS, "<minTokens");
                require(_bindTokens.length < BConst.MAX_BOUND_TOKENS, ">maxTokens");
                require(_bindTokens.length == _bindDenorms.length, "erLengMism");
        
                uint totalWeight = 0;
                uint256[] memory balances = new uint[](_bindTokens.length);
                for (uint i = 0; i < _bindTokens.length; i++) {
                    address token = _bindTokens[i];
                    uint denorm = _bindDenorms[i];
                    uint balance = BToken(token).balanceOf(address(this));
                    balances[i] = balance;
                    require(!_records[token].bound, "bound");
                    require(denorm >= BConst.MIN_WEIGHT, "<minWeight");
                    require(denorm <= BConst.MAX_WEIGHT, ">maxWeight");
                    require(balance >= BConst.MIN_BALANCE, "<minBal");
                    _records[token] = Record({
                        bound : true,
                        index : i,
                        denorm : denorm,
                        balance : balance
                        });
                    totalWeight = badd(totalWeight, denorm);
                }
                require(totalWeight <= BConst.MAX_TOTAL_WEIGHT, ">maxTWeight");
                _totalWeight = totalWeight;
                _tokens = _bindTokens;
                finalized = true;
                _mintPoolShare(_initPoolSupply);
                _pushPoolShare(msg.sender, _initPoolSupply);
                emit LOG_FINALIZE(swapFee, _initPoolSupply, version, _bindTokens, _bindDenorms, balances);
            }
        
            // Absorb any tokens that have been sent to this contract into the pool
            function gulp(address token)
            external
            _lock_
            {
                require(_records[token].bound, "!bound");
                _records[token].balance = IERC20(token).balanceOf(address(this));
            }
        
            function getSpotPrice(address tokenIn, address tokenOut)
            external view
            _viewlock_
            returns (uint spotPrice)
            {
                require(_records[tokenIn].bound, "!bound");
                require(_records[tokenOut].bound, "!bound");
                Record storage inRecord = _records[tokenIn];
                Record storage outRecord = _records[tokenOut];
                return calcSpotPrice(inRecord.balance, inRecord.denorm, outRecord.balance, outRecord.denorm, swapFee);
            }
        
            function getSpotPriceSansFee(address tokenIn, address tokenOut)
            external view
            _viewlock_
            returns (uint spotPrice)
            {
                require(_records[tokenIn].bound, "!bound");
                require(_records[tokenOut].bound, "!bound");
                Record storage inRecord = _records[tokenIn];
                Record storage outRecord = _records[tokenOut];
                return calcSpotPrice(inRecord.balance, inRecord.denorm, outRecord.balance, outRecord.denorm, 0);
            }
        
            function joinPool(uint poolAmountOut, uint[] calldata maxAmountsIn)
            external virtual
            _lock_
            {
                require(finalized, "!fnl");
        
                uint poolTotal = totalSupply();
                uint ratio = bdiv(poolAmountOut, poolTotal);
                require(ratio != 0, "erMApr");
        
                for (uint i = 0; i < _tokens.length; i++) {
                    address t = _tokens[i];
                    uint bal = _records[t].balance;
                    uint tokenAmountIn = bmul(ratio, bal);
                    require(tokenAmountIn != 0, "erMApr");
                    require(tokenAmountIn <= maxAmountsIn[i], "<limIn");
                    _records[t].balance = badd(_records[t].balance, tokenAmountIn);
                    emit LOG_JOIN(msg.sender, t, tokenAmountIn);
                    _pullUnderlying(t, msg.sender, tokenAmountIn);
                }
                _mintPoolShare(poolAmountOut);
                _pushPoolShare(msg.sender, poolAmountOut);
            }
        
            function exitPool(uint poolAmountIn, uint[] calldata minAmountsOut)
            external virtual
            _lock_
            {
                require(finalized, "!fnl");
        
                uint poolTotal = totalSupply();
                uint _exitFee = bmul(poolAmountIn, exitFee);
                uint pAiAfterExitFee = bsub(poolAmountIn, _exitFee);
                uint ratio = bdiv(pAiAfterExitFee, poolTotal);
                require(ratio != 0, "erMApr");
        
                _pullPoolShare(msg.sender, poolAmountIn);
                if (_exitFee > 0) {
                    _pushPoolShare(factory, _exitFee);
                }
                _burnPoolShare(pAiAfterExitFee);
        
                for (uint i = 0; i < _tokens.length; i++) {
                    address t = _tokens[i];
                    uint bal = _records[t].balance;
                    uint tokenAmountOut = bmul(ratio, bal);
                    require(tokenAmountOut != 0, "erMApr");
                    require(tokenAmountOut >= minAmountsOut[i], "<limO");
                    _records[t].balance = bsub(_records[t].balance, tokenAmountOut);
                    emit LOG_EXIT(msg.sender, t, tokenAmountOut);
                    _pushUnderlying(t, msg.sender, tokenAmountOut);
                }
            }
        
            function swapExactAmountIn(
                address tokenIn,
                uint tokenAmountIn,
                address tokenOut,
                uint minAmountOut,
                uint maxPrice
            )
            external
            _lock_
            returns (uint tokenAmountOut, uint spotPriceAfter)
            {
        
                require(_records[tokenIn].bound, "!bound");
                require(_records[tokenOut].bound, "!bound");
        
                Record storage inRecord = _records[address(tokenIn)];
                Record storage outRecord = _records[address(tokenOut)];
        
                require(tokenAmountIn <= bmul(inRecord.balance, BConst.MAX_IN_RATIO), ">maxIRat");
        
                uint spotPriceBefore = calcSpotPrice(
                    inRecord.balance,
                    inRecord.denorm,
                    outRecord.balance,
                    outRecord.denorm,
                    swapFee
                );
                require(spotPriceBefore <= maxPrice, "badLimPrice");
        
                tokenAmountOut = calcOutGivenIn(
                    inRecord.balance,
                    inRecord.denorm,
                    outRecord.balance,
                    outRecord.denorm,
                    tokenAmountIn,
                    swapFee
                );
                require(tokenAmountOut >= minAmountOut, "<limO");
        
                inRecord.balance = badd(inRecord.balance, tokenAmountIn);
                outRecord.balance = bsub(outRecord.balance, tokenAmountOut);
        
                spotPriceAfter = calcSpotPrice(
                    inRecord.balance,
                    inRecord.denorm,
                    outRecord.balance,
                    outRecord.denorm,
                    swapFee
                );
                require(spotPriceAfter >= spotPriceBefore, "erMApr");
                require(spotPriceAfter <= maxPrice, ">limPrice");
                require(spotPriceBefore <= bdiv(tokenAmountIn, tokenAmountOut), "erMApr");
        
                emit LOG_SWAP(msg.sender, tokenIn, tokenOut, tokenAmountIn, tokenAmountOut);
        
                _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);
                uint _subTokenAmountIn;
                (_subTokenAmountIn, tokenAmountOut) = _pushCollectedFundGivenOut(tokenIn, tokenAmountIn, tokenOut, tokenAmountOut);
                if (_subTokenAmountIn > 0) inRecord.balance = bsub(inRecord.balance, _subTokenAmountIn);
                _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);
        
                return (tokenAmountOut, spotPriceAfter);
            }
        
            function swapExactAmountOut(
                address tokenIn,
                uint maxAmountIn,
                address tokenOut,
                uint tokenAmountOut,
                uint maxPrice
            )
            external
            _lock_
            returns (uint tokenAmountIn, uint spotPriceAfter)
            {
                require(_records[tokenIn].bound, "!bound");
                require(_records[tokenOut].bound, "!bound");
        
                Record storage inRecord = _records[address(tokenIn)];
                Record storage outRecord = _records[address(tokenOut)];
        
                require(tokenAmountOut <= bmul(outRecord.balance, BConst.MAX_OUT_RATIO), ">maxORat");
        
                uint spotPriceBefore = calcSpotPrice(
                    inRecord.balance,
                    inRecord.denorm,
                    outRecord.balance,
                    outRecord.denorm,
                    swapFee
                );
                require(spotPriceBefore <= maxPrice, "badLimPrice");
        
                tokenAmountIn = calcInGivenOut(
                    inRecord.balance,
                    inRecord.denorm,
                    outRecord.balance,
                    outRecord.denorm,
                    tokenAmountOut,
                    swapFee
                );
                require(tokenAmountIn <= maxAmountIn, "<limIn");
        
                inRecord.balance = badd(inRecord.balance, tokenAmountIn);
                outRecord.balance = bsub(outRecord.balance, tokenAmountOut);
        
                spotPriceAfter = calcSpotPrice(
                    inRecord.balance,
                    inRecord.denorm,
                    outRecord.balance,
                    outRecord.denorm,
                    swapFee
                );
                require(spotPriceAfter >= spotPriceBefore, "erMApr");
                require(spotPriceAfter <= maxPrice, ">limPrice");
                require(spotPriceBefore <= bdiv(tokenAmountIn, tokenAmountOut), "erMApr");
        
                emit LOG_SWAP(msg.sender, tokenIn, tokenOut, tokenAmountIn, tokenAmountOut);
        
                _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);
                _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);
                uint _collectedFeeAmount = _pushCollectedFundGivenIn(tokenIn, tokenAmountIn);
                if (_collectedFeeAmount > 0) inRecord.balance = bsub(inRecord.balance, _collectedFeeAmount);
        
                return (tokenAmountIn, spotPriceAfter);
            }
        
            function joinswapExternAmountIn(address tokenIn, uint tokenAmountIn, uint minPoolAmountOut)
            external
            _lock_
            returns (uint poolAmountOut)
        
            {
                require(finalized, "!fnl");
                require(_records[tokenIn].bound, "!bound");
                require(tokenAmountIn <= bmul(_records[tokenIn].balance, BConst.MAX_IN_RATIO), ">maxIRat");
        
                Record storage inRecord = _records[tokenIn];
        
                poolAmountOut = calcPoolOutGivenSingleIn(
                    inRecord.balance,
                    inRecord.denorm,
                    _totalSupply,
                    _totalWeight,
                    tokenAmountIn,
                    swapFee
                );
        
                require(poolAmountOut >= minPoolAmountOut, "<limO");
        
                inRecord.balance = badd(inRecord.balance, tokenAmountIn);
        
                emit LOG_JOIN(msg.sender, tokenIn, tokenAmountIn);
        
                _mintPoolShare(poolAmountOut);
                _pullUnderlying(tokenIn, msg.sender, tokenAmountIn);
                uint _subTokenAmountIn;
                (_subTokenAmountIn, poolAmountOut) = _pushCollectedFundGivenOut(tokenIn, tokenAmountIn, address(this), poolAmountOut);
                if (_subTokenAmountIn > 0) inRecord.balance = bsub(inRecord.balance, _subTokenAmountIn);
                _pushPoolShare(msg.sender, poolAmountOut);
        
                return poolAmountOut;
            }
        
            function exitswapPoolAmountIn(address tokenOut, uint poolAmountIn, uint minAmountOut)
            external
            _lock_
            returns (uint tokenAmountOut)
            {
                require(finalized, "!fnl");
                require(_records[tokenOut].bound, "!bound");
        
                Record storage outRecord = _records[tokenOut];
        
                tokenAmountOut = calcSingleOutGivenPoolIn(
                    outRecord.balance,
                    outRecord.denorm,
                    _totalSupply,
                    _totalWeight,
                    poolAmountIn,
                    swapFee,
                    exitFee
                );
        
                require(tokenAmountOut >= minAmountOut, "<limO");
        
                require(tokenAmountOut <= bmul(_records[tokenOut].balance, BConst.MAX_OUT_RATIO), ">maxORat");
        
                outRecord.balance = bsub(outRecord.balance, tokenAmountOut);
        
                uint _exitFee = bmul(poolAmountIn, exitFee);
        
                emit LOG_EXIT(msg.sender, tokenOut, tokenAmountOut);
        
                _pullPoolShare(msg.sender, poolAmountIn);
                _burnPoolShare(bsub(poolAmountIn, _exitFee));
                if (_exitFee > 0) {
                    _pushPoolShare(factory, _exitFee);
                }
                (, tokenAmountOut) = _pushCollectedFundGivenOut(address(this), poolAmountIn, tokenOut, tokenAmountOut);
                _pushUnderlying(tokenOut, msg.sender, tokenAmountOut);
        
                return tokenAmountOut;
            }
        
            // ==
            // 'Underlying' token-manipulation functions make external calls but are NOT locked
            // You must `_lock_` or otherwise ensure reentry-safety
            //
            // Fixed ERC-20 transfer revert for some special token such as USDT
            function _pullUnderlying(address erc20, address from, uint amount) internal {
                // bytes4(keccak256(bytes('transferFrom(address,address,uint256)')));
                (bool success, bytes memory data) = erc20.call(abi.encodeWithSelector(0x23b872dd, from, address(this), amount));
                require(success && (data.length == 0 || abi.decode(data, (bool))), '!_pullU');
            }
        
            function _pushUnderlying(address erc20, address to, uint amount) internal
            {
                // bytes4(keccak256(bytes('transfer(address,uint256)')));
                (bool success, bytes memory data) = erc20.call(abi.encodeWithSelector(0xa9059cbb, to, amount));
                require(success && (data.length == 0 || abi.decode(data, (bool))), '!_pushU');
            }
        
            function _pullPoolShare(address from, uint amount)
            internal
            {
                _pull(from, amount);
            }
        
            function _pushPoolShare(address to, uint amount)
            internal
            {
                _push(to, amount);
            }
        
            function _mintPoolShare(uint amount)
            internal
            {
                _mint(amount);
            }
        
            function _burnPoolShare(uint amount)
            internal
            {
                _burn(amount);
            }
        
            function _pushCollectedFundGivenOut(address _tokenIn, uint _tokenAmountIn, address _tokenOut, uint _tokenAmountOut) internal returns (uint subTokenAmountIn, uint tokenAmountOut) {
                subTokenAmountIn = 0;
                tokenAmountOut = _tokenAmountOut;
                if (collectedFee > 0) {
                    address _collectedToken = IBFactory(factory).collectedToken();
                    if (_collectedToken == _tokenIn) {
                        subTokenAmountIn = bdiv(bmul(_tokenAmountIn, collectedFee), BConst.BONE);
                        _pushUnderlying(_tokenIn, factory, subTokenAmountIn);
                        emit LOG_COLLECTED_FUND(_tokenIn, subTokenAmountIn);
                    } else {
                        uint _collectedFeeAmount = bdiv(bmul(_tokenAmountOut, collectedFee), BConst.BONE);
                        _pushUnderlying(_tokenOut, factory, _collectedFeeAmount);
                        tokenAmountOut = bsub(_tokenAmountOut, _collectedFeeAmount);
                        emit LOG_COLLECTED_FUND(_tokenOut, _collectedFeeAmount);
                    }
                }
            }
        
            // always push out _tokenIn (already have)
            function _pushCollectedFundGivenIn(address _tokenIn, uint _tokenAmountIn) internal returns (uint collectedFeeAmount) {
                collectedFeeAmount = 0;
                if (collectedFee > 0) {
                    address _collectedToken = IBFactory(factory).collectedToken();
                    if (_collectedToken != address(0)) {
                        collectedFeeAmount = bdiv(bmul(_tokenAmountIn, collectedFee), BConst.BONE);
                        _pushUnderlying(_tokenIn, factory, collectedFeeAmount);
                        emit LOG_COLLECTED_FUND(_tokenIn, collectedFeeAmount);
                    }
                }
            }
        }
        
        interface IFaaSPool {
            function stake(uint) external;
            function withdraw(uint) external;
            function getReward(uint8 _pid, address _account) external;
            function getAllRewards(address _account) external;
            function pendingReward(uint8 _pid, address _account) external view returns (uint);
            function emergencyWithdraw() external;
        }
        
        interface IFaaSRewardFund {
            function balance(IERC20 _token) external view returns (uint);
            function safeTransfer(IERC20 _token, address _to, uint _value) external;
        }
        
        // This implements BPool contract, and allows for generalized staking, yield farming, and token distribution.
        contract FaaSPoolLite is BPoolLite, IFaaSPool {
            using SafeMath for uint;
        
            // Info of each user.
            struct UserInfo {
                uint amount;
                mapping(uint8 => uint) rewardDebt;
                mapping(uint8 => uint) accumulatedEarned; // will accumulate every time user harvest
                mapping(uint8 => uint) lockReward;
                mapping(uint8 => uint) lockRewardReleased;
                uint lastStakeTime;
            }
        
            // Info of each rewardPool funding.
            struct RewardPoolInfo {
                IERC20 rewardToken;     // Address of rewardPool token contract.
                uint lastRewardBlock;   // Last block number that rewardPool distribution occurs.
                uint endRewardBlock;    // Block number which rewardPool distribution ends.
                uint rewardPerBlock;    // Reward token amount to distribute per block.
                uint accRewardPerShare; // Accumulated rewardPool per share, times 1e18.
        
                uint lockRewardPercent; // Lock reward percent - 0 to disable lock & vesting
                uint startVestingBlock; // Block number which vesting starts.
                uint endVestingBlock;   // Block number which vesting ends.
                uint numOfVestingBlocks;
        
                uint totalPaidRewards;
            }
        
            mapping(address => UserInfo) private userInfo;
            RewardPoolInfo[] public rewardPoolInfo;
        
            IFaaSRewardFund public rewardFund;
            address public exchangeProxy;
            uint public unstakingFrozenTime = 0 days;
        
            event Deposit(address indexed account, uint256 amount);
            event Withdraw(address indexed account, uint256 amount);
            event RewardPaid(uint8 pid, address indexed account, uint256 amount);
        
            constructor(address _factory) public BPoolLite(_factory) {
            }
        
            modifier onlyController() {
                require(msg.sender == controller, "!cntler");
                _;
            }
        
            function finalizeRewardFundInfo(IFaaSRewardFund _rewardFund, uint _unstakingFrozenTime) external onlyController {
                require(address(rewardFund) == address(0), "rewardFund!=null");
                assert(unstakingFrozenTime <= 30 days);
                // do not lock fund for too long, please!
                unstakingFrozenTime = _unstakingFrozenTime;
                rewardFund = _rewardFund;
            }
        
            function setExchangeProxy(address _exchangeProxy) public onlyController {
                exchangeProxy = _exchangeProxy;
            }
        
            function addRewardPool(IERC20 _rewardToken, uint256 _startBlock, uint256 _endRewardBlock, uint256 _rewardPerBlock,
                uint256 _lockRewardPercent, uint256 _startVestingBlock, uint256 _endVestingBlock) external onlyController {
                require(rewardPoolInfo.length < 8, "exceed rwdPoolLim");
                require(_startVestingBlock <= _endVestingBlock, "sVB>eVB");
                _startBlock = (block.number > _startBlock) ? block.number : _startBlock;
                require(_startBlock < _endRewardBlock, "sB>=eB");
                updateReward();
                rewardPoolInfo.push(RewardPoolInfo({
                    rewardToken : _rewardToken,
                    lastRewardBlock : _startBlock,
                    endRewardBlock : _endRewardBlock,
                    rewardPerBlock : _rewardPerBlock,
                    accRewardPerShare : 0,
                    lockRewardPercent : _lockRewardPercent,
                    startVestingBlock : _startVestingBlock,
                    endVestingBlock : _endVestingBlock,
                    numOfVestingBlocks: _endVestingBlock - _startVestingBlock,
                    totalPaidRewards: 0
                    }));
            }
        
            function updateRewardPool(uint8 _pid, uint256 _endRewardBlock, uint256 _rewardPerBlock) public onlyController {
                updateReward(_pid);
                RewardPoolInfo storage rewardPool = rewardPoolInfo[_pid];
                require(block.number <= rewardPool.endRewardBlock, "late");
                rewardPool.endRewardBlock = _endRewardBlock;
                rewardPool.rewardPerBlock = _rewardPerBlock;
            }
        
            function joinPool(uint poolAmountOut, uint[] calldata maxAmountsIn) external override {
                joinPoolFor(msg.sender, poolAmountOut, maxAmountsIn);
            }
        
            function joinPoolFor(address account, uint poolAmountOut, uint[] calldata maxAmountsIn) public _lock_ {
                require(msg.sender == account || msg.sender == exchangeProxy, "!(prx||own)");
                _joinPool(account, poolAmountOut, maxAmountsIn);
                _stakePoolShare(account, poolAmountOut);
            }
        
            function joinPoolNotStake(uint poolAmountOut, uint[] calldata maxAmountsIn) external _lock_ {
                _joinPool(msg.sender, poolAmountOut, maxAmountsIn);
                _pushPoolShare(msg.sender, poolAmountOut);
            }
        
            function _joinPool(address account, uint poolAmountOut, uint[] calldata maxAmountsIn) internal {
                require(finalized, "!fnl");
        
                uint rewardTotal = totalSupply();
                uint ratio = bdiv(poolAmountOut, rewardTotal);
                require(ratio != 0, "erMApr");
        
                for (uint i = 0; i < _tokens.length; i++) {
                    address t = _tokens[i];
                    uint bal = _records[t].balance;
                    uint tokenAmountIn = bmul(ratio, bal);
                    require(tokenAmountIn != 0 && tokenAmountIn <= maxAmountsIn[i], "erMApr||<limIn");
                    _records[t].balance = badd(_records[t].balance, tokenAmountIn);
                    emit LOG_JOIN(account, t, tokenAmountIn);
                    _pullUnderlying(t, msg.sender, tokenAmountIn);
                }
                _mintPoolShare(poolAmountOut);
            }
        
            function stake(uint _shares) external override {
                uint _before = balanceOf(address(this));
                _pullPoolShare(msg.sender, _shares);
                uint _after = balanceOf(address(this));
                _shares = bsub(_after, _before); // Additional check for deflationary tokens
                _stakePoolShare(msg.sender, _shares);
            }
        
            function _stakePoolShare(address _account, uint _shares) internal {
                UserInfo storage user = userInfo[_account];
                getAllRewards(_account);
                user.amount = user.amount.add(_shares);
                uint8 rewardPoolLength = uint8(rewardPoolInfo.length);
                for (uint8 _pid = 0; _pid < rewardPoolLength; ++_pid) {
                    user.rewardDebt[_pid] = user.amount.mul(rewardPoolInfo[_pid].accRewardPerShare).div(1e18);
                }
                user.lastStakeTime = block.timestamp;
                emit Deposit(_account, _shares);
            }
        
            function unfrozenStakeTime(address _account) public view returns (uint) {
                return userInfo[_account].lastStakeTime + unstakingFrozenTime;
            }
        
            function withdraw(uint _amount) public override {
                UserInfo storage user = userInfo[msg.sender];
                require(user.amount >= _amount, "am>us.am");
                require(block.timestamp >= user.lastStakeTime.add(unstakingFrozenTime), "frozen");
                getAllRewards(msg.sender);
                user.amount = bsub(user.amount, _amount);
                uint8 rewardPoolLength = uint8(rewardPoolInfo.length);
                for (uint8 _pid = 0; _pid < rewardPoolLength; ++_pid) {
                    user.rewardDebt[_pid] = user.amount.mul(rewardPoolInfo[_pid].accRewardPerShare).div(1e18);
                }
                _pushPoolShare(msg.sender, _amount);
                emit Withdraw(msg.sender, _amount);
            }
        
            // using PUSH pattern for using by Proxy if needed
            function getAllRewards(address _account) public override {
                uint8 rewardPoolLength = uint8(rewardPoolInfo.length);
                for (uint8 _pid = 0; _pid < rewardPoolLength; ++_pid) {
                    getReward(_pid, _account);
                }
            }
        
            function getReward(uint8 _pid, address _account) public override {
                updateReward(_pid);
                UserInfo storage user = userInfo[_account];
                RewardPoolInfo storage rewardPool = rewardPoolInfo[_pid];
                uint _pendingReward = user.amount.mul(rewardPool.accRewardPerShare).div(1e18).sub(user.rewardDebt[_pid]);
                uint _lockRewardPercent = rewardPool.lockRewardPercent;
                if (_lockRewardPercent > 0) {
                    if (block.number > rewardPool.endVestingBlock) {
                        uint _unlockReward = user.lockReward[_pid].sub(user.lockRewardReleased[_pid]);
                        if (_unlockReward > 0) {
                            _pendingReward = _pendingReward.add(_unlockReward);
                            user.lockRewardReleased[_pid] = user.lockRewardReleased[_pid].add(_unlockReward);
                        }
                    } else {
                        if (_pendingReward > 0) {
                            uint _toLocked = _pendingReward.mul(_lockRewardPercent).div(100);
                            _pendingReward = _pendingReward.sub(_toLocked);
                            user.lockReward[_pid] = user.lockReward[_pid].add(_toLocked);
                        }
                        if (block.number > rewardPool.startVestingBlock) {
                            uint _toReleased = user.lockReward[_pid].mul(block.number.sub(rewardPool.startVestingBlock)).div(rewardPool.numOfVestingBlocks);
                            uint _lockRewardReleased = user.lockRewardReleased[_pid];
                            if (_toReleased > _lockRewardReleased) {
                                uint _unlockReward = _toReleased.sub(_lockRewardReleased);
                                user.lockRewardReleased[_pid] = _lockRewardReleased.add(_unlockReward);
                                _pendingReward = _pendingReward.add(_unlockReward);
                            }
                        }
                    }
                }
                if (_pendingReward > 0) {
                    user.accumulatedEarned[_pid] = user.accumulatedEarned[_pid].add(_pendingReward);
                    rewardPool.totalPaidRewards = rewardPool.totalPaidRewards.add(_pendingReward);
                    rewardFund.safeTransfer(rewardPool.rewardToken, _account, _pendingReward);
                    emit RewardPaid(_pid, _account, _pendingReward);
                    user.rewardDebt[_pid] = user.amount.mul(rewardPoolInfo[_pid].accRewardPerShare).div(1e18);
                }
            }
        
            function pendingReward(uint8 _pid, address _account) public override view returns (uint _pending) {
                UserInfo storage user = userInfo[_account];
                RewardPoolInfo storage rewardPool = rewardPoolInfo[_pid];
                uint _accRewardPerShare = rewardPool.accRewardPerShare;
                uint lpSupply = balanceOf(address(this));
                uint _endRewardBlockApplicable = block.number > rewardPool.endRewardBlock ? rewardPool.endRewardBlock : block.number;
                if (_endRewardBlockApplicable > rewardPool.lastRewardBlock && lpSupply != 0) {
                    uint _numBlocks = _endRewardBlockApplicable.sub(rewardPool.lastRewardBlock);
                    uint _incRewardPerShare = _numBlocks.mul(rewardPool.rewardPerBlock).mul(1e18).div(lpSupply);
                    _accRewardPerShare = _accRewardPerShare.add(_incRewardPerShare);
                }
                _pending = user.amount.mul(_accRewardPerShare).div(1e18).sub(user.rewardDebt[_pid]);
            }
        
            // Withdraw without caring about rewards. EMERGENCY ONLY.
            function emergencyWithdraw() external override {
                UserInfo storage user = userInfo[msg.sender];
                uint _amount = user.amount;
                _pushPoolShare(msg.sender, _amount);
                user.amount = 0;
                uint8 rewardPoolLength = uint8(rewardPoolInfo.length);
                for (uint8 _pid = 0; _pid < rewardPoolLength; ++_pid) {
                    user.rewardDebt[_pid] = 0;
                }
                emit Withdraw(msg.sender, _amount);
            }
        
            function getUserInfo(uint8 _pid, address _account) public view returns (uint amount, uint rewardDebt, uint accumulatedEarned, uint lockReward, uint lockRewardReleased) {
                UserInfo storage user = userInfo[_account];
                amount = user.amount;
                rewardDebt = user.rewardDebt[_pid];
                accumulatedEarned = user.accumulatedEarned[_pid];
                lockReward = user.lockReward[_pid];
                lockRewardReleased = user.lockRewardReleased[_pid];
            }
        
            function exitPool(uint poolAmountIn, uint[] calldata minAmountsOut) external override _lock_ {
                require(finalized, "!fnl");
        
                uint rewardTotal = totalSupply();
                uint _exitFee = bmul(poolAmountIn, exitFee);
                uint pAiAfterExitFee = bsub(poolAmountIn, _exitFee);
                uint ratio = bdiv(pAiAfterExitFee, rewardTotal);
                require(ratio != 0, "erMApr");
        
                uint _externalShares = balanceOf(msg.sender);
                if (_externalShares < poolAmountIn) {
                    uint _withdrawShares = bsub(poolAmountIn, _externalShares);
                    uint _stakedShares = userInfo[msg.sender].amount;
                    require(_stakedShares >= _withdrawShares, "stk<wdr");
                    withdraw(_withdrawShares);
                }
        
                _pullPoolShare(msg.sender, poolAmountIn);
                if (_exitFee > 0) {
                    _pushPoolShare(factory, _exitFee);
                }
                _burnPoolShare(pAiAfterExitFee);
        
                for (uint i = 0; i < _tokens.length; i++) {
                    address t = _tokens[i];
                    uint bal = _records[t].balance;
                    uint tokenAmountOut = bmul(ratio, bal);
                    require(tokenAmountOut != 0, "erMApr");
                    require(tokenAmountOut >= minAmountsOut[i], "<limO");
                    _records[t].balance = bsub(_records[t].balance, tokenAmountOut);
                    emit LOG_EXIT(msg.sender, t, tokenAmountOut);
                    _pushUnderlying(t, msg.sender, tokenAmountOut);
                }
            }
        
            function updateReward() public {
                uint8 rewardPoolLength = uint8(rewardPoolInfo.length);
                for (uint8 _pid = 0; _pid < rewardPoolLength; ++_pid) {
                    updateReward(_pid);
                }
            }
        
            function updateReward(uint8 _pid) public {
                RewardPoolInfo storage rewardPool = rewardPoolInfo[_pid];
                uint _endRewardBlockApplicable = block.number > rewardPool.endRewardBlock ? rewardPool.endRewardBlock : block.number;
                if (_endRewardBlockApplicable > rewardPool.lastRewardBlock) {
                    uint lpSupply = balanceOf(address(this));
                    if (lpSupply > 0) {
                        uint _numBlocks = _endRewardBlockApplicable.sub(rewardPool.lastRewardBlock);
                        uint _incRewardPerShare = _numBlocks.mul(rewardPool.rewardPerBlock).mul(1e18).div(lpSupply);
                        rewardPool.accRewardPerShare = rewardPool.accRewardPerShare.add(_incRewardPerShare);
                    }
                    rewardPool.lastRewardBlock = _endRewardBlockApplicable;
                }
            }
        }

        File 2 of 6: FaaSRewardFund
        /**
         *Submitted for verification at Etherscan.io on 2020-11-22
        */
        
        // SPDX-License-Identifier: MIT
        
        pragma solidity 0.6.12;
        
        pragma experimental ABIEncoderV2;
        
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20 {
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
        
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
        
            /**
             * @dev Moves `amount` tokens from the caller's account to `recipient`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address recipient, uint256 amount) external returns (bool);
        
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
        
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
        
            /**
             * @dev Moves `amount` tokens from `sender` to `recipient` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
        
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
        
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
        }
        
        /**
         * @dev Wrappers over Solidity's arithmetic operations with added overflow
         * checks.
         *
         * Arithmetic operations in Solidity wrap on overflow. This can easily result
         * in bugs, because programmers usually assume that an overflow raises an
         * error, which is the standard behavior in high level programming languages.
         * `SafeMath` restores this intuition by reverting the transaction when an
         * operation overflows.
         *
         * Using this library instead of the unchecked operations eliminates an entire
         * class of bugs, so it's recommended to use it always.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `+` operator.
             *
             * Requirements:
             *
             * - Addition cannot overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a, "SafeMath: addition overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                return sub(a, b, "SafeMath: subtraction overflow");
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b <= a, errorMessage);
                uint256 c = a - b;
        
                return c;
            }
        
            /**
             * @dev Returns the multiplication of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `*` operator.
             *
             * Requirements:
             *
             * - Multiplication cannot overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                if (a == 0) {
                    return 0;
                }
        
                uint256 c = a * b;
                require(c / a == b, "SafeMath: multiplication overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return div(a, b, "SafeMath: division by zero");
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                uint256 c = a / b;
                // assert(a == b * c + a % b); // There is no case in which this doesn't hold
        
                return c;
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                return mod(a, b, "SafeMath: modulo by zero");
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts with custom message when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b != 0, errorMessage);
                return a % b;
            }
        }
        
        /**
         * @dev Collection of functions related to the address type
         */
        library Address {
            /**
             * @dev Returns true if `account` is a contract.
             *
             * [IMPORTANT]
             * ====
             * It is unsafe to assume that an address for which this function returns
             * false is an externally-owned account (EOA) and not a contract.
             *
             * Among others, `isContract` will return false for the following
             * types of addresses:
             *
             *  - an externally-owned account
             *  - a contract in construction
             *  - an address where a contract will be created
             *  - an address where a contract lived, but was destroyed
             * ====
             */
            function isContract(address account) internal view returns (bool) {
                // This method relies in extcodesize, which returns 0 for contracts in
                // construction, since the code is only stored at the end of the
                // constructor execution.
        
                uint256 size;
                // solhint-disable-next-line no-inline-assembly
                assembly { size := extcodesize(account) }
                return size > 0;
            }
        
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                require(address(this).balance >= amount, "Address: insufficient balance");
        
                // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                (bool success, ) = recipient.call{ value: amount }("");
                require(success, "Address: unable to send value, recipient may have reverted");
            }
        
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain`call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason, it is bubbled up by this
             * function (like regular Solidity function calls).
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
              return functionCall(target, data, "Address: low-level call failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
             * `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
                return _functionCallWithValue(target, data, 0, errorMessage);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
             * with `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) {
                require(address(this).balance >= value, "Address: insufficient balance for call");
                return _functionCallWithValue(target, data, value, errorMessage);
            }
        
            function _functionCallWithValue(address target, bytes memory data, uint256 weiValue, string memory errorMessage) private returns (bytes memory) {
                require(isContract(target), "Address: call to non-contract");
        
                // solhint-disable-next-line avoid-low-level-calls
                (bool success, bytes memory returndata) = target.call{ value : weiValue}(data);
                if (success) {
                    return returndata;
                } else {
                    // Look for revert reason and bubble it up if present
                    if (returndata.length > 0) {
                        // The easiest way to bubble the revert reason is using memory via assembly
        
                        // solhint-disable-next-line no-inline-assembly
                        assembly {
                            let returndata_size := mload(returndata)
                            revert(add(32, returndata), returndata_size)
                        }
                    } else {
                        revert(errorMessage);
                    }
                }
            }
        }
        
        /**
         * @title SafeERC20
         * @dev Wrappers around ERC20 operations that throw on failure (when the token
         * contract returns false). Tokens that return no value (and instead revert or
         * throw on failure) are also supported, non-reverting calls are assumed to be
         * successful.
         * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
         * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
         */
        library SafeERC20 {
            using SafeMath for uint256;
            using Address for address;
        
            function safeTransfer(IERC20 token, address to, uint256 value) internal {
                _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
            }
        
            function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
                _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
            }
        
            /**
             * @dev Deprecated. This function has issues similar to the ones found in
             * {IERC20-approve}, and its usage is discouraged.
             *
             * Whenever possible, use {safeIncreaseAllowance} and
             * {safeDecreaseAllowance} instead.
             */
            function safeApprove(IERC20 token, address spender, uint256 value) internal {
                // safeApprove should only be called when setting an initial allowance,
                // or when resetting it to zero. To increase and decrease it, use
                // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                // solhint-disable-next-line max-line-length
                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 safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
                uint256 newAllowance = token.allowance(address(this), spender).add(value);
                _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
            }
        
            function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
                uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero");
                _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
            }
        
            /**
             * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
             * on the return value: the return value is optional (but if data is returned, it must not be false).
             * @param token The token targeted by the call.
             * @param data The call data (encoded using abi.encode or one of its variants).
             */
            function _callOptionalReturn(IERC20 token, bytes memory data) private {
                // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
                // the target address contains contract code and also asserts for success in the low-level call.
        
                bytes memory returndata = address(token).functionCall(data, "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");
                }
            }
        }
        
        interface IBPool is IERC20 {
            function version() external view returns(uint);
            function swapExactAmountIn(address, uint, address, uint, uint) external returns (uint, uint);
        
            function swapExactAmountOut(address, uint, address, uint, uint) external returns (uint, uint);
        
            function calcInGivenOut(uint, uint, uint, uint, uint, uint) external pure returns (uint);
        
            function calcOutGivenIn(uint, uint, uint, uint, uint, uint) external pure returns (uint);
        
            function getDenormalizedWeight(address) external view returns (uint);
        
            function swapFee() external view returns (uint);
        
            function setSwapFee(uint _swapFee) external;
        
            function bind(address token, uint balance, uint denorm) external;
        
            function rebind(address token, uint balance, uint denorm) external;
        
            function finalize(
                uint _swapFee,
                uint _initPoolSupply,
                address[] calldata _bindTokens,
                uint[] calldata _bindDenorms
            ) external;
        
            function setPublicSwap(bool _publicSwap) external;
            function setController(address _controller) external;
            function setExchangeProxy(address _exchangeProxy) external;
            function getFinalTokens() external view returns (address[] memory tokens);
        
        
            function getTotalDenormalizedWeight() external view returns (uint);
        
            function getBalance(address token) external view returns (uint);
        
        
            function joinPool(uint poolAmountOut, uint[] calldata maxAmountsIn) external;
            function joinPoolFor(address account, uint rewardAmountOut, uint[] calldata maxAmountsIn) external;
            function joinswapPoolAmountOut(address tokenIn, uint poolAmountOut, uint maxAmountIn) external returns (uint tokenAmountIn);
        
            function exitPool(uint poolAmountIn, uint[] calldata minAmountsOut) external;
            function exitswapPoolAmountIn(address tokenOut, uint poolAmountIn, uint minAmountOut) external returns (uint tokenAmountOut);
            function exitswapExternAmountOut(address tokenOut, uint tokenAmountOut, uint maxPoolAmountIn) external returns (uint poolAmountIn);
            function joinswapExternAmountIn(
                address tokenIn,
                uint tokenAmountIn,
                uint minPoolAmountOut
            ) external returns (uint poolAmountOut);
            function finalizeRewardFundInfo(address _rewardFund, uint _unstakingFrozenTime) external;
            function addRewardPool(IERC20 _rewardToken, uint256 _startBlock, uint256 _endRewardBlock, uint256 _rewardPerBlock,
                uint256 _lockRewardPercent, uint256 _startVestingBlock, uint256 _endVestingBlock) external;
        }
        
        interface IFreeFromUpTo {
            function freeFromUpTo(address from, uint256 value) external returns (uint256 freed);
        }
        
        interface IBFactory {
            function newBPool() external returns (IBPool);
        }
        
        interface IValueLiquidRegistry {
            function getBestPoolsWithLimit(address, address, uint) external view returns (address[] memory);
        }
        
        interface IWETH {
            function deposit() external payable;
            function allowance(address owner, address spender) external view returns (uint256);
            function transfer(address to, uint value) external returns (bool);
            function withdraw(uint) external;
            function balanceOf(address account) external view returns (uint256);
        }
        
        // Token pool of arbitrary ERC20 token.
        // This is owned and used by a parent FaaSPool.
        contract FaaSRewardFund {
            using SafeERC20 for IERC20;
            using SafeMath for uint256;
        
            event NewAdmin(address indexed newAdmin);
            event NewPendingAdmin(address indexed newPendingAdmin);
            event NewDelay(uint indexed newDelay);
            event CancelTransaction(bytes32 indexed txHash, address indexed target, uint value, string signature, bytes data, uint eta);
            event ExecuteTransaction(bytes32 indexed txHash, address indexed target, uint value, string signature, bytes data, uint eta);
            event QueueTransaction(bytes32 indexed txHash, address indexed target, uint value, string signature, bytes data, uint eta);
        
            uint public constant GRACE_PERIOD = 14 days;
            uint public constant MINIMUM_DELAY = 1 days;
            uint public constant MAXIMUM_DELAY = 30 days;
            bool private _initialized;
            address public faasPool;
            address public admin;
            address public pendingAdmin;
            uint public delay;
            bool public admin_initialized;
            mapping(bytes32 => bool) public queuedTransactions;
        
            constructor() public {
                admin_initialized = false;
                _initialized = false;
            }
        
            function initialized(address admin_, uint delay_, address _faasPool) public {
                require(_initialized == false, "Timelock::constructor: Delay must exceed minimum delay.");
                require(delay_ >= MINIMUM_DELAY, "Timelock::constructor: Delay must exceed minimum delay.");
                require(delay_ <= MAXIMUM_DELAY, "Timelock::constructor: Delay must not exceed maximum delay.");
                admin = admin_;
                faasPool = _faasPool;
                delay = delay_;
                _initialized = true;
            }
        
            // XXX: function() external payable { }
            receive() external payable {}
        
            function setDelay(uint delay_) public {
                require(msg.sender == address(this), "Timelock::setDelay: Call must come from Timelock.");
                require(delay_ >= MINIMUM_DELAY, "Timelock::setDelay: Delay must exceed minimum delay.");
                require(delay_ <= MAXIMUM_DELAY, "Timelock::setDelay: Delay must not exceed maximum delay.");
                delay = delay_;
        
                emit NewDelay(delay);
            }
        
            function acceptAdmin() public {
                require(msg.sender == pendingAdmin, "Timelock::acceptAdmin: Call must come from pendingAdmin.");
                admin = msg.sender;
                pendingAdmin = address(0);
        
                emit NewAdmin(admin);
            }
        
            function setPendingAdmin(address pendingAdmin_) public {
                // allows one time setting of admin for deployment purposes
                if (admin_initialized) {
                    require(msg.sender == address(this), "Timelock::setPendingAdmin: Call must come from Timelock.");
                } else {
                    require(msg.sender == admin, "Timelock::setPendingAdmin: First call must come from admin.");
                    admin_initialized = true;
                }
                pendingAdmin = pendingAdmin_;
        
                emit NewPendingAdmin(pendingAdmin);
            }
        
            function queueTransaction(address target, uint value, string memory signature, bytes memory data, uint eta) public returns (bytes32) {
                require(msg.sender == admin, "Timelock::queueTransaction: Call must come from admin.");
                require(eta >= getBlockTimestamp().add(delay), "Timelock::queueTransaction: Estimated execution block must satisfy delay.");
        
                bytes32 txHash = keccak256(abi.encode(target, value, signature, data, eta));
                queuedTransactions[txHash] = true;
        
                emit QueueTransaction(txHash, target, value, signature, data, eta);
                return txHash;
            }
        
            function cancelTransaction(address target, uint value, string memory signature, bytes memory data, uint eta) public {
                require(msg.sender == admin, "Timelock::cancelTransaction: Call must come from admin.");
        
                bytes32 txHash = keccak256(abi.encode(target, value, signature, data, eta));
                queuedTransactions[txHash] = false;
        
                emit CancelTransaction(txHash, target, value, signature, data, eta);
            }
        
            function executeTransaction(address target, uint value, string memory signature, bytes memory data, uint eta) public payable returns (bytes memory) {
                require(msg.sender == admin, "Timelock::executeTransaction: Call must come from admin.");
        
                bytes32 txHash = keccak256(abi.encode(target, value, signature, data, eta));
                require(queuedTransactions[txHash], "Timelock::executeTransaction: Transaction hasn't been queued.");
                require(getBlockTimestamp() >= eta, "Timelock::executeTransaction: Transaction hasn't surpassed time lock.");
                require(getBlockTimestamp() <= eta.add(GRACE_PERIOD), "Timelock::executeTransaction: Transaction is stale.");
        
                queuedTransactions[txHash] = false;
        
                bytes memory callData;
        
                if (bytes(signature).length == 0) {
                    callData = data;
                } else {
                    callData = abi.encodePacked(bytes4(keccak256(bytes(signature))), data);
                }
        
                // solium-disable-next-line security/no-call-value
                (bool success, bytes memory returnData) = target.call{value : value}(callData);
                require(success, "Timelock::executeTransaction: Transaction execution reverted.");
        
                emit ExecuteTransaction(txHash, target, value, signature, data, eta);
        
                return returnData;
            }
        
            function getBlockTimestamp() internal view returns (uint) {
                return block.timestamp;
            }
        
        
            function balance(IERC20 _token) public view returns (uint256) {
                return _token.balanceOf(address(this));
            }
        
            function safeTransfer(IERC20 _token, address _to, uint256 _value) external {
                require(msg.sender == faasPool, "!faasPool");
                uint256 _tokenBal = balance(_token);
                _token.safeTransfer(_to, _tokenBal > _value ? _value : _tokenBal);
            }
        
            /**
             * This function allows governance to take unsupported tokens out of the contract. This is in an effort to make someone whole, should they seriously mess up.
             * There is no guarantee governance will vote to return these. It also allows for removal of airdropped tokens.
             */
            function governanceRecoverUnsupported(IERC20 _token, uint _amount, address _to) external {
                require(msg.sender == address(this), "!timelock");
                if ((address(_token) == address(0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE))) {
                    (bool xfer,) = _to.call{value : _amount}("");
                    require(xfer, "ERR_ETH_FAILED");
                } else {
                    _token.safeTransfer(_to, _amount);
                }
            }
        }
        
        contract FaasPoolProxy {
            using SafeMath for uint256;
            using SafeERC20 for IERC20;
            using Address for address;
        
            IFreeFromUpTo public constant chi = IFreeFromUpTo(0x0000000000004946c0e9F43F4Dee607b0eF1fA1c);
        
            modifier discountCHI(uint8 flag) {
                if ((flag & 0x1) == 0) {
                    _;
                } else {
                    uint256 gasStart = gasleft();
                    _;
                    uint256 gasSpent = 21000 + gasStart - gasleft() + 16 * msg.data.length;
                    chi.freeFromUpTo(msg.sender, (gasSpent + 14154) / 41130);
                }
            }
        
            IWETH weth;
            address private constant ETH_ADDRESS = address(0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE);
            address public governance;
            address public exchangeProxy;
        
            constructor(address _weth,address _exchangeProxy) public {
                weth = IWETH(_weth);
                governance = tx.origin;
                exchangeProxy = _exchangeProxy;
            }
            struct PoolRewardInfo {
                IERC20 rewardToken;
                uint256 startBlock;
                uint256 endRewardBlock;
                uint256 rewardPerBlock;
                uint256 lockRewardPercent;
                uint256 startVestingBlock;
                uint256 endVestingBlock;
                uint unstakingFrozenTime;
                uint rewardFundAmount;
            }
        
            struct PoolInfo {
                IBFactory factory;
                address[] tokens;
                uint[] balances;
                uint[] denorms;
                uint swapFee;
                uint initPoolSupply;
            }
        
            receive() external payable {}
        
            function setExchangeProxy(address _exchangeProxy) external {
                require(msg.sender == governance, "!governance");
                exchangeProxy = _exchangeProxy;
            }
            function createInternal(
                PoolInfo calldata poolInfo
        
            ) internal returns (IBPool pool) {
                address[] memory tokens = poolInfo.tokens;
                require(tokens.length == poolInfo.balances.length, "ERR_LENGTH_MISMATCH");
                require(tokens.length == poolInfo.denorms.length, "ERR_LENGTH_MISMATCH");
                pool = poolInfo.factory.newBPool();
                bool containsETH = false;
                for (uint i = 0; i < tokens.length; i++) {
                    if (transferFromAllTo(tokens[i], poolInfo.balances[i], address(pool))) {
                        containsETH = true;
                        tokens[i] = address(weth);
                    }
                }
                require(msg.value == 0 || containsETH, "!invalid payable");
                pool.finalize(poolInfo.swapFee, poolInfo.initPoolSupply, tokens, poolInfo.denorms);
        
            }
        
            function createFaaSReward(
                PoolInfo calldata poolInfo,
                PoolRewardInfo calldata poolRewardInfo,
                uint8 flag
            ) payable external discountCHI(flag) returns (IBPool pool) {
                pool = createInternal(poolInfo);
                {
                    FaaSRewardFund faasRewardFund = new FaaSRewardFund();
                    pool.finalizeRewardFundInfo(address(faasRewardFund), poolRewardInfo.unstakingFrozenTime);
                    pool.addRewardPool(
                        poolRewardInfo.rewardToken,
                        poolRewardInfo.startBlock,
                        poolRewardInfo.endRewardBlock,
                        poolRewardInfo.rewardPerBlock,
                        poolRewardInfo.lockRewardPercent,
                        poolRewardInfo.startVestingBlock,
                        poolRewardInfo.endVestingBlock);
                    transferFromAllTo(address(poolRewardInfo.rewardToken), poolRewardInfo.rewardFundAmount, address(faasRewardFund));
                    faasRewardFund.initialized(msg.sender, poolRewardInfo.unstakingFrozenTime + 1 days, address(pool));
                    pool.setExchangeProxy(exchangeProxy);
                    pool.setController(address(faasRewardFund));
                }
                uint lpAmount = pool.balanceOf(address(this));
                if (lpAmount > 0) {
                    IERC20(pool).safeTransfer(msg.sender, lpAmount);
                }
            }
        
            function isETH(IERC20 token) internal pure returns (bool) {
                return (address(token) == ETH_ADDRESS);
            }
        
            function transferFromAllTo(address token, uint amount, address to) internal returns (bool containsETH) {
                if (isETH(IERC20(token))) {
                    require(amount == msg.value, "!invalid amount");
                    weth.deposit{value : amount}();
                    weth.transfer(to, amount);
                    containsETH = true;
                } else {
                    IERC20(token).safeTransferFrom(msg.sender, to, amount);
                }
                return containsETH;
            }
        
            function transferFromAllAndApprove(address token, uint amount, address spender) internal returns (bool containsETH) {
                if (isETH(IERC20(token))) {
                    require(amount == msg.value, "!invalid amount");
                    weth.deposit{value : amount}();
                    if (weth.allowance(address(this), spender) > 0) {
                        IERC20(address(weth)).safeApprove(address(spender), 0);
                    }
                    IERC20(address(weth)).safeApprove(spender, amount);
                    containsETH = true;
                } else {
                    IERC20(token).safeTransferFrom(msg.sender, address(this), amount);
                    if (IERC20(token).allowance(address(this), spender) > 0) {
                        IERC20(token).safeApprove(spender, 0);
                    }
                    IERC20(token).safeApprove(spender, amount);
                }
                return containsETH;
            }
        }

        File 3 of 6: Share
        // SPDX-License-Identifier: MIT
        
        pragma solidity 0.6.12;
        
        /**
         * @dev Wrappers over Solidity's arithmetic operations with added overflow
         * checks.
         *
         * Arithmetic operations in Solidity wrap on overflow. This can easily result
         * in bugs, because programmers usually assume that an overflow raises an
         * error, which is the standard behavior in high level programming languages.
         * `SafeMath` restores this intuition by reverting the transaction when an
         * operation overflows.
         *
         * Using this library instead of the unchecked operations eliminates an entire
         * class of bugs, so it's recommended to use it always.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `+` operator.
             *
             * Requirements:
             *
             * - Addition cannot overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a, "SafeMath: addition overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                return sub(a, b, "SafeMath: subtraction overflow");
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b <= a, errorMessage);
                uint256 c = a - b;
        
                return c;
            }
        
            /**
             * @dev Returns the multiplication of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `*` operator.
             *
             * Requirements:
             *
             * - Multiplication cannot overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                if (a == 0) {
                    return 0;
                }
        
                uint256 c = a * b;
                require(c / a == b, "SafeMath: multiplication overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return div(a, b, "SafeMath: division by zero");
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                uint256 c = a / b;
                // assert(a == b * c + a % b); // There is no case in which this doesn't hold
        
                return c;
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                return mod(a, b, "SafeMath: modulo by zero");
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts with custom message when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b != 0, errorMessage);
                return a % b;
            }
        }
        
        /*
         * @dev Provides information about the current execution context, including the
         * sender of the transaction and its data. While these are generally available
         * via msg.sender and msg.data, they should not be accessed in such a direct
         * manner, since when dealing with GSN meta-transactions the account sending and
         * paying for execution may not be the actual sender (as far as an application
         * is concerned).
         *
         * This contract is only required for intermediate, library-like contracts.
         */
        abstract contract Context {
            function _msgSender() internal view virtual returns (address payable) {
                return msg.sender;
            }
        
            function _msgData() internal view virtual returns (bytes memory) {
                this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
                return msg.data;
            }
        }
        
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20 {
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
        
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
        
            /**
             * @dev Moves `amount` tokens from the caller's account to `recipient`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address recipient, uint256 amount) external returns (bool);
        
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
        
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
        
            /**
             * @dev Moves `amount` tokens from `sender` to `recipient` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address sender,
                address recipient,
                uint256 amount
            ) external returns (bool);
        
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
        
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
        }
        
        /**
         * @dev Collection of functions related to the address type
         */
        library Address {
            /**
             * @dev Returns true if `account` is a contract.
             *
             * [IMPORTANT]
             * ====
             * It is unsafe to assume that an address for which this function returns
             * false is an externally-owned account (EOA) and not a contract.
             *
             * Among others, `isContract` will return false for the following
             * types of addresses:
             *
             *  - an externally-owned account
             *  - a contract in construction
             *  - an address where a contract will be created
             *  - an address where a contract lived, but was destroyed
             * ====
             */
            function isContract(address account) internal view returns (bool) {
                // This method relies in extcodesize, which returns 0 for contracts in
                // construction, since the code is only stored at the end of the
                // constructor execution.
        
                uint256 size;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    size := extcodesize(account)
                }
                return size > 0;
            }
        
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                require(address(this).balance >= amount, "Address: insufficient balance");
        
                // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                (bool success, ) = recipient.call{value: amount}("");
                require(success, "Address: unable to send value, recipient may have reverted");
            }
        
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain`call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason, it is bubbled up by this
             * function (like regular Solidity function calls).
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                return functionCall(target, data, "Address: low-level call failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
             * `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal returns (bytes memory) {
                return _functionCallWithValue(target, data, 0, errorMessage);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value
            ) internal returns (bytes memory) {
                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
             * with `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value,
                string memory errorMessage
            ) internal returns (bytes memory) {
                require(address(this).balance >= value, "Address: insufficient balance for call");
                return _functionCallWithValue(target, data, value, errorMessage);
            }
        
            function _functionCallWithValue(
                address target,
                bytes memory data,
                uint256 weiValue,
                string memory errorMessage
            ) private returns (bytes memory) {
                require(isContract(target), "Address: call to non-contract");
        
                // solhint-disable-next-line avoid-low-level-calls
                (bool success, bytes memory returndata) = target.call{value: weiValue}(data);
                if (success) {
                    return returndata;
                } else {
                    // Look for revert reason and bubble it up if present
                    if (returndata.length > 0) {
                        // The easiest way to bubble the revert reason is using memory via assembly
        
                        // solhint-disable-next-line no-inline-assembly
                        assembly {
                            let returndata_size := mload(returndata)
                            revert(add(32, returndata), returndata_size)
                        }
                    } else {
                        revert(errorMessage);
                    }
                }
            }
        }
        
        /**
         * @dev Implementation of the {IERC20} interface.
         *
         * This implementation is agnostic to the way tokens are created. This means
         * that a supply mechanism has to be added in a derived contract using {_mint}.
         * For a generic mechanism see {ERC20PresetMinterPauser}.
         *
         * TIP: For a detailed writeup see our guide
         * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
         * to implement supply mechanisms].
         *
         * We have followed general OpenZeppelin guidelines: functions revert instead
         * of returning `false` on failure. This behavior is nonetheless conventional
         * and does not conflict with the expectations of ERC20 applications.
         *
         * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
         * This allows applications to reconstruct the allowance for all accounts just
         * by listening to said events. Other implementations of the EIP may not emit
         * these events, as it isn't required by the specification.
         *
         * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
         * functions have been added to mitigate the well-known issues around setting
         * allowances. See {IERC20-approve}.
         */
        contract ERC20 is Context, IERC20 {
            using SafeMath for uint256;
            using Address for address;
        
            mapping(address => uint256) private _balances;
        
            mapping(address => mapping(address => uint256)) private _allowances;
        
            uint256 private _totalSupply;
        
            string private _name;
            string private _symbol;
            uint8 private _decimals;
        
            /**
             * @dev Sets the values for {name} and {symbol}, initializes {decimals} with
             * a default value of 18.
             *
             * To select a different value for {decimals}, use {_setupDecimals}.
             *
             * All three of these values are immutable: they can only be set once during
             * construction.
             */
            constructor(string memory name, string memory symbol) public {
                _name = name;
                _symbol = symbol;
                _decimals = 18;
            }
        
            /**
             * @dev Returns the name of the token.
             */
            function name() public view returns (string memory) {
                return _name;
            }
        
            /**
             * @dev Returns the symbol of the token, usually a shorter version of the
             * name.
             */
            function symbol() public view returns (string memory) {
                return _symbol;
            }
        
            /**
             * @dev Returns the number of decimals used to get its user representation.
             * For example, if `decimals` equals `2`, a balance of `505` tokens should
             * be displayed to a user as `5,05` (`505 / 10 ** 2`).
             *
             * Tokens usually opt for a value of 18, imitating the relationship between
             * Ether and Wei. This is the value {ERC20} uses, unless {_setupDecimals} is
             * called.
             *
             * NOTE: This information is only used for _display_ purposes: it in
             * no way affects any of the arithmetic of the contract, including
             * {IERC20-balanceOf} and {IERC20-transfer}.
             */
            function decimals() public view returns (uint8) {
                return _decimals;
            }
        
            /**
             * @dev See {IERC20-totalSupply}.
             */
            function totalSupply() public view override returns (uint256) {
                return _totalSupply;
            }
        
            /**
             * @dev See {IERC20-balanceOf}.
             */
            function balanceOf(address account) public view override returns (uint256) {
                return _balances[account];
            }
        
            /**
             * @dev See {IERC20-transfer}.
             *
             * Requirements:
             *
             * - `recipient` cannot be the zero address.
             * - the caller must have a balance of at least `amount`.
             */
            function transfer(address recipient, uint256 amount) public virtual override returns (bool) {
                _transfer(_msgSender(), recipient, amount);
                return true;
            }
        
            /**
             * @dev See {IERC20-allowance}.
             */
            function allowance(address owner, address spender) public view virtual override returns (uint256) {
                return _allowances[owner][spender];
            }
        
            /**
             * @dev See {IERC20-approve}.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             */
            function approve(address spender, uint256 amount) public virtual override returns (bool) {
                _approve(_msgSender(), spender, amount);
                return true;
            }
        
            /**
             * @dev See {IERC20-transferFrom}.
             *
             * Emits an {Approval} event indicating the updated allowance. This is not
             * required by the EIP. See the note at the beginning of {ERC20};
             *
             * Requirements:
             * - `sender` and `recipient` cannot be the zero address.
             * - `sender` must have a balance of at least `amount`.
             * - the caller must have allowance for ``sender``'s tokens of at least
             * `amount`.
             */
            function transferFrom(
                address sender,
                address recipient,
                uint256 amount
            ) public virtual override returns (bool) {
                _transfer(sender, recipient, amount);
                _approve(sender, _msgSender(), _allowances[sender][_msgSender()].sub(amount, "ERC20: transfer amount exceeds allowance"));
                return true;
            }
        
            /**
             * @dev Atomically increases the allowance granted to `spender` by the caller.
             *
             * This is an alternative to {approve} that can be used as a mitigation for
             * problems described in {IERC20-approve}.
             *
             * Emits an {Approval} event indicating the updated allowance.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             */
            function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
                _approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));
                return true;
            }
        
            /**
             * @dev Atomically decreases the allowance granted to `spender` by the caller.
             *
             * This is an alternative to {approve} that can be used as a mitigation for
             * problems described in {IERC20-approve}.
             *
             * Emits an {Approval} event indicating the updated allowance.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             * - `spender` must have allowance for the caller of at least
             * `subtractedValue`.
             */
            function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
                _approve(_msgSender(), spender, _allowances[_msgSender()][spender].sub(subtractedValue, "ERC20: decreased allowance below zero"));
                return true;
            }
        
            /**
             * @dev Moves tokens `amount` from `sender` to `recipient`.
             *
             * This is internal function is equivalent to {transfer}, and can be used to
             * e.g. implement automatic token fees, slashing mechanisms, etc.
             *
             * Emits a {Transfer} event.
             *
             * Requirements:
             *
             * - `sender` cannot be the zero address.
             * - `recipient` cannot be the zero address.
             * - `sender` must have a balance of at least `amount`.
             */
            function _transfer(
                address sender,
                address recipient,
                uint256 amount
            ) internal virtual {
                require(sender != address(0), "ERC20: transfer from the zero address");
                require(recipient != address(0), "ERC20: transfer to the zero address");
        
                _beforeTokenTransfer(sender, recipient, amount);
        
                _balances[sender] = _balances[sender].sub(amount, "ERC20: transfer amount exceeds balance");
                _balances[recipient] = _balances[recipient].add(amount);
                emit Transfer(sender, recipient, 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 virtual {
                require(account != address(0), "ERC20: mint to the zero address");
        
                _beforeTokenTransfer(address(0), account, amount);
        
                _totalSupply = _totalSupply.add(amount);
                _balances[account] = _balances[account].add(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 virtual {
                require(account != address(0), "ERC20: burn from the zero address");
        
                _beforeTokenTransfer(account, address(0), amount);
        
                _balances[account] = _balances[account].sub(amount, "ERC20: burn amount exceeds balance");
                _totalSupply = _totalSupply.sub(amount);
                emit Transfer(account, address(0), amount);
            }
        
            /**
             * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
             *
             * This internal function is equivalent to `approve`, and can be used to
             * e.g. set automatic allowances for certain subsystems, etc.
             *
             * Emits an {Approval} event.
             *
             * Requirements:
             *
             * - `owner` cannot be the zero address.
             * - `spender` cannot be the zero address.
             */
            function _approve(
                address owner,
                address spender,
                uint256 amount
            ) internal virtual {
                require(owner != address(0), "ERC20: approve from the zero address");
                require(spender != address(0), "ERC20: approve to the zero address");
        
                _allowances[owner][spender] = amount;
                emit Approval(owner, spender, amount);
            }
        
            /**
             * @dev Sets {decimals} to a value other than the default one of 18.
             *
             * WARNING: This function should only be called from the constructor. Most
             * applications that interact with token contracts will not expect
             * {decimals} to ever change, and may work incorrectly if it does.
             */
            function _setupDecimals(uint8 decimals_) internal {
                _decimals = decimals_;
            }
        
            /**
             * @dev Hook that is called before any transfer of tokens. This includes
             * minting and burning.
             *
             * Calling conditions:
             *
             * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
             * will be to transferred to `to`.
             * - when `from` is zero, `amount` tokens will be minted for `to`.
             * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 amount
            ) internal virtual {}
        }
        
        /**
         * @dev Extension of {ERC20} that allows token holders to destroy both their own
         * tokens and those that they have an allowance for, in a way that can be
         * recognized off-chain (via event analysis).
         */
        abstract contract ERC20Burnable is Context, ERC20 {
            /**
             * @dev Destroys `amount` tokens from the caller.
             *
             * See {ERC20-_burn}.
             */
            function burn(uint256 amount) public virtual {
                _burn(_msgSender(), amount);
            }
        
            /**
             * @dev Destroys `amount` tokens from `account`, deducting from the caller's
             * allowance.
             *
             * See {ERC20-_burn} and {ERC20-allowance}.
             *
             * Requirements:
             *
             * - the caller must have allowance for ``accounts``'s tokens of at least
             * `amount`.
             */
            function burnFrom(address account, uint256 amount) public virtual {
                uint256 decreasedAllowance = allowance(account, _msgSender()).sub(amount, "ERC20: burn amount exceeds allowance");
        
                _approve(account, _msgSender(), decreasedAllowance);
                _burn(account, amount);
            }
        }
        
        /**
         * @dev Contract module which provides a basic access control mechanism, where
         * there is an account (an owner) that can be granted exclusive access to
         * specific functions.
         *
         * By default, the owner account will be the one that deploys the contract. This
         * can later be changed with {transferOwnership}.
         *
         * This module is used through inheritance. It will make available the modifier
         * `onlyOwner`, which can be applied to your functions to restrict their use to
         * the owner.
         */
        contract Ownable is Context {
            address private _owner;
        
            event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        
            /**
             * @dev Initializes the contract setting the deployer as the initial owner.
             */
            constructor() internal {
                address msgSender = _msgSender();
                _owner = msgSender;
                emit OwnershipTransferred(address(0), msgSender);
            }
        
            /**
             * @dev Returns the address of the current owner.
             */
            function owner() public view returns (address) {
                return _owner;
            }
        
            /**
             * @dev Throws if called by any account other than the owner.
             */
            modifier onlyOwner() {
                require(_owner == _msgSender(), "Ownable: caller is not the owner");
                _;
            }
        
            /**
             * @dev Leaves the contract without owner. It will not be possible to call
             * `onlyOwner` functions anymore. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby removing any functionality that is only available to the owner.
             */
            function renounceOwnership() public virtual onlyOwner {
                emit OwnershipTransferred(_owner, address(0));
                _owner = address(0);
            }
        
            /**
             * @dev Transfers ownership of the contract to a new account (`newOwner`).
             * Can only be called by the current owner.
             */
            function transferOwnership(address newOwner) public virtual onlyOwner {
                require(newOwner != address(0), "Ownable: new owner is the zero address");
                emit OwnershipTransferred(_owner, newOwner);
                _owner = newOwner;
            }
        }
        
        contract Operator is Context, Ownable {
            address private _operator;
        
            event OperatorTransferred(address indexed previousOperator, address indexed newOperator);
        
            constructor() internal {
                _operator = _msgSender();
                emit OperatorTransferred(address(0), _operator);
            }
        
            function operator() public view returns (address) {
                return _operator;
            }
        
            modifier onlyOperator() {
                require(_operator == msg.sender, "operator: caller is not the operator");
                _;
            }
        
            function isOperator() public view returns (bool) {
                return _msgSender() == _operator;
            }
        
            function transferOperator(address newOperator_) public onlyOwner {
                _transferOperator(newOperator_);
            }
        
            function _transferOperator(address newOperator_) internal {
                require(newOperator_ != address(0), "operator: zero address given for new operator");
                emit OperatorTransferred(address(0), newOperator_);
                _operator = newOperator_;
            }
        }
        
        contract Share is ERC20Burnable, Operator {
            using SafeMath for uint256;
        
            uint256 public constant FARMING_POOL_REWARD_ALLOCATION = 800000 ether;
            uint256 public constant TREASURY_FUND_POOL_ALLOCATION = 100000 ether;
            uint256 public constant DEV_FUND_POOL_ALLOCATION = 99999 ether;
        
            uint256 public constant VESTING_DURATION = 365 days;
            uint256 public startTime = 1608040800; // Tuesday, December 15, 2020 2:00:00 PM UTC
            uint256 public endTime = startTime + VESTING_DURATION; // Wednesday, December 15, 2021 2:00:00 PM
        
            uint256 public treasuryFundRewardRate = TREASURY_FUND_POOL_ALLOCATION / VESTING_DURATION;
            uint256 public devFundRewardRate = DEV_FUND_POOL_ALLOCATION / VESTING_DURATION;
        
            address public treasuryFund;
            address public devFund;
        
            uint256 public treasuryFundLastClaimed = startTime;
            uint256 public devFundLastClaimed = startTime;
        
            bool public rewardPoolDistributed = false;
        
            constructor() public ERC20("Basis Dollar Share", "BSDS") {
                _mint(msg.sender, 1 ether); // mint 1 Basis Share for initial pools deployment
                devFund = msg.sender;
            }
        
            function setTreasuryFund(address _treasuryFund) external {
                require(msg.sender == devFund, "!dev");
                treasuryFund = _treasuryFund;
            }
        
            function setDevFund(address _devFund) external {
                require(msg.sender == devFund, "!dev");
                require(_devFund != address(0), "zero");
                devFund = _devFund;
            }
        
            function unclaimedTreasuryFund() public view returns (uint256 _pending) {
                uint256 _now = block.timestamp;
                if (_now > endTime) _now = endTime;
                if (treasuryFundLastClaimed >= _now) return 0;
                _pending = _now.sub(treasuryFundLastClaimed).mul(treasuryFundRewardRate);
            }
        
            function unclaimedDevFund() public view returns (uint256 _pending) {
                uint256 _now = block.timestamp;
                if (_now > endTime) _now = endTime;
                if (devFundLastClaimed >= _now) return 0;
                _pending = _now.sub(devFundLastClaimed).mul(devFundRewardRate);
            }
        
            /**
             * @dev Claim pending rewards to treasury and dev fund
             */
            function claimRewards() external {
                uint256 _pending = unclaimedTreasuryFund();
                if (_pending > 0 && treasuryFund != address(0)) {
                    _mint(treasuryFund, _pending);
                    treasuryFundLastClaimed = block.timestamp;
                }
                _pending = unclaimedDevFund();
                if (_pending > 0 && devFund != address(0)) {
                    _mint(devFund, _pending);
                    devFundLastClaimed = block.timestamp;
                }
            }
        
            /**
             * @notice distribute to reward pool (only once)
             */
            function distributeReward(address _farmingIncentiveFund) external onlyOperator {
                require(!rewardPoolDistributed, "only can distribute once");
                require(_farmingIncentiveFund != address(0), "!_farmingIncentiveFund");
                rewardPoolDistributed = true;
                _mint(_farmingIncentiveFund, FARMING_POOL_REWARD_ALLOCATION);
            }
        
            function burn(uint256 amount) public override onlyOperator {
                super.burn(amount);
            }
        
            function burnFrom(address account, uint256 amount) public override onlyOperator {
                super.burnFrom(account, amount);
            }
        }

        File 4 of 6: Dollar
        // SPDX-License-Identifier: MIT
        
        pragma solidity 0.6.12;
        
        /*
         * @dev Provides information about the current execution context, including the
         * sender of the transaction and its data. While these are generally available
         * via msg.sender and msg.data, they should not be accessed in such a direct
         * manner, since when dealing with GSN meta-transactions the account sending and
         * paying for execution may not be the actual sender (as far as an application
         * is concerned).
         *
         * This contract is only required for intermediate, library-like contracts.
         */
        abstract contract Context {
            function _msgSender() internal view virtual returns (address payable) {
                return msg.sender;
            }
        
            function _msgData() internal view virtual returns (bytes memory) {
                this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
                return msg.data;
            }
        }
        
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20 {
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
        
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
        
            /**
             * @dev Moves `amount` tokens from the caller's account to `recipient`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address recipient, uint256 amount) external returns (bool);
        
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
        
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
        
            /**
             * @dev Moves `amount` tokens from `sender` to `recipient` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address sender,
                address recipient,
                uint256 amount
            ) external returns (bool);
        
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
        
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
        }
        
        /**
         * @dev Wrappers over Solidity's arithmetic operations with added overflow
         * checks.
         *
         * Arithmetic operations in Solidity wrap on overflow. This can easily result
         * in bugs, because programmers usually assume that an overflow raises an
         * error, which is the standard behavior in high level programming languages.
         * `SafeMath` restores this intuition by reverting the transaction when an
         * operation overflows.
         *
         * Using this library instead of the unchecked operations eliminates an entire
         * class of bugs, so it's recommended to use it always.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `+` operator.
             *
             * Requirements:
             *
             * - Addition cannot overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a, "SafeMath: addition overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                return sub(a, b, "SafeMath: subtraction overflow");
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b <= a, errorMessage);
                uint256 c = a - b;
        
                return c;
            }
        
            /**
             * @dev Returns the multiplication of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `*` operator.
             *
             * Requirements:
             *
             * - Multiplication cannot overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                if (a == 0) {
                    return 0;
                }
        
                uint256 c = a * b;
                require(c / a == b, "SafeMath: multiplication overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return div(a, b, "SafeMath: division by zero");
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                uint256 c = a / b;
                // assert(a == b * c + a % b); // There is no case in which this doesn't hold
        
                return c;
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                return mod(a, b, "SafeMath: modulo by zero");
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts with custom message when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b != 0, errorMessage);
                return a % b;
            }
        }
        
        /**
         * @dev Collection of functions related to the address type
         */
        library Address {
            /**
             * @dev Returns true if `account` is a contract.
             *
             * [IMPORTANT]
             * ====
             * It is unsafe to assume that an address for which this function returns
             * false is an externally-owned account (EOA) and not a contract.
             *
             * Among others, `isContract` will return false for the following
             * types of addresses:
             *
             *  - an externally-owned account
             *  - a contract in construction
             *  - an address where a contract will be created
             *  - an address where a contract lived, but was destroyed
             * ====
             */
            function isContract(address account) internal view returns (bool) {
                // This method relies in extcodesize, which returns 0 for contracts in
                // construction, since the code is only stored at the end of the
                // constructor execution.
        
                uint256 size;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    size := extcodesize(account)
                }
                return size > 0;
            }
        
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                require(address(this).balance >= amount, "Address: insufficient balance");
        
                // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                (bool success, ) = recipient.call{value: amount}("");
                require(success, "Address: unable to send value, recipient may have reverted");
            }
        
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain`call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason, it is bubbled up by this
             * function (like regular Solidity function calls).
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                return functionCall(target, data, "Address: low-level call failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
             * `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal returns (bytes memory) {
                return _functionCallWithValue(target, data, 0, errorMessage);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value
            ) internal returns (bytes memory) {
                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
             * with `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value,
                string memory errorMessage
            ) internal returns (bytes memory) {
                require(address(this).balance >= value, "Address: insufficient balance for call");
                return _functionCallWithValue(target, data, value, errorMessage);
            }
        
            function _functionCallWithValue(
                address target,
                bytes memory data,
                uint256 weiValue,
                string memory errorMessage
            ) private returns (bytes memory) {
                require(isContract(target), "Address: call to non-contract");
        
                // solhint-disable-next-line avoid-low-level-calls
                (bool success, bytes memory returndata) = target.call{value: weiValue}(data);
                if (success) {
                    return returndata;
                } else {
                    // Look for revert reason and bubble it up if present
                    if (returndata.length > 0) {
                        // The easiest way to bubble the revert reason is using memory via assembly
        
                        // solhint-disable-next-line no-inline-assembly
                        assembly {
                            let returndata_size := mload(returndata)
                            revert(add(32, returndata), returndata_size)
                        }
                    } else {
                        revert(errorMessage);
                    }
                }
            }
        }
        
        /**
         * @dev Implementation of the {IERC20} interface.
         *
         * This implementation is agnostic to the way tokens are created. This means
         * that a supply mechanism has to be added in a derived contract using {_mint}.
         * For a generic mechanism see {ERC20PresetMinterPauser}.
         *
         * TIP: For a detailed writeup see our guide
         * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
         * to implement supply mechanisms].
         *
         * We have followed general OpenZeppelin guidelines: functions revert instead
         * of returning `false` on failure. This behavior is nonetheless conventional
         * and does not conflict with the expectations of ERC20 applications.
         *
         * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
         * This allows applications to reconstruct the allowance for all accounts just
         * by listening to said events. Other implementations of the EIP may not emit
         * these events, as it isn't required by the specification.
         *
         * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
         * functions have been added to mitigate the well-known issues around setting
         * allowances. See {IERC20-approve}.
         */
        contract ERC20 is Context, IERC20 {
            using SafeMath for uint256;
            using Address for address;
        
            mapping(address => uint256) private _balances;
        
            mapping(address => mapping(address => uint256)) private _allowances;
        
            uint256 private _totalSupply;
        
            string private _name;
            string private _symbol;
            uint8 private _decimals;
        
            /**
             * @dev Sets the values for {name} and {symbol}, initializes {decimals} with
             * a default value of 18.
             *
             * To select a different value for {decimals}, use {_setupDecimals}.
             *
             * All three of these values are immutable: they can only be set once during
             * construction.
             */
            constructor(string memory name, string memory symbol) public {
                _name = name;
                _symbol = symbol;
                _decimals = 18;
            }
        
            /**
             * @dev Returns the name of the token.
             */
            function name() public view returns (string memory) {
                return _name;
            }
        
            /**
             * @dev Returns the symbol of the token, usually a shorter version of the
             * name.
             */
            function symbol() public view returns (string memory) {
                return _symbol;
            }
        
            /**
             * @dev Returns the number of decimals used to get its user representation.
             * For example, if `decimals` equals `2`, a balance of `505` tokens should
             * be displayed to a user as `5,05` (`505 / 10 ** 2`).
             *
             * Tokens usually opt for a value of 18, imitating the relationship between
             * Ether and Wei. This is the value {ERC20} uses, unless {_setupDecimals} is
             * called.
             *
             * NOTE: This information is only used for _display_ purposes: it in
             * no way affects any of the arithmetic of the contract, including
             * {IERC20-balanceOf} and {IERC20-transfer}.
             */
            function decimals() public view returns (uint8) {
                return _decimals;
            }
        
            /**
             * @dev See {IERC20-totalSupply}.
             */
            function totalSupply() public view override returns (uint256) {
                return _totalSupply;
            }
        
            /**
             * @dev See {IERC20-balanceOf}.
             */
            function balanceOf(address account) public view override returns (uint256) {
                return _balances[account];
            }
        
            /**
             * @dev See {IERC20-transfer}.
             *
             * Requirements:
             *
             * - `recipient` cannot be the zero address.
             * - the caller must have a balance of at least `amount`.
             */
            function transfer(address recipient, uint256 amount) public virtual override returns (bool) {
                _transfer(_msgSender(), recipient, amount);
                return true;
            }
        
            /**
             * @dev See {IERC20-allowance}.
             */
            function allowance(address owner, address spender) public view virtual override returns (uint256) {
                return _allowances[owner][spender];
            }
        
            /**
             * @dev See {IERC20-approve}.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             */
            function approve(address spender, uint256 amount) public virtual override returns (bool) {
                _approve(_msgSender(), spender, amount);
                return true;
            }
        
            /**
             * @dev See {IERC20-transferFrom}.
             *
             * Emits an {Approval} event indicating the updated allowance. This is not
             * required by the EIP. See the note at the beginning of {ERC20};
             *
             * Requirements:
             * - `sender` and `recipient` cannot be the zero address.
             * - `sender` must have a balance of at least `amount`.
             * - the caller must have allowance for ``sender``'s tokens of at least
             * `amount`.
             */
            function transferFrom(
                address sender,
                address recipient,
                uint256 amount
            ) public virtual override returns (bool) {
                _transfer(sender, recipient, amount);
                _approve(sender, _msgSender(), _allowances[sender][_msgSender()].sub(amount, "ERC20: transfer amount exceeds allowance"));
                return true;
            }
        
            /**
             * @dev Atomically increases the allowance granted to `spender` by the caller.
             *
             * This is an alternative to {approve} that can be used as a mitigation for
             * problems described in {IERC20-approve}.
             *
             * Emits an {Approval} event indicating the updated allowance.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             */
            function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
                _approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));
                return true;
            }
        
            /**
             * @dev Atomically decreases the allowance granted to `spender` by the caller.
             *
             * This is an alternative to {approve} that can be used as a mitigation for
             * problems described in {IERC20-approve}.
             *
             * Emits an {Approval} event indicating the updated allowance.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             * - `spender` must have allowance for the caller of at least
             * `subtractedValue`.
             */
            function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
                _approve(_msgSender(), spender, _allowances[_msgSender()][spender].sub(subtractedValue, "ERC20: decreased allowance below zero"));
                return true;
            }
        
            /**
             * @dev Moves tokens `amount` from `sender` to `recipient`.
             *
             * This is internal function is equivalent to {transfer}, and can be used to
             * e.g. implement automatic token fees, slashing mechanisms, etc.
             *
             * Emits a {Transfer} event.
             *
             * Requirements:
             *
             * - `sender` cannot be the zero address.
             * - `recipient` cannot be the zero address.
             * - `sender` must have a balance of at least `amount`.
             */
            function _transfer(
                address sender,
                address recipient,
                uint256 amount
            ) internal virtual {
                require(sender != address(0), "ERC20: transfer from the zero address");
                require(recipient != address(0), "ERC20: transfer to the zero address");
        
                _beforeTokenTransfer(sender, recipient, amount);
        
                _balances[sender] = _balances[sender].sub(amount, "ERC20: transfer amount exceeds balance");
                _balances[recipient] = _balances[recipient].add(amount);
                emit Transfer(sender, recipient, 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 virtual {
                require(account != address(0), "ERC20: mint to the zero address");
        
                _beforeTokenTransfer(address(0), account, amount);
        
                _totalSupply = _totalSupply.add(amount);
                _balances[account] = _balances[account].add(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 virtual {
                require(account != address(0), "ERC20: burn from the zero address");
        
                _beforeTokenTransfer(account, address(0), amount);
        
                _balances[account] = _balances[account].sub(amount, "ERC20: burn amount exceeds balance");
                _totalSupply = _totalSupply.sub(amount);
                emit Transfer(account, address(0), amount);
            }
        
            /**
             * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
             *
             * This internal function is equivalent to `approve`, and can be used to
             * e.g. set automatic allowances for certain subsystems, etc.
             *
             * Emits an {Approval} event.
             *
             * Requirements:
             *
             * - `owner` cannot be the zero address.
             * - `spender` cannot be the zero address.
             */
            function _approve(
                address owner,
                address spender,
                uint256 amount
            ) internal virtual {
                require(owner != address(0), "ERC20: approve from the zero address");
                require(spender != address(0), "ERC20: approve to the zero address");
        
                _allowances[owner][spender] = amount;
                emit Approval(owner, spender, amount);
            }
        
            /**
             * @dev Sets {decimals} to a value other than the default one of 18.
             *
             * WARNING: This function should only be called from the constructor. Most
             * applications that interact with token contracts will not expect
             * {decimals} to ever change, and may work incorrectly if it does.
             */
            function _setupDecimals(uint8 decimals_) internal {
                _decimals = decimals_;
            }
        
            /**
             * @dev Hook that is called before any transfer of tokens. This includes
             * minting and burning.
             *
             * Calling conditions:
             *
             * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
             * will be to transferred to `to`.
             * - when `from` is zero, `amount` tokens will be minted for `to`.
             * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 amount
            ) internal virtual {}
        }
        
        /**
         * @dev Extension of {ERC20} that allows token holders to destroy both their own
         * tokens and those that they have an allowance for, in a way that can be
         * recognized off-chain (via event analysis).
         */
        abstract contract ERC20Burnable is Context, ERC20 {
            /**
             * @dev Destroys `amount` tokens from the caller.
             *
             * See {ERC20-_burn}.
             */
            function burn(uint256 amount) public virtual {
                _burn(_msgSender(), amount);
            }
        
            /**
             * @dev Destroys `amount` tokens from `account`, deducting from the caller's
             * allowance.
             *
             * See {ERC20-_burn} and {ERC20-allowance}.
             *
             * Requirements:
             *
             * - the caller must have allowance for ``accounts``'s tokens of at least
             * `amount`.
             */
            function burnFrom(address account, uint256 amount) public virtual {
                uint256 decreasedAllowance = allowance(account, _msgSender()).sub(amount, "ERC20: burn amount exceeds allowance");
        
                _approve(account, _msgSender(), decreasedAllowance);
                _burn(account, amount);
            }
        }
        
        /**
         * @dev Contract module which provides a basic access control mechanism, where
         * there is an account (an owner) that can be granted exclusive access to
         * specific functions.
         *
         * By default, the owner account will be the one that deploys the contract. This
         * can later be changed with {transferOwnership}.
         *
         * This module is used through inheritance. It will make available the modifier
         * `onlyOwner`, which can be applied to your functions to restrict their use to
         * the owner.
         */
        contract Ownable is Context {
            address private _owner;
        
            event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
        
            /**
             * @dev Initializes the contract setting the deployer as the initial owner.
             */
            constructor() internal {
                address msgSender = _msgSender();
                _owner = msgSender;
                emit OwnershipTransferred(address(0), msgSender);
            }
        
            /**
             * @dev Returns the address of the current owner.
             */
            function owner() public view returns (address) {
                return _owner;
            }
        
            /**
             * @dev Throws if called by any account other than the owner.
             */
            modifier onlyOwner() {
                require(_owner == _msgSender(), "Ownable: caller is not the owner");
                _;
            }
        
            /**
             * @dev Leaves the contract without owner. It will not be possible to call
             * `onlyOwner` functions anymore. Can only be called by the current owner.
             *
             * NOTE: Renouncing ownership will leave the contract without an owner,
             * thereby removing any functionality that is only available to the owner.
             */
            function renounceOwnership() public virtual onlyOwner {
                emit OwnershipTransferred(_owner, address(0));
                _owner = address(0);
            }
        
            /**
             * @dev Transfers ownership of the contract to a new account (`newOwner`).
             * Can only be called by the current owner.
             */
            function transferOwnership(address newOwner) public virtual onlyOwner {
                require(newOwner != address(0), "Ownable: new owner is the zero address");
                emit OwnershipTransferred(_owner, newOwner);
                _owner = newOwner;
            }
        }
        
        contract Operator is Context, Ownable {
            address private _operator;
        
            event OperatorTransferred(address indexed previousOperator, address indexed newOperator);
        
            constructor() internal {
                _operator = _msgSender();
                emit OperatorTransferred(address(0), _operator);
            }
        
            function operator() public view returns (address) {
                return _operator;
            }
        
            modifier onlyOperator() {
                require(_operator == msg.sender, "operator: caller is not the operator");
                _;
            }
        
            function isOperator() public view returns (bool) {
                return _msgSender() == _operator;
            }
        
            function transferOperator(address newOperator_) public onlyOwner {
                _transferOperator(newOperator_);
            }
        
            function _transferOperator(address newOperator_) internal {
                require(newOperator_ != address(0), "operator: zero address given for new operator");
                emit OperatorTransferred(address(0), newOperator_);
                _operator = newOperator_;
            }
        }
        
        contract Dollar is ERC20Burnable, Operator {
            uint256 public constant STABLES_POOL_REWARD_ALLOCATION = 500000 ether;
        
            bool public rewardPoolDistributed = false;
        
            /**
             * @notice Constructs the Basis Dollar ERC-20 contract.
             */
            constructor() public ERC20("Basis Dollar", "BSD") {
                // Mints 1 Basis Dollar to contract creator for initial pool setup
                _mint(msg.sender, 1 ether);
            }
        
            /**
             * @notice Operator mints basis dollar to a recipient
             * @param recipient_ The address of recipient
             * @param amount_ The amount of basis dollar to mint to
             * @return whether the process has been done
             */
            function mint(address recipient_, uint256 amount_) public onlyOperator returns (bool) {
                uint256 balanceBefore = balanceOf(recipient_);
                _mint(recipient_, amount_);
                uint256 balanceAfter = balanceOf(recipient_);
        
                return balanceAfter > balanceBefore;
            }
        
            function burn(uint256 amount) public override onlyOperator {
                super.burn(amount);
            }
        
            function burnFrom(address account, uint256 amount) public override onlyOperator {
                super.burnFrom(account, amount);
            }
        
            /**
             * @notice distribute to reward pool (only once)
             */
            function distributeReward(address _stablesPool) external onlyOperator {
                require(!rewardPoolDistributed, "only can distribute once");
                require(_stablesPool != address(0), "!_stablesPool");
                rewardPoolDistributed = true;
                _mint(_stablesPool, STABLES_POOL_REWARD_ALLOCATION);
            }
        }

        File 5 of 6: FiatTokenProxy
        pragma solidity ^0.4.24;
        
        // File: zos-lib/contracts/upgradeability/Proxy.sol
        
        /**
         * @title Proxy
         * @dev Implements delegation of calls to other contracts, with proper
         * forwarding of return values and bubbling of failures.
         * It defines a fallback function that delegates all calls to the address
         * returned by the abstract _implementation() internal function.
         */
        contract Proxy {
          /**
           * @dev Fallback function.
           * Implemented entirely in `_fallback`.
           */
          function () payable external {
            _fallback();
          }
        
          /**
           * @return The Address of the implementation.
           */
          function _implementation() internal view returns (address);
        
          /**
           * @dev Delegates execution to an implementation contract.
           * This is a low level function that doesn't return to its internal call site.
           * It will return to the external caller whatever the implementation returns.
           * @param implementation Address to delegate.
           */
          function _delegate(address implementation) internal {
            assembly {
              // Copy msg.data. We take full control of memory in this inline assembly
              // block because it will not return to Solidity code. We overwrite the
              // Solidity scratch pad at memory position 0.
              calldatacopy(0, 0, calldatasize)
        
              // Call the implementation.
              // out and outsize are 0 because we don't know the size yet.
              let result := delegatecall(gas, implementation, 0, calldatasize, 0, 0)
        
              // Copy the returned data.
              returndatacopy(0, 0, returndatasize)
        
              switch result
              // delegatecall returns 0 on error.
              case 0 { revert(0, returndatasize) }
              default { return(0, returndatasize) }
            }
          }
        
          /**
           * @dev Function that is run as the first thing in the fallback function.
           * Can be redefined in derived contracts to add functionality.
           * Redefinitions must call super._willFallback().
           */
          function _willFallback() internal {
          }
        
          /**
           * @dev fallback implementation.
           * Extracted to enable manual triggering.
           */
          function _fallback() internal {
            _willFallback();
            _delegate(_implementation());
          }
        }
        
        // File: openzeppelin-solidity/contracts/AddressUtils.sol
        
        /**
         * Utility library of inline functions on addresses
         */
        library AddressUtils {
        
          /**
           * Returns whether the target address is a contract
           * @dev This function will return false if invoked during the constructor of a contract,
           * as the code is not actually created until after the constructor finishes.
           * @param addr address to check
           * @return whether the target address is a contract
           */
          function isContract(address addr) internal view returns (bool) {
            uint256 size;
            // XXX Currently there is no better way to check if there is a contract in an address
            // than to check the size of the code at that address.
            // See https://ethereum.stackexchange.com/a/14016/36603
            // for more details about how this works.
            // TODO Check this again before the Serenity release, because all addresses will be
            // contracts then.
            // solium-disable-next-line security/no-inline-assembly
            assembly { size := extcodesize(addr) }
            return size > 0;
          }
        
        }
        
        // File: zos-lib/contracts/upgradeability/UpgradeabilityProxy.sol
        
        /**
         * @title UpgradeabilityProxy
         * @dev This contract implements a proxy that allows to change the
         * implementation address to which it will delegate.
         * Such a change is called an implementation upgrade.
         */
        contract UpgradeabilityProxy is Proxy {
          /**
           * @dev Emitted when the implementation is upgraded.
           * @param implementation Address of the new implementation.
           */
          event Upgraded(address implementation);
        
          /**
           * @dev Storage slot with the address of the current implementation.
           * This is the keccak-256 hash of "org.zeppelinos.proxy.implementation", and is
           * validated in the constructor.
           */
          bytes32 private constant IMPLEMENTATION_SLOT = 0x7050c9e0f4ca769c69bd3a8ef740bc37934f8e2c036e5a723fd8ee048ed3f8c3;
        
          /**
           * @dev Contract constructor.
           * @param _implementation Address of the initial implementation.
           */
          constructor(address _implementation) public {
            assert(IMPLEMENTATION_SLOT == keccak256("org.zeppelinos.proxy.implementation"));
        
            _setImplementation(_implementation);
          }
        
          /**
           * @dev Returns the current implementation.
           * @return Address of the current implementation
           */
          function _implementation() internal view returns (address impl) {
            bytes32 slot = IMPLEMENTATION_SLOT;
            assembly {
              impl := sload(slot)
            }
          }
        
          /**
           * @dev Upgrades the proxy to a new implementation.
           * @param newImplementation Address of the new implementation.
           */
          function _upgradeTo(address newImplementation) internal {
            _setImplementation(newImplementation);
            emit Upgraded(newImplementation);
          }
        
          /**
           * @dev Sets the implementation address of the proxy.
           * @param newImplementation Address of the new implementation.
           */
          function _setImplementation(address newImplementation) private {
            require(AddressUtils.isContract(newImplementation), "Cannot set a proxy implementation to a non-contract address");
        
            bytes32 slot = IMPLEMENTATION_SLOT;
        
            assembly {
              sstore(slot, newImplementation)
            }
          }
        }
        
        // File: zos-lib/contracts/upgradeability/AdminUpgradeabilityProxy.sol
        
        /**
         * @title AdminUpgradeabilityProxy
         * @dev This contract combines an upgradeability proxy with an authorization
         * mechanism for administrative tasks.
         * All external functions in this contract must be guarded by the
         * `ifAdmin` modifier. See ethereum/solidity#3864 for a Solidity
         * feature proposal that would enable this to be done automatically.
         */
        contract AdminUpgradeabilityProxy is UpgradeabilityProxy {
          /**
           * @dev Emitted when the administration has been transferred.
           * @param previousAdmin Address of the previous admin.
           * @param newAdmin Address of the new admin.
           */
          event AdminChanged(address previousAdmin, address newAdmin);
        
          /**
           * @dev Storage slot with the admin of the contract.
           * This is the keccak-256 hash of "org.zeppelinos.proxy.admin", and is
           * validated in the constructor.
           */
          bytes32 private constant ADMIN_SLOT = 0x10d6a54a4754c8869d6886b5f5d7fbfa5b4522237ea5c60d11bc4e7a1ff9390b;
        
          /**
           * @dev Modifier to check whether the `msg.sender` is the admin.
           * If it is, it will run the function. Otherwise, it will delegate the call
           * to the implementation.
           */
          modifier ifAdmin() {
            if (msg.sender == _admin()) {
              _;
            } else {
              _fallback();
            }
          }
        
          /**
           * Contract constructor.
           * It sets the `msg.sender` as the proxy administrator.
           * @param _implementation address of the initial implementation.
           */
          constructor(address _implementation) UpgradeabilityProxy(_implementation) public {
            assert(ADMIN_SLOT == keccak256("org.zeppelinos.proxy.admin"));
        
            _setAdmin(msg.sender);
          }
        
          /**
           * @return The address of the proxy admin.
           */
          function admin() external view ifAdmin returns (address) {
            return _admin();
          }
        
          /**
           * @return The address of the implementation.
           */
          function implementation() external view ifAdmin returns (address) {
            return _implementation();
          }
        
          /**
           * @dev Changes the admin of the proxy.
           * Only the current admin can call this function.
           * @param newAdmin Address to transfer proxy administration to.
           */
          function changeAdmin(address newAdmin) external ifAdmin {
            require(newAdmin != address(0), "Cannot change the admin of a proxy to the zero address");
            emit AdminChanged(_admin(), newAdmin);
            _setAdmin(newAdmin);
          }
        
          /**
           * @dev Upgrade the backing implementation of the proxy.
           * Only the admin can call this function.
           * @param newImplementation Address of the new implementation.
           */
          function upgradeTo(address newImplementation) external ifAdmin {
            _upgradeTo(newImplementation);
          }
        
          /**
           * @dev Upgrade the backing implementation of the proxy and call a function
           * on the new implementation.
           * This is useful to initialize the proxied contract.
           * @param newImplementation Address of the new implementation.
           * @param data Data to send as msg.data in the low level call.
           * It should include the signature and the parameters of the function to be
           * called, as described in
           * https://solidity.readthedocs.io/en/develop/abi-spec.html#function-selector-and-argument-encoding.
           */
          function upgradeToAndCall(address newImplementation, bytes data) payable external ifAdmin {
            _upgradeTo(newImplementation);
            require(address(this).call.value(msg.value)(data));
          }
        
          /**
           * @return The admin slot.
           */
          function _admin() internal view returns (address adm) {
            bytes32 slot = ADMIN_SLOT;
            assembly {
              adm := sload(slot)
            }
          }
        
          /**
           * @dev Sets the address of the proxy admin.
           * @param newAdmin Address of the new proxy admin.
           */
          function _setAdmin(address newAdmin) internal {
            bytes32 slot = ADMIN_SLOT;
        
            assembly {
              sstore(slot, newAdmin)
            }
          }
        
          /**
           * @dev Only fall back when the sender is not the admin.
           */
          function _willFallback() internal {
            require(msg.sender != _admin(), "Cannot call fallback function from the proxy admin");
            super._willFallback();
          }
        }
        
        // File: contracts/FiatTokenProxy.sol
        
        /**
        * Copyright CENTRE SECZ 2018
        *
        * Permission is hereby granted, free of charge, to any person obtaining a copy 
        * of this software and associated documentation files (the "Software"), to deal 
        * in the Software without restriction, including without limitation the rights 
        * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 
        * copies of the Software, and to permit persons to whom the Software is furnished to 
        * do so, subject to the following conditions:
        *
        * The above copyright notice and this permission notice shall be included in all 
        * copies or substantial portions of the Software.
        *
        * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 
        * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 
        * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 
        * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
        * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 
        * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
        */
        
        pragma solidity ^0.4.24;
        
        
        /**
         * @title FiatTokenProxy
         * @dev This contract proxies FiatToken calls and enables FiatToken upgrades
        */ 
        contract FiatTokenProxy is AdminUpgradeabilityProxy {
            constructor(address _implementation) public AdminUpgradeabilityProxy(_implementation) {
            }
        }

        File 6 of 6: FiatTokenV2_1
        // File: @openzeppelin/contracts/math/SafeMath.sol
        
        // SPDX-License-Identifier: MIT
        
        pragma solidity ^0.6.0;
        
        /**
         * @dev Wrappers over Solidity's arithmetic operations with added overflow
         * checks.
         *
         * Arithmetic operations in Solidity wrap on overflow. This can easily result
         * in bugs, because programmers usually assume that an overflow raises an
         * error, which is the standard behavior in high level programming languages.
         * `SafeMath` restores this intuition by reverting the transaction when an
         * operation overflows.
         *
         * Using this library instead of the unchecked operations eliminates an entire
         * class of bugs, so it's recommended to use it always.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `+` operator.
             *
             * Requirements:
             *
             * - Addition cannot overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a, "SafeMath: addition overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                return sub(a, b, "SafeMath: subtraction overflow");
            }
        
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b <= a, errorMessage);
                uint256 c = a - b;
        
                return c;
            }
        
            /**
             * @dev Returns the multiplication of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `*` operator.
             *
             * Requirements:
             *
             * - Multiplication cannot overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                if (a == 0) {
                    return 0;
                }
        
                uint256 c = a * b;
                require(c / a == b, "SafeMath: multiplication overflow");
        
                return c;
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                return div(a, b, "SafeMath: division by zero");
            }
        
            /**
             * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                uint256 c = a / b;
                // assert(a == b * c + a % b); // There is no case in which this doesn't hold
        
                return c;
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                return mod(a, b, "SafeMath: modulo by zero");
            }
        
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * Reverts with custom message when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(
                uint256 a,
                uint256 b,
                string memory errorMessage
            ) internal pure returns (uint256) {
                require(b != 0, errorMessage);
                return a % b;
            }
        }
        
        // File: @openzeppelin/contracts/token/ERC20/IERC20.sol
        
        pragma solidity ^0.6.0;
        
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20 {
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
        
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
        
            /**
             * @dev Moves `amount` tokens from the caller's account to `recipient`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address recipient, uint256 amount)
                external
                returns (bool);
        
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender)
                external
                view
                returns (uint256);
        
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
        
            /**
             * @dev Moves `amount` tokens from `sender` to `recipient` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address sender,
                address recipient,
                uint256 amount
            ) external returns (bool);
        
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
        
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(
                address indexed owner,
                address indexed spender,
                uint256 value
            );
        }
        
        // File: contracts/v1/AbstractFiatTokenV1.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        abstract contract AbstractFiatTokenV1 is IERC20 {
            function _approve(
                address owner,
                address spender,
                uint256 value
            ) internal virtual;
        
            function _transfer(
                address from,
                address to,
                uint256 value
            ) internal virtual;
        }
        
        // File: contracts/v1/Ownable.sol
        
        /**
         * Copyright (c) 2018 zOS Global Limited.
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        pragma solidity 0.6.12;
        
        /**
         * @notice The Ownable contract has an owner address, and provides basic
         * authorization control functions
         * @dev Forked from https://github.com/OpenZeppelin/openzeppelin-labs/blob/3887ab77b8adafba4a26ace002f3a684c1a3388b/upgradeability_ownership/contracts/ownership/Ownable.sol
         * Modifications:
         * 1. Consolidate OwnableStorage into this contract (7/13/18)
         * 2. Reformat, conform to Solidity 0.6 syntax, and add error messages (5/13/20)
         * 3. Make public functions external (5/27/20)
         */
        contract Ownable {
            // Owner of the contract
            address private _owner;
        
            /**
             * @dev Event to show ownership has been transferred
             * @param previousOwner representing the address of the previous owner
             * @param newOwner representing the address of the new owner
             */
            event OwnershipTransferred(address previousOwner, address newOwner);
        
            /**
             * @dev The constructor sets the original owner of the contract to the sender account.
             */
            constructor() public {
                setOwner(msg.sender);
            }
        
            /**
             * @dev Tells the address of the owner
             * @return the address of the owner
             */
            function owner() external view returns (address) {
                return _owner;
            }
        
            /**
             * @dev Sets a new owner address
             */
            function setOwner(address newOwner) internal {
                _owner = newOwner;
            }
        
            /**
             * @dev Throws if called by any account other than the owner.
             */
            modifier onlyOwner() {
                require(msg.sender == _owner, "Ownable: caller is not the owner");
                _;
            }
        
            /**
             * @dev Allows the current owner to transfer control of the contract to a newOwner.
             * @param newOwner The address to transfer ownership to.
             */
            function transferOwnership(address newOwner) external onlyOwner {
                require(
                    newOwner != address(0),
                    "Ownable: new owner is the zero address"
                );
                emit OwnershipTransferred(_owner, newOwner);
                setOwner(newOwner);
            }
        }
        
        // File: contracts/v1/Pausable.sol
        
        /**
         * Copyright (c) 2016 Smart Contract Solutions, Inc.
         * Copyright (c) 2018-2020 CENTRE SECZ0
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @notice Base contract which allows children to implement an emergency stop
         * mechanism
         * @dev Forked from https://github.com/OpenZeppelin/openzeppelin-contracts/blob/feb665136c0dae9912e08397c1a21c4af3651ef3/contracts/lifecycle/Pausable.sol
         * Modifications:
         * 1. Added pauser role, switched pause/unpause to be onlyPauser (6/14/2018)
         * 2. Removed whenNotPause/whenPaused from pause/unpause (6/14/2018)
         * 3. Removed whenPaused (6/14/2018)
         * 4. Switches ownable library to use ZeppelinOS (7/12/18)
         * 5. Remove constructor (7/13/18)
         * 6. Reformat, conform to Solidity 0.6 syntax and add error messages (5/13/20)
         * 7. Make public functions external (5/27/20)
         */
        contract Pausable is Ownable {
            event Pause();
            event Unpause();
            event PauserChanged(address indexed newAddress);
        
            address public pauser;
            bool public paused = false;
        
            /**
             * @dev Modifier to make a function callable only when the contract is not paused.
             */
            modifier whenNotPaused() {
                require(!paused, "Pausable: paused");
                _;
            }
        
            /**
             * @dev throws if called by any account other than the pauser
             */
            modifier onlyPauser() {
                require(msg.sender == pauser, "Pausable: caller is not the pauser");
                _;
            }
        
            /**
             * @dev called by the owner to pause, triggers stopped state
             */
            function pause() external onlyPauser {
                paused = true;
                emit Pause();
            }
        
            /**
             * @dev called by the owner to unpause, returns to normal state
             */
            function unpause() external onlyPauser {
                paused = false;
                emit Unpause();
            }
        
            /**
             * @dev update the pauser role
             */
            function updatePauser(address _newPauser) external onlyOwner {
                require(
                    _newPauser != address(0),
                    "Pausable: new pauser is the zero address"
                );
                pauser = _newPauser;
                emit PauserChanged(pauser);
            }
        }
        
        // File: contracts/v1/Blacklistable.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title Blacklistable Token
         * @dev Allows accounts to be blacklisted by a "blacklister" role
         */
        contract Blacklistable is Ownable {
            address public blacklister;
            mapping(address => bool) internal blacklisted;
        
            event Blacklisted(address indexed _account);
            event UnBlacklisted(address indexed _account);
            event BlacklisterChanged(address indexed newBlacklister);
        
            /**
             * @dev Throws if called by any account other than the blacklister
             */
            modifier onlyBlacklister() {
                require(
                    msg.sender == blacklister,
                    "Blacklistable: caller is not the blacklister"
                );
                _;
            }
        
            /**
             * @dev Throws if argument account is blacklisted
             * @param _account The address to check
             */
            modifier notBlacklisted(address _account) {
                require(
                    !blacklisted[_account],
                    "Blacklistable: account is blacklisted"
                );
                _;
            }
        
            /**
             * @dev Checks if account is blacklisted
             * @param _account The address to check
             */
            function isBlacklisted(address _account) external view returns (bool) {
                return blacklisted[_account];
            }
        
            /**
             * @dev Adds account to blacklist
             * @param _account The address to blacklist
             */
            function blacklist(address _account) external onlyBlacklister {
                blacklisted[_account] = true;
                emit Blacklisted(_account);
            }
        
            /**
             * @dev Removes account from blacklist
             * @param _account The address to remove from the blacklist
             */
            function unBlacklist(address _account) external onlyBlacklister {
                blacklisted[_account] = false;
                emit UnBlacklisted(_account);
            }
        
            function updateBlacklister(address _newBlacklister) external onlyOwner {
                require(
                    _newBlacklister != address(0),
                    "Blacklistable: new blacklister is the zero address"
                );
                blacklister = _newBlacklister;
                emit BlacklisterChanged(blacklister);
            }
        }
        
        // File: contracts/v1/FiatTokenV1.sol
        
        /**
         *
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title FiatToken
         * @dev ERC20 Token backed by fiat reserves
         */
        contract FiatTokenV1 is AbstractFiatTokenV1, Ownable, Pausable, Blacklistable {
            using SafeMath for uint256;
        
            string public name;
            string public symbol;
            uint8 public decimals;
            string public currency;
            address public masterMinter;
            bool internal initialized;
        
            mapping(address => uint256) internal balances;
            mapping(address => mapping(address => uint256)) internal allowed;
            uint256 internal totalSupply_ = 0;
            mapping(address => bool) internal minters;
            mapping(address => uint256) internal minterAllowed;
        
            event Mint(address indexed minter, address indexed to, uint256 amount);
            event Burn(address indexed burner, uint256 amount);
            event MinterConfigured(address indexed minter, uint256 minterAllowedAmount);
            event MinterRemoved(address indexed oldMinter);
            event MasterMinterChanged(address indexed newMasterMinter);
        
            function initialize(
                string memory tokenName,
                string memory tokenSymbol,
                string memory tokenCurrency,
                uint8 tokenDecimals,
                address newMasterMinter,
                address newPauser,
                address newBlacklister,
                address newOwner
            ) public {
                require(!initialized, "FiatToken: contract is already initialized");
                require(
                    newMasterMinter != address(0),
                    "FiatToken: new masterMinter is the zero address"
                );
                require(
                    newPauser != address(0),
                    "FiatToken: new pauser is the zero address"
                );
                require(
                    newBlacklister != address(0),
                    "FiatToken: new blacklister is the zero address"
                );
                require(
                    newOwner != address(0),
                    "FiatToken: new owner is the zero address"
                );
        
                name = tokenName;
                symbol = tokenSymbol;
                currency = tokenCurrency;
                decimals = tokenDecimals;
                masterMinter = newMasterMinter;
                pauser = newPauser;
                blacklister = newBlacklister;
                setOwner(newOwner);
                initialized = true;
            }
        
            /**
             * @dev Throws if called by any account other than a minter
             */
            modifier onlyMinters() {
                require(minters[msg.sender], "FiatToken: caller is not a minter");
                _;
            }
        
            /**
             * @dev Function to mint tokens
             * @param _to The address that will receive the minted tokens.
             * @param _amount The amount of tokens to mint. Must be less than or equal
             * to the minterAllowance of the caller.
             * @return A boolean that indicates if the operation was successful.
             */
            function mint(address _to, uint256 _amount)
                external
                whenNotPaused
                onlyMinters
                notBlacklisted(msg.sender)
                notBlacklisted(_to)
                returns (bool)
            {
                require(_to != address(0), "FiatToken: mint to the zero address");
                require(_amount > 0, "FiatToken: mint amount not greater than 0");
        
                uint256 mintingAllowedAmount = minterAllowed[msg.sender];
                require(
                    _amount <= mintingAllowedAmount,
                    "FiatToken: mint amount exceeds minterAllowance"
                );
        
                totalSupply_ = totalSupply_.add(_amount);
                balances[_to] = balances[_to].add(_amount);
                minterAllowed[msg.sender] = mintingAllowedAmount.sub(_amount);
                emit Mint(msg.sender, _to, _amount);
                emit Transfer(address(0), _to, _amount);
                return true;
            }
        
            /**
             * @dev Throws if called by any account other than the masterMinter
             */
            modifier onlyMasterMinter() {
                require(
                    msg.sender == masterMinter,
                    "FiatToken: caller is not the masterMinter"
                );
                _;
            }
        
            /**
             * @dev Get minter allowance for an account
             * @param minter The address of the minter
             */
            function minterAllowance(address minter) external view returns (uint256) {
                return minterAllowed[minter];
            }
        
            /**
             * @dev Checks if account is a minter
             * @param account The address to check
             */
            function isMinter(address account) external view returns (bool) {
                return minters[account];
            }
        
            /**
             * @notice Amount of remaining tokens spender is allowed to transfer on
             * behalf of the token owner
             * @param owner     Token owner's address
             * @param spender   Spender's address
             * @return Allowance amount
             */
            function allowance(address owner, address spender)
                external
                override
                view
                returns (uint256)
            {
                return allowed[owner][spender];
            }
        
            /**
             * @dev Get totalSupply of token
             */
            function totalSupply() external override view returns (uint256) {
                return totalSupply_;
            }
        
            /**
             * @dev Get token balance of an account
             * @param account address The account
             */
            function balanceOf(address account)
                external
                override
                view
                returns (uint256)
            {
                return balances[account];
            }
        
            /**
             * @notice Set spender's allowance over the caller's tokens to be a given
             * value.
             * @param spender   Spender's address
             * @param value     Allowance amount
             * @return True if successful
             */
            function approve(address spender, uint256 value)
                external
                override
                whenNotPaused
                notBlacklisted(msg.sender)
                notBlacklisted(spender)
                returns (bool)
            {
                _approve(msg.sender, spender, value);
                return true;
            }
        
            /**
             * @dev Internal function to set allowance
             * @param owner     Token owner's address
             * @param spender   Spender's address
             * @param value     Allowance amount
             */
            function _approve(
                address owner,
                address spender,
                uint256 value
            ) internal override {
                require(owner != address(0), "ERC20: approve from the zero address");
                require(spender != address(0), "ERC20: approve to the zero address");
                allowed[owner][spender] = value;
                emit Approval(owner, spender, value);
            }
        
            /**
             * @notice Transfer tokens by spending allowance
             * @param from  Payer's address
             * @param to    Payee's address
             * @param value Transfer amount
             * @return True if successful
             */
            function transferFrom(
                address from,
                address to,
                uint256 value
            )
                external
                override
                whenNotPaused
                notBlacklisted(msg.sender)
                notBlacklisted(from)
                notBlacklisted(to)
                returns (bool)
            {
                require(
                    value <= allowed[from][msg.sender],
                    "ERC20: transfer amount exceeds allowance"
                );
                _transfer(from, to, value);
                allowed[from][msg.sender] = allowed[from][msg.sender].sub(value);
                return true;
            }
        
            /**
             * @notice Transfer tokens from the caller
             * @param to    Payee's address
             * @param value Transfer amount
             * @return True if successful
             */
            function transfer(address to, uint256 value)
                external
                override
                whenNotPaused
                notBlacklisted(msg.sender)
                notBlacklisted(to)
                returns (bool)
            {
                _transfer(msg.sender, to, value);
                return true;
            }
        
            /**
             * @notice Internal function to process transfers
             * @param from  Payer's address
             * @param to    Payee's address
             * @param value Transfer amount
             */
            function _transfer(
                address from,
                address to,
                uint256 value
            ) internal override {
                require(from != address(0), "ERC20: transfer from the zero address");
                require(to != address(0), "ERC20: transfer to the zero address");
                require(
                    value <= balances[from],
                    "ERC20: transfer amount exceeds balance"
                );
        
                balances[from] = balances[from].sub(value);
                balances[to] = balances[to].add(value);
                emit Transfer(from, to, value);
            }
        
            /**
             * @dev Function to add/update a new minter
             * @param minter The address of the minter
             * @param minterAllowedAmount The minting amount allowed for the minter
             * @return True if the operation was successful.
             */
            function configureMinter(address minter, uint256 minterAllowedAmount)
                external
                whenNotPaused
                onlyMasterMinter
                returns (bool)
            {
                minters[minter] = true;
                minterAllowed[minter] = minterAllowedAmount;
                emit MinterConfigured(minter, minterAllowedAmount);
                return true;
            }
        
            /**
             * @dev Function to remove a minter
             * @param minter The address of the minter to remove
             * @return True if the operation was successful.
             */
            function removeMinter(address minter)
                external
                onlyMasterMinter
                returns (bool)
            {
                minters[minter] = false;
                minterAllowed[minter] = 0;
                emit MinterRemoved(minter);
                return true;
            }
        
            /**
             * @dev allows a minter to burn some of its own tokens
             * Validates that caller is a minter and that sender is not blacklisted
             * amount is less than or equal to the minter's account balance
             * @param _amount uint256 the amount of tokens to be burned
             */
            function burn(uint256 _amount)
                external
                whenNotPaused
                onlyMinters
                notBlacklisted(msg.sender)
            {
                uint256 balance = balances[msg.sender];
                require(_amount > 0, "FiatToken: burn amount not greater than 0");
                require(balance >= _amount, "FiatToken: burn amount exceeds balance");
        
                totalSupply_ = totalSupply_.sub(_amount);
                balances[msg.sender] = balance.sub(_amount);
                emit Burn(msg.sender, _amount);
                emit Transfer(msg.sender, address(0), _amount);
            }
        
            function updateMasterMinter(address _newMasterMinter) external onlyOwner {
                require(
                    _newMasterMinter != address(0),
                    "FiatToken: new masterMinter is the zero address"
                );
                masterMinter = _newMasterMinter;
                emit MasterMinterChanged(masterMinter);
            }
        }
        
        // File: @openzeppelin/contracts/utils/Address.sol
        
        pragma solidity ^0.6.2;
        
        /**
         * @dev Collection of functions related to the address type
         */
        library Address {
            /**
             * @dev Returns true if `account` is a contract.
             *
             * [IMPORTANT]
             * ====
             * It is unsafe to assume that an address for which this function returns
             * false is an externally-owned account (EOA) and not a contract.
             *
             * Among others, `isContract` will return false for the following
             * types of addresses:
             *
             *  - an externally-owned account
             *  - a contract in construction
             *  - an address where a contract will be created
             *  - an address where a contract lived, but was destroyed
             * ====
             */
            function isContract(address account) internal view returns (bool) {
                // According to EIP-1052, 0x0 is the value returned for not-yet created accounts
                // and 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470 is returned
                // for accounts without code, i.e. `keccak256('')`
                bytes32 codehash;
        
                    bytes32 accountHash
                 = 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    codehash := extcodehash(account)
                }
                return (codehash != accountHash && codehash != 0x0);
            }
        
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                require(
                    address(this).balance >= amount,
                    "Address: insufficient balance"
                );
        
                // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
                (bool success, ) = recipient.call{ value: amount }("");
                require(
                    success,
                    "Address: unable to send value, recipient may have reverted"
                );
            }
        
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain`call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason, it is bubbled up by this
             * function (like regular Solidity function calls).
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data)
                internal
                returns (bytes memory)
            {
                return functionCall(target, data, "Address: low-level call failed");
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
             * `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal returns (bytes memory) {
                return _functionCallWithValue(target, data, 0, errorMessage);
            }
        
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value
            ) internal returns (bytes memory) {
                return
                    functionCallWithValue(
                        target,
                        data,
                        value,
                        "Address: low-level call with value failed"
                    );
            }
        
            /**
             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
             * with `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value,
                string memory errorMessage
            ) internal returns (bytes memory) {
                require(
                    address(this).balance >= value,
                    "Address: insufficient balance for call"
                );
                return _functionCallWithValue(target, data, value, errorMessage);
            }
        
            function _functionCallWithValue(
                address target,
                bytes memory data,
                uint256 weiValue,
                string memory errorMessage
            ) private returns (bytes memory) {
                require(isContract(target), "Address: call to non-contract");
        
                // solhint-disable-next-line avoid-low-level-calls
                (bool success, bytes memory returndata) = target.call{
                    value: weiValue
                }(data);
                if (success) {
                    return returndata;
                } else {
                    // Look for revert reason and bubble it up if present
                    if (returndata.length > 0) {
                        // The easiest way to bubble the revert reason is using memory via assembly
        
                        // solhint-disable-next-line no-inline-assembly
                        assembly {
                            let returndata_size := mload(returndata)
                            revert(add(32, returndata), returndata_size)
                        }
                    } else {
                        revert(errorMessage);
                    }
                }
            }
        }
        
        // File: @openzeppelin/contracts/token/ERC20/SafeERC20.sol
        
        pragma solidity ^0.6.0;
        
        /**
         * @title SafeERC20
         * @dev Wrappers around ERC20 operations that throw on failure (when the token
         * contract returns false). Tokens that return no value (and instead revert or
         * throw on failure) are also supported, non-reverting calls are assumed to be
         * successful.
         * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
         * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
         */
        library SafeERC20 {
            using SafeMath for uint256;
            using Address for address;
        
            function safeTransfer(
                IERC20 token,
                address to,
                uint256 value
            ) internal {
                _callOptionalReturn(
                    token,
                    abi.encodeWithSelector(token.transfer.selector, to, value)
                );
            }
        
            function safeTransferFrom(
                IERC20 token,
                address from,
                address to,
                uint256 value
            ) internal {
                _callOptionalReturn(
                    token,
                    abi.encodeWithSelector(token.transferFrom.selector, from, to, value)
                );
            }
        
            /**
             * @dev Deprecated. This function has issues similar to the ones found in
             * {IERC20-approve}, and its usage is discouraged.
             *
             * Whenever possible, use {safeIncreaseAllowance} and
             * {safeDecreaseAllowance} instead.
             */
            function safeApprove(
                IERC20 token,
                address spender,
                uint256 value
            ) internal {
                // safeApprove should only be called when setting an initial allowance,
                // or when resetting it to zero. To increase and decrease it, use
                // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
                // solhint-disable-next-line max-line-length
                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 safeIncreaseAllowance(
                IERC20 token,
                address spender,
                uint256 value
            ) internal {
                uint256 newAllowance = token.allowance(address(this), spender).add(
                    value
                );
                _callOptionalReturn(
                    token,
                    abi.encodeWithSelector(
                        token.approve.selector,
                        spender,
                        newAllowance
                    )
                );
            }
        
            function safeDecreaseAllowance(
                IERC20 token,
                address spender,
                uint256 value
            ) internal {
                uint256 newAllowance = token.allowance(address(this), spender).sub(
                    value,
                    "SafeERC20: decreased allowance below zero"
                );
                _callOptionalReturn(
                    token,
                    abi.encodeWithSelector(
                        token.approve.selector,
                        spender,
                        newAllowance
                    )
                );
            }
        
            /**
             * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
             * on the return value: the return value is optional (but if data is returned, it must not be false).
             * @param token The token targeted by the call.
             * @param data The call data (encoded using abi.encode or one of its variants).
             */
            function _callOptionalReturn(IERC20 token, bytes memory data) private {
                // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
                // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
                // the target address contains contract code and also asserts for success in the low-level call.
        
                bytes memory returndata = address(token).functionCall(
                    data,
                    "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"
                    );
                }
            }
        }
        
        // File: contracts/v1.1/Rescuable.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        contract Rescuable is Ownable {
            using SafeERC20 for IERC20;
        
            address private _rescuer;
        
            event RescuerChanged(address indexed newRescuer);
        
            /**
             * @notice Returns current rescuer
             * @return Rescuer's address
             */
            function rescuer() external view returns (address) {
                return _rescuer;
            }
        
            /**
             * @notice Revert if called by any account other than the rescuer.
             */
            modifier onlyRescuer() {
                require(msg.sender == _rescuer, "Rescuable: caller is not the rescuer");
                _;
            }
        
            /**
             * @notice Rescue ERC20 tokens locked up in this contract.
             * @param tokenContract ERC20 token contract address
             * @param to        Recipient address
             * @param amount    Amount to withdraw
             */
            function rescueERC20(
                IERC20 tokenContract,
                address to,
                uint256 amount
            ) external onlyRescuer {
                tokenContract.safeTransfer(to, amount);
            }
        
            /**
             * @notice Assign the rescuer role to a given address.
             * @param newRescuer New rescuer's address
             */
            function updateRescuer(address newRescuer) external onlyOwner {
                require(
                    newRescuer != address(0),
                    "Rescuable: new rescuer is the zero address"
                );
                _rescuer = newRescuer;
                emit RescuerChanged(newRescuer);
            }
        }
        
        // File: contracts/v1.1/FiatTokenV1_1.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title FiatTokenV1_1
         * @dev ERC20 Token backed by fiat reserves
         */
        contract FiatTokenV1_1 is FiatTokenV1, Rescuable {
        
        }
        
        // File: contracts/v2/AbstractFiatTokenV2.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        abstract contract AbstractFiatTokenV2 is AbstractFiatTokenV1 {
            function _increaseAllowance(
                address owner,
                address spender,
                uint256 increment
            ) internal virtual;
        
            function _decreaseAllowance(
                address owner,
                address spender,
                uint256 decrement
            ) internal virtual;
        }
        
        // File: contracts/util/ECRecover.sol
        
        /**
         * Copyright (c) 2016-2019 zOS Global Limited
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title ECRecover
         * @notice A library that provides a safe ECDSA recovery function
         */
        library ECRecover {
            /**
             * @notice Recover signer's address from a signed message
             * @dev Adapted from: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/65e4ffde586ec89af3b7e9140bdc9235d1254853/contracts/cryptography/ECDSA.sol
             * Modifications: Accept v, r, and s as separate arguments
             * @param digest    Keccak-256 hash digest of the signed message
             * @param v         v of the signature
             * @param r         r of the signature
             * @param s         s of the signature
             * @return Signer address
             */
            function recover(
                bytes32 digest,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal pure returns (address) {
                // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
                // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
                // the valid range for s in (281): 0 < s < secp256k1n ÷ 2 + 1, and for v in (282): v ∈ {27, 28}. Most
                // signatures from current libraries generate a unique signature with an s-value in the lower half order.
                //
                // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
                // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
                // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
                // these malleable signatures as well.
                if (
                    uint256(s) >
                    0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0
                ) {
                    revert("ECRecover: invalid signature 's' value");
                }
        
                if (v != 27 && v != 28) {
                    revert("ECRecover: invalid signature 'v' value");
                }
        
                // If the signature is valid (and not malleable), return the signer address
                address signer = ecrecover(digest, v, r, s);
                require(signer != address(0), "ECRecover: invalid signature");
        
                return signer;
            }
        }
        
        // File: contracts/util/EIP712.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title EIP712
         * @notice A library that provides EIP712 helper functions
         */
        library EIP712 {
            /**
             * @notice Make EIP712 domain separator
             * @param name      Contract name
             * @param version   Contract version
             * @return Domain separator
             */
            function makeDomainSeparator(string memory name, string memory version)
                internal
                view
                returns (bytes32)
            {
                uint256 chainId;
                assembly {
                    chainId := chainid()
                }
                return
                    keccak256(
                        abi.encode(
                            // keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)")
                            0x8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f,
                            keccak256(bytes(name)),
                            keccak256(bytes(version)),
                            chainId,
                            address(this)
                        )
                    );
            }
        
            /**
             * @notice Recover signer's address from a EIP712 signature
             * @param domainSeparator   Domain separator
             * @param v                 v of the signature
             * @param r                 r of the signature
             * @param s                 s of the signature
             * @param typeHashAndData   Type hash concatenated with data
             * @return Signer's address
             */
            function recover(
                bytes32 domainSeparator,
                uint8 v,
                bytes32 r,
                bytes32 s,
                bytes memory typeHashAndData
            ) internal pure returns (address) {
                bytes32 digest = keccak256(
                    abi.encodePacked(
                        "\x19\x01",
                        domainSeparator,
                        keccak256(typeHashAndData)
                    )
                );
                return ECRecover.recover(digest, v, r, s);
            }
        }
        
        // File: contracts/v2/EIP712Domain.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title EIP712 Domain
         */
        contract EIP712Domain {
            /**
             * @dev EIP712 Domain Separator
             */
            bytes32 public DOMAIN_SEPARATOR;
        }
        
        // File: contracts/v2/EIP3009.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title EIP-3009
         * @notice Provide internal implementation for gas-abstracted transfers
         * @dev Contracts that inherit from this must wrap these with publicly
         * accessible functions, optionally adding modifiers where necessary
         */
        abstract contract EIP3009 is AbstractFiatTokenV2, EIP712Domain {
            // keccak256("TransferWithAuthorization(address from,address to,uint256 value,uint256 validAfter,uint256 validBefore,bytes32 nonce)")
            bytes32
                public constant TRANSFER_WITH_AUTHORIZATION_TYPEHASH = 0x7c7c6cdb67a18743f49ec6fa9b35f50d52ed05cbed4cc592e13b44501c1a2267;
        
            // keccak256("ReceiveWithAuthorization(address from,address to,uint256 value,uint256 validAfter,uint256 validBefore,bytes32 nonce)")
            bytes32
                public constant RECEIVE_WITH_AUTHORIZATION_TYPEHASH = 0xd099cc98ef71107a616c4f0f941f04c322d8e254fe26b3c6668db87aae413de8;
        
            // keccak256("CancelAuthorization(address authorizer,bytes32 nonce)")
            bytes32
                public constant CANCEL_AUTHORIZATION_TYPEHASH = 0x158b0a9edf7a828aad02f63cd515c68ef2f50ba807396f6d12842833a1597429;
        
            /**
             * @dev authorizer address => nonce => bool (true if nonce is used)
             */
            mapping(address => mapping(bytes32 => bool)) private _authorizationStates;
        
            event AuthorizationUsed(address indexed authorizer, bytes32 indexed nonce);
            event AuthorizationCanceled(
                address indexed authorizer,
                bytes32 indexed nonce
            );
        
            /**
             * @notice Returns the state of an authorization
             * @dev Nonces are randomly generated 32-byte data unique to the
             * authorizer's address
             * @param authorizer    Authorizer's address
             * @param nonce         Nonce of the authorization
             * @return True if the nonce is used
             */
            function authorizationState(address authorizer, bytes32 nonce)
                external
                view
                returns (bool)
            {
                return _authorizationStates[authorizer][nonce];
            }
        
            /**
             * @notice Execute a transfer with a signed authorization
             * @param from          Payer's address (Authorizer)
             * @param to            Payee's address
             * @param value         Amount to be transferred
             * @param validAfter    The time after which this is valid (unix time)
             * @param validBefore   The time before which this is valid (unix time)
             * @param nonce         Unique nonce
             * @param v             v of the signature
             * @param r             r of the signature
             * @param s             s of the signature
             */
            function _transferWithAuthorization(
                address from,
                address to,
                uint256 value,
                uint256 validAfter,
                uint256 validBefore,
                bytes32 nonce,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal {
                _requireValidAuthorization(from, nonce, validAfter, validBefore);
        
                bytes memory data = abi.encode(
                    TRANSFER_WITH_AUTHORIZATION_TYPEHASH,
                    from,
                    to,
                    value,
                    validAfter,
                    validBefore,
                    nonce
                );
                require(
                    EIP712.recover(DOMAIN_SEPARATOR, v, r, s, data) == from,
                    "FiatTokenV2: invalid signature"
                );
        
                _markAuthorizationAsUsed(from, nonce);
                _transfer(from, to, value);
            }
        
            /**
             * @notice Receive a transfer with a signed authorization from the payer
             * @dev This has an additional check to ensure that the payee's address
             * matches the caller of this function to prevent front-running attacks.
             * @param from          Payer's address (Authorizer)
             * @param to            Payee's address
             * @param value         Amount to be transferred
             * @param validAfter    The time after which this is valid (unix time)
             * @param validBefore   The time before which this is valid (unix time)
             * @param nonce         Unique nonce
             * @param v             v of the signature
             * @param r             r of the signature
             * @param s             s of the signature
             */
            function _receiveWithAuthorization(
                address from,
                address to,
                uint256 value,
                uint256 validAfter,
                uint256 validBefore,
                bytes32 nonce,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal {
                require(to == msg.sender, "FiatTokenV2: caller must be the payee");
                _requireValidAuthorization(from, nonce, validAfter, validBefore);
        
                bytes memory data = abi.encode(
                    RECEIVE_WITH_AUTHORIZATION_TYPEHASH,
                    from,
                    to,
                    value,
                    validAfter,
                    validBefore,
                    nonce
                );
                require(
                    EIP712.recover(DOMAIN_SEPARATOR, v, r, s, data) == from,
                    "FiatTokenV2: invalid signature"
                );
        
                _markAuthorizationAsUsed(from, nonce);
                _transfer(from, to, value);
            }
        
            /**
             * @notice Attempt to cancel an authorization
             * @param authorizer    Authorizer's address
             * @param nonce         Nonce of the authorization
             * @param v             v of the signature
             * @param r             r of the signature
             * @param s             s of the signature
             */
            function _cancelAuthorization(
                address authorizer,
                bytes32 nonce,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal {
                _requireUnusedAuthorization(authorizer, nonce);
        
                bytes memory data = abi.encode(
                    CANCEL_AUTHORIZATION_TYPEHASH,
                    authorizer,
                    nonce
                );
                require(
                    EIP712.recover(DOMAIN_SEPARATOR, v, r, s, data) == authorizer,
                    "FiatTokenV2: invalid signature"
                );
        
                _authorizationStates[authorizer][nonce] = true;
                emit AuthorizationCanceled(authorizer, nonce);
            }
        
            /**
             * @notice Check that an authorization is unused
             * @param authorizer    Authorizer's address
             * @param nonce         Nonce of the authorization
             */
            function _requireUnusedAuthorization(address authorizer, bytes32 nonce)
                private
                view
            {
                require(
                    !_authorizationStates[authorizer][nonce],
                    "FiatTokenV2: authorization is used or canceled"
                );
            }
        
            /**
             * @notice Check that authorization is valid
             * @param authorizer    Authorizer's address
             * @param nonce         Nonce of the authorization
             * @param validAfter    The time after which this is valid (unix time)
             * @param validBefore   The time before which this is valid (unix time)
             */
            function _requireValidAuthorization(
                address authorizer,
                bytes32 nonce,
                uint256 validAfter,
                uint256 validBefore
            ) private view {
                require(
                    now > validAfter,
                    "FiatTokenV2: authorization is not yet valid"
                );
                require(now < validBefore, "FiatTokenV2: authorization is expired");
                _requireUnusedAuthorization(authorizer, nonce);
            }
        
            /**
             * @notice Mark an authorization as used
             * @param authorizer    Authorizer's address
             * @param nonce         Nonce of the authorization
             */
            function _markAuthorizationAsUsed(address authorizer, bytes32 nonce)
                private
            {
                _authorizationStates[authorizer][nonce] = true;
                emit AuthorizationUsed(authorizer, nonce);
            }
        }
        
        // File: contracts/v2/EIP2612.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title EIP-2612
         * @notice Provide internal implementation for gas-abstracted approvals
         */
        abstract contract EIP2612 is AbstractFiatTokenV2, EIP712Domain {
            // keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)")
            bytes32
                public constant PERMIT_TYPEHASH = 0x6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9;
        
            mapping(address => uint256) private _permitNonces;
        
            /**
             * @notice Nonces for permit
             * @param owner Token owner's address (Authorizer)
             * @return Next nonce
             */
            function nonces(address owner) external view returns (uint256) {
                return _permitNonces[owner];
            }
        
            /**
             * @notice Verify a signed approval permit and execute if valid
             * @param owner     Token owner's address (Authorizer)
             * @param spender   Spender's address
             * @param value     Amount of allowance
             * @param deadline  The time at which this expires (unix time)
             * @param v         v of the signature
             * @param r         r of the signature
             * @param s         s of the signature
             */
            function _permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal {
                require(deadline >= now, "FiatTokenV2: permit is expired");
        
                bytes memory data = abi.encode(
                    PERMIT_TYPEHASH,
                    owner,
                    spender,
                    value,
                    _permitNonces[owner]++,
                    deadline
                );
                require(
                    EIP712.recover(DOMAIN_SEPARATOR, v, r, s, data) == owner,
                    "EIP2612: invalid signature"
                );
        
                _approve(owner, spender, value);
            }
        }
        
        // File: contracts/v2/FiatTokenV2.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        /**
         * @title FiatToken V2
         * @notice ERC20 Token backed by fiat reserves, version 2
         */
        contract FiatTokenV2 is FiatTokenV1_1, EIP3009, EIP2612 {
            uint8 internal _initializedVersion;
        
            /**
             * @notice Initialize v2
             * @param newName   New token name
             */
            function initializeV2(string calldata newName) external {
                // solhint-disable-next-line reason-string
                require(initialized && _initializedVersion == 0);
                name = newName;
                DOMAIN_SEPARATOR = EIP712.makeDomainSeparator(newName, "2");
                _initializedVersion = 1;
            }
        
            /**
             * @notice Increase the allowance by a given increment
             * @param spender   Spender's address
             * @param increment Amount of increase in allowance
             * @return True if successful
             */
            function increaseAllowance(address spender, uint256 increment)
                external
                whenNotPaused
                notBlacklisted(msg.sender)
                notBlacklisted(spender)
                returns (bool)
            {
                _increaseAllowance(msg.sender, spender, increment);
                return true;
            }
        
            /**
             * @notice Decrease the allowance by a given decrement
             * @param spender   Spender's address
             * @param decrement Amount of decrease in allowance
             * @return True if successful
             */
            function decreaseAllowance(address spender, uint256 decrement)
                external
                whenNotPaused
                notBlacklisted(msg.sender)
                notBlacklisted(spender)
                returns (bool)
            {
                _decreaseAllowance(msg.sender, spender, decrement);
                return true;
            }
        
            /**
             * @notice Execute a transfer with a signed authorization
             * @param from          Payer's address (Authorizer)
             * @param to            Payee's address
             * @param value         Amount to be transferred
             * @param validAfter    The time after which this is valid (unix time)
             * @param validBefore   The time before which this is valid (unix time)
             * @param nonce         Unique nonce
             * @param v             v of the signature
             * @param r             r of the signature
             * @param s             s of the signature
             */
            function transferWithAuthorization(
                address from,
                address to,
                uint256 value,
                uint256 validAfter,
                uint256 validBefore,
                bytes32 nonce,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external whenNotPaused notBlacklisted(from) notBlacklisted(to) {
                _transferWithAuthorization(
                    from,
                    to,
                    value,
                    validAfter,
                    validBefore,
                    nonce,
                    v,
                    r,
                    s
                );
            }
        
            /**
             * @notice Receive a transfer with a signed authorization from the payer
             * @dev This has an additional check to ensure that the payee's address
             * matches the caller of this function to prevent front-running attacks.
             * @param from          Payer's address (Authorizer)
             * @param to            Payee's address
             * @param value         Amount to be transferred
             * @param validAfter    The time after which this is valid (unix time)
             * @param validBefore   The time before which this is valid (unix time)
             * @param nonce         Unique nonce
             * @param v             v of the signature
             * @param r             r of the signature
             * @param s             s of the signature
             */
            function receiveWithAuthorization(
                address from,
                address to,
                uint256 value,
                uint256 validAfter,
                uint256 validBefore,
                bytes32 nonce,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external whenNotPaused notBlacklisted(from) notBlacklisted(to) {
                _receiveWithAuthorization(
                    from,
                    to,
                    value,
                    validAfter,
                    validBefore,
                    nonce,
                    v,
                    r,
                    s
                );
            }
        
            /**
             * @notice Attempt to cancel an authorization
             * @dev Works only if the authorization is not yet used.
             * @param authorizer    Authorizer's address
             * @param nonce         Nonce of the authorization
             * @param v             v of the signature
             * @param r             r of the signature
             * @param s             s of the signature
             */
            function cancelAuthorization(
                address authorizer,
                bytes32 nonce,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external whenNotPaused {
                _cancelAuthorization(authorizer, nonce, v, r, s);
            }
        
            /**
             * @notice Update allowance with a signed permit
             * @param owner       Token owner's address (Authorizer)
             * @param spender     Spender's address
             * @param value       Amount of allowance
             * @param deadline    Expiration time, seconds since the epoch
             * @param v           v of the signature
             * @param r           r of the signature
             * @param s           s of the signature
             */
            function permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external whenNotPaused notBlacklisted(owner) notBlacklisted(spender) {
                _permit(owner, spender, value, deadline, v, r, s);
            }
        
            /**
             * @notice Internal function to increase the allowance by a given increment
             * @param owner     Token owner's address
             * @param spender   Spender's address
             * @param increment Amount of increase
             */
            function _increaseAllowance(
                address owner,
                address spender,
                uint256 increment
            ) internal override {
                _approve(owner, spender, allowed[owner][spender].add(increment));
            }
        
            /**
             * @notice Internal function to decrease the allowance by a given decrement
             * @param owner     Token owner's address
             * @param spender   Spender's address
             * @param decrement Amount of decrease
             */
            function _decreaseAllowance(
                address owner,
                address spender,
                uint256 decrement
            ) internal override {
                _approve(
                    owner,
                    spender,
                    allowed[owner][spender].sub(
                        decrement,
                        "ERC20: decreased allowance below zero"
                    )
                );
            }
        }
        
        // File: contracts/v2/FiatTokenV2_1.sol
        
        /**
         * Copyright (c) 2018-2020 CENTRE SECZ
         *
         * Permission is hereby granted, free of charge, to any person obtaining a copy
         * of this software and associated documentation files (the "Software"), to deal
         * in the Software without restriction, including without limitation the rights
         * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         * copies of the Software, and to permit persons to whom the Software is
         * furnished to do so, subject to the following conditions:
         *
         * The above copyright notice and this permission notice shall be included in
         * copies or substantial portions of the Software.
         *
         * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         * SOFTWARE.
         */
        
        pragma solidity 0.6.12;
        
        // solhint-disable func-name-mixedcase
        
        /**
         * @title FiatToken V2.1
         * @notice ERC20 Token backed by fiat reserves, version 2.1
         */
        contract FiatTokenV2_1 is FiatTokenV2 {
            /**
             * @notice Initialize v2.1
             * @param lostAndFound  The address to which the locked funds are sent
             */
            function initializeV2_1(address lostAndFound) external {
                // solhint-disable-next-line reason-string
                require(_initializedVersion == 1);
        
                uint256 lockedAmount = balances[address(this)];
                if (lockedAmount > 0) {
                    _transfer(address(this), lostAndFound, lockedAmount);
                }
                blacklisted[address(this)] = true;
        
                _initializedVersion = 2;
            }
        
            /**
             * @notice Version string for the EIP712 domain separator
             * @return Version string
             */
            function version() external view returns (string memory) {
                return "2";
            }
        }