false
true
0

Contract Address Details

0x05F0d687A8fb3a42c6Be874Ce143554A44D5Cefb

Contract Name
LendingPool
Creator
0x64e7ff–1c04a3 at 0x5aa209–f58e38
Balance
0 PLS ( )
Tokens
Fetching tokens...
Transactions
1 Transactions
Transfers
0 Transfers
Gas Used
0
Last Balance Update
26325511
Warning! Contract bytecode has been changed and doesn't match the verified one. Therefore, interaction with this smart contract may be risky.
Contract name:
LendingPool




Optimization enabled
true
Compiler version
v0.7.6+commit.7338295f




Optimization runs
200
EVM Version
istanbul




Verified at
2023-07-16T13:24:40.834904Z

contracts/protocol/lendingpool/LendingPool.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;
pragma experimental ABIEncoderV2;

import {SafeMath} from "../../dependencies/openzeppelin/contracts/SafeMath.sol";
import {IERC20} from "../../dependencies/openzeppelin/contracts/IERC20.sol";
import {SafeERC20} from "../../dependencies/openzeppelin/contracts/SafeERC20.sol";
import {Address} from "../../dependencies/openzeppelin/contracts/Address.sol";
import {ILendingPoolAddressesProvider} from "../../interfaces/ILendingPoolAddressesProvider.sol";
import {IAToken} from "../../interfaces/IAToken.sol";
import {IVariableDebtToken} from "../../interfaces/IVariableDebtToken.sol";
import {IFlashLoanReceiver} from "../../flashloan/interfaces/IFlashLoanReceiver.sol";
import {IPriceOracleGetter} from "../../interfaces/IPriceOracleGetter.sol";
import {IStableDebtToken} from "../../interfaces/IStableDebtToken.sol";
import {ILendingPool} from "../../interfaces/ILendingPool.sol";
import {VersionedInitializable} from "../libraries/aave-upgradeability/VersionedInitializable.sol";
import {Helpers} from "../libraries/helpers/Helpers.sol";
import {Errors} from "../libraries/helpers/Errors.sol";
import {WadRayMath} from "../libraries/math/WadRayMath.sol";
import {PercentageMath} from "../libraries/math/PercentageMath.sol";
import {ReserveLogic} from "../libraries/logic/ReserveLogic.sol";
import {GenericLogic} from "../libraries/logic/GenericLogic.sol";
import {ValidationLogic} from "../libraries/logic/ValidationLogic.sol";
import {ReserveConfiguration} from "../libraries/configuration/ReserveConfiguration.sol";
import {UserConfiguration} from "../libraries/configuration/UserConfiguration.sol";
import {DataTypes} from "../libraries/types/DataTypes.sol";
import {LendingPoolStorage} from "./LendingPoolStorage.sol";

/**
 * @title LendingPool contract
 * @dev Main point of interaction with an Aave protocol's market
 * - Users can:
 *   # Deposit
 *   # Withdraw
 *   # Borrow
 *   # Repay
 *   # Swap their loans between variable and stable rate
 *   # Enable/disable their deposits as collateral rebalance stable rate borrow positions
 *   # Liquidate positions
 *   # Execute Flash Loans
 * - To be covered by a proxy contract, owned by the LendingPoolAddressesProvider of the specific market
 * - All admin functions are callable by the LendingPoolConfigurator contract defined also in the
 *   LendingPoolAddressesProvider
 * @author Aave
 **/
contract LendingPool is
    VersionedInitializable,
    ILendingPool,
    LendingPoolStorage
{
    using SafeMath for uint256;
    using WadRayMath for uint256;
    using PercentageMath for uint256;
    using SafeERC20 for IERC20;
    using ReserveLogic for DataTypes.ReserveData;
    using ReserveConfiguration for DataTypes.ReserveConfigurationMap;
    using UserConfiguration for DataTypes.UserConfigurationMap;

    uint256 public constant LENDINGPOOL_REVISION = 0x1;

    mapping(address => bool) public override isBlacklisted;

    modifier whenNotPaused() {
        _whenNotPaused();
        _;
    }

    modifier onlyLendingPoolConfigurator() {
        _onlyLendingPoolConfigurator();
        _;
    }

    function _whenNotPaused() internal view {
        require(!_paused, Errors.LP_IS_PAUSED);
    }

    function _onlyLendingPoolConfigurator() internal view {
        require(
            _addressesProvider.getLendingPoolConfigurator() == msg.sender,
            Errors.LP_CALLER_NOT_LENDING_POOL_CONFIGURATOR
        );
    }

    function _notBlacklisted(address user) internal view {
        require(!isBlacklisted[user], Errors.LP_USER_IS_BLACKLISTED);
    }

    function getRevision() internal pure override returns (uint256) {
        return LENDINGPOOL_REVISION;
    }

    /**
     * @dev Function is invoked by the proxy contract when the LendingPool contract is added to the
     * LendingPoolAddressesProvider of the market.
     * - Caching the address of the LendingPoolAddressesProvider in order to reduce gas consumption
     *   on subsequent operations
     * @param provider The address of the LendingPoolAddressesProvider
     **/
    function initialize(ILendingPoolAddressesProvider provider)
        public
        initializer
    {
        _addressesProvider = provider;
        _maxStableRateBorrowSizePercent = 2500;
        _flashLoanPremiumTotal = 9;
        _maxNumberOfReserves = 128;
    }

    /**
     * @dev Deposits an `amount` of underlying asset into the reserve, receiving in return overlying aTokens.
     * - E.g. User deposits 100 USDC and gets in return 100 aUSDC
     * @param asset The address of the underlying asset to deposit
     * @param amount The amount to be deposited
     * @param onBehalfOf The address that will receive the aTokens, same as msg.sender if the user
     *   wants to receive them on his own wallet, or a different address if the beneficiary of aTokens
     *   is a different wallet
     * @param referralCode Code used to register the integrator originating the operation, for potential rewards.
     *   0 if the action is executed directly by the user, without any middle-man
     **/
    function deposit(
        address asset,
        uint256 amount,
        address onBehalfOf,
        uint16 referralCode
    ) external override whenNotPaused {
        _notBlacklisted(onBehalfOf);

        DataTypes.ReserveData storage reserve = _reserves[asset];

        ValidationLogic.validateDeposit(
            reserve,
            _reserveLimits[asset],
            onBehalfOf,
            amount
        );

        address aToken = reserve.aTokenAddress;

        reserve.updateState();
        reserve.updateInterestRates(asset, aToken, amount, 0);

        IERC20(asset).safeTransferFrom(msg.sender, aToken, amount);

        bool isFirstDeposit = IAToken(aToken).mint(
            onBehalfOf,
            amount,
            reserve.liquidityIndex
        );

        if (isFirstDeposit) {
            _usersConfig[onBehalfOf].setUsingAsCollateral(reserve.id, true);
            emit ReserveUsedAsCollateralEnabled(asset, onBehalfOf);
        }

        emit Deposit(asset, msg.sender, onBehalfOf, amount, referralCode);
    }

    /**
     * @dev Withdraws an `amount` of underlying asset from the reserve, burning the equivalent aTokens owned
     * E.g. User has 100 aUSDC, calls withdraw() and receives 100 USDC, burning the 100 aUSDC
     * @param asset The address of the underlying asset to withdraw
     * @param amount The underlying amount to be withdrawn
     *   - Send the value type(uint256).max in order to withdraw the whole aToken balance
     * @param to Address that will receive the underlying, same as msg.sender if the user
     *   wants to receive it on his own wallet, or a different address if the beneficiary is a
     *   different wallet
     * @return The final amount withdrawn
     **/
    function withdraw(
        address asset,
        uint256 amount,
        address to
    ) external override whenNotPaused returns (uint256) {
        DataTypes.ReserveData storage reserve = _reserves[asset];

        address aToken = reserve.aTokenAddress;

        uint256 userBalance = IAToken(aToken).balanceOf(msg.sender);

        uint256 amountToWithdraw = amount;

        if (amount == type(uint256).max) {
            amountToWithdraw = userBalance;
        }

        ValidationLogic.validateWithdraw(
            asset,
            amountToWithdraw,
            userBalance,
            _reserves,
            _reserveLimits[asset],
            _usersConfig[msg.sender],
            _reservesList,
            _reservesCount,
            _addressesProvider.getPriceOracle()
        );

        reserve.updateState();

        reserve.updateInterestRates(asset, aToken, 0, amountToWithdraw);

        if (amountToWithdraw == userBalance) {
            _usersConfig[msg.sender].setUsingAsCollateral(reserve.id, false);
            emit ReserveUsedAsCollateralDisabled(asset, msg.sender);
        }

        IAToken(aToken).burn(
            msg.sender,
            to,
            amountToWithdraw,
            reserve.liquidityIndex
        );

        emit Withdraw(asset, msg.sender, to, amountToWithdraw);

        return amountToWithdraw;
    }

    /**
     * @dev Allows users to borrow a specific `amount` of the reserve underlying asset, provided that the borrower
     * already deposited enough collateral, or he was given enough allowance by a credit delegator on the
     * corresponding debt token (StableDebtToken or VariableDebtToken)
     * - E.g. User borrows 100 USDC passing as `onBehalfOf` his own address, receiving the 100 USDC in his wallet
     *   and 100 stable/variable debt tokens, depending on the `interestRateMode`
     * @param asset The address of the underlying asset to borrow
     * @param amount The amount to be borrowed
     * @param interestRateMode The interest rate mode at which the user wants to borrow: 1 for Stable, 2 for Variable
     * @param referralCode Code used to register the integrator originating the operation, for potential rewards.
     *   0 if the action is executed directly by the user, without any middle-man
     * @param onBehalfOf Address of the user who will receive the debt. Should be the address of the borrower itself
     * calling the function if he wants to borrow against his own collateral, or the address of the credit delegator
     * if he has been given credit delegation allowance
     **/
    function borrow(
        address asset,
        uint256 amount,
        uint256 interestRateMode,
        uint16 referralCode,
        address onBehalfOf
    ) external override whenNotPaused {
        DataTypes.ReserveData storage reserve = _reserves[asset];

        _executeBorrow(
            ExecuteBorrowParams(
                asset,
                msg.sender,
                onBehalfOf,
                amount,
                interestRateMode,
                reserve.aTokenAddress,
                referralCode,
                true
            )
        );
    }

    /**
     * @notice Repays a borrowed `amount` on a specific reserve, burning the equivalent debt tokens owned
     * - E.g. User repays 100 USDC, burning 100 variable/stable debt tokens of the `onBehalfOf` address
     * @param asset The address of the borrowed underlying asset previously borrowed
     * @param amount The amount to repay
     * - Send the value type(uint256).max in order to repay the whole debt for `asset` on the specific `debtMode`
     * @param rateMode The interest rate mode at of the debt the user wants to repay: 1 for Stable, 2 for Variable
     * @param onBehalfOf Address of the user who will get his debt reduced/removed. Should be the address of the
     * user calling the function if he wants to reduce/remove his own debt, or the address of any other
     * other borrower whose debt should be removed
     * @return The final amount repaid
     **/
    function repay(
        address asset,
        uint256 amount,
        uint256 rateMode,
        address onBehalfOf
    ) public override whenNotPaused returns (uint256) {
        DataTypes.ReserveData storage reserve = _reserves[asset];

        (uint256 stableDebt, uint256 variableDebt) = Helpers.getUserCurrentDebt(
            onBehalfOf,
            reserve
        );

        DataTypes.InterestRateMode interestRateMode = DataTypes
            .InterestRateMode(rateMode);

        ValidationLogic.validateRepay(
            reserve,
            amount,
            interestRateMode,
            onBehalfOf,
            stableDebt,
            variableDebt
        );

        uint256 paybackAmount = interestRateMode ==
            DataTypes.InterestRateMode.STABLE
            ? stableDebt
            : variableDebt;

        if (amount < paybackAmount) {
            paybackAmount = amount;
        }

        reserve.updateState();

        if (interestRateMode == DataTypes.InterestRateMode.STABLE) {
            IStableDebtToken(reserve.stableDebtTokenAddress).burn(
                onBehalfOf,
                paybackAmount
            );
        } else {
            IVariableDebtToken(reserve.variableDebtTokenAddress).burn(
                onBehalfOf,
                paybackAmount,
                reserve.variableBorrowIndex
            );
        }

        address aToken = reserve.aTokenAddress;
        reserve.updateInterestRates(asset, aToken, paybackAmount, 0);

        if (stableDebt.add(variableDebt).sub(paybackAmount) == 0) {
            _usersConfig[onBehalfOf].setBorrowing(reserve.id, false);
        }

        IERC20(asset).safeTransferFrom(msg.sender, aToken, paybackAmount);

        IAToken(aToken).handleRepayment(msg.sender, paybackAmount);

        emit Repay(asset, onBehalfOf, msg.sender, paybackAmount);

        return paybackAmount;
    }

    /**
     * @notice Repay all debts of the user and blacklist that user.
     * @param user Address of the user who will get his debt removed. Should be the address of the
     * user calling the function if he wants to reduce/remove his own debt, or the address of any other
     * other borrower whose debt should be removed
     **/
    function repayAllAndBlacklist(address user)
        external
        override
        whenNotPaused
    {
        // verify that the user has a health factor of zero
        (
            ,
            ,
            ,
            ,
            // totalCollateralETH
            // totalDebtETH
            // ltv
            // currentLiquidationThreshold
            uint256 healthFactor
        ) = GenericLogic.calculateUserAccountData(
                user,
                _reserves,
                _usersConfig[user],
                _reservesList,
                _reservesCount,
                _addressesProvider.getPriceOracle()
            );
        require(healthFactor == 0, Errors.LP_USER_NOT_ELIGIBLE_FOR_BLACKLIST);

        for (uint256 i = 0; i < _reservesCount; i++) {
            address asset = _reservesList[i];
            (uint256 stableDebt, uint256 variableDebt) = Helpers
                .getUserCurrentDebt(user, _reserves[asset]);
            if (stableDebt > 0) {
                repay(asset, stableDebt, 1, user);
            }
            if (variableDebt > 0) {
                repay(asset, variableDebt, 2, user);
            }
        }

        isBlacklisted[user] = true;
        emit Blacklist(user);
    }

    /**
     * @dev Allows a borrower to swap his debt between stable and variable mode, or viceversa
     * @param asset The address of the underlying asset borrowed
     * @param rateMode The rate mode that the user wants to swap to
     **/
    function swapBorrowRateMode(address asset, uint256 rateMode)
        external
        override
        whenNotPaused
    {
        DataTypes.ReserveData storage reserve = _reserves[asset];

        (uint256 stableDebt, uint256 variableDebt) = Helpers.getUserCurrentDebt(
            msg.sender,
            reserve
        );

        DataTypes.InterestRateMode interestRateMode = DataTypes
            .InterestRateMode(rateMode);

        ValidationLogic.validateSwapRateMode(
            reserve,
            _usersConfig[msg.sender],
            stableDebt,
            variableDebt,
            interestRateMode
        );

        reserve.updateState();

        if (interestRateMode == DataTypes.InterestRateMode.STABLE) {
            IStableDebtToken(reserve.stableDebtTokenAddress).burn(
                msg.sender,
                stableDebt
            );
            IVariableDebtToken(reserve.variableDebtTokenAddress).mint(
                msg.sender,
                msg.sender,
                stableDebt,
                reserve.variableBorrowIndex
            );
        } else {
            IVariableDebtToken(reserve.variableDebtTokenAddress).burn(
                msg.sender,
                variableDebt,
                reserve.variableBorrowIndex
            );
            IStableDebtToken(reserve.stableDebtTokenAddress).mint(
                msg.sender,
                msg.sender,
                variableDebt,
                reserve.currentStableBorrowRate
            );
        }

        reserve.updateInterestRates(asset, reserve.aTokenAddress, 0, 0);

        emit Swap(asset, msg.sender, rateMode);
    }

    /**
     * @dev Rebalances the stable interest rate of a user to the current stable rate defined on the reserve.
     * - Users can be rebalanced if the following conditions are satisfied:
     *     1. Usage ratio is above 95%
     *     2. the current deposit APY is below REBALANCE_UP_THRESHOLD * maxVariableBorrowRate, which means that too much has been
     *        borrowed at a stable rate and depositors are not earning enough
     * @param asset The address of the underlying asset borrowed
     * @param user The address of the user to be rebalanced
     **/
    function rebalanceStableBorrowRate(address asset, address user)
        external
        override
        whenNotPaused
    {
        DataTypes.ReserveData storage reserve = _reserves[asset];

        IERC20 stableDebtToken = IERC20(reserve.stableDebtTokenAddress);
        IERC20 variableDebtToken = IERC20(reserve.variableDebtTokenAddress);
        address aTokenAddress = reserve.aTokenAddress;

        uint256 stableDebt = IERC20(stableDebtToken).balanceOf(user);

        ValidationLogic.validateRebalanceStableBorrowRate(
            reserve,
            asset,
            stableDebtToken,
            variableDebtToken,
            aTokenAddress
        );

        reserve.updateState();

        IStableDebtToken(address(stableDebtToken)).burn(user, stableDebt);
        IStableDebtToken(address(stableDebtToken)).mint(
            user,
            user,
            stableDebt,
            reserve.currentStableBorrowRate
        );

        reserve.updateInterestRates(asset, aTokenAddress, 0, 0);

        emit RebalanceStableBorrowRate(asset, user);
    }

    /**
     * @dev Allows depositors to enable/disable a specific deposited asset as collateral
     * @param asset The address of the underlying asset deposited
     * @param useAsCollateral `true` if the user wants to use the deposit as collateral, `false` otherwise
     **/
    function setUserUseReserveAsCollateral(address asset, bool useAsCollateral)
        external
        override
        whenNotPaused
    {
        _notBlacklisted(msg.sender); // can not change collateral setting after getting blacklisted (so no more borrowing)

        DataTypes.ReserveData storage reserve = _reserves[asset];

        ValidationLogic.validateSetUseReserveAsCollateral(
            reserve,
            asset,
            useAsCollateral,
            _reserves,
            _usersConfig[msg.sender],
            _reservesList,
            _reservesCount,
            _addressesProvider.getPriceOracle()
        );

        _usersConfig[msg.sender].setUsingAsCollateral(
            reserve.id,
            useAsCollateral
        );

        if (useAsCollateral) {
            emit ReserveUsedAsCollateralEnabled(asset, msg.sender);
        } else {
            emit ReserveUsedAsCollateralDisabled(asset, msg.sender);
        }
    }

    /**
     * @dev Function to liquidate a non-healthy position collateral-wise, with Health Factor below 1
     * - The caller (liquidator) covers `debtToCover` amount of debt of the user getting liquidated, and receives
     *   a proportionally amount of the `collateralAsset` plus a bonus to cover market risk
     * @param collateralAsset The address of the underlying asset used as collateral, to receive as result of the liquidation
     * @param debtAsset The address of the underlying borrowed asset to be repaid with the liquidation
     * @param user The address of the borrower getting liquidated
     * @param debtToCover The debt amount of borrowed `asset` the liquidator wants to cover
     * @param receiveAToken `true` if the liquidators wants to receive the collateral aTokens, `false` if he wants
     * to receive the underlying collateral asset directly
     **/
    function liquidationCall(
        address collateralAsset,
        address debtAsset,
        address user,
        uint256 debtToCover,
        bool receiveAToken
    ) external override whenNotPaused {
        address collateralManager = _addressesProvider
            .getLendingPoolCollateralManager();

        //solium-disable-next-line
        (bool success, bytes memory result) = collateralManager.delegatecall(
            abi.encodeWithSignature(
                "liquidationCall(address,address,address,uint256,bool)",
                collateralAsset,
                debtAsset,
                user,
                debtToCover,
                receiveAToken
            )
        );

        require(success, Errors.LP_LIQUIDATION_CALL_FAILED);

        (uint256 returnCode, string memory returnMessage) = abi.decode(
            result,
            (uint256, string)
        );

        require(returnCode == 0, string(abi.encodePacked(returnMessage)));
    }

    struct FlashLoanLocalVars {
        IFlashLoanReceiver receiver;
        address oracle;
        uint256 i;
        address currentAsset;
        address currentATokenAddress;
        uint256 currentAmount;
        uint256 currentPremium;
        uint256 currentAmountPlusPremium;
        address debtToken;
    }

    /**
     * @dev Allows smartcontracts to access the liquidity of the pool within one transaction,
     * as long as the amount taken plus a fee is returned.
     * IMPORTANT There are security concerns for developers of flashloan receiver contracts that must be kept into consideration.
     * For further details please visit https://developers.aave.com
     * @param receiverAddress The address of the contract receiving the funds, implementing the IFlashLoanReceiver interface
     * @param assets The addresses of the assets being flash-borrowed
     * @param amounts The amounts amounts being flash-borrowed
     * @param params Variadic packed params to pass to the receiver as extra information
     * @param referralCode Code used to register the integrator originating the operation, for potential rewards.
     *   0 if the action is executed directly by the user, without any middle-man
     **/
    function flashLoan(
        address receiverAddress,
        address[] calldata assets,
        uint256[] calldata amounts,
        bytes calldata params,
        uint16 referralCode
    ) external override whenNotPaused {
        FlashLoanLocalVars memory vars;
        ValidationLogic.validateFlashloan(assets, amounts);
        address[] memory aTokenAddresses = new address[](assets.length);
        uint256[] memory premiums = new uint256[](assets.length);
        vars.receiver = IFlashLoanReceiver(receiverAddress);
        for (vars.i = 0; vars.i < assets.length; vars.i++) {
            aTokenAddresses[vars.i] = _reserves[assets[vars.i]].aTokenAddress;
            premiums[vars.i] = amounts[vars.i].mul(_flashLoanPremiumTotal).div(
                10000
            );
            IAToken(aTokenAddresses[vars.i]).transferUnderlyingTo(
                receiverAddress,
                amounts[vars.i]
            );
        }
        require(
            vars.receiver.executeOperation(
                assets,
                amounts,
                premiums,
                msg.sender,
                params
            ),
            Errors.LP_INVALID_FLASH_LOAN_EXECUTOR_RETURN
        );
        for (vars.i = 0; vars.i < assets.length; vars.i++) {
            vars.currentAsset = assets[vars.i];
            vars.currentAmount = amounts[vars.i];
            vars.currentPremium = premiums[vars.i];
            vars.currentATokenAddress = aTokenAddresses[vars.i];
            vars.currentAmountPlusPremium = vars.currentAmount.add(
                vars.currentPremium
            );

            _reserves[vars.currentAsset].updateState();
            _reserves[vars.currentAsset].cumulateToLiquidityIndex(
                IERC20(vars.currentATokenAddress).totalSupply(),
                vars.currentPremium
            );
            _reserves[vars.currentAsset].updateInterestRates(
                vars.currentAsset,
                vars.currentATokenAddress,
                vars.currentAmountPlusPremium,
                0
            );
            IERC20(vars.currentAsset).safeTransferFrom(
                receiverAddress,
                vars.currentATokenAddress,
                vars.currentAmountPlusPremium
            );

            emit FlashLoan(
                receiverAddress,
                msg.sender,
                vars.currentAsset,
                vars.currentAmount,
                vars.currentPremium,
                referralCode
            );
        }
    }

    /**
     * @dev Returns the state and configuration of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The state of the reserve
     **/
    function getReserveData(address asset)
        external
        view
        override
        returns (DataTypes.ReserveData memory)
    {
        return _reserves[asset];
    }

    /**
     * @dev Returns the limits of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The limits of the reserve
     **/
    function getReserveLimits(address asset)
        external
        view
        override
        returns (DataTypes.ReserveLimits memory)
    {
        return _reserveLimits[asset];
    }

    /**
     * @dev Returns the user account data across all the reserves
     * @param user The address of the user
     * @return totalCollateralETH the total collateral in ETH of the user
     * @return totalDebtETH the total debt in ETH of the user
     * @return availableBorrowsETH the borrowing power left of the user
     * @return currentLiquidationThreshold the liquidation threshold of the user
     * @return ltv the loan to value of the user
     * @return healthFactor the current health factor of the user
     **/
    function getUserAccountData(address user)
        external
        view
        override
        returns (
            uint256 totalCollateralETH,
            uint256 totalDebtETH,
            uint256 availableBorrowsETH,
            uint256 currentLiquidationThreshold,
            uint256 ltv,
            uint256 healthFactor
        )
    {
        (
            totalCollateralETH,
            totalDebtETH,
            ltv,
            currentLiquidationThreshold,
            healthFactor
        ) = GenericLogic.calculateUserAccountData(
            user,
            _reserves,
            _usersConfig[user],
            _reservesList,
            _reservesCount,
            _addressesProvider.getPriceOracle()
        );

        availableBorrowsETH = GenericLogic.calculateAvailableBorrowsETH(
            totalCollateralETH,
            totalDebtETH,
            ltv
        );
    }

    /**
     * @dev Returns the configuration of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The configuration of the reserve
     **/
    function getConfiguration(address asset)
        external
        view
        override
        returns (DataTypes.ReserveConfigurationMap memory)
    {
        return _reserves[asset].configuration;
    }

    /**
     * @dev Returns the configuration of the user across all the reserves
     * @param user The user address
     * @return The configuration of the user
     **/
    function getUserConfiguration(address user)
        external
        view
        override
        returns (DataTypes.UserConfigurationMap memory)
    {
        return _usersConfig[user];
    }

    /**
     * @dev Returns the normalized income per unit of asset
     * @param asset The address of the underlying asset of the reserve
     * @return The reserve's normalized income
     */
    function getReserveNormalizedIncome(address asset)
        external
        view
        virtual
        override
        returns (uint256)
    {
        return _reserves[asset].getNormalizedIncome();
    }

    /**
     * @dev Returns the normalized variable debt per unit of asset
     * @param asset The address of the underlying asset of the reserve
     * @return The reserve normalized variable debt
     */
    function getReserveNormalizedVariableDebt(address asset)
        external
        view
        override
        returns (uint256)
    {
        return _reserves[asset].getNormalizedDebt();
    }

    /**
     * @dev Returns if the LendingPool is paused
     */
    function paused() external view override returns (bool) {
        return _paused;
    }

    /**
     * @dev Returns the list of the initialized reserves
     **/
    function getReservesList()
        external
        view
        override
        returns (address[] memory)
    {
        address[] memory _activeReserves = new address[](_reservesCount);

        for (uint256 i = 0; i < _reservesCount; i++) {
            _activeReserves[i] = _reservesList[i];
        }
        return _activeReserves;
    }

    /**
     * @dev Returns the cached LendingPoolAddressesProvider connected to this contract
     **/
    function getAddressesProvider()
        external
        view
        override
        returns (ILendingPoolAddressesProvider)
    {
        return _addressesProvider;
    }

    /**
     * @dev Returns the percentage of available liquidity that can be borrowed at once at stable rate
     */
    function MAX_STABLE_RATE_BORROW_SIZE_PERCENT()
        public
        view
        returns (uint256)
    {
        return _maxStableRateBorrowSizePercent;
    }

    /**
     * @dev Returns the fee on flash loans
     */
    function FLASHLOAN_PREMIUM_TOTAL() public view override returns (uint256) {
        return _flashLoanPremiumTotal;
    }

    /**
     * @dev Returns the maximum number of reserves supported to be listed in this LendingPool
     */
    function MAX_NUMBER_RESERVES() public view returns (uint256) {
        return _maxNumberOfReserves;
    }

    /**
     * @dev Validates and finalizes an aToken transfer
     * - Only callable by the overlying aToken of the `asset`
     * @param asset The address of the underlying asset of the aToken
     * @param from The user from which the aTokens are transferred
     * @param to The user receiving the aTokens
     * @param amount The amount being transferred/withdrawn
     * @param balanceFromBefore The aToken balance of the `from` user before the transfer
     * @param balanceToBefore The aToken balance of the `to` user before the transfer
     */
    function finalizeTransfer(
        address asset,
        address from,
        address to,
        uint256 amount,
        uint256 balanceFromBefore,
        uint256 balanceToBefore
    ) external override whenNotPaused {
        require(
            msg.sender == _reserves[asset].aTokenAddress,
            Errors.LP_CALLER_MUST_BE_AN_ATOKEN
        );

        address oracle = _addressesProvider.getPriceOracle();
        ValidationLogic.validateTransferLimits(
            from,
            to,
            _reserves[asset],
            _reserveLimits[asset]
        );
        ValidationLogic.validateTransfer(
            from,
            _reserves,
            _usersConfig[from],
            _reservesList,
            _reservesCount,
            oracle
        );

        uint256 reserveId = _reserves[asset].id;

        if (from != to) {
            if (balanceFromBefore.sub(amount) == 0) {
                DataTypes.UserConfigurationMap
                    storage fromConfig = _usersConfig[from];
                fromConfig.setUsingAsCollateral(reserveId, false);
                emit ReserveUsedAsCollateralDisabled(asset, from);
            }

            if (balanceToBefore == 0 && amount != 0) {
                DataTypes.UserConfigurationMap storage toConfig = _usersConfig[
                    to
                ];
                toConfig.setUsingAsCollateral(reserveId, true);
                emit ReserveUsedAsCollateralEnabled(asset, to);
            }
        }
    }

    /**
     * @dev Initializes a reserve, activating it, assigning an aToken and debt tokens and an
     * interest rate strategy
     * - Only callable by the LendingPoolConfigurator contract
     * @param asset The address of the underlying asset of the reserve
     * @param aTokenAddress The address of the aToken that will be assigned to the reserve
     * @param stableDebtAddress The address of the StableDebtToken that will be assigned to the reserve
     * @param variableDebtAddress The address of the VariableDebtToken that will be assigned to the reserve
     * @param interestRateStrategyAddress The address of the interest rate strategy contract
     **/
    function initReserve(
        address asset,
        address aTokenAddress,
        address stableDebtAddress,
        address variableDebtAddress,
        address interestRateStrategyAddress
    ) external override onlyLendingPoolConfigurator {
        require(Address.isContract(asset), Errors.LP_NOT_CONTRACT);
        _reserves[asset].init(
            aTokenAddress,
            stableDebtAddress,
            variableDebtAddress,
            interestRateStrategyAddress
        );
        _addReserveToList(asset);
    }

    /**
     * @dev Updates the address of the interest rate strategy contract
     * - Only callable by the LendingPoolConfigurator contract
     * @param asset The address of the underlying asset of the reserve
     * @param rateStrategyAddress The address of the interest rate strategy contract
     **/
    function setReserveInterestRateStrategyAddress(
        address asset,
        address rateStrategyAddress
    ) external override onlyLendingPoolConfigurator {
        _reserves[asset].interestRateStrategyAddress = rateStrategyAddress;
    }

    /**
     * @dev Sets the configuration bitmap of the reserve as a whole
     * - Only callable by the LendingPoolConfigurator contract
     * @param asset The address of the underlying asset of the reserve
     * @param configuration The new configuration bitmap
     **/
    function setConfiguration(address asset, uint256 configuration)
        external
        override
        onlyLendingPoolConfigurator
    {
        _reserves[asset].configuration.data = configuration;
    }

    /**
     * @dev Sets the limits of the reserve as a whole
     * - Only callable by the LendingPoolConfigurator contract
     * @param asset The address of the underlying asset of the reserve
     * @param reserveLimits The new reserve limits
     **/
    function setLimits(
        address asset,
        DataTypes.ReserveLimits calldata reserveLimits
    ) external override onlyLendingPoolConfigurator {
        _reserveLimits[asset] = reserveLimits;
    }

    /**
     * @dev Set the _pause state of a reserve
     * - Only callable by the LendingPoolConfigurator contract
     * @param val `true` to pause the reserve, `false` to un-pause it
     */
    function setPause(bool val) external override onlyLendingPoolConfigurator {
        _paused = val;
        if (_paused) {
            emit Paused();
        } else {
            emit Unpaused();
        }
    }

    struct ExecuteBorrowParams {
        address asset;
        address user;
        address onBehalfOf;
        uint256 amount;
        uint256 interestRateMode;
        address aTokenAddress;
        uint16 referralCode;
        bool releaseUnderlying;
    }

    function _executeBorrow(ExecuteBorrowParams memory vars) internal {
        DataTypes.ReserveData storage reserve = _reserves[vars.asset];
        DataTypes.UserConfigurationMap storage userConfig = _usersConfig[
            vars.onBehalfOf
        ];

        address oracle = _addressesProvider.getPriceOracle();
        uint256 price = IPriceOracleGetter(oracle).getAssetPrice(vars.asset);
        uint256 amountInETH = price.mul(vars.amount).div(
            10**reserve.configuration.getDecimals()
        );

        ValidationLogic.validateBorrowLimits(
            reserve,
            _reserveLimits[vars.asset],
            vars.onBehalfOf,
            vars.amount
        );
        ValidationLogic.validateBorrow(
            vars.asset,
            reserve,
            vars.onBehalfOf,
            vars.amount,
            amountInETH,
            vars.interestRateMode,
            _maxStableRateBorrowSizePercent,
            _reserves,
            userConfig,
            _reservesList,
            _reservesCount,
            oracle
        );

        reserve.updateState();

        uint256 currentStableRate = 0;

        bool isFirstBorrowing = false;
        if (
            DataTypes.InterestRateMode(vars.interestRateMode) ==
            DataTypes.InterestRateMode.STABLE
        ) {
            currentStableRate = reserve.currentStableBorrowRate;

            isFirstBorrowing = IStableDebtToken(reserve.stableDebtTokenAddress)
                .mint(
                    vars.user,
                    vars.onBehalfOf,
                    vars.amount,
                    currentStableRate
                );
        } else {
            isFirstBorrowing = IVariableDebtToken(
                reserve.variableDebtTokenAddress
            ).mint(
                    vars.user,
                    vars.onBehalfOf,
                    vars.amount,
                    reserve.variableBorrowIndex
                );
        }

        if (isFirstBorrowing) {
            userConfig.setBorrowing(reserve.id, true);
        }

        reserve.updateInterestRates(
            vars.asset,
            vars.aTokenAddress,
            0,
            vars.releaseUnderlying ? vars.amount : 0
        );

        if (vars.releaseUnderlying) {
            IAToken(vars.aTokenAddress).transferUnderlyingTo(
                vars.user,
                vars.amount
            );
        }

        emit Borrow(
            vars.asset,
            vars.user,
            vars.onBehalfOf,
            vars.amount,
            vars.interestRateMode,
            DataTypes.InterestRateMode(vars.interestRateMode) ==
                DataTypes.InterestRateMode.STABLE
                ? currentStableRate
                : reserve.currentVariableBorrowRate,
            vars.referralCode
        );
    }

    function _addReserveToList(address asset) internal {
        uint256 reservesCount = _reservesCount;

        require(
            reservesCount < _maxNumberOfReserves,
            Errors.LP_NO_MORE_RESERVES_ALLOWED
        );

        bool reserveAlreadyAdded = _reserves[asset].id != 0 ||
            _reservesList[0] == asset;

        if (!reserveAlreadyAdded) {
            _reserves[asset].id = uint8(reservesCount);
            _reservesList[reservesCount] = asset;

            _reservesCount = reservesCount + 1;
        }
    }
}
        

contracts/dependencies/openzeppelin/contracts/Address.sol

// SPDX-License-Identifier: MIT

pragma solidity >=0.6.2 <0.8.0;

/**
 * @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 on 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"
        );
        require(isContract(target), "Address: call to non-contract");

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.call{value: value}(
            data
        );
        return _verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(address target, bytes memory data)
        internal
        view
        returns (bytes memory)
    {
        return
            functionStaticCall(
                target,
                data,
                "Address: low-level static call failed"
            );
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal view returns (bytes memory) {
        require(isContract(target), "Address: static call to non-contract");

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.staticcall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(address target, bytes memory data)
        internal
        returns (bytes memory)
    {
        return
            functionDelegateCall(
                target,
                data,
                "Address: low-level delegate call failed"
            );
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal returns (bytes memory) {
        require(isContract(target), "Address: delegate call to non-contract");

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.delegatecall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    function _verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) private pure returns (bytes memory) {
        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);
            }
        }
    }
}
          

contracts/dependencies/openzeppelin/contracts/IERC20.sol

// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.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
    );
}
          

contracts/dependencies/openzeppelin/contracts/SafeERC20.sol

// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

import "./IERC20.sol";
import "./SafeMath.sol";
import "./Address.sol";

/**
 * @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"
            );
        }
    }
}
          

contracts/dependencies/openzeppelin/contracts/SafeMath.sol

// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.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, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryAdd(uint256 a, uint256 b)
        internal
        pure
        returns (bool, uint256)
    {
        uint256 c = a + b;
        if (c < a) return (false, 0);
        return (true, c);
    }

    /**
     * @dev Returns the substraction of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function trySub(uint256 a, uint256 b)
        internal
        pure
        returns (bool, uint256)
    {
        if (b > a) return (false, 0);
        return (true, a - b);
    }

    /**
     * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryMul(uint256 a, uint256 b)
        internal
        pure
        returns (bool, 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 (true, 0);
        uint256 c = a * b;
        if (c / a != b) return (false, 0);
        return (true, c);
    }

    /**
     * @dev Returns the division of two unsigned integers, with a division by zero flag.
     *
     * _Available since v3.4._
     */
    function tryDiv(uint256 a, uint256 b)
        internal
        pure
        returns (bool, uint256)
    {
        if (b == 0) return (false, 0);
        return (true, a / b);
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
     *
     * _Available since v3.4._
     */
    function tryMod(uint256 a, uint256 b)
        internal
        pure
        returns (bool, uint256)
    {
        if (b == 0) return (false, 0);
        return (true, a % b);
    }

    /**
     * @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) {
        require(b <= a, "SafeMath: subtraction overflow");
        return a - b;
    }

    /**
     * @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) {
        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, reverting 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) {
        require(b > 0, "SafeMath: division by zero");
        return a / b;
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * reverting 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) {
        require(b > 0, "SafeMath: modulo by zero");
        return a % b;
    }

    /**
     * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
     * overflow (when the result is negative).
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {trySub}.
     *
     * 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);
        return a - b;
    }

    /**
     * @dev Returns the integer division of two unsigned integers, reverting with custom message on
     * division by zero. The result is rounded towards zero.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {tryDiv}.
     *
     * 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);
        return a / b;
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * reverting with custom message when dividing by zero.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {tryMod}.
     *
     * 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;
    }
}
          

contracts/flashloan/interfaces/IFlashLoanReceiver.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {ILendingPoolAddressesProvider} from "../../interfaces/ILendingPoolAddressesProvider.sol";
import {ILendingPool} from "../../interfaces/ILendingPool.sol";

/**
 * @title IFlashLoanReceiver interface
 * @notice Interface for the Aave fee IFlashLoanReceiver.
 * @author Aave
 * @dev implement this interface to develop a flashloan-compatible flashLoanReceiver contract
 **/
interface IFlashLoanReceiver {
    function executeOperation(
        address[] calldata assets,
        uint256[] calldata amounts,
        uint256[] calldata premiums,
        address initiator,
        bytes calldata params
    ) external returns (bool);

    function ADDRESSES_PROVIDER()
        external
        view
        returns (ILendingPoolAddressesProvider);

    function LENDING_POOL() external view returns (ILendingPool);
}
          

contracts/interfaces/IAToken.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {IERC20} from "../dependencies/openzeppelin/contracts/IERC20.sol";
import {IScaledBalanceToken} from "./IScaledBalanceToken.sol";
import {IInitializableAToken} from "./IInitializableAToken.sol";
import {IPhiatIncentivesController} from "./IPhiatIncentivesController.sol";

interface IAToken is IERC20, IScaledBalanceToken, IInitializableAToken {
    /**
     * @dev Emitted after the mint action
     * @param from The address performing the mint
     * @param value The amount being
     * @param index The new liquidity index of the reserve
     **/
    event Mint(address indexed from, uint256 value, uint256 index);

    /**
     * @dev Mints `amount` aTokens to `user`
     * @param user The address receiving the minted tokens
     * @param amount The amount of tokens getting minted
     * @param index The new liquidity index of the reserve
     * @return `true` if the the previous balance of the user was 0
     */
    function mint(
        address user,
        uint256 amount,
        uint256 index
    ) external returns (bool);

    /**
     * @dev Emitted after aTokens are burned
     * @param from The owner of the aTokens, getting them burned
     * @param target The address that will receive the underlying
     * @param value The amount being burned
     * @param index The new liquidity index of the reserve
     **/
    event Burn(
        address indexed from,
        address indexed target,
        uint256 value,
        uint256 index
    );

    /**
     * @dev Emitted during the transfer action
     * @param from The user whose tokens are being transferred
     * @param to The recipient
     * @param value The amount being transferred
     * @param index The new liquidity index of the reserve
     **/
    event BalanceTransfer(
        address indexed from,
        address indexed to,
        uint256 value,
        uint256 index
    );

    /**
     * @dev Burns aTokens from `user` and sends the equivalent amount of underlying to `receiverOfUnderlying`
     * @param user The owner of the aTokens, getting them burned
     * @param receiverOfUnderlying The address that will receive the underlying
     * @param amount The amount being burned
     * @param index The new liquidity index of the reserve
     **/
    function burn(
        address user,
        address receiverOfUnderlying,
        uint256 amount,
        uint256 index
    ) external;

    /**
     * @dev Mints aTokens to the reserve treasury
     * @param amount The amount of tokens getting minted
     * @param index The new liquidity index of the reserve
     */
    function mintToTreasury(uint256 amount, uint256 index) external;

    /**
     * @dev Transfers aTokens in the event of a borrow being liquidated, in case the liquidators reclaims the aToken
     * @param from The address getting liquidated, current owner of the aTokens
     * @param to The recipient
     * @param value The amount of tokens getting transferred
     **/
    function transferOnLiquidation(
        address from,
        address to,
        uint256 value
    ) external;

    /**
     * @dev Transfers the underlying asset to `target`. Used by the LendingPool to transfer
     * assets in borrow(), withdraw() and flashLoan()
     * @param user The recipient of the underlying
     * @param amount The amount getting transferred
     * @return The amount transferred
     **/
    function transferUnderlyingTo(address user, uint256 amount)
        external
        returns (uint256);

    /**
     * @dev Invoked to execute actions on the aToken side after a repayment.
     * @param user The user executing the repayment
     * @param amount The amount getting repaid
     **/
    function handleRepayment(address user, uint256 amount) external;

    /**
     * @dev Returns the address of the incentives controller contract
     **/
    function getIncentivesController()
        external
        view
        returns (IPhiatIncentivesController);

    /**
     * @dev Returns the address of treasury
     **/
    function RESERVE_TREASURY_ADDRESS() external view returns (address);

    /**
     * @dev Returns the address of the underlying asset of this aToken (E.g. WETH for aWETH)
     **/
    function UNDERLYING_ASSET_ADDRESS() external view returns (address);
}
          

contracts/interfaces/IInitializableAToken.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {ILendingPool} from "./ILendingPool.sol";

/**
 * @title IInitializableAToken
 * @notice Interface for the initialize function on AToken
 * @author Aave
 **/
interface IInitializableAToken {
    /**
     * @dev Emitted when an aToken is initialized
     * @param underlyingAsset The address of the underlying asset
     * @param pool The address of the associated lending pool
     * @param treasury The address of the treasury
     * @param aTokenDecimals the decimals of the underlying
     * @param aTokenName the name of the aToken
     * @param aTokenSymbol the symbol of the aToken
     * @param params A set of encoded parameters for additional initialization
     **/
    event Initialized(
        address indexed underlyingAsset,
        address indexed pool,
        address treasury,
        uint8 aTokenDecimals,
        string aTokenName,
        string aTokenSymbol,
        bytes params
    );

    /**
     * @dev Initializes the aToken
     * @param pool The address of the lending pool where this aToken will be used
     * @param treasury The address of the Aave treasury, receiving the fees on this aToken
     * @param underlyingAsset The address of the underlying asset of this aToken (E.g. WETH for aWETH)
     * @param aTokenDecimals The decimals of the aToken, same as the underlying asset's
     * @param aTokenName The name of the aToken
     * @param aTokenSymbol The symbol of the aToken
     */
    function initialize(
        ILendingPool pool,
        address treasury,
        address underlyingAsset,
        uint8 aTokenDecimals,
        string calldata aTokenName,
        string calldata aTokenSymbol,
        bytes calldata params
    ) external;
}
          

contracts/interfaces/IInitializableDebtToken.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {ILendingPool} from "./ILendingPool.sol";

/**
 * @title IInitializableDebtToken
 * @notice Interface for the initialize function common between debt tokens
 * @author Aave
 **/
interface IInitializableDebtToken {
    /**
     * @dev Emitted when a debt token is initialized
     * @param underlyingAsset The address of the underlying asset
     * @param pool The address of the associated lending pool
     * @param debtTokenDecimals the decimals of the debt token
     * @param debtTokenName the name of the debt token
     * @param debtTokenSymbol the symbol of the debt token
     * @param params A set of encoded parameters for additional initialization
     **/
    event Initialized(
        address indexed underlyingAsset,
        address indexed pool,
        uint8 debtTokenDecimals,
        string debtTokenName,
        string debtTokenSymbol,
        bytes params
    );

    /**
     * @dev Initializes the debt token.
     * @param pool The address of the lending pool where this aToken will be used
     * @param underlyingAsset The address of the underlying asset of this aToken (E.g. WETH for aWETH)
     * @param debtTokenDecimals The decimals of the debtToken, same as the underlying asset's
     * @param debtTokenName The name of the token
     * @param debtTokenSymbol The symbol of the token
     */
    function initialize(
        ILendingPool pool,
        address underlyingAsset,
        uint8 debtTokenDecimals,
        string memory debtTokenName,
        string memory debtTokenSymbol,
        bytes calldata params
    ) external;
}
          

contracts/interfaces/ILendingPool.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;
pragma experimental ABIEncoderV2;

import {ILendingPoolAddressesProvider} from "./ILendingPoolAddressesProvider.sol";
import {DataTypes} from "../protocol/libraries/types/DataTypes.sol";

interface ILendingPool {
    /**
     * @dev Emitted on deposit()
     * @param reserve The address of the underlying asset of the reserve
     * @param user The address initiating the deposit
     * @param onBehalfOf The beneficiary of the deposit, receiving the aTokens
     * @param amount The amount deposited
     * @param referral The referral code used
     **/
    event Deposit(
        address indexed reserve,
        address user,
        address indexed onBehalfOf,
        uint256 amount,
        uint16 indexed referral
    );

    /**
     * @dev Emitted on withdraw()
     * @param reserve The address of the underlyng asset being withdrawn
     * @param user The address initiating the withdrawal, owner of aTokens
     * @param to Address that will receive the underlying
     * @param amount The amount to be withdrawn
     **/
    event Withdraw(
        address indexed reserve,
        address indexed user,
        address indexed to,
        uint256 amount
    );

    /**
     * @dev Emitted on borrow() and flashLoan() when debt needs to be opened
     * @param reserve The address of the underlying asset being borrowed
     * @param user The address of the user initiating the borrow(), receiving the funds on borrow() or just
     * initiator of the transaction on flashLoan()
     * @param onBehalfOf The address that will be getting the debt
     * @param amount The amount borrowed out
     * @param borrowRateMode The rate mode: 1 for Stable, 2 for Variable
     * @param borrowRate The numeric rate at which the user has borrowed
     * @param referral The referral code used
     **/
    event Borrow(
        address indexed reserve,
        address user,
        address indexed onBehalfOf,
        uint256 amount,
        uint256 borrowRateMode,
        uint256 borrowRate,
        uint16 indexed referral
    );

    /**
     * @dev Emitted on repay()
     * @param reserve The address of the underlying asset of the reserve
     * @param user The beneficiary of the repayment, getting his debt reduced
     * @param repayer The address of the user initiating the repay(), providing the funds
     * @param amount The amount repaid
     **/
    event Repay(
        address indexed reserve,
        address indexed user,
        address indexed repayer,
        uint256 amount
    );

    /**
     * @dev Emitted on swapBorrowRateMode()
     * @param reserve The address of the underlying asset of the reserve
     * @param user The address of the user swapping his rate mode
     * @param rateMode The rate mode that the user wants to swap to
     **/
    event Swap(address indexed reserve, address indexed user, uint256 rateMode);

    /**
     * @dev Emitted on setUserUseReserveAsCollateral()
     * @param reserve The address of the underlying asset of the reserve
     * @param user The address of the user enabling the usage as collateral
     **/
    event ReserveUsedAsCollateralEnabled(
        address indexed reserve,
        address indexed user
    );

    /**
     * @dev Emitted on setUserUseReserveAsCollateral()
     * @param reserve The address of the underlying asset of the reserve
     * @param user The address of the user enabling the usage as collateral
     **/
    event ReserveUsedAsCollateralDisabled(
        address indexed reserve,
        address indexed user
    );

    /**
     * @dev Emitted on rebalanceStableBorrowRate()
     * @param reserve The address of the underlying asset of the reserve
     * @param user The address of the user for which the rebalance has been executed
     **/
    event RebalanceStableBorrowRate(
        address indexed reserve,
        address indexed user
    );

    /**
     * @dev Emitted on flashLoan()
     * @param target The address of the flash loan receiver contract
     * @param initiator The address initiating the flash loan
     * @param asset The address of the asset being flash borrowed
     * @param amount The amount flash borrowed
     * @param premium The fee flash borrowed
     * @param referralCode The referral code used
     **/
    event FlashLoan(
        address indexed target,
        address indexed initiator,
        address indexed asset,
        uint256 amount,
        uint256 premium,
        uint16 referralCode
    );

    /**
     * @dev Emitted when the pause is triggered.
     */
    event Paused();

    /**
     * @dev Emitted when the pause is lifted.
     */
    event Unpaused();

    /**
     * @dev Emitted when a borrower is liquidated. This event is emitted by the LendingPool via
     * LendingPoolCollateral manager using a DELEGATECALL
     * This allows to have the events in the generated ABI for LendingPool.
     * @param collateralAsset The address of the underlying asset used as collateral, to receive as result of the liquidation
     * @param debtAsset The address of the underlying borrowed asset to be repaid with the liquidation
     * @param user The address of the borrower getting liquidated
     * @param debtToCover The debt amount of borrowed `asset` the liquidator wants to cover
     * @param liquidatedCollateralAmount The amount of collateral received by the liiquidator
     * @param liquidator The address of the liquidator
     * @param receiveAToken `true` if the liquidators wants to receive the collateral aTokens, `false` if he wants
     * to receive the underlying collateral asset directly
     **/
    event LiquidationCall(
        address indexed collateralAsset,
        address indexed debtAsset,
        address indexed user,
        uint256 debtToCover,
        uint256 liquidatedCollateralAmount,
        address liquidator,
        bool receiveAToken
    );

    /**
     * @dev Emitted when a borrower is blacklisted.
     * @param user The address of the borrower getting liquidated
     **/
    event Blacklist(address indexed user);

    /**
     * @dev Emitted when the state of a reserve is updated. NOTE: This event is actually declared
     * in the ReserveLogic library and emitted in the updateInterestRates() function. Since the function is internal,
     * the event will actually be fired by the LendingPool contract. The event is therefore replicated here so it
     * gets added to the LendingPool ABI
     * @param reserve The address of the underlying asset of the reserve
     * @param liquidityRate The new liquidity rate
     * @param stableBorrowRate The new stable borrow rate
     * @param variableBorrowRate The new variable borrow rate
     * @param liquidityIndex The new liquidity index
     * @param variableBorrowIndex The new variable borrow index
     **/
    event ReserveDataUpdated(
        address indexed reserve,
        uint256 liquidityRate,
        uint256 stableBorrowRate,
        uint256 variableBorrowRate,
        uint256 liquidityIndex,
        uint256 variableBorrowIndex
    );

    function isBlacklisted(address user) external returns (bool);

    /**
     * @dev Deposits an `amount` of underlying asset into the reserve, receiving in return overlying aTokens.
     * - E.g. User deposits 100 USDC and gets in return 100 aUSDC
     * @param asset The address of the underlying asset to deposit
     * @param amount The amount to be deposited
     * @param onBehalfOf The address that will receive the aTokens, same as msg.sender if the user
     *   wants to receive them on his own wallet, or a different address if the beneficiary of aTokens
     *   is a different wallet
     * @param referralCode Code used to register the integrator originating the operation, for potential rewards.
     *   0 if the action is executed directly by the user, without any middle-man
     **/
    function deposit(
        address asset,
        uint256 amount,
        address onBehalfOf,
        uint16 referralCode
    ) external;

    /**
     * @dev Withdraws an `amount` of underlying asset from the reserve, burning the equivalent aTokens owned
     * E.g. User has 100 aUSDC, calls withdraw() and receives 100 USDC, burning the 100 aUSDC
     * @param asset The address of the underlying asset to withdraw
     * @param amount The underlying amount to be withdrawn
     *   - Send the value type(uint256).max in order to withdraw the whole aToken balance
     * @param to Address that will receive the underlying, same as msg.sender if the user
     *   wants to receive it on his own wallet, or a different address if the beneficiary is a
     *   different wallet
     * @return The final amount withdrawn
     **/
    function withdraw(
        address asset,
        uint256 amount,
        address to
    ) external returns (uint256);

    /**
     * @dev Allows users to borrow a specific `amount` of the reserve underlying asset, provided that the borrower
     * already deposited enough collateral, or he was given enough allowance by a credit delegator on the
     * corresponding debt token (StableDebtToken or VariableDebtToken)
     * - E.g. User borrows 100 USDC passing as `onBehalfOf` his own address, receiving the 100 USDC in his wallet
     *   and 100 stable/variable debt tokens, depending on the `interestRateMode`
     * @param asset The address of the underlying asset to borrow
     * @param amount The amount to be borrowed
     * @param interestRateMode The interest rate mode at which the user wants to borrow: 1 for Stable, 2 for Variable
     * @param referralCode Code used to register the integrator originating the operation, for potential rewards.
     *   0 if the action is executed directly by the user, without any middle-man
     * @param onBehalfOf Address of the user who will receive the debt. Should be the address of the borrower itself
     * calling the function if he wants to borrow against his own collateral, or the address of the credit delegator
     * if he has been given credit delegation allowance
     **/
    function borrow(
        address asset,
        uint256 amount,
        uint256 interestRateMode,
        uint16 referralCode,
        address onBehalfOf
    ) external;

    /**
     * @notice Repays a borrowed `amount` on a specific reserve, burning the equivalent debt tokens owned
     * - E.g. User repays 100 USDC, burning 100 variable/stable debt tokens of the `onBehalfOf` address
     * @param asset The address of the borrowed underlying asset previously borrowed
     * @param amount The amount to repay
     * - Send the value type(uint256).max in order to repay the whole debt for `asset` on the specific `debtMode`
     * @param rateMode The interest rate mode at of the debt the user wants to repay: 1 for Stable, 2 for Variable
     * @param onBehalfOf Address of the user who will get his debt reduced/removed. Should be the address of the
     * user calling the function if he wants to reduce/remove his own debt, or the address of any other
     * other borrower whose debt should be removed
     * @return The final amount repaid
     **/
    function repay(
        address asset,
        uint256 amount,
        uint256 rateMode,
        address onBehalfOf
    ) external returns (uint256);

    /**
     * @notice Repay all debts of the user and blacklist that user.
     * @param user Address of the user who will get his debt removed. Should be the address of the
     * user calling the function if he wants to reduce/remove his own debt, or the address of any other
     * other borrower whose debt should be removed
     **/
    function repayAllAndBlacklist(address user) external;

    /**
     * @dev Allows a borrower to swap his debt between stable and variable mode, or viceversa
     * @param asset The address of the underlying asset borrowed
     * @param rateMode The rate mode that the user wants to swap to
     **/
    function swapBorrowRateMode(address asset, uint256 rateMode) external;

    /**
     * @dev Rebalances the stable interest rate of a user to the current stable rate defined on the reserve.
     * - Users can be rebalanced if the following conditions are satisfied:
     *     1. Usage ratio is above 95%
     *     2. the current deposit APY is below REBALANCE_UP_THRESHOLD * maxVariableBorrowRate, which means that too much has been
     *        borrowed at a stable rate and depositors are not earning enough
     * @param asset The address of the underlying asset borrowed
     * @param user The address of the user to be rebalanced
     **/
    function rebalanceStableBorrowRate(address asset, address user) external;

    /**
     * @dev Allows depositors to enable/disable a specific deposited asset as collateral
     * @param asset The address of the underlying asset deposited
     * @param useAsCollateral `true` if the user wants to use the deposit as collateral, `false` otherwise
     **/
    function setUserUseReserveAsCollateral(address asset, bool useAsCollateral)
        external;

    /**
     * @dev Function to liquidate a non-healthy position collateral-wise, with Health Factor below 1
     * - The caller (liquidator) covers `debtToCover` amount of debt of the user getting liquidated, and receives
     *   a proportionally amount of the `collateralAsset` plus a bonus to cover market risk
     * @param collateralAsset The address of the underlying asset used as collateral, to receive as result of the liquidation
     * @param debtAsset The address of the underlying borrowed asset to be repaid with the liquidation
     * @param user The address of the borrower getting liquidated
     * @param debtToCover The debt amount of borrowed `asset` the liquidator wants to cover
     * @param receiveAToken `true` if the liquidators wants to receive the collateral aTokens, `false` if he wants
     * to receive the underlying collateral asset directly
     **/
    function liquidationCall(
        address collateralAsset,
        address debtAsset,
        address user,
        uint256 debtToCover,
        bool receiveAToken
    ) external;

    /**
     * @dev Allows smartcontracts to access the liquidity of the pool within one transaction,
     * as long as the amount taken plus a fee is returned.
     * IMPORTANT There are security concerns for developers of flashloan receiver contracts that must be kept into consideration.
     * For further details please visit https://developers.aave.com
     * @param receiverAddress The address of the contract receiving the funds, implementing the IFlashLoanReceiver interface
     * @param assets The addresses of the assets being flash-borrowed
     * @param amounts The amounts amounts being flash-borrowed
     * @param params Variadic packed params to pass to the receiver as extra information
     * @param referralCode Code used to register the integrator originating the operation, for potential rewards.
     *   0 if the action is executed directly by the user, without any middle-man
     **/
    function flashLoan(
        address receiverAddress,
        address[] calldata assets,
        uint256[] calldata amounts,
        bytes calldata params,
        uint16 referralCode
    ) external;

    /**
     * @dev Returns the user account data across all the reserves
     * @param user The address of the user
     * @return totalCollateralETH the total collateral in ETH of the user
     * @return totalDebtETH the total debt in ETH of the user
     * @return availableBorrowsETH the borrowing power left of the user
     * @return currentLiquidationThreshold the liquidation threshold of the user
     * @return ltv the loan to value of the user
     * @return healthFactor the current health factor of the user
     **/
    function getUserAccountData(address user)
        external
        view
        returns (
            uint256 totalCollateralETH,
            uint256 totalDebtETH,
            uint256 availableBorrowsETH,
            uint256 currentLiquidationThreshold,
            uint256 ltv,
            uint256 healthFactor
        );

    function initReserve(
        address reserve,
        address aTokenAddress,
        address stableDebtAddress,
        address variableDebtAddress,
        address interestRateStrategyAddress
    ) external;

    function setReserveInterestRateStrategyAddress(
        address reserve,
        address rateStrategyAddress
    ) external;

    function setConfiguration(address reserve, uint256 configuration) external;

    function setLimits(
        address reserve,
        DataTypes.ReserveLimits calldata reserveLimits
    ) external;

    /**
     * @dev Returns the configuration of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The configuration of the reserve
     **/
    function getConfiguration(address asset)
        external
        view
        returns (DataTypes.ReserveConfigurationMap memory);

    /**
     * @dev Returns the configuration of the user across all the reserves
     * @param user The user address
     * @return The configuration of the user
     **/
    function getUserConfiguration(address user)
        external
        view
        returns (DataTypes.UserConfigurationMap memory);

    /**
     * @dev Returns the normalized income normalized income of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The reserve's normalized income
     */
    function getReserveNormalizedIncome(address asset)
        external
        view
        returns (uint256);

    /**
     * @dev Returns the normalized variable debt per unit of asset
     * @param asset The address of the underlying asset of the reserve
     * @return The reserve normalized variable debt
     */
    function getReserveNormalizedVariableDebt(address asset)
        external
        view
        returns (uint256);

    /**
     * @dev Returns the state and configuration of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The state of the reserve
     **/
    function getReserveData(address asset)
        external
        view
        returns (DataTypes.ReserveData memory);

    /**
     * @dev Returns the limits of the reserve
     * @param asset The address of the underlying asset of the reserve
     * @return The limits of the reserve
     **/
    function getReserveLimits(address asset)
        external
        view
        returns (DataTypes.ReserveLimits memory);

    function finalizeTransfer(
        address asset,
        address from,
        address to,
        uint256 amount,
        uint256 balanceFromAfter,
        uint256 balanceToBefore
    ) external;

    function getReservesList() external view returns (address[] memory);

    function getAddressesProvider()
        external
        view
        returns (ILendingPoolAddressesProvider);

    function setPause(bool val) external;

    function paused() external view returns (bool);

    function FLASHLOAN_PREMIUM_TOTAL() external view returns (uint256);
}
          

contracts/interfaces/ILendingPoolAddressesProvider.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

/**
 * @title LendingPoolAddressesProvider contract
 * @dev Main registry of addresses part of or connected to the protocol, including permissioned roles
 * - Acting also as factory of proxies and admin of those, so with right to change its implementations
 * - Owned by the Aave Governance
 * @author Aave
 **/
interface ILendingPoolAddressesProvider {
    event MarketIdSet(string newMarketId);
    event LendingPoolUpdated(address indexed newAddress);
    event ConfigurationAdminUpdated(address indexed newAddress);
    event EmergencyAdminUpdated(address indexed newAddress);
    event LendingPoolConfiguratorUpdated(address indexed newAddress);
    event LendingPoolCollateralManagerUpdated(address indexed newAddress);
    event PriceOracleUpdated(address indexed newAddress);
    event LendingRateOracleUpdated(address indexed newAddress);
    event IncentivesControllerUpdated(address indexed newAddress);
    event ProxyCreated(bytes32 id, address indexed newAddress);
    event AddressSet(bytes32 id, address indexed newAddress, bool hasProxy);

    function getMarketId() external view returns (string memory);

    function setMarketId(string calldata marketId) external;

    function setAddress(bytes32 id, address newAddress) external;

    function setAddressAsProxy(bytes32 id, address impl) external;

    function getAddress(bytes32 id) external view returns (address);

    function getLendingPool() external view returns (address);

    function setLendingPoolImpl(address pool) external;

    function getLendingPoolConfigurator() external view returns (address);

    function setLendingPoolConfiguratorImpl(address configurator) external;

    function getLendingPoolCollateralManager() external view returns (address);

    function setLendingPoolCollateralManager(address manager) external;

    function getPoolAdmin() external view returns (address);

    function setPoolAdmin(address admin) external;

    function getEmergencyAdmin() external view returns (address);

    function setEmergencyAdmin(address admin) external;

    function getPriceOracle() external view returns (address);

    function setPriceOracle(address priceOracle) external;

    function getLendingRateOracle() external view returns (address);

    function setLendingRateOracle(address lendingRateOracle) external;

    function getIncentivesController() external view returns (address);

    function setIncentivesController(address lendingRateOracle) external;
}
          

contracts/interfaces/IPhiatIncentivesController.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;
pragma experimental ABIEncoderV2;

interface IPhiatIncentivesController {
    event RewardsAccrued(address indexed user, uint256 amount);

    event RewardsClaimed(address indexed user, uint256 amount);

    /**
     * @dev add asset to accumulate rewards
     * @param asset The address of the reference asset of the distribution
     */
    function addAsset(address asset) external;

    /**
     * @dev Returns the configuration of the distribution for a certain asset
     * @param asset The address of the reference asset of the distribution
     * @return The asset index, the emission per second and the last updated timestamp
     **/
    function getAssetData(address asset)
        external
        view
        returns (
            uint256,
            uint256,
            uint256
        );

    /**
     * LEGACY **************************
     * @dev Returns the configuration of the distribution for a certain asset
     * @param asset The address of the reference asset of the distribution
     * @return The asset index, the emission per second and the last updated timestamp
     **/
    function assets(address asset)
        external
        view
        returns (
            uint128,
            uint128,
            uint256
        );

    /**
     * @dev Called by the corresponding asset on any update that affects the rewards distribution
     * @param user The address of the user
     * @param userBalance The (old) balance of the user of the asset in the lending pool
     *      latest user balance (if different) can be retrieved from sender's balanceOf function
     * @param totalSupply The (old) total supply of the asset in the lending pool
     *      latest total supply (if different) can be retrieved from sender's totalSupply function
     **/
    function handleAction(
        address user,
        uint256 userBalance,
        uint256 totalSupply
    ) external;

    /**
     * @dev returns the unclaimed rewards of the user
     * @param user the address of the user
     * @return the unclaimed user rewards
     */
    function getUserUnclaimedRewards(address user)
        external
        view
        returns (uint256);

    /**
     * @dev returns the unclaimed rewards of the user
     * @param user the address of the user
     * @param asset The asset to incentivize
     * @return the user index for the asset
     */
    function getUserAssetData(address user, address asset)
        external
        view
        returns (uint256);

    /**
     * @dev claim all user rewards
     * @param user the address of the user
     */
    function claimRewards(address user) external;

    /**
     * @dev for backward compatibility with previous implementation of the Incentives controller
     */
    function REWARD_TOKEN() external view returns (address);

    /**
     * @dev for backward compatibility with previous implementation of the Incentives controller
     */
    function PRECISION() external view returns (uint8);

    /**
     * @dev Gets the distribution end timestamp of the emissions
     */
    function DISTRIBUTION_END() external view returns (uint256);
}
          

contracts/interfaces/IPriceOracleGetter.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

/**
 * @title IPriceOracleGetter interface
 * @notice Interface for the Aave price oracle.
 **/

interface IPriceOracleGetter {
    /**
     * @dev returns the asset price in ETH
     * @param asset the address of the asset
     * @return the ETH price of the asset
     **/
    function getAssetPrice(address asset) external view returns (uint256);
}
          

contracts/interfaces/IReserveInterestRateStrategy.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

/**
 * @title IReserveInterestRateStrategyInterface interface
 * @dev Interface for the calculation of the interest rates
 * @author Aave
 */
interface IReserveInterestRateStrategy {
    function baseVariableBorrowRate() external view returns (uint256);

    function getMaxVariableBorrowRate() external view returns (uint256);

    function calculateInterestRates(
        address reserve,
        uint256 availableLiquidity,
        uint256 totalStableDebt,
        uint256 totalVariableDebt,
        uint256 averageStableBorrowRate,
        uint256 reserveFactor
    )
        external
        view
        returns (
            uint256,
            uint256,
            uint256
        );

    function calculateInterestRates(
        address reserve,
        address aToken,
        uint256 liquidityAdded,
        uint256 liquidityTaken,
        uint256 totalStableDebt,
        uint256 totalVariableDebt,
        uint256 averageStableBorrowRate,
        uint256 reserveFactor
    )
        external
        view
        returns (
            uint256 liquidityRate,
            uint256 stableBorrowRate,
            uint256 variableBorrowRate
        );
}
          

contracts/interfaces/IScaledBalanceToken.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

interface IScaledBalanceToken {
    /**
     * @dev Returns the scaled balance of the user. The scaled balance is the sum of all the
     * updated stored balance divided by the reserve's liquidity index at the moment of the update
     * @param user The user whose balance is calculated
     * @return The scaled balance of the user
     **/
    function scaledBalanceOf(address user) external view returns (uint256);

    /**
     * @dev Returns the scaled balance of the user and the scaled total supply.
     * @param user The address of the user
     * @return The scaled balance of the user
     * @return The scaled balance and the scaled total supply
     **/
    function getScaledUserBalanceAndSupply(address user)
        external
        view
        returns (uint256, uint256);

    /**
     * @dev Returns the scaled total supply of the variable debt token. Represents sum(debt/index)
     * @return The scaled total supply
     **/
    function scaledTotalSupply() external view returns (uint256);
}
          

contracts/interfaces/IStableDebtToken.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {IInitializableDebtToken} from "./IInitializableDebtToken.sol";
import {IPhiatIncentivesController} from "./IPhiatIncentivesController.sol";

/**
 * @title IStableDebtToken
 * @notice Defines the interface for the stable debt token
 * @dev It does not inherit from IERC20 to save in code size
 * @author Aave
 **/

interface IStableDebtToken is IInitializableDebtToken {
    /**
     * @dev Emitted when new stable debt is minted
     * @param user The address of the user who triggered the minting
     * @param onBehalfOf The recipient of stable debt tokens
     * @param amount The amount minted
     * @param currentBalance The current balance of the user
     * @param balanceIncrease The increase in balance since the last action of the user
     * @param newRate The rate of the debt after the minting
     * @param avgStableRate The new average stable rate after the minting
     * @param newTotalSupply The new total supply of the stable debt token after the action
     **/
    event Mint(
        address indexed user,
        address indexed onBehalfOf,
        uint256 amount,
        uint256 currentBalance,
        uint256 balanceIncrease,
        uint256 newRate,
        uint256 avgStableRate,
        uint256 newTotalSupply
    );

    /**
     * @dev Emitted when new stable debt is burned
     * @param user The address of the user
     * @param amount The amount being burned
     * @param currentBalance The current balance of the user
     * @param balanceIncrease The the increase in balance since the last action of the user
     * @param avgStableRate The new average stable rate after the burning
     * @param newTotalSupply The new total supply of the stable debt token after the action
     **/
    event Burn(
        address indexed user,
        uint256 amount,
        uint256 currentBalance,
        uint256 balanceIncrease,
        uint256 avgStableRate,
        uint256 newTotalSupply
    );

    /**
     * @dev Mints debt token to the `onBehalfOf` address.
     * - The resulting rate is the weighted average between the rate of the new debt
     * and the rate of the previous debt
     * @param user The address receiving the borrowed underlying, being the delegatee in case
     * of credit delegate, or same as `onBehalfOf` otherwise
     * @param onBehalfOf The address receiving the debt tokens
     * @param amount The amount of debt tokens to mint
     * @param rate The rate of the debt being minted
     **/
    function mint(
        address user,
        address onBehalfOf,
        uint256 amount,
        uint256 rate
    ) external returns (bool);

    /**
     * @dev Burns debt of `user`
     * - The resulting rate is the weighted average between the rate of the new debt
     * and the rate of the previous debt
     * @param user The address of the user getting his debt burned
     * @param amount The amount of debt tokens getting burned
     **/
    function burn(address user, uint256 amount) external;

    /**
     * @dev Returns the average rate of all the stable rate loans.
     * @return The average stable rate
     **/
    function getAverageStableRate() external view returns (uint256);

    /**
     * @dev Returns the stable rate of the user debt
     * @return The stable rate of the user
     **/
    function getUserStableRate(address user) external view returns (uint256);

    /**
     * @dev Returns the timestamp of the last update of the user
     * @return The timestamp
     **/
    function getUserLastUpdated(address user) external view returns (uint40);

    /**
     * @dev Returns the principal, the total supply and the average stable rate
     **/
    function getSupplyData()
        external
        view
        returns (
            uint256,
            uint256,
            uint256,
            uint40
        );

    /**
     * @dev Returns the timestamp of the last update of the total supply
     * @return The timestamp
     **/
    function getTotalSupplyLastUpdated() external view returns (uint40);

    /**
     * @dev Returns the total supply and the average stable rate
     **/
    function getTotalSupplyAndAvgRate()
        external
        view
        returns (uint256, uint256);

    /**
     * @dev Returns the principal debt balance of the user
     * @return The debt balance of the user since the last burn/mint action
     **/
    function principalBalanceOf(address user) external view returns (uint256);

    /**
     * @dev Returns the address of the incentives controller contract
     **/
    function getIncentivesController()
        external
        view
        returns (IPhiatIncentivesController);
}
          

contracts/interfaces/IVariableDebtToken.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {IScaledBalanceToken} from "./IScaledBalanceToken.sol";
import {IInitializableDebtToken} from "./IInitializableDebtToken.sol";
import {IPhiatIncentivesController} from "./IPhiatIncentivesController.sol";

/**
 * @title IVariableDebtToken
 * @author Aave
 * @notice Defines the basic interface for a variable debt token.
 **/
interface IVariableDebtToken is IScaledBalanceToken, IInitializableDebtToken {
    /**
     * @dev Emitted after the mint action
     * @param from The address performing the mint
     * @param onBehalfOf The address of the user on which behalf minting has been performed
     * @param value The amount to be minted
     * @param index The last index of the reserve
     **/
    event Mint(
        address indexed from,
        address indexed onBehalfOf,
        uint256 value,
        uint256 index
    );

    /**
     * @dev Mints debt token to the `onBehalfOf` address
     * @param user The address receiving the borrowed underlying, being the delegatee in case
     * of credit delegate, or same as `onBehalfOf` otherwise
     * @param onBehalfOf The address receiving the debt tokens
     * @param amount The amount of debt being minted
     * @param index The variable debt index of the reserve
     * @return `true` if the the previous balance of the user is 0
     **/
    function mint(
        address user,
        address onBehalfOf,
        uint256 amount,
        uint256 index
    ) external returns (bool);

    /**
     * @dev Emitted when variable debt is burnt
     * @param user The user which debt has been burned
     * @param amount The amount of debt being burned
     * @param index The index of the user
     **/
    event Burn(address indexed user, uint256 amount, uint256 index);

    /**
     * @dev Burns user variable debt
     * @param user The user which debt is burnt
     * @param index The variable debt index of the reserve
     **/
    function burn(
        address user,
        uint256 amount,
        uint256 index
    ) external;

    /**
     * @dev Returns the address of the incentives controller contract
     **/
    function getIncentivesController()
        external
        view
        returns (IPhiatIncentivesController);
}
          

contracts/protocol/lendingpool/LendingPoolStorage.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {UserConfiguration} from "../libraries/configuration/UserConfiguration.sol";
import {ReserveConfiguration} from "../libraries/configuration/ReserveConfiguration.sol";
import {ReserveLogic} from "../libraries/logic/ReserveLogic.sol";
import {ILendingPoolAddressesProvider} from "../../interfaces/ILendingPoolAddressesProvider.sol";
import {DataTypes} from "../libraries/types/DataTypes.sol";

contract LendingPoolStorage {
    using ReserveLogic for DataTypes.ReserveData;
    using ReserveConfiguration for DataTypes.ReserveConfigurationMap;
    using UserConfiguration for DataTypes.UserConfigurationMap;

    ILendingPoolAddressesProvider internal _addressesProvider;

    mapping(address => DataTypes.ReserveData) internal _reserves;
    mapping(address => DataTypes.ReserveLimits) internal _reserveLimits;
    mapping(address => DataTypes.UserConfigurationMap) internal _usersConfig;

    // the list of the available reserves, structured as a mapping for gas savings reasons
    mapping(uint256 => address) internal _reservesList;

    uint256 internal _reservesCount;

    bool internal _paused;

    uint256 internal _maxStableRateBorrowSizePercent;

    uint256 internal _flashLoanPremiumTotal;

    uint256 internal _maxNumberOfReserves;
}
          

contracts/protocol/libraries/aave-upgradeability/VersionedInitializable.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

/**
 * @title VersionedInitializable
 *
 * @dev Helper contract to implement initializer functions. To use it, replace
 * the constructor with a function that has the `initializer` modifier.
 * WARNING: Unlike constructors, initializer functions must be manually
 * invoked. This applies both to deploying an Initializable contract, as well
 * as extending an Initializable contract via inheritance.
 * WARNING: When used with inheritance, manual care must be taken to not invoke
 * a parent initializer twice, or ensure that all initializers are idempotent,
 * because this is not dealt with automatically as with constructors.
 *
 * @author Aave, inspired by the OpenZeppelin Initializable contract
 */
abstract contract VersionedInitializable {
    /**
     * @dev Indicates that the contract has been initialized.
     */
    uint256 private lastInitializedRevision = 0;

    /**
     * @dev Indicates that the contract is in the process of being initialized.
     */
    bool private initializing;

    /**
     * @dev Modifier to use in the initializer function of a contract.
     */
    modifier initializer() {
        uint256 revision = getRevision();
        require(
            initializing ||
                isConstructor() ||
                revision > lastInitializedRevision,
            "Contract instance has already been initialized"
        );

        bool isTopLevelCall = !initializing;
        if (isTopLevelCall) {
            initializing = true;
            lastInitializedRevision = revision;
        }

        _;

        if (isTopLevelCall) {
            initializing = false;
        }
    }

    /**
     * @dev returns the revision number of the contract
     * Needs to be defined in the inherited class as a constant.
     **/
    function getRevision() internal pure virtual returns (uint256);

    /**
     * @dev Returns true if and only if the function is running in the constructor
     **/
    function isConstructor() private view returns (bool) {
        // extcodesize checks the size of the code stored in an address, and
        // address returns the current address. Since the code is still not
        // deployed when running a constructor, any checks on its code size will
        // yield zero, making it an effective way to detect if a contract is
        // under construction or not.
        uint256 cs;
        //solium-disable-next-line
        assembly {
            cs := extcodesize(address())
        }
        return cs == 0;
    }

    // Reserved storage space to allow for layout changes in the future.
    uint256[50] private ______gap;
}
          

contracts/protocol/libraries/configuration/ReserveConfiguration.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {Errors} from "../helpers/Errors.sol";
import {DataTypes} from "../types/DataTypes.sol";

/**
 * @title ReserveConfiguration library
 * @author Aave
 * @notice Implements the bitmap logic to handle the reserve configuration
 */
library ReserveConfiguration {
    uint256 constant LTV_MASK =                   0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000; // prettier-ignore
    uint256 constant LIQUIDATION_THRESHOLD_MASK = 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000FFFF; // prettier-ignore
    uint256 constant LIQUIDATION_BONUS_MASK =     0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000FFFFFFFF; // prettier-ignore
    uint256 constant DECIMALS_MASK =              0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF00FFFFFFFFFFFF; // prettier-ignore
    uint256 constant ACTIVE_MASK =                0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFF; // prettier-ignore
    uint256 constant FROZEN_MASK =                0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFDFFFFFFFFFFFFFF; // prettier-ignore
    uint256 constant BORROWING_MASK =             0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFBFFFFFFFFFFFFFF; // prettier-ignore
    uint256 constant STABLE_BORROWING_MASK =      0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFFFFF; // prettier-ignore
    uint256 constant RESERVE_FACTOR_MASK =        0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000FFFFFFFFFFFFFFFF; // prettier-ignore

    /// @dev For the LTV, the start bit is 0 (up to 15), hence no bitshifting is needed
    uint256 constant LIQUIDATION_THRESHOLD_START_BIT_POSITION = 16;
    uint256 constant LIQUIDATION_BONUS_START_BIT_POSITION = 32;
    uint256 constant RESERVE_DECIMALS_START_BIT_POSITION = 48;
    uint256 constant IS_ACTIVE_START_BIT_POSITION = 56;
    uint256 constant IS_FROZEN_START_BIT_POSITION = 57;
    uint256 constant BORROWING_ENABLED_START_BIT_POSITION = 58;
    uint256 constant STABLE_BORROWING_ENABLED_START_BIT_POSITION = 59;
    uint256 constant RESERVE_FACTOR_START_BIT_POSITION = 64;

    uint256 constant MAX_VALID_LTV = 65535;
    uint256 constant MAX_VALID_LIQUIDATION_THRESHOLD = 65535;
    uint256 constant MAX_VALID_LIQUIDATION_BONUS = 65535;
    uint256 constant MAX_VALID_DECIMALS = 255;
    uint256 constant MAX_VALID_RESERVE_FACTOR = 65535;

    /**
     * @dev Sets the Loan to Value of the reserve
     * @param self The reserve configuration
     * @param ltv the new ltv
     **/
    function setLtv(DataTypes.ReserveConfigurationMap memory self, uint256 ltv)
        internal
        pure
    {
        require(ltv <= MAX_VALID_LTV, Errors.RC_INVALID_LTV);

        self.data = (self.data & LTV_MASK) | ltv;
    }

    /**
     * @dev Gets the Loan to Value of the reserve
     * @param self The reserve configuration
     * @return The loan to value
     **/
    function getLtv(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (uint256)
    {
        return self.data & ~LTV_MASK;
    }

    /**
     * @dev Sets the liquidation threshold of the reserve
     * @param self The reserve configuration
     * @param threshold The new liquidation threshold
     **/
    function setLiquidationThreshold(
        DataTypes.ReserveConfigurationMap memory self,
        uint256 threshold
    ) internal pure {
        require(
            threshold <= MAX_VALID_LIQUIDATION_THRESHOLD,
            Errors.RC_INVALID_LIQ_THRESHOLD
        );

        self.data =
            (self.data & LIQUIDATION_THRESHOLD_MASK) |
            (threshold << LIQUIDATION_THRESHOLD_START_BIT_POSITION);
    }

    /**
     * @dev Gets the liquidation threshold of the reserve
     * @param self The reserve configuration
     * @return The liquidation threshold
     **/
    function getLiquidationThreshold(
        DataTypes.ReserveConfigurationMap storage self
    ) internal view returns (uint256) {
        return
            (self.data & ~LIQUIDATION_THRESHOLD_MASK) >>
            LIQUIDATION_THRESHOLD_START_BIT_POSITION;
    }

    /**
     * @dev Sets the liquidation bonus of the reserve
     * @param self The reserve configuration
     * @param bonus The new liquidation bonus
     **/
    function setLiquidationBonus(
        DataTypes.ReserveConfigurationMap memory self,
        uint256 bonus
    ) internal pure {
        require(
            bonus <= MAX_VALID_LIQUIDATION_BONUS,
            Errors.RC_INVALID_LIQ_BONUS
        );

        self.data =
            (self.data & LIQUIDATION_BONUS_MASK) |
            (bonus << LIQUIDATION_BONUS_START_BIT_POSITION);
    }

    /**
     * @dev Gets the liquidation bonus of the reserve
     * @param self The reserve configuration
     * @return The liquidation bonus
     **/
    function getLiquidationBonus(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (uint256)
    {
        return
            (self.data & ~LIQUIDATION_BONUS_MASK) >>
            LIQUIDATION_BONUS_START_BIT_POSITION;
    }

    /**
     * @dev Sets the decimals of the underlying asset of the reserve
     * @param self The reserve configuration
     * @param decimals The decimals
     **/
    function setDecimals(
        DataTypes.ReserveConfigurationMap memory self,
        uint256 decimals
    ) internal pure {
        require(decimals <= MAX_VALID_DECIMALS, Errors.RC_INVALID_DECIMALS);

        self.data =
            (self.data & DECIMALS_MASK) |
            (decimals << RESERVE_DECIMALS_START_BIT_POSITION);
    }

    /**
     * @dev Gets the decimals of the underlying asset of the reserve
     * @param self The reserve configuration
     * @return The decimals of the asset
     **/
    function getDecimals(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (uint256)
    {
        return
            (self.data & ~DECIMALS_MASK) >> RESERVE_DECIMALS_START_BIT_POSITION;
    }

    /**
     * @dev Sets the active state of the reserve
     * @param self The reserve configuration
     * @param active The active state
     **/
    function setActive(
        DataTypes.ReserveConfigurationMap memory self,
        bool active
    ) internal pure {
        self.data =
            (self.data & ACTIVE_MASK) |
            (uint256(active ? 1 : 0) << IS_ACTIVE_START_BIT_POSITION);
    }

    /**
     * @dev Gets the active state of the reserve
     * @param self The reserve configuration
     * @return The active state
     **/
    function getActive(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (bool)
    {
        return (self.data & ~ACTIVE_MASK) != 0;
    }

    /**
     * @dev Sets the frozen state of the reserve
     * @param self The reserve configuration
     * @param frozen The frozen state
     **/
    function setFrozen(
        DataTypes.ReserveConfigurationMap memory self,
        bool frozen
    ) internal pure {
        self.data =
            (self.data & FROZEN_MASK) |
            (uint256(frozen ? 1 : 0) << IS_FROZEN_START_BIT_POSITION);
    }

    /**
     * @dev Gets the frozen state of the reserve
     * @param self The reserve configuration
     * @return The frozen state
     **/
    function getFrozen(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (bool)
    {
        return (self.data & ~FROZEN_MASK) != 0;
    }

    /**
     * @dev Enables or disables borrowing on the reserve
     * @param self The reserve configuration
     * @param enabled True if the borrowing needs to be enabled, false otherwise
     **/
    function setBorrowingEnabled(
        DataTypes.ReserveConfigurationMap memory self,
        bool enabled
    ) internal pure {
        self.data =
            (self.data & BORROWING_MASK) |
            (uint256(enabled ? 1 : 0) << BORROWING_ENABLED_START_BIT_POSITION);
    }

    /**
     * @dev Gets the borrowing state of the reserve
     * @param self The reserve configuration
     * @return The borrowing state
     **/
    function getBorrowingEnabled(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (bool)
    {
        return (self.data & ~BORROWING_MASK) != 0;
    }

    /**
     * @dev Enables or disables stable rate borrowing on the reserve
     * @param self The reserve configuration
     * @param enabled True if the stable rate borrowing needs to be enabled, false otherwise
     **/
    function setStableRateBorrowingEnabled(
        DataTypes.ReserveConfigurationMap memory self,
        bool enabled
    ) internal pure {
        self.data =
            (self.data & STABLE_BORROWING_MASK) |
            (uint256(enabled ? 1 : 0) <<
                STABLE_BORROWING_ENABLED_START_BIT_POSITION);
    }

    /**
     * @dev Gets the stable rate borrowing state of the reserve
     * @param self The reserve configuration
     * @return The stable rate borrowing state
     **/
    function getStableRateBorrowingEnabled(
        DataTypes.ReserveConfigurationMap storage self
    ) internal view returns (bool) {
        return (self.data & ~STABLE_BORROWING_MASK) != 0;
    }

    /**
     * @dev Sets the reserve factor of the reserve
     * @param self The reserve configuration
     * @param reserveFactor The reserve factor
     **/
    function setReserveFactor(
        DataTypes.ReserveConfigurationMap memory self,
        uint256 reserveFactor
    ) internal pure {
        require(
            reserveFactor <= MAX_VALID_RESERVE_FACTOR,
            Errors.RC_INVALID_RESERVE_FACTOR
        );

        self.data =
            (self.data & RESERVE_FACTOR_MASK) |
            (reserveFactor << RESERVE_FACTOR_START_BIT_POSITION);
    }

    /**
     * @dev Gets the reserve factor of the reserve
     * @param self The reserve configuration
     * @return The reserve factor
     **/
    function getReserveFactor(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (uint256)
    {
        return
            (self.data & ~RESERVE_FACTOR_MASK) >>
            RESERVE_FACTOR_START_BIT_POSITION;
    }

    /**
     * @dev Gets the configuration flags of the reserve
     * @param self The reserve configuration
     * @return The state flags representing active, frozen, borrowing enabled, stableRateBorrowing enabled
     **/
    function getFlags(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (
            bool,
            bool,
            bool,
            bool
        )
    {
        uint256 dataLocal = self.data;

        return (
            (dataLocal & ~ACTIVE_MASK) != 0,
            (dataLocal & ~FROZEN_MASK) != 0,
            (dataLocal & ~BORROWING_MASK) != 0,
            (dataLocal & ~STABLE_BORROWING_MASK) != 0
        );
    }

    /**
     * @dev Gets the configuration paramters of the reserve
     * @param self The reserve configuration
     * @return The state params representing ltv, liquidation threshold, liquidation bonus, the reserve decimals
     **/
    function getParams(DataTypes.ReserveConfigurationMap storage self)
        internal
        view
        returns (
            uint256,
            uint256,
            uint256,
            uint256,
            uint256
        )
    {
        uint256 dataLocal = self.data;

        return (
            dataLocal & ~LTV_MASK,
            (dataLocal & ~LIQUIDATION_THRESHOLD_MASK) >>
                LIQUIDATION_THRESHOLD_START_BIT_POSITION,
            (dataLocal & ~LIQUIDATION_BONUS_MASK) >>
                LIQUIDATION_BONUS_START_BIT_POSITION,
            (dataLocal & ~DECIMALS_MASK) >> RESERVE_DECIMALS_START_BIT_POSITION,
            (dataLocal & ~RESERVE_FACTOR_MASK) >>
                RESERVE_FACTOR_START_BIT_POSITION
        );
    }

    /**
     * @dev Gets the configuration paramters of the reserve from a memory object
     * @param self The reserve configuration
     * @return The state params representing ltv, liquidation threshold, liquidation bonus, the reserve decimals
     **/
    function getParamsMemory(DataTypes.ReserveConfigurationMap memory self)
        internal
        pure
        returns (
            uint256,
            uint256,
            uint256,
            uint256,
            uint256
        )
    {
        return (
            self.data & ~LTV_MASK,
            (self.data & ~LIQUIDATION_THRESHOLD_MASK) >>
                LIQUIDATION_THRESHOLD_START_BIT_POSITION,
            (self.data & ~LIQUIDATION_BONUS_MASK) >>
                LIQUIDATION_BONUS_START_BIT_POSITION,
            (self.data & ~DECIMALS_MASK) >> RESERVE_DECIMALS_START_BIT_POSITION,
            (self.data & ~RESERVE_FACTOR_MASK) >>
                RESERVE_FACTOR_START_BIT_POSITION
        );
    }

    /**
     * @dev Gets the configuration flags of the reserve from a memory object
     * @param self The reserve configuration
     * @return The state flags representing active, frozen, borrowing enabled, stableRateBorrowing enabled
     **/
    function getFlagsMemory(DataTypes.ReserveConfigurationMap memory self)
        internal
        pure
        returns (
            bool,
            bool,
            bool,
            bool
        )
    {
        return (
            (self.data & ~ACTIVE_MASK) != 0,
            (self.data & ~FROZEN_MASK) != 0,
            (self.data & ~BORROWING_MASK) != 0,
            (self.data & ~STABLE_BORROWING_MASK) != 0
        );
    }
}
          

contracts/protocol/libraries/configuration/UserConfiguration.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {Errors} from "../helpers/Errors.sol";
import {DataTypes} from "../types/DataTypes.sol";

/**
 * @title UserConfiguration library
 * @author Aave
 * @notice Implements the bitmap logic to handle the user configuration
 */
library UserConfiguration {
    uint256 internal constant BORROWING_MASK =
        0x5555555555555555555555555555555555555555555555555555555555555555;

    /**
     * @dev Sets if the user is borrowing the reserve identified by reserveIndex
     * @param self The configuration object
     * @param reserveIndex The index of the reserve in the bitmap
     * @param borrowing True if the user is borrowing the reserve, false otherwise
     **/
    function setBorrowing(
        DataTypes.UserConfigurationMap storage self,
        uint256 reserveIndex,
        bool borrowing
    ) internal {
        require(reserveIndex < 128, Errors.UL_INVALID_INDEX);
        self.data =
            (self.data & ~(1 << (reserveIndex * 2))) |
            (uint256(borrowing ? 1 : 0) << (reserveIndex * 2));
    }

    /**
     * @dev Sets if the user is using as collateral the reserve identified by reserveIndex
     * @param self The configuration object
     * @param reserveIndex The index of the reserve in the bitmap
     * @param usingAsCollateral True if the user is usin the reserve as collateral, false otherwise
     **/
    function setUsingAsCollateral(
        DataTypes.UserConfigurationMap storage self,
        uint256 reserveIndex,
        bool usingAsCollateral
    ) internal {
        require(reserveIndex < 128, Errors.UL_INVALID_INDEX);
        self.data =
            (self.data & ~(1 << (reserveIndex * 2 + 1))) |
            (uint256(usingAsCollateral ? 1 : 0) << (reserveIndex * 2 + 1));
    }

    /**
     * @dev Used to validate if a user has been using the reserve for borrowing or as collateral
     * @param self The configuration object
     * @param reserveIndex The index of the reserve in the bitmap
     * @return True if the user has been using a reserve for borrowing or as collateral, false otherwise
     **/
    function isUsingAsCollateralOrBorrowing(
        DataTypes.UserConfigurationMap memory self,
        uint256 reserveIndex
    ) internal pure returns (bool) {
        require(reserveIndex < 128, Errors.UL_INVALID_INDEX);
        return (self.data >> (reserveIndex * 2)) & 3 != 0;
    }

    /**
     * @dev Used to validate if a user has been using the reserve for borrowing
     * @param self The configuration object
     * @param reserveIndex The index of the reserve in the bitmap
     * @return True if the user has been using a reserve for borrowing, false otherwise
     **/
    function isBorrowing(
        DataTypes.UserConfigurationMap memory self,
        uint256 reserveIndex
    ) internal pure returns (bool) {
        require(reserveIndex < 128, Errors.UL_INVALID_INDEX);
        return (self.data >> (reserveIndex * 2)) & 1 != 0;
    }

    /**
     * @dev Used to validate if a user has been using the reserve as collateral
     * @param self The configuration object
     * @param reserveIndex The index of the reserve in the bitmap
     * @return True if the user has been using a reserve as collateral, false otherwise
     **/
    function isUsingAsCollateral(
        DataTypes.UserConfigurationMap memory self,
        uint256 reserveIndex
    ) internal pure returns (bool) {
        require(reserveIndex < 128, Errors.UL_INVALID_INDEX);
        return (self.data >> (reserveIndex * 2 + 1)) & 1 != 0;
    }

    /**
     * @dev Used to validate if a user has been borrowing from any reserve
     * @param self The configuration object
     * @return True if the user has been borrowing any reserve, false otherwise
     **/
    function isBorrowingAny(DataTypes.UserConfigurationMap memory self)
        internal
        pure
        returns (bool)
    {
        return self.data & BORROWING_MASK != 0;
    }

    /**
     * @dev Used to validate if a user has not been using any reserve
     * @param self The configuration object
     * @return True if the user has been borrowing any reserve, false otherwise
     **/
    function isEmpty(DataTypes.UserConfigurationMap memory self)
        internal
        pure
        returns (bool)
    {
        return self.data == 0;
    }
}
          

contracts/protocol/libraries/helpers/Errors.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

/**
 * @title Errors library
 * @author Aave
 * @notice Defines the error messages emitted by the different contracts of the Aave protocol
 * @dev Error messages prefix glossary:
 *  - VL = ValidationLogic
 *  - AVL = Additional ValidationLogic
 *  - MATH = Math libraries
 *  - CT = Common errors between tokens (AToken, VariableDebtToken and StableDebtToken)
 *  - AT = AToken
 *  - SDT = StableDebtToken
 *  - VDT = VariableDebtToken
 *  - LP = LendingPool
 *  - LPAPR = LendingPoolAddressesProviderRegistry
 *  - LPC = LendingPoolConfiguration
 *  - RL = ReserveLogic
 *  - LPCM = LendingPoolCollateralManager
 *  - P = Pausable
 */
library Errors {
    //common errors
    string public constant CALLER_NOT_POOL_ADMIN = "33"; // 'The caller must be the pool admin'
    string public constant BORROW_ALLOWANCE_NOT_ENOUGH = "59"; // User borrows on behalf, but allowance are too small

    //contract specific errors
    string public constant VL_INVALID_AMOUNT = "1"; // 'Amount must be greater than 0'
    string public constant VL_NO_ACTIVE_RESERVE = "2"; // 'Action requires an active reserve'
    string public constant VL_RESERVE_FROZEN = "3"; // 'Action cannot be performed because the reserve is frozen'
    string public constant VL_CURRENT_AVAILABLE_LIQUIDITY_NOT_ENOUGH = "4"; // 'The current liquidity is not enough'
    string public constant VL_NOT_ENOUGH_AVAILABLE_USER_BALANCE = "5"; // 'User cannot withdraw more than the available balance'
    string public constant VL_TRANSFER_NOT_ALLOWED = "6"; // 'Transfer cannot be allowed.'
    string public constant VL_BORROWING_NOT_ENABLED = "7"; // 'Borrowing is not enabled'
    string public constant VL_INVALID_INTEREST_RATE_MODE_SELECTED = "8"; // 'Invalid interest rate mode selected'
    string public constant VL_COLLATERAL_BALANCE_IS_0 = "9"; // 'The collateral balance is 0'
    string public constant VL_HEALTH_FACTOR_LOWER_THAN_LIQUIDATION_THRESHOLD =
        "10"; // 'Health factor is lesser than the liquidation threshold'
    string public constant VL_COLLATERAL_CANNOT_COVER_NEW_BORROW = "11"; // 'There is not enough collateral to cover a new borrow'
    string public constant VL_STABLE_BORROWING_NOT_ENABLED = "12"; // stable borrowing not enabled
    string public constant VL_COLLATERAL_SAME_AS_BORROWING_CURRENCY = "13"; // collateral is (mostly) the same currency that is being borrowed
    string public constant VL_AMOUNT_BIGGER_THAN_MAX_LOAN_SIZE_STABLE = "14"; // 'The requested amount is greater than the max loan size in stable rate mode
    string public constant VL_NO_DEBT_OF_SELECTED_TYPE = "15"; // 'for repayment of stable debt, the user needs to have stable debt, otherwise, he needs to have variable debt'
    string public constant VL_NO_EXPLICIT_AMOUNT_TO_REPAY_ON_BEHALF = "16"; // 'To repay on behalf of an user an explicit amount to repay is needed'
    string public constant VL_NO_STABLE_RATE_LOAN_IN_RESERVE = "17"; // 'User does not have a stable rate loan in progress on this reserve'
    string public constant VL_NO_VARIABLE_RATE_LOAN_IN_RESERVE = "18"; // 'User does not have a variable rate loan in progress on this reserve'
    string public constant VL_UNDERLYING_BALANCE_NOT_GREATER_THAN_0 = "19"; // 'The underlying balance needs to be greater than 0'
    string public constant VL_DEPOSIT_ALREADY_IN_USE = "20"; // 'User deposit is already being used as collateral'
    string public constant LP_NOT_ENOUGH_STABLE_BORROW_BALANCE = "21"; // 'User does not have any stable rate loan for this reserve'
    string public constant LP_INTEREST_RATE_REBALANCE_CONDITIONS_NOT_MET = "22"; // 'Interest rate rebalance conditions were not met'
    string public constant LP_LIQUIDATION_CALL_FAILED = "23"; // 'Liquidation call failed'
    string public constant LP_NOT_ENOUGH_LIQUIDITY_TO_BORROW = "24"; // 'There is not enough liquidity available to borrow'
    string public constant LP_REQUESTED_AMOUNT_TOO_SMALL = "25"; // 'The requested amount is too small for a FlashLoan.'
    string public constant LP_INCONSISTENT_PROTOCOL_ACTUAL_BALANCE = "26"; // 'The actual balance of the protocol is inconsistent'
    string public constant LP_CALLER_NOT_LENDING_POOL_CONFIGURATOR = "27"; // 'The caller of the function is not the lending pool configurator'
    string public constant LP_INCONSISTENT_FLASHLOAN_PARAMS = "28";
    string public constant CT_CALLER_MUST_BE_LENDING_POOL = "29"; // 'The caller of this function must be a lending pool'
    string public constant CT_CANNOT_GIVE_ALLOWANCE_TO_HIMSELF = "30"; // 'User cannot give allowance to himself'
    string public constant CT_TRANSFER_AMOUNT_NOT_GT_0 = "31"; // 'Transferred amount needs to be greater than zero'
    string public constant RL_RESERVE_ALREADY_INITIALIZED = "32"; // 'Reserve has already been initialized'
    string public constant LPC_RESERVE_LIQUIDITY_NOT_0 = "34"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_ATOKEN_POOL_ADDRESS = "35"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_STABLE_DEBT_TOKEN_POOL_ADDRESS = "36"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_VARIABLE_DEBT_TOKEN_POOL_ADDRESS = "37"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_STABLE_DEBT_TOKEN_UNDERLYING_ADDRESS =
        "38"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_VARIABLE_DEBT_TOKEN_UNDERLYING_ADDRESS =
        "39"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_ADDRESSES_PROVIDER_ID = "40"; // 'The liquidity of the reserve needs to be 0'
    string public constant LPC_INVALID_CONFIGURATION = "75"; // 'Invalid risk parameters for the reserve'
    string public constant LPC_CALLER_NOT_EMERGENCY_ADMIN = "76"; // 'The caller must be the emergency admin'
    string public constant LPAPR_PROVIDER_NOT_REGISTERED = "41"; // 'Provider is not registered'
    string public constant LPCM_HEALTH_FACTOR_NOT_BELOW_THRESHOLD = "42"; // 'Health factor is not below the threshold'
    string public constant LPCM_COLLATERAL_CANNOT_BE_LIQUIDATED = "43"; // 'The collateral chosen cannot be liquidated'
    string public constant LPCM_SPECIFIED_CURRENCY_NOT_BORROWED_BY_USER = "44"; // 'User did not borrow the specified currency'
    string public constant LPCM_NOT_ENOUGH_LIQUIDITY_TO_LIQUIDATE = "45"; // "There isn't enough liquidity available to liquidate"
    string public constant LPCM_NO_ERRORS = "46"; // 'No errors'
    string public constant LP_INVALID_FLASHLOAN_MODE = "47"; //Invalid flashloan mode selected
    string public constant MATH_MULTIPLICATION_OVERFLOW = "48";
    string public constant MATH_ADDITION_OVERFLOW = "49";
    string public constant MATH_DIVISION_BY_ZERO = "50";
    string public constant RL_LIQUIDITY_INDEX_OVERFLOW = "51"; //  Liquidity index overflows uint128
    string public constant RL_VARIABLE_BORROW_INDEX_OVERFLOW = "52"; //  Variable borrow index overflows uint128
    string public constant RL_LIQUIDITY_RATE_OVERFLOW = "53"; //  Liquidity rate overflows uint128
    string public constant RL_VARIABLE_BORROW_RATE_OVERFLOW = "54"; //  Variable borrow rate overflows uint128
    string public constant RL_STABLE_BORROW_RATE_OVERFLOW = "55"; //  Stable borrow rate overflows uint128
    string public constant CT_INVALID_MINT_AMOUNT = "56"; //invalid amount to mint
    string public constant LP_FAILED_REPAY_WITH_COLLATERAL = "57";
    string public constant CT_INVALID_BURN_AMOUNT = "58"; //invalid amount to burn
    string public constant LP_FAILED_COLLATERAL_SWAP = "60";
    string public constant LP_INVALID_EQUAL_ASSETS_TO_SWAP = "61";
    string public constant LP_REENTRANCY_NOT_ALLOWED = "62";
    string public constant LP_CALLER_MUST_BE_AN_ATOKEN = "63";
    string public constant LP_IS_PAUSED = "64"; // 'Pool is paused'
    string public constant LP_NO_MORE_RESERVES_ALLOWED = "65";
    string public constant LP_INVALID_FLASH_LOAN_EXECUTOR_RETURN = "66";
    string public constant RC_INVALID_LTV = "67";
    string public constant RC_INVALID_LIQ_THRESHOLD = "68";
    string public constant RC_INVALID_LIQ_BONUS = "69";
    string public constant RC_INVALID_DECIMALS = "70";
    string public constant RC_INVALID_RESERVE_FACTOR = "71";
    string public constant LPAPR_INVALID_ADDRESSES_PROVIDER_ID = "72";
    string public constant VL_INCONSISTENT_FLASHLOAN_PARAMS = "73";
    string public constant LP_INCONSISTENT_PARAMS_LENGTH = "74";
    string public constant UL_INVALID_INDEX = "77";
    string public constant LP_NOT_CONTRACT = "78";
    string public constant SDT_STABLE_DEBT_OVERFLOW = "79";
    string public constant SDT_BURN_EXCEEDS_BALANCE = "80";

    string public constant AVL_EXCEED_MAX_GLOBAL_DEPOSIT_SIZE = "81";
    string public constant AVL_EXCEED_MAX_INDIVIDUAL_DEPOSIT_SIZE = "82";
    string public constant AVL_BELOW_MIN_INDIVIDUAL_DEPOSIT_SIZE = "83";
    string public constant AVL_EXCEED_MAX_GLOBAL_BORROW_SIZE = "84";
    string public constant AVL_EXCEED_MAX_INDIVIDUAL_BORROW_SIZE = "85";
    string public constant AVL_EXCEED_BORROW_MAX_PERCENTAGE = "86";
    string public constant AVL_INVALID_BORROW_MAX_PERCENTAGE = "87";

    string public constant LP_USER_IS_BLACKLISTED = "88";
    string public constant LP_USER_NOT_ELIGIBLE_FOR_BLACKLIST = "89";

    enum CollateralManagerErrors {
        NO_ERROR,
        NO_COLLATERAL_AVAILABLE,
        COLLATERAL_CANNOT_BE_LIQUIDATED,
        CURRRENCY_NOT_BORROWED,
        HEALTH_FACTOR_ABOVE_THRESHOLD,
        NOT_ENOUGH_LIQUIDITY,
        NO_ACTIVE_RESERVE,
        HEALTH_FACTOR_LOWER_THAN_LIQUIDATION_THRESHOLD,
        INVALID_EQUAL_ASSETS_TO_SWAP,
        FROZEN_RESERVE
    }
}
          

contracts/protocol/libraries/helpers/Helpers.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {IERC20} from "../../../dependencies/openzeppelin/contracts/IERC20.sol";
import {DataTypes} from "../types/DataTypes.sol";

/**
 * @title Helpers library
 * @author Aave
 */
library Helpers {
    /**
     * @dev Fetches the user current stable and variable debt balances
     * @param user The user address
     * @param reserve The reserve data object
     * @return The stable and variable debt balance
     **/
    function getUserCurrentDebt(
        address user,
        DataTypes.ReserveData storage reserve
    ) internal view returns (uint256, uint256) {
        return (
            IERC20(reserve.stableDebtTokenAddress).balanceOf(user),
            IERC20(reserve.variableDebtTokenAddress).balanceOf(user)
        );
    }

    function getUserCurrentDebtMemory(
        address user,
        DataTypes.ReserveData memory reserve
    ) internal view returns (uint256, uint256) {
        return (
            IERC20(reserve.stableDebtTokenAddress).balanceOf(user),
            IERC20(reserve.variableDebtTokenAddress).balanceOf(user)
        );
    }
}
          

contracts/protocol/libraries/logic/GenericLogic.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;
pragma experimental ABIEncoderV2;

import {SafeMath} from "../../../dependencies/openzeppelin/contracts/SafeMath.sol";
import {IERC20} from "../../../dependencies/openzeppelin/contracts/IERC20.sol";
import {ReserveLogic} from "./ReserveLogic.sol";
import {ReserveConfiguration} from "../configuration/ReserveConfiguration.sol";
import {UserConfiguration} from "../configuration/UserConfiguration.sol";
import {WadRayMath} from "../math/WadRayMath.sol";
import {PercentageMath} from "../math/PercentageMath.sol";
import {IPriceOracleGetter} from "../../../interfaces/IPriceOracleGetter.sol";
import {DataTypes} from "../types/DataTypes.sol";

/**
 * @title GenericLogic library
 * @author Aave
 * @title Implements protocol-level logic to calculate and validate the state of a user
 */
library GenericLogic {
    using ReserveLogic for DataTypes.ReserveData;
    using SafeMath for uint256;
    using WadRayMath for uint256;
    using PercentageMath for uint256;
    using ReserveConfiguration for DataTypes.ReserveConfigurationMap;
    using UserConfiguration for DataTypes.UserConfigurationMap;

    uint256 public constant HEALTH_FACTOR_LIQUIDATION_THRESHOLD = 1 ether;

    struct balanceDecreaseAllowedLocalVars {
        uint256 decimals;
        uint256 liquidationThreshold;
        uint256 totalCollateralInETH;
        uint256 totalDebtInETH;
        uint256 avgLiquidationThreshold;
        uint256 amountToDecreaseInETH;
        uint256 collateralBalanceAfterDecrease;
        uint256 liquidationThresholdAfterDecrease;
        uint256 healthFactorAfterDecrease;
        bool reserveUsageAsCollateralEnabled;
    }

    /**
     * @dev Checks if a specific balance decrease is allowed
     * (i.e. doesn't bring the user borrow position health factor under HEALTH_FACTOR_LIQUIDATION_THRESHOLD)
     * @param asset The address of the underlying asset of the reserve
     * @param user The address of the user
     * @param amount The amount to decrease
     * @param reservesData The data of all the reserves
     * @param userConfig The user configuration
     * @param reserves The list of all the active reserves
     * @param oracle The address of the oracle contract
     * @return true if the decrease of the balance is allowed
     **/
    function balanceDecreaseAllowed(
        address asset,
        address user,
        uint256 amount,
        mapping(address => DataTypes.ReserveData) storage reservesData,
        DataTypes.UserConfigurationMap calldata userConfig,
        mapping(uint256 => address) storage reserves,
        uint256 reservesCount,
        address oracle
    ) external view returns (bool) {
        if (
            !userConfig.isBorrowingAny() ||
            !userConfig.isUsingAsCollateral(reservesData[asset].id)
        ) {
            return true;
        }

        balanceDecreaseAllowedLocalVars memory vars;

        (, vars.liquidationThreshold, , vars.decimals, ) = reservesData[asset]
            .configuration
            .getParams();

        if (vars.liquidationThreshold == 0) {
            return true;
        }

        (
            vars.totalCollateralInETH,
            vars.totalDebtInETH,
            ,
            vars.avgLiquidationThreshold,

        ) = calculateUserAccountData(
            user,
            reservesData,
            userConfig,
            reserves,
            reservesCount,
            oracle
        );

        if (vars.totalDebtInETH == 0) {
            return true;
        }

        vars.amountToDecreaseInETH = IPriceOracleGetter(oracle)
            .getAssetPrice(asset)
            .mul(amount)
            .div(10**vars.decimals);

        vars.collateralBalanceAfterDecrease = vars.totalCollateralInETH.sub(
            vars.amountToDecreaseInETH
        );

        //if there is a borrow, there can't be 0 collateral
        if (vars.collateralBalanceAfterDecrease == 0) {
            return false;
        }

        vars.liquidationThresholdAfterDecrease = vars
            .totalCollateralInETH
            .mul(vars.avgLiquidationThreshold)
            .sub(vars.amountToDecreaseInETH.mul(vars.liquidationThreshold))
            .div(vars.collateralBalanceAfterDecrease);

        uint256 healthFactorAfterDecrease = calculateHealthFactorFromBalances(
            vars.collateralBalanceAfterDecrease,
            vars.totalDebtInETH,
            vars.liquidationThresholdAfterDecrease
        );

        return
            healthFactorAfterDecrease >=
            GenericLogic.HEALTH_FACTOR_LIQUIDATION_THRESHOLD;
    }

    struct CalculateUserAccountDataVars {
        uint256 reserveUnitPrice;
        uint256 tokenUnit;
        uint256 compoundedLiquidityBalance;
        uint256 compoundedBorrowBalance;
        uint256 decimals;
        uint256 ltv;
        uint256 liquidationThreshold;
        uint256 i;
        uint256 healthFactor;
        uint256 totalCollateralInETH;
        uint256 totalDebtInETH;
        uint256 avgLtv;
        uint256 avgLiquidationThreshold;
        uint256 reservesLength;
        bool healthFactorBelowThreshold;
        address currentReserveAddress;
        bool usageAsCollateralEnabled;
        bool userUsesReserveAsCollateral;
    }

    /**
     * @dev Calculates the user data across the reserves.
     * this includes the total liquidity/collateral/borrow balances in ETH,
     * the average Loan To Value, the average Liquidation Ratio, and the Health factor.
     * @param user The address of the user
     * @param reservesData Data of all the reserves
     * @param userConfig The configuration of the user
     * @param reserves The list of the available reserves
     * @param oracle The price oracle address
     * @return The total collateral and total debt of the user in ETH, the avg ltv, liquidation threshold and the HF
     **/
    function calculateUserAccountData(
        address user,
        mapping(address => DataTypes.ReserveData) storage reservesData,
        DataTypes.UserConfigurationMap memory userConfig,
        mapping(uint256 => address) storage reserves,
        uint256 reservesCount,
        address oracle
    )
        internal
        view
        returns (
            uint256,
            uint256,
            uint256,
            uint256,
            uint256
        )
    {
        CalculateUserAccountDataVars memory vars;

        if (userConfig.isEmpty()) {
            return (0, 0, 0, 0, uint256(-1));
        }
        for (vars.i = 0; vars.i < reservesCount; vars.i++) {
            if (!userConfig.isUsingAsCollateralOrBorrowing(vars.i)) {
                continue;
            }

            vars.currentReserveAddress = reserves[vars.i];
            DataTypes.ReserveData storage currentReserve = reservesData[
                vars.currentReserveAddress
            ];

            (
                vars.ltv,
                vars.liquidationThreshold,
                ,
                vars.decimals,

            ) = currentReserve.configuration.getParams();

            vars.tokenUnit = 10**vars.decimals;
            vars.reserveUnitPrice = IPriceOracleGetter(oracle).getAssetPrice(
                vars.currentReserveAddress
            );

            if (
                vars.liquidationThreshold != 0 &&
                userConfig.isUsingAsCollateral(vars.i)
            ) {
                vars.compoundedLiquidityBalance = IERC20(
                    currentReserve.aTokenAddress
                ).balanceOf(user);

                uint256 liquidityBalanceETH = vars
                    .reserveUnitPrice
                    .mul(vars.compoundedLiquidityBalance)
                    .div(vars.tokenUnit);

                vars.totalCollateralInETH = vars.totalCollateralInETH.add(
                    liquidityBalanceETH
                );

                vars.avgLtv = vars.avgLtv.add(
                    liquidityBalanceETH.mul(vars.ltv)
                );
                vars.avgLiquidationThreshold = vars.avgLiquidationThreshold.add(
                    liquidityBalanceETH.mul(vars.liquidationThreshold)
                );
            }

            if (userConfig.isBorrowing(vars.i)) {
                vars.compoundedBorrowBalance = IERC20(
                    currentReserve.stableDebtTokenAddress
                ).balanceOf(user);
                vars.compoundedBorrowBalance = vars.compoundedBorrowBalance.add(
                    IERC20(currentReserve.variableDebtTokenAddress).balanceOf(
                        user
                    )
                );

                vars.totalDebtInETH = vars.totalDebtInETH.add(
                    vars.reserveUnitPrice.mul(vars.compoundedBorrowBalance).div(
                        vars.tokenUnit
                    )
                );
            }
        }

        vars.avgLtv = vars.totalCollateralInETH > 0
            ? vars.avgLtv.div(vars.totalCollateralInETH)
            : 0;
        vars.avgLiquidationThreshold = vars.totalCollateralInETH > 0
            ? vars.avgLiquidationThreshold.div(vars.totalCollateralInETH)
            : 0;

        vars.healthFactor = calculateHealthFactorFromBalances(
            vars.totalCollateralInETH,
            vars.totalDebtInETH,
            vars.avgLiquidationThreshold
        );
        return (
            vars.totalCollateralInETH,
            vars.totalDebtInETH,
            vars.avgLtv,
            vars.avgLiquidationThreshold,
            vars.healthFactor
        );
    }

    /**
     * @dev Calculates the health factor from the corresponding balances
     * @param totalCollateralInETH The total collateral in ETH
     * @param totalDebtInETH The total debt in ETH
     * @param liquidationThreshold The avg liquidation threshold
     * @return The health factor calculated from the balances provided
     **/
    function calculateHealthFactorFromBalances(
        uint256 totalCollateralInETH,
        uint256 totalDebtInETH,
        uint256 liquidationThreshold
    ) internal pure returns (uint256) {
        if (totalDebtInETH == 0) return uint256(-1);

        return
            (totalCollateralInETH.percentMul(liquidationThreshold)).wadDiv(
                totalDebtInETH
            );
    }

    /**
     * @dev Calculates the equivalent amount in ETH that an user can borrow, depending on the available collateral and the
     * average Loan To Value
     * @param totalCollateralInETH The total collateral in ETH
     * @param totalDebtInETH The total borrow balance
     * @param ltv The average loan to value
     * @return the amount available to borrow in ETH for the user
     **/

    function calculateAvailableBorrowsETH(
        uint256 totalCollateralInETH,
        uint256 totalDebtInETH,
        uint256 ltv
    ) internal pure returns (uint256) {
        uint256 availableBorrowsETH = totalCollateralInETH.percentMul(ltv);

        if (availableBorrowsETH < totalDebtInETH) {
            return 0;
        }

        availableBorrowsETH = availableBorrowsETH.sub(totalDebtInETH);
        return availableBorrowsETH;
    }
}
          

contracts/protocol/libraries/logic/ReserveLogic.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {SafeMath} from "../../../dependencies/openzeppelin/contracts/SafeMath.sol";
import {IERC20} from "../../../dependencies/openzeppelin/contracts/IERC20.sol";
import {SafeERC20} from "../../../dependencies/openzeppelin/contracts/SafeERC20.sol";
import {IAToken} from "../../../interfaces/IAToken.sol";
import {IStableDebtToken} from "../../../interfaces/IStableDebtToken.sol";
import {IVariableDebtToken} from "../../../interfaces/IVariableDebtToken.sol";
import {IReserveInterestRateStrategy} from "../../../interfaces/IReserveInterestRateStrategy.sol";
import {ReserveConfiguration} from "../configuration/ReserveConfiguration.sol";
import {MathUtils} from "../math/MathUtils.sol";
import {WadRayMath} from "../math/WadRayMath.sol";
import {PercentageMath} from "../math/PercentageMath.sol";
import {Errors} from "../helpers/Errors.sol";
import {DataTypes} from "../types/DataTypes.sol";

/**
 * @title ReserveLogic library
 * @author Aave
 * @notice Implements the logic to update the reserves state
 */
library ReserveLogic {
    using SafeMath for uint256;
    using WadRayMath for uint256;
    using PercentageMath for uint256;
    using SafeERC20 for IERC20;

    /**
     * @dev Emitted when the state of a reserve is updated
     * @param asset The address of the underlying asset of the reserve
     * @param liquidityRate The new liquidity rate
     * @param stableBorrowRate The new stable borrow rate
     * @param variableBorrowRate The new variable borrow rate
     * @param liquidityIndex The new liquidity index
     * @param variableBorrowIndex The new variable borrow index
     **/
    event ReserveDataUpdated(
        address indexed asset,
        uint256 liquidityRate,
        uint256 stableBorrowRate,
        uint256 variableBorrowRate,
        uint256 liquidityIndex,
        uint256 variableBorrowIndex
    );

    using ReserveLogic for DataTypes.ReserveData;
    using ReserveConfiguration for DataTypes.ReserveConfigurationMap;

    /**
     * @dev Returns the ongoing normalized income for the reserve
     * A value of 1e27 means there is no income. As time passes, the income is accrued
     * A value of 2*1e27 means for each unit of asset one unit of income has been accrued
     * @param reserve The reserve object
     * @return the normalized income. expressed in ray
     **/
    function getNormalizedIncome(DataTypes.ReserveData storage reserve)
        internal
        view
        returns (uint256)
    {
        uint40 timestamp = reserve.lastUpdateTimestamp;

        //solium-disable-next-line
        if (timestamp == uint40(block.timestamp)) {
            //if the index was updated in the same block, no need to perform any calculation
            return reserve.liquidityIndex;
        }

        uint256 cumulated = MathUtils
            .calculateLinearInterest(reserve.currentLiquidityRate, timestamp)
            .rayMul(reserve.liquidityIndex);

        return cumulated;
    }

    /**
     * @dev Returns the ongoing normalized variable debt for the reserve
     * A value of 1e27 means there is no debt. As time passes, the income is accrued
     * A value of 2*1e27 means that for each unit of debt, one unit worth of interest has been accumulated
     * @param reserve The reserve object
     * @return The normalized variable debt. expressed in ray
     **/
    function getNormalizedDebt(DataTypes.ReserveData storage reserve)
        internal
        view
        returns (uint256)
    {
        uint40 timestamp = reserve.lastUpdateTimestamp;

        //solium-disable-next-line
        if (timestamp == uint40(block.timestamp)) {
            //if the index was updated in the same block, no need to perform any calculation
            return reserve.variableBorrowIndex;
        }

        uint256 cumulated = MathUtils
            .calculateCompoundedInterest(
                reserve.currentVariableBorrowRate,
                timestamp
            )
            .rayMul(reserve.variableBorrowIndex);

        return cumulated;
    }

    /**
     * @dev Updates the liquidity cumulative index and the variable borrow index.
     * @param reserve the reserve object
     **/
    function updateState(DataTypes.ReserveData storage reserve) internal {
        uint256 scaledVariableDebt = IVariableDebtToken(
            reserve.variableDebtTokenAddress
        ).scaledTotalSupply();
        uint256 previousVariableBorrowIndex = reserve.variableBorrowIndex;
        uint256 previousLiquidityIndex = reserve.liquidityIndex;
        uint40 lastUpdatedTimestamp = reserve.lastUpdateTimestamp;

        (
            uint256 newLiquidityIndex,
            uint256 newVariableBorrowIndex
        ) = _updateIndexes(
                reserve,
                scaledVariableDebt,
                previousLiquidityIndex,
                previousVariableBorrowIndex,
                lastUpdatedTimestamp
            );

        _mintToTreasury(
            reserve,
            scaledVariableDebt,
            previousVariableBorrowIndex,
            newLiquidityIndex,
            newVariableBorrowIndex,
            lastUpdatedTimestamp
        );
    }

    /**
     * @dev Accumulates a predefined amount of asset to the reserve as a fixed, instantaneous income. Used for example to accumulate
     * the flashloan fee to the reserve, and spread it between all the depositors
     * @param reserve The reserve object
     * @param totalLiquidity The total liquidity available in the reserve
     * @param amount The amount to accomulate
     **/
    function cumulateToLiquidityIndex(
        DataTypes.ReserveData storage reserve,
        uint256 totalLiquidity,
        uint256 amount
    ) internal {
        uint256 amountToLiquidityRatio = amount.wadToRay().rayDiv(
            totalLiquidity.wadToRay()
        );

        uint256 result = amountToLiquidityRatio.add(WadRayMath.ray());

        result = result.rayMul(reserve.liquidityIndex);
        require(
            result <= type(uint128).max,
            Errors.RL_LIQUIDITY_INDEX_OVERFLOW
        );

        reserve.liquidityIndex = uint128(result);
    }

    /**
     * @dev Initializes a reserve
     * @param reserve The reserve object
     * @param aTokenAddress The address of the overlying atoken contract
     * @param interestRateStrategyAddress The address of the interest rate strategy contract
     **/
    function init(
        DataTypes.ReserveData storage reserve,
        address aTokenAddress,
        address stableDebtTokenAddress,
        address variableDebtTokenAddress,
        address interestRateStrategyAddress
    ) external {
        require(
            reserve.aTokenAddress == address(0),
            Errors.RL_RESERVE_ALREADY_INITIALIZED
        );

        reserve.liquidityIndex = uint128(WadRayMath.ray());
        reserve.variableBorrowIndex = uint128(WadRayMath.ray());
        reserve.aTokenAddress = aTokenAddress;
        reserve.stableDebtTokenAddress = stableDebtTokenAddress;
        reserve.variableDebtTokenAddress = variableDebtTokenAddress;
        reserve.interestRateStrategyAddress = interestRateStrategyAddress;
    }

    struct UpdateInterestRatesLocalVars {
        address stableDebtTokenAddress;
        uint256 availableLiquidity;
        uint256 totalStableDebt;
        uint256 newLiquidityRate;
        uint256 newStableRate;
        uint256 newVariableRate;
        uint256 avgStableRate;
        uint256 totalVariableDebt;
    }

    /**
     * @dev Updates the reserve current stable borrow rate, the current variable borrow rate and the current liquidity rate
     * @param reserve The address of the reserve to be updated
     * @param liquidityAdded The amount of liquidity added to the protocol (deposit or repay) in the previous action
     * @param liquidityTaken The amount of liquidity taken from the protocol (redeem or borrow)
     **/
    function updateInterestRates(
        DataTypes.ReserveData storage reserve,
        address reserveAddress,
        address aTokenAddress,
        uint256 liquidityAdded,
        uint256 liquidityTaken
    ) internal {
        UpdateInterestRatesLocalVars memory vars;

        vars.stableDebtTokenAddress = reserve.stableDebtTokenAddress;

        (vars.totalStableDebt, vars.avgStableRate) = IStableDebtToken(
            vars.stableDebtTokenAddress
        ).getTotalSupplyAndAvgRate();

        //calculates the total variable debt locally using the scaled total supply instead
        //of totalSupply(), as it's noticeably cheaper. Also, the index has been
        //updated by the previous updateState() call
        vars.totalVariableDebt = IVariableDebtToken(
            reserve.variableDebtTokenAddress
        ).scaledTotalSupply().rayMul(reserve.variableBorrowIndex);

        (
            vars.newLiquidityRate,
            vars.newStableRate,
            vars.newVariableRate
        ) = IReserveInterestRateStrategy(reserve.interestRateStrategyAddress)
            .calculateInterestRates(
                reserveAddress,
                aTokenAddress,
                liquidityAdded,
                liquidityTaken,
                vars.totalStableDebt,
                vars.totalVariableDebt,
                vars.avgStableRate,
                reserve.configuration.getReserveFactor()
            );
        require(
            vars.newLiquidityRate <= type(uint128).max,
            Errors.RL_LIQUIDITY_RATE_OVERFLOW
        );
        require(
            vars.newStableRate <= type(uint128).max,
            Errors.RL_STABLE_BORROW_RATE_OVERFLOW
        );
        require(
            vars.newVariableRate <= type(uint128).max,
            Errors.RL_VARIABLE_BORROW_RATE_OVERFLOW
        );

        reserve.currentLiquidityRate = uint128(vars.newLiquidityRate);
        reserve.currentStableBorrowRate = uint128(vars.newStableRate);
        reserve.currentVariableBorrowRate = uint128(vars.newVariableRate);

        emit ReserveDataUpdated(
            reserveAddress,
            vars.newLiquidityRate,
            vars.newStableRate,
            vars.newVariableRate,
            reserve.liquidityIndex,
            reserve.variableBorrowIndex
        );
    }

    struct MintToTreasuryLocalVars {
        uint256 currentStableDebt;
        uint256 principalStableDebt;
        uint256 previousStableDebt;
        uint256 currentVariableDebt;
        uint256 previousVariableDebt;
        uint256 avgStableRate;
        uint256 cumulatedStableInterest;
        uint256 totalDebtAccrued;
        uint256 amountToMint;
        uint256 reserveFactor;
        uint40 stableSupplyUpdatedTimestamp;
    }

    /**
     * @dev Mints part of the repaid interest to the reserve treasury as a function of the reserveFactor for the
     * specific asset.
     * @param reserve The reserve reserve to be updated
     * @param scaledVariableDebt The current scaled total variable debt
     * @param previousVariableBorrowIndex The variable borrow index before the last accumulation of the interest
     * @param newLiquidityIndex The new liquidity index
     * @param newVariableBorrowIndex The variable borrow index after the last accumulation of the interest
     **/
    function _mintToTreasury(
        DataTypes.ReserveData storage reserve,
        uint256 scaledVariableDebt,
        uint256 previousVariableBorrowIndex,
        uint256 newLiquidityIndex,
        uint256 newVariableBorrowIndex,
        uint40 timestamp
    ) internal {
        MintToTreasuryLocalVars memory vars;

        vars.reserveFactor = reserve.configuration.getReserveFactor();

        if (vars.reserveFactor == 0) {
            return;
        }

        //fetching the principal, total stable debt and the avg stable rate
        (
            vars.principalStableDebt,
            vars.currentStableDebt,
            vars.avgStableRate,
            vars.stableSupplyUpdatedTimestamp
        ) = IStableDebtToken(reserve.stableDebtTokenAddress).getSupplyData();

        //calculate the last principal variable debt
        vars.previousVariableDebt = scaledVariableDebt.rayMul(
            previousVariableBorrowIndex
        );

        //calculate the new total supply after accumulation of the index
        vars.currentVariableDebt = scaledVariableDebt.rayMul(
            newVariableBorrowIndex
        );

        //calculate the stable debt until the last timestamp update
        vars.cumulatedStableInterest = MathUtils.calculateCompoundedInterest(
            vars.avgStableRate,
            vars.stableSupplyUpdatedTimestamp,
            timestamp
        );

        vars.previousStableDebt = vars.principalStableDebt.rayMul(
            vars.cumulatedStableInterest
        );

        //debt accrued is the sum of the current debt minus the sum of the debt at the last update
        vars.totalDebtAccrued = vars
            .currentVariableDebt
            .add(vars.currentStableDebt)
            .sub(vars.previousVariableDebt)
            .sub(vars.previousStableDebt);

        vars.amountToMint = vars.totalDebtAccrued.percentMul(
            vars.reserveFactor
        );

        if (vars.amountToMint != 0) {
            IAToken(reserve.aTokenAddress).mintToTreasury(
                vars.amountToMint,
                newLiquidityIndex
            );
        }
    }

    /**
     * @dev Updates the reserve indexes and the timestamp of the update
     * @param reserve The reserve reserve to be updated
     * @param scaledVariableDebt The scaled variable debt
     * @param liquidityIndex The last stored liquidity index
     * @param variableBorrowIndex The last stored variable borrow index
     **/
    function _updateIndexes(
        DataTypes.ReserveData storage reserve,
        uint256 scaledVariableDebt,
        uint256 liquidityIndex,
        uint256 variableBorrowIndex,
        uint40 timestamp
    ) internal returns (uint256, uint256) {
        uint256 currentLiquidityRate = reserve.currentLiquidityRate;

        uint256 newLiquidityIndex = liquidityIndex;
        uint256 newVariableBorrowIndex = variableBorrowIndex;

        //only cumulating if there is any income being produced
        if (currentLiquidityRate > 0) {
            uint256 cumulatedLiquidityInterest = MathUtils
                .calculateLinearInterest(currentLiquidityRate, timestamp);
            newLiquidityIndex = cumulatedLiquidityInterest.rayMul(
                liquidityIndex
            );
            require(
                newLiquidityIndex <= type(uint128).max,
                Errors.RL_LIQUIDITY_INDEX_OVERFLOW
            );

            reserve.liquidityIndex = uint128(newLiquidityIndex);

            //as the liquidity rate might come only from stable rate loans, we need to ensure
            //that there is actual variable debt before accumulating
            if (scaledVariableDebt != 0) {
                uint256 cumulatedVariableBorrowInterest = MathUtils
                    .calculateCompoundedInterest(
                        reserve.currentVariableBorrowRate,
                        timestamp
                    );
                newVariableBorrowIndex = cumulatedVariableBorrowInterest.rayMul(
                        variableBorrowIndex
                    );
                require(
                    newVariableBorrowIndex <= type(uint128).max,
                    Errors.RL_VARIABLE_BORROW_INDEX_OVERFLOW
                );
                reserve.variableBorrowIndex = uint128(newVariableBorrowIndex);
            }
        }

        //solium-disable-next-line
        reserve.lastUpdateTimestamp = uint40(block.timestamp);
        return (newLiquidityIndex, newVariableBorrowIndex);
    }
}
          

contracts/protocol/libraries/logic/ValidationLogic.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;
pragma experimental ABIEncoderV2;

import {SafeMath} from "../../../dependencies/openzeppelin/contracts/SafeMath.sol";
import {IERC20} from "../../../dependencies/openzeppelin/contracts/IERC20.sol";
import {ReserveLogic} from "./ReserveLogic.sol";
import {GenericLogic} from "./GenericLogic.sol";
import {WadRayMath} from "../math/WadRayMath.sol";
import {PercentageMath} from "../math/PercentageMath.sol";
import {SafeERC20} from "../../../dependencies/openzeppelin/contracts/SafeERC20.sol";
import {ReserveConfiguration} from "../configuration/ReserveConfiguration.sol";
import {UserConfiguration} from "../configuration/UserConfiguration.sol";
import {Errors} from "../helpers/Errors.sol";
import {Helpers} from "../helpers/Helpers.sol";
import {IAToken} from "../../../interfaces/IAToken.sol";
import {IPriceOracleGetter} from "../../../interfaces/IPriceOracleGetter.sol";
import {IReserveInterestRateStrategy} from "../../../interfaces/IReserveInterestRateStrategy.sol";
import {DataTypes} from "../types/DataTypes.sol";

/**
 * @title ReserveLogic library
 * @author Aave
 * @notice Implements functions to validate the different actions of the protocol
 */
library ValidationLogic {
    using ReserveLogic for DataTypes.ReserveData;
    using SafeMath for uint256;
    using WadRayMath for uint256;
    using PercentageMath for uint256;
    using SafeERC20 for IERC20;
    using ReserveConfiguration for DataTypes.ReserveConfigurationMap;
    using UserConfiguration for DataTypes.UserConfigurationMap;

    uint256 public constant REBALANCE_UP_LIQUIDITY_RATE_THRESHOLD = 4000;
    uint256 public constant REBALANCE_UP_USAGE_RATIO_THRESHOLD = 0.95 * 1e27; //usage ratio of 95%

    /**
     * @dev Validates a deposit action
     * @param reserve The reserve object on which the user is depositing
     * @param reserveLimits The reserve limits object on which the user is depositing
     * @param userAddress The address of the user
     * @param amount The amount to be deposited
     */
    function validateDeposit(
        DataTypes.ReserveData storage reserve,
        DataTypes.ReserveLimits storage reserveLimits,
        address userAddress,
        uint256 amount
    ) external view {
        (bool isActive, bool isFrozen, , ) = reserve.configuration.getFlags();

        require(amount != 0, Errors.VL_INVALID_AMOUNT);
        require(isActive, Errors.VL_NO_ACTIVE_RESERVE);
        require(!isFrozen, Errors.VL_RESERVE_FROZEN);

        if (
            reserveLimits.maxGlobalDepositSize > 0 ||
            reserveLimits.maxIndividualDepositSize > 0 ||
            reserveLimits.minIndividualDepositSize > 0
        ) {
            IERC20 aToken = IERC20(reserve.aTokenAddress);

            if (reserveLimits.maxGlobalDepositSize > 0) {
                require(
                    aToken.totalSupply().add(amount) <=
                        reserveLimits.maxGlobalDepositSize,
                    Errors.AVL_EXCEED_MAX_GLOBAL_DEPOSIT_SIZE
                );
            }
            if (
                reserveLimits.maxIndividualDepositSize > 0 ||
                reserveLimits.minIndividualDepositSize > 0
            ) {
                uint256 userATokenBalance = aToken.balanceOf(userAddress).add(
                    amount
                );
                if (reserveLimits.maxIndividualDepositSize > 0) {
                    require(
                        userATokenBalance <=
                            reserveLimits.maxIndividualDepositSize,
                        Errors.AVL_EXCEED_MAX_INDIVIDUAL_DEPOSIT_SIZE
                    );
                }
                if (reserveLimits.minIndividualDepositSize > 0) {
                    require(
                        userATokenBalance >=
                            reserveLimits.minIndividualDepositSize,
                        Errors.AVL_BELOW_MIN_INDIVIDUAL_DEPOSIT_SIZE
                    );
                }
            }
        }
    }

    /**
     * @dev Validates a withdraw action
     * @param reserveAddress The address of the reserve
     * @param amount The amount to be withdrawn
     * @param userBalance The balance of the user
     * @param reservesData The reserves state
     * @param reserveLimits The reserve limits object on which the user is withdrawing
     * @param userConfig The user configuration
     * @param reserves The addresses of the reserves
     * @param reservesCount The number of reserves
     * @param oracle The price oracle
     */
    function validateWithdraw(
        address reserveAddress,
        uint256 amount,
        uint256 userBalance,
        mapping(address => DataTypes.ReserveData) storage reservesData,
        DataTypes.ReserveLimits storage reserveLimits,
        DataTypes.UserConfigurationMap storage userConfig,
        mapping(uint256 => address) storage reserves,
        uint256 reservesCount,
        address oracle
    ) external view {
        require(amount != 0, Errors.VL_INVALID_AMOUNT);
        require(
            amount <= userBalance,
            Errors.VL_NOT_ENOUGH_AVAILABLE_USER_BALANCE
        );

        (bool isActive, , , ) = reservesData[reserveAddress]
            .configuration
            .getFlags();
        require(isActive, Errors.VL_NO_ACTIVE_RESERVE);

        if (
            userBalance > amount && reserveLimits.minIndividualDepositSize > 0
        ) {
            require(
                userBalance.sub(amount) >=
                    reserveLimits.minIndividualDepositSize,
                Errors.AVL_BELOW_MIN_INDIVIDUAL_DEPOSIT_SIZE
            );
        }

        require(
            GenericLogic.balanceDecreaseAllowed(
                reserveAddress,
                msg.sender,
                amount,
                reservesData,
                userConfig,
                reserves,
                reservesCount,
                oracle
            ),
            Errors.VL_TRANSFER_NOT_ALLOWED
        );
    }

    struct ValidateBorrowLocalVars {
        uint256 currentLtv;
        uint256 currentLiquidationThreshold;
        uint256 amountOfCollateralNeededETH;
        uint256 userCollateralBalanceETH;
        uint256 userBorrowBalanceETH;
        uint256 availableLiquidity;
        uint256 healthFactor;
        bool isActive;
        bool isFrozen;
        bool borrowingEnabled;
        bool stableRateBorrowingEnabled;
    }

    /**
     * @dev Validates a borrow action
     * @param asset The address of the asset to borrow
     * @param reserve The reserve state from which the user is borrowing
     * @param userAddress The address of the user
     * @param amount The amount to be borrowed
     * @param amountInETH The amount to be borrowed, in ETH
     * @param interestRateMode The interest rate mode at which the user is borrowing
     * @param maxStableLoanPercent The max amount of the liquidity that can be borrowed at stable rate, in percentage
     * @param reservesData The state of all the reserves
     * @param userConfig The state of the user for the specific reserve
     * @param reserves The addresses of all the active reserves
     * @param oracle The price oracle
     */
    function validateBorrow(
        address asset,
        DataTypes.ReserveData storage reserve,
        address userAddress,
        uint256 amount,
        uint256 amountInETH,
        uint256 interestRateMode,
        uint256 maxStableLoanPercent,
        mapping(address => DataTypes.ReserveData) storage reservesData,
        DataTypes.UserConfigurationMap storage userConfig,
        mapping(uint256 => address) storage reserves,
        uint256 reservesCount,
        address oracle
    ) external view {
        ValidateBorrowLocalVars memory vars;

        (
            vars.isActive,
            vars.isFrozen,
            vars.borrowingEnabled,
            vars.stableRateBorrowingEnabled
        ) = reserve.configuration.getFlags();

        require(vars.isActive, Errors.VL_NO_ACTIVE_RESERVE);
        require(!vars.isFrozen, Errors.VL_RESERVE_FROZEN);
        require(amount != 0, Errors.VL_INVALID_AMOUNT);

        require(vars.borrowingEnabled, Errors.VL_BORROWING_NOT_ENABLED);

        //validate interest rate mode
        require(
            uint256(DataTypes.InterestRateMode.VARIABLE) == interestRateMode ||
                uint256(DataTypes.InterestRateMode.STABLE) == interestRateMode,
            Errors.VL_INVALID_INTEREST_RATE_MODE_SELECTED
        );

        (
            vars.userCollateralBalanceETH,
            vars.userBorrowBalanceETH,
            vars.currentLtv,
            vars.currentLiquidationThreshold,
            vars.healthFactor
        ) = GenericLogic.calculateUserAccountData(
            userAddress,
            reservesData,
            userConfig,
            reserves,
            reservesCount,
            oracle
        );

        require(
            vars.userCollateralBalanceETH > 0,
            Errors.VL_COLLATERAL_BALANCE_IS_0
        );

        require(
            vars.healthFactor >
                GenericLogic.HEALTH_FACTOR_LIQUIDATION_THRESHOLD,
            Errors.VL_HEALTH_FACTOR_LOWER_THAN_LIQUIDATION_THRESHOLD
        );

        //add the current already borrowed amount to the amount requested to calculate the total collateral needed.
        vars.amountOfCollateralNeededETH = vars
            .userBorrowBalanceETH
            .add(amountInETH)
            .percentDiv(vars.currentLtv); //LTV is calculated in percentage

        require(
            vars.amountOfCollateralNeededETH <= vars.userCollateralBalanceETH,
            Errors.VL_COLLATERAL_CANNOT_COVER_NEW_BORROW
        );

        /**
         * Following conditions need to be met if the user is borrowing at a stable rate:
         * 1. Reserve must be enabled for stable rate borrowing
         * 2. Users cannot borrow from the reserve if their collateral is (mostly) the same currency
         *    they are borrowing, to prevent abuses.
         * 3. Users will be able to borrow only a portion of the total available liquidity
         **/

        if (interestRateMode == uint256(DataTypes.InterestRateMode.STABLE)) {
            //check if the borrow mode is stable and if stable rate borrowing is enabled on this reserve

            require(
                vars.stableRateBorrowingEnabled,
                Errors.VL_STABLE_BORROWING_NOT_ENABLED
            );

            require(
                !userConfig.isUsingAsCollateral(reserve.id) ||
                    reserve.configuration.getLtv() == 0 ||
                    amount >
                    IERC20(reserve.aTokenAddress).balanceOf(userAddress),
                Errors.VL_COLLATERAL_SAME_AS_BORROWING_CURRENCY
            );

            vars.availableLiquidity = IERC20(asset).balanceOf(
                reserve.aTokenAddress
            );

            //calculate the max available loan size in stable rate mode as a percentage of the
            //available liquidity
            uint256 maxLoanSizeStable = vars.availableLiquidity.percentMul(
                maxStableLoanPercent
            );

            require(
                amount <= maxLoanSizeStable,
                Errors.VL_AMOUNT_BIGGER_THAN_MAX_LOAN_SIZE_STABLE
            );
        }
    }

    /**
     * @dev Validates limits of a borrow action
     * @param reserve The reserve state from which the user is borrowing
     * @param reserveLimits The limits of reserve from which the user is borrowing
     * @param userAddress The address of the user
     * @param amount The amount to be borrowed
     */
    function validateBorrowLimits(
        DataTypes.ReserveData storage reserve,
        DataTypes.ReserveLimits storage reserveLimits,
        address userAddress,
        uint256 amount
    ) external view {
        if (
            reserveLimits.maxGlobalBorrowSize > 0 ||
            reserveLimits.maxIndividualBorrowSize > 0 ||
            reserveLimits.maxBorrowBps > 0
        ) {
            IERC20 stableDebtToken = IERC20(reserve.stableDebtTokenAddress);
            IERC20 variableDebtToken = IERC20(reserve.variableDebtTokenAddress);

            if (reserveLimits.maxGlobalBorrowSize > 0) {
                require(
                    stableDebtToken
                        .totalSupply()
                        .add(variableDebtToken.totalSupply())
                        .add(amount) <= reserveLimits.maxGlobalBorrowSize,
                    Errors.AVL_EXCEED_MAX_GLOBAL_BORROW_SIZE
                );
            }
            if (reserveLimits.maxIndividualBorrowSize > 0) {
                require(
                    stableDebtToken
                        .balanceOf(userAddress)
                        .add(variableDebtToken.balanceOf(userAddress))
                        .add(amount) <= reserveLimits.maxIndividualBorrowSize,
                    Errors.AVL_EXCEED_MAX_INDIVIDUAL_BORROW_SIZE
                );
            }
            if (reserveLimits.maxBorrowBps > 0) {
                require(
                    stableDebtToken
                        .totalSupply()
                        .add(variableDebtToken.totalSupply())
                        .add(amount)
                        .mul(10000) <=
                        IERC20(reserve.aTokenAddress).totalSupply().mul(
                            reserveLimits.maxBorrowBps
                        ),
                    Errors.AVL_EXCEED_BORROW_MAX_PERCENTAGE
                );
            }
        }
    }

    /**
     * @dev Validates a repay action
     * @param reserve The reserve state from which the user is repaying
     * @param amountSent The amount sent for the repayment. Can be an actual value or uint(-1)
     * @param onBehalfOf The address of the user msg.sender is repaying for
     * @param stableDebt The borrow balance of the user
     * @param variableDebt The borrow balance of the user
     */
    function validateRepay(
        DataTypes.ReserveData storage reserve,
        uint256 amountSent,
        DataTypes.InterestRateMode rateMode,
        address onBehalfOf,
        uint256 stableDebt,
        uint256 variableDebt
    ) external view {
        bool isActive = reserve.configuration.getActive();

        require(isActive, Errors.VL_NO_ACTIVE_RESERVE);

        require(amountSent > 0, Errors.VL_INVALID_AMOUNT);

        require(
            (stableDebt > 0 &&
                DataTypes.InterestRateMode(rateMode) ==
                DataTypes.InterestRateMode.STABLE) ||
                (variableDebt > 0 &&
                    DataTypes.InterestRateMode(rateMode) ==
                    DataTypes.InterestRateMode.VARIABLE),
            Errors.VL_NO_DEBT_OF_SELECTED_TYPE
        );

        require(
            amountSent != uint256(-1) || msg.sender == onBehalfOf,
            Errors.VL_NO_EXPLICIT_AMOUNT_TO_REPAY_ON_BEHALF
        );
    }

    /**
     * @dev Validates a swap of borrow rate mode.
     * @param reserve The reserve state on which the user is swapping the rate
     * @param userConfig The user reserves configuration
     * @param stableDebt The stable debt of the user
     * @param variableDebt The variable debt of the user
     * @param currentRateMode The rate mode of the borrow
     */
    function validateSwapRateMode(
        DataTypes.ReserveData storage reserve,
        DataTypes.UserConfigurationMap storage userConfig,
        uint256 stableDebt,
        uint256 variableDebt,
        DataTypes.InterestRateMode currentRateMode
    ) external view {
        (bool isActive, bool isFrozen, , bool stableRateEnabled) = reserve
            .configuration
            .getFlags();

        require(isActive, Errors.VL_NO_ACTIVE_RESERVE);
        require(!isFrozen, Errors.VL_RESERVE_FROZEN);

        if (currentRateMode == DataTypes.InterestRateMode.STABLE) {
            require(stableDebt > 0, Errors.VL_NO_STABLE_RATE_LOAN_IN_RESERVE);
        } else if (currentRateMode == DataTypes.InterestRateMode.VARIABLE) {
            require(
                variableDebt > 0,
                Errors.VL_NO_VARIABLE_RATE_LOAN_IN_RESERVE
            );
            /**
             * user wants to swap to stable, before swapping we need to ensure that
             * 1. stable borrow rate is enabled on the reserve
             * 2. user is not trying to abuse the reserve by depositing
             * more collateral than he is borrowing, artificially lowering
             * the interest rate, borrowing at variable, and switching to stable
             **/
            require(stableRateEnabled, Errors.VL_STABLE_BORROWING_NOT_ENABLED);

            require(
                !userConfig.isUsingAsCollateral(reserve.id) ||
                    reserve.configuration.getLtv() == 0 ||
                    stableDebt.add(variableDebt) >
                    IERC20(reserve.aTokenAddress).balanceOf(msg.sender),
                Errors.VL_COLLATERAL_SAME_AS_BORROWING_CURRENCY
            );
        } else {
            revert(Errors.VL_INVALID_INTEREST_RATE_MODE_SELECTED);
        }
    }

    /**
     * @dev Validates a stable borrow rate rebalance action
     * @param reserve The reserve state on which the user is getting rebalanced
     * @param reserveAddress The address of the reserve
     * @param stableDebtToken The stable debt token instance
     * @param variableDebtToken The variable debt token instance
     * @param aTokenAddress The address of the aToken contract
     */
    function validateRebalanceStableBorrowRate(
        DataTypes.ReserveData storage reserve,
        address reserveAddress,
        IERC20 stableDebtToken,
        IERC20 variableDebtToken,
        address aTokenAddress
    ) external view {
        (bool isActive, , , ) = reserve.configuration.getFlags();

        require(isActive, Errors.VL_NO_ACTIVE_RESERVE);

        //if the usage ratio is below 95%, no rebalances are needed
        uint256 totalDebt = stableDebtToken
            .totalSupply()
            .add(variableDebtToken.totalSupply())
            .wadToRay();
        uint256 availableLiquidity = IERC20(reserveAddress)
            .balanceOf(aTokenAddress)
            .wadToRay();
        uint256 usageRatio = totalDebt == 0
            ? 0
            : totalDebt.rayDiv(availableLiquidity.add(totalDebt));

        //if the liquidity rate is below REBALANCE_UP_THRESHOLD of the max variable APR at 95% usage,
        //then we allow rebalancing of the stable rate positions.

        uint256 currentLiquidityRate = reserve.currentLiquidityRate;
        uint256 maxVariableBorrowRate = IReserveInterestRateStrategy(
            reserve.interestRateStrategyAddress
        ).getMaxVariableBorrowRate();

        require(
            usageRatio >= REBALANCE_UP_USAGE_RATIO_THRESHOLD &&
                currentLiquidityRate <=
                maxVariableBorrowRate.percentMul(
                    REBALANCE_UP_LIQUIDITY_RATE_THRESHOLD
                ),
            Errors.LP_INTEREST_RATE_REBALANCE_CONDITIONS_NOT_MET
        );
    }

    /**
     * @dev Validates the action of setting an asset as collateral
     * @param reserve The state of the reserve that the user is enabling or disabling as collateral
     * @param reserveAddress The address of the reserve
     * @param reservesData The data of all the reserves
     * @param userConfig The state of the user for the specific reserve
     * @param reserves The addresses of all the active reserves
     * @param oracle The price oracle
     */
    function validateSetUseReserveAsCollateral(
        DataTypes.ReserveData storage reserve,
        address reserveAddress,
        bool useAsCollateral,
        mapping(address => DataTypes.ReserveData) storage reservesData,
        DataTypes.UserConfigurationMap storage userConfig,
        mapping(uint256 => address) storage reserves,
        uint256 reservesCount,
        address oracle
    ) external view {
        uint256 underlyingBalance = IERC20(reserve.aTokenAddress).balanceOf(
            msg.sender
        );

        require(
            underlyingBalance > 0,
            Errors.VL_UNDERLYING_BALANCE_NOT_GREATER_THAN_0
        );

        require(
            useAsCollateral ||
                GenericLogic.balanceDecreaseAllowed(
                    reserveAddress,
                    msg.sender,
                    underlyingBalance,
                    reservesData,
                    userConfig,
                    reserves,
                    reservesCount,
                    oracle
                ),
            Errors.VL_DEPOSIT_ALREADY_IN_USE
        );
    }

    /**
     * @dev Validates a flashloan action
     * @param assets The assets being flashborrowed
     * @param amounts The amounts for each asset being borrowed
     **/
    function validateFlashloan(
        address[] memory assets,
        uint256[] memory amounts
    ) internal pure {
        require(
            assets.length == amounts.length,
            Errors.VL_INCONSISTENT_FLASHLOAN_PARAMS
        );
    }

    /**
     * @dev Validates the liquidation action
     * @param collateralReserve The reserve data of the collateral
     * @param principalReserve The reserve data of the principal
     * @param userConfig The user configuration
     * @param userHealthFactor The user's health factor
     * @param userStableDebt Total stable debt balance of the user
     * @param userVariableDebt Total variable debt balance of the user
     **/
    function validateLiquidationCall(
        DataTypes.ReserveData storage collateralReserve,
        DataTypes.ReserveData storage principalReserve,
        DataTypes.UserConfigurationMap storage userConfig,
        uint256 userHealthFactor,
        uint256 userStableDebt,
        uint256 userVariableDebt
    ) external view returns (uint256, string memory) {
        if (
            !collateralReserve.configuration.getActive() ||
            !principalReserve.configuration.getActive()
        ) {
            return (
                uint256(Errors.CollateralManagerErrors.NO_ACTIVE_RESERVE),
                Errors.VL_NO_ACTIVE_RESERVE
            );
        }

        if (
            userHealthFactor >= GenericLogic.HEALTH_FACTOR_LIQUIDATION_THRESHOLD
        ) {
            return (
                uint256(
                    Errors.CollateralManagerErrors.HEALTH_FACTOR_ABOVE_THRESHOLD
                ),
                Errors.LPCM_HEALTH_FACTOR_NOT_BELOW_THRESHOLD
            );
        }

        bool isCollateralEnabled = collateralReserve
            .configuration
            .getLiquidationThreshold() >
            0 &&
            userConfig.isUsingAsCollateral(collateralReserve.id);

        //if collateral isn't enabled as collateral by user, it cannot be liquidated
        if (!isCollateralEnabled) {
            return (
                uint256(
                    Errors
                        .CollateralManagerErrors
                        .COLLATERAL_CANNOT_BE_LIQUIDATED
                ),
                Errors.LPCM_COLLATERAL_CANNOT_BE_LIQUIDATED
            );
        }

        if (userStableDebt == 0 && userVariableDebt == 0) {
            return (
                uint256(Errors.CollateralManagerErrors.CURRRENCY_NOT_BORROWED),
                Errors.LPCM_SPECIFIED_CURRENCY_NOT_BORROWED_BY_USER
            );
        }

        return (
            uint256(Errors.CollateralManagerErrors.NO_ERROR),
            Errors.LPCM_NO_ERRORS
        );
    }

    /**
     * @dev Validates an aToken transfer
     * @param from The user from which the aTokens are being transferred
     * @param reservesData The state of all the reserves
     * @param userConfig The state of the user for the specific reserve
     * @param reserves The addresses of all the active reserves
     * @param oracle The price oracle
     */
    function validateTransfer(
        address from,
        mapping(address => DataTypes.ReserveData) storage reservesData,
        DataTypes.UserConfigurationMap storage userConfig,
        mapping(uint256 => address) storage reserves,
        uint256 reservesCount,
        address oracle
    ) external view {
        (, , , , uint256 healthFactor) = GenericLogic.calculateUserAccountData(
            from,
            reservesData,
            userConfig,
            reserves,
            reservesCount,
            oracle
        );

        require(
            healthFactor >= GenericLogic.HEALTH_FACTOR_LIQUIDATION_THRESHOLD,
            Errors.VL_TRANSFER_NOT_ALLOWED
        );
    }

    /**
     * @dev Validates limits of an aToken transfer
     * @param from The user from which the aTokens are transferred
     * @param to The user receiving the aTokens
     * @param reserve The reserve object on which the user is depositing
     * @param reserveLimits The reserve limits object on which the user is depositing
     */
    function validateTransferLimits(
        address from,
        address to,
        DataTypes.ReserveData storage reserve,
        DataTypes.ReserveLimits storage reserveLimits
    ) external view {
        if (
            from != IAToken(reserve.aTokenAddress).RESERVE_TREASURY_ADDRESS() &&
            (reserveLimits.maxIndividualDepositSize > 0 ||
                reserveLimits.minIndividualDepositSize > 0)
        ) {
            // verify individual dposit size limit if not coming from treasury
            IERC20 aToken = IERC20(reserve.aTokenAddress);
            if (reserveLimits.maxIndividualDepositSize > 0) {
                // balance of "to" already includes amount transferred in
                // as this validation happens after token transfer function is called
                require(
                    aToken.balanceOf(to) <=
                        reserveLimits.maxIndividualDepositSize,
                    Errors.AVL_EXCEED_MAX_INDIVIDUAL_DEPOSIT_SIZE
                );
            }
            if (reserveLimits.minIndividualDepositSize > 0) {
                // balance of "from" already includes amount transferred out
                // as this validation happens after token transfer function is called
                uint256 balance = aToken.balanceOf(from);
                if (balance > 0) {
                    require(
                        balance >= reserveLimits.minIndividualDepositSize,
                        Errors.AVL_BELOW_MIN_INDIVIDUAL_DEPOSIT_SIZE
                    );
                }
                // balance of "to" is not of concern here even if it is still lower
                // than the minimum limit
            }
        }
    }
}
          

contracts/protocol/libraries/math/MathUtils.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {SafeMath} from "../../../dependencies/openzeppelin/contracts/SafeMath.sol";
import {WadRayMath} from "./WadRayMath.sol";

library MathUtils {
    using SafeMath for uint256;
    using WadRayMath for uint256;

    /// @dev Ignoring leap years
    uint256 internal constant SECONDS_PER_YEAR = 365 days;

    /**
     * @dev Function to calculate the interest accumulated using a linear interest rate formula
     * @param rate The interest rate, in ray
     * @param lastUpdateTimestamp The timestamp of the last update of the interest
     * @return The interest rate linearly accumulated during the timeDelta, in ray
     **/

    function calculateLinearInterest(uint256 rate, uint40 lastUpdateTimestamp)
        internal
        view
        returns (uint256)
    {
        //solium-disable-next-line
        uint256 timeDifference = block.timestamp.sub(
            uint256(lastUpdateTimestamp)
        );

        return
            (rate.mul(timeDifference) / SECONDS_PER_YEAR).add(WadRayMath.ray());
    }

    /**
     * @dev Function to calculate the interest using a compounded interest rate formula
     * To avoid expensive exponentiation, the calculation is performed using a binomial approximation:
     *
     *  (1+x)^n = 1+n*x+[n/2*(n-1)]*x^2+[n/6*(n-1)*(n-2)*x^3...
     *
     * The approximation slightly underpays liquidity providers and undercharges borrowers, with the advantage of great gas cost reductions
     * The whitepaper contains reference to the approximation and a table showing the margin of error per different time periods
     *
     * @param rate The interest rate, in ray
     * @param lastUpdateTimestamp The timestamp of the last update of the interest
     * @return The interest rate compounded during the timeDelta, in ray
     **/
    function calculateCompoundedInterest(
        uint256 rate,
        uint40 lastUpdateTimestamp,
        uint256 currentTimestamp
    ) internal pure returns (uint256) {
        //solium-disable-next-line
        uint256 exp = currentTimestamp.sub(uint256(lastUpdateTimestamp));

        if (exp == 0) {
            return WadRayMath.ray();
        }

        uint256 expMinusOne = exp - 1;

        uint256 expMinusTwo = exp > 2 ? exp - 2 : 0;

        uint256 ratePerSecond = rate / SECONDS_PER_YEAR;

        uint256 basePowerTwo = ratePerSecond.rayMul(ratePerSecond);
        uint256 basePowerThree = basePowerTwo.rayMul(ratePerSecond);

        uint256 secondTerm = exp.mul(expMinusOne).mul(basePowerTwo) / 2;
        uint256 thirdTerm = exp.mul(expMinusOne).mul(expMinusTwo).mul(
            basePowerThree
        ) / 6;

        return
            WadRayMath.ray().add(ratePerSecond.mul(exp)).add(secondTerm).add(
                thirdTerm
            );
    }

    /**
     * @dev Calculates the compounded interest between the timestamp of the last update and the current block timestamp
     * @param rate The interest rate (in ray)
     * @param lastUpdateTimestamp The timestamp from which the interest accumulation needs to be calculated
     **/
    function calculateCompoundedInterest(
        uint256 rate,
        uint40 lastUpdateTimestamp
    ) internal view returns (uint256) {
        return
            calculateCompoundedInterest(
                rate,
                lastUpdateTimestamp,
                block.timestamp
            );
    }
}
          

contracts/protocol/libraries/math/PercentageMath.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {Errors} from "../helpers/Errors.sol";

/**
 * @title PercentageMath library
 * @author Aave
 * @notice Provides functions to perform percentage calculations
 * @dev Percentages are defined by default with 2 decimals of precision (100.00). The precision is indicated by PERCENTAGE_FACTOR
 * @dev Operations are rounded half up
 **/

library PercentageMath {
    uint256 constant PERCENTAGE_FACTOR = 1e4; //percentage plus two decimals
    uint256 constant HALF_PERCENT = PERCENTAGE_FACTOR / 2;

    /**
     * @dev Executes a percentage multiplication
     * @param value The value of which the percentage needs to be calculated
     * @param percentage The percentage of the value to be calculated
     * @return The percentage of value
     **/
    function percentMul(uint256 value, uint256 percentage)
        internal
        pure
        returns (uint256)
    {
        if (value == 0 || percentage == 0) {
            return 0;
        }

        require(
            value <= (type(uint256).max - HALF_PERCENT) / percentage,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );

        return (value * percentage + HALF_PERCENT) / PERCENTAGE_FACTOR;
    }

    /**
     * @dev Executes a percentage division
     * @param value The value of which the percentage needs to be calculated
     * @param percentage The percentage of the value to be calculated
     * @return The value divided the percentage
     **/
    function percentDiv(uint256 value, uint256 percentage)
        internal
        pure
        returns (uint256)
    {
        require(percentage != 0, Errors.MATH_DIVISION_BY_ZERO);
        uint256 halfPercentage = percentage / 2;

        require(
            value <= (type(uint256).max - halfPercentage) / PERCENTAGE_FACTOR,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );

        return (value * PERCENTAGE_FACTOR + halfPercentage) / percentage;
    }
}
          

contracts/protocol/libraries/math/WadRayMath.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

import {Errors} from "../helpers/Errors.sol";

/**
 * @title WadRayMath library
 * @author Aave
 * @dev Provides mul and div function for wads (decimal numbers with 18 digits precision) and rays (decimals with 27 digits)
 **/

library WadRayMath {
    uint256 internal constant WAD = 1e18;
    uint256 internal constant halfWAD = WAD / 2;

    uint256 internal constant RAY = 1e27;
    uint256 internal constant halfRAY = RAY / 2;

    uint256 internal constant WAD_RAY_RATIO = 1e9;

    /**
     * @return One ray, 1e27
     **/
    function ray() internal pure returns (uint256) {
        return RAY;
    }

    /**
     * @return One wad, 1e18
     **/

    function wad() internal pure returns (uint256) {
        return WAD;
    }

    /**
     * @return Half ray, 1e27/2
     **/
    function halfRay() internal pure returns (uint256) {
        return halfRAY;
    }

    /**
     * @return Half ray, 1e18/2
     **/
    function halfWad() internal pure returns (uint256) {
        return halfWAD;
    }

    /**
     * @dev Multiplies two wad, rounding half up to the nearest wad
     * @param a Wad
     * @param b Wad
     * @return The result of a*b, in wad
     **/
    function wadMul(uint256 a, uint256 b) internal pure returns (uint256) {
        if (a == 0 || b == 0) {
            return 0;
        }

        require(
            a <= (type(uint256).max - halfWAD) / b,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );

        return (a * b + halfWAD) / WAD;
    }

    /**
     * @dev Divides two wad, rounding half up to the nearest wad
     * @param a Wad
     * @param b Wad
     * @return The result of a/b, in wad
     **/
    function wadDiv(uint256 a, uint256 b) internal pure returns (uint256) {
        require(b != 0, Errors.MATH_DIVISION_BY_ZERO);
        uint256 halfB = b / 2;

        require(
            a <= (type(uint256).max - halfB) / WAD,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );

        return (a * WAD + halfB) / b;
    }

    /**
     * @dev Multiplies two ray, rounding half up to the nearest ray
     * @param a Ray
     * @param b Ray
     * @return The result of a*b, in ray
     **/
    function rayMul(uint256 a, uint256 b) internal pure returns (uint256) {
        if (a == 0 || b == 0) {
            return 0;
        }

        require(
            a <= (type(uint256).max - halfRAY) / b,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );

        return (a * b + halfRAY) / RAY;
    }

    /**
     * @dev Divides two ray, rounding half up to the nearest ray
     * @param a Ray
     * @param b Ray
     * @return The result of a/b, in ray
     **/
    function rayDiv(uint256 a, uint256 b) internal pure returns (uint256) {
        require(b != 0, Errors.MATH_DIVISION_BY_ZERO);
        uint256 halfB = b / 2;

        require(
            a <= (type(uint256).max - halfB) / RAY,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );

        return (a * RAY + halfB) / b;
    }

    /**
     * @dev Casts ray down to wad
     * @param a Ray
     * @return a casted to wad, rounded half up to the nearest wad
     **/
    function rayToWad(uint256 a) internal pure returns (uint256) {
        uint256 halfRatio = WAD_RAY_RATIO / 2;
        uint256 result = halfRatio + a;
        require(result >= halfRatio, Errors.MATH_ADDITION_OVERFLOW);

        return result / WAD_RAY_RATIO;
    }

    /**
     * @dev Converts wad up to ray
     * @param a Wad
     * @return a converted in ray
     **/
    function wadToRay(uint256 a) internal pure returns (uint256) {
        uint256 result = a * WAD_RAY_RATIO;
        require(
            result / WAD_RAY_RATIO == a,
            Errors.MATH_MULTIPLICATION_OVERFLOW
        );
        return result;
    }
}
          

contracts/protocol/libraries/types/DataTypes.sol

// SPDX-License-Identifier: agpl-3.0
pragma solidity 0.7.6;

library DataTypes {
    // refer to the whitepaper, section 1.1 basic concepts for a formal description of these properties.
    struct ReserveData {
        //stores the reserve configuration
        ReserveConfigurationMap configuration;
        //the liquidity index. Expressed in ray
        uint128 liquidityIndex;
        //variable borrow index. Expressed in ray
        uint128 variableBorrowIndex;
        //the current supply rate. Expressed in ray
        uint128 currentLiquidityRate;
        //the current variable borrow rate. Expressed in ray
        uint128 currentVariableBorrowRate;
        //the current stable borrow rate. Expressed in ray
        uint128 currentStableBorrowRate;
        uint40 lastUpdateTimestamp;
        //tokens addresses
        address aTokenAddress;
        address stableDebtTokenAddress;
        address variableDebtTokenAddress;
        //address of the interest rate strategy
        address interestRateStrategyAddress;
        //the id of the reserve. Represents the position in the list of the active reserves
        uint8 id;
    }

    struct ReserveConfigurationMap {
        //bit 0-15: LTV
        //bit 16-31: Liq. threshold
        //bit 32-47: Liq. bonus
        //bit 48-55: Decimals
        //bit 56: Reserve is active
        //bit 57: reserve is frozen
        //bit 58: borrowing is enabled
        //bit 59: stable rate borrowing enabled
        //bit 60-63: reserved
        //bit 64-79: reserve factor
        uint256 data;
    }

    struct ReserveLimits {
        uint256 maxGlobalDepositSize;
        uint256 maxIndividualDepositSize;
        uint256 minIndividualDepositSize;
        uint256 maxGlobalBorrowSize;
        uint256 maxIndividualBorrowSize;
        uint256 maxBorrowBps;
    }

    struct UserConfigurationMap {
        uint256 data;
    }

    enum InterestRateMode {
        NONE,
        STABLE,
        VARIABLE
    }
}
          

Compiler Settings

{"outputSelection":{"*":{"*":["abi","evm.bytecode","evm.deployedBytecode","evm.methodIdentifiers"]}},"optimizer":{"runs":200,"enabled":true},"libraries":{"contracts/protocol/libraries/logic/ValidationLogic.sol":{"ValidationLogic":"0x3223474c18d19a969a7d6dc6b4ebb210a0843ef2"},"contracts/protocol/libraries/logic/ReserveLogic.sol":{"ReserveLogic":"0x0da77bca9b08c307c64911313214514d1b939886"}},"evmVersion":"istanbul"}
              

Contract ABI

[{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"FLASHLOAN_PREMIUM_TOTAL","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"LENDINGPOOL_REVISION","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"MAX_NUMBER_RESERVES","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"MAX_STABLE_RATE_BORROW_SIZE_PERCENT","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"borrow","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"uint256","name":"amount","internalType":"uint256"},{"type":"uint256","name":"interestRateMode","internalType":"uint256"},{"type":"uint16","name":"referralCode","internalType":"uint16"},{"type":"address","name":"onBehalfOf","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"deposit","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"uint256","name":"amount","internalType":"uint256"},{"type":"address","name":"onBehalfOf","internalType":"address"},{"type":"uint16","name":"referralCode","internalType":"uint16"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"finalizeTransfer","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"address","name":"from","internalType":"address"},{"type":"address","name":"to","internalType":"address"},{"type":"uint256","name":"amount","internalType":"uint256"},{"type":"uint256","name":"balanceFromBefore","internalType":"uint256"},{"type":"uint256","name":"balanceToBefore","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"flashLoan","inputs":[{"type":"address","name":"receiverAddress","internalType":"address"},{"type":"address[]","name":"assets","internalType":"address[]"},{"type":"uint256[]","name":"amounts","internalType":"uint256[]"},{"type":"bytes","name":"params","internalType":"bytes"},{"type":"uint16","name":"referralCode","internalType":"uint16"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"contract ILendingPoolAddressesProvider"}],"name":"getAddressesProvider","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct DataTypes.ReserveConfigurationMap","components":[{"type":"uint256"}]}],"name":"getConfiguration","inputs":[{"type":"address","name":"asset","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct DataTypes.ReserveData","components":[{"type":"tuple","components":[{"type":"uint256"}]},{"type":"uint128"},{"type":"uint128"},{"type":"uint128"},{"type":"uint128"},{"type":"uint128"},{"type":"uint40"},{"type":"address"},{"type":"address"},{"type":"address"},{"type":"address"},{"type":"uint8"}]}],"name":"getReserveData","inputs":[{"type":"address","name":"asset","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct DataTypes.ReserveLimits","components":[{"type":"uint256"},{"type":"uint256"},{"type":"uint256"},{"type":"uint256"},{"type":"uint256"},{"type":"uint256"}]}],"name":"getReserveLimits","inputs":[{"type":"address","name":"asset","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"getReserveNormalizedIncome","inputs":[{"type":"address","name":"asset","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"getReserveNormalizedVariableDebt","inputs":[{"type":"address","name":"asset","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address[]","name":"","internalType":"address[]"}],"name":"getReservesList","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"uint256","name":"totalCollateralETH","internalType":"uint256"},{"type":"uint256","name":"totalDebtETH","internalType":"uint256"},{"type":"uint256","name":"availableBorrowsETH","internalType":"uint256"},{"type":"uint256","name":"currentLiquidationThreshold","internalType":"uint256"},{"type":"uint256","name":"ltv","internalType":"uint256"},{"type":"uint256","name":"healthFactor","internalType":"uint256"}],"name":"getUserAccountData","inputs":[{"type":"address","name":"user","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct DataTypes.UserConfigurationMap","components":[{"type":"uint256"}]}],"name":"getUserConfiguration","inputs":[{"type":"address","name":"user","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"initReserve","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"address","name":"aTokenAddress","internalType":"address"},{"type":"address","name":"stableDebtAddress","internalType":"address"},{"type":"address","name":"variableDebtAddress","internalType":"address"},{"type":"address","name":"interestRateStrategyAddress","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"initialize","inputs":[{"type":"address","name":"provider","internalType":"contract ILendingPoolAddressesProvider"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"bool","name":"","internalType":"bool"}],"name":"isBlacklisted","inputs":[{"type":"address","name":"","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"liquidationCall","inputs":[{"type":"address","name":"collateralAsset","internalType":"address"},{"type":"address","name":"debtAsset","internalType":"address"},{"type":"address","name":"user","internalType":"address"},{"type":"uint256","name":"debtToCover","internalType":"uint256"},{"type":"bool","name":"receiveAToken","internalType":"bool"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"bool","name":"","internalType":"bool"}],"name":"paused","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"rebalanceStableBorrowRate","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"address","name":"user","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"repay","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"uint256","name":"amount","internalType":"uint256"},{"type":"uint256","name":"rateMode","internalType":"uint256"},{"type":"address","name":"onBehalfOf","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"repayAllAndBlacklist","inputs":[{"type":"address","name":"user","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setConfiguration","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"uint256","name":"configuration","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setLimits","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"tuple","name":"reserveLimits","internalType":"struct DataTypes.ReserveLimits","components":[{"type":"uint256"},{"type":"uint256"},{"type":"uint256"},{"type":"uint256"},{"type":"uint256"},{"type":"uint256"}]}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setPause","inputs":[{"type":"bool","name":"val","internalType":"bool"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setReserveInterestRateStrategyAddress","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"address","name":"rateStrategyAddress","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setUserUseReserveAsCollateral","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"bool","name":"useAsCollateral","internalType":"bool"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"swapBorrowRateMode","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"uint256","name":"rateMode","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[{"type":"uint256","name":"","internalType":"uint256"}],"name":"withdraw","inputs":[{"type":"address","name":"asset","internalType":"address"},{"type":"uint256","name":"amount","internalType":"uint256"},{"type":"address","name":"to","internalType":"address"}]},{"type":"event","name":"Blacklist","inputs":[{"type":"address","name":"user","indexed":true}],"anonymous":false},{"type":"event","name":"Borrow","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":false},{"type":"address","name":"onBehalfOf","indexed":true},{"type":"uint256","name":"amount","indexed":false},{"type":"uint256","name":"borrowRateMode","indexed":false},{"type":"uint256","name":"borrowRate","indexed":false},{"type":"uint16","name":"referral","indexed":true}],"anonymous":false},{"type":"event","name":"Deposit","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":false},{"type":"address","name":"onBehalfOf","indexed":true},{"type":"uint256","name":"amount","indexed":false},{"type":"uint16","name":"referral","indexed":true}],"anonymous":false},{"type":"event","name":"FlashLoan","inputs":[{"type":"address","name":"target","indexed":true},{"type":"address","name":"initiator","indexed":true},{"type":"address","name":"asset","indexed":true},{"type":"uint256","name":"amount","indexed":false},{"type":"uint256","name":"premium","indexed":false},{"type":"uint16","name":"referralCode","indexed":false}],"anonymous":false},{"type":"event","name":"LiquidationCall","inputs":[{"type":"address","name":"collateralAsset","indexed":true},{"type":"address","name":"debtAsset","indexed":true},{"type":"address","name":"user","indexed":true},{"type":"uint256","name":"debtToCover","indexed":false},{"type":"uint256","name":"liquidatedCollateralAmount","indexed":false},{"type":"address","name":"liquidator","indexed":false},{"type":"bool","name":"receiveAToken","indexed":false}],"anonymous":false},{"type":"event","name":"Paused","inputs":[],"anonymous":false},{"type":"event","name":"RebalanceStableBorrowRate","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":true}],"anonymous":false},{"type":"event","name":"Repay","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":true},{"type":"address","name":"repayer","indexed":true},{"type":"uint256","name":"amount","indexed":false}],"anonymous":false},{"type":"event","name":"ReserveDataUpdated","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"uint256","name":"liquidityRate","indexed":false},{"type":"uint256","name":"stableBorrowRate","indexed":false},{"type":"uint256","name":"variableBorrowRate","indexed":false},{"type":"uint256","name":"liquidityIndex","indexed":false},{"type":"uint256","name":"variableBorrowIndex","indexed":false}],"anonymous":false},{"type":"event","name":"ReserveUsedAsCollateralDisabled","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":true}],"anonymous":false},{"type":"event","name":"ReserveUsedAsCollateralEnabled","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":true}],"anonymous":false},{"type":"event","name":"Swap","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":true},{"type":"uint256","name":"rateMode","indexed":false}],"anonymous":false},{"type":"event","name":"Unpaused","inputs":[],"anonymous":false},{"type":"event","name":"Withdraw","inputs":[{"type":"address","name":"reserve","indexed":true},{"type":"address","name":"user","indexed":true},{"type":"address","name":"to","indexed":true},{"type":"uint256","name":"amount","indexed":false}],"anonymous":false}]
              

Contract Creation Code

0x60806040526000805534801561001457600080fd5b50615d2a80620000256000396000f3fe608060405234801561001057600080fd5b50600436106101ef5760003560e01c8063a415bcad1161010f578063d1946dbc116100a2578063e8eda9df11610071578063e8eda9df1461043c578063f8119d511461044f578063fe575a8714610457578063fe65acfe1461046a576101ef565b8063d1946dbc146103f9578063d5ed39331461040e578063e7e7d62a14610421578063e82fec2f14610434576101ef565b8063c44b11f7116100de578063c44b11f7146103ad578063c4d66de8146103c0578063cd112382146103d3578063d15e0053146103e6576101ef565b8063a415bcad1461034f578063b8d2927614610362578063bedb86fb14610375578063bf92857c14610388576101ef565b80635a3b74b9116101875780637a708e92116101565780637a708e9214610301578063845e79bc146103145780638afaff021461033457806394ba89a21461033c576101ef565b80635a3b74b9146102b35780635c975abb146102c657806363a5d53b146102db57806369328dec146102ee576101ef565b806335ea6a75116101c357806335ea6a751461024d578063386497fd1461026d5780634417a58314610280578063573ade81146102a0576101ef565b8062a718a9146101f4578063074b2e4314610209578063158c13e8146102275780631d2118f91461023a575b600080fd5b61020761020236600461511f565b61047f565b005b610211610653565b60405161021e9190615ba7565b60405180910390f35b61020761023536600461503f565b610659565b610207610248366004615077565b61081b565b61026061025b36600461503f565b610854565b60405161021e9190615967565b61021161027b36600461503f565b610936565b61029361028e36600461503f565b61095d565b60405161021e919061595d565b6102116102ae3660046153e7565b610990565b6102076102c13660046152bf565b610cbc565b6102ce610e8a565b60405161021e919061591f565b6102076102e93660046152ec565b610e93565b6102116102fc366004615356565b610ec5565b61020761030f3660046150af565b611215565b61032761032236600461503f565b6112f7565b60405161021e9190615b63565b610211611361565b61020761034a36600461532b565b611366565b61020761035d366004615430565b6116d3565b61020761037036600461532b565b611753565b61020761038336600461546e565b611777565b61039b61039636600461503f565b6117f2565b60405161021e96959493929190615bca565b6102936103bb36600461503f565b6118b1565b6102076103ce36600461503f565b6118e4565b6102076103e1366004615077565b6119ab565b6102116103f436600461503f565b611c21565b610401611c42565b60405161021e91906158d2565b61020761041c366004615178565b611ce8565b61020761042f3660046151dc565b61200f565b610211612609565b61020761044a366004615397565b61260f565b610211612854565b6102ce61046536600461503f565b61285a565b61047261286f565b60405161021e9190615620565b61048761287e565b6034546040805163712d917160e01b815290516000926001600160a01b03169163712d9171916004808301926020929190829003018186803b1580156104cc57600080fd5b505afa1580156104e0573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610504919061505b565b9050600080826001600160a01b0316888888888860405160240161052c9594939291906156a8565b60408051601f198184030181529181526020820180516001600160e01b031662a718a960e01b179052516105609190615604565b600060405180830381855af49150503d806000811461059b576040519150601f19603f3d011682016040523d82523d6000602084013e6105a0565b606091505b50915091508160405180604001604052806002815260200161323360f01b815250906105e85760405162461bcd60e51b81526004016105df919061592a565b60405180910390fd5b506000808280602001905181019061060091906154be565b9150915081600014816040516020016106199190615604565b604051602081830303815290604052906106465760405162461bcd60e51b81526004016105df919061592a565b5050505050505050505050565b603c5490565b61066161287e565b6001600160a01b03808216600090815260376020908152604080832081518084018352905481526039546034548351631f94a27560e31b81529351959661070d968996603596603895949093169263fca513a89260048083019392829003018186803b1580156106d057600080fd5b505afa1580156106e4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610708919061505b565b6128bc565b9450505050508060001460405180604001604052806002815260200161383960f01b815250906107505760405162461bcd60e51b81526004016105df919061592a565b5060005b6039548110156107cd576000818152603860209081526040808320546001600160a01b0316808452603590925282209091908190610793908790612d7d565b909250905081156107ad576107ab8383600189610990565b505b80156107c2576107c08382600289610990565b505b505050600101610754565b506001600160a01b0382166000818152603e6020526040808220805460ff19166001179055517f7811ed4e4b787e4f0f046ffb4a5ec1e064b0eb6d191ffabb66270ed8be89b3259190a25050565b610823612e82565b6001600160a01b03918216600090815260356020526040902060070180546001600160a01b03191691909216179055565b61085c614da2565b506001600160a01b0381811660009081526035602090815260409182902082516101a08101845281546101808201908152815260018201546001600160801b0380821694830194909452600160801b908190048416948201949094526002820154808416606083015284900483166080820152600382015492831660a08201529290910464ffffffffff1660c08301526004810154831660e0830152600581015483166101008301526006810154831661012083015260070154918216610140820152600160a01b90910460ff166101608201525b919050565b6001600160a01b038116600090815260356020526040812061095790612f41565b92915050565b610965614e0d565b506001600160a01b031660009081526037602090815260409182902082519182019092529054815290565b600061099a61287e565b6001600160a01b038516600090815260356020526040812090806109be8584612d7d565b9150915060008660028111156109d057fe5b60405163fa0c214960e01b8152909150733223474c18d19a969a7d6dc6b4ebb210a0843ef29063fa0c214990610a149087908c9086908c908a908a90600401615b22565b60006040518083038186803b158015610a2c57600080fd5b505af4158015610a40573d6000803e3d6000fd5b506000925060019150610a509050565b826002811115610a5c57fe5b14610a675782610a69565b835b905080891015610a765750875b610a7f85612fbe565b6001826002811115610a8d57fe5b1415610afe576005850154604051632770a7eb60e21b81526001600160a01b0390911690639dc29fac90610ac7908a908590600401615665565b600060405180830381600087803b158015610ae157600080fd5b505af1158015610af5573d6000803e3d6000fd5b50505050610b7c565b60068501546001860154604051637a94c56560e11b81526001600160a01b039092169163f5298aca91610b49918b918691600160801b9091046001600160801b03169060040161567e565b600060405180830381600087803b158015610b6357600080fd5b505af1158015610b77573d6000803e3d6000fd5b505050505b60048501546001600160a01b0316610b98868c83856000613088565b610bac82610ba687876134f8565b90613552565b610be45760078601546001600160a01b0389166000908152603760205260408120610be4929091600160a01b90910460ff16906135af565b610bf96001600160a01b038c1633838561364d565b6040516388dd91a160e01b81526001600160a01b038216906388dd91a190610c279033908690600401615665565b600060405180830381600087803b158015610c4157600080fd5b505af1158015610c55573d6000803e3d6000fd5b50505050336001600160a01b0316886001600160a01b03168c6001600160a01b03167f4cdde6e09bb755c9a5589ebaec640bbfedff1362d4b255ebf8339782b9942faa85604051610ca69190615ba7565b60405180910390a4509998505050505050505050565b610cc461287e565b610ccd336136a7565b6001600160a01b03808316600090815260356020818152604080842033855260378352938190206039546034548351631f94a27560e31b815293519697733223474c18d19a969a7d6dc6b4ebb210a0843ef297635fa297e5978a978d978d9792969295603895939493169263fca513a892600480840193919291829003018186803b158015610d5b57600080fd5b505afa158015610d6f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d93919061505b565b6040518963ffffffff1660e01b8152600401610db6989796959493929190615a88565b60006040518083038186803b158015610dce57600080fd5b505af4158015610de2573d6000803e3d6000fd5b505050506007810154336000908152603760205260409020610e0e91600160a01b900460ff1684613702565b8115610e4e5760405133906001600160a01b038516907e058a56ea94653cdf4f152d227ace22d4c00ad99e2a43f58cb7d9e3feb295f290600090a3610e85565b60405133906001600160a01b038516907f44c58d81365b66dd4b1a7f36c25aa97b8c71c361ee4937adc1a00000227db5dd90600090a35b505050565b603a5460ff1690565b610e9b612e82565b6001600160a01b03821660009081526036602052604090208190610ebf8282615c1e565b50505050565b6000610ecf61287e565b6001600160a01b0380851660009081526035602052604080822060048082015492516370a0823160e01b8152919492909216929183916370a0823191610f1791339101615620565b60206040518083038186803b158015610f2f57600080fd5b505afa158015610f43573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610f6791906154a6565b905085600019811415610f775750805b733223474c18d19a969a7d6dc6b4ebb210a0843ef263c5b18e768983856035603660008f6001600160a01b03166001600160a01b0316815260200190815260200160002060376000336001600160a01b03166001600160a01b031681526020019081526020016000206038603954603460009054906101000a90046001600160a01b03166001600160a01b031663fca513a86040518163ffffffff1660e01b815260040160206040518083038186803b15801561103357600080fd5b505afa158015611047573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061106b919061505b565b6040518a63ffffffff1660e01b815260040161108f999897969594939291906157a1565b60006040518083038186803b1580156110a757600080fd5b505af41580156110bb573d6000803e3d6000fd5b505050506110c884612fbe565b6110d6848985600085613088565b81811415611140576007840154336000908152603760205260408120611109929091600160a01b90910460ff1690613702565b60405133906001600160a01b038a16907f44c58d81365b66dd4b1a7f36c25aa97b8c71c361ee4937adc1a00000227db5dd90600090a35b6001840154604051636b81068560e11b81526001600160a01b0385169163d7020d0a916111819133918b9187916001600160801b0390911690600401615634565b600060405180830381600087803b15801561119b57600080fd5b505af11580156111af573d6000803e3d6000fd5b50505050856001600160a01b0316336001600160a01b0316896001600160a01b03167f3115d1449a7b732c986cba18244e897a450f61e1bb8d589cd2e69e6c8924f9f7846040516112009190615ba7565b60405180910390a493505050505b9392505050565b61121d612e82565b611226856137a6565b6040518060400160405280600281526020016106e760f31b8152509061125f5760405162461bcd60e51b81526004016105df919061592a565b506001600160a01b038516600090815260356020526040908190209051630acce25f60e21b8152730da77bca9b08c307c64911313214514d1b93988691632b33897c916112b791908890889088908890600401615a5a565b60006040518083038186803b1580156112cf57600080fd5b505af41580156112e3573d6000803e3d6000fd5b505050506112f0856137ac565b5050505050565b6112ff614e20565b506001600160a01b0316600090815260366020908152604091829020825160c08101845281548152600182015492810192909252600281015492820192909252600382015460608201526004820154608082015260059091015460a082015290565b600181565b61136e61287e565b6001600160a01b038216600090815260356020526040812090806113923384612d7d565b9150915060008460028111156113a457fe5b3360009081526037602052604090819020905163a8695b1d60e01b8152919250733223474c18d19a969a7d6dc6b4ebb210a0843ef29163a8695b1d916113f591889190889088908890600401615aee565b60006040518083038186803b15801561140d57600080fd5b505af4158015611421573d6000803e3d6000fd5b5050505061142e84612fbe565b600181600281111561143c57fe5b141561154c576005840154604051632770a7eb60e21b81526001600160a01b0390911690639dc29fac906114769033908790600401615665565b600060405180830381600087803b15801561149057600080fd5b505af11580156114a4573d6000803e3d6000fd5b505050506006840154600185015460405163b3f1c93d60e01b81526001600160a01b039092169163b3f1c93d916114f491339182918991600160801b90046001600160801b031690600401615634565b602060405180830381600087803b15801561150e57600080fd5b505af1158015611522573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611546919061548a565b50611662565b60068401546001850154604051637a94c56560e11b81526001600160a01b039092169163f5298aca916115979133918791600160801b9091046001600160801b03169060040161567e565b600060405180830381600087803b1580156115b157600080fd5b505af11580156115c5573d6000803e3d6000fd5b505050506005840154600385015460405163b3f1c93d60e01b81526001600160a01b039092169163b3f1c93d9161160e913391829188916001600160801b031690600401615634565b602060405180830381600087803b15801561162857600080fd5b505af115801561163c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611660919061548a565b505b600484015461168090859088906001600160a01b0316600080613088565b336001600160a01b0316866001600160a01b03167fea368a40e9570069bb8e6511d668293ad2e1f03b0d982431fd223de9f3b70ca6876040516116c39190615ba7565b60405180910390a3505050505050565b6116db61287e565b6001600160a01b038086166000818152603560209081526040918290208251610100810184529384523391840191909152848416918301919091526060820187905260808201869052600481015490921660a082015261ffff841660c0820152600160e082015261174b906138b5565b505050505050565b61175b612e82565b6001600160a01b03909116600090815260356020526040902055565b61177f612e82565b603a805460ff1916821515179081905560ff16156117c5576040517f9e87fac88ff661f02d44f95383c817fece4bce600a3dab7a54406878b965e75290600090a16117ef565b6040517fa45f47fdea8a1efdd9029a5691c7f759c32b7c698632b563573e155625d1693390600090a15b50565b60008060008060008061188e876035603760008b6001600160a01b03166001600160a01b031681526020019081526020016000206040518060200160405290816000820154815250506038603954603460009054906101000a90046001600160a01b03166001600160a01b031663fca513a86040518163ffffffff1660e01b815260040160206040518083038186803b1580156106d057600080fd5b939950919750909450925090506118a6868684613e5c565b935091939550919395565b6118b9614e0d565b506001600160a01b031660009081526035602090815260409182902082519182019092529054815290565b60006118ee613e90565b60015490915060ff16806119055750611905613e95565b80611911575060005481115b61194c5760405162461bcd60e51b815260040180806020018281038252602e815260200180615c9d602e913960400191505060405180910390fd5b60015460ff1615801561196b576001805460ff19168117905560008290555b603480546001600160a01b0319166001600160a01b0385161790556109c4603b556009603c556080603d558015610e85576001805460ff19169055505050565b6119b361287e565b6001600160a01b038083166000908152603560205260408082206005810154600682015460048084015494516370a0823160e01b81529396928316959183169490921692909185916370a0823191611a0d918a9101615620565b60206040518083038186803b158015611a2557600080fd5b505afa158015611a39573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a5d91906154a6565b60405163548cad0960e01b8152909150733223474c18d19a969a7d6dc6b4ebb210a0843ef29063548cad0990611a9f9088908b90899089908990600401615a5a565b60006040518083038186803b158015611ab757600080fd5b505af4158015611acb573d6000803e3d6000fd5b50505050611ad885612fbe565b604051632770a7eb60e21b81526001600160a01b03851690639dc29fac90611b069089908590600401615665565b600060405180830381600087803b158015611b2057600080fd5b505af1158015611b34573d6000803e3d6000fd5b505050600386015460405163b3f1c93d60e01b81526001600160a01b038716925063b3f1c93d91611b77918a91829187916001600160801b031690600401615634565b602060405180830381600087803b158015611b9157600080fd5b505af1158015611ba5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bc9919061548a565b50611bd8858884600080613088565b856001600160a01b0316876001600160a01b03167f9f439ae0c81e41a04d3fdfe07aed54e6a179fb0db15be7702eb66fa8ef6f530060405160405180910390a350505050505050565b6001600160a01b038116600090815260356020526040812061095790613e9b565b6060600060395467ffffffffffffffff81118015611c5f57600080fd5b50604051908082528060200260200182016040528015611c89578160200160208202803683370190505b50905060005b603954811015611ce25760008181526038602052604090205482516001600160a01b0390911690839083908110611cc257fe5b6001600160a01b0390921660209283029190910190910152600101611c8f565b50905090565b611cf061287e565b6001600160a01b038681166000908152603560209081526040918290206004015482518084019093526002835261363360f01b918301919091529091163314611d4c5760405162461bcd60e51b81526004016105df919061592a565b5060345460408051631f94a27560e31b815290516000926001600160a01b03169163fca513a8916004808301926020929190829003018186803b158015611d9257600080fd5b505afa158015611da6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dca919061505b565b6001600160a01b03881660009081526035602090815260408083206036909252918290209151633ef9e4fb60e01b8152929350733223474c18d19a969a7d6dc6b4ebb210a0843ef292633ef9e4fb92611e2a928b928b92906004016156dc565b60006040518083038186803b158015611e4257600080fd5b505af4158015611e56573d6000803e3d6000fd5b5050506001600160a01b03871660009081526037602052604090819020603954915163516d1bc760e01b8152733223474c18d19a969a7d6dc6b4ebb210a0843ef2935063516d1bc792611eb5928b926035926038918990600401615705565b60006040518083038186803b158015611ecd57600080fd5b505af4158015611ee1573d6000803e3d6000fd5b505050506001600160a01b03878116600090815260356020526040902060070154600160a01b900460ff16908681169088161461200557611f228486613552565b611f8e576001600160a01b038716600090815260376020526040812090611f4c9082908490613702565b876001600160a01b0316896001600160a01b03167f44c58d81365b66dd4b1a7f36c25aa97b8c71c361ee4937adc1a00000227db5dd60405160405180910390a3505b82158015611f9b57508415155b15612005576001600160a01b0386166000908152603760205260409020611fc481836001613702565b866001600160a01b0316896001600160a01b03167e058a56ea94653cdf4f152d227ace22d4c00ad99e2a43f58cb7d9e3feb295f260405160405180910390a3505b5050505050505050565b61201761287e565b61201f614e56565b61208c88888080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050604080516020808c0282810182019093528b82529093508b92508a918291850190849080828437600092019190915250613efa92505050565b60008767ffffffffffffffff811180156120a557600080fd5b506040519080825280602002602001820160405280156120cf578160200160208202803683370190505b50905060008867ffffffffffffffff811180156120eb57600080fd5b50604051908082528060200260200182016040528015612115578160200160208202803683370190505b506001600160a01b038c1684526000604085015290505b60408301518911156122d957603560008b8b866040015181811061214c57fe5b9050602002016020810190612161919061503f565b6001600160a01b03166001600160a01b0316815260200190815260200160002060040160009054906101000a90046001600160a01b0316828460400151815181106121a857fe5b60200260200101906001600160a01b031690816001600160a01b0316815250506121ff6127106121f9603c548b8b88604001518181106121e457fe5b90506020020135613f3890919063ffffffff16565b90613f91565b8184604001518151811061220f57fe5b6020026020010181815250508183604001518151811061222b57fe5b60200260200101516001600160a01b0316634efecaa58c8a8a876040015181811061225257fe5b905060200201356040518363ffffffff1660e01b8152600401612276929190615665565b602060405180830381600087803b15801561229057600080fd5b505af11580156122a4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122c891906154a6565b50604083018051600101905261212c565b8251604051632483d72160e21b81526001600160a01b039091169063920f5c8490612316908d908d908d908d90889033908f908f90600401615813565b602060405180830381600087803b15801561233057600080fd5b505af1158015612344573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612368919061548a565b604051806040016040528060028152602001611b1b60f11b815250906123a15760405162461bcd60e51b81526004016105df919061592a565b50600060408401525b604083015189111561064657898984604001518181106123c657fe5b90506020020160208101906123db919061503f565b6001600160a01b031660608401526040830151889088908181106123fb57fe5b905060200201358360a00181815250508083604001518151811061241b57fe5b60200260200101518360c00181815250508183604001518151811061243c57fe5b60209081029190910101516001600160a01b0316608084015260c083015160a0840151612468916134f8565b60e084015260608301516001600160a01b0316600090815260356020526040902061249290612fbe565b61253283608001516001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156124d257600080fd5b505afa1580156124e6573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061250a91906154a6565b60c085015160608601516001600160a01b031660009081526035602052604090209190613ff8565b6060830151608084015160e08501516001600160a01b038316600090815260356020526040812061256894909390929091613088565b6125948b84608001518560e0015186606001516001600160a01b031661364d909392919063ffffffff16565b82606001516001600160a01b0316336001600160a01b03168c6001600160a01b03167f631042c832b07452973831137f2d73e395028b44b250dedc5abb0ee766e168ac8660a001518760c00151896040516125f193929190615bb0565b60405180910390a460408301805160010190526123aa565b603b5490565b61261761287e565b612620826136a7565b6001600160a01b03841660009081526035602090815260408083206036909252918290209151630e1601cd60e11b81529091733223474c18d19a969a7d6dc6b4ebb210a0843ef291631c2c039a916126819185919088908a90600401615aca565b60006040518083038186803b15801561269957600080fd5b505af41580156126ad573d6000803e3d6000fd5b5050505060048101546001600160a01b03166126c882612fbe565b6126d6828783886000613088565b6126eb6001600160a01b03871633838861364d565b6001820154604051630ab714fb60e11b81526000916001600160a01b0384169163156e29f69161272d9189918b916001600160801b039091169060040161567e565b602060405180830381600087803b15801561274757600080fd5b505af115801561275b573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061277f919061548a565b905080156127f95760078301546001600160a01b03861660009081526037602052604090206127b991600160a01b900460ff166001613702565b846001600160a01b0316876001600160a01b03167e058a56ea94653cdf4f152d227ace22d4c00ad99e2a43f58cb7d9e3feb295f260405160405180910390a35b8361ffff16856001600160a01b0316886001600160a01b03167fde6857219544bb5b7746f48ed30be6386fefc61b2f864cacf559893bf50fd951338a604051612843929190615665565b60405180910390a450505050505050565b603d5490565b603e6020526000908152604090205460ff1681565b6034546001600160a01b031690565b603a546040805180820190915260028152610d8d60f21b60208201529060ff16156117ef5760405162461bcd60e51b81526004016105df919061592a565b60008060008060006128cc614ea2565b6128d58a6140e3565b156128f3576000806000806000199550955095509550955050612d6f565b600060e08201525b878160e001511015612cce5760e0810151612917908b906140e8565b61292057612cbe565b60e0810151600090815260208a81526040808320546001600160a01b03166101e085018190528352908d9052902061295781614167565b506080860181905260c08601929092525060a0840191909152600a0a60208301526101e082015160405163b3596f0760e01b81526001600160a01b038a169163b3596f07916129a99190600401615620565b60206040518083038186803b1580156129c157600080fd5b505afa1580156129d5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906129f991906154a6565b825260c082015115801590612a19575060e0820151612a19908c90614192565b15612b37578060040160009054906101000a90046001600160a01b03166001600160a01b03166370a082318e6040518263ffffffff1660e01b8152600401612a619190615620565b60206040518083038186803b158015612a7957600080fd5b505afa158015612a8d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612ab191906154a6565b6040830181905260208301518351600092612ad092916121f991613f38565b610120840151909150612ae390826134f8565b61012084015260a0830151612b0990612afd908390613f38565b610160850151906134f8565b61016084015260c0830151612b2f90612b23908390613f38565b610180850151906134f8565b610180840152505b60e0820151612b47908c90614218565b15612cbc578060050160009054906101000a90046001600160a01b03166001600160a01b03166370a082318e6040518263ffffffff1660e01b8152600401612b8f9190615620565b60206040518083038186803b158015612ba757600080fd5b505afa158015612bbb573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bdf91906154a6565b826060018181525050612c898160060160009054906101000a90046001600160a01b03166001600160a01b03166370a082318f6040518263ffffffff1660e01b8152600401612c2e9190615620565b60206040518083038186803b158015612c4657600080fd5b505afa158015612c5a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612c7e91906154a6565b6060840151906134f8565b6060830181905260208301518351612cb592612ca992916121f991613f38565b610140840151906134f8565b6101408301525b505b60e08101805160010190526128fb565b600081610120015111612ce2576000612cf7565b610120810151610160820151612cf791613f91565b610160820152610120810151612d0e576000612d23565b610120810151610180820151612d2391613f91565b6101808201819052610120820151610140830151612d4092614297565b610100820181905261012082015161014083015161016084015161018090940151919850965091945090925090505b965096509650965096915050565b6005810154604080516370a0823160e01b81526001600160a01b0385811660048301529151600093849316916370a08231916024808301926020929190829003018186803b158015612dce57600080fd5b505afa158015612de2573d6000803e3d6000fd5b505050506040513d6020811015612df857600080fd5b50516006840154604080516370a0823160e01b81526001600160a01b038881166004830152915191909216916370a08231916024808301926020929190829003018186803b158015612e4957600080fd5b505afa158015612e5d573d6000803e3d6000fd5b505050506040513d6020811015612e7357600080fd5b505190925090505b9250929050565b603454604080516385c858b160e01b8152905133926001600160a01b0316916385c858b1916004808301926020929190829003018186803b158015612ec657600080fd5b505afa158015612eda573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612efe919061505b565b6001600160a01b03161460405180604001604052806002815260200161323760f01b815250906117ef5760405162461bcd60e51b81526004016105df919061592a565b600381015460009064ffffffffff600160801b9091048116904216811415612f7f5750506001810154600160801b90046001600160801b0316610931565b60018301546002840154600091612fb6916001600160801b03600160801b92839004811692612fb0920416856142bb565b906142c8565b949350505050565b60068101546040805163b1bf962d60e01b815290516000926001600160a01b03169163b1bf962d916004808301926020929190829003018186803b15801561300557600080fd5b505afa158015613019573d6000803e3d6000fd5b505050506040513d602081101561302f57600080fd5b505160018301546003840154919250600160801b8082046001600160801b03908116939216910464ffffffffff1660008061306d8787868887614381565b9150915061307f87878785858861453a565b50505050505050565b613090614f3c565b60058601546001600160a01b031680825260408051637b98f4df60e11b8152815163f731e9be92600480840193919291829003018186803b1580156130d457600080fd5b505afa1580156130e8573d6000803e3d6000fd5b505050506040513d60408110156130fe57600080fd5b50805160209182015160c084015260408084019190915260018801546006890154825163b1bf962d60e01b815292516131a294600160801b9093046001600160801b0316936001600160a01b039092169263b1bf962d9260048082019391829003018186803b15801561317057600080fd5b505afa158015613184573d6000803e3d6000fd5b505050506040513d602081101561319a57600080fd5b5051906142c8565b60e082018190526007870154604083015160c08401516001600160a01b03909216926329db497d92899289928992899291906131dd8f61470f565b6040518963ffffffff1660e01b815260040180896001600160a01b03168152602001886001600160a01b031681526020018781526020018681526020018581526020018481526020018381526020018281526020019850505050505050505060606040518083038186803b15801561325457600080fd5b505afa158015613268573d6000803e3d6000fd5b505050506040513d606081101561327e57600080fd5b50805160208083015160409384015160a086015260808501526060840182905282518084019093526002835261353360f01b908301526001600160801b0310156133465760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561330b5781810151838201526020016132f3565b50505050905090810190601f1680156133385780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b506080810151604080518082019091526002815261353560f01b6020820152906001600160801b0310156133bb5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060a08101516040805180820190915260028152610d4d60f21b6020820152906001600160801b0310156134305760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060608181015160028801805460808086015160038c0180546001600160801b03199081166001600160801b038085169190911790925560a0808a015191909516828816178216600160801b82841681029190911790965560018e01546040805198895260208901949094528784019190915280821697870197909752939095049092169183019190915291516001600160a01b038816927f804c9b842b2748a22bb64b345453a3de7ca54a6ca45ce00d415894979e22897a928290030190a2505050505050565b60008282018381101561120e576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b6000828211156135a9576040805162461bcd60e51b815260206004820152601e60248201527f536166654d6174683a207375627472616374696f6e206f766572666c6f770000604482015290519081900360640190fd5b50900390565b604080518082019091526002815261373760f01b6020820152608083106136175760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50816002028161362857600061362b565b60015b60ff16901b826002026001901b19846000015416178360000181905550505050565b604080516001600160a01b0380861660248301528416604482015260648082018490528251808303909101815260849091019091526020810180516001600160e01b03166323b872dd60e01b179052610ebf90859061471a565b6001600160a01b0381166000908152603e60209081526040918290205482518084019093526002835261070760f31b9183019190915260ff16156136fe5760405162461bcd60e51b81526004016105df919061592a565b5050565b604080518082019091526002815261373760f01b60208201526080831061376a5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50816002026001018161377e576000613781565b60015b60ff16901b826002026001016001901b19846000015416178360000181905550505050565b3b151590565b603954603d54604080518082019091526002815261363560f01b60208201529082106137eb5760405162461bcd60e51b81526004016105df919061592a565b506001600160a01b038216600090815260356020526040812060070154600160a01b900460ff1615158061385457506000805260386020527fe14cf4d84b2ff434db2c3d715ad03acb36d95ed6f766d46660154cee72012d71546001600160a01b038481169116145b905080610e8557506001600160a01b03919091166000818152603560209081526040808320600701805460ff60a01b1916600160a01b60ff8816021790558483526038909152902080546001600160a01b0319169091179055600101603955565b80516001600160a01b0390811660009081526035602090815260408083208186015185168452603783528184206034548351631f94a27560e31b81529351929691959491169263fca513a89260048083019392829003018186803b15801561391c57600080fd5b505afa158015613930573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613954919061505b565b845160405163b3596f0760e01b81529192506000916001600160a01b0384169163b3596f07916139879190600401615620565b60206040518083038186803b15801561399f57600080fd5b505afa1580156139b3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906139d791906154a6565b90506000613a016139e7866147cb565b600a0a6121f9886060015185613f3890919063ffffffff16565b86516001600160a01b0316600090815260366020526040908190208189015160608a0151925163bfbba6cb60e01b8152939450733223474c18d19a969a7d6dc6b4ebb210a0843ef29363bfbba6cb93613a63938b939092909190600401615aca565b60006040518083038186803b158015613a7b57600080fd5b505af4158015613a8f573d6000803e3d6000fd5b50505050733223474c18d19a969a7d6dc6b4ebb210a0843ef263721a92f987600001518789604001518a60600151868c60800151603b5460358d60386039548f6040518d63ffffffff1660e01b8152600401613af69c9b9a9998979695949392919061573d565b60006040518083038186803b158015613b0e57600080fd5b505af4158015613b22573d6000803e3d6000fd5b50505050613b2f85612fbe565b600080600188608001516002811115613b4457fe5b6002811115613b4f57fe5b1415613c04576003870154600588015460208a01516040808c015160608d0151915163b3f1c93d60e01b81526001600160801b0390951696506001600160a01b039093169363b3f1c93d93613bab9392909188906004016156dc565b602060405180830381600087803b158015613bc557600080fd5b505af1158015613bd9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613bfd919061548a565b9050613cb3565b600687015460208901516040808b015160608c015160018c0154925163b3f1c93d60e01b81526001600160a01b039095169463b3f1c93d94613c5e9490939291600160801b9091046001600160801b031690600401615634565b602060405180830381600087803b158015613c7857600080fd5b505af1158015613c8c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613cb0919061548a565b90505b8015613cd5576007870154613cd5908790600160a01b900460ff1660016135af565b613d0488600001518960a0015160008b60e00151613cf4576000613cfa565b8b606001515b8b93929190613088565b8760e0015115613d9c578760a001516001600160a01b0316634efecaa589602001518a606001516040518363ffffffff1660e01b8152600401613d48929190615665565b602060405180830381600087803b158015613d6257600080fd5b505af1158015613d76573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613d9a91906154a6565b505b8760c0015161ffff1688604001516001600160a01b031689600001516001600160a01b03167fc6a898309e823ee50bac64e45ca8adba6690e99e7841c45d754e2a38e9019d9b8b602001518c606001518d6080015160016002811115613dfe57fe5b8f608001516002811115613e0e57fe5b6002811115613e1957fe5b14613e385760028e0154600160801b90046001600160801b0316613e3a565b885b604051613e4a94939291906157ed565b60405180910390a45050505050505050565b600080613e6985846147d5565b905083811015613e7d57600091505061120e565b613e878185613552565b95945050505050565b600190565b303b1590565b600381015460009064ffffffffff600160801b9091048116904216811415613ed257505060018101546001600160801b0316610931565b60018301546002840154600091612fb6916001600160801b0391821691612fb0911685614872565b805182511460405180604001604052806002815260200161373360f01b81525090610e855760405162461bcd60e51b81526004016105df919061592a565b600082613f4757506000610957565b82820282848281613f5457fe5b041461120e5760405162461bcd60e51b8152600401808060200182810382526021815260200180615c7c6021913960400191505060405180910390fd5b6000808211613fe7576040805162461bcd60e51b815260206004820152601a60248201527f536166654d6174683a206469766973696f6e206279207a65726f000000000000604482015290519081900360640190fd5b818381613ff057fe5b049392505050565b6000614015614006846148b0565b61400f846148b0565b9061492e565b9050600061402b614024614a35565b83906134f8565b60018601549091506140479082906001600160801b03166142c8565b604080518082019091526002815261353160f01b60208201529091506001600160801b038211156140b95760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060019490940180546001600160801b0319166001600160801b0390951694909417909355505050565b511590565b60006080821060405180604001604052806002815260200161373760f01b815250906141555760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50509051600360029092021c16151590565b5461ffff80821692601083901c821692602081901c831692603082901c60ff169260409290921c1690565b60006080821060405180604001604052806002815260200161373760f01b815250906141ff5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5050815160016002830281019190911c16151592915050565b60006080821060405180604001604052806002815260200161373760f01b815250906142855760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50509051600160029092021c16151590565b6000826142a7575060001961120e565b612fb6836142b586856147d5565b90614a45565b600061120e838342614b3b565b60008215806142d5575081155b156142e257506000610957565b816b019d971e4fe8401e7400000019816142f857fe5b0483111560405180604001604052806002815260200161068760f31b815250906143635760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b506b033b2e3c9fd0803ce80000006002815b048385020181613ff057fe5b600285015460009081906001600160801b03168585821561450b5760006143a88488614872565b90506143b4818a6142c8565b604080518082019091526002815261353160f01b60208201529093506001600160801b038411156144265760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060018b0180546001600160801b0319166001600160801b03851617905589156145095760028b015460009061446c90600160801b90046001600160801b0316896142bb565b9050614478818a6142c8565b6040805180820190915260028152611a9960f11b60208201529093506001600160801b038411156144ea5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b505060018b0180546001600160801b03808516600160801b0291161790555b505b600399909901805464ffffffffff60801b1916600160801b4264ffffffffff1602179055989650505050505050565b614542614f8a565b61454b8761470f565b610120820181905261455d575061174b565b8660050160009054906101000a90046001600160a01b03166001600160a01b031663797743386040518163ffffffff1660e01b815260040160806040518083038186803b1580156145ad57600080fd5b505afa1580156145c1573d6000803e3d6000fd5b505050506040513d60808110156145d757600080fd5b508051602080830151604084015160609094015164ffffffffff1661014086015260a08501939093529183529082015261461186866142c8565b608082015261462086846142c8565b606082015260a0810151610140820151614642919064ffffffffff8516614b3b565b60c082018190526020820151614657916142c8565b6040820181905260808201518251606084015161467c9392610ba692909183916134f8565b60e0820181905261012082015161469391906147d5565b61010082018190521561307f5760048088015461010083015160408051637df5bd3b60e01b81529384019190915260248301879052516001600160a01b0390911691637df5bd3b91604480830192600092919082900301818387803b1580156146fb57600080fd5b505af1158015610646573d6000803e3d6000fd5b5460401c61ffff1690565b600061476f826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316614c179092919063ffffffff16565b805190915015610e855780806020019051602081101561478e57600080fd5b5051610e855760405162461bcd60e51b815260040180806020018281038252602a815260200180615ccb602a913960400191505060405180910390fd5b5460301c60ff1690565b60008215806147e2575081155b156147ef57506000610957565b8161138819816147fb57fe5b0483111560405180604001604052806002815260200161068760f31b815250906148665760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50612710600281614375565b6000806148864264ffffffffff8516613552565b9050612fb6614893614a35565b6301e133806148a28785613f38565b816148a957fe5b04906134f8565b6000633b9aca0082810290839082041460405180604001604052806002815260200161068760f31b815250906149275760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5092915050565b604080518082019091526002815261035360f41b6020820152600090826149965760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060408051808201909152600280825261068760f31b60208301528304906b033b2e3c9fd0803ce8000000821904851115614a125760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5082816b033b2e3c9fd0803ce800000086020181614a2c57fe5b04949350505050565b6b033b2e3c9fd0803ce800000090565b604080518082019091526002815261035360f41b602082015260009082614aad5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060408051808201909152600280825261068760f31b6020830152830490670de0b6b3a7640000821904851115614b255760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b508281670de0b6b3a764000086020181614a2c57fe5b600080614b4f8364ffffffffff8616613552565b905080614b6657614b5e614a35565b91505061120e565b6000198101600060028311614b7c576000614b81565b600283035b90506301e1338087046000614b9682806142c8565b90506000614ba482846142c8565b905060006002614bbe84614bb88a8a613f38565b90613f38565b81614bc557fe5b04905060006006614bdc84614bb889818d8d613f38565b81614be357fe5b049050614c0781614c018481614bf98a8e613f38565b614c01614a35565b906134f8565b9c9b505050505050505050505050565b6060612fb6848460008585614c2b856137a6565b614c7c576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b600080866001600160a01b031685876040518082805190602001908083835b60208310614cba5780518252601f199092019160209182019101614c9b565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114614d1c576040519150601f19603f3d011682016040523d82523d6000602084013e614d21565b606091505b5091509150614d31828286614d3c565b979650505050505050565b60608315614d4b57508161120e565b825115614d5b5782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561330b5781810151838201526020016132f3565b604051806101800160405280614db6614e0d565b815260006020820181905260408201819052606082018190526080820181905260a0820181905260c0820181905260e082018190526101008201819052610120820181905261014082018190526101609091015290565b6040518060200160405280600081525090565b6040518060c001604052806000815260200160008152602001600081526020016000815260200160008152602001600081525090565b6040805161012081018252600080825260208201819052918101829052606081018290526080810182905260a0810182905260c0810182905260e0810182905261010081019190915290565b604051806102400160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160001515815260200160006001600160a01b031681526020016000151581526020016000151581525090565b60405180610100016040528060006001600160a01b03168152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b60405180610160016040528060008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600064ffffffffff1681525090565b60008083601f840112614ffc578081fd5b50813567ffffffffffffffff811115615013578182fd5b6020830191508360208083028501011115612e7b57600080fd5b803561ffff8116811461093157600080fd5b600060208284031215615050578081fd5b813561120e81615c58565b60006020828403121561506c578081fd5b815161120e81615c58565b60008060408385031215615089578081fd5b823561509481615c58565b915060208301356150a481615c58565b809150509250929050565b600080600080600060a086880312156150c6578081fd5b85356150d181615c58565b945060208601356150e181615c58565b935060408601356150f181615c58565b9250606086013561510181615c58565b9150608086013561511181615c58565b809150509295509295909350565b600080600080600060a08688031215615136578081fd5b853561514181615c58565b9450602086013561515181615c58565b9350604086013561516181615c58565b925060608601359150608086013561511181615c6d565b60008060008060008060c08789031215615190578081fd5b863561519b81615c58565b955060208701356151ab81615c58565b945060408701356151bb81615c58565b959894975094956060810135955060808101359460a0909101359350915050565b60008060008060008060008060a0898b0312156151f7578182fd5b883561520281615c58565b9750602089013567ffffffffffffffff8082111561521e578384fd5b61522a8c838d01614feb565b909950975060408b0135915080821115615242578384fd5b61524e8c838d01614feb565b909750955060608b0135915080821115615266578384fd5b818b0191508b601f830112615279578384fd5b813581811115615287578485fd5b8c6020828501011115615298578485fd5b6020830195508094505050506152b060808a0161502d565b90509295985092959890939650565b600080604083850312156152d1578182fd5b82356152dc81615c58565b915060208301356150a481615c6d565b60008082840360e08112156152ff578283fd5b833561530a81615c58565b925060c0601f198201121561531d578182fd5b506020830190509250929050565b6000806040838503121561533d578182fd5b823561534881615c58565b946020939093013593505050565b60008060006060848603121561536a578081fd5b833561537581615c58565b925060208401359150604084013561538c81615c58565b809150509250925092565b600080600080608085870312156153ac578182fd5b84356153b781615c58565b93506020850135925060408501356153ce81615c58565b91506153dc6060860161502d565b905092959194509250565b600080600080608085870312156153fc578182fd5b843561540781615c58565b93506020850135925060408501359150606085013561542581615c58565b939692955090935050565b600080600080600060a08688031215615447578283fd5b853561545281615c58565b945060208601359350604086013592506151016060870161502d565b60006020828403121561547f578081fd5b813561120e81615c6d565b60006020828403121561549b578081fd5b815161120e81615c6d565b6000602082840312156154b7578081fd5b5051919050565b600080604083850312156154d0578182fd5b82519150602083015167ffffffffffffffff808211156154ee578283fd5b818501915085601f830112615501578283fd5b81518181111561550d57fe5b604051601f8201601f19168101602001838111828210171561552b57fe5b604052818152838201602001881015615542578485fd5b615553826020830160208701615bf2565b809450505050509250929050565b6001600160a01b03169052565b6000815180845260208085019450808401835b8381101561559d57815187529582019590820190600101615581565b509495945050505050565b60008284528282602086013780602084860101526020601f19601f85011685010190509392505050565b600381106155dc57fe5b9052565b519052565b6001600160801b03169052565b64ffffffffff169052565b60ff169052565b60008251615616818460208701615bf2565b9190910192915050565b6001600160a01b0391909116815260200190565b6001600160a01b03948516815292909316602083015260408201526001600160801b03909116606082015260800190565b6001600160a01b03929092168252602082015260400190565b6001600160a01b0393909316835260208301919091526001600160801b0316604082015260600190565b6001600160a01b03958616815293851660208501529190931660408301526060820192909252901515608082015260a00190565b6001600160a01b0394851681529290931660208301526040820152606081019190915260800190565b6001600160a01b039687168152602081019590955260408501939093526060840191909152608083015290911660a082015260c00190565b6001600160a01b039c8d168152602081019b909b52988b1660408b015260608a0197909752608089019590955260a088019390935260c087019190915260e08601526101008501526101208401526101408301529091166101608201526101800190565b6001600160a01b03998a168152602081019890985260408801969096526060870194909452608086019290925260a085015260c084015260e08301529091166101008201526101200190565b6001600160a01b0394909416845260208401929092526040830152606082015260800190565b60a0808252810188905260008960c08301825b8b81101561585657823561583981615c58565b6001600160a01b0316825260209283019290910190600101615826565b5083810360208501528881526001600160fb1b03891115615875578283fd5b602089029150818a602083013701602081810183815284830390910160408501526158a0818961556e565b9150506158b06060840187615561565b82810360808401526158c38185876155a8565b9b9a5050505050505050505050565b6020808252825182820181905260009190848201906040850190845b818110156159135783516001600160a01b0316835292840192918401916001016158ee565b50909695505050505050565b901515815260200190565b6000602082528251806020840152615949816040850160208701615bf2565b601f01601f19169190910160400192915050565b9051815260200190565b60006101808201905061597b8284516155e0565b602083015161598d60208401826155e5565b5060408301516159a060408401826155e5565b5060608301516159b360608401826155e5565b5060808301516159c660808401826155e5565b5060a08301516159d960a08401826155e5565b5060c08301516159ec60c08401826155f2565b5060e08301516159ff60e0840182615561565b5061010080840151615a1382850182615561565b505061012080840151615a2882850182615561565b505061014080840151615a3d82850182615561565b505061016080840151615a52828501826155fd565b505092915050565b9485526001600160a01b03938416602086015291831660408501528216606084015216608082015260a00190565b9788526001600160a01b03968716602089015294151560408801526060870193909352608086019190915260a085015260c08401521660e08201526101000190565b93845260208401929092526001600160a01b03166040830152606082015260800190565b600060a082019050868252856020830152846040830152836060830152615b1860808301846155d2565b9695505050505050565b8681526020810186905260c08101615b3d60408301876155d2565b6001600160a01b03949094166060820152608081019290925260a0909101529392505050565b600060c082019050825182526020830151602083015260408301516040830152606083015160608301526080830151608083015260a083015160a083015292915050565b90815260200190565b928352602083019190915261ffff16604082015260600190565b958652602086019490945260408501929092526060840152608083015260a082015260c00190565b60005b83811015615c0d578181015183820152602001615bf5565b83811115610ebf5750506000910152565b813581556020820135600182015560408201356002820155606082013560038201556080820135600482015560a082013560058201555050565b6001600160a01b03811681146117ef57600080fd5b80151581146117ef57600080fdfe536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f77436f6e747261637420696e7374616e63652068617320616c7265616479206265656e20696e697469616c697a65645361666545524332303a204552433230206f7065726174696f6e20646964206e6f742073756363656564a26469706673582212200cd40c425c6db626729fc3c8744a3eba9342d8768634406c3fa2ba81877612aa64736f6c63430007060033

Deployed ByteCode

0x608060405234801561001057600080fd5b50600436106101ef5760003560e01c8063a415bcad1161010f578063d1946dbc116100a2578063e8eda9df11610071578063e8eda9df1461043c578063f8119d511461044f578063fe575a8714610457578063fe65acfe1461046a576101ef565b8063d1946dbc146103f9578063d5ed39331461040e578063e7e7d62a14610421578063e82fec2f14610434576101ef565b8063c44b11f7116100de578063c44b11f7146103ad578063c4d66de8146103c0578063cd112382146103d3578063d15e0053146103e6576101ef565b8063a415bcad1461034f578063b8d2927614610362578063bedb86fb14610375578063bf92857c14610388576101ef565b80635a3b74b9116101875780637a708e92116101565780637a708e9214610301578063845e79bc146103145780638afaff021461033457806394ba89a21461033c576101ef565b80635a3b74b9146102b35780635c975abb146102c657806363a5d53b146102db57806369328dec146102ee576101ef565b806335ea6a75116101c357806335ea6a751461024d578063386497fd1461026d5780634417a58314610280578063573ade81146102a0576101ef565b8062a718a9146101f4578063074b2e4314610209578063158c13e8146102275780631d2118f91461023a575b600080fd5b61020761020236600461511f565b61047f565b005b610211610653565b60405161021e9190615ba7565b60405180910390f35b61020761023536600461503f565b610659565b610207610248366004615077565b61081b565b61026061025b36600461503f565b610854565b60405161021e9190615967565b61021161027b36600461503f565b610936565b61029361028e36600461503f565b61095d565b60405161021e919061595d565b6102116102ae3660046153e7565b610990565b6102076102c13660046152bf565b610cbc565b6102ce610e8a565b60405161021e919061591f565b6102076102e93660046152ec565b610e93565b6102116102fc366004615356565b610ec5565b61020761030f3660046150af565b611215565b61032761032236600461503f565b6112f7565b60405161021e9190615b63565b610211611361565b61020761034a36600461532b565b611366565b61020761035d366004615430565b6116d3565b61020761037036600461532b565b611753565b61020761038336600461546e565b611777565b61039b61039636600461503f565b6117f2565b60405161021e96959493929190615bca565b6102936103bb36600461503f565b6118b1565b6102076103ce36600461503f565b6118e4565b6102076103e1366004615077565b6119ab565b6102116103f436600461503f565b611c21565b610401611c42565b60405161021e91906158d2565b61020761041c366004615178565b611ce8565b61020761042f3660046151dc565b61200f565b610211612609565b61020761044a366004615397565b61260f565b610211612854565b6102ce61046536600461503f565b61285a565b61047261286f565b60405161021e9190615620565b61048761287e565b6034546040805163712d917160e01b815290516000926001600160a01b03169163712d9171916004808301926020929190829003018186803b1580156104cc57600080fd5b505afa1580156104e0573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610504919061505b565b9050600080826001600160a01b0316888888888860405160240161052c9594939291906156a8565b60408051601f198184030181529181526020820180516001600160e01b031662a718a960e01b179052516105609190615604565b600060405180830381855af49150503d806000811461059b576040519150601f19603f3d011682016040523d82523d6000602084013e6105a0565b606091505b50915091508160405180604001604052806002815260200161323360f01b815250906105e85760405162461bcd60e51b81526004016105df919061592a565b60405180910390fd5b506000808280602001905181019061060091906154be565b9150915081600014816040516020016106199190615604565b604051602081830303815290604052906106465760405162461bcd60e51b81526004016105df919061592a565b5050505050505050505050565b603c5490565b61066161287e565b6001600160a01b03808216600090815260376020908152604080832081518084018352905481526039546034548351631f94a27560e31b81529351959661070d968996603596603895949093169263fca513a89260048083019392829003018186803b1580156106d057600080fd5b505afa1580156106e4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610708919061505b565b6128bc565b9450505050508060001460405180604001604052806002815260200161383960f01b815250906107505760405162461bcd60e51b81526004016105df919061592a565b5060005b6039548110156107cd576000818152603860209081526040808320546001600160a01b0316808452603590925282209091908190610793908790612d7d565b909250905081156107ad576107ab8383600189610990565b505b80156107c2576107c08382600289610990565b505b505050600101610754565b506001600160a01b0382166000818152603e6020526040808220805460ff19166001179055517f7811ed4e4b787e4f0f046ffb4a5ec1e064b0eb6d191ffabb66270ed8be89b3259190a25050565b610823612e82565b6001600160a01b03918216600090815260356020526040902060070180546001600160a01b03191691909216179055565b61085c614da2565b506001600160a01b0381811660009081526035602090815260409182902082516101a08101845281546101808201908152815260018201546001600160801b0380821694830194909452600160801b908190048416948201949094526002820154808416606083015284900483166080820152600382015492831660a08201529290910464ffffffffff1660c08301526004810154831660e0830152600581015483166101008301526006810154831661012083015260070154918216610140820152600160a01b90910460ff166101608201525b919050565b6001600160a01b038116600090815260356020526040812061095790612f41565b92915050565b610965614e0d565b506001600160a01b031660009081526037602090815260409182902082519182019092529054815290565b600061099a61287e565b6001600160a01b038516600090815260356020526040812090806109be8584612d7d565b9150915060008660028111156109d057fe5b60405163fa0c214960e01b8152909150733223474c18d19a969a7d6dc6b4ebb210a0843ef29063fa0c214990610a149087908c9086908c908a908a90600401615b22565b60006040518083038186803b158015610a2c57600080fd5b505af4158015610a40573d6000803e3d6000fd5b506000925060019150610a509050565b826002811115610a5c57fe5b14610a675782610a69565b835b905080891015610a765750875b610a7f85612fbe565b6001826002811115610a8d57fe5b1415610afe576005850154604051632770a7eb60e21b81526001600160a01b0390911690639dc29fac90610ac7908a908590600401615665565b600060405180830381600087803b158015610ae157600080fd5b505af1158015610af5573d6000803e3d6000fd5b50505050610b7c565b60068501546001860154604051637a94c56560e11b81526001600160a01b039092169163f5298aca91610b49918b918691600160801b9091046001600160801b03169060040161567e565b600060405180830381600087803b158015610b6357600080fd5b505af1158015610b77573d6000803e3d6000fd5b505050505b60048501546001600160a01b0316610b98868c83856000613088565b610bac82610ba687876134f8565b90613552565b610be45760078601546001600160a01b0389166000908152603760205260408120610be4929091600160a01b90910460ff16906135af565b610bf96001600160a01b038c1633838561364d565b6040516388dd91a160e01b81526001600160a01b038216906388dd91a190610c279033908690600401615665565b600060405180830381600087803b158015610c4157600080fd5b505af1158015610c55573d6000803e3d6000fd5b50505050336001600160a01b0316886001600160a01b03168c6001600160a01b03167f4cdde6e09bb755c9a5589ebaec640bbfedff1362d4b255ebf8339782b9942faa85604051610ca69190615ba7565b60405180910390a4509998505050505050505050565b610cc461287e565b610ccd336136a7565b6001600160a01b03808316600090815260356020818152604080842033855260378352938190206039546034548351631f94a27560e31b815293519697733223474c18d19a969a7d6dc6b4ebb210a0843ef297635fa297e5978a978d978d9792969295603895939493169263fca513a892600480840193919291829003018186803b158015610d5b57600080fd5b505afa158015610d6f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d93919061505b565b6040518963ffffffff1660e01b8152600401610db6989796959493929190615a88565b60006040518083038186803b158015610dce57600080fd5b505af4158015610de2573d6000803e3d6000fd5b505050506007810154336000908152603760205260409020610e0e91600160a01b900460ff1684613702565b8115610e4e5760405133906001600160a01b038516907e058a56ea94653cdf4f152d227ace22d4c00ad99e2a43f58cb7d9e3feb295f290600090a3610e85565b60405133906001600160a01b038516907f44c58d81365b66dd4b1a7f36c25aa97b8c71c361ee4937adc1a00000227db5dd90600090a35b505050565b603a5460ff1690565b610e9b612e82565b6001600160a01b03821660009081526036602052604090208190610ebf8282615c1e565b50505050565b6000610ecf61287e565b6001600160a01b0380851660009081526035602052604080822060048082015492516370a0823160e01b8152919492909216929183916370a0823191610f1791339101615620565b60206040518083038186803b158015610f2f57600080fd5b505afa158015610f43573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610f6791906154a6565b905085600019811415610f775750805b733223474c18d19a969a7d6dc6b4ebb210a0843ef263c5b18e768983856035603660008f6001600160a01b03166001600160a01b0316815260200190815260200160002060376000336001600160a01b03166001600160a01b031681526020019081526020016000206038603954603460009054906101000a90046001600160a01b03166001600160a01b031663fca513a86040518163ffffffff1660e01b815260040160206040518083038186803b15801561103357600080fd5b505afa158015611047573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061106b919061505b565b6040518a63ffffffff1660e01b815260040161108f999897969594939291906157a1565b60006040518083038186803b1580156110a757600080fd5b505af41580156110bb573d6000803e3d6000fd5b505050506110c884612fbe565b6110d6848985600085613088565b81811415611140576007840154336000908152603760205260408120611109929091600160a01b90910460ff1690613702565b60405133906001600160a01b038a16907f44c58d81365b66dd4b1a7f36c25aa97b8c71c361ee4937adc1a00000227db5dd90600090a35b6001840154604051636b81068560e11b81526001600160a01b0385169163d7020d0a916111819133918b9187916001600160801b0390911690600401615634565b600060405180830381600087803b15801561119b57600080fd5b505af11580156111af573d6000803e3d6000fd5b50505050856001600160a01b0316336001600160a01b0316896001600160a01b03167f3115d1449a7b732c986cba18244e897a450f61e1bb8d589cd2e69e6c8924f9f7846040516112009190615ba7565b60405180910390a493505050505b9392505050565b61121d612e82565b611226856137a6565b6040518060400160405280600281526020016106e760f31b8152509061125f5760405162461bcd60e51b81526004016105df919061592a565b506001600160a01b038516600090815260356020526040908190209051630acce25f60e21b8152730da77bca9b08c307c64911313214514d1b93988691632b33897c916112b791908890889088908890600401615a5a565b60006040518083038186803b1580156112cf57600080fd5b505af41580156112e3573d6000803e3d6000fd5b505050506112f0856137ac565b5050505050565b6112ff614e20565b506001600160a01b0316600090815260366020908152604091829020825160c08101845281548152600182015492810192909252600281015492820192909252600382015460608201526004820154608082015260059091015460a082015290565b600181565b61136e61287e565b6001600160a01b038216600090815260356020526040812090806113923384612d7d565b9150915060008460028111156113a457fe5b3360009081526037602052604090819020905163a8695b1d60e01b8152919250733223474c18d19a969a7d6dc6b4ebb210a0843ef29163a8695b1d916113f591889190889088908890600401615aee565b60006040518083038186803b15801561140d57600080fd5b505af4158015611421573d6000803e3d6000fd5b5050505061142e84612fbe565b600181600281111561143c57fe5b141561154c576005840154604051632770a7eb60e21b81526001600160a01b0390911690639dc29fac906114769033908790600401615665565b600060405180830381600087803b15801561149057600080fd5b505af11580156114a4573d6000803e3d6000fd5b505050506006840154600185015460405163b3f1c93d60e01b81526001600160a01b039092169163b3f1c93d916114f491339182918991600160801b90046001600160801b031690600401615634565b602060405180830381600087803b15801561150e57600080fd5b505af1158015611522573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611546919061548a565b50611662565b60068401546001850154604051637a94c56560e11b81526001600160a01b039092169163f5298aca916115979133918791600160801b9091046001600160801b03169060040161567e565b600060405180830381600087803b1580156115b157600080fd5b505af11580156115c5573d6000803e3d6000fd5b505050506005840154600385015460405163b3f1c93d60e01b81526001600160a01b039092169163b3f1c93d9161160e913391829188916001600160801b031690600401615634565b602060405180830381600087803b15801561162857600080fd5b505af115801561163c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611660919061548a565b505b600484015461168090859088906001600160a01b0316600080613088565b336001600160a01b0316866001600160a01b03167fea368a40e9570069bb8e6511d668293ad2e1f03b0d982431fd223de9f3b70ca6876040516116c39190615ba7565b60405180910390a3505050505050565b6116db61287e565b6001600160a01b038086166000818152603560209081526040918290208251610100810184529384523391840191909152848416918301919091526060820187905260808201869052600481015490921660a082015261ffff841660c0820152600160e082015261174b906138b5565b505050505050565b61175b612e82565b6001600160a01b03909116600090815260356020526040902055565b61177f612e82565b603a805460ff1916821515179081905560ff16156117c5576040517f9e87fac88ff661f02d44f95383c817fece4bce600a3dab7a54406878b965e75290600090a16117ef565b6040517fa45f47fdea8a1efdd9029a5691c7f759c32b7c698632b563573e155625d1693390600090a15b50565b60008060008060008061188e876035603760008b6001600160a01b03166001600160a01b031681526020019081526020016000206040518060200160405290816000820154815250506038603954603460009054906101000a90046001600160a01b03166001600160a01b031663fca513a86040518163ffffffff1660e01b815260040160206040518083038186803b1580156106d057600080fd5b939950919750909450925090506118a6868684613e5c565b935091939550919395565b6118b9614e0d565b506001600160a01b031660009081526035602090815260409182902082519182019092529054815290565b60006118ee613e90565b60015490915060ff16806119055750611905613e95565b80611911575060005481115b61194c5760405162461bcd60e51b815260040180806020018281038252602e815260200180615c9d602e913960400191505060405180910390fd5b60015460ff1615801561196b576001805460ff19168117905560008290555b603480546001600160a01b0319166001600160a01b0385161790556109c4603b556009603c556080603d558015610e85576001805460ff19169055505050565b6119b361287e565b6001600160a01b038083166000908152603560205260408082206005810154600682015460048084015494516370a0823160e01b81529396928316959183169490921692909185916370a0823191611a0d918a9101615620565b60206040518083038186803b158015611a2557600080fd5b505afa158015611a39573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a5d91906154a6565b60405163548cad0960e01b8152909150733223474c18d19a969a7d6dc6b4ebb210a0843ef29063548cad0990611a9f9088908b90899089908990600401615a5a565b60006040518083038186803b158015611ab757600080fd5b505af4158015611acb573d6000803e3d6000fd5b50505050611ad885612fbe565b604051632770a7eb60e21b81526001600160a01b03851690639dc29fac90611b069089908590600401615665565b600060405180830381600087803b158015611b2057600080fd5b505af1158015611b34573d6000803e3d6000fd5b505050600386015460405163b3f1c93d60e01b81526001600160a01b038716925063b3f1c93d91611b77918a91829187916001600160801b031690600401615634565b602060405180830381600087803b158015611b9157600080fd5b505af1158015611ba5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bc9919061548a565b50611bd8858884600080613088565b856001600160a01b0316876001600160a01b03167f9f439ae0c81e41a04d3fdfe07aed54e6a179fb0db15be7702eb66fa8ef6f530060405160405180910390a350505050505050565b6001600160a01b038116600090815260356020526040812061095790613e9b565b6060600060395467ffffffffffffffff81118015611c5f57600080fd5b50604051908082528060200260200182016040528015611c89578160200160208202803683370190505b50905060005b603954811015611ce25760008181526038602052604090205482516001600160a01b0390911690839083908110611cc257fe5b6001600160a01b0390921660209283029190910190910152600101611c8f565b50905090565b611cf061287e565b6001600160a01b038681166000908152603560209081526040918290206004015482518084019093526002835261363360f01b918301919091529091163314611d4c5760405162461bcd60e51b81526004016105df919061592a565b5060345460408051631f94a27560e31b815290516000926001600160a01b03169163fca513a8916004808301926020929190829003018186803b158015611d9257600080fd5b505afa158015611da6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dca919061505b565b6001600160a01b03881660009081526035602090815260408083206036909252918290209151633ef9e4fb60e01b8152929350733223474c18d19a969a7d6dc6b4ebb210a0843ef292633ef9e4fb92611e2a928b928b92906004016156dc565b60006040518083038186803b158015611e4257600080fd5b505af4158015611e56573d6000803e3d6000fd5b5050506001600160a01b03871660009081526037602052604090819020603954915163516d1bc760e01b8152733223474c18d19a969a7d6dc6b4ebb210a0843ef2935063516d1bc792611eb5928b926035926038918990600401615705565b60006040518083038186803b158015611ecd57600080fd5b505af4158015611ee1573d6000803e3d6000fd5b505050506001600160a01b03878116600090815260356020526040902060070154600160a01b900460ff16908681169088161461200557611f228486613552565b611f8e576001600160a01b038716600090815260376020526040812090611f4c9082908490613702565b876001600160a01b0316896001600160a01b03167f44c58d81365b66dd4b1a7f36c25aa97b8c71c361ee4937adc1a00000227db5dd60405160405180910390a3505b82158015611f9b57508415155b15612005576001600160a01b0386166000908152603760205260409020611fc481836001613702565b866001600160a01b0316896001600160a01b03167e058a56ea94653cdf4f152d227ace22d4c00ad99e2a43f58cb7d9e3feb295f260405160405180910390a3505b5050505050505050565b61201761287e565b61201f614e56565b61208c88888080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050604080516020808c0282810182019093528b82529093508b92508a918291850190849080828437600092019190915250613efa92505050565b60008767ffffffffffffffff811180156120a557600080fd5b506040519080825280602002602001820160405280156120cf578160200160208202803683370190505b50905060008867ffffffffffffffff811180156120eb57600080fd5b50604051908082528060200260200182016040528015612115578160200160208202803683370190505b506001600160a01b038c1684526000604085015290505b60408301518911156122d957603560008b8b866040015181811061214c57fe5b9050602002016020810190612161919061503f565b6001600160a01b03166001600160a01b0316815260200190815260200160002060040160009054906101000a90046001600160a01b0316828460400151815181106121a857fe5b60200260200101906001600160a01b031690816001600160a01b0316815250506121ff6127106121f9603c548b8b88604001518181106121e457fe5b90506020020135613f3890919063ffffffff16565b90613f91565b8184604001518151811061220f57fe5b6020026020010181815250508183604001518151811061222b57fe5b60200260200101516001600160a01b0316634efecaa58c8a8a876040015181811061225257fe5b905060200201356040518363ffffffff1660e01b8152600401612276929190615665565b602060405180830381600087803b15801561229057600080fd5b505af11580156122a4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122c891906154a6565b50604083018051600101905261212c565b8251604051632483d72160e21b81526001600160a01b039091169063920f5c8490612316908d908d908d908d90889033908f908f90600401615813565b602060405180830381600087803b15801561233057600080fd5b505af1158015612344573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612368919061548a565b604051806040016040528060028152602001611b1b60f11b815250906123a15760405162461bcd60e51b81526004016105df919061592a565b50600060408401525b604083015189111561064657898984604001518181106123c657fe5b90506020020160208101906123db919061503f565b6001600160a01b031660608401526040830151889088908181106123fb57fe5b905060200201358360a00181815250508083604001518151811061241b57fe5b60200260200101518360c00181815250508183604001518151811061243c57fe5b60209081029190910101516001600160a01b0316608084015260c083015160a0840151612468916134f8565b60e084015260608301516001600160a01b0316600090815260356020526040902061249290612fbe565b61253283608001516001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156124d257600080fd5b505afa1580156124e6573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061250a91906154a6565b60c085015160608601516001600160a01b031660009081526035602052604090209190613ff8565b6060830151608084015160e08501516001600160a01b038316600090815260356020526040812061256894909390929091613088565b6125948b84608001518560e0015186606001516001600160a01b031661364d909392919063ffffffff16565b82606001516001600160a01b0316336001600160a01b03168c6001600160a01b03167f631042c832b07452973831137f2d73e395028b44b250dedc5abb0ee766e168ac8660a001518760c00151896040516125f193929190615bb0565b60405180910390a460408301805160010190526123aa565b603b5490565b61261761287e565b612620826136a7565b6001600160a01b03841660009081526035602090815260408083206036909252918290209151630e1601cd60e11b81529091733223474c18d19a969a7d6dc6b4ebb210a0843ef291631c2c039a916126819185919088908a90600401615aca565b60006040518083038186803b15801561269957600080fd5b505af41580156126ad573d6000803e3d6000fd5b5050505060048101546001600160a01b03166126c882612fbe565b6126d6828783886000613088565b6126eb6001600160a01b03871633838861364d565b6001820154604051630ab714fb60e11b81526000916001600160a01b0384169163156e29f69161272d9189918b916001600160801b039091169060040161567e565b602060405180830381600087803b15801561274757600080fd5b505af115801561275b573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061277f919061548a565b905080156127f95760078301546001600160a01b03861660009081526037602052604090206127b991600160a01b900460ff166001613702565b846001600160a01b0316876001600160a01b03167e058a56ea94653cdf4f152d227ace22d4c00ad99e2a43f58cb7d9e3feb295f260405160405180910390a35b8361ffff16856001600160a01b0316886001600160a01b03167fde6857219544bb5b7746f48ed30be6386fefc61b2f864cacf559893bf50fd951338a604051612843929190615665565b60405180910390a450505050505050565b603d5490565b603e6020526000908152604090205460ff1681565b6034546001600160a01b031690565b603a546040805180820190915260028152610d8d60f21b60208201529060ff16156117ef5760405162461bcd60e51b81526004016105df919061592a565b60008060008060006128cc614ea2565b6128d58a6140e3565b156128f3576000806000806000199550955095509550955050612d6f565b600060e08201525b878160e001511015612cce5760e0810151612917908b906140e8565b61292057612cbe565b60e0810151600090815260208a81526040808320546001600160a01b03166101e085018190528352908d9052902061295781614167565b506080860181905260c08601929092525060a0840191909152600a0a60208301526101e082015160405163b3596f0760e01b81526001600160a01b038a169163b3596f07916129a99190600401615620565b60206040518083038186803b1580156129c157600080fd5b505afa1580156129d5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906129f991906154a6565b825260c082015115801590612a19575060e0820151612a19908c90614192565b15612b37578060040160009054906101000a90046001600160a01b03166001600160a01b03166370a082318e6040518263ffffffff1660e01b8152600401612a619190615620565b60206040518083038186803b158015612a7957600080fd5b505afa158015612a8d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612ab191906154a6565b6040830181905260208301518351600092612ad092916121f991613f38565b610120840151909150612ae390826134f8565b61012084015260a0830151612b0990612afd908390613f38565b610160850151906134f8565b61016084015260c0830151612b2f90612b23908390613f38565b610180850151906134f8565b610180840152505b60e0820151612b47908c90614218565b15612cbc578060050160009054906101000a90046001600160a01b03166001600160a01b03166370a082318e6040518263ffffffff1660e01b8152600401612b8f9190615620565b60206040518083038186803b158015612ba757600080fd5b505afa158015612bbb573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bdf91906154a6565b826060018181525050612c898160060160009054906101000a90046001600160a01b03166001600160a01b03166370a082318f6040518263ffffffff1660e01b8152600401612c2e9190615620565b60206040518083038186803b158015612c4657600080fd5b505afa158015612c5a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612c7e91906154a6565b6060840151906134f8565b6060830181905260208301518351612cb592612ca992916121f991613f38565b610140840151906134f8565b6101408301525b505b60e08101805160010190526128fb565b600081610120015111612ce2576000612cf7565b610120810151610160820151612cf791613f91565b610160820152610120810151612d0e576000612d23565b610120810151610180820151612d2391613f91565b6101808201819052610120820151610140830151612d4092614297565b610100820181905261012082015161014083015161016084015161018090940151919850965091945090925090505b965096509650965096915050565b6005810154604080516370a0823160e01b81526001600160a01b0385811660048301529151600093849316916370a08231916024808301926020929190829003018186803b158015612dce57600080fd5b505afa158015612de2573d6000803e3d6000fd5b505050506040513d6020811015612df857600080fd5b50516006840154604080516370a0823160e01b81526001600160a01b038881166004830152915191909216916370a08231916024808301926020929190829003018186803b158015612e4957600080fd5b505afa158015612e5d573d6000803e3d6000fd5b505050506040513d6020811015612e7357600080fd5b505190925090505b9250929050565b603454604080516385c858b160e01b8152905133926001600160a01b0316916385c858b1916004808301926020929190829003018186803b158015612ec657600080fd5b505afa158015612eda573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612efe919061505b565b6001600160a01b03161460405180604001604052806002815260200161323760f01b815250906117ef5760405162461bcd60e51b81526004016105df919061592a565b600381015460009064ffffffffff600160801b9091048116904216811415612f7f5750506001810154600160801b90046001600160801b0316610931565b60018301546002840154600091612fb6916001600160801b03600160801b92839004811692612fb0920416856142bb565b906142c8565b949350505050565b60068101546040805163b1bf962d60e01b815290516000926001600160a01b03169163b1bf962d916004808301926020929190829003018186803b15801561300557600080fd5b505afa158015613019573d6000803e3d6000fd5b505050506040513d602081101561302f57600080fd5b505160018301546003840154919250600160801b8082046001600160801b03908116939216910464ffffffffff1660008061306d8787868887614381565b9150915061307f87878785858861453a565b50505050505050565b613090614f3c565b60058601546001600160a01b031680825260408051637b98f4df60e11b8152815163f731e9be92600480840193919291829003018186803b1580156130d457600080fd5b505afa1580156130e8573d6000803e3d6000fd5b505050506040513d60408110156130fe57600080fd5b50805160209182015160c084015260408084019190915260018801546006890154825163b1bf962d60e01b815292516131a294600160801b9093046001600160801b0316936001600160a01b039092169263b1bf962d9260048082019391829003018186803b15801561317057600080fd5b505afa158015613184573d6000803e3d6000fd5b505050506040513d602081101561319a57600080fd5b5051906142c8565b60e082018190526007870154604083015160c08401516001600160a01b03909216926329db497d92899289928992899291906131dd8f61470f565b6040518963ffffffff1660e01b815260040180896001600160a01b03168152602001886001600160a01b031681526020018781526020018681526020018581526020018481526020018381526020018281526020019850505050505050505060606040518083038186803b15801561325457600080fd5b505afa158015613268573d6000803e3d6000fd5b505050506040513d606081101561327e57600080fd5b50805160208083015160409384015160a086015260808501526060840182905282518084019093526002835261353360f01b908301526001600160801b0310156133465760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561330b5781810151838201526020016132f3565b50505050905090810190601f1680156133385780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b506080810151604080518082019091526002815261353560f01b6020820152906001600160801b0310156133bb5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060a08101516040805180820190915260028152610d4d60f21b6020820152906001600160801b0310156134305760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060608181015160028801805460808086015160038c0180546001600160801b03199081166001600160801b038085169190911790925560a0808a015191909516828816178216600160801b82841681029190911790965560018e01546040805198895260208901949094528784019190915280821697870197909752939095049092169183019190915291516001600160a01b038816927f804c9b842b2748a22bb64b345453a3de7ca54a6ca45ce00d415894979e22897a928290030190a2505050505050565b60008282018381101561120e576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b6000828211156135a9576040805162461bcd60e51b815260206004820152601e60248201527f536166654d6174683a207375627472616374696f6e206f766572666c6f770000604482015290519081900360640190fd5b50900390565b604080518082019091526002815261373760f01b6020820152608083106136175760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50816002028161362857600061362b565b60015b60ff16901b826002026001901b19846000015416178360000181905550505050565b604080516001600160a01b0380861660248301528416604482015260648082018490528251808303909101815260849091019091526020810180516001600160e01b03166323b872dd60e01b179052610ebf90859061471a565b6001600160a01b0381166000908152603e60209081526040918290205482518084019093526002835261070760f31b9183019190915260ff16156136fe5760405162461bcd60e51b81526004016105df919061592a565b5050565b604080518082019091526002815261373760f01b60208201526080831061376a5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50816002026001018161377e576000613781565b60015b60ff16901b826002026001016001901b19846000015416178360000181905550505050565b3b151590565b603954603d54604080518082019091526002815261363560f01b60208201529082106137eb5760405162461bcd60e51b81526004016105df919061592a565b506001600160a01b038216600090815260356020526040812060070154600160a01b900460ff1615158061385457506000805260386020527fe14cf4d84b2ff434db2c3d715ad03acb36d95ed6f766d46660154cee72012d71546001600160a01b038481169116145b905080610e8557506001600160a01b03919091166000818152603560209081526040808320600701805460ff60a01b1916600160a01b60ff8816021790558483526038909152902080546001600160a01b0319169091179055600101603955565b80516001600160a01b0390811660009081526035602090815260408083208186015185168452603783528184206034548351631f94a27560e31b81529351929691959491169263fca513a89260048083019392829003018186803b15801561391c57600080fd5b505afa158015613930573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613954919061505b565b845160405163b3596f0760e01b81529192506000916001600160a01b0384169163b3596f07916139879190600401615620565b60206040518083038186803b15801561399f57600080fd5b505afa1580156139b3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906139d791906154a6565b90506000613a016139e7866147cb565b600a0a6121f9886060015185613f3890919063ffffffff16565b86516001600160a01b0316600090815260366020526040908190208189015160608a0151925163bfbba6cb60e01b8152939450733223474c18d19a969a7d6dc6b4ebb210a0843ef29363bfbba6cb93613a63938b939092909190600401615aca565b60006040518083038186803b158015613a7b57600080fd5b505af4158015613a8f573d6000803e3d6000fd5b50505050733223474c18d19a969a7d6dc6b4ebb210a0843ef263721a92f987600001518789604001518a60600151868c60800151603b5460358d60386039548f6040518d63ffffffff1660e01b8152600401613af69c9b9a9998979695949392919061573d565b60006040518083038186803b158015613b0e57600080fd5b505af4158015613b22573d6000803e3d6000fd5b50505050613b2f85612fbe565b600080600188608001516002811115613b4457fe5b6002811115613b4f57fe5b1415613c04576003870154600588015460208a01516040808c015160608d0151915163b3f1c93d60e01b81526001600160801b0390951696506001600160a01b039093169363b3f1c93d93613bab9392909188906004016156dc565b602060405180830381600087803b158015613bc557600080fd5b505af1158015613bd9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613bfd919061548a565b9050613cb3565b600687015460208901516040808b015160608c015160018c0154925163b3f1c93d60e01b81526001600160a01b039095169463b3f1c93d94613c5e9490939291600160801b9091046001600160801b031690600401615634565b602060405180830381600087803b158015613c7857600080fd5b505af1158015613c8c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613cb0919061548a565b90505b8015613cd5576007870154613cd5908790600160a01b900460ff1660016135af565b613d0488600001518960a0015160008b60e00151613cf4576000613cfa565b8b606001515b8b93929190613088565b8760e0015115613d9c578760a001516001600160a01b0316634efecaa589602001518a606001516040518363ffffffff1660e01b8152600401613d48929190615665565b602060405180830381600087803b158015613d6257600080fd5b505af1158015613d76573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613d9a91906154a6565b505b8760c0015161ffff1688604001516001600160a01b031689600001516001600160a01b03167fc6a898309e823ee50bac64e45ca8adba6690e99e7841c45d754e2a38e9019d9b8b602001518c606001518d6080015160016002811115613dfe57fe5b8f608001516002811115613e0e57fe5b6002811115613e1957fe5b14613e385760028e0154600160801b90046001600160801b0316613e3a565b885b604051613e4a94939291906157ed565b60405180910390a45050505050505050565b600080613e6985846147d5565b905083811015613e7d57600091505061120e565b613e878185613552565b95945050505050565b600190565b303b1590565b600381015460009064ffffffffff600160801b9091048116904216811415613ed257505060018101546001600160801b0316610931565b60018301546002840154600091612fb6916001600160801b0391821691612fb0911685614872565b805182511460405180604001604052806002815260200161373360f01b81525090610e855760405162461bcd60e51b81526004016105df919061592a565b600082613f4757506000610957565b82820282848281613f5457fe5b041461120e5760405162461bcd60e51b8152600401808060200182810382526021815260200180615c7c6021913960400191505060405180910390fd5b6000808211613fe7576040805162461bcd60e51b815260206004820152601a60248201527f536166654d6174683a206469766973696f6e206279207a65726f000000000000604482015290519081900360640190fd5b818381613ff057fe5b049392505050565b6000614015614006846148b0565b61400f846148b0565b9061492e565b9050600061402b614024614a35565b83906134f8565b60018601549091506140479082906001600160801b03166142c8565b604080518082019091526002815261353160f01b60208201529091506001600160801b038211156140b95760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060019490940180546001600160801b0319166001600160801b0390951694909417909355505050565b511590565b60006080821060405180604001604052806002815260200161373760f01b815250906141555760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50509051600360029092021c16151590565b5461ffff80821692601083901c821692602081901c831692603082901c60ff169260409290921c1690565b60006080821060405180604001604052806002815260200161373760f01b815250906141ff5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5050815160016002830281019190911c16151592915050565b60006080821060405180604001604052806002815260200161373760f01b815250906142855760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50509051600160029092021c16151590565b6000826142a7575060001961120e565b612fb6836142b586856147d5565b90614a45565b600061120e838342614b3b565b60008215806142d5575081155b156142e257506000610957565b816b019d971e4fe8401e7400000019816142f857fe5b0483111560405180604001604052806002815260200161068760f31b815250906143635760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b506b033b2e3c9fd0803ce80000006002815b048385020181613ff057fe5b600285015460009081906001600160801b03168585821561450b5760006143a88488614872565b90506143b4818a6142c8565b604080518082019091526002815261353160f01b60208201529093506001600160801b038411156144265760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060018b0180546001600160801b0319166001600160801b03851617905589156145095760028b015460009061446c90600160801b90046001600160801b0316896142bb565b9050614478818a6142c8565b6040805180820190915260028152611a9960f11b60208201529093506001600160801b038411156144ea5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b505060018b0180546001600160801b03808516600160801b0291161790555b505b600399909901805464ffffffffff60801b1916600160801b4264ffffffffff1602179055989650505050505050565b614542614f8a565b61454b8761470f565b610120820181905261455d575061174b565b8660050160009054906101000a90046001600160a01b03166001600160a01b031663797743386040518163ffffffff1660e01b815260040160806040518083038186803b1580156145ad57600080fd5b505afa1580156145c1573d6000803e3d6000fd5b505050506040513d60808110156145d757600080fd5b508051602080830151604084015160609094015164ffffffffff1661014086015260a08501939093529183529082015261461186866142c8565b608082015261462086846142c8565b606082015260a0810151610140820151614642919064ffffffffff8516614b3b565b60c082018190526020820151614657916142c8565b6040820181905260808201518251606084015161467c9392610ba692909183916134f8565b60e0820181905261012082015161469391906147d5565b61010082018190521561307f5760048088015461010083015160408051637df5bd3b60e01b81529384019190915260248301879052516001600160a01b0390911691637df5bd3b91604480830192600092919082900301818387803b1580156146fb57600080fd5b505af1158015610646573d6000803e3d6000fd5b5460401c61ffff1690565b600061476f826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316614c179092919063ffffffff16565b805190915015610e855780806020019051602081101561478e57600080fd5b5051610e855760405162461bcd60e51b815260040180806020018281038252602a815260200180615ccb602a913960400191505060405180910390fd5b5460301c60ff1690565b60008215806147e2575081155b156147ef57506000610957565b8161138819816147fb57fe5b0483111560405180604001604052806002815260200161068760f31b815250906148665760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b50612710600281614375565b6000806148864264ffffffffff8516613552565b9050612fb6614893614a35565b6301e133806148a28785613f38565b816148a957fe5b04906134f8565b6000633b9aca0082810290839082041460405180604001604052806002815260200161068760f31b815250906149275760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5092915050565b604080518082019091526002815261035360f41b6020820152600090826149965760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060408051808201909152600280825261068760f31b60208301528304906b033b2e3c9fd0803ce8000000821904851115614a125760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5082816b033b2e3c9fd0803ce800000086020181614a2c57fe5b04949350505050565b6b033b2e3c9fd0803ce800000090565b604080518082019091526002815261035360f41b602082015260009082614aad5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b5060408051808201909152600280825261068760f31b6020830152830490670de0b6b3a7640000821904851115614b255760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561330b5781810151838201526020016132f3565b508281670de0b6b3a764000086020181614a2c57fe5b600080614b4f8364ffffffffff8616613552565b905080614b6657614b5e614a35565b91505061120e565b6000198101600060028311614b7c576000614b81565b600283035b90506301e1338087046000614b9682806142c8565b90506000614ba482846142c8565b905060006002614bbe84614bb88a8a613f38565b90613f38565b81614bc557fe5b04905060006006614bdc84614bb889818d8d613f38565b81614be357fe5b049050614c0781614c018481614bf98a8e613f38565b614c01614a35565b906134f8565b9c9b505050505050505050505050565b6060612fb6848460008585614c2b856137a6565b614c7c576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b600080866001600160a01b031685876040518082805190602001908083835b60208310614cba5780518252601f199092019160209182019101614c9b565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114614d1c576040519150601f19603f3d011682016040523d82523d6000602084013e614d21565b606091505b5091509150614d31828286614d3c565b979650505050505050565b60608315614d4b57508161120e565b825115614d5b5782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561330b5781810151838201526020016132f3565b604051806101800160405280614db6614e0d565b815260006020820181905260408201819052606082018190526080820181905260a0820181905260c0820181905260e082018190526101008201819052610120820181905261014082018190526101609091015290565b6040518060200160405280600081525090565b6040518060c001604052806000815260200160008152602001600081526020016000815260200160008152602001600081525090565b6040805161012081018252600080825260208201819052918101829052606081018290526080810182905260a0810182905260c0810182905260e0810182905261010081019190915290565b604051806102400160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160001515815260200160006001600160a01b031681526020016000151581526020016000151581525090565b60405180610100016040528060006001600160a01b03168152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b60405180610160016040528060008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081526020016000815260200160008152602001600064ffffffffff1681525090565b60008083601f840112614ffc578081fd5b50813567ffffffffffffffff811115615013578182fd5b6020830191508360208083028501011115612e7b57600080fd5b803561ffff8116811461093157600080fd5b600060208284031215615050578081fd5b813561120e81615c58565b60006020828403121561506c578081fd5b815161120e81615c58565b60008060408385031215615089578081fd5b823561509481615c58565b915060208301356150a481615c58565b809150509250929050565b600080600080600060a086880312156150c6578081fd5b85356150d181615c58565b945060208601356150e181615c58565b935060408601356150f181615c58565b9250606086013561510181615c58565b9150608086013561511181615c58565b809150509295509295909350565b600080600080600060a08688031215615136578081fd5b853561514181615c58565b9450602086013561515181615c58565b9350604086013561516181615c58565b925060608601359150608086013561511181615c6d565b60008060008060008060c08789031215615190578081fd5b863561519b81615c58565b955060208701356151ab81615c58565b945060408701356151bb81615c58565b959894975094956060810135955060808101359460a0909101359350915050565b60008060008060008060008060a0898b0312156151f7578182fd5b883561520281615c58565b9750602089013567ffffffffffffffff8082111561521e578384fd5b61522a8c838d01614feb565b909950975060408b0135915080821115615242578384fd5b61524e8c838d01614feb565b909750955060608b0135915080821115615266578384fd5b818b0191508b601f830112615279578384fd5b813581811115615287578485fd5b8c6020828501011115615298578485fd5b6020830195508094505050506152b060808a0161502d565b90509295985092959890939650565b600080604083850312156152d1578182fd5b82356152dc81615c58565b915060208301356150a481615c6d565b60008082840360e08112156152ff578283fd5b833561530a81615c58565b925060c0601f198201121561531d578182fd5b506020830190509250929050565b6000806040838503121561533d578182fd5b823561534881615c58565b946020939093013593505050565b60008060006060848603121561536a578081fd5b833561537581615c58565b925060208401359150604084013561538c81615c58565b809150509250925092565b600080600080608085870312156153ac578182fd5b84356153b781615c58565b93506020850135925060408501356153ce81615c58565b91506153dc6060860161502d565b905092959194509250565b600080600080608085870312156153fc578182fd5b843561540781615c58565b93506020850135925060408501359150606085013561542581615c58565b939692955090935050565b600080600080600060a08688031215615447578283fd5b853561545281615c58565b945060208601359350604086013592506151016060870161502d565b60006020828403121561547f578081fd5b813561120e81615c6d565b60006020828403121561549b578081fd5b815161120e81615c6d565b6000602082840312156154b7578081fd5b5051919050565b600080604083850312156154d0578182fd5b82519150602083015167ffffffffffffffff808211156154ee578283fd5b818501915085601f830112615501578283fd5b81518181111561550d57fe5b604051601f8201601f19168101602001838111828210171561552b57fe5b604052818152838201602001881015615542578485fd5b615553826020830160208701615bf2565b809450505050509250929050565b6001600160a01b03169052565b6000815180845260208085019450808401835b8381101561559d57815187529582019590820190600101615581565b509495945050505050565b60008284528282602086013780602084860101526020601f19601f85011685010190509392505050565b600381106155dc57fe5b9052565b519052565b6001600160801b03169052565b64ffffffffff169052565b60ff169052565b60008251615616818460208701615bf2565b9190910192915050565b6001600160a01b0391909116815260200190565b6001600160a01b03948516815292909316602083015260408201526001600160801b03909116606082015260800190565b6001600160a01b03929092168252602082015260400190565b6001600160a01b0393909316835260208301919091526001600160801b0316604082015260600190565b6001600160a01b03958616815293851660208501529190931660408301526060820192909252901515608082015260a00190565b6001600160a01b0394851681529290931660208301526040820152606081019190915260800190565b6001600160a01b039687168152602081019590955260408501939093526060840191909152608083015290911660a082015260c00190565b6001600160a01b039c8d168152602081019b909b52988b1660408b015260608a0197909752608089019590955260a088019390935260c087019190915260e08601526101008501526101208401526101408301529091166101608201526101800190565b6001600160a01b03998a168152602081019890985260408801969096526060870194909452608086019290925260a085015260c084015260e08301529091166101008201526101200190565b6001600160a01b0394909416845260208401929092526040830152606082015260800190565b60a0808252810188905260008960c08301825b8b81101561585657823561583981615c58565b6001600160a01b0316825260209283019290910190600101615826565b5083810360208501528881526001600160fb1b03891115615875578283fd5b602089029150818a602083013701602081810183815284830390910160408501526158a0818961556e565b9150506158b06060840187615561565b82810360808401526158c38185876155a8565b9b9a5050505050505050505050565b6020808252825182820181905260009190848201906040850190845b818110156159135783516001600160a01b0316835292840192918401916001016158ee565b50909695505050505050565b901515815260200190565b6000602082528251806020840152615949816040850160208701615bf2565b601f01601f19169190910160400192915050565b9051815260200190565b60006101808201905061597b8284516155e0565b602083015161598d60208401826155e5565b5060408301516159a060408401826155e5565b5060608301516159b360608401826155e5565b5060808301516159c660808401826155e5565b5060a08301516159d960a08401826155e5565b5060c08301516159ec60c08401826155f2565b5060e08301516159ff60e0840182615561565b5061010080840151615a1382850182615561565b505061012080840151615a2882850182615561565b505061014080840151615a3d82850182615561565b505061016080840151615a52828501826155fd565b505092915050565b9485526001600160a01b03938416602086015291831660408501528216606084015216608082015260a00190565b9788526001600160a01b03968716602089015294151560408801526060870193909352608086019190915260a085015260c08401521660e08201526101000190565b93845260208401929092526001600160a01b03166040830152606082015260800190565b600060a082019050868252856020830152846040830152836060830152615b1860808301846155d2565b9695505050505050565b8681526020810186905260c08101615b3d60408301876155d2565b6001600160a01b03949094166060820152608081019290925260a0909101529392505050565b600060c082019050825182526020830151602083015260408301516040830152606083015160608301526080830151608083015260a083015160a083015292915050565b90815260200190565b928352602083019190915261ffff16604082015260600190565b958652602086019490945260408501929092526060840152608083015260a082015260c00190565b60005b83811015615c0d578181015183820152602001615bf5565b83811115610ebf5750506000910152565b813581556020820135600182015560408201356002820155606082013560038201556080820135600482015560a082013560058201555050565b6001600160a01b03811681146117ef57600080fd5b80151581146117ef57600080fdfe536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f77436f6e747261637420696e7374616e63652068617320616c7265616479206265656e20696e697469616c697a65645361666545524332303a204552433230206f7065726174696f6e20646964206e6f742073756363656564a26469706673582212200cd40c425c6db626729fc3c8744a3eba9342d8768634406c3fa2ba81877612aa64736f6c63430007060033

External libraries