2020-05-29 16:45:37 +00:00
|
|
|
// SPDX-License-Identifier: agpl-3.0
|
|
|
|
pragma solidity ^0.6.8;
|
|
|
|
|
2020-10-15 13:25:27 +00:00
|
|
|
import {SafeMath} from '../dependencies/openzeppelin/contracts//SafeMath.sol';
|
|
|
|
import {IERC20} from '../dependencies/openzeppelin/contracts//IERC20.sol';
|
2020-10-15 13:16:05 +00:00
|
|
|
import {VersionedInitializable} from '../libraries/aave-upgradeability/VersionedInitializable.sol';
|
2020-08-21 10:38:08 +00:00
|
|
|
import {IAToken} from '../tokenization/interfaces/IAToken.sol';
|
2020-08-20 07:51:21 +00:00
|
|
|
import {IStableDebtToken} from '../tokenization/interfaces/IStableDebtToken.sol';
|
|
|
|
import {IVariableDebtToken} from '../tokenization/interfaces/IVariableDebtToken.sol';
|
|
|
|
import {IPriceOracleGetter} from '../interfaces/IPriceOracleGetter.sol';
|
|
|
|
import {GenericLogic} from '../libraries/logic/GenericLogic.sol';
|
|
|
|
import {ReserveLogic} from '../libraries/logic/ReserveLogic.sol';
|
|
|
|
import {UserConfiguration} from '../libraries/configuration/UserConfiguration.sol';
|
|
|
|
import {Helpers} from '../libraries/helpers/Helpers.sol';
|
|
|
|
import {WadRayMath} from '../libraries/math/WadRayMath.sol';
|
|
|
|
import {PercentageMath} from '../libraries/math/PercentageMath.sol';
|
2020-10-15 13:25:27 +00:00
|
|
|
import {SafeERC20} from '../dependencies/openzeppelin/contracts/SafeERC20.sol';
|
2020-09-02 16:53:39 +00:00
|
|
|
import {Errors} from '../libraries/helpers/Errors.sol';
|
2020-09-14 08:52:31 +00:00
|
|
|
import {ValidationLogic} from '../libraries/logic/ValidationLogic.sol';
|
2020-09-16 10:16:51 +00:00
|
|
|
import {LendingPoolStorage} from './LendingPoolStorage.sol';
|
2020-05-29 16:45:37 +00:00
|
|
|
|
|
|
|
/**
|
2020-09-16 10:41:12 +00:00
|
|
|
* @title LendingPoolCollateralManager contract
|
2020-07-13 08:54:08 +00:00
|
|
|
* @author Aave
|
2020-09-16 12:09:42 +00:00
|
|
|
* @notice Implements actions involving management of collateral in the protocol.
|
|
|
|
* @notice this contract will be ran always through delegatecall
|
2020-09-16 12:45:49 +00:00
|
|
|
* @dev LendingPoolCollateralManager inherits VersionedInitializable from OpenZeppelin to have the same storage layout as LendingPool
|
2020-07-13 08:54:08 +00:00
|
|
|
**/
|
2020-09-21 07:31:54 +00:00
|
|
|
contract LendingPoolCollateralManager is VersionedInitializable, LendingPoolStorage {
|
2020-08-12 17:36:58 +00:00
|
|
|
using SafeERC20 for IERC20;
|
2020-07-13 08:54:08 +00:00
|
|
|
using SafeMath for uint256;
|
|
|
|
using WadRayMath for uint256;
|
2020-07-27 11:47:48 +00:00
|
|
|
using PercentageMath for uint256;
|
2020-07-13 08:54:08 +00:00
|
|
|
|
2020-09-09 11:06:46 +00:00
|
|
|
// IMPORTANT The storage layout of the LendingPool is reproduced here because this contract
|
|
|
|
// is gonna be used through DELEGATECALL
|
|
|
|
|
2020-08-21 12:14:13 +00:00
|
|
|
uint256 internal constant LIQUIDATION_CLOSE_FACTOR_PERCENT = 5000;
|
2020-07-13 08:54:08 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev emitted when a borrower is liquidated
|
2020-08-21 12:14:13 +00:00
|
|
|
* @param collateral the address of the collateral being liquidated
|
|
|
|
* @param principal the address of the reserve
|
|
|
|
* @param user the address of the user being liquidated
|
|
|
|
* @param purchaseAmount the total amount liquidated
|
|
|
|
* @param liquidatedCollateralAmount the amount of collateral being liquidated
|
|
|
|
* @param liquidator the address of the liquidator
|
|
|
|
* @param receiveAToken true if the liquidator wants to receive aTokens, false otherwise
|
2020-07-13 08:54:08 +00:00
|
|
|
**/
|
|
|
|
event LiquidationCall(
|
2020-08-21 12:14:13 +00:00
|
|
|
address indexed collateral,
|
|
|
|
address indexed principal,
|
|
|
|
address indexed user,
|
|
|
|
uint256 purchaseAmount,
|
|
|
|
uint256 liquidatedCollateralAmount,
|
|
|
|
address liquidator,
|
|
|
|
bool receiveAToken
|
2020-07-13 08:54:08 +00:00
|
|
|
);
|
|
|
|
|
2020-11-10 15:18:31 +00:00
|
|
|
/**
|
|
|
|
* @dev emitted when a user disables a reserve as collateral
|
|
|
|
* @param reserve the address of the reserve
|
|
|
|
* @param user the address of the user
|
|
|
|
**/
|
|
|
|
event ReserveUsedAsCollateralDisabled(address indexed reserve, address indexed user);
|
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
struct LiquidationCallLocalVars {
|
|
|
|
uint256 userCollateralBalance;
|
|
|
|
uint256 userStableDebt;
|
|
|
|
uint256 userVariableDebt;
|
|
|
|
uint256 maxPrincipalAmountToLiquidate;
|
|
|
|
uint256 actualAmountToLiquidate;
|
|
|
|
uint256 liquidationRatio;
|
|
|
|
uint256 maxAmountCollateralToLiquidate;
|
|
|
|
uint256 userStableRate;
|
|
|
|
uint256 maxCollateralToLiquidate;
|
|
|
|
uint256 principalAmountNeeded;
|
|
|
|
uint256 healthFactor;
|
2020-08-20 12:32:20 +00:00
|
|
|
IAToken collateralAtoken;
|
2020-07-13 08:54:08 +00:00
|
|
|
bool isCollateralEnabled;
|
2020-11-12 09:16:10 +00:00
|
|
|
ReserveLogic.InterestRateMode borrowRateMode;
|
2020-09-14 08:52:31 +00:00
|
|
|
address principalAToken;
|
|
|
|
uint256 errorCode;
|
|
|
|
string errorMsg;
|
2020-07-13 08:54:08 +00:00
|
|
|
}
|
|
|
|
|
2020-09-15 08:28:39 +00:00
|
|
|
struct AvailableCollateralToLiquidateLocalVars {
|
|
|
|
uint256 userCompoundedBorrowBalance;
|
|
|
|
uint256 liquidationBonus;
|
|
|
|
uint256 collateralPrice;
|
|
|
|
uint256 principalCurrencyPrice;
|
|
|
|
uint256 maxAmountCollateralToLiquidate;
|
|
|
|
uint256 principalDecimals;
|
|
|
|
uint256 collateralDecimals;
|
|
|
|
}
|
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
/**
|
|
|
|
* @dev as the contract extends the VersionedInitializable contract to match the state
|
|
|
|
* of the LendingPool contract, the getRevision() function is needed.
|
|
|
|
*/
|
|
|
|
function getRevision() internal override pure returns (uint256) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev users can invoke this function to liquidate an undercollateralized position.
|
2020-08-21 12:14:13 +00:00
|
|
|
* @param collateral the address of the collateral to liquidated
|
|
|
|
* @param principal the address of the principal reserve
|
|
|
|
* @param user the address of the borrower
|
|
|
|
* @param purchaseAmount the amount of principal that the liquidator wants to repay
|
|
|
|
* @param receiveAToken true if the liquidators wants to receive the aTokens, false if
|
2020-07-13 08:54:08 +00:00
|
|
|
* he wants to receive the underlying asset directly
|
|
|
|
**/
|
|
|
|
function liquidationCall(
|
2020-08-21 12:14:13 +00:00
|
|
|
address collateral,
|
|
|
|
address principal,
|
|
|
|
address user,
|
|
|
|
uint256 purchaseAmount,
|
|
|
|
bool receiveAToken
|
2020-08-20 12:32:20 +00:00
|
|
|
) external returns (uint256, string memory) {
|
2020-09-16 10:16:51 +00:00
|
|
|
ReserveLogic.ReserveData storage collateralReserve = _reserves[collateral];
|
|
|
|
ReserveLogic.ReserveData storage principalReserve = _reserves[principal];
|
|
|
|
UserConfiguration.Map storage userConfig = _usersConfig[user];
|
2020-07-13 08:54:08 +00:00
|
|
|
|
|
|
|
LiquidationCallLocalVars memory vars;
|
|
|
|
|
2020-07-23 15:18:06 +00:00
|
|
|
(, , , , vars.healthFactor) = GenericLogic.calculateUserAccountData(
|
2020-08-21 12:14:13 +00:00
|
|
|
user,
|
2020-09-16 10:16:51 +00:00
|
|
|
_reserves,
|
2020-11-10 15:18:31 +00:00
|
|
|
userConfig,
|
2020-09-16 10:16:51 +00:00
|
|
|
_reservesList,
|
2020-10-06 13:51:48 +00:00
|
|
|
_reservesCount,
|
2020-09-16 10:16:51 +00:00
|
|
|
_addressesProvider.getPriceOracle()
|
2020-05-29 16:45:37 +00:00
|
|
|
);
|
|
|
|
|
2020-08-21 12:14:13 +00:00
|
|
|
//if the user hasn't borrowed the specific currency defined by asset, it cannot be liquidated
|
2020-08-06 07:52:15 +00:00
|
|
|
(vars.userStableDebt, vars.userVariableDebt) = Helpers.getUserCurrentDebt(
|
2020-08-21 12:14:13 +00:00
|
|
|
user,
|
2020-07-13 08:54:08 +00:00
|
|
|
principalReserve
|
|
|
|
);
|
2020-05-29 16:45:37 +00:00
|
|
|
|
2020-09-14 08:52:31 +00:00
|
|
|
(vars.errorCode, vars.errorMsg) = ValidationLogic.validateLiquidationCall(
|
|
|
|
collateralReserve,
|
|
|
|
principalReserve,
|
|
|
|
userConfig,
|
|
|
|
vars.healthFactor,
|
|
|
|
vars.userStableDebt,
|
|
|
|
vars.userVariableDebt
|
|
|
|
);
|
|
|
|
|
2020-09-16 12:09:42 +00:00
|
|
|
if (Errors.CollateralManagerErrors(vars.errorCode) != Errors.CollateralManagerErrors.NO_ERROR) {
|
2020-09-14 08:52:31 +00:00
|
|
|
return (vars.errorCode, vars.errorMsg);
|
2020-07-13 08:54:08 +00:00
|
|
|
}
|
|
|
|
|
2020-09-14 08:52:31 +00:00
|
|
|
vars.collateralAtoken = IAToken(collateralReserve.aTokenAddress);
|
|
|
|
|
|
|
|
vars.userCollateralBalance = vars.collateralAtoken.balanceOf(user);
|
|
|
|
|
2020-07-27 11:47:48 +00:00
|
|
|
vars.maxPrincipalAmountToLiquidate = vars.userStableDebt.add(vars.userVariableDebt).percentMul(
|
|
|
|
LIQUIDATION_CLOSE_FACTOR_PERCENT
|
|
|
|
);
|
2020-07-13 08:54:08 +00:00
|
|
|
|
2020-08-21 12:14:13 +00:00
|
|
|
vars.actualAmountToLiquidate = purchaseAmount > vars.maxPrincipalAmountToLiquidate
|
2020-07-13 08:54:08 +00:00
|
|
|
? vars.maxPrincipalAmountToLiquidate
|
2020-08-21 12:14:13 +00:00
|
|
|
: purchaseAmount;
|
2020-07-13 08:54:08 +00:00
|
|
|
|
|
|
|
(
|
|
|
|
vars.maxCollateralToLiquidate,
|
|
|
|
vars.principalAmountNeeded
|
2020-10-22 09:50:04 +00:00
|
|
|
) = _calculateAvailableCollateralToLiquidate(
|
2020-07-13 08:54:08 +00:00
|
|
|
collateralReserve,
|
|
|
|
principalReserve,
|
2020-08-21 12:14:13 +00:00
|
|
|
collateral,
|
|
|
|
principal,
|
2020-07-13 08:54:08 +00:00
|
|
|
vars.actualAmountToLiquidate,
|
|
|
|
vars.userCollateralBalance
|
|
|
|
);
|
|
|
|
|
|
|
|
//if principalAmountNeeded < vars.ActualAmountToLiquidate, there isn't enough
|
2020-08-21 12:14:13 +00:00
|
|
|
//of collateral to cover the actual amount that is being liquidated, hence we liquidate
|
2020-07-13 08:54:08 +00:00
|
|
|
//a smaller amount
|
2020-05-29 16:45:37 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
if (vars.principalAmountNeeded < vars.actualAmountToLiquidate) {
|
|
|
|
vars.actualAmountToLiquidate = vars.principalAmountNeeded;
|
|
|
|
}
|
|
|
|
|
|
|
|
//if liquidator reclaims the underlying asset, we make sure there is enough available collateral in the reserve
|
2020-08-21 12:14:13 +00:00
|
|
|
if (!receiveAToken) {
|
|
|
|
uint256 currentAvailableCollateral = IERC20(collateral).balanceOf(
|
2020-07-13 10:24:36 +00:00
|
|
|
address(vars.collateralAtoken)
|
|
|
|
);
|
2020-07-13 08:54:08 +00:00
|
|
|
if (currentAvailableCollateral < vars.maxCollateralToLiquidate) {
|
|
|
|
return (
|
2020-09-16 12:09:42 +00:00
|
|
|
uint256(Errors.CollateralManagerErrors.NOT_ENOUGH_LIQUIDITY),
|
2020-10-14 09:03:32 +00:00
|
|
|
Errors.LPCM_NOT_ENOUGH_LIQUIDITY_TO_LIQUIDATE
|
2020-05-29 16:45:37 +00:00
|
|
|
);
|
2020-07-13 08:54:08 +00:00
|
|
|
}
|
|
|
|
}
|
2020-05-29 16:45:37 +00:00
|
|
|
|
2020-07-15 14:44:20 +00:00
|
|
|
//update the principal reserve
|
2020-09-14 07:53:21 +00:00
|
|
|
principalReserve.updateState();
|
2020-09-10 10:51:52 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
if (vars.userVariableDebt >= vars.actualAmountToLiquidate) {
|
2020-09-14 13:09:16 +00:00
|
|
|
IVariableDebtToken(principalReserve.variableDebtTokenAddress).burn(
|
2020-08-21 12:14:13 +00:00
|
|
|
user,
|
2020-09-14 13:09:16 +00:00
|
|
|
vars.actualAmountToLiquidate,
|
|
|
|
principalReserve.variableBorrowIndex
|
2020-07-13 08:54:08 +00:00
|
|
|
);
|
|
|
|
} else {
|
2020-09-30 15:59:47 +00:00
|
|
|
//if the user does not have variable debt, no need to try to burn variable
|
|
|
|
//debt tokens
|
|
|
|
if (vars.userVariableDebt > 0) {
|
|
|
|
IVariableDebtToken(principalReserve.variableDebtTokenAddress).burn(
|
|
|
|
user,
|
|
|
|
vars.userVariableDebt,
|
|
|
|
principalReserve.variableBorrowIndex
|
|
|
|
);
|
|
|
|
}
|
2020-09-14 13:09:16 +00:00
|
|
|
IStableDebtToken(principalReserve.stableDebtTokenAddress).burn(
|
2020-08-21 12:14:13 +00:00
|
|
|
user,
|
2020-07-13 08:54:08 +00:00
|
|
|
vars.actualAmountToLiquidate.sub(vars.userVariableDebt)
|
|
|
|
);
|
2020-05-29 16:45:37 +00:00
|
|
|
}
|
|
|
|
|
2020-10-26 10:04:13 +00:00
|
|
|
principalReserve.updateInterestRates(
|
|
|
|
principal,
|
|
|
|
principalReserve.aTokenAddress,
|
|
|
|
vars.actualAmountToLiquidate,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
//if liquidator reclaims the aToken, he receives the equivalent atoken amount
|
2020-08-21 12:14:13 +00:00
|
|
|
if (receiveAToken) {
|
|
|
|
vars.collateralAtoken.transferOnLiquidation(user, msg.sender, vars.maxCollateralToLiquidate);
|
2020-07-13 08:54:08 +00:00
|
|
|
} else {
|
|
|
|
//otherwise receives the underlying asset
|
2020-07-15 14:44:20 +00:00
|
|
|
|
|
|
|
//updating collateral reserve
|
2020-09-14 07:53:21 +00:00
|
|
|
collateralReserve.updateState();
|
2020-08-25 10:37:38 +00:00
|
|
|
collateralReserve.updateInterestRates(
|
|
|
|
collateral,
|
|
|
|
address(vars.collateralAtoken),
|
|
|
|
0,
|
|
|
|
vars.maxCollateralToLiquidate
|
|
|
|
);
|
2020-07-15 14:44:20 +00:00
|
|
|
|
2020-07-13 08:54:08 +00:00
|
|
|
//burn the equivalent amount of atoken
|
2020-09-14 13:09:16 +00:00
|
|
|
vars.collateralAtoken.burn(
|
|
|
|
user,
|
|
|
|
msg.sender,
|
|
|
|
vars.maxCollateralToLiquidate,
|
|
|
|
collateralReserve.liquidityIndex
|
|
|
|
);
|
2020-05-29 16:45:37 +00:00
|
|
|
}
|
|
|
|
|
2020-11-10 15:18:31 +00:00
|
|
|
//if the collateral being liquidated is equal to the user balance,
|
|
|
|
//we set the currency as not being used as collateral anymore
|
|
|
|
|
|
|
|
if (vars.maxCollateralToLiquidate == vars.userCollateralBalance) {
|
|
|
|
userConfig.setUsingAsCollateral(collateralReserve.id, false);
|
|
|
|
emit ReserveUsedAsCollateralDisabled(collateral, user);
|
|
|
|
}
|
|
|
|
|
2020-07-13 10:24:36 +00:00
|
|
|
//transfers the principal currency to the aToken
|
2020-08-21 12:14:13 +00:00
|
|
|
IERC20(principal).safeTransferFrom(
|
2020-07-13 10:24:36 +00:00
|
|
|
msg.sender,
|
|
|
|
principalReserve.aTokenAddress,
|
2020-08-12 17:36:58 +00:00
|
|
|
vars.actualAmountToLiquidate
|
2020-07-13 10:24:36 +00:00
|
|
|
);
|
2020-07-13 08:54:08 +00:00
|
|
|
|
|
|
|
emit LiquidationCall(
|
2020-08-21 12:14:13 +00:00
|
|
|
collateral,
|
|
|
|
principal,
|
|
|
|
user,
|
2020-07-13 08:54:08 +00:00
|
|
|
vars.actualAmountToLiquidate,
|
|
|
|
vars.maxCollateralToLiquidate,
|
|
|
|
msg.sender,
|
2020-08-21 12:14:13 +00:00
|
|
|
receiveAToken
|
2020-07-13 08:54:08 +00:00
|
|
|
);
|
|
|
|
|
2020-10-14 09:03:32 +00:00
|
|
|
return (uint256(Errors.CollateralManagerErrors.NO_ERROR), Errors.LPCM_NO_ERRORS);
|
2020-07-13 08:54:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev calculates how much of a specific collateral can be liquidated, given
|
|
|
|
* a certain amount of principal currency. This function needs to be called after
|
|
|
|
* all the checks to validate the liquidation have been performed, otherwise it might fail.
|
2020-08-21 12:14:13 +00:00
|
|
|
* @param collateralAddress the collateral to be liquidated
|
|
|
|
* @param principalAddress the principal currency to be liquidated
|
|
|
|
* @param purchaseAmount the amount of principal being liquidated
|
|
|
|
* @param userCollateralBalance the collatera balance for the specific collateral asset of the user being liquidated
|
2020-07-13 08:54:08 +00:00
|
|
|
* @return collateralAmount the maximum amount that is possible to liquidated given all the liquidation constraints (user balance, close factor)
|
|
|
|
* @return principalAmountNeeded the purchase amount
|
|
|
|
**/
|
2020-10-22 09:50:04 +00:00
|
|
|
function _calculateAvailableCollateralToLiquidate(
|
2020-08-25 08:53:58 +00:00
|
|
|
ReserveLogic.ReserveData storage collateralReserve,
|
|
|
|
ReserveLogic.ReserveData storage principalReserve,
|
2020-08-21 12:14:13 +00:00
|
|
|
address collateralAddress,
|
|
|
|
address principalAddress,
|
|
|
|
uint256 purchaseAmount,
|
|
|
|
uint256 userCollateralBalance
|
2020-08-20 13:31:52 +00:00
|
|
|
) internal view returns (uint256, uint256) {
|
|
|
|
uint256 collateralAmount = 0;
|
|
|
|
uint256 principalAmountNeeded = 0;
|
2020-09-16 10:16:51 +00:00
|
|
|
IPriceOracleGetter oracle = IPriceOracleGetter(_addressesProvider.getPriceOracle());
|
2020-07-13 08:54:08 +00:00
|
|
|
|
|
|
|
AvailableCollateralToLiquidateLocalVars memory vars;
|
|
|
|
|
2020-08-21 12:14:13 +00:00
|
|
|
vars.collateralPrice = oracle.getAssetPrice(collateralAddress);
|
|
|
|
vars.principalCurrencyPrice = oracle.getAssetPrice(principalAddress);
|
2020-07-23 15:18:06 +00:00
|
|
|
|
2020-10-12 12:25:03 +00:00
|
|
|
(, , vars.liquidationBonus, vars.collateralDecimals, ) = collateralReserve
|
2020-07-23 15:18:06 +00:00
|
|
|
.configuration
|
|
|
|
.getParams();
|
2020-08-25 08:53:58 +00:00
|
|
|
vars.principalDecimals = principalReserve.configuration.getDecimals();
|
2020-07-13 08:54:08 +00:00
|
|
|
|
|
|
|
//this is the maximum possible amount of the selected collateral that can be liquidated, given the
|
|
|
|
//max amount of principal currency that is available for liquidation.
|
|
|
|
vars.maxAmountCollateralToLiquidate = vars
|
|
|
|
.principalCurrencyPrice
|
2020-08-21 12:14:13 +00:00
|
|
|
.mul(purchaseAmount)
|
2020-07-13 08:54:08 +00:00
|
|
|
.mul(10**vars.collateralDecimals)
|
2020-10-26 09:48:43 +00:00
|
|
|
.percentMul(vars.liquidationBonus)
|
|
|
|
.div(vars.collateralPrice.mul(10**vars.principalDecimals));
|
2020-07-13 08:54:08 +00:00
|
|
|
|
2020-08-21 12:14:13 +00:00
|
|
|
if (vars.maxAmountCollateralToLiquidate > userCollateralBalance) {
|
|
|
|
collateralAmount = userCollateralBalance;
|
2020-07-13 08:54:08 +00:00
|
|
|
principalAmountNeeded = vars
|
|
|
|
.collateralPrice
|
|
|
|
.mul(collateralAmount)
|
|
|
|
.mul(10**vars.principalDecimals)
|
|
|
|
.div(vars.principalCurrencyPrice.mul(10**vars.collateralDecimals))
|
2020-07-27 11:47:48 +00:00
|
|
|
.percentDiv(vars.liquidationBonus);
|
2020-07-13 08:54:08 +00:00
|
|
|
} else {
|
|
|
|
collateralAmount = vars.maxAmountCollateralToLiquidate;
|
2020-08-21 12:14:13 +00:00
|
|
|
principalAmountNeeded = purchaseAmount;
|
2020-05-29 16:45:37 +00:00
|
|
|
}
|
2020-07-13 08:54:08 +00:00
|
|
|
return (collateralAmount, principalAmountNeeded);
|
|
|
|
}
|
2020-05-29 16:45:37 +00:00
|
|
|
}
|