// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {IERC20} from'../../dependencies/openzeppelin/contracts/IERC20.sol';
import {SafeERC20} from'../../dependencies/openzeppelin/contracts/SafeERC20.sol';
import {ILendingPool} from'../../interfaces/ILendingPool.sol';
import {IAToken} from'../../interfaces/IAToken.sol';
import {WadRayMath} from'../libraries/math/WadRayMath.sol';
import {Errors} from'../libraries/helpers/Errors.sol';
import {VersionedInitializable} from'../libraries/aave-upgradeability/VersionedInitializable.sol';
import {IncentivizedERC20} from'./IncentivizedERC20.sol';
import {IAaveIncentivesController} from'../../interfaces/IAaveIncentivesController.sol';
import {SafeMath} from'../../dependencies/openzeppelin/contracts/SafeMath.sol';
/**
* @title Aave ERC20 AToken
* @dev Implementation of the interest bearing token for the Aave protocol
* @author Aave
*/contractATokenisVersionedInitializable,
IncentivizedERC20('ATOKEN_IMPL', 'ATOKEN_IMPL', 0),
IAToken{
usingWadRayMathforuint256;
usingSafeERC20forIERC20;
usingSafeMathforuint256;
bytespublicconstant EIP712_REVISION =bytes('1');
bytes32internalconstant EIP712_DOMAIN =keccak256('EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)');
bytes32publicconstant PERMIT_TYPEHASH =keccak256('Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)');
uint256publicconstant ATOKEN_REVISION =0x1;
/// @dev owner => next valid nonce to submit with permit()mapping(address=>uint256) public _nonces;
bytes32public DOMAIN_SEPARATOR;
addressinternal _treasury;
IAaveIncentivesController internal _incentivesController;
modifieronlyLendingPool{
require(_msgSender() ==address(_pool), Errors.CT_CALLER_MUST_BE_LENDING_POOL);
_;
}
functiongetRevision() internalpurevirtualoverridereturns (uint256) {
return ATOKEN_REVISION;
}
/**
* @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 incentivesController The smart contract managing potential incentives distribution
* @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
*/functioninitialize(
ILendingPool pool,
address treasury,
address underlyingAsset,
IAaveIncentivesController incentivesController,
uint8 aTokenDecimals,
stringcalldata aTokenName,
stringcalldata aTokenSymbol,
bytescalldata params
) externaloverrideinitializer{
uint256 chainId;
//solium-disable-next-lineassembly {
chainId :=chainid()
}
DOMAIN_SEPARATOR =keccak256(
abi.encode(
EIP712_DOMAIN,
keccak256(bytes(aTokenName)),
keccak256(EIP712_REVISION),
chainId,
address(this)
)
);
_setName(aTokenName);
_setSymbol(aTokenSymbol);
_setDecimals(aTokenDecimals);
_pool = pool;
_treasury = treasury;
_underlyingAsset = underlyingAsset;
_incentivesController = incentivesController;
emit Initialized(
underlyingAsset,
address(pool),
treasury,
address(incentivesController),
aTokenDecimals,
aTokenName,
aTokenSymbol,
params
);
}
/**
* @dev Burns aTokens from `user` and sends the equivalent amount of underlying to `receiverOfUnderlying`
* - Only callable by the LendingPool, as extra state updates there need to be managed
* @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
**/functionburn(address user,
address receiverOfUnderlying,
uint256 amount,
uint256 index
) externaloverrideonlyLendingPool{
uint256 amountScaled = amount.rayDiv(index);
require(amountScaled !=0, Errors.CT_INVALID_BURN_AMOUNT);
_burn(user, amountScaled);
IERC20(_underlyingAsset).safeTransfer(receiverOfUnderlying, amount);
emit Transfer(user, address(0), amount);
emit Burn(user, receiverOfUnderlying, amount, index);
}
/**
* @dev Mints `amount` aTokens to `user`
* - Only callable by the LendingPool, as extra state updates there need to be managed
* @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
*/functionmint(address user,
uint256 amount,
uint256 index
) externaloverrideonlyLendingPoolreturns (bool) {
uint256 previousBalance =super.balanceOf(user);
uint256 amountScaled = amount.rayDiv(index);
require(amountScaled !=0, Errors.CT_INVALID_MINT_AMOUNT);
_mint(user, amountScaled);
emit Transfer(address(0), user, amount);
emit Mint(user, amount, index);
return previousBalance ==0;
}
/**
* @dev Mints aTokens to the reserve treasury
* - Only callable by the LendingPool
* @param amount The amount of tokens getting minted
* @param index The new liquidity index of the reserve
*/functionmintToTreasury(uint256 amount, uint256 index) externaloverrideonlyLendingPool{
if (amount ==0) {
return;
}
address treasury = _treasury;
// Compared to the normal mint, we don't check for rounding errors.// The amount to mint can easily be very small since it is a fraction of the interest ccrued.// In that case, the treasury will experience a (very small) loss, but it// wont cause potentially valid transactions to fail.
_mint(treasury, amount.rayDiv(index));
emit Transfer(address(0), treasury, amount);
emit Mint(treasury, amount, index);
}
/**
* @dev Transfers aTokens in the event of a borrow being liquidated, in case the liquidators reclaims the aToken
* - Only callable by the LendingPool
* @param from The address getting liquidated, current owner of the aTokens
* @param to The recipient
* @param value The amount of tokens getting transferred
**/functiontransferOnLiquidation(addressfrom,
address to,
uint256 value
) externaloverrideonlyLendingPool{
// Being a normal transfer, the Transfer() and BalanceTransfer() are emitted// so no need to emit a specific event here
_transfer(from, to, value, false);
emit Transfer(from, to, value);
}
/**
* @dev Calculates the balance of the user: principal balance + interest generated by the principal
* @param user The user whose balance is calculated
* @return The balance of the user
**/functionbalanceOf(address user)
publicviewoverride(IncentivizedERC20, IERC20)
returns (uint256)
{
returnsuper.balanceOf(user).rayMul(_pool.getReserveNormalizedIncome(_underlyingAsset));
}
/**
* @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
**/functionscaledBalanceOf(address user) externalviewoverridereturns (uint256) {
returnsuper.balanceOf(user);
}
/**
* @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
**/functiongetScaledUserBalanceAndSupply(address user)
externalviewoverridereturns (uint256, uint256)
{
return (super.balanceOf(user), super.totalSupply());
}
/**
* @dev calculates the total supply of the specific aToken
* since the balance of every single user increases over time, the total supply
* does that too.
* @return the current total supply
**/functiontotalSupply() publicviewoverride(IncentivizedERC20, IERC20) returns (uint256) {
uint256 currentSupplyScaled =super.totalSupply();
if (currentSupplyScaled ==0) {
return0;
}
return currentSupplyScaled.rayMul(_pool.getReserveNormalizedIncome(_underlyingAsset));
}
/**
* @dev Returns the scaled total supply of the variable debt token. Represents sum(debt/index)
* @return the scaled total supply
**/functionscaledTotalSupply() publicviewvirtualoverridereturns (uint256) {
returnsuper.totalSupply();
}
/**
* @dev Returns the address of the Aave treasury, receiving the fees on this aToken
**/functionRESERVE_TREASURY_ADDRESS() publicviewreturns (address) {
return _treasury;
}
/**
* @dev Returns the address of the underlying asset of this aToken (E.g. WETH for aWETH)
**/functionUNDERLYING_ASSET_ADDRESS() publicoverrideviewreturns (address) {
return _underlyingAsset;
}
/**
* @dev Returns the address of the lending pool where this aToken is used
**/functionPOOL() publicviewreturns (ILendingPool) {
return _pool;
}
/**
* @dev For internal usage in the logic of the parent contract IncentivizedERC20
**/function_getIncentivesController() internalviewoverridereturns (IAaveIncentivesController) {
return _incentivesController;
}
/**
* @dev Returns the address of the incentives controller contract
**/functiongetIncentivesController() externalviewoverridereturns (IAaveIncentivesController) {
return _getIncentivesController();
}
/**
* @dev Transfers the underlying asset to `target`. Used by the LendingPool to transfer
* assets in borrow(), withdraw() and flashLoan()
* @param target The recipient of the aTokens
* @param amount The amount getting transferred
* @return The amount transferred
**/functiontransferUnderlyingTo(address target, uint256 amount)
externaloverrideonlyLendingPoolreturns (uint256)
{
IERC20(_underlyingAsset).safeTransfer(target, amount);
return amount;
}
/**
* @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
**/functionhandleRepayment(address user, uint256 amount) externaloverrideonlyLendingPool{}
/**
* @dev implements the permit function as for
* https://github.com/ethereum/EIPs/blob/8a34d644aacf0f9f8f00815307fd7dd5da07655f/EIPS/eip-2612.md
* @param owner The owner of the funds
* @param spender The spender
* @param value The amount
* @param deadline The deadline timestamp, type(uint256).max for max deadline
* @param v Signature param
* @param s Signature param
* @param r Signature param
*/functionpermit(address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external{
require(owner !=address(0), 'INVALID_OWNER');
//solium-disable-next-linerequire(block.timestamp<= deadline, 'INVALID_EXPIRATION');
uint256 currentValidNonce = _nonces[owner];
bytes32 digest =keccak256(
abi.encodePacked(
'\x19\x01',
DOMAIN_SEPARATOR,
keccak256(abi.encode(PERMIT_TYPEHASH, owner, spender, value, currentValidNonce, deadline))
)
);
require(owner ==ecrecover(digest, v, r, s), 'INVALID_SIGNATURE');
_nonces[owner] = currentValidNonce.add(1);
_approve(owner, spender, value);
}
/**
* @dev Transfers the aTokens between two users. Validates the transfer
* (ie checks for valid HF after the transfer) if required
* @param from The source address
* @param to The destination address
* @param amount The amount getting transferred
* @param validate `true` if the transfer needs to be validated
**/function_transfer(addressfrom,
address to,
uint256 amount,
bool validate
) internal{
address underlyingAsset = _underlyingAsset;
ILendingPool pool = _pool;
uint256 index = pool.getReserveNormalizedIncome(underlyingAsset);
uint256 fromBalanceBefore =super.balanceOf(from).rayMul(index);
uint256 toBalanceBefore =super.balanceOf(to).rayMul(index);
super._transfer(from, to, amount.rayDiv(index));
if (validate) {
pool.finalizeTransfer(underlyingAsset, from, to, amount, fromBalanceBefore, toBalanceBefore);
}
emit BalanceTransfer(from, to, amount, index);
}
/**
* @dev Overrides the parent _transfer to force validated transfer() and transferFrom()
* @param from The source address
* @param to The destination address
* @param amount The amount getting transferred
**/function_transfer(addressfrom,
address to,
uint256 amount
) internaloverride{
_transfer(from, to, amount, true);
}
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {Ownable} from'../dependencies/openzeppelin/contracts/Ownable.sol';
import {IERC20} from'../dependencies/openzeppelin/contracts/IERC20.sol';
import {IPriceOracleGetter} from'../interfaces/IPriceOracleGetter.sol';
import {IPriceFeed} from'../interfaces/IPriceFeed.sol';
import {SafeERC20} from'../dependencies/openzeppelin/contracts/SafeERC20.sol';
/// @title AaveOracle/// @author Aave/// @notice Proxy smart contract to get the price of an asset from a price sourcecontractAaveOracleisIPriceOracleGetter, Ownable{
usingSafeERC20forIERC20;
eventAssetSourceUpdated(addressindexed asset, addressindexed source);
mapping(address=> IPriceFeed) private assetsSources;
/// @notice Constructor/// @param assets The addresses of the assets/// @param sources The address of the source of each assetconstructor(address[] memory assets,
address[] memory sources
) {
_setAssetsSources(assets, sources);
}
/// @notice External function called by the Aave governance to set or replace sources of assets/// @param assets The addresses of the assets/// @param sources The address of the source of each assetfunctionsetAssetSources(address[] calldata assets, address[] calldata sources)
externalonlyOwner{
_setAssetsSources(assets, sources);
}
/// @notice Internal function to set the sources for each asset/// @param assets The addresses of the assets/// @param sources The address of the source of each assetfunction_setAssetsSources(address[] memory assets, address[] memory sources) internal{
require(assets.length== sources.length, 'INCONSISTENT_PARAMS_LENGTH');
for (uint256 i =0; i < assets.length; i++) {
assetsSources[assets[i]] = IPriceFeed(sources[i]);
emit AssetSourceUpdated(assets[i], sources[i]);
}
}
/// @notice Gets an asset price by address/// @dev All assets are priced relative to USD/// @param asset The asset addressfunctionupdateAssetPrice(address asset) publicoverridereturns (uint256) {
IPriceFeed source = assetsSources[asset];
return source.updatePrice();
}
/// @notice Gets an asset price by address/// @dev All assets are priced relative to USD/// @param asset The asset addressfunctiongetAssetPrice(address asset) publicviewoverridereturns (uint256) {
IPriceFeed source = assetsSources[asset];
return source.fetchPrice();
}
/// @notice Gets a list of prices from a list of assets addresses/// @param assets The list of assets addressesfunctiongetAssetsPrices(address[] calldata assets) externalviewreturns (uint256[] memory) {
uint256[] memory prices =newuint256[](assets.length);
for (uint256 i =0; i < assets.length; i++) {
prices[i] = getAssetPrice(assets[i]);
}
return prices;
}
/// @notice Gets the address of the source for an asset address/// @param asset The address of the asset/// @return address The address of the sourcefunctiongetSourceOfAsset(address asset) externalviewreturns (address) {
returnaddress(assetsSources[asset]);
}
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @dev Collection of functions related to the address type
*/libraryAddress{
/**
* @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
* ====
*/functionisContract(address account) internalviewreturns (bool) {
// According to EIP-1052, 0x0 is the value returned for not-yet created accounts// and 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470 is returned// for accounts without code, i.e. `keccak256('')`bytes32 codehash;
bytes32 accountHash =0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470;
// solhint-disable-next-line no-inline-assemblyassembly {
codehash :=extcodehash(account)
}
return (codehash != accountHash && codehash !=0x0);
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
*
* IMPORTANT: because control is transferred to `recipient`, care must be
* taken to not create reentrancy vulnerabilities. Consider using
* {ReentrancyGuard} or the
* https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/functionsendValue(addresspayable 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');
}
}
Contract Source Code
File 6 of 91: AdminUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./BaseAdminUpgradeabilityProxy.sol';
/**
* @title AdminUpgradeabilityProxy
* @dev Extends from BaseAdminUpgradeabilityProxy with a constructor for
* initializing the implementation, admin, and init data.
*/contractAdminUpgradeabilityProxyisBaseAdminUpgradeabilityProxy, UpgradeabilityProxy{
/**
* Contract constructor.
* @param _logic address of the initial implementation.
* @param _admin Address of the proxy administrator.
* @param _data Data to send as msg.data to the implementation to initialize the proxied contract.
* It should include the signature and the parameters of the function to be called, as described in
* https://solidity.readthedocs.io/en/v0.4.24/abi-spec.html#function-selector-and-argument-encoding.
* This parameter is optional, if no data is given the initialization call to proxied contract will be skipped.
*/constructor(address _logic,
address _admin,
bytesmemory _data
) payableUpgradeabilityProxy(_logic, _data) {
assert(ADMIN_SLOT ==bytes32(uint256(keccak256('eip1967.proxy.admin')) -1));
_setAdmin(_admin);
}
/**
* @dev Only fall back when the sender is not the admin.
*/function_willFallback() internaloverride(BaseAdminUpgradeabilityProxy, Proxy) {
BaseAdminUpgradeabilityProxy._willFallback();
}
}
Contract Source Code
File 7 of 91: BaseAdminUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./UpgradeabilityProxy.sol';
/**
* @title BaseAdminUpgradeabilityProxy
* @dev This contract combines an upgradeability proxy with an authorization
* mechanism for administrative tasks.
* All external functions in this contract must be guarded by the
* `ifAdmin` modifier. See ethereum/solidity#3864 for a Solidity
* feature proposal that would enable this to be done automatically.
*/contractBaseAdminUpgradeabilityProxyisBaseUpgradeabilityProxy{
/**
* @dev Emitted when the administration has been transferred.
* @param previousAdmin Address of the previous admin.
* @param newAdmin Address of the new admin.
*/eventAdminChanged(address previousAdmin, address newAdmin);
/**
* @dev Storage slot with the admin of the contract.
* This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is
* validated in the constructor.
*/bytes32internalconstant ADMIN_SLOT =0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103;
/**
* @dev Modifier to check whether the `msg.sender` is the admin.
* If it is, it will run the function. Otherwise, it will delegate the call
* to the implementation.
*/modifierifAdmin() {
if (msg.sender== _admin()) {
_;
} else {
_fallback();
}
}
/**
* @return The address of the proxy admin.
*/functionadmin() externalifAdminreturns (address) {
return _admin();
}
/**
* @return The address of the implementation.
*/functionimplementation() externalifAdminreturns (address) {
return _implementation();
}
/**
* @dev Changes the admin of the proxy.
* Only the current admin can call this function.
* @param newAdmin Address to transfer proxy administration to.
*/functionchangeAdmin(address newAdmin) externalifAdmin{
require(newAdmin !=address(0), 'Cannot change the admin of a proxy to the zero address');
emit AdminChanged(_admin(), newAdmin);
_setAdmin(newAdmin);
}
/**
* @dev Upgrade the backing implementation of the proxy.
* Only the admin can call this function.
* @param newImplementation Address of the new implementation.
*/functionupgradeTo(address newImplementation) externalifAdmin{
_upgradeTo(newImplementation);
}
/**
* @dev Upgrade the backing implementation of the proxy and call a function
* on the new implementation.
* This is useful to initialize the proxied contract.
* @param newImplementation Address of the new implementation.
* @param data Data to send as msg.data in the low level call.
* It should include the signature and the parameters of the function to be called, as described in
* https://solidity.readthedocs.io/en/v0.4.24/abi-spec.html#function-selector-and-argument-encoding.
*/functionupgradeToAndCall(address newImplementation, bytescalldata data)
externalpayableifAdmin{
_upgradeTo(newImplementation);
(bool success, ) = newImplementation.delegatecall(data);
require(success);
}
/**
* @return adm The admin slot.
*/function_admin() internalviewreturns (address adm) {
bytes32 slot = ADMIN_SLOT;
//solium-disable-next-lineassembly {
adm :=sload(slot)
}
}
/**
* @dev Sets the address of the proxy admin.
* @param newAdmin Address of the new proxy admin.
*/function_setAdmin(address newAdmin) internal{
bytes32 slot = ADMIN_SLOT;
//solium-disable-next-lineassembly {
sstore(slot, newAdmin)
}
}
/**
* @dev Only fall back when the sender is not the admin.
*/function_willFallback() internalvirtualoverride{
require(msg.sender!= _admin(), 'Cannot call fallback function from the proxy admin');
super._willFallback();
}
}
Contract Source Code
File 8 of 91: BaseImmutableAdminUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'../../../dependencies/openzeppelin/upgradeability/BaseUpgradeabilityProxy.sol';
/**
* @title BaseImmutableAdminUpgradeabilityProxy
* @author Aave, inspired by the OpenZeppelin upgradeability proxy pattern
* @dev This contract combines an upgradeability proxy with an authorization
* mechanism for administrative tasks. The admin role is stored in an immutable, which
* helps saving transactions costs
* All external functions in this contract must be guarded by the
* `ifAdmin` modifier. See ethereum/solidity#3864 for a Solidity
* feature proposal that would enable this to be done automatically.
*/contractBaseImmutableAdminUpgradeabilityProxyisBaseUpgradeabilityProxy{
addressimmutable ADMIN;
constructor(address admin) {
ADMIN = admin;
}
modifierifAdmin() {
if (msg.sender== ADMIN) {
_;
} else {
_fallback();
}
}
/**
* @return The address of the proxy admin.
*/functionadmin() externalifAdminreturns (address) {
return ADMIN;
}
/**
* @return The address of the implementation.
*/functionimplementation() externalifAdminreturns (address) {
return _implementation();
}
/**
* @dev Upgrade the backing implementation of the proxy.
* Only the admin can call this function.
* @param newImplementation Address of the new implementation.
*/functionupgradeTo(address newImplementation) externalifAdmin{
_upgradeTo(newImplementation);
}
/**
* @dev Upgrade the backing implementation of the proxy and call a function
* on the new implementation.
* This is useful to initialize the proxied contract.
* @param newImplementation Address of the new implementation.
* @param data Data to send as msg.data in the low level call.
* It should include the signature and the parameters of the function to be called, as described in
* https://solidity.readthedocs.io/en/v0.4.24/abi-spec.html#function-selector-and-argument-encoding.
*/functionupgradeToAndCall(address newImplementation, bytescalldata data)
externalpayableifAdmin{
_upgradeTo(newImplementation);
(bool success, ) = newImplementation.delegatecall(data);
require(success);
}
/**
* @dev Only fall back when the sender is not the admin.
*/function_willFallback() internalvirtualoverride{
require(msg.sender!= ADMIN, 'Cannot call fallback function from the proxy admin');
super._willFallback();
}
}
Contract Source Code
File 9 of 91: BaseUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./Proxy.sol';
import'../contracts/Address.sol';
/**
* @title BaseUpgradeabilityProxy
* @dev This contract implements a proxy that allows to change the
* implementation address to which it will delegate.
* Such a change is called an implementation upgrade.
*/contractBaseUpgradeabilityProxyisProxy{
/**
* @dev Emitted when the implementation is upgraded.
* @param implementation Address of the new implementation.
*/eventUpgraded(addressindexed implementation);
/**
* @dev Storage slot with the address of the current implementation.
* This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is
* validated in the constructor.
*/bytes32internalconstant IMPLEMENTATION_SLOT =0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc;
/**
* @dev Returns the current implementation.
* @return impl Address of the current implementation
*/function_implementation() internalviewoverridereturns (address impl) {
bytes32 slot = IMPLEMENTATION_SLOT;
//solium-disable-next-lineassembly {
impl :=sload(slot)
}
}
/**
* @dev Upgrades the proxy to a new implementation.
* @param newImplementation Address of the new implementation.
*/function_upgradeTo(address newImplementation) internal{
_setImplementation(newImplementation);
emit Upgraded(newImplementation);
}
/**
* @dev Sets the implementation address of the proxy.
* @param newImplementation Address of the new implementation.
*/function_setImplementation(address newImplementation) internal{
require(
Address.isContract(newImplementation),
'Cannot set a proxy implementation to a non-contract address'
);
bytes32 slot = IMPLEMENTATION_SLOT;
//solium-disable-next-lineassembly {
sstore(slot, newImplementation)
}
}
}
Contract Source Code
File 10 of 91: ChefIncentivesController.sol
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;import"../interfaces/IMultiFeeDistribution.sol";
import"../interfaces/IOnwardIncentivesController.sol";
import"../dependencies/openzeppelin/contracts/IERC20.sol";
import"../dependencies/openzeppelin/contracts/SafeERC20.sol";
import"../dependencies/openzeppelin/contracts/SafeMath.sol";
import"../dependencies/openzeppelin/contracts/Ownable.sol";
// based on the Sushi MasterChef// https://github.com/sushiswap/sushiswap/blob/master/contracts/MasterChef.solcontractChefIncentivesControllerisOwnable{
usingSafeMathforuint256;
usingSafeERC20forIERC20;
// Info of each user.structUserInfo {
uint256 amount;
uint256 rewardDebt;
}
// Info of each pool.structPoolInfo {
uint256 totalSupply;
uint256 allocPoint; // How many allocation points assigned to this pool.uint256 lastRewardTime; // Last second that reward distribution occurs.uint256 accRewardPerShare; // Accumulated rewards per share, times 1e12. See below.
IOnwardIncentivesController onwardIncentives;
}
// Info about token emissions for a given time period.structEmissionPoint {
uint128 startTimeOffset;
uint128 rewardsPerSecond;
}
addresspublic poolConfigurator;
IMultiFeeDistribution public rewardMinter;
uint256public rewardsPerSecond;
uint256publicimmutable maxMintableTokens;
uint256public mintedTokens;
// Info of each pool.address[] public registeredTokens;
mapping(address=> PoolInfo) public poolInfo;
// Data about the future reward rates. emissionSchedule stored in reverse chronological order,// whenever the number of blocks since the start block exceeds the next block offset a new// reward rate is applied.
EmissionPoint[] public emissionSchedule;
// token => user => Info of each user that stakes LP tokens.mapping(address=>mapping(address=> UserInfo)) public userInfo;
// user => base claimable balancemapping(address=>uint256) public userBaseClaimable;
// Total allocation poitns. Must be the sum of all allocation points in all pools.uint256public totalAllocPoint =0;
// The block number when reward mining starts.uint256public startTime;
// account earning rewards => receiver of rewards for this account// if receiver is set to address(0), rewards are paid to the earner// this is used to aid 3rd party contract integrationsmapping (address=>address) public claimReceiver;
eventBalanceUpdated(addressindexed token,
addressindexed user,
uint256 balance,
uint256 totalSupply
);
constructor(uint128[] memory _startTimeOffset,
uint128[] memory _rewardsPerSecond,
address _poolConfigurator,
IMultiFeeDistribution _rewardMinter,
uint256 _maxMintable
)
Ownable()
{
poolConfigurator = _poolConfigurator;
rewardMinter = _rewardMinter;
uint256 length = _startTimeOffset.length;
for (uint256 i = length -1; i +1!=0; i--) {
emissionSchedule.push(
EmissionPoint({
startTimeOffset: _startTimeOffset[i],
rewardsPerSecond: _rewardsPerSecond[i]
})
);
}
maxMintableTokens = _maxMintable;
}
// Start the partyfunctionstart() publiconlyOwner{
require(startTime ==0);
startTime =block.timestamp;
}
// Add a new lp to the pool. Can only be called by the poolConfigurator.functionaddPool(address _token, uint256 _allocPoint) external{
require(msg.sender== poolConfigurator);
require(poolInfo[_token].lastRewardTime ==0);
_updateEmissions();
totalAllocPoint = totalAllocPoint.add(_allocPoint);
registeredTokens.push(_token);
poolInfo[_token] = PoolInfo({
totalSupply: 0,
allocPoint: _allocPoint,
lastRewardTime: block.timestamp,
accRewardPerShare: 0,
onwardIncentives: IOnwardIncentivesController(0)
});
}
// Update the given pool's allocation point. Can only be called by the owner.functionbatchUpdateAllocPoint(address[] calldata _tokens,
uint256[] calldata _allocPoints
) publiconlyOwner{
require(_tokens.length== _allocPoints.length);
_massUpdatePools();
uint256 _totalAllocPoint = totalAllocPoint;
for (uint256 i =0; i < _tokens.length; i++) {
PoolInfo storage pool = poolInfo[_tokens[i]];
require(pool.lastRewardTime >0);
_totalAllocPoint = _totalAllocPoint.sub(pool.allocPoint).add(_allocPoints[i]);
pool.allocPoint = _allocPoints[i];
}
totalAllocPoint = _totalAllocPoint;
}
functionsetOnwardIncentives(address _token,
IOnwardIncentivesController _incentives
)
externalonlyOwner{
require(poolInfo[_token].lastRewardTime !=0);
poolInfo[_token].onwardIncentives = _incentives;
}
functionsetClaimReceiver(address _user, address _receiver) external{
require(msg.sender== _user ||msg.sender== owner());
claimReceiver[_user] = _receiver;
}
functionpoolLength() externalviewreturns (uint256) {
return registeredTokens.length;
}
functionclaimableReward(address _user, address[] calldata _tokens)
externalviewreturns (uint256[] memory)
{
uint256[] memory claimable =newuint256[](_tokens.length);
for (uint256 i =0; i < _tokens.length; i++) {
address token = _tokens[i];
PoolInfo storage pool = poolInfo[token];
UserInfo storage user = userInfo[token][_user];
uint256 accRewardPerShare = pool.accRewardPerShare;
uint256 lpSupply = pool.totalSupply;
if (block.timestamp> pool.lastRewardTime && lpSupply !=0) {
uint256 duration =block.timestamp.sub(pool.lastRewardTime);
uint256 reward = duration.mul(rewardsPerSecond).mul(pool.allocPoint).div(totalAllocPoint);
accRewardPerShare = accRewardPerShare.add(reward.mul(1e12).div(lpSupply));
}
claimable[i] = user.amount.mul(accRewardPerShare).div(1e12).sub(user.rewardDebt);
}
return claimable;
}
function_updateEmissions() internal{
uint256 length = emissionSchedule.length;
if (startTime >0&& length >0) {
EmissionPoint memory e = emissionSchedule[length-1];
if (block.timestamp.sub(startTime) > e.startTimeOffset) {
_massUpdatePools();
rewardsPerSecond =uint256(e.rewardsPerSecond);
emissionSchedule.pop();
}
}
}
// Update reward variables for all poolsfunction_massUpdatePools() internal{
uint256 totalAP = totalAllocPoint;
uint256 length = registeredTokens.length;
for (uint256 i =0; i < length; ++i) {
_updatePool(poolInfo[registeredTokens[i]], totalAP);
}
}
// Update reward variables of the given pool to be up-to-date.function_updatePool(PoolInfo storage pool, uint256 _totalAllocPoint) internal{
if (block.timestamp<= pool.lastRewardTime) {
return;
}
uint256 lpSupply = pool.totalSupply;
if (lpSupply ==0) {
pool.lastRewardTime =block.timestamp;
return;
}
uint256 duration =block.timestamp.sub(pool.lastRewardTime);
uint256 reward = duration.mul(rewardsPerSecond).mul(pool.allocPoint).div(_totalAllocPoint);
pool.accRewardPerShare = pool.accRewardPerShare.add(reward.mul(1e12).div(lpSupply));
pool.lastRewardTime =block.timestamp;
}
function_mint(address _user, uint256 _amount) internal{
uint256 minted = mintedTokens;
if (minted.add(_amount) > maxMintableTokens) {
_amount = maxMintableTokens.sub(minted);
}
if (_amount >0) {
mintedTokens = minted.add(_amount);
address receiver = claimReceiver[_user];
if (receiver ==address(0)) receiver = _user;
rewardMinter.mint(receiver, _amount, true);
}
}
functionhandleAction(address _user, uint256 _balance, uint256 _totalSupply) external{
PoolInfo storage pool = poolInfo[msg.sender];
require(pool.lastRewardTime >0);
_updateEmissions();
_updatePool(pool, totalAllocPoint);
UserInfo storage user = userInfo[msg.sender][_user];
uint256 amount = user.amount;
uint256 accRewardPerShare = pool.accRewardPerShare;
if (amount >0) {
uint256 pending = amount.mul(accRewardPerShare).div(1e12).sub(user.rewardDebt);
if (pending >0) {
userBaseClaimable[_user] = userBaseClaimable[_user].add(pending);
}
}
user.amount = _balance;
user.rewardDebt = _balance.mul(accRewardPerShare).div(1e12);
pool.totalSupply = _totalSupply;
if (pool.onwardIncentives != IOnwardIncentivesController(0)) {
pool.onwardIncentives.handleAction(msg.sender, _user, _balance, _totalSupply);
}
emit BalanceUpdated(msg.sender, _user, _balance, _totalSupply);
}
// Claim pending rewards for one or more pools.// Rewards are not received directly, they are minted by the rewardMinter.functionclaim(address _user, address[] calldata _tokens) external{
_updateEmissions();
uint256 pending = userBaseClaimable[_user];
userBaseClaimable[_user] =0;
uint256 _totalAllocPoint = totalAllocPoint;
for (uint i =0; i < _tokens.length; i++) {
PoolInfo storage pool = poolInfo[_tokens[i]];
require(pool.lastRewardTime >0);
_updatePool(pool, _totalAllocPoint);
UserInfo storage user = userInfo[_tokens[i]][_user];
uint256 rewardDebt = user.amount.mul(pool.accRewardPerShare).div(1e12);
pending = pending.add(rewardDebt.sub(user.rewardDebt));
user.rewardDebt = rewardDebt;
}
_mint(_user, pending);
}
}
Contract Source Code
File 11 of 91: Context.sol
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;/*
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with GSN meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/abstractcontractContext{
function_msgSender() internalviewvirtualreturns (addresspayable) {
returnmsg.sender;
}
function_msgData() internalviewvirtualreturns (bytesmemory) {
this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691returnmsg.data;
}
}
Contract Source Code
File 12 of 91: DataTypes.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;libraryDataTypes{
// refer to the whitepaper, section 1.1 basic concepts for a formal description of these properties.structReserveData {
//stores the reserve configuration
ReserveConfigurationMap configuration;
//the liquidity index. Expressed in rayuint128 liquidityIndex;
//variable borrow index. Expressed in rayuint128 variableBorrowIndex;
//the current supply rate. Expressed in rayuint128 currentLiquidityRate;
//the current variable borrow rate. Expressed in rayuint128 currentVariableBorrowRate;
//the current stable borrow rate. Expressed in rayuint128 currentStableBorrowRate;
uint40 lastUpdateTimestamp;
//tokens addressesaddress aTokenAddress;
address stableDebtTokenAddress;
address variableDebtTokenAddress;
//address of the interest rate strategyaddress interestRateStrategyAddress;
//the id of the reserve. Represents the position in the list of the active reservesuint8 id;
}
structReserveConfigurationMap {
//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 factoruint256 data;
}
structUserConfigurationMap {
uint256 data;
}
enumInterestRateMode {NONE, STABLE, VARIABLE}
}
Contract Source Code
File 13 of 91: DebtTokenBase.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {ILendingPool} from'../../../interfaces/ILendingPool.sol';
import {ICreditDelegationToken} from'../../../interfaces/ICreditDelegationToken.sol';
import {
VersionedInitializable
} from'../../libraries/aave-upgradeability/VersionedInitializable.sol';
import {IncentivizedERC20} from'../IncentivizedERC20.sol';
import {Errors} from'../../libraries/helpers/Errors.sol';
import {SafeMath} from'../../../dependencies/openzeppelin/contracts/SafeMath.sol';
/**
* @title DebtTokenBase
* @notice Base contract for different types of debt tokens, like StableDebtToken or VariableDebtToken
* @author Aave
*/abstractcontractDebtTokenBaseisIncentivizedERC20('DEBTTOKEN_IMPL', 'DEBTTOKEN_IMPL', 0),
VersionedInitializable,
ICreditDelegationToken{
usingSafeMathforuint256;
mapping(address=>mapping(address=>uint256)) internal _borrowAllowances;
/**
* @dev Only lending pool can call functions marked by this modifier
**/modifieronlyLendingPool{
require(_msgSender() ==address(_getLendingPool()), Errors.CT_CALLER_MUST_BE_LENDING_POOL);
_;
}
/**
* @dev delegates borrowing power to a user on the specific debt token
* @param delegatee the address receiving the delegated borrowing power
* @param amount the maximum amount being delegated. Delegation will still
* respect the liquidation constraints (even if delegated, a delegatee cannot
* force a delegator HF to go below 1)
**/functionapproveDelegation(address delegatee, uint256 amount) externaloverride{
_borrowAllowances[_msgSender()][delegatee] = amount;
emit BorrowAllowanceDelegated(_msgSender(), delegatee, _getUnderlyingAssetAddress(), amount);
}
/**
* @dev returns the borrow allowance of the user
* @param fromUser The user to giving allowance
* @param toUser The user to give allowance to
* @return the current allowance of toUser
**/functionborrowAllowance(address fromUser, address toUser)
externalviewoverridereturns (uint256)
{
return _borrowAllowances[fromUser][toUser];
}
/**
* @dev Being non transferrable, the debt token does not implement any of the
* standard ERC20 functions for transfer and allowance.
**/functiontransfer(address recipient, uint256 amount) publicvirtualoverridereturns (bool) {
recipient;
amount;
revert('TRANSFER_NOT_SUPPORTED');
}
functionallowance(address owner, address spender)
publicviewvirtualoverridereturns (uint256)
{
owner;
spender;
revert('ALLOWANCE_NOT_SUPPORTED');
}
functionapprove(address spender, uint256 amount) publicvirtualoverridereturns (bool) {
spender;
amount;
revert('APPROVAL_NOT_SUPPORTED');
}
functiontransferFrom(address sender,
address recipient,
uint256 amount
) publicvirtualoverridereturns (bool) {
sender;
recipient;
amount;
revert('TRANSFER_NOT_SUPPORTED');
}
functionincreaseAllowance(address spender, uint256 addedValue)
publicvirtualoverridereturns (bool)
{
spender;
addedValue;
revert('ALLOWANCE_NOT_SUPPORTED');
}
functiondecreaseAllowance(address spender, uint256 subtractedValue)
publicvirtualoverridereturns (bool)
{
spender;
subtractedValue;
revert('ALLOWANCE_NOT_SUPPORTED');
}
function_decreaseBorrowAllowance(address delegator,
address delegatee,
uint256 amount
) internal{
uint256 newAllowance =
_borrowAllowances[delegator][delegatee].sub(amount, Errors.BORROW_ALLOWANCE_NOT_ENOUGH);
_borrowAllowances[delegator][delegatee] = newAllowance;
emit BorrowAllowanceDelegated(delegator, delegatee, _getUnderlyingAssetAddress(), newAllowance);
}
function_getUnderlyingAssetAddress() internalviewvirtualreturns (address);
function_getLendingPool() internalviewvirtualreturns (ILendingPool);
}
Contract Source Code
File 14 of 91: DefaultReserveInterestRateStrategy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {SafeMath} from'../../dependencies/openzeppelin/contracts/SafeMath.sol';
import {IReserveInterestRateStrategy} from'../../interfaces/IReserveInterestRateStrategy.sol';
import {WadRayMath} from'../libraries/math/WadRayMath.sol';
import {PercentageMath} from'../libraries/math/PercentageMath.sol';
import {ILendingPoolAddressesProvider} from'../../interfaces/ILendingPoolAddressesProvider.sol';
import {ILendingRateOracle} from'../../interfaces/ILendingRateOracle.sol';
import {IERC20} from'../../dependencies/openzeppelin/contracts/IERC20.sol';
/**
* @title DefaultReserveInterestRateStrategy contract
* @notice Implements the calculation of the interest rates depending on the reserve state
* @dev The model of interest rate is based on 2 slopes, one before the `OPTIMAL_UTILIZATION_RATE`
* point of utilization and another from that one to 100%
* - An instance of this same contract, can't be used across different Aave markets, due to the caching
* of the LendingPoolAddressesProvider
* @author Aave
**/contractDefaultReserveInterestRateStrategyisIReserveInterestRateStrategy{
usingWadRayMathforuint256;
usingSafeMathforuint256;
usingPercentageMathforuint256;
/**
* @dev this constant represents the utilization rate at which the pool aims to obtain most competitive borrow rates.
* Expressed in ray
**/uint256publicimmutable OPTIMAL_UTILIZATION_RATE;
/**
* @dev This constant represents the excess utilization rate above the optimal. It's always equal to
* 1-optimal utilization rate. Added as a constant here for gas optimizations.
* Expressed in ray
**/uint256publicimmutable EXCESS_UTILIZATION_RATE;
ILendingPoolAddressesProvider publicimmutable addressesProvider;
// Base variable borrow rate when Utilization rate = 0. Expressed in rayuint256internalimmutable _baseVariableBorrowRate;
// Slope of the variable interest curve when utilization rate > 0 and <= OPTIMAL_UTILIZATION_RATE. Expressed in rayuint256internalimmutable _variableRateSlope1;
// Slope of the variable interest curve when utilization rate > OPTIMAL_UTILIZATION_RATE. Expressed in rayuint256internalimmutable _variableRateSlope2;
// Slope of the stable interest curve when utilization rate > 0 and <= OPTIMAL_UTILIZATION_RATE. Expressed in rayuint256internalimmutable _stableRateSlope1;
// Slope of the stable interest curve when utilization rate > OPTIMAL_UTILIZATION_RATE. Expressed in rayuint256internalimmutable _stableRateSlope2;
constructor(
ILendingPoolAddressesProvider provider,
uint256 optimalUtilizationRate,
uint256 baseVariableBorrowRate,
uint256 variableRateSlope1,
uint256 variableRateSlope2,
uint256 stableRateSlope1,
uint256 stableRateSlope2
) {
OPTIMAL_UTILIZATION_RATE = optimalUtilizationRate;
EXCESS_UTILIZATION_RATE = WadRayMath.ray().sub(optimalUtilizationRate);
addressesProvider = provider;
_baseVariableBorrowRate = baseVariableBorrowRate;
_variableRateSlope1 = variableRateSlope1;
_variableRateSlope2 = variableRateSlope2;
_stableRateSlope1 = stableRateSlope1;
_stableRateSlope2 = stableRateSlope2;
}
functionvariableRateSlope1() externalviewreturns (uint256) {
return _variableRateSlope1;
}
functionvariableRateSlope2() externalviewreturns (uint256) {
return _variableRateSlope2;
}
functionstableRateSlope1() externalviewreturns (uint256) {
return _stableRateSlope1;
}
functionstableRateSlope2() externalviewreturns (uint256) {
return _stableRateSlope2;
}
functionbaseVariableBorrowRate() externalviewoverridereturns (uint256) {
return _baseVariableBorrowRate;
}
functiongetMaxVariableBorrowRate() externalviewoverridereturns (uint256) {
return _baseVariableBorrowRate.add(_variableRateSlope1).add(_variableRateSlope2);
}
/**
* @dev Calculates the interest rates depending on the reserve's state and configurations
* @param reserve The address of the reserve
* @param liquidityAdded The liquidity added during the operation
* @param liquidityTaken The liquidity taken during the operation
* @param totalStableDebt The total borrowed from the reserve a stable rate
* @param totalVariableDebt The total borrowed from the reserve at a variable rate
* @param averageStableBorrowRate The weighted average of all the stable rate loans
* @param reserveFactor The reserve portion of the interest that goes to the treasury of the market
* @return The liquidity rate, the stable borrow rate and the variable borrow rate
**/functioncalculateInterestRates(address reserve,
address aToken,
uint256 liquidityAdded,
uint256 liquidityTaken,
uint256 totalStableDebt,
uint256 totalVariableDebt,
uint256 averageStableBorrowRate,
uint256 reserveFactor
)
externalviewoverridereturns (uint256,
uint256,
uint256)
{
uint256 availableLiquidity = IERC20(reserve).balanceOf(aToken);
//avoid stack too deep
availableLiquidity = availableLiquidity.add(liquidityAdded).sub(liquidityTaken);
return
calculateInterestRates(
reserve,
availableLiquidity,
totalStableDebt,
totalVariableDebt,
averageStableBorrowRate,
reserveFactor
);
}
structCalcInterestRatesLocalVars {
uint256 totalDebt;
uint256 currentVariableBorrowRate;
uint256 currentStableBorrowRate;
uint256 currentLiquidityRate;
uint256 utilizationRate;
}
/**
* @dev Calculates the interest rates depending on the reserve's state and configurations.
* NOTE This function is kept for compatibility with the previous DefaultInterestRateStrategy interface.
* New protocol implementation uses the new calculateInterestRates() interface
* @param reserve The address of the reserve
* @param availableLiquidity The liquidity available in the corresponding aToken
* @param totalStableDebt The total borrowed from the reserve a stable rate
* @param totalVariableDebt The total borrowed from the reserve at a variable rate
* @param averageStableBorrowRate The weighted average of all the stable rate loans
* @param reserveFactor The reserve portion of the interest that goes to the treasury of the market
* @return The liquidity rate, the stable borrow rate and the variable borrow rate
**/functioncalculateInterestRates(address reserve,
uint256 availableLiquidity,
uint256 totalStableDebt,
uint256 totalVariableDebt,
uint256 averageStableBorrowRate,
uint256 reserveFactor
)
publicviewoverridereturns (uint256,
uint256,
uint256)
{
CalcInterestRatesLocalVars memory vars;
vars.totalDebt = totalStableDebt.add(totalVariableDebt);
vars.currentVariableBorrowRate =0;
vars.currentStableBorrowRate =0;
vars.currentLiquidityRate =0;
vars.utilizationRate = vars.totalDebt ==0
? 0
: vars.totalDebt.rayDiv(availableLiquidity.add(vars.totalDebt));
vars.currentStableBorrowRate = ILendingRateOracle(addressesProvider.getLendingRateOracle())
.getMarketBorrowRate(reserve);
if (vars.utilizationRate > OPTIMAL_UTILIZATION_RATE) {
uint256 excessUtilizationRateRatio =
vars.utilizationRate.sub(OPTIMAL_UTILIZATION_RATE).rayDiv(EXCESS_UTILIZATION_RATE);
vars.currentStableBorrowRate = vars.currentStableBorrowRate.add(_stableRateSlope1).add(
_stableRateSlope2.rayMul(excessUtilizationRateRatio)
);
vars.currentVariableBorrowRate = _baseVariableBorrowRate.add(_variableRateSlope1).add(
_variableRateSlope2.rayMul(excessUtilizationRateRatio)
);
} else {
vars.currentStableBorrowRate = vars.currentStableBorrowRate.add(
_stableRateSlope1.rayMul(vars.utilizationRate.rayDiv(OPTIMAL_UTILIZATION_RATE))
);
vars.currentVariableBorrowRate = _baseVariableBorrowRate.add(
vars.utilizationRate.rayMul(_variableRateSlope1).rayDiv(OPTIMAL_UTILIZATION_RATE)
);
}
vars.currentLiquidityRate = _getOverallBorrowRate(
totalStableDebt,
totalVariableDebt,
vars
.currentVariableBorrowRate,
averageStableBorrowRate
)
.rayMul(vars.utilizationRate)
.percentMul(PercentageMath.PERCENTAGE_FACTOR.sub(reserveFactor));
return (
vars.currentLiquidityRate,
vars.currentStableBorrowRate,
vars.currentVariableBorrowRate
);
}
/**
* @dev Calculates the overall borrow rate as the weighted average between the total variable debt and total stable debt
* @param totalStableDebt The total borrowed from the reserve a stable rate
* @param totalVariableDebt The total borrowed from the reserve at a variable rate
* @param currentVariableBorrowRate The current variable borrow rate of the reserve
* @param currentAverageStableBorrowRate The current weighted average of all the stable rate loans
* @return The weighted averaged borrow rate
**/function_getOverallBorrowRate(uint256 totalStableDebt,
uint256 totalVariableDebt,
uint256 currentVariableBorrowRate,
uint256 currentAverageStableBorrowRate
) internalpurereturns (uint256) {
uint256 totalDebt = totalStableDebt.add(totalVariableDebt);
if (totalDebt ==0) return0;
uint256 weightedVariableRate = totalVariableDebt.wadToRay().rayMul(currentVariableBorrowRate);
uint256 weightedStableRate = totalStableDebt.wadToRay().rayMul(currentAverageStableBorrowRate);
uint256 overallBorrowRate =
weightedVariableRate.add(weightedStableRate).rayDiv(totalDebt.wadToRay());
return overallBorrowRate;
}
}
Contract Source Code
File 15 of 91: ERC20.sol
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;import'./Context.sol';
import'./IERC20.sol';
import'./SafeMath.sol';
import'./Address.sol';
/**
* @dev Implementation of the {IERC20} interface.
*
* This implementation is agnostic to the way tokens are created. This means
* that a supply mechanism has to be added in a derived contract using {_mint}.
* For a generic mechanism see {ERC20PresetMinterPauser}.
*
* TIP: For a detailed writeup see our guide
* https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
* to implement supply mechanisms].
*
* We have followed general OpenZeppelin guidelines: functions revert instead
* of returning `false` on failure. This behavior is nonetheless conventional
* and does not conflict with the expectations of ERC20 applications.
*
* Additionally, an {Approval} event is emitted on calls to {transferFrom}.
* This allows applications to reconstruct the allowance for all accounts just
* by listening to said events. Other implementations of the EIP may not emit
* these events, as it isn't required by the specification.
*
* Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
* functions have been added to mitigate the well-known issues around setting
* allowances. See {IERC20-approve}.
*/contractERC20isContext, IERC20{
usingSafeMathforuint256;
usingAddressforaddress;
mapping(address=>uint256) private _balances;
mapping(address=>mapping(address=>uint256)) private _allowances;
uint256private _totalSupply;
stringprivate _name;
stringprivate _symbol;
uint8private _decimals;
/**
* @dev Sets the values for {name} and {symbol}, initializes {decimals} with
* a default value of 18.
*
* To select a different value for {decimals}, use {_setupDecimals}.
*
* All three of these values are immutable: they can only be set once during
* construction.
*/constructor(stringmemory name, stringmemory symbol) {
_name = name;
_symbol = symbol;
_decimals =18;
}
/**
* @dev Returns the name of the token.
*/functionname() publicviewreturns (stringmemory) {
return _name;
}
/**
* @dev Returns the symbol of the token, usually a shorter version of the
* name.
*/functionsymbol() publicviewreturns (stringmemory) {
return _symbol;
}
/**
* @dev Returns the number of decimals used to get its user representation.
* For example, if `decimals` equals `2`, a balance of `505` tokens should
* be displayed to a user as `5,05` (`505 / 10 ** 2`).
*
* Tokens usually opt for a value of 18, imitating the relationship between
* Ether and Wei. This is the value {ERC20} uses, unless {_setupDecimals} is
* called.
*
* NOTE: This information is only used for _display_ purposes: it in
* no way affects any of the arithmetic of the contract, including
* {IERC20-balanceOf} and {IERC20-transfer}.
*/functiondecimals() publicviewreturns (uint8) {
return _decimals;
}
/**
* @dev See {IERC20-totalSupply}.
*/functiontotalSupply() publicviewoverridereturns (uint256) {
return _totalSupply;
}
/**
* @dev See {IERC20-balanceOf}.
*/functionbalanceOf(address account) publicviewoverridereturns (uint256) {
return _balances[account];
}
/**
* @dev See {IERC20-transfer}.
*
* Requirements:
*
* - `recipient` cannot be the zero address.
* - the caller must have a balance of at least `amount`.
*/functiontransfer(address recipient, uint256 amount) publicvirtualoverridereturns (bool) {
_transfer(_msgSender(), recipient, amount);
returntrue;
}
/**
* @dev See {IERC20-allowance}.
*/functionallowance(address owner, address spender)
publicviewvirtualoverridereturns (uint256)
{
return _allowances[owner][spender];
}
/**
* @dev See {IERC20-approve}.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/functionapprove(address spender, uint256 amount) publicvirtualoverridereturns (bool) {
_approve(_msgSender(), spender, amount);
returntrue;
}
/**
* @dev See {IERC20-transferFrom}.
*
* Emits an {Approval} event indicating the updated allowance. This is not
* required by the EIP. See the note at the beginning of {ERC20};
*
* Requirements:
* - `sender` and `recipient` cannot be the zero address.
* - `sender` must have a balance of at least `amount`.
* - the caller must have allowance for ``sender``'s tokens of at least
* `amount`.
*/functiontransferFrom(address sender,
address recipient,
uint256 amount
) publicvirtualoverridereturns (bool) {
_transfer(sender, recipient, amount);
_approve(
sender,
_msgSender(),
_allowances[sender][_msgSender()].sub(amount, 'ERC20: transfer amount exceeds allowance')
);
returntrue;
}
/**
* @dev Atomically increases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/functionincreaseAllowance(address spender, uint256 addedValue) publicvirtualreturns (bool) {
_approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));
returntrue;
}
/**
* @dev Atomically decreases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `spender` must have allowance for the caller of at least
* `subtractedValue`.
*/functiondecreaseAllowance(address spender, uint256 subtractedValue)
publicvirtualreturns (bool)
{
_approve(
_msgSender(),
spender,
_allowances[_msgSender()][spender].sub(
subtractedValue,
'ERC20: decreased allowance below zero'
)
);
returntrue;
}
/**
* @dev Moves tokens `amount` from `sender` to `recipient`.
*
* This is internal function is equivalent to {transfer}, and can be used to
* e.g. implement automatic token fees, slashing mechanisms, etc.
*
* Emits a {Transfer} event.
*
* Requirements:
*
* - `sender` cannot be the zero address.
* - `recipient` cannot be the zero address.
* - `sender` must have a balance of at least `amount`.
*/function_transfer(address sender,
address recipient,
uint256 amount
) internalvirtual{
require(sender !=address(0), 'ERC20: transfer from the zero address');
require(recipient !=address(0), 'ERC20: transfer to the zero address');
_beforeTokenTransfer(sender, recipient, amount);
_balances[sender] = _balances[sender].sub(amount, 'ERC20: transfer amount exceeds balance');
_balances[recipient] = _balances[recipient].add(amount);
emit Transfer(sender, recipient, amount);
}
/** @dev Creates `amount` tokens and assigns them to `account`, increasing
* the total supply.
*
* Emits a {Transfer} event with `from` set to the zero address.
*
* Requirements
*
* - `to` cannot be the zero address.
*/function_mint(address account, uint256 amount) internalvirtual{
require(account !=address(0), 'ERC20: mint to the zero address');
_beforeTokenTransfer(address(0), account, amount);
_totalSupply = _totalSupply.add(amount);
_balances[account] = _balances[account].add(amount);
emit Transfer(address(0), account, amount);
}
/**
* @dev Destroys `amount` tokens from `account`, reducing the
* total supply.
*
* Emits a {Transfer} event with `to` set to the zero address.
*
* Requirements
*
* - `account` cannot be the zero address.
* - `account` must have at least `amount` tokens.
*/function_burn(address account, uint256 amount) internalvirtual{
require(account !=address(0), 'ERC20: burn from the zero address');
_beforeTokenTransfer(account, address(0), amount);
_balances[account] = _balances[account].sub(amount, 'ERC20: burn amount exceeds balance');
_totalSupply = _totalSupply.sub(amount);
emit Transfer(account, address(0), amount);
}
/**
* @dev Sets `amount` as the allowance of `spender` over the `owner`s tokens.
*
* This is internal function is equivalent to `approve`, and can be used to
* e.g. set automatic allowances for certain subsystems, etc.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `owner` cannot be the zero address.
* - `spender` cannot be the zero address.
*/function_approve(address owner,
address spender,
uint256 amount
) internalvirtual{
require(owner !=address(0), 'ERC20: approve from the zero address');
require(spender !=address(0), 'ERC20: approve to the zero address');
_allowances[owner][spender] = amount;
emit Approval(owner, spender, amount);
}
/**
* @dev Sets {decimals} to a value other than the default one of 18.
*
* WARNING: This function should only be called from the constructor. Most
* applications that interact with token contracts will not expect
* {decimals} to ever change, and may work incorrectly if it does.
*/function_setupDecimals(uint8 decimals_) internal{
_decimals = decimals_;
}
/**
* @dev Hook that is called before any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* will be to transferred to `to`.
* - when `from` is zero, `amount` tokens will be minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens will be burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/function_beforeTokenTransfer(addressfrom,
address to,
uint256 amount
) internalvirtual{}
}
Contract Source Code
File 16 of 91: Errors.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
* - 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
*/libraryErrors{
//common errorsstringpublicconstant CALLER_NOT_POOL_ADMIN ='33'; // 'The caller must be the pool admin'stringpublicconstant BORROW_ALLOWANCE_NOT_ENOUGH ='59'; // User borrows on behalf, but allowance are too small//contract specific errorsstringpublicconstant VL_INVALID_AMOUNT ='1'; // 'Amount must be greater than 0'stringpublicconstant VL_NO_ACTIVE_RESERVE ='2'; // 'Action requires an active reserve'stringpublicconstant VL_RESERVE_FROZEN ='3'; // 'Action cannot be performed because the reserve is frozen'stringpublicconstant VL_CURRENT_AVAILABLE_LIQUIDITY_NOT_ENOUGH ='4'; // 'The current liquidity is not enough'stringpublicconstant VL_NOT_ENOUGH_AVAILABLE_USER_BALANCE ='5'; // 'User cannot withdraw more than the available balance'stringpublicconstant VL_TRANSFER_NOT_ALLOWED ='6'; // 'Transfer cannot be allowed.'stringpublicconstant VL_BORROWING_NOT_ENABLED ='7'; // 'Borrowing is not enabled'stringpublicconstant VL_INVALID_INTEREST_RATE_MODE_SELECTED ='8'; // 'Invalid interest rate mode selected'stringpublicconstant VL_COLLATERAL_BALANCE_IS_0 ='9'; // 'The collateral balance is 0'stringpublicconstant VL_HEALTH_FACTOR_LOWER_THAN_LIQUIDATION_THRESHOLD ='10'; // 'Health factor is lesser than the liquidation threshold'stringpublicconstant VL_COLLATERAL_CANNOT_COVER_NEW_BORROW ='11'; // 'There is not enough collateral to cover a new borrow'stringpublicconstant VL_STABLE_BORROWING_NOT_ENABLED ='12'; // stable borrowing not enabledstringpublicconstant VL_COLLATERAL_SAME_AS_BORROWING_CURRENCY ='13'; // collateral is (mostly) the same currency that is being borrowedstringpublicconstant VL_AMOUNT_BIGGER_THAN_MAX_LOAN_SIZE_STABLE ='14'; // 'The requested amount is greater than the max loan size in stable rate modestringpublicconstant 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'stringpublicconstant VL_NO_EXPLICIT_AMOUNT_TO_REPAY_ON_BEHALF ='16'; // 'To repay on behalf of an user an explicit amount to repay is needed'stringpublicconstant VL_NO_STABLE_RATE_LOAN_IN_RESERVE ='17'; // 'User does not have a stable rate loan in progress on this reserve'stringpublicconstant VL_NO_VARIABLE_RATE_LOAN_IN_RESERVE ='18'; // 'User does not have a variable rate loan in progress on this reserve'stringpublicconstant VL_UNDERLYING_BALANCE_NOT_GREATER_THAN_0 ='19'; // 'The underlying balance needs to be greater than 0'stringpublicconstant VL_DEPOSIT_ALREADY_IN_USE ='20'; // 'User deposit is already being used as collateral'stringpublicconstant LP_NOT_ENOUGH_STABLE_BORROW_BALANCE ='21'; // 'User does not have any stable rate loan for this reserve'stringpublicconstant LP_INTEREST_RATE_REBALANCE_CONDITIONS_NOT_MET ='22'; // 'Interest rate rebalance conditions were not met'stringpublicconstant LP_LIQUIDATION_CALL_FAILED ='23'; // 'Liquidation call failed'stringpublicconstant LP_NOT_ENOUGH_LIQUIDITY_TO_BORROW ='24'; // 'There is not enough liquidity available to borrow'stringpublicconstant LP_REQUESTED_AMOUNT_TOO_SMALL ='25'; // 'The requested amount is too small for a FlashLoan.'stringpublicconstant LP_INCONSISTENT_PROTOCOL_ACTUAL_BALANCE ='26'; // 'The actual balance of the protocol is inconsistent'stringpublicconstant LP_CALLER_NOT_LENDING_POOL_CONFIGURATOR ='27'; // 'The caller of the function is not the lending pool configurator'stringpublicconstant LP_INCONSISTENT_FLASHLOAN_PARAMS ='28';
stringpublicconstant CT_CALLER_MUST_BE_LENDING_POOL ='29'; // 'The caller of this function must be a lending pool'stringpublicconstant CT_CANNOT_GIVE_ALLOWANCE_TO_HIMSELF ='30'; // 'User cannot give allowance to himself'stringpublicconstant CT_TRANSFER_AMOUNT_NOT_GT_0 ='31'; // 'Transferred amount needs to be greater than zero'stringpublicconstant RL_RESERVE_ALREADY_INITIALIZED ='32'; // 'Reserve has already been initialized'stringpublicconstant LPC_RESERVE_LIQUIDITY_NOT_0 ='34'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_ATOKEN_POOL_ADDRESS ='35'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_STABLE_DEBT_TOKEN_POOL_ADDRESS ='36'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_VARIABLE_DEBT_TOKEN_POOL_ADDRESS ='37'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_STABLE_DEBT_TOKEN_UNDERLYING_ADDRESS ='38'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_VARIABLE_DEBT_TOKEN_UNDERLYING_ADDRESS ='39'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_ADDRESSES_PROVIDER_ID ='40'; // 'The liquidity of the reserve needs to be 0'stringpublicconstant LPC_INVALID_CONFIGURATION ='75'; // 'Invalid risk parameters for the reserve'stringpublicconstant LPC_CALLER_NOT_EMERGENCY_ADMIN ='76'; // 'The caller must be the emergency admin'stringpublicconstant LPAPR_PROVIDER_NOT_REGISTERED ='41'; // 'Provider is not registered'stringpublicconstant LPCM_HEALTH_FACTOR_NOT_BELOW_THRESHOLD ='42'; // 'Health factor is not below the threshold'stringpublicconstant LPCM_COLLATERAL_CANNOT_BE_LIQUIDATED ='43'; // 'The collateral chosen cannot be liquidated'stringpublicconstant LPCM_SPECIFIED_CURRENCY_NOT_BORROWED_BY_USER ='44'; // 'User did not borrow the specified currency'stringpublicconstant LPCM_NOT_ENOUGH_LIQUIDITY_TO_LIQUIDATE ='45'; // "There isn't enough liquidity available to liquidate"stringpublicconstant LPCM_NO_ERRORS ='46'; // 'No errors'stringpublicconstant LP_INVALID_FLASHLOAN_MODE ='47'; //Invalid flashloan mode selectedstringpublicconstant MATH_MULTIPLICATION_OVERFLOW ='48';
stringpublicconstant MATH_ADDITION_OVERFLOW ='49';
stringpublicconstant MATH_DIVISION_BY_ZERO ='50';
stringpublicconstant RL_LIQUIDITY_INDEX_OVERFLOW ='51'; // Liquidity index overflows uint128stringpublicconstant RL_VARIABLE_BORROW_INDEX_OVERFLOW ='52'; // Variable borrow index overflows uint128stringpublicconstant RL_LIQUIDITY_RATE_OVERFLOW ='53'; // Liquidity rate overflows uint128stringpublicconstant RL_VARIABLE_BORROW_RATE_OVERFLOW ='54'; // Variable borrow rate overflows uint128stringpublicconstant RL_STABLE_BORROW_RATE_OVERFLOW ='55'; // Stable borrow rate overflows uint128stringpublicconstant CT_INVALID_MINT_AMOUNT ='56'; //invalid amount to mintstringpublicconstant LP_FAILED_REPAY_WITH_COLLATERAL ='57';
stringpublicconstant CT_INVALID_BURN_AMOUNT ='58'; //invalid amount to burnstringpublicconstant LP_FAILED_COLLATERAL_SWAP ='60';
stringpublicconstant LP_INVALID_EQUAL_ASSETS_TO_SWAP ='61';
stringpublicconstant LP_REENTRANCY_NOT_ALLOWED ='62';
stringpublicconstant LP_CALLER_MUST_BE_AN_ATOKEN ='63';
stringpublicconstant LP_IS_PAUSED ='64'; // 'Pool is paused'stringpublicconstant LP_NO_MORE_RESERVES_ALLOWED ='65';
stringpublicconstant LP_INVALID_FLASH_LOAN_EXECUTOR_RETURN ='66';
stringpublicconstant RC_INVALID_LTV ='67';
stringpublicconstant RC_INVALID_LIQ_THRESHOLD ='68';
stringpublicconstant RC_INVALID_LIQ_BONUS ='69';
stringpublicconstant RC_INVALID_DECIMALS ='70';
stringpublicconstant RC_INVALID_RESERVE_FACTOR ='71';
stringpublicconstant LPAPR_INVALID_ADDRESSES_PROVIDER_ID ='72';
stringpublicconstant VL_INCONSISTENT_FLASHLOAN_PARAMS ='73';
stringpublicconstant LP_INCONSISTENT_PARAMS_LENGTH ='74';
stringpublicconstant UL_INVALID_INDEX ='77';
stringpublicconstant LP_NOT_CONTRACT ='78';
stringpublicconstant SDT_STABLE_DEBT_OVERFLOW ='79';
stringpublicconstant SDT_BURN_EXCEEDS_BALANCE ='80';
enumCollateralManagerErrors {
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
}
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;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
*/libraryGenericLogic{
usingReserveLogicforDataTypes.ReserveData;
usingSafeMathforuint256;
usingWadRayMathforuint256;
usingPercentageMathforuint256;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
usingUserConfigurationforDataTypes.UserConfigurationMap;
uint256publicconstant HEALTH_FACTOR_LIQUIDATION_THRESHOLD =1ether;
structbalanceDecreaseAllowedLocalVars {
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
**/functionbalanceDecreaseAllowed(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
) externalviewreturns (bool) {
if (!userConfig.isBorrowingAny() ||!userConfig.isUsingAsCollateral(reservesData[asset].id)) {
returntrue;
}
balanceDecreaseAllowedLocalVars memory vars;
(, vars.liquidationThreshold, , vars.decimals, ) = reservesData[asset]
.configuration
.getParams();
if (vars.liquidationThreshold ==0) {
returntrue;
}
(
vars.totalCollateralInETH,
vars.totalDebtInETH,
,
vars.avgLiquidationThreshold,
) = calculateUserAccountData(user, reservesData, userConfig, reserves, reservesCount, oracle);
if (vars.totalDebtInETH ==0) {
returntrue;
}
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 collateralif (vars.collateralBalanceAfterDecrease ==0) {
returnfalse;
}
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;
}
structCalculateUserAccountDataVars {
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
**/functioncalculateUserAccountData(address user,
mapping(address => DataTypes.ReserveData) storage reservesData,
DataTypes.UserConfigurationMap memory userConfig,
mapping(uint256 => address) storage reserves,
uint256 reservesCount,
address oracle
)
internalviewreturns (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
**/functioncalculateHealthFactorFromBalances(uint256 totalCollateralInETH,
uint256 totalDebtInETH,
uint256 liquidationThreshold
) internalpurereturns (uint256) {
if (totalDebtInETH ==0) returnuint256(-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
**/functioncalculateAvailableBorrowsETH(uint256 totalCollateralInETH,
uint256 totalDebtInETH,
uint256 ltv
) internalpurereturns (uint256) {
uint256 availableBorrowsETH = totalCollateralInETH.percentMul(ltv);
if (availableBorrowsETH < totalDebtInETH) {
return0;
}
availableBorrowsETH = availableBorrowsETH.sub(totalDebtInETH);
return availableBorrowsETH;
}
}
Contract Source Code
File 19 of 91: Helpers.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {IERC20} from'../../../dependencies/openzeppelin/contracts/IERC20.sol';
import {DataTypes} from'../types/DataTypes.sol';
/**
* @title Helpers library
* @author Aave
*/libraryHelpers{
/**
* @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
**/functiongetUserCurrentDebt(address user, DataTypes.ReserveData storage reserve)
internalviewreturns (uint256, uint256)
{
return (
IERC20(reserve.stableDebtTokenAddress).balanceOf(user),
IERC20(reserve.variableDebtTokenAddress).balanceOf(user)
);
}
functiongetUserCurrentDebtMemory(address user, DataTypes.ReserveData memory reserve)
internalviewreturns (uint256, uint256)
{
return (
IERC20(reserve.stableDebtTokenAddress).balanceOf(user),
IERC20(reserve.variableDebtTokenAddress).balanceOf(user)
);
}
}
Contract Source Code
File 20 of 91: IAToken.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {IERC20} from'../dependencies/openzeppelin/contracts/IERC20.sol';
import {IScaledBalanceToken} from'./IScaledBalanceToken.sol';
import {IInitializableAToken} from'./IInitializableAToken.sol';
import {IAaveIncentivesController} from'./IAaveIncentivesController.sol';
interfaceIATokenisIERC20, 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
**/eventMint(addressindexedfrom, 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
*/functionmint(address user,
uint256 amount,
uint256 index
) externalreturns (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
**/eventBurn(addressindexedfrom, addressindexed 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
**/eventBalanceTransfer(addressindexedfrom, addressindexed 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
**/functionburn(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
*/functionmintToTreasury(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
**/functiontransferOnLiquidation(addressfrom,
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
**/functiontransferUnderlyingTo(address user, uint256 amount) externalreturns (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
**/functionhandleRepayment(address user, uint256 amount) external;
/**
* @dev Returns the address of the incentives controller contract
**/functiongetIncentivesController() externalviewreturns (IAaveIncentivesController);
/**
* @dev Returns the address of the underlying asset of this aToken (E.g. WETH for aWETH)
**/functionUNDERLYING_ASSET_ADDRESS() externalviewreturns (address);
}
Contract Source Code
File 21 of 91: IAaveIncentivesController.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;interfaceIAaveIncentivesController{
eventRewardsAccrued(addressindexed user, uint256 amount);
eventRewardsClaimed(addressindexed user, addressindexed to, uint256 amount);
eventRewardsClaimed(addressindexed user,
addressindexed to,
addressindexed claimer,
uint256 amount
);
eventClaimerSet(addressindexed user, addressindexed claimer);
/*
* @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
**/functiongetAssetData(address asset)
externalviewreturns (uint256,
uint256,
uint256);
/**
* @dev Whitelists an address to claim the rewards on behalf of another address
* @param user The address of the user
* @param claimer The address of the claimer
*/functionsetClaimer(address user, address claimer) external;
/**
* @dev Returns the whitelisted claimer for a certain address (0x0 if not set)
* @param user The address of the user
* @return The claimer address
*/functiongetClaimer(address user) externalviewreturns (address);
/**
* @dev Configure assets for a certain rewards emission
* @param assets The assets to incentivize
* @param emissionsPerSecond The emission for each asset
*/functionconfigureAssets(address[] calldata assets, uint256[] calldata emissionsPerSecond)
external;
/**
* @dev Called by the corresponding asset on any update that affects the rewards distribution
* @param user The address of the user
* @param userBalance The balance of the user of the asset in the lending pool
* @param totalSupply The total supply of the asset in the lending pool
**/functionhandleAction(address user,
uint256 userBalance,
uint256 totalSupply
) external;
/**
* @dev Returns the total of rewards of an user, already accrued + not yet accrued
* @param user The address of the user
* @return The rewards
**/functiongetRewardsBalance(address[] calldata assets, address user)
externalviewreturns (uint256);
/**
* @dev Claims reward for an user, on all the assets of the lending pool, accumulating the pending rewards
* @param amount Amount of rewards to claim
* @param to Address that will be receiving the rewards
* @return Rewards claimed
**/functionclaimRewards(address[] calldata assets,
uint256 amount,
address to
) externalreturns (uint256);
/**
* @dev Claims reward for an user on behalf, on all the assets of the lending pool, accumulating the pending rewards. The caller must
* be whitelisted via "allowClaimOnBehalf" function by the RewardsAdmin role manager
* @param amount Amount of rewards to claim
* @param user Address to check and claim rewards
* @param to Address that will be receiving the rewards
* @return Rewards claimed
**/functionclaimRewardsOnBehalf(address[] calldata assets,
uint256 amount,
address user,
address to
) externalreturns (uint256);
/**
* @dev returns the unclaimed rewards of the user
* @param user the address of the user
* @return the unclaimed user rewards
*/functiongetUserUnclaimedRewards(address user) externalviewreturns (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
*/functiongetUserAssetData(address user, address asset) externalviewreturns (uint256);
/**
* @dev for backward compatibility with previous implementation of the Incentives controller
*/functionREWARD_TOKEN() externalviewreturns (address);
/**
* @dev for backward compatibility with previous implementation of the Incentives controller
*/functionPRECISION() externalviewreturns (uint8);
/**
* @dev Gets the distribution end timestamp of the emissions
*/functionDISTRIBUTION_END() externalviewreturns (uint256);
}
Contract Source Code
File 22 of 91: IBandStdReference.sol
pragmasolidity 0.7.6;interfaceIBandStdReference{
/// Returns the price data for the given base/quote pair. Revert if not available.functiongetReferenceData(stringmemory _base, stringmemory _quote)
externalviewreturns (uint256 rate, uint256 lastUpdatedBase, uint256 lastUpdatedRate);
}
Contract Source Code
File 23 of 91: IChainlinkAggregator.sol
// SPDX-License-Identifier: MIT// Code from https://github.com/smartcontractkit/chainlink/blob/master/evm-contracts/src/v0.6/interfaces/AggregatorV3Interface.solpragmasolidity 0.7.6;interfaceIChainlinkAggregator{
functiondecimals() externalviewreturns (uint8);
functiondescription() externalviewreturns (stringmemory);
functionversion() externalviewreturns (uint256);
// getRoundData and latestRoundData should both raise "No data present"// if they do not have data to report, instead of returning unset values// which could be misinterpreted as actual reported values.functiongetRoundData(uint80 _roundId)
externalviewreturns (uint80 roundId,
int256 answer,
uint256 startedAt,
uint256 updatedAt,
uint80 answeredInRound
);
functionlatestRoundData()
externalviewreturns (uint80 roundId,
int256 answer,
uint256 startedAt,
uint256 updatedAt,
uint80 answeredInRound
);
}
Contract Source Code
File 24 of 91: IChefIncentivesController.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;interfaceIChefIncentivesController{
/**
* @dev Called by the corresponding asset on any update that affects the rewards distribution
* @param user The address of the user
* @param userBalance The balance of the user of the asset in the lending pool
* @param totalSupply The total supply of the asset in the lending pool
**/functionhandleAction(address user,
uint256 userBalance,
uint256 totalSupply
) external;
functionaddPool(address _token, uint256 _allocPoint) external;
functionclaim(address _user, address[] calldata _tokens) external;
functionsetClaimReceiver(address _user, address _receiver) external;
}
Contract Source Code
File 25 of 91: ICreditDelegationToken.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;interfaceICreditDelegationToken{
eventBorrowAllowanceDelegated(addressindexed fromUser,
addressindexed toUser,
address asset,
uint256 amount
);
/**
* @dev delegates borrowing power to a user on the specific debt token
* @param delegatee the address receiving the delegated borrowing power
* @param amount the maximum amount being delegated. Delegation will still
* respect the liquidation constraints (even if delegated, a delegatee cannot
* force a delegator HF to go below 1)
**/functionapproveDelegation(address delegatee, uint256 amount) external;
/**
* @dev returns the borrow allowance of the user
* @param fromUser The user to giving allowance
* @param toUser The user to give allowance to
* @return the current allowance of toUser
**/functionborrowAllowance(address fromUser, address toUser) externalviewreturns (uint256);
}
Contract Source Code
File 26 of 91: IERC20.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/interfaceIERC20{
/**
* @dev Returns the amount of tokens in existence.
*/functiontotalSupply() externalviewreturns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/functionbalanceOf(address account) externalviewreturns (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.
*/functiontransfer(address recipient, uint256 amount) externalreturns (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.
*/functionallowance(address owner, address spender) externalviewreturns (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.
*/functionapprove(address spender, uint256 amount) externalreturns (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.
*/functiontransferFrom(address sender,
address recipient,
uint256 amount
) externalreturns (bool);
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/eventTransfer(addressindexedfrom, addressindexed 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.
*/eventApproval(addressindexed owner, addressindexed spender, uint256 value);
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {ILendingPool} from'./ILendingPool.sol';
import {IAaveIncentivesController} from'./IAaveIncentivesController.sol';
/**
* @title IInitializableAToken
* @notice Interface for the initialize function on AToken
* @author Aave
**/interfaceIInitializableAToken{
/**
* @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 incentivesController The address of the incentives controller for this aToken
* @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
**/eventInitialized(addressindexed underlyingAsset,
addressindexed pool,
address treasury,
address incentivesController,
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 incentivesController The smart contract managing potential incentives distribution
* @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
*/functioninitialize(
ILendingPool pool,
address treasury,
address underlyingAsset,
IAaveIncentivesController incentivesController,
uint8 aTokenDecimals,
stringcalldata aTokenName,
stringcalldata aTokenSymbol,
bytescalldata params
) external;
}
Contract Source Code
File 31 of 91: IInitializableDebtToken.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {ILendingPool} from'./ILendingPool.sol';
import {IAaveIncentivesController} from'./IAaveIncentivesController.sol';
/**
* @title IInitializableDebtToken
* @notice Interface for the initialize function common between debt tokens
* @author Aave
**/interfaceIInitializableDebtToken{
/**
* @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 incentivesController The address of the incentives controller for this aToken
* @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
**/eventInitialized(addressindexed underlyingAsset,
addressindexed pool,
address incentivesController,
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 incentivesController The smart contract managing potential incentives distribution
* @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
*/functioninitialize(
ILendingPool pool,
address underlyingAsset,
IAaveIncentivesController incentivesController,
uint8 debtTokenDecimals,
stringmemory debtTokenName,
stringmemory debtTokenSymbol,
bytescalldata params
) external;
}
Contract Source Code
File 32 of 91: ILendingPool.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;import {ILendingPoolAddressesProvider} from'./ILendingPoolAddressesProvider.sol';
import {DataTypes} from'../protocol/libraries/types/DataTypes.sol';
interfaceILendingPool{
/**
* @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
**/eventDeposit(addressindexed reserve,
address user,
addressindexed onBehalfOf,
uint256 amount,
uint16indexed 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
**/eventWithdraw(addressindexed reserve, addressindexed user, addressindexed 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
**/eventBorrow(addressindexed reserve,
address user,
addressindexed onBehalfOf,
uint256 amount,
uint256 borrowRateMode,
uint256 borrowRate,
uint16indexed 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
**/eventRepay(addressindexed reserve,
addressindexed user,
addressindexed 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
**/eventSwap(addressindexed reserve, addressindexed 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
**/eventReserveUsedAsCollateralEnabled(addressindexed reserve, addressindexed 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
**/eventReserveUsedAsCollateralDisabled(addressindexed reserve, addressindexed 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
**/eventRebalanceStableBorrowRate(addressindexed reserve, addressindexed 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
**/eventFlashLoan(addressindexed target,
addressindexed initiator,
addressindexed asset,
uint256 amount,
uint256 premium,
uint16 referralCode
);
/**
* @dev Emitted when the pause is triggered.
*/eventPaused();
/**
* @dev Emitted when the pause is lifted.
*/eventUnpaused();
/**
* @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
**/eventLiquidationCall(addressindexed collateralAsset,
addressindexed debtAsset,
addressindexed user,
uint256 debtToCover,
uint256 liquidatedCollateralAmount,
address liquidator,
bool receiveAToken
);
/**
* @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
**/eventReserveDataUpdated(addressindexed reserve,
uint256 liquidityRate,
uint256 stableBorrowRate,
uint256 variableBorrowRate,
uint256 liquidityIndex,
uint256 variableBorrowIndex
);
/**
* @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
**/functiondeposit(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
**/functionwithdraw(address asset,
uint256 amount,
address to
) externalreturns (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
**/functionborrow(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
**/functionrepay(address asset,
uint256 amount,
uint256 rateMode,
address onBehalfOf
) externalreturns (uint256);
/**
* @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
**/functionswapBorrowRateMode(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
**/functionrebalanceStableBorrowRate(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
**/functionsetUserUseReserveAsCollateral(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
**/functionliquidationCall(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 modes Types of the debt to open if the flash loan is not returned:
* 0 -> Don't open any debt, just revert if funds can't be transferred from the receiver
* 1 -> Open debt at stable rate for the value of the amount flash-borrowed to the `onBehalfOf` address
* 2 -> Open debt at variable rate for the value of the amount flash-borrowed to the `onBehalfOf` address
* @param onBehalfOf The address that will receive the debt in the case of using on `modes` 1 or 2
* @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
**/functionflashLoan(address receiverAddress,
address[] calldata assets,
uint256[] calldata amounts,
uint256[] calldata modes,
address onBehalfOf,
bytescalldata 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
**/functiongetUserAccountData(address user)
externalviewreturns (uint256 totalCollateralETH,
uint256 totalDebtETH,
uint256 availableBorrowsETH,
uint256 currentLiquidationThreshold,
uint256 ltv,
uint256 healthFactor
);
functioninitReserve(address reserve,
address aTokenAddress,
address stableDebtAddress,
address variableDebtAddress,
address interestRateStrategyAddress
) external;
functionsetReserveInterestRateStrategyAddress(address reserve, address rateStrategyAddress)
external;
functionsetConfiguration(address reserve, uint256 configuration) 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
**/functiongetConfiguration(address asset)
externalviewreturns (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
**/functiongetUserConfiguration(address user)
externalviewreturns (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
*/functiongetReserveNormalizedIncome(address asset) externalviewreturns (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
*/functiongetReserveNormalizedVariableDebt(address asset) externalviewreturns (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
**/functiongetReserveData(address asset) externalviewreturns (DataTypes.ReserveData memory);
functionfinalizeTransfer(address asset,
addressfrom,
address to,
uint256 amount,
uint256 balanceFromAfter,
uint256 balanceToBefore
) external;
functiongetReservesList() externalviewreturns (address[] memory);
functiongetAddressesProvider() externalviewreturns (ILendingPoolAddressesProvider);
functionsetPause(bool val) external;
functionpaused() externalviewreturns (bool);
}
Contract Source Code
File 33 of 91: ILendingPoolAddressesProvider.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
**/interfaceILendingPoolAddressesProvider{
eventMarketIdSet(string newMarketId);
eventLendingPoolUpdated(addressindexed newAddress);
eventConfigurationAdminUpdated(addressindexed newAddress);
eventEmergencyAdminUpdated(addressindexed newAddress);
eventLendingPoolConfiguratorUpdated(addressindexed newAddress);
eventLendingPoolCollateralManagerUpdated(addressindexed newAddress);
eventPriceOracleUpdated(addressindexed newAddress);
eventLendingRateOracleUpdated(addressindexed newAddress);
eventProxyCreated(bytes32 id, addressindexed newAddress);
eventAddressSet(bytes32 id, addressindexed newAddress, bool hasProxy);
functiongetMarketId() externalviewreturns (stringmemory);
functionsetMarketId(stringcalldata marketId) external;
functionsetAddress(bytes32 id, address newAddress) external;
functionsetAddressAsProxy(bytes32 id, address impl) external;
functiongetAddress(bytes32 id) externalviewreturns (address);
functiongetLendingPool() externalviewreturns (address);
functionsetLendingPoolImpl(address pool) external;
functiongetLendingPoolConfigurator() externalviewreturns (address);
functionsetLendingPoolConfiguratorImpl(address configurator) external;
functiongetLendingPoolCollateralManager() externalviewreturns (address);
functionsetLendingPoolCollateralManager(address manager) external;
functiongetPoolAdmin() externalviewreturns (address);
functionsetPoolAdmin(address admin) external;
functiongetEmergencyAdmin() externalviewreturns (address);
functionsetEmergencyAdmin(address admin) external;
functiongetPriceOracle() externalviewreturns (address);
functionsetPriceOracle(address priceOracle) external;
functiongetLendingRateOracle() externalviewreturns (address);
functionsetLendingRateOracle(address lendingRateOracle) external;
}
Contract Source Code
File 34 of 91: ILendingPoolAddressesProviderRegistry.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @title LendingPoolAddressesProviderRegistry contract
* @dev Main registry of LendingPoolAddressesProvider of multiple Aave protocol's markets
* - Used for indexing purposes of Aave protocol's markets
* - The id assigned to a LendingPoolAddressesProvider refers to the market it is connected with,
* for example with `0` for the Aave main market and `1` for the next created
* @author Aave
**/interfaceILendingPoolAddressesProviderRegistry{
eventAddressesProviderRegistered(addressindexed newAddress);
eventAddressesProviderUnregistered(addressindexed newAddress);
functiongetAddressesProvidersList() externalviewreturns (address[] memory);
functiongetAddressesProviderIdByAddress(address addressesProvider)
externalviewreturns (uint256);
functionregisterAddressesProvider(address provider, uint256 id) external;
functionunregisterAddressesProvider(address provider) external;
}
Contract Source Code
File 35 of 91: ILendingPoolCollateralManager.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @title ILendingPoolCollateralManager
* @author Aave
* @notice Defines the actions involving management of collateral in the protocol.
**/interfaceILendingPoolCollateralManager{
/**
* @dev Emitted when a borrower is liquidated
* @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 debtToCover 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
**/eventLiquidationCall(addressindexed collateral,
addressindexed principal,
addressindexed user,
uint256 debtToCover,
uint256 liquidatedCollateralAmount,
address liquidator,
bool receiveAToken
);
/**
* @dev Emitted when a reserve is disabled as collateral for an user
* @param reserve The address of the reserve
* @param user The address of the user
**/eventReserveUsedAsCollateralDisabled(addressindexed reserve, addressindexed user);
/**
* @dev Emitted when a reserve is enabled as collateral for an user
* @param reserve The address of the reserve
* @param user The address of the user
**/eventReserveUsedAsCollateralEnabled(addressindexed reserve, addressindexed user);
/**
* @dev Users can invoke this function to liquidate an undercollateralized position.
* @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 debtToCover The amount of principal that the liquidator wants to repay
* @param receiveAToken true if the liquidators wants to receive the aTokens, false if
* he wants to receive the underlying asset directly
**/functionliquidationCall(address collateral,
address principal,
address user,
uint256 debtToCover,
bool receiveAToken
) externalreturns (uint256, stringmemory);
}
Contract Source Code
File 36 of 91: ILendingPoolConfigurator.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;interfaceILendingPoolConfigurator{
structInitReserveInput {
address aTokenImpl;
address stableDebtTokenImpl;
address variableDebtTokenImpl;
uint8 underlyingAssetDecimals;
address interestRateStrategyAddress;
address underlyingAsset;
address treasury;
address incentivesController;
uint256 allocPoint;
string underlyingAssetName;
string aTokenName;
string aTokenSymbol;
string variableDebtTokenName;
string variableDebtTokenSymbol;
string stableDebtTokenName;
string stableDebtTokenSymbol;
bytes params;
}
structUpdateATokenInput {
address asset;
address treasury;
address incentivesController;
string name;
string symbol;
address implementation;
bytes params;
}
structUpdateDebtTokenInput {
address asset;
address incentivesController;
string name;
string symbol;
address implementation;
bytes params;
}
/**
* @dev Emitted when a reserve is initialized.
* @param asset The address of the underlying asset of the reserve
* @param aToken The address of the associated aToken contract
* @param stableDebtToken The address of the associated stable rate debt token
* @param variableDebtToken The address of the associated variable rate debt token
* @param interestRateStrategyAddress The address of the interest rate strategy for the reserve
**/eventReserveInitialized(addressindexed asset,
addressindexed aToken,
address stableDebtToken,
address variableDebtToken,
address interestRateStrategyAddress
);
/**
* @dev Emitted when borrowing is enabled on a reserve
* @param asset The address of the underlying asset of the reserve
* @param stableRateEnabled True if stable rate borrowing is enabled, false otherwise
**/eventBorrowingEnabledOnReserve(addressindexed asset, bool stableRateEnabled);
/**
* @dev Emitted when borrowing is disabled on a reserve
* @param asset The address of the underlying asset of the reserve
**/eventBorrowingDisabledOnReserve(addressindexed asset);
/**
* @dev Emitted when the collateralization risk parameters for the specified asset are updated.
* @param asset The address of the underlying asset of the reserve
* @param ltv The loan to value of the asset when used as collateral
* @param liquidationThreshold The threshold at which loans using this asset as collateral will be considered undercollateralized
* @param liquidationBonus The bonus liquidators receive to liquidate this asset
**/eventCollateralConfigurationChanged(addressindexed asset,
uint256 ltv,
uint256 liquidationThreshold,
uint256 liquidationBonus
);
/**
* @dev Emitted when stable rate borrowing is enabled on a reserve
* @param asset The address of the underlying asset of the reserve
**/eventStableRateEnabledOnReserve(addressindexed asset);
/**
* @dev Emitted when stable rate borrowing is disabled on a reserve
* @param asset The address of the underlying asset of the reserve
**/eventStableRateDisabledOnReserve(addressindexed asset);
/**
* @dev Emitted when a reserve is activated
* @param asset The address of the underlying asset of the reserve
**/eventReserveActivated(addressindexed asset);
/**
* @dev Emitted when a reserve is deactivated
* @param asset The address of the underlying asset of the reserve
**/eventReserveDeactivated(addressindexed asset);
/**
* @dev Emitted when a reserve is frozen
* @param asset The address of the underlying asset of the reserve
**/eventReserveFrozen(addressindexed asset);
/**
* @dev Emitted when a reserve is unfrozen
* @param asset The address of the underlying asset of the reserve
**/eventReserveUnfrozen(addressindexed asset);
/**
* @dev Emitted when a reserve factor is updated
* @param asset The address of the underlying asset of the reserve
* @param factor The new reserve factor
**/eventReserveFactorChanged(addressindexed asset, uint256 factor);
/**
* @dev Emitted when the reserve decimals are updated
* @param asset The address of the underlying asset of the reserve
* @param decimals The new decimals
**/eventReserveDecimalsChanged(addressindexed asset, uint256 decimals);
/**
* @dev Emitted when a reserve interest strategy contract is updated
* @param asset The address of the underlying asset of the reserve
* @param strategy The new address of the interest strategy contract
**/eventReserveInterestRateStrategyChanged(addressindexed asset, address strategy);
/**
* @dev Emitted when an aToken implementation is upgraded
* @param asset The address of the underlying asset of the reserve
* @param proxy The aToken proxy address
* @param implementation The new aToken implementation
**/eventATokenUpgraded(addressindexed asset,
addressindexed proxy,
addressindexed implementation
);
/**
* @dev Emitted when the implementation of a stable debt token is upgraded
* @param asset The address of the underlying asset of the reserve
* @param proxy The stable debt token proxy address
* @param implementation The new aToken implementation
**/eventStableDebtTokenUpgraded(addressindexed asset,
addressindexed proxy,
addressindexed implementation
);
/**
* @dev Emitted when the implementation of a variable debt token is upgraded
* @param asset The address of the underlying asset of the reserve
* @param proxy The variable debt token proxy address
* @param implementation The new aToken implementation
**/eventVariableDebtTokenUpgraded(addressindexed asset,
addressindexed proxy,
addressindexed implementation
);
}
Contract Source Code
File 37 of 91: ILendingRateOracle.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @title ILendingRateOracle interface
* @notice Interface for the Aave borrow rate oracle. Provides the average market borrow rate to be used as a base for the stable borrow rate calculations
**/interfaceILendingRateOracle{
/**
@dev returns the market borrow rate in ray
**/functiongetMarketBorrowRate(address asset) externalviewreturns (uint256);
/**
@dev sets the market borrow rate. Rate value must be in ray
**/functionsetMarketBorrowRate(address asset, uint256 rate) external;
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/************
@title IPriceOracle interface
@notice Interface for the Aave price oracle.*/interfaceIPriceOracle{
/***********
@dev returns the asset price in ETH
*/functiongetAssetPrice(address asset) externalviewreturns (uint256);
/***********
@dev sets the asset price, in wei
*/functionsetAssetPrice(address asset, uint256 price) external;
}
Contract Source Code
File 42 of 91: IPriceOracleGetter.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @title IPriceOracleGetter interface
* @notice Interface for the Aave price oracle.
**/interfaceIPriceOracleGetter{
/**
* @dev returns the asset price in ETH
* @param asset the address of the asset
* @return the ETH price of the asset
**/functiongetAssetPrice(address asset) externalviewreturns (uint256);
functionupdateAssetPrice(address asset) externalreturns (uint256);
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;interfaceIScaledBalanceToken{
/**
* @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
**/functionscaledBalanceOf(address user) externalviewreturns (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
**/functiongetScaledUserBalanceAndSupply(address user) externalviewreturns (uint256, uint256);
/**
* @dev Returns the scaled total supply of the variable debt token. Represents sum(debt/index)
* @return The scaled total supply
**/functionscaledTotalSupply() externalviewreturns (uint256);
}
Contract Source Code
File 45 of 91: IStableDebtToken.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {IInitializableDebtToken} from'./IInitializableDebtToken.sol';
import {IAaveIncentivesController} from'./IAaveIncentivesController.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
**/interfaceIStableDebtTokenisIInitializableDebtToken{
/**
* @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
**/eventMint(addressindexed user,
addressindexed 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
**/eventBurn(addressindexed 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
**/functionmint(address user,
address onBehalfOf,
uint256 amount,
uint256 rate
) externalreturns (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
**/functionburn(address user, uint256 amount) external;
/**
* @dev Returns the average rate of all the stable rate loans.
* @return The average stable rate
**/functiongetAverageStableRate() externalviewreturns (uint256);
/**
* @dev Returns the stable rate of the user debt
* @return The stable rate of the user
**/functiongetUserStableRate(address user) externalviewreturns (uint256);
/**
* @dev Returns the timestamp of the last update of the user
* @return The timestamp
**/functiongetUserLastUpdated(address user) externalviewreturns (uint40);
/**
* @dev Returns the principal, the total supply and the average stable rate
**/functiongetSupplyData()
externalviewreturns (uint256,
uint256,
uint256,
uint40);
/**
* @dev Returns the timestamp of the last update of the total supply
* @return The timestamp
**/functiongetTotalSupplyLastUpdated() externalviewreturns (uint40);
/**
* @dev Returns the total supply and the average stable rate
**/functiongetTotalSupplyAndAvgRate() externalviewreturns (uint256, uint256);
/**
* @dev Returns the principal debt balance of the user
* @return The debt balance of the user since the last burn/mint action
**/functionprincipalBalanceOf(address user) externalviewreturns (uint256);
/**
* @dev Returns the address of the incentives controller contract
**/functiongetIncentivesController() externalviewreturns (IAaveIncentivesController);
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {IScaledBalanceToken} from'./IScaledBalanceToken.sol';
import {IInitializableDebtToken} from'./IInitializableDebtToken.sol';
import {IAaveIncentivesController} from'./IAaveIncentivesController.sol';
/**
* @title IVariableDebtToken
* @author Aave
* @notice Defines the basic interface for a variable debt token.
**/interfaceIVariableDebtTokenisIScaledBalanceToken, 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
**/eventMint(addressindexedfrom, addressindexed 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
**/functionmint(address user,
address onBehalfOf,
uint256 amount,
uint256 index
) externalreturns (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
**/eventBurn(addressindexed 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
**/functionburn(address user,
uint256 amount,
uint256 index
) external;
/**
* @dev Returns the address of the incentives controller contract
**/functiongetIncentivesController() externalviewreturns (IAaveIncentivesController);
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {Context} from'../../dependencies/openzeppelin/contracts/Context.sol';
import {IERC20} from'../../dependencies/openzeppelin/contracts/IERC20.sol';
import {IERC20Detailed} from'../../dependencies/openzeppelin/contracts/IERC20Detailed.sol';
import {SafeMath} from'../../dependencies/openzeppelin/contracts/SafeMath.sol';
import {IAaveIncentivesController} from'../../interfaces/IAaveIncentivesController.sol';
import {ILendingPoolAddressesProvider} from'../../interfaces/ILendingPoolAddressesProvider.sol';
import {IPriceOracle} from'../../interfaces/IPriceOracle.sol';
import {ILendingPool} from'../../interfaces/ILendingPool.sol';
/**
* @title ERC20
* @notice Basic ERC20 implementation
* @author Aave, inspired by the Openzeppelin ERC20 implementation
**/abstractcontractIncentivizedERC20isContext, IERC20, IERC20Detailed{
usingSafeMathforuint256;
mapping(address=>uint256) internal _balances;
mapping(address=>mapping(address=>uint256)) private _allowances;
uint256internal _totalSupply;
stringprivate _name;
stringprivate _symbol;
uint8private _decimals;
ILendingPool internal _pool;
addressinternal _underlyingAsset;
constructor(stringmemory name,
stringmemory symbol,
uint8 decimals
) {
_name = name;
_symbol = symbol;
_decimals = decimals;
}
/**
* @return The name of the token
**/functionname() publicviewoverridereturns (stringmemory) {
return _name;
}
/**
* @return The symbol of the token
**/functionsymbol() publicviewoverridereturns (stringmemory) {
return _symbol;
}
/**
* @return The decimals of the token
**/functiondecimals() publicviewoverridereturns (uint8) {
return _decimals;
}
/**
* @return The total supply of the token
**/functiontotalSupply() publicviewvirtualoverridereturns (uint256) {
return _totalSupply;
}
/**
* @return The balance of the token
**/functionbalanceOf(address account) publicviewvirtualoverridereturns (uint256) {
return _balances[account];
}
/**
* @return Abstract function implemented by the child aToken/debtToken.
* Done this way in order to not break compatibility with previous versions of aTokens/debtTokens
**/function_getIncentivesController() internalviewvirtualreturns(IAaveIncentivesController);
/**
* @dev Executes a transfer of tokens from _msgSender() to recipient
* @param recipient The recipient of the tokens
* @param amount The amount of tokens being transferred
* @return `true` if the transfer succeeds, `false` otherwise
**/functiontransfer(address recipient, uint256 amount) publicvirtualoverridereturns (bool) {
_transfer(_msgSender(), recipient, amount);
emit Transfer(_msgSender(), recipient, amount);
returntrue;
}
/**
* @dev Returns the allowance of spender on the tokens owned by owner
* @param owner The owner of the tokens
* @param spender The user allowed to spend the owner's tokens
* @return The amount of owner's tokens spender is allowed to spend
**/functionallowance(address owner, address spender)
publicviewvirtualoverridereturns (uint256)
{
return _allowances[owner][spender];
}
/**
* @dev Allows `spender` to spend the tokens owned by _msgSender()
* @param spender The user allowed to spend _msgSender() tokens
* @return `true`
**/functionapprove(address spender, uint256 amount) publicvirtualoverridereturns (bool) {
_approve(_msgSender(), spender, amount);
returntrue;
}
/**
* @dev Executes a transfer of token from sender to recipient, if _msgSender() is allowed to do so
* @param sender The owner of the tokens
* @param recipient The recipient of the tokens
* @param amount The amount of tokens being transferred
* @return `true` if the transfer succeeds, `false` otherwise
**/functiontransferFrom(address sender,
address recipient,
uint256 amount
) publicvirtualoverridereturns (bool) {
_transfer(sender, recipient, amount);
_approve(
sender,
_msgSender(),
_allowances[sender][_msgSender()].sub(amount, 'ERC20: transfer amount exceeds allowance')
);
emit Transfer(sender, recipient, amount);
returntrue;
}
/**
* @dev Increases the allowance of spender to spend _msgSender() tokens
* @param spender The user allowed to spend on behalf of _msgSender()
* @param addedValue The amount being added to the allowance
* @return `true`
**/functionincreaseAllowance(address spender, uint256 addedValue) publicvirtualreturns (bool) {
_approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));
returntrue;
}
/**
* @dev Decreases the allowance of spender to spend _msgSender() tokens
* @param spender The user allowed to spend on behalf of _msgSender()
* @param subtractedValue The amount being subtracted to the allowance
* @return `true`
**/functiondecreaseAllowance(address spender, uint256 subtractedValue)
publicvirtualreturns (bool)
{
_approve(
_msgSender(),
spender,
_allowances[_msgSender()][spender].sub(
subtractedValue,
'ERC20: decreased allowance below zero'
)
);
returntrue;
}
function_transfer(address sender,
address recipient,
uint256 amount
) internalvirtual{
require(sender !=address(0), 'ERC20: transfer from the zero address');
require(recipient !=address(0), 'ERC20: transfer to the zero address');
_beforeTokenTransfer(sender, recipient, amount);
uint256 senderBalance = _balances[sender].sub(amount, 'ERC20: transfer amount exceeds balance');
_balances[sender] = senderBalance;
uint256 recipientBalance = _balances[recipient].add(amount);
_balances[recipient] = recipientBalance;
if (address(_getIncentivesController()) !=address(0)) {
uint256 currentTotalSupply = _totalSupply;
_getIncentivesController().handleAction(sender, senderBalance, currentTotalSupply);
if (sender != recipient) {
_getIncentivesController().handleAction(recipient, recipientBalance, currentTotalSupply);
}
}
}
function_mint(address account, uint256 amount) internalvirtual{
require(account !=address(0), 'ERC20: mint to the zero address');
_beforeTokenTransfer(address(0), account, amount);
uint256 currentTotalSupply = _totalSupply.add(amount);
_totalSupply = currentTotalSupply;
uint256 accountBalance = _balances[account].add(amount);
_balances[account] = accountBalance;
if (address(_getIncentivesController()) !=address(0)) {
_getIncentivesController().handleAction(account, accountBalance, currentTotalSupply);
}
}
function_burn(address account, uint256 amount) internalvirtual{
require(account !=address(0), 'ERC20: burn from the zero address');
_beforeTokenTransfer(account, address(0), amount);
uint256 currentTotalSupply = _totalSupply.sub(amount);
_totalSupply = currentTotalSupply;
uint256 accountBalance = _balances[account].sub(amount, 'ERC20: burn amount exceeds balance');
_balances[account] = accountBalance;
if (address(_getIncentivesController()) !=address(0)) {
_getIncentivesController().handleAction(account, accountBalance, currentTotalSupply);
}
}
function_approve(address owner,
address spender,
uint256 amount
) internalvirtual{
require(owner !=address(0), 'ERC20: approve from the zero address');
require(spender !=address(0), 'ERC20: approve to the zero address');
_allowances[owner][spender] = amount;
emit Approval(owner, spender, amount);
}
function_setName(stringmemory newName) internal{
_name = newName;
}
function_setSymbol(stringmemory newSymbol) internal{
_symbol = newSymbol;
}
function_setDecimals(uint8 newDecimals) internal{
_decimals = newDecimals;
}
function_beforeTokenTransfer(addressfrom,
address to,
uint256 amount
) internalvirtual{}
functiongetAssetPrice() externalviewreturns (uint256) {
ILendingPoolAddressesProvider provider = _pool.getAddressesProvider();
address oracle = provider.getPriceOracle();
return IPriceOracle(oracle).getAssetPrice(_underlyingAsset);
}
}
Contract Source Code
File 51 of 91: InitializableAdminUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./BaseAdminUpgradeabilityProxy.sol';
import'./InitializableUpgradeabilityProxy.sol';
/**
* @title InitializableAdminUpgradeabilityProxy
* @dev Extends from BaseAdminUpgradeabilityProxy with an initializer for
* initializing the implementation, admin, and init data.
*/contractInitializableAdminUpgradeabilityProxyisBaseAdminUpgradeabilityProxy,
InitializableUpgradeabilityProxy{
/**
* Contract initializer.
* @param logic address of the initial implementation.
* @param admin Address of the proxy administrator.
* @param data Data to send as msg.data to the implementation to initialize the proxied contract.
* It should include the signature and the parameters of the function to be called, as described in
* https://solidity.readthedocs.io/en/v0.4.24/abi-spec.html#function-selector-and-argument-encoding.
* This parameter is optional, if no data is given the initialization call to proxied contract will be skipped.
*/functioninitialize(address logic,
address admin,
bytesmemory data
) publicpayable{
require(_implementation() ==address(0));
InitializableUpgradeabilityProxy.initialize(logic, data);
assert(ADMIN_SLOT ==bytes32(uint256(keccak256('eip1967.proxy.admin')) -1));
_setAdmin(admin);
}
/**
* @dev Only fall back when the sender is not the admin.
*/function_willFallback() internaloverride(BaseAdminUpgradeabilityProxy, Proxy) {
BaseAdminUpgradeabilityProxy._willFallback();
}
}
Contract Source Code
File 52 of 91: InitializableImmutableAdminUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./BaseImmutableAdminUpgradeabilityProxy.sol';
import'../../../dependencies/openzeppelin/upgradeability/InitializableUpgradeabilityProxy.sol';
/**
* @title InitializableAdminUpgradeabilityProxy
* @dev Extends BaseAdminUpgradeabilityProxy with an initializer function
*/contractInitializableImmutableAdminUpgradeabilityProxyisBaseImmutableAdminUpgradeabilityProxy,
InitializableUpgradeabilityProxy{
constructor(address admin) BaseImmutableAdminUpgradeabilityProxy(admin) {}
/**
* @dev Only fall back when the sender is not the admin.
*/function_willFallback() internaloverride(BaseImmutableAdminUpgradeabilityProxy, Proxy) {
BaseImmutableAdminUpgradeabilityProxy._willFallback();
}
}
Contract Source Code
File 53 of 91: InitializableUpgradeabilityProxy.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./BaseUpgradeabilityProxy.sol';
/**
* @title InitializableUpgradeabilityProxy
* @dev Extends BaseUpgradeabilityProxy with an initializer for initializing
* implementation and init data.
*/contractInitializableUpgradeabilityProxyisBaseUpgradeabilityProxy{
/**
* @dev Contract initializer.
* @param _logic Address of the initial implementation.
* @param _data Data to send as msg.data to the implementation to initialize the proxied contract.
* It should include the signature and the parameters of the function to be called, as described in
* https://solidity.readthedocs.io/en/v0.4.24/abi-spec.html#function-selector-and-argument-encoding.
* This parameter is optional, if no data is given the initialization call to proxied contract will be skipped.
*/functioninitialize(address _logic, bytesmemory _data) publicpayable{
require(_implementation() ==address(0));
assert(IMPLEMENTATION_SLOT ==bytes32(uint256(keccak256('eip1967.proxy.implementation')) -1));
_setImplementation(_logic);
if (_data.length>0) {
(bool success, ) = _logic.delegatecall(_data);
require(success);
}
}
}
Contract Source Code
File 54 of 91: LendingPool.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;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
**/contractLendingPoolisVersionedInitializable, ILendingPool, LendingPoolStorage{
usingSafeMathforuint256;
usingWadRayMathforuint256;
usingPercentageMathforuint256;
usingSafeERC20forIERC20;
usingReserveLogicforDataTypes.ReserveData;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
usingUserConfigurationforDataTypes.UserConfigurationMap;
uint256publicconstant LENDINGPOOL_REVISION =0x2;
modifierwhenNotPaused() {
_whenNotPaused();
_;
}
modifieronlyLendingPoolConfigurator() {
_onlyLendingPoolConfigurator();
_;
}
function_whenNotPaused() internalview{
require(!_paused, Errors.LP_IS_PAUSED);
}
function_onlyLendingPoolConfigurator() internalview{
require(
_addressesProvider.getLendingPoolConfigurator() ==msg.sender,
Errors.LP_CALLER_NOT_LENDING_POOL_CONFIGURATOR
);
}
functiongetRevision() internalpureoverridereturns (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
**/functioninitialize(ILendingPoolAddressesProvider provider) publicinitializer{
_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
**/functiondeposit(address asset,
uint256 amount,
address onBehalfOf,
uint16 referralCode
) externaloverridewhenNotPaused{
DataTypes.ReserveData storage reserve = _reserves[asset];
ValidationLogic.validateDeposit(reserve, 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
**/functionwithdraw(address asset,
uint256 amount,
address to
) externaloverridewhenNotPausedreturns (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,
_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
**/functionborrow(address asset,
uint256 amount,
uint256 interestRateMode,
uint16 referralCode,
address onBehalfOf
) externaloverridewhenNotPaused{
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
**/functionrepay(address asset,
uint256 amount,
uint256 rateMode,
address onBehalfOf
) externaloverridewhenNotPausedreturns (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;
}
/**
* @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
**/functionswapBorrowRateMode(address asset, uint256 rateMode) externaloverridewhenNotPaused{
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
**/functionrebalanceStableBorrowRate(address asset, address user) externaloverridewhenNotPaused{
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
**/functionsetUserUseReserveAsCollateral(address asset, bool useAsCollateral)
externaloverridewhenNotPaused{
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
**/functionliquidationCall(address collateralAsset,
address debtAsset,
address user,
uint256 debtToCover,
bool receiveAToken
) externaloverridewhenNotPaused{
address collateralManager = _addressesProvider.getLendingPoolCollateralManager();
//solium-disable-next-line
(bool success, bytesmemory 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, stringmemory returnMessage) =abi.decode(result, (uint256, string));
require(returnCode ==0, string(abi.encodePacked(returnMessage)));
}
structFlashLoanLocalVars {
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 modes Types of the debt to open if the flash loan is not returned:
* 0 -> Don't open any debt, just revert if funds can't be transferred from the receiver
* 1 -> Open debt at stable rate for the value of the amount flash-borrowed to the `onBehalfOf` address
* 2 -> Open debt at variable rate for the value of the amount flash-borrowed to the `onBehalfOf` address
* @param onBehalfOf The address that will receive the debt in the case of using on `modes` 1 or 2
* @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
**/functionflashLoan(address receiverAddress,
address[] calldata assets,
uint256[] calldata amounts,
uint256[] calldata modes,
address onBehalfOf,
bytescalldata params,
uint16 referralCode
) externaloverridewhenNotPaused{
// 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);// if (DataTypes.InterestRateMode(modes[vars.i]) == DataTypes.InterestRateMode.NONE) {// _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// );// } else {// // If the user chose to not return the funds, the system checks if there is enough collateral and// // eventually opens a debt position// _executeBorrow(// ExecuteBorrowParams(// vars.currentAsset,// msg.sender,// onBehalfOf,// vars.currentAmount,// modes[vars.i],// vars.currentATokenAddress,// referralCode,// false// )// );// }// 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
**/functiongetReserveData(address asset)
externalviewoverridereturns (DataTypes.ReserveData memory)
{
return _reserves[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
**/functiongetUserAccountData(address user)
externalviewoverridereturns (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
**/functiongetConfiguration(address asset)
externalviewoverridereturns (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
**/functiongetUserConfiguration(address user)
externalviewoverridereturns (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
*/functiongetReserveNormalizedIncome(address asset)
externalviewvirtualoverridereturns (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
*/functiongetReserveNormalizedVariableDebt(address asset)
externalviewoverridereturns (uint256)
{
return _reserves[asset].getNormalizedDebt();
}
/**
* @dev Returns if the LendingPool is paused
*/functionpaused() externalviewoverridereturns (bool) {
return _paused;
}
/**
* @dev Returns the list of the initialized reserves
**/functiongetReservesList() externalviewoverridereturns (address[] memory) {
address[] memory _activeReserves =newaddress[](_reservesCount);
for (uint256 i =0; i < _reservesCount; i++) {
_activeReserves[i] = _reservesList[i];
}
return _activeReserves;
}
/**
* @dev Returns the cached LendingPoolAddressesProvider connected to this contract
**/functiongetAddressesProvider() externalviewoverridereturns (ILendingPoolAddressesProvider) {
return _addressesProvider;
}
/**
* @dev Returns the percentage of available liquidity that can be borrowed at once at stable rate
*/functionMAX_STABLE_RATE_BORROW_SIZE_PERCENT() publicviewreturns (uint256) {
return _maxStableRateBorrowSizePercent;
}
/**
* @dev Returns the fee on flash loans
*/functionFLASHLOAN_PREMIUM_TOTAL() publicviewreturns (uint256) {
return _flashLoanPremiumTotal;
}
/**
* @dev Returns the maximum number of reserves supported to be listed in this LendingPool
*/functionMAX_NUMBER_RESERVES() publicviewreturns (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
*/functionfinalizeTransfer(address asset,
addressfrom,
address to,
uint256 amount,
uint256 balanceFromBefore,
uint256 balanceToBefore
) externaloverridewhenNotPaused{
require(msg.sender== _reserves[asset].aTokenAddress, Errors.LP_CALLER_MUST_BE_AN_ATOKEN);
ValidationLogic.validateTransfer(
from,
_reserves,
_usersConfig[from],
_reservesList,
_reservesCount,
_addressesProvider.getPriceOracle()
);
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 aTokenAddress The address of the VariableDebtToken that will be assigned to the reserve
* @param interestRateStrategyAddress The address of the interest rate strategy contract
**/functioninitReserve(address asset,
address aTokenAddress,
address stableDebtAddress,
address variableDebtAddress,
address interestRateStrategyAddress
) externaloverrideonlyLendingPoolConfigurator{
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
**/functionsetReserveInterestRateStrategyAddress(address asset, address rateStrategyAddress)
externaloverrideonlyLendingPoolConfigurator{
_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
**/functionsetConfiguration(address asset, uint256 configuration)
externaloverrideonlyLendingPoolConfigurator{
_reserves[asset].configuration.data = configuration;
}
/**
* @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
*/functionsetPause(bool val) externaloverrideonlyLendingPoolConfigurator{
_paused = val;
if (_paused) {
emit Paused();
} else {
emit Unpaused();
}
}
structExecuteBorrowParams {
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 amountInETH =
IPriceOracleGetter(oracle).updateAssetPrice(vars.asset).mul(vars.amount).div(
10**reserve.configuration.getDecimals()
);
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;
}
}
}
Contract Source Code
File 55 of 91: LendingPoolAddressesProvider.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {Ownable} from'../../dependencies/openzeppelin/contracts/Ownable.sol';
// Prettier ignore to prevent buidler flatter bug// prettier-ignoreimport {InitializableImmutableAdminUpgradeabilityProxy} from'../libraries/aave-upgradeability/InitializableImmutableAdminUpgradeabilityProxy.sol';
import {ILendingPoolAddressesProvider} from'../../interfaces/ILendingPoolAddressesProvider.sol';
/**
* @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
**/contractLendingPoolAddressesProviderisOwnable, ILendingPoolAddressesProvider{
stringprivate _marketId;
mapping(bytes32=>address) private _addresses;
bytes32privateconstant LENDING_POOL ='LENDING_POOL';
bytes32privateconstant LENDING_POOL_CONFIGURATOR ='LENDING_POOL_CONFIGURATOR';
bytes32privateconstant POOL_ADMIN ='POOL_ADMIN';
bytes32privateconstant EMERGENCY_ADMIN ='EMERGENCY_ADMIN';
bytes32privateconstant LENDING_POOL_COLLATERAL_MANAGER ='COLLATERAL_MANAGER';
bytes32privateconstant PRICE_ORACLE ='PRICE_ORACLE';
bytes32privateconstant LENDING_RATE_ORACLE ='LENDING_RATE_ORACLE';
constructor(stringmemory marketId) {
_setMarketId(marketId);
}
/**
* @dev Returns the id of the Aave market to which this contracts points to
* @return The market id
**/functiongetMarketId() externalviewoverridereturns (stringmemory) {
return _marketId;
}
/**
* @dev Allows to set the market which this LendingPoolAddressesProvider represents
* @param marketId The market id
*/functionsetMarketId(stringmemory marketId) externaloverrideonlyOwner{
_setMarketId(marketId);
}
/**
* @dev General function to update the implementation of a proxy registered with
* certain `id`. If there is no proxy registered, it will instantiate one and
* set as implementation the `implementationAddress`
* IMPORTANT Use this function carefully, only for ids that don't have an explicit
* setter function, in order to avoid unexpected consequences
* @param id The id
* @param implementationAddress The address of the new implementation
*/functionsetAddressAsProxy(bytes32 id, address implementationAddress)
externaloverrideonlyOwner{
_updateImpl(id, implementationAddress);
emit AddressSet(id, implementationAddress, true);
}
/**
* @dev Sets an address for an id replacing the address saved in the addresses map
* IMPORTANT Use this function carefully, as it will do a hard replacement
* @param id The id
* @param newAddress The address to set
*/functionsetAddress(bytes32 id, address newAddress) externaloverrideonlyOwner{
_addresses[id] = newAddress;
emit AddressSet(id, newAddress, false);
}
/**
* @dev Returns an address by id
* @return The address
*/functiongetAddress(bytes32 id) publicviewoverridereturns (address) {
return _addresses[id];
}
/**
* @dev Returns the address of the LendingPool proxy
* @return The LendingPool proxy address
**/functiongetLendingPool() externalviewoverridereturns (address) {
return getAddress(LENDING_POOL);
}
/**
* @dev Updates the implementation of the LendingPool, or creates the proxy
* setting the new `pool` implementation on the first time calling it
* @param pool The new LendingPool implementation
**/functionsetLendingPoolImpl(address pool) externaloverrideonlyOwner{
_updateImpl(LENDING_POOL, pool);
emit LendingPoolUpdated(pool);
}
/**
* @dev Returns the address of the LendingPoolConfigurator proxy
* @return The LendingPoolConfigurator proxy address
**/functiongetLendingPoolConfigurator() externalviewoverridereturns (address) {
return getAddress(LENDING_POOL_CONFIGURATOR);
}
/**
* @dev Updates the implementation of the LendingPoolConfigurator, or creates the proxy
* setting the new `configurator` implementation on the first time calling it
* @param configurator The new LendingPoolConfigurator implementation
**/functionsetLendingPoolConfiguratorImpl(address configurator) externaloverrideonlyOwner{
_updateImpl(LENDING_POOL_CONFIGURATOR, configurator);
emit LendingPoolConfiguratorUpdated(configurator);
}
/**
* @dev Returns the address of the LendingPoolCollateralManager. Since the manager is used
* through delegateCall within the LendingPool contract, the proxy contract pattern does not work properly hence
* the addresses are changed directly
* @return The address of the LendingPoolCollateralManager
**/functiongetLendingPoolCollateralManager() externalviewoverridereturns (address) {
return getAddress(LENDING_POOL_COLLATERAL_MANAGER);
}
/**
* @dev Updates the address of the LendingPoolCollateralManager
* @param manager The new LendingPoolCollateralManager address
**/functionsetLendingPoolCollateralManager(address manager) externaloverrideonlyOwner{
_addresses[LENDING_POOL_COLLATERAL_MANAGER] = manager;
emit LendingPoolCollateralManagerUpdated(manager);
}
/**
* @dev The functions below are getters/setters of addresses that are outside the context
* of the protocol hence the upgradable proxy pattern is not used
**/functiongetPoolAdmin() externalviewoverridereturns (address) {
return getAddress(POOL_ADMIN);
}
functionsetPoolAdmin(address admin) externaloverrideonlyOwner{
_addresses[POOL_ADMIN] = admin;
emit ConfigurationAdminUpdated(admin);
}
functiongetEmergencyAdmin() externalviewoverridereturns (address) {
return getAddress(EMERGENCY_ADMIN);
}
functionsetEmergencyAdmin(address emergencyAdmin) externaloverrideonlyOwner{
_addresses[EMERGENCY_ADMIN] = emergencyAdmin;
emit EmergencyAdminUpdated(emergencyAdmin);
}
functiongetPriceOracle() externalviewoverridereturns (address) {
return getAddress(PRICE_ORACLE);
}
functionsetPriceOracle(address priceOracle) externaloverrideonlyOwner{
_addresses[PRICE_ORACLE] = priceOracle;
emit PriceOracleUpdated(priceOracle);
}
functiongetLendingRateOracle() externalviewoverridereturns (address) {
return getAddress(LENDING_RATE_ORACLE);
}
functionsetLendingRateOracle(address lendingRateOracle) externaloverrideonlyOwner{
_addresses[LENDING_RATE_ORACLE] = lendingRateOracle;
emit LendingRateOracleUpdated(lendingRateOracle);
}
/**
* @dev Internal function to update the implementation of a specific proxied component of the protocol
* - If there is no proxy registered in the given `id`, it creates the proxy setting `newAdress`
* as implementation and calls the initialize() function on the proxy
* - If there is already a proxy registered, it just updates the implementation to `newAddress` and
* calls the initialize() function via upgradeToAndCall() in the proxy
* @param id The id of the proxy to be updated
* @param newAddress The address of the new implementation
**/function_updateImpl(bytes32 id, address newAddress) internal{
addresspayable proxyAddress =payable(_addresses[id]);
InitializableImmutableAdminUpgradeabilityProxy proxy =
InitializableImmutableAdminUpgradeabilityProxy(proxyAddress);
bytesmemory params =abi.encodeWithSignature('initialize(address)', address(this));
if (proxyAddress ==address(0)) {
proxy =new InitializableImmutableAdminUpgradeabilityProxy(address(this));
proxy.initialize(newAddress, params);
_addresses[id] =address(proxy);
emit ProxyCreated(id, address(proxy));
} else {
proxy.upgradeToAndCall(newAddress, params);
}
}
function_setMarketId(stringmemory marketId) internal{
_marketId = marketId;
emit MarketIdSet(marketId);
}
}
Contract Source Code
File 56 of 91: LendingPoolAddressesProviderRegistry.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {Ownable} from'../../dependencies/openzeppelin/contracts/Ownable.sol';
import {
ILendingPoolAddressesProviderRegistry
} from'../../interfaces/ILendingPoolAddressesProviderRegistry.sol';
import {Errors} from'../libraries/helpers/Errors.sol';
/**
* @title LendingPoolAddressesProviderRegistry contract
* @dev Main registry of LendingPoolAddressesProvider of multiple Aave protocol's markets
* - Used for indexing purposes of Aave protocol's markets
* - The id assigned to a LendingPoolAddressesProvider refers to the market it is connected with,
* for example with `0` for the Aave main market and `1` for the next created
* @author Aave
**/contractLendingPoolAddressesProviderRegistryisOwnable, ILendingPoolAddressesProviderRegistry{
mapping(address=>uint256) private _addressesProviders;
address[] private _addressesProvidersList;
/**
* @dev Returns the list of registered addresses provider
* @return The list of addresses provider, potentially containing address(0) elements
**/functiongetAddressesProvidersList() externalviewoverridereturns (address[] memory) {
address[] memory addressesProvidersList = _addressesProvidersList;
uint256 maxLength = addressesProvidersList.length;
address[] memory activeProviders =newaddress[](maxLength);
for (uint256 i =0; i < maxLength; i++) {
if (_addressesProviders[addressesProvidersList[i]] >0) {
activeProviders[i] = addressesProvidersList[i];
}
}
return activeProviders;
}
/**
* @dev Registers an addresses provider
* @param provider The address of the new LendingPoolAddressesProvider
* @param id The id for the new LendingPoolAddressesProvider, referring to the market it belongs to
**/functionregisterAddressesProvider(address provider, uint256 id) externaloverrideonlyOwner{
require(id !=0, Errors.LPAPR_INVALID_ADDRESSES_PROVIDER_ID);
_addressesProviders[provider] = id;
_addToAddressesProvidersList(provider);
emit AddressesProviderRegistered(provider);
}
/**
* @dev Removes a LendingPoolAddressesProvider from the list of registered addresses provider
* @param provider The LendingPoolAddressesProvider address
**/functionunregisterAddressesProvider(address provider) externaloverrideonlyOwner{
require(_addressesProviders[provider] >0, Errors.LPAPR_PROVIDER_NOT_REGISTERED);
_addressesProviders[provider] =0;
emit AddressesProviderUnregistered(provider);
}
/**
* @dev Returns the id on a registered LendingPoolAddressesProvider
* @return The id or 0 if the LendingPoolAddressesProvider is not registered
*/functiongetAddressesProviderIdByAddress(address addressesProvider)
externalviewoverridereturns (uint256)
{
return _addressesProviders[addressesProvider];
}
function_addToAddressesProvidersList(address provider) internal{
uint256 providersCount = _addressesProvidersList.length;
for (uint256 i =0; i < providersCount; i++) {
if (_addressesProvidersList[i] == provider) {
return;
}
}
_addressesProvidersList.push(provider);
}
}
Contract Source Code
File 57 of 91: LendingPoolCollateralManager.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {SafeMath} from'../../dependencies/openzeppelin/contracts//SafeMath.sol';
import {IERC20} from'../../dependencies/openzeppelin/contracts//IERC20.sol';
import {IAToken} from'../../interfaces/IAToken.sol';
import {IStableDebtToken} from'../../interfaces/IStableDebtToken.sol';
import {IVariableDebtToken} from'../../interfaces/IVariableDebtToken.sol';
import {IPriceOracleGetter} from'../../interfaces/IPriceOracleGetter.sol';
import {ILendingPoolCollateralManager} from'../../interfaces/ILendingPoolCollateralManager.sol';
import {VersionedInitializable} from'../libraries/aave-upgradeability/VersionedInitializable.sol';
import {GenericLogic} from'../libraries/logic/GenericLogic.sol';
import {Helpers} from'../libraries/helpers/Helpers.sol';
import {WadRayMath} from'../libraries/math/WadRayMath.sol';
import {PercentageMath} from'../libraries/math/PercentageMath.sol';
import {SafeERC20} from'../../dependencies/openzeppelin/contracts/SafeERC20.sol';
import {Errors} from'../libraries/helpers/Errors.sol';
import {ValidationLogic} from'../libraries/logic/ValidationLogic.sol';
import {DataTypes} from'../libraries/types/DataTypes.sol';
import {LendingPoolStorage} from'./LendingPoolStorage.sol';
import {UserConfiguration} from'../libraries/configuration/UserConfiguration.sol';
import {ReserveConfiguration} from'../libraries/configuration/ReserveConfiguration.sol';
import {ReserveLogic} from'../libraries/logic/ReserveLogic.sol';
/**
* @title LendingPoolCollateralManager contract
* @author Aave
* @dev Implements actions involving management of collateral in the protocol, the main one being the liquidations
* IMPORTANT This contract will run always via DELEGATECALL, through the LendingPool, so the chain of inheritance
* is the same as the LendingPool, to have compatible storage layouts
**/contractLendingPoolCollateralManagerisILendingPoolCollateralManager,
VersionedInitializable,
LendingPoolStorage{
usingSafeERC20forIERC20;
usingSafeMathforuint256;
usingWadRayMathforuint256;
usingPercentageMathforuint256;
usingReserveLogicforDataTypes.ReserveData;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
usingUserConfigurationforDataTypes.UserConfigurationMap;
uint256internalconstant LIQUIDATION_CLOSE_FACTOR_PERCENT =5000;
structLiquidationCallLocalVars {
uint256 userCollateralBalance;
uint256 userStableDebt;
uint256 userVariableDebt;
uint256 maxLiquidatableDebt;
uint256 actualDebtToLiquidate;
uint256 liquidationRatio;
uint256 maxAmountCollateralToLiquidate;
uint256 userStableRate;
uint256 maxCollateralToLiquidate;
uint256 debtAmountNeeded;
uint256 healthFactor;
uint256 liquidatorPreviousATokenBalance;
IAToken collateralAtoken;
bool isCollateralEnabled;
DataTypes.InterestRateMode borrowRateMode;
uint256 errorCode;
string errorMsg;
}
/**
* @dev As thIS contract extends the VersionedInitializable contract to match the state
* of the LendingPool contract, the getRevision() function is needed, but the value is not
* important, as the initialize() function will never be called here
*/functiongetRevision() internalpureoverridereturns (uint256) {
return0;
}
/**
* @dev Function to liquidate a position if its Health Factor drops 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
**/functionliquidationCall(address collateralAsset,
address debtAsset,
address user,
uint256 debtToCover,
bool receiveAToken
) externaloverridereturns (uint256, stringmemory) {
DataTypes.ReserveData storage collateralReserve = _reserves[collateralAsset];
DataTypes.ReserveData storage debtReserve = _reserves[debtAsset];
DataTypes.UserConfigurationMap storage userConfig = _usersConfig[user];
LiquidationCallLocalVars memory vars;
(, , , , vars.healthFactor) = GenericLogic.calculateUserAccountData(
user,
_reserves,
userConfig,
_reservesList,
_reservesCount,
_addressesProvider.getPriceOracle()
);
(vars.userStableDebt, vars.userVariableDebt) = Helpers.getUserCurrentDebt(user, debtReserve);
(vars.errorCode, vars.errorMsg) = ValidationLogic.validateLiquidationCall(
collateralReserve,
debtReserve,
userConfig,
vars.healthFactor,
vars.userStableDebt,
vars.userVariableDebt
);
if (Errors.CollateralManagerErrors(vars.errorCode) != Errors.CollateralManagerErrors.NO_ERROR) {
return (vars.errorCode, vars.errorMsg);
}
vars.collateralAtoken = IAToken(collateralReserve.aTokenAddress);
vars.userCollateralBalance = vars.collateralAtoken.balanceOf(user);
vars.maxLiquidatableDebt = vars.userStableDebt.add(vars.userVariableDebt).percentMul(
LIQUIDATION_CLOSE_FACTOR_PERCENT
);
vars.actualDebtToLiquidate = debtToCover > vars.maxLiquidatableDebt
? vars.maxLiquidatableDebt
: debtToCover;
(
vars.maxCollateralToLiquidate,
vars.debtAmountNeeded
) = _calculateAvailableCollateralToLiquidate(
collateralReserve,
debtReserve,
collateralAsset,
debtAsset,
vars.actualDebtToLiquidate,
vars.userCollateralBalance
);
// If debtAmountNeeded < actualDebtToLiquidate, there isn't enough// collateral to cover the actual amount that is being liquidated, hence we liquidate// a smaller amountif (vars.debtAmountNeeded < vars.actualDebtToLiquidate) {
vars.actualDebtToLiquidate = vars.debtAmountNeeded;
}
// If the liquidator reclaims the underlying asset, we make sure there is enough available liquidity in the// collateral reserveif (!receiveAToken) {
uint256 currentAvailableCollateral =
IERC20(collateralAsset).balanceOf(address(vars.collateralAtoken));
if (currentAvailableCollateral < vars.maxCollateralToLiquidate) {
return (
uint256(Errors.CollateralManagerErrors.NOT_ENOUGH_LIQUIDITY),
Errors.LPCM_NOT_ENOUGH_LIQUIDITY_TO_LIQUIDATE
);
}
}
debtReserve.updateState();
if (vars.userVariableDebt >= vars.actualDebtToLiquidate) {
IVariableDebtToken(debtReserve.variableDebtTokenAddress).burn(
user,
vars.actualDebtToLiquidate,
debtReserve.variableBorrowIndex
);
} else {
// If the user doesn't have variable debt, no need to try to burn variable debt tokensif (vars.userVariableDebt >0) {
IVariableDebtToken(debtReserve.variableDebtTokenAddress).burn(
user,
vars.userVariableDebt,
debtReserve.variableBorrowIndex
);
}
IStableDebtToken(debtReserve.stableDebtTokenAddress).burn(
user,
vars.actualDebtToLiquidate.sub(vars.userVariableDebt)
);
}
debtReserve.updateInterestRates(
debtAsset,
debtReserve.aTokenAddress,
vars.actualDebtToLiquidate,
0
);
if (receiveAToken) {
vars.liquidatorPreviousATokenBalance = IERC20(vars.collateralAtoken).balanceOf(msg.sender);
vars.collateralAtoken.transferOnLiquidation(user, msg.sender, vars.maxCollateralToLiquidate);
if (vars.liquidatorPreviousATokenBalance ==0) {
DataTypes.UserConfigurationMap storage liquidatorConfig = _usersConfig[msg.sender];
liquidatorConfig.setUsingAsCollateral(collateralReserve.id, true);
emit ReserveUsedAsCollateralEnabled(collateralAsset, msg.sender);
}
} else {
collateralReserve.updateState();
collateralReserve.updateInterestRates(
collateralAsset,
address(vars.collateralAtoken),
0,
vars.maxCollateralToLiquidate
);
// Burn the equivalent amount of aToken, sending the underlying to the liquidator
vars.collateralAtoken.burn(
user,
msg.sender,
vars.maxCollateralToLiquidate,
collateralReserve.liquidityIndex
);
}
// If the collateral being liquidated is equal to the user balance,// we set the currency as not being used as collateral anymoreif (vars.maxCollateralToLiquidate == vars.userCollateralBalance) {
userConfig.setUsingAsCollateral(collateralReserve.id, false);
emit ReserveUsedAsCollateralDisabled(collateralAsset, user);
}
// Transfers the debt asset being repaid to the aToken, where the liquidity is kept
IERC20(debtAsset).safeTransferFrom(
msg.sender,
debtReserve.aTokenAddress,
vars.actualDebtToLiquidate
);
emit LiquidationCall(
collateralAsset,
debtAsset,
user,
vars.actualDebtToLiquidate,
vars.maxCollateralToLiquidate,
msg.sender,
receiveAToken
);
return (uint256(Errors.CollateralManagerErrors.NO_ERROR), Errors.LPCM_NO_ERRORS);
}
structAvailableCollateralToLiquidateLocalVars {
uint256 userCompoundedBorrowBalance;
uint256 liquidationBonus;
uint256 collateralPrice;
uint256 debtAssetPrice;
uint256 maxAmountCollateralToLiquidate;
uint256 debtAssetDecimals;
uint256 collateralDecimals;
}
/**
* @dev Calculates how much of a specific collateral can be liquidated, given
* a certain amount of debt asset.
* - This function needs to be called after all the checks to validate the liquidation have been performed,
* otherwise it might fail.
* @param collateralReserve The data of the collateral reserve
* @param debtReserve The data of the debt reserve
* @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 debtToCover The debt amount of borrowed `asset` the liquidator wants to cover
* @param userCollateralBalance The collateral balance for the specific `collateralAsset` of the user being liquidated
* @return collateralAmount: The maximum amount that is possible to liquidate given all the liquidation constraints
* (user balance, close factor)
* debtAmountNeeded: The amount to repay with the liquidation
**/function_calculateAvailableCollateralToLiquidate(
DataTypes.ReserveData storage collateralReserve,
DataTypes.ReserveData storage debtReserve,
address collateralAsset,
address debtAsset,
uint256 debtToCover,
uint256 userCollateralBalance
) internalreturns (uint256, uint256) {
uint256 collateralAmount =0;
uint256 debtAmountNeeded =0;
IPriceOracleGetter oracle = IPriceOracleGetter(_addressesProvider.getPriceOracle());
AvailableCollateralToLiquidateLocalVars memory vars;
vars.collateralPrice = oracle.updateAssetPrice(collateralAsset);
vars.debtAssetPrice = oracle.updateAssetPrice(debtAsset);
(, , vars.liquidationBonus, vars.collateralDecimals, ) = collateralReserve
.configuration
.getParams();
vars.debtAssetDecimals = debtReserve.configuration.getDecimals();
// This is the maximum possible amount of the selected collateral that can be liquidated, given the// max amount of liquidatable debt
vars.maxAmountCollateralToLiquidate = vars
.debtAssetPrice
.mul(debtToCover)
.mul(10**vars.collateralDecimals)
.percentMul(vars.liquidationBonus)
.div(vars.collateralPrice.mul(10**vars.debtAssetDecimals));
if (vars.maxAmountCollateralToLiquidate > userCollateralBalance) {
collateralAmount = userCollateralBalance;
debtAmountNeeded = vars
.collateralPrice
.mul(collateralAmount)
.mul(10**vars.debtAssetDecimals)
.div(vars.debtAssetPrice.mul(10**vars.collateralDecimals))
.percentDiv(vars.liquidationBonus);
} else {
collateralAmount = vars.maxAmountCollateralToLiquidate;
debtAmountNeeded = debtToCover;
}
return (collateralAmount, debtAmountNeeded);
}
}
Contract Source Code
File 58 of 91: LendingPoolConfigurator.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;import {SafeMath} from'../../dependencies/openzeppelin/contracts/SafeMath.sol';
import {VersionedInitializable} from'../libraries/aave-upgradeability/VersionedInitializable.sol';
import {
InitializableImmutableAdminUpgradeabilityProxy
} from'../libraries/aave-upgradeability/InitializableImmutableAdminUpgradeabilityProxy.sol';
import {ReserveConfiguration} from'../libraries/configuration/ReserveConfiguration.sol';
import {ILendingPoolAddressesProvider} from'../../interfaces/ILendingPoolAddressesProvider.sol';
import {ILendingPool} from'../../interfaces/ILendingPool.sol';
import {IERC20Detailed} from'../../dependencies/openzeppelin/contracts/IERC20Detailed.sol';
import {Errors} from'../libraries/helpers/Errors.sol';
import {PercentageMath} from'../libraries/math/PercentageMath.sol';
import {DataTypes} from'../libraries/types/DataTypes.sol';
import {IInitializableDebtToken} from'../../interfaces/IInitializableDebtToken.sol';
import {IInitializableAToken} from'../../interfaces/IInitializableAToken.sol';
import {IChefIncentivesController} from'../../interfaces/IChefIncentivesController.sol';
import {ILendingPoolConfigurator} from'../../interfaces/ILendingPoolConfigurator.sol';
import {IMultiFeeDistribution} from'../../interfaces/IMultiFeeDistribution.sol';
/**
* @title LendingPoolConfigurator contract
* @author Aave
* @dev Implements the configuration methods for the Aave protocol
**/contractLendingPoolConfiguratorisVersionedInitializable, ILendingPoolConfigurator{
usingSafeMathforuint256;
usingPercentageMathforuint256;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
ILendingPoolAddressesProvider internal addressesProvider;
ILendingPool internal pool;
modifieronlyPoolAdmin{
require(addressesProvider.getPoolAdmin() ==msg.sender, Errors.CALLER_NOT_POOL_ADMIN);
_;
}
modifieronlyEmergencyAdmin{
require(
addressesProvider.getEmergencyAdmin() ==msg.sender,
Errors.LPC_CALLER_NOT_EMERGENCY_ADMIN
);
_;
}
uint256internalconstant CONFIGURATOR_REVISION =0x1;
functiongetRevision() internalpureoverridereturns (uint256) {
return CONFIGURATOR_REVISION;
}
functioninitialize(ILendingPoolAddressesProvider provider) publicinitializer{
addressesProvider = provider;
pool = ILendingPool(addressesProvider.getLendingPool());
}
/**
* @dev Initializes reserves in batch
**/functionbatchInitReserve(InitReserveInput[] calldata input) externalonlyPoolAdmin{
ILendingPool cachedPool = pool;
for (uint256 i =0; i < input.length; i++) {
_initReserve(cachedPool, input[i]);
}
}
function_initReserve(ILendingPool pool, InitReserveInput calldata input) internal{
IChefIncentivesController incentivesController = IChefIncentivesController(input.incentivesController);
address aTokenProxyAddress =
_initTokenWithProxy(
input.aTokenImpl,
abi.encodeWithSelector(
IInitializableAToken.initialize.selector,
pool,
input.treasury,
input.underlyingAsset,
incentivesController,
input.underlyingAssetDecimals,
input.aTokenName,
input.aTokenSymbol,
input.params
)
);
incentivesController.addPool(aTokenProxyAddress, input.allocPoint);
IMultiFeeDistribution(input.treasury).addReward(aTokenProxyAddress);
address stableDebtTokenProxyAddress =
_initTokenWithProxy(
input.stableDebtTokenImpl,
abi.encodeWithSelector(
IInitializableDebtToken.initialize.selector,
pool,
input.underlyingAsset,
IChefIncentivesController(input.incentivesController),
input.underlyingAssetDecimals,
input.stableDebtTokenName,
input.stableDebtTokenSymbol,
input.params
)
);
// stableDebt is not added to incentives controller// POLTER does not support stable lendingaddress variableDebtTokenProxyAddress =
_initTokenWithProxy(
input.variableDebtTokenImpl,
abi.encodeWithSelector(
IInitializableDebtToken.initialize.selector,
pool,
input.underlyingAsset,
IChefIncentivesController(input.incentivesController),
input.underlyingAssetDecimals,
input.variableDebtTokenName,
input.variableDebtTokenSymbol,
input.params
)
);
incentivesController.addPool(variableDebtTokenProxyAddress, input.allocPoint);
pool.initReserve(
input.underlyingAsset,
aTokenProxyAddress,
stableDebtTokenProxyAddress,
variableDebtTokenProxyAddress,
input.interestRateStrategyAddress
);
DataTypes.ReserveConfigurationMap memory currentConfig =
pool.getConfiguration(input.underlyingAsset);
currentConfig.setDecimals(input.underlyingAssetDecimals);
currentConfig.setActive(true);
currentConfig.setFrozen(false);
pool.setConfiguration(input.underlyingAsset, currentConfig.data);
emit ReserveInitialized(
input.underlyingAsset,
aTokenProxyAddress,
stableDebtTokenProxyAddress,
variableDebtTokenProxyAddress,
input.interestRateStrategyAddress
);
}
/**
* @dev Updates the aToken implementation for the reserve
**/functionupdateAToken(UpdateATokenInput calldata input) externalonlyPoolAdmin{
ILendingPool cachedPool = pool;
DataTypes.ReserveData memory reserveData = cachedPool.getReserveData(input.asset);
(, , , uint256 decimals, ) = cachedPool.getConfiguration(input.asset).getParamsMemory();
bytesmemory encodedCall =abi.encodeWithSelector(
IInitializableAToken.initialize.selector,
cachedPool,
input.treasury,
input.asset,
input.incentivesController,
decimals,
input.name,
input.symbol,
input.params
);
_upgradeTokenImplementation(
reserveData.aTokenAddress,
input.implementation,
encodedCall
);
emit ATokenUpgraded(input.asset, reserveData.aTokenAddress, input.implementation);
}
/**
* @dev Updates the stable debt token implementation for the reserve
**/functionupdateStableDebtToken(UpdateDebtTokenInput calldata input) externalonlyPoolAdmin{
ILendingPool cachedPool = pool;
DataTypes.ReserveData memory reserveData = cachedPool.getReserveData(input.asset);
(, , , uint256 decimals, ) = cachedPool.getConfiguration(input.asset).getParamsMemory();
bytesmemory encodedCall =abi.encodeWithSelector(
IInitializableDebtToken.initialize.selector,
cachedPool,
input.asset,
input.incentivesController,
decimals,
input.name,
input.symbol,
input.params
);
_upgradeTokenImplementation(
reserveData.stableDebtTokenAddress,
input.implementation,
encodedCall
);
emit StableDebtTokenUpgraded(
input.asset,
reserveData.stableDebtTokenAddress,
input.implementation
);
}
/**
* @dev Updates the variable debt token implementation for the asset
**/functionupdateVariableDebtToken(UpdateDebtTokenInput calldata input)
externalonlyPoolAdmin{
ILendingPool cachedPool = pool;
DataTypes.ReserveData memory reserveData = cachedPool.getReserveData(input.asset);
(, , , uint256 decimals, ) = cachedPool.getConfiguration(input.asset).getParamsMemory();
bytesmemory encodedCall =abi.encodeWithSelector(
IInitializableDebtToken.initialize.selector,
cachedPool,
input.asset,
input.incentivesController,
decimals,
input.name,
input.symbol,
input.params
);
_upgradeTokenImplementation(
reserveData.variableDebtTokenAddress,
input.implementation,
encodedCall
);
emit VariableDebtTokenUpgraded(
input.asset,
reserveData.variableDebtTokenAddress,
input.implementation
);
}
/**
* @dev Enables borrowing on a reserve
* @param asset The address of the underlying asset of the reserve
* @param stableBorrowRateEnabled True if stable borrow rate needs to be enabled by default on this reserve
**/functionenableBorrowingOnReserve(address asset, bool stableBorrowRateEnabled)
externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setBorrowingEnabled(true);
currentConfig.setStableRateBorrowingEnabled(stableBorrowRateEnabled);
pool.setConfiguration(asset, currentConfig.data);
emit BorrowingEnabledOnReserve(asset, stableBorrowRateEnabled);
}
/**
* @dev Disables borrowing on a reserve
* @param asset The address of the underlying asset of the reserve
**/functiondisableBorrowingOnReserve(address asset) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setBorrowingEnabled(false);
pool.setConfiguration(asset, currentConfig.data);
emit BorrowingDisabledOnReserve(asset);
}
/**
* @dev Configures the reserve collateralization parameters
* all the values are expressed in percentages with two decimals of precision. A valid value is 10000, which means 100.00%
* @param asset The address of the underlying asset of the reserve
* @param ltv The loan to value of the asset when used as collateral
* @param liquidationThreshold The threshold at which loans using this asset as collateral will be considered undercollateralized
* @param liquidationBonus The bonus liquidators receive to liquidate this asset. The values is always above 100%. A value of 105%
* means the liquidator will receive a 5% bonus
**/functionconfigureReserveAsCollateral(address asset,
uint256 ltv,
uint256 liquidationThreshold,
uint256 liquidationBonus
) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
//validation of the parameters: the LTV can//only be lower or equal than the liquidation threshold//(otherwise a loan against the asset would cause instantaneous liquidation)require(ltv <= liquidationThreshold, Errors.LPC_INVALID_CONFIGURATION);
if (liquidationThreshold !=0) {
//liquidation bonus must be bigger than 100.00%, otherwise the liquidator would receive less//collateral than needed to cover the debtrequire(
liquidationBonus > PercentageMath.PERCENTAGE_FACTOR,
Errors.LPC_INVALID_CONFIGURATION
);
//if threshold * bonus is less than PERCENTAGE_FACTOR, it's guaranteed that at the moment//a loan is taken there is enough collateral available to cover the liquidation bonusrequire(
liquidationThreshold.percentMul(liquidationBonus) <= PercentageMath.PERCENTAGE_FACTOR,
Errors.LPC_INVALID_CONFIGURATION
);
} else {
require(liquidationBonus ==0, Errors.LPC_INVALID_CONFIGURATION);
//if the liquidation threshold is being set to 0,// the reserve is being disabled as collateral. To do so,//we need to ensure no liquidity is deposited
_checkNoLiquidity(asset);
}
currentConfig.setLtv(ltv);
currentConfig.setLiquidationThreshold(liquidationThreshold);
currentConfig.setLiquidationBonus(liquidationBonus);
pool.setConfiguration(asset, currentConfig.data);
emit CollateralConfigurationChanged(asset, ltv, liquidationThreshold, liquidationBonus);
}
/**
* @dev Enable stable rate borrowing on a reserve
* @param asset The address of the underlying asset of the reserve
**/functionenableReserveStableRate(address asset) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setStableRateBorrowingEnabled(true);
pool.setConfiguration(asset, currentConfig.data);
emit StableRateEnabledOnReserve(asset);
}
/**
* @dev Disable stable rate borrowing on a reserve
* @param asset The address of the underlying asset of the reserve
**/functiondisableReserveStableRate(address asset) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setStableRateBorrowingEnabled(false);
pool.setConfiguration(asset, currentConfig.data);
emit StableRateDisabledOnReserve(asset);
}
/**
* @dev Activates a reserve
* @param asset The address of the underlying asset of the reserve
**/functionactivateReserve(address asset) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setActive(true);
pool.setConfiguration(asset, currentConfig.data);
emit ReserveActivated(asset);
}
/**
* @dev Deactivates a reserve
* @param asset The address of the underlying asset of the reserve
**/functiondeactivateReserve(address asset) externalonlyPoolAdmin{
_checkNoLiquidity(asset);
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setActive(false);
pool.setConfiguration(asset, currentConfig.data);
emit ReserveDeactivated(asset);
}
/**
* @dev Freezes a reserve. A frozen reserve doesn't allow any new deposit, borrow or rate swap
* but allows repayments, liquidations, rate rebalances and withdrawals
* @param asset The address of the underlying asset of the reserve
**/functionfreezeReserve(address asset) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setFrozen(true);
pool.setConfiguration(asset, currentConfig.data);
emit ReserveFrozen(asset);
}
/**
* @dev Unfreezes a reserve
* @param asset The address of the underlying asset of the reserve
**/functionunfreezeReserve(address asset) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setFrozen(false);
pool.setConfiguration(asset, currentConfig.data);
emit ReserveUnfrozen(asset);
}
/**
* @dev Updates the reserve factor of a reserve
* @param asset The address of the underlying asset of the reserve
* @param reserveFactor The new reserve factor of the reserve
**/functionsetReserveFactor(address asset, uint256 reserveFactor) externalonlyPoolAdmin{
DataTypes.ReserveConfigurationMap memory currentConfig = pool.getConfiguration(asset);
currentConfig.setReserveFactor(reserveFactor);
pool.setConfiguration(asset, currentConfig.data);
emit ReserveFactorChanged(asset, reserveFactor);
}
/**
* @dev Sets the interest rate strategy of a reserve
* @param asset The address of the underlying asset of the reserve
* @param rateStrategyAddress The new address of the interest strategy contract
**/functionsetReserveInterestRateStrategyAddress(address asset, address rateStrategyAddress)
externalonlyPoolAdmin{
pool.setReserveInterestRateStrategyAddress(asset, rateStrategyAddress);
emit ReserveInterestRateStrategyChanged(asset, rateStrategyAddress);
}
/**
* @dev pauses or unpauses all the actions of the protocol, including aToken transfers
* @param val true if protocol needs to be paused, false otherwise
**/functionsetPoolPause(bool val) externalonlyEmergencyAdmin{
pool.setPause(val);
}
function_initTokenWithProxy(address implementation, bytesmemory initParams)
internalreturns (address)
{
InitializableImmutableAdminUpgradeabilityProxy proxy =new InitializableImmutableAdminUpgradeabilityProxy(address(this));
proxy.initialize(implementation, initParams);
returnaddress(proxy);
}
function_upgradeTokenImplementation(address proxyAddress,
address implementation,
bytesmemory initParams
) internal{
InitializableImmutableAdminUpgradeabilityProxy proxy =
InitializableImmutableAdminUpgradeabilityProxy(payable(proxyAddress));
proxy.upgradeToAndCall(implementation, initParams);
}
function_checkNoLiquidity(address asset) internalview{
DataTypes.ReserveData memory reserveData = pool.getReserveData(asset);
uint256 availableLiquidity = IERC20Detailed(asset).balanceOf(reserveData.aTokenAddress);
require(
availableLiquidity ==0&& reserveData.currentLiquidityRate ==0,
Errors.LPC_RESERVE_LIQUIDITY_NOT_0
);
}
}
Contract Source Code
File 59 of 91: LendingPoolStorage.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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';
contractLendingPoolStorage{
usingReserveLogicforDataTypes.ReserveData;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
usingUserConfigurationforDataTypes.UserConfigurationMap;
ILendingPoolAddressesProvider internal _addressesProvider;
mapping(address=> DataTypes.ReserveData) internal _reserves;
mapping(address=> DataTypes.UserConfigurationMap) internal _usersConfig;
// the list of the available reserves, structured as a mapping for gas savings reasonsmapping(uint256=>address) internal _reservesList;
uint256internal _reservesCount;
boolinternal _paused;
uint256internal _maxStableRateBorrowSizePercent;
uint256internal _flashLoanPremiumTotal;
uint256internal _maxNumberOfReserves;
}
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;import"../interfaces/IMultiFeeDistribution.sol";
import"../interfaces/IOnwardIncentivesController.sol";
import"../dependencies/openzeppelin/contracts/IERC20.sol";
import"../dependencies/openzeppelin/contracts/SafeERC20.sol";
import"../dependencies/openzeppelin/contracts/SafeMath.sol";
import"../dependencies/openzeppelin/contracts/Ownable.sol";
// based on the Sushi MasterChef// https://github.com/sushiswap/sushiswap/blob/master/contracts/MasterChef.solcontractMasterChefisOwnable{
usingSafeMathforuint256;
usingSafeERC20forIERC20;
// Info of each user.structUserInfo {
uint256 amount;
uint256 rewardDebt;
}
// Info of each pool.structPoolInfo {
uint256 allocPoint; // How many allocation points assigned to this pool.uint256 lastRewardTime; // Last second that reward distribution occurs.uint256 accRewardPerShare; // Accumulated rewards per share, times 1e12. See below.
IOnwardIncentivesController onwardIncentives;
}
// Info about token emissions for a given time period.structEmissionPoint {
uint128 startTimeOffset;
uint128 rewardsPerSecond;
}
addresspublic poolConfigurator;
IMultiFeeDistribution public rewardMinter;
uint256public rewardsPerSecond;
uint256publicimmutable maxMintableTokens;
uint256public mintedTokens;
// Info of each pool.address[] public registeredTokens;
mapping(address=> PoolInfo) public poolInfo;
// Data about the future reward rates. emissionSchedule stored in reverse chronological order,// whenever the number of blocks since the start block exceeds the next block offset a new// reward rate is applied.
EmissionPoint[] public emissionSchedule;
// token => user => Info of each user that stakes LP tokens.mapping(address=>mapping(address=> UserInfo)) public userInfo;
// user => base claimable balancemapping(address=>uint256) public userBaseClaimable;
// Total allocation poitns. Must be the sum of all allocation points in all pools.uint256public totalAllocPoint =0;
// The block number when reward mining starts.uint256public startTime;
// account earning rewards => receiver of rewards for this account// if receiver is set to address(0), rewards are paid to the earner// this is used to aid 3rd party contract integrationsmapping (address=>address) public claimReceiver;
eventDeposit(addressindexed token,
addressindexed user,
uint256 amount
);
eventWithdraw(addressindexed token,
addressindexed user,
uint256 amount
);
eventEmergencyWithdraw(addressindexed token,
addressindexed user,
uint256 amount
);
constructor(uint128[] memory _startTimeOffset,
uint128[] memory _rewardsPerSecond,
address _poolConfigurator,
IMultiFeeDistribution _rewardMinter,
uint256 _maxMintable
)
Ownable()
{
poolConfigurator = _poolConfigurator;
rewardMinter = _rewardMinter;
uint256 length = _startTimeOffset.length;
for (uint256 i = length -1; i +1!=0; i--) {
emissionSchedule.push(
EmissionPoint({
startTimeOffset: _startTimeOffset[i],
rewardsPerSecond: _rewardsPerSecond[i]
})
);
}
maxMintableTokens = _maxMintable;
}
// Start the partyfunctionstart() publiconlyOwner{
require(startTime ==0);
startTime =block.timestamp;
}
// Add a new lp to the pool. Can only be called by the poolConfigurator.functionaddPool(address _token, uint256 _allocPoint) externalonlyOwner{
require(poolInfo[_token].lastRewardTime ==0);
_updateEmissions();
totalAllocPoint = totalAllocPoint.add(_allocPoint);
registeredTokens.push(_token);
poolInfo[_token] = PoolInfo({
allocPoint: _allocPoint,
lastRewardTime: block.timestamp,
accRewardPerShare: 0,
onwardIncentives: IOnwardIncentivesController(0)
});
}
// Update the given pool's allocation point. Can only be called by the owner.functionbatchUpdateAllocPoint(address[] calldata _tokens,
uint256[] calldata _allocPoints
) publiconlyOwner{
require(_tokens.length== _allocPoints.length);
_massUpdatePools();
uint256 _totalAllocPoint = totalAllocPoint;
for (uint256 i =0; i < _tokens.length; i++) {
PoolInfo storage pool = poolInfo[_tokens[i]];
require(pool.lastRewardTime >0);
_totalAllocPoint = _totalAllocPoint.sub(pool.allocPoint).add(_allocPoints[i]);
pool.allocPoint = _allocPoints[i];
}
totalAllocPoint = _totalAllocPoint;
}
functionsetOnwardIncentives(address _token,
IOnwardIncentivesController _incentives
)
externalonlyOwner{
require(poolInfo[_token].lastRewardTime !=0);
poolInfo[_token].onwardIncentives = _incentives;
}
functionsetClaimReceiver(address _user, address _receiver) external{
require(msg.sender== _user ||msg.sender== owner());
claimReceiver[_user] = _receiver;
}
functionpoolLength() externalviewreturns (uint256) {
return registeredTokens.length;
}
functionclaimableReward(address _user, address[] calldata _tokens)
externalviewreturns (uint256[] memory)
{
uint256[] memory claimable =newuint256[](_tokens.length);
for (uint256 i =0; i < _tokens.length; i++) {
address token = _tokens[i];
PoolInfo storage pool = poolInfo[token];
UserInfo storage user = userInfo[token][_user];
uint256 accRewardPerShare = pool.accRewardPerShare;
uint256 lpSupply = IERC20(token).balanceOf(address(this));
if (block.timestamp> pool.lastRewardTime && lpSupply !=0) {
uint256 duration =block.timestamp.sub(pool.lastRewardTime);
uint256 reward = duration.mul(rewardsPerSecond).mul(pool.allocPoint).div(totalAllocPoint);
accRewardPerShare = accRewardPerShare.add(reward.mul(1e12).div(lpSupply));
}
claimable[i] = user.amount.mul(accRewardPerShare).div(1e12).sub(user.rewardDebt);
}
return claimable;
}
function_updateEmissions() internal{
uint256 length = emissionSchedule.length;
if (startTime >0&& length >0) {
EmissionPoint memory e = emissionSchedule[length-1];
if (block.timestamp.sub(startTime) > e.startTimeOffset) {
_massUpdatePools();
rewardsPerSecond =uint256(e.rewardsPerSecond);
emissionSchedule.pop();
}
}
}
// Update reward variables for all poolsfunction_massUpdatePools() internal{
uint256 totalAP = totalAllocPoint;
uint256 length = registeredTokens.length;
for (uint256 i =0; i < length; ++i) {
_updatePool(registeredTokens[i], totalAP);
}
}
// Update reward variables of the given pool to be up-to-date.function_updatePool(address _token, uint256 _totalAllocPoint) internal{
PoolInfo storage pool = poolInfo[_token];
if (block.timestamp<= pool.lastRewardTime) {
return;
}
uint256 lpSupply = IERC20(_token).balanceOf(address(this));
if (lpSupply ==0) {
pool.lastRewardTime =block.timestamp;
return;
}
uint256 duration =block.timestamp.sub(pool.lastRewardTime);
uint256 reward = duration.mul(rewardsPerSecond).mul(pool.allocPoint).div(_totalAllocPoint);
pool.accRewardPerShare = pool.accRewardPerShare.add(reward.mul(1e12).div(lpSupply));
pool.lastRewardTime =block.timestamp;
}
function_mint(address _user, uint256 _amount) internal{
uint256 minted = mintedTokens;
if (minted.add(_amount) > maxMintableTokens) {
_amount = maxMintableTokens.sub(minted);
}
if (_amount >0) {
mintedTokens = minted.add(_amount);
address receiver = claimReceiver[_user];
if (receiver ==address(0)) receiver = _user;
rewardMinter.mint(receiver, _amount, true);
}
}
// Deposit LP tokens into the contract. Also triggers a claim.functiondeposit(address _token, uint256 _amount) external{
PoolInfo storage pool = poolInfo[_token];
require(pool.lastRewardTime >0);
_updateEmissions();
_updatePool(_token, totalAllocPoint);
UserInfo storage user = userInfo[_token][msg.sender];
uint256 userAmount = user.amount;
uint256 accRewardPerShare = pool.accRewardPerShare;
if (userAmount >0) {
uint256 pending = userAmount.mul(accRewardPerShare).div(1e12).sub(user.rewardDebt);
if (pending >0) {
userBaseClaimable[msg.sender] = userBaseClaimable[msg.sender].add(pending);
}
}
IERC20(_token).safeTransferFrom(
address(msg.sender),
address(this),
_amount
);
userAmount = userAmount.add(_amount);
user.amount = userAmount;
user.rewardDebt = userAmount.mul(accRewardPerShare).div(1e12);
if (pool.onwardIncentives != IOnwardIncentivesController(0)) {
uint256 lpSupply = IERC20(_token).balanceOf(address(this));
pool.onwardIncentives.handleAction(_token, msg.sender, userAmount, lpSupply);
}
emit Deposit(_token, msg.sender, _amount);
}
// Withdraw LP tokens. Also triggers a claim.functionwithdraw(address _token, uint256 _amount) external{
PoolInfo storage pool = poolInfo[_token];
require(pool.lastRewardTime >0);
UserInfo storage user = userInfo[_token][msg.sender];
uint256 userAmount = user.amount;
require(userAmount >= _amount, "withdraw: not good");
_updateEmissions();
_updatePool(_token, totalAllocPoint);
uint256 accRewardPerShare = pool.accRewardPerShare;
uint256 pending = userAmount.mul(accRewardPerShare).div(1e12).sub(user.rewardDebt);
if (pending >0) {
userBaseClaimable[msg.sender] = userBaseClaimable[msg.sender].add(pending);
}
userAmount = userAmount.sub(_amount);
user.amount = userAmount;
user.rewardDebt = userAmount.mul(accRewardPerShare).div(1e12);
IERC20(_token).safeTransfer(address(msg.sender), _amount);
if (pool.onwardIncentives != IOnwardIncentivesController(0)) {
uint256 lpSupply = IERC20(_token).balanceOf(address(this));
pool.onwardIncentives.handleAction(_token, msg.sender, userAmount, lpSupply);
}
emit Withdraw(_token, msg.sender, _amount);
}
// Withdraw without caring about rewards. EMERGENCY ONLY.functionemergencyWithdraw(address _token) external{
PoolInfo storage pool = poolInfo[_token];
UserInfo storage user = userInfo[_token][msg.sender];
uint256 amount = user.amount;
user.amount =0;
user.rewardDebt =0;
IERC20(_token).safeTransfer(address(msg.sender), amount);
emit EmergencyWithdraw(_token, msg.sender, amount);
if (pool.onwardIncentives != IOnwardIncentivesController(0)) {
uint256 lpSupply = IERC20(_token).balanceOf(address(this));
try pool.onwardIncentives.handleAction(_token, msg.sender, 0, lpSupply) {} catch {}
}
}
// Claim pending rewards for one or more pools.// Rewards are not received directly, they are minted by the rewardMinter.functionclaim(address _user, address[] calldata _tokens) external{
_updateEmissions();
uint256 pending = userBaseClaimable[_user];
userBaseClaimable[_user] =0;
uint256 _totalAllocPoint = totalAllocPoint;
for (uint i =0; i < _tokens.length; i++) {
PoolInfo storage pool = poolInfo[_tokens[i]];
require(pool.lastRewardTime >0);
_updatePool(_tokens[i], _totalAllocPoint);
UserInfo storage user = userInfo[_tokens[i]][_user];
uint256 rewardDebt = user.amount.mul(pool.accRewardPerShare).div(1e12);
pending = pending.add(rewardDebt.sub(user.rewardDebt));
user.rewardDebt = rewardDebt;
}
_mint(_user, pending);
}
}
Contract Source Code
File 62 of 91: MathUtils.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {SafeMath} from'../../../dependencies/openzeppelin/contracts/SafeMath.sol';
import {WadRayMath} from'./WadRayMath.sol';
libraryMathUtils{
usingSafeMathforuint256;
usingWadRayMathforuint256;
/// @dev Ignoring leap yearsuint256internalconstant SECONDS_PER_YEAR =365days;
/**
* @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
**/functioncalculateLinearInterest(uint256 rate, uint40 lastUpdateTimestamp)
internalviewreturns (uint256)
{
//solium-disable-next-lineuint256 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
**/functioncalculateCompoundedInterest(uint256 rate,
uint40 lastUpdateTimestamp,
uint256 currentTimestamp
) internalpurereturns (uint256) {
//solium-disable-next-lineuint256 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
**/functioncalculateCompoundedInterest(uint256 rate, uint40 lastUpdateTimestamp)
internalviewreturns (uint256)
{
return calculateCompoundedInterest(rate, lastUpdateTimestamp, block.timestamp);
}
}
Contract Source Code
File 63 of 91: MerkleDistributor.sol
// SPDX-License-Identifier: UNLICENSEDpragmasolidity 0.7.6;import"../interfaces/IMultiFeeDistribution.sol";
import"../dependencies/openzeppelin/contracts/Ownable.sol";
import"../dependencies/openzeppelin/contracts/SafeMath.sol";
contractMerkleDistributorisOwnable{
usingSafeMathforuint256;
structClaimRecord {
bytes32 merkleRoot;
uint256 validUntil;
uint256 total;
uint256 claimed;
}
uint256publicimmutable maxMintableTokens;
uint256public mintedTokens;
uint256public reservedTokens;
uint256publicimmutable startTime;
uint256publicconstant duration =86400*365;
uint256publicconstant minDuration =86400*7;
IMultiFeeDistribution public rewardMinter;
ClaimRecord[] public claims;
eventClaimed(addressindexed account,
uint256indexed merkleIndex,
uint256 index,
uint256 amount,
address receiver
);
// This is a packed array of booleans.mapping(uint256=>mapping(uint256=>uint256)) private claimedBitMap;
constructor(IMultiFeeDistribution _rewardMinter, uint256 _maxMintable) Ownable() {
rewardMinter = _rewardMinter;
maxMintableTokens = _maxMintable;
startTime =block.timestamp;
}
functionmintableBalance() publicviewreturns (uint256) {
uint elapsedTime =block.timestamp.sub(startTime);
if (elapsedTime > duration) elapsedTime = duration;
return maxMintableTokens.mul(elapsedTime).div(duration).sub(mintedTokens).sub(reservedTokens);
}
functionaddClaimRecord(bytes32 _root, uint256 _duration, uint256 _total) externalonlyOwner{
require(_duration >= minDuration);
uint mintable = mintableBalance();
require(mintable >= _total);
claims.push(ClaimRecord({
merkleRoot: _root,
validUntil: block.timestamp+ _duration,
total: _total,
claimed: 0
}));
reservedTokens = reservedTokens.add(_total);
}
functionreleaseExpiredClaimReserves(uint256[] calldata _claimIndexes) external{
for (uint256 i =0; i < _claimIndexes.length; i++) {
ClaimRecord storage c = claims[_claimIndexes[i]];
require(block.timestamp> c.validUntil, 'MerkleDistributor: Drop still active.');
reservedTokens = reservedTokens.sub(c.total.sub(c.claimed));
c.total =0;
c.claimed =0;
}
}
functionisClaimed(uint256 _claimIndex, uint256 _index) publicviewreturns (bool) {
uint256 claimedWordIndex = _index /256;
uint256 claimedBitIndex = _index %256;
uint256 claimedWord = claimedBitMap[_claimIndex][claimedWordIndex];
uint256 mask = (1<< claimedBitIndex);
return claimedWord & mask == mask;
}
function_setClaimed(uint256 _claimIndex, uint256 _index) private{
uint256 claimedWordIndex = _index /256;
uint256 claimedBitIndex = _index %256;
claimedBitMap[_claimIndex][claimedWordIndex] = claimedBitMap[_claimIndex][claimedWordIndex] | (1<< claimedBitIndex);
}
functionclaim(uint256 _claimIndex,
uint256 _index,
uint256 _amount,
address _receiver,
bytes32[] calldata _merkleProof
) external{
require(_claimIndex < claims.length, 'MerkleDistributor: Invalid merkleIndex');
require(!isClaimed(_claimIndex, _index), 'MerkleDistributor: Drop already claimed.');
ClaimRecord storage c = claims[_claimIndex];
require(c.validUntil >block.timestamp, 'MerkleDistributor: Drop has expired.');
c.claimed = c.claimed.add(_amount);
require(c.total >= c.claimed, 'MerkleDistributor: Exceeds allocated total for drop.');
reservedTokens = reservedTokens.sub(_amount);
mintedTokens = mintedTokens.add(_amount);
// Verify the merkle proof.bytes32 node =keccak256(abi.encodePacked(_index, msg.sender, _amount));
require(verify(_merkleProof, c.merkleRoot, node), 'MerkleDistributor: Invalid proof.');
// Mark it claimed and send the token.
_setClaimed(_claimIndex, _index);
rewardMinter.mint(_receiver, _amount, true);
emit Claimed(msg.sender, _claimIndex, _index, _amount, _receiver);
}
functionverify(bytes32[] calldata _proof, bytes32 _root, bytes32 _leaf) internalpurereturns (bool) {
bytes32 computedHash = _leaf;
for (uint256 i =0; i < _proof.length; i++) {
bytes32 proofElement = _proof[i];
if (computedHash <= proofElement) {
// Hash(current computed hash + current element of the proof)
computedHash =keccak256(abi.encodePacked(computedHash, proofElement));
} else {
// Hash(current element of the proof + current computed hash)
computedHash =keccak256(abi.encodePacked(proofElement, computedHash));
}
}
// Check if the computed hash (root) is equal to the provided rootreturn computedHash == _root;
}
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import"../../dependencies/openzeppelin/contracts/ERC20.sol";
contractMockERC20isERC20{
constructor(stringmemory name, stringmemory symbol, uint8 decimals) ERC20(name, symbol) public{
_setupDecimals(decimals);
_mint(msg.sender, 1000*10**decimals);
}
// deposit wraps received FTM tokens as wFTM in 1:1 ratio by minting// the received amount of FTMs in wFTM on the sender's address.functiondeposit() publicpayablereturns (uint256) {
// there has to be some value to be convertedif (msg.value==0) {
return0x01;
}
// we already received FTMs, mint the appropriate amount of wFTM
_mint(msg.sender, msg.value);
// all went well herereturn0x0;
}
// withdraw unwraps FTM tokens by burning specified amount// of wFTM from the caller address and sending the same amount// of FTMs back in exchange.functionwithdraw(uint256 amount) publicreturns (uint256) {
// there has to be some value to be convertedif (amount ==0) {
return0x01;
}
// burn wFTM from the sender first to prevent re-entrance issue
_burn(msg.sender, amount);
// if wFTM were burned, transfer native tokens back to the sendermsg.sender.transfer(amount);
// all went well herereturn0x0;
}
}
Contract Source Code
File 67 of 91: MultiFeeDistribution.sol
pragmasolidity 0.7.6;pragmaabicoderv2;import"../interfaces/IChefIncentivesController.sol";
import"../interfaces/IMultiFeeDistribution.sol";
import"../dependencies/openzeppelin/contracts/IERC20.sol";
import"../dependencies/openzeppelin/contracts/SafeERC20.sol";
import"../dependencies/openzeppelin/contracts/SafeMath.sol";
import"../dependencies/openzeppelin/contracts/Ownable.sol";
interfaceIMintableTokenisIERC20{
functionmint(address _receiver, uint256 _amount) externalreturns (bool);
functionsetMinter(address _minter) externalreturns (bool);
}
// Based on Ellipsis EPS Staker// https://github.com/ellipsis-finance/ellipsis/blob/master/contracts/EpsStaker.solcontractMultiFeeDistributionisIMultiFeeDistribution, Ownable{
usingSafeMathforuint256;
usingSafeERC20forIERC20;
usingSafeERC20forIMintableToken;
/* ========== STATE VARIABLES ========== */structReward {
uint256 periodFinish;
uint256 rewardRate;
uint256 lastUpdateTime;
uint256 rewardPerTokenStored;
// tracks already-added balances to handle accrued interest in aToken rewards// for the stakingToken this value is unused and will always be 0uint256 balance;
}
structBalances {
uint256 total;
uint256 unlocked;
uint256 locked;
uint256 earned;
}
structLockedBalance {
uint256 amount;
uint256 unlockTime;
}
structRewardData {
address token;
uint256 amount;
}
IChefIncentivesController public incentivesController;
IMintableToken publicimmutable stakingToken;
address[] public rewardTokens;
mapping(address=> Reward) public rewardData;
// Duration that rewards are streamed overuint256publicconstant rewardsDuration =86400*7;
// Duration of lock/earned penalty perioduint256publicconstant lockDuration = rewardsDuration *13;
// Addresses approved to call mintmapping(address=>bool) public minters;
boolpublic mintersAreSet;
// user -> reward token -> amountmapping(address=>mapping(address=>uint256)) public userRewardPerTokenPaid;
mapping(address=>mapping(address=>uint256)) public rewards;
uint256public totalSupply;
uint256public lockedSupply;
// Private mappings for balance datamapping(address=> Balances) private balances;
mapping(address=> LockedBalance[]) private userLocks;
mapping(address=> LockedBalance[]) private userEarnings;
/* ========== CONSTRUCTOR ========== */constructor(address _stakingToken) Ownable() {
stakingToken = IMintableToken(_stakingToken);
IMintableToken(_stakingToken).setMinter(address(this));
// First reward MUST be the staking token or things will break// related to the 50% penalty and distribution to locked balances
rewardTokens.push(_stakingToken);
rewardData[_stakingToken].lastUpdateTime =block.timestamp;
}
/* ========== ADMIN CONFIGURATION ========== */functionsetMinters(address[] memory _minters) externalonlyOwner{
require(!mintersAreSet);
for (uint i; i < _minters.length; i++) {
minters[_minters[i]] =true;
}
mintersAreSet =true;
}
functionsetIncentivesController(IChefIncentivesController _controller) externalonlyOwner{
incentivesController = _controller;
}
// Add a new reward token to be distributed to stakersfunctionaddReward(address _rewardsToken) externaloverrideonlyOwner{
require(rewardData[_rewardsToken].lastUpdateTime ==0);
rewardTokens.push(_rewardsToken);
rewardData[_rewardsToken].lastUpdateTime =block.timestamp;
rewardData[_rewardsToken].periodFinish =block.timestamp;
}
/* ========== VIEWS ========== */function_rewardPerToken(address _rewardsToken, uint256 _supply) internalviewreturns (uint256) {
if (_supply ==0) {
return rewardData[_rewardsToken].rewardPerTokenStored;
}
return
rewardData[_rewardsToken].rewardPerTokenStored.add(
lastTimeRewardApplicable(_rewardsToken).sub(
rewardData[_rewardsToken].lastUpdateTime).mul(
rewardData[_rewardsToken].rewardRate).mul(1e18).div(_supply)
);
}
function_earned(address _user,
address _rewardsToken,
uint256 _balance,
uint256 _currentRewardPerToken
) internalviewreturns (uint256) {
return _balance.mul(
_currentRewardPerToken.sub(userRewardPerTokenPaid[_user][_rewardsToken])
).div(1e18).add(rewards[_user][_rewardsToken]);
}
functionlastTimeRewardApplicable(address _rewardsToken) publicviewreturns (uint256) {
uint periodFinish = rewardData[_rewardsToken].periodFinish;
returnblock.timestamp< periodFinish ? block.timestamp : periodFinish;
}
functionrewardPerToken(address _rewardsToken) externalviewreturns (uint256) {
uint256 supply = _rewardsToken ==address(stakingToken) ? lockedSupply : totalSupply;
return _rewardPerToken(_rewardsToken, supply);
}
functiongetRewardForDuration(address _rewardsToken) externalviewreturns (uint256) {
return rewardData[_rewardsToken].rewardRate.mul(rewardsDuration).div(1e12);
}
// Address and claimable amount of all reward tokens for the given accountfunctionclaimableRewards(address account) externalviewreturns (RewardData[] memory rewards) {
rewards =new RewardData[](rewardTokens.length);
for (uint256 i =0; i < rewards.length; i++) {
// If i == 0 this is the stakingReward, distribution is based on locked balancesuint256 balance = i ==0 ? balances[account].locked : balances[account].total;
uint256 supply = i ==0 ? lockedSupply : totalSupply;
rewards[i].token = rewardTokens[i];
rewards[i].amount = _earned(account, rewards[i].token, balance, _rewardPerToken(rewardTokens[i], supply)).div(1e12);
}
return rewards;
}
// Total balance of an account, including unlocked, locked and earned tokensfunctiontotalBalance(address user) viewexternalreturns (uint256 amount) {
return balances[user].total;
}
// Total withdrawable balance for an account to which no penalty is appliedfunctionunlockedBalance(address user) viewexternalreturns (uint256 amount) {
amount = balances[user].unlocked;
LockedBalance[] storage earnings = userEarnings[msg.sender];
for (uint i =0; i < earnings.length; i++) {
if (earnings[i].unlockTime >block.timestamp) {
break;
}
amount = amount.add(earnings[i].amount);
}
return amount;
}
// Information on the "earned" balances of a user// Earned balances may be withdrawn immediately for a 50% penaltyfunctionearnedBalances(address user
) viewexternalreturns (uint256 total,
LockedBalance[] memory earningsData
) {
LockedBalance[] storage earnings = userEarnings[user];
uint256 idx;
for (uint i =0; i < earnings.length; i++) {
if (earnings[i].unlockTime >block.timestamp) {
if (idx ==0) {
earningsData =new LockedBalance[](earnings.length- i);
}
earningsData[idx] = earnings[i];
idx++;
total = total.add(earnings[i].amount);
}
}
return (total, earningsData);
}
// Information on a user's locked balancesfunctionlockedBalances(address user
) viewexternalreturns (uint256 total,
uint256 unlockable,
uint256 locked,
LockedBalance[] memory lockData
) {
LockedBalance[] storage locks = userLocks[user];
uint256 idx;
for (uint i =0; i < locks.length; i++) {
if (locks[i].unlockTime >block.timestamp) {
if (idx ==0) {
lockData =new LockedBalance[](locks.length- i);
}
lockData[idx] = locks[i];
idx++;
locked = locked.add(locks[i].amount);
} else {
unlockable = unlockable.add(locks[i].amount);
}
}
return (balances[user].locked, unlockable, locked, lockData);
}
// Final balance received and penalty balance paid by user upon calling exitfunctionwithdrawableBalance(address user
) viewpublicreturns (uint256 amount,
uint256 penaltyAmount
) {
Balances storage bal = balances[user];
uint256 earned = bal.earned;
if (earned >0) {
uint256 amountWithoutPenalty;
uint256 length = userEarnings[user].length;
for (uint i =0; i < length; i++) {
uint256 earnedAmount = userEarnings[user][i].amount;
if (earnedAmount ==0) continue;
if (userEarnings[user][i].unlockTime >block.timestamp) {
break;
}
amountWithoutPenalty = amountWithoutPenalty.add(earnedAmount);
}
penaltyAmount = earned.sub(amountWithoutPenalty).div(2);
}
amount = bal.unlocked.add(earned).sub(penaltyAmount);
return (amount, penaltyAmount);
}
/* ========== MUTATIVE FUNCTIONS ========== */// Stake tokens to receive rewards// Locked tokens cannot be withdrawn for lockDuration and are eligible to receive stakingReward rewardsfunctionstake(uint256 amount, bool lock) external{
require(amount >0, "Cannot stake 0");
_updateReward(msg.sender);
totalSupply = totalSupply.add(amount);
Balances storage bal = balances[msg.sender];
bal.total = bal.total.add(amount);
if (lock) {
lockedSupply = lockedSupply.add(amount);
bal.locked = bal.locked.add(amount);
uint256 unlockTime =block.timestamp.div(rewardsDuration).mul(rewardsDuration).add(lockDuration);
uint256 idx = userLocks[msg.sender].length;
if (idx ==0|| userLocks[msg.sender][idx-1].unlockTime < unlockTime) {
userLocks[msg.sender].push(LockedBalance({amount: amount, unlockTime: unlockTime}));
} else {
userLocks[msg.sender][idx-1].amount = userLocks[msg.sender][idx-1].amount.add(amount);
}
} else {
bal.unlocked = bal.unlocked.add(amount);
}
stakingToken.safeTransferFrom(msg.sender, address(this), amount);
emit Staked(msg.sender, amount, lock);
}
// Mint new tokens// Minted tokens receive rewards normally but incur a 50% penalty when// withdrawn before lockDuration has passed.functionmint(address user, uint256 amount, bool withPenalty) externaloverride{
require(minters[msg.sender]);
if (amount ==0) return;
_updateReward(user);
stakingToken.mint(address(this), amount);
if (user ==address(this)) {
// minting to this contract adds the new tokens as incentives for lockers
_notifyReward(address(stakingToken), amount);
return;
}
totalSupply = totalSupply.add(amount);
Balances storage bal = balances[user];
bal.total = bal.total.add(amount);
if (withPenalty) {
bal.earned = bal.earned.add(amount);
uint256 unlockTime =block.timestamp.div(rewardsDuration).mul(rewardsDuration).add(lockDuration);
LockedBalance[] storage earnings = userEarnings[user];
uint256 idx = earnings.length;
if (idx ==0|| earnings[idx-1].unlockTime < unlockTime) {
earnings.push(LockedBalance({amount: amount, unlockTime: unlockTime}));
} else {
earnings[idx-1].amount = earnings[idx-1].amount.add(amount);
}
} else {
bal.unlocked = bal.unlocked.add(amount);
}
emit Staked(user, amount, false);
}
// Withdraw staked tokens// First withdraws unlocked tokens, then earned tokens. Withdrawing earned tokens// incurs a 50% penalty which is distributed based on locked balances.functionwithdraw(uint256 amount) public{
require(amount >0, "Cannot withdraw 0");
_updateReward(msg.sender);
Balances storage bal = balances[msg.sender];
uint256 penaltyAmount;
if (amount <= bal.unlocked) {
bal.unlocked = bal.unlocked.sub(amount);
} else {
uint256 remaining = amount.sub(bal.unlocked);
require(bal.earned >= remaining, "Insufficient unlocked balance");
bal.unlocked =0;
bal.earned = bal.earned.sub(remaining);
for (uint i =0; ; i++) {
uint256 earnedAmount = userEarnings[msg.sender][i].amount;
if (earnedAmount ==0) continue;
if (penaltyAmount ==0&& userEarnings[msg.sender][i].unlockTime >block.timestamp) {
penaltyAmount = remaining;
require(bal.earned >= remaining, "Insufficient balance after penalty");
bal.earned = bal.earned.sub(remaining);
if (bal.earned ==0) {
delete userEarnings[msg.sender];
break;
}
remaining = remaining.mul(2);
}
if (remaining <= earnedAmount) {
userEarnings[msg.sender][i].amount = earnedAmount.sub(remaining);
break;
} else {
delete userEarnings[msg.sender][i];
remaining = remaining.sub(earnedAmount);
}
}
}
uint256 adjustedAmount = amount.add(penaltyAmount);
bal.total = bal.total.sub(adjustedAmount);
totalSupply = totalSupply.sub(adjustedAmount);
stakingToken.safeTransfer(msg.sender, amount);
if (penaltyAmount >0) {
incentivesController.claim(address(this), newaddress[](0));
_notifyReward(address(stakingToken), penaltyAmount);
}
emit Withdrawn(msg.sender, amount, penaltyAmount);
}
function_getReward(address[] memory _rewardTokens) internal{
uint256 length = _rewardTokens.length;
for (uint i; i < length; i++) {
address token = _rewardTokens[i];
uint256 reward = rewards[msg.sender][token].div(1e12);
if (token !=address(stakingToken)) {
// for rewards other than stakingToken, every 24 hours we check if new// rewards were sent to the contract or accrued via aToken interest
Reward storage r = rewardData[token];
uint256 periodFinish = r.periodFinish;
require(periodFinish >0, "Unknown reward token");
uint256 balance = r.balance;
if (periodFinish <block.timestamp.add(rewardsDuration -86400)) {
uint256 unseen = IERC20(token).balanceOf(address(this)).sub(balance);
if (unseen >0) {
_notifyReward(token, unseen);
balance = balance.add(unseen);
}
}
r.balance= balance.sub(reward);
}
if (reward ==0) continue;
rewards[msg.sender][token] =0;
IERC20(token).safeTransfer(msg.sender, reward);
emit RewardPaid(msg.sender, token, reward);
}
}
// Claim all pending staking rewardsfunctiongetReward(address[] memory _rewardTokens) public{
_updateReward(msg.sender);
_getReward(_rewardTokens);
}
// Withdraw full unlocked balance and optionally claim pending rewardsfunctionexit(bool claimRewards) external{
_updateReward(msg.sender);
(uint256 amount, uint256 penaltyAmount) = withdrawableBalance(msg.sender);
delete userEarnings[msg.sender];
Balances storage bal = balances[msg.sender];
bal.total = bal.total.sub(bal.unlocked).sub(bal.earned);
bal.unlocked =0;
bal.earned =0;
totalSupply = totalSupply.sub(amount.add(penaltyAmount));
stakingToken.safeTransfer(msg.sender, amount);
if (penaltyAmount >0) {
incentivesController.claim(address(this), newaddress[](0));
_notifyReward(address(stakingToken), penaltyAmount);
}
if (claimRewards) {
_getReward(rewardTokens);
}
emit Withdrawn(msg.sender, amount, penaltyAmount);
}
// Withdraw all currently locked tokens where the unlock time has passedfunctionwithdrawExpiredLocks() external{
_updateReward(msg.sender);
LockedBalance[] storage locks = userLocks[msg.sender];
Balances storage bal = balances[msg.sender];
uint256 amount;
uint256 length = locks.length;
if (locks[length-1].unlockTime <=block.timestamp) {
amount = bal.locked;
delete userLocks[msg.sender];
} else {
for (uint i =0; i < length; i++) {
if (locks[i].unlockTime >block.timestamp) break;
amount = amount.add(locks[i].amount);
delete locks[i];
}
}
bal.locked = bal.locked.sub(amount);
bal.total = bal.total.sub(amount);
totalSupply = totalSupply.sub(amount);
lockedSupply = lockedSupply.sub(amount);
stakingToken.safeTransfer(msg.sender, amount);
emit Withdrawn(msg.sender, amount, 0);
}
/* ========== RESTRICTED FUNCTIONS ========== */function_notifyReward(address _rewardsToken, uint256 reward) internal{
Reward storage r = rewardData[_rewardsToken];
if (block.timestamp>= r.periodFinish) {
r.rewardRate = reward.mul(1e12).div(rewardsDuration);
} else {
uint256 remaining = r.periodFinish.sub(block.timestamp);
uint256 leftover = remaining.mul(r.rewardRate).div(1e12);
r.rewardRate = reward.add(leftover).mul(1e12).div(rewardsDuration);
}
r.lastUpdateTime =block.timestamp;
r.periodFinish =block.timestamp.add(rewardsDuration);
}
// Added to support recovering LP Rewards from other systems such as BAL to be distributed to holdersfunctionrecoverERC20(address tokenAddress, uint256 tokenAmount) externalonlyOwner{
require(tokenAddress !=address(stakingToken), "Cannot withdraw staking token");
require(rewardData[tokenAddress].lastUpdateTime ==0, "Cannot withdraw reward token");
IERC20(tokenAddress).safeTransfer(owner(), tokenAmount);
emit Recovered(tokenAddress, tokenAmount);
}
function_updateReward(address account) internal{
address token =address(stakingToken);
uint256 balance;
Reward storage r = rewardData[token];
uint256 rpt = _rewardPerToken(token, lockedSupply);
r.rewardPerTokenStored = rpt;
r.lastUpdateTime = lastTimeRewardApplicable(token);
if (account !=address(this)) {
// Special case, use the locked balances and supply for stakingReward rewards
rewards[account][token] = _earned(account, token, balances[account].locked, rpt);
userRewardPerTokenPaid[account][token] = rpt;
balance = balances[account].total;
}
uint256 supply = totalSupply;
uint256 length = rewardTokens.length;
for (uint i =1; i < length; i++) {
token = rewardTokens[i];
r = rewardData[token];
rpt = _rewardPerToken(token, supply);
r.rewardPerTokenStored = rpt;
r.lastUpdateTime = lastTimeRewardApplicable(token);
if (account !=address(this)) {
rewards[account][token] = _earned(account, token, balance, rpt);
userRewardPerTokenPaid[account][token] = rpt;
}
}
}
/* ========== EVENTS ========== */eventRewardAdded(uint256 reward);
eventStaked(addressindexed user, uint256 amount, bool locked);
eventWithdrawn(addressindexed user, uint256 receivedAmount, uint256 penaltyPaid);
eventRewardPaid(addressindexed user, addressindexed rewardsToken, uint256 reward);
eventRewardsDurationUpdated(address token, uint256 newDuration);
eventRecovered(address token, uint256 amount);
}
Contract Source Code
File 68 of 91: Ownable.sol
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;import'./Context.sol';
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership}.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/contractOwnableisContext{
addressprivate _owner;
eventOwnershipTransferred(addressindexed previousOwner, addressindexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/constructor() {
address msgSender = _msgSender();
_owner = msgSender;
emit OwnershipTransferred(address(0), msgSender);
}
/**
* @dev Returns the address of the current owner.
*/functionowner() publicviewreturns (address) {
return _owner;
}
/**
* @dev Throws if called by any account other than the owner.
*/modifieronlyOwner() {
require(_owner == _msgSender(), 'Ownable: caller is not the owner');
_;
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore. Can only be called by the current owner.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*/functionrenounceOwnership() publicvirtualonlyOwner{
emit OwnershipTransferred(_owner, address(0));
_owner =address(0);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/functiontransferOwnership(address newOwner) publicvirtualonlyOwner{
require(newOwner !=address(0), 'Ownable: new owner is the zero address');
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}
Contract Source Code
File 69 of 91: PercentageMath.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
**/libraryPercentageMath{
uint256constant PERCENTAGE_FACTOR =1e4; //percentage plus two decimalsuint256constant 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
**/functionpercentMul(uint256 value, uint256 percentage) internalpurereturns (uint256) {
if (value ==0|| percentage ==0) {
return0;
}
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
**/functionpercentDiv(uint256 value, uint256 percentage) internalpurereturns (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;
}
}
Contract Source Code
File 70 of 91: PolterToken.sol
pragmasolidity 0.7.6;// SPDX-License-Identifier: MITimport"../dependencies/openzeppelin/contracts/SafeMath.sol";
import"../dependencies/openzeppelin/contracts/IERC20.sol";
contractPolterTokenisIERC20{
usingSafeMathforuint256;
stringpublicconstant symbol ="POLTER";
stringpublicconstant name ="Polter.Finance Protocol Token";
uint8publicconstant decimals =18;
uint256publicoverride totalSupply;
uint256publicimmutable maxTotalSupply;
addresspublic minter;
mapping(address=>uint256) publicoverride balanceOf;
mapping(address=>mapping(address=>uint256)) publicoverride allowance;
constructor(uint256 _maxTotalSupply) {
maxTotalSupply = _maxTotalSupply;
emit Transfer(address(0), msg.sender, 0);
}
functionsetMinter(address _minter) externalreturns (bool) {
require(minter ==address(0));
minter = _minter;
returntrue;
}
functionapprove(address _spender, uint256 _value) externaloverridereturns (bool) {
allowance[msg.sender][_spender] = _value;
emit Approval(msg.sender, _spender, _value);
returntrue;
}
/** shared logic for transfer and transferFrom */function_transfer(address _from, address _to, uint256 _value) internal{
require(balanceOf[_from] >= _value, "Insufficient balance");
balanceOf[_from] = balanceOf[_from].sub(_value);
balanceOf[_to] = balanceOf[_to].add(_value);
emit Transfer(_from, _to, _value);
}
/**
@notice Transfer tokens to a specified address
@param _to The address to transfer to
@param _value The amount to be transferred
@return Success boolean
*/functiontransfer(address _to, uint256 _value) publicoverridereturns (bool) {
_transfer(msg.sender, _to, _value);
returntrue;
}
/**
@notice Transfer tokens from one address to another
@param _from The address which you want to send tokens from
@param _to The address which you want to transfer to
@param _value The amount of tokens to be transferred
@return Success boolean
*/functiontransferFrom(address _from,
address _to,
uint256 _value
)
publicoverridereturns (bool)
{
uint256 allowed = allowance[_from][msg.sender];
require(allowed >= _value, "Insufficient allowance");
if (allowed !=uint(-1)) {
allowance[_from][msg.sender] = allowed.sub(_value);
}
_transfer(_from, _to, _value);
returntrue;
}
functionmint(address _to, uint256 _value) externalreturns (bool) {
require(msg.sender== minter);
balanceOf[_to] = balanceOf[_to].add(_value);
totalSupply = totalSupply.add(_value);
require(maxTotalSupply >= totalSupply);
emit Transfer(address(0), _to, _value);
returntrue;
}
}
Contract Source Code
File 71 of 91: PriceFeed.sol
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;import"../interfaces/IPriceFeed.sol";
import"../interfaces/IBandStdReference.sol";
import"../interfaces/IChainlinkAggregator.sol";
import"../dependencies/openzeppelin/contracts/SafeMath.sol";
/*
* PriceFeed for mainnet deployment, to be connected to Chainlink's live ETH:USD aggregator reference
* contract, and a wrapper contract bandOracle, which connects to BandMaster contract.
*
* The PriceFeed uses Chainlink as primary oracle, and Band as fallback. It contains logic for
* switching oracles based on oracle failures, timeouts, and conditions for returning to the primary
* Chainlink oracle.
*/contractPriceFeedisIPriceFeed{
usingSafeMathforuint256;
uintconstantpublic DECIMAL_PRECISION =1e18;
IChainlinkAggregator public chainlinkOracle; // Mainnet Chainlink aggregator
IBandStdReference public bandOracle; // Wrapper contract that calls the Band systemstringpublic bandBase;
stringpublicconstant bandQuote ="USD";
// Use to convert a price answer to an 18-digit precision uintuintconstantpublic TARGET_DIGITS =18;
// Maximum time period allowed since Chainlink's latest round data timestamp, beyond which Chainlink is considered frozen.// For stablecoins we recommend 90000, as Chainlink updates once per day when there is no significant price movement// For volatile assets we recommend 14400 (4 hours)uintimmutablepublic TIMEOUT;
// Maximum deviation allowed between two consecutive Chainlink oracle prices. 18-digit precision.uintconstantpublic MAX_PRICE_DEVIATION_FROM_PREVIOUS_ROUND =5e17; // 50%/*
* The maximum relative price difference between two oracle responses allowed in order for the PriceFeed
* to return to using the Chainlink oracle. 18-digit precision.
*/uintconstantpublic MAX_PRICE_DIFFERENCE_BETWEEN_ORACLES =5e16; // 5%// The last good price seen from an oracle by Liquityuintpublic lastGoodPrice;
structChainlinkResponse {
uint80 roundId;
uint256 answer;
uint256 timestamp;
bool success;
uint8 decimals;
}
structBandResponse {
uint256 value;
uint256 timestamp;
bool success;
}
enumStatus {
chainlinkWorking,
usingBandChainlinkUntrusted,
bothOraclesUntrusted,
usingBandChainlinkFrozen,
usingChainlinkBandUntrusted
}
// The current status of the PricFeed, which determines the conditions for the next price fetch attempt
Status public status;
eventLastGoodPriceUpdated(uint _lastGoodPrice);
eventPriceFeedStatusChanged(Status newStatus);
// --- Dependency setters ---constructor(
IChainlinkAggregator _chainlinkOracleAddress,
IBandStdReference _bandOracleAddress,
uint256 _timeout,
stringmemory _bandBase
) {
chainlinkOracle = _chainlinkOracleAddress;
bandOracle = _bandOracleAddress;
TIMEOUT = _timeout;
bandBase = _bandBase;
// Explicitly set initial system status
status = Status.chainlinkWorking;
// Get an initial price from Chainlink to serve as first reference for lastGoodPrice
ChainlinkResponse memory chainlinkResponse = _getCurrentChainlinkResponse();
ChainlinkResponse memory prevChainlinkResponse = _getPrevChainlinkResponse(chainlinkResponse.roundId, chainlinkResponse.decimals);
require(
!_chainlinkIsBroken(chainlinkResponse, prevChainlinkResponse) &&block.timestamp.sub(chainlinkResponse.timestamp) < _timeout,
"PriceFeed: Chainlink must be working and current"
);
lastGoodPrice = _scaleChainlinkPriceByDigits(uint256(chainlinkResponse.answer), chainlinkResponse.decimals);
}
// --- Functions ---/*
* fetchPrice():
* Returns the latest price obtained from the Oracle. Called by Liquity functions that require a current price.
*
* Also callable by anyone externally.
*
* Non-view function - it stores the last good price seen by Liquity.
*
* Uses a main oracle (Chainlink) and a fallback oracle (Band) in case Chainlink fails. If both fail,
* it uses the last good price seen by Liquity.
*
*/functionfetchPrice() externalviewoverridereturns (uint) {
(,uint price) = _fetchPrice();
return price;
}
functionupdatePrice() externaloverridereturns (uint) {
(Status newStatus, uint price) = _fetchPrice();
lastGoodPrice = price;
if (status != newStatus) {
status = newStatus;
emit PriceFeedStatusChanged(newStatus);
}
return price;
}
function_fetchPrice() internalviewreturns (Status, uint) {
// Get current and previous price data from Chainlink, and current price data from Band
ChainlinkResponse memory chainlinkResponse = _getCurrentChainlinkResponse();
ChainlinkResponse memory prevChainlinkResponse = _getPrevChainlinkResponse(chainlinkResponse.roundId, chainlinkResponse.decimals);
BandResponse memory bandResponse = _getCurrentBandResponse();
// --- CASE 1: System fetched last price from Chainlink ---if (status == Status.chainlinkWorking) {
// If Chainlink is broken, try Bandif (_chainlinkIsBroken(chainlinkResponse, prevChainlinkResponse)) {
// If Band is broken then both oracles are untrusted, so return the last good priceif (_bandIsBroken(bandResponse)) {
return (Status.bothOraclesUntrusted, lastGoodPrice);
}
/*
* If Band is only frozen but otherwise returning valid data, return the last good price.
*/if (_bandIsFrozen(bandResponse)) {
return (Status.usingBandChainlinkUntrusted, lastGoodPrice);
}
// If Chainlink is broken and Band is working, switch to Band and return current Band pricereturn (Status.usingBandChainlinkUntrusted, bandResponse.value);
}
// If Chainlink is frozen, try Bandif (_chainlinkIsFrozen(chainlinkResponse)) {
// If Band is broken too, remember Band broke, and return last good priceif (_bandIsBroken(bandResponse)) {
return (Status.usingChainlinkBandUntrusted, lastGoodPrice);
}
// If Band is frozen or working, remember Chainlink froze, and switch to Bandif (_bandIsFrozen(bandResponse)) {return (Status.usingBandChainlinkFrozen, lastGoodPrice);}
// If Band is working, use itreturn (Status.usingBandChainlinkFrozen, bandResponse.value);
}
// If Chainlink price has changed by > 50% between two consecutive rounds, compare it to Band's priceif (_chainlinkPriceChangeAboveMax(chainlinkResponse, prevChainlinkResponse)) {
// If Band is broken, both oracles are untrusted, and return last good priceif (_bandIsBroken(bandResponse)) {
return (Status.bothOraclesUntrusted, lastGoodPrice);
}
// If Band is frozen, switch to Band and return last good priceif (_bandIsFrozen(bandResponse)) {
return (Status.usingBandChainlinkUntrusted, lastGoodPrice);
}
/*
* If Band is live and both oracles have a similar price, conclude that Chainlink's large price deviation between
* two consecutive rounds was likely a legitmate market price movement, and so continue using Chainlink
*/if (_bothOraclesSimilarPrice(chainlinkResponse, bandResponse)) {
return (Status.chainlinkWorking, chainlinkResponse.answer);
}
// If Band is live but the oracles differ too much in price, conclude that Chainlink's initial price deviation was// an oracle failure. Switch to Band, and use Band pricereturn (Status.usingBandChainlinkUntrusted, bandResponse.value);
}
// If Chainlink is working and Band is broken, remember Band is brokenif (_bandIsBroken(bandResponse)) {
return (Status.usingChainlinkBandUntrusted, chainlinkResponse.answer);
}
// If Chainlink is working, return Chainlink current price (no status change)return (Status.chainlinkWorking, chainlinkResponse.answer);
}
// --- CASE 2: The system fetched last price from Band ---if (status == Status.usingBandChainlinkUntrusted) {
// If both Band and Chainlink are live, unbroken, and reporting similar prices, switch back to Chainlinkif (_bothOraclesLiveAndUnbrokenAndSimilarPrice(chainlinkResponse, prevChainlinkResponse, bandResponse)) {
return (Status.chainlinkWorking, chainlinkResponse.answer);
}
if (_bandIsBroken(bandResponse)) {
return (Status.bothOraclesUntrusted, lastGoodPrice);
}
/*
* If Band is only frozen but otherwise returning valid data, just return the last good price.
* Band may need to be tipped to return current data.
*/if (_bandIsFrozen(bandResponse)) {return (Status.usingBandChainlinkUntrusted, lastGoodPrice);}
// Otherwise, use Band pricereturn (Status.usingBandChainlinkUntrusted, bandResponse.value);
}
// --- CASE 3: Both oracles were untrusted at the last price fetch ---if (status == Status.bothOraclesUntrusted) {
/*
* If both oracles are now live, unbroken and similar price, we assume that they are reporting
* accurately, and so we switch back to Chainlink.
*/if (_bothOraclesLiveAndUnbrokenAndSimilarPrice(chainlinkResponse, prevChainlinkResponse, bandResponse)) {
return (Status.chainlinkWorking, chainlinkResponse.answer);
}
// Otherwise, return the last good price - both oracles are still untrusted (no status change)return (Status.bothOraclesUntrusted, lastGoodPrice);
}
// --- CASE 4: Using Band, and Chainlink is frozen ---if (status == Status.usingBandChainlinkFrozen) {
if (_chainlinkIsBroken(chainlinkResponse, prevChainlinkResponse)) {
// If both Oracles are broken, return last good priceif (_bandIsBroken(bandResponse)) {
return (Status.bothOraclesUntrusted, lastGoodPrice);
}
// If Chainlink is broken, remember it and switch to using Bandif (_bandIsFrozen(bandResponse)) {return (Status.usingBandChainlinkUntrusted, lastGoodPrice);}
// If Band is working, return Band current pricereturn (Status.usingBandChainlinkUntrusted, bandResponse.value);
}
if (_chainlinkIsFrozen(chainlinkResponse)) {
// if Chainlink is frozen and Band is broken, remember Band broke, and return last good priceif (_bandIsBroken(bandResponse)) {
return (Status.usingChainlinkBandUntrusted, lastGoodPrice);
}
// If both are frozen, just use lastGoodPriceif (_bandIsFrozen(bandResponse)) {return (Status.usingBandChainlinkFrozen, lastGoodPrice);}
// if Chainlink is frozen and Band is working, keep using Band (no status change)return (Status.usingBandChainlinkFrozen, bandResponse.value);
}
// if Chainlink is live and Band is broken, remember Band broke, and return Chainlink priceif (_bandIsBroken(bandResponse)) {
return (Status.usingChainlinkBandUntrusted, chainlinkResponse.answer);
}
// If Chainlink is live and Band is frozen, just use last good price (no status change) since we have no basis for comparisonif (_bandIsFrozen(bandResponse)) {return (Status.usingBandChainlinkFrozen, lastGoodPrice);}
// If Chainlink is live and Band is working, compare prices. Switch to Chainlink// if prices are within 5%, and return Chainlink price.if (_bothOraclesSimilarPrice(chainlinkResponse, bandResponse)) {
return (Status.chainlinkWorking, chainlinkResponse.answer);
}
// Otherwise if Chainlink is live but price not within 5% of Band, distrust Chainlink, and return Band pricereturn (Status.usingBandChainlinkUntrusted, bandResponse.value);
}
// --- CASE 5: Using Chainlink, Band is untrusted ---if (status == Status.usingChainlinkBandUntrusted) {
// If Chainlink breaks, now both oracles are untrustedif (_chainlinkIsBroken(chainlinkResponse, prevChainlinkResponse)) {
return (Status.bothOraclesUntrusted, lastGoodPrice);
}
// If Chainlink is frozen, return last good price (no status change)if (_chainlinkIsFrozen(chainlinkResponse)) {
return (Status.usingChainlinkBandUntrusted, lastGoodPrice);
}
// If Chainlink and Band are both live, unbroken and similar price, switch back to chainlinkWorking and return Chainlink priceif (_bothOraclesLiveAndUnbrokenAndSimilarPrice(chainlinkResponse, prevChainlinkResponse, bandResponse)) {
return (Status.chainlinkWorking, chainlinkResponse.answer);
}
// If Chainlink is live but deviated >50% from it's previous price and Band is still untrusted, switch// to bothOraclesUntrusted and return last good priceif (_chainlinkPriceChangeAboveMax(chainlinkResponse, prevChainlinkResponse)) {
return (Status.bothOraclesUntrusted, lastGoodPrice);
}
// Otherwise if Chainlink is live and deviated <50% from it's previous price and Band is still untrusted,// return Chainlink price (no status change)return (Status.usingChainlinkBandUntrusted, chainlinkResponse.answer);
}
}
// --- Helper functions ---/* Chainlink is considered broken if its current or previous round data is in any way bad. We check the previous round
* for two reasons:
*
* 1) It is necessary data for the price deviation check in case 1,
* and
* 2) Chainlink is the PriceFeed's preferred primary oracle - having two consecutive valid round responses adds
* peace of mind when using or returning to Chainlink.
*/function_chainlinkIsBroken(ChainlinkResponse memory _currentResponse, ChainlinkResponse memory _prevResponse) internalviewreturns (bool) {
return _badChainlinkResponse(_currentResponse) || _badChainlinkResponse(_prevResponse);
}
function_badChainlinkResponse(ChainlinkResponse memory _response) internalviewreturns (bool) {
// Check for response call revertedif (!_response.success) {returntrue;}
// Check for an invalid roundId that is 0if (_response.roundId ==0) {returntrue;}
// Check for an invalid timeStamp that is 0, or in the futureif (_response.timestamp ==0|| _response.timestamp >block.timestamp) {returntrue;}
// Check for non-positive price (original value returned from chainlink is int256)if (int256(_response.answer) <=0) {returntrue;}
returnfalse;
}
function_chainlinkIsFrozen(ChainlinkResponse memory _response) internalviewreturns (bool) {
returnblock.timestamp.sub(_response.timestamp) > TIMEOUT;
}
function_chainlinkPriceChangeAboveMax(ChainlinkResponse memory _currentResponse, ChainlinkResponse memory _prevResponse) internalpurereturns (bool) {
uint currentScaledPrice = _currentResponse.answer;
uint prevScaledPrice = _prevResponse.answer;
uint minPrice = (currentScaledPrice < prevScaledPrice) ? currentScaledPrice : prevScaledPrice;
uint maxPrice = (currentScaledPrice >= prevScaledPrice) ? currentScaledPrice : prevScaledPrice;
/*
* Use the larger price as the denominator:
* - If price decreased, the percentage deviation is in relation to the the previous price.
* - If price increased, the percentage deviation is in relation to the current price.
*/uint percentDeviation = maxPrice.sub(minPrice).mul(DECIMAL_PRECISION).div(maxPrice);
// Return true if price has more than doubled, or more than halved.return percentDeviation > MAX_PRICE_DEVIATION_FROM_PREVIOUS_ROUND;
}
function_bandIsBroken(BandResponse memory _response) internalviewreturns (bool) {
// Check for response call revertedif (!_response.success) {returntrue;}
// Check for an invalid timeStamp that is 0, or in the futureif (_response.timestamp ==0|| _response.timestamp >block.timestamp) {returntrue;}
// Check for zero priceif (_response.value==0) {returntrue;}
returnfalse;
}
function_bandIsFrozen(BandResponse memory _bandResponse) internalviewreturns (bool) {
returnblock.timestamp.sub(_bandResponse.timestamp) > TIMEOUT;
}
function_bothOraclesLiveAndUnbrokenAndSimilarPrice
(
ChainlinkResponse memory _chainlinkResponse,
ChainlinkResponse memory _prevChainlinkResponse,
BandResponse memory _bandResponse
)
internalviewreturns (bool)
{
// Return false if either oracle is broken or frozenif
(
_bandIsBroken(_bandResponse) ||
_bandIsFrozen(_bandResponse) ||
_chainlinkIsBroken(_chainlinkResponse, _prevChainlinkResponse) ||
_chainlinkIsFrozen(_chainlinkResponse)
)
{
returnfalse;
}
return _bothOraclesSimilarPrice(_chainlinkResponse, _bandResponse);
}
function_bothOraclesSimilarPrice( ChainlinkResponse memory _chainlinkResponse, BandResponse memory _bandResponse) internalpurereturns (bool) {
uint scaledChainlinkPrice = _chainlinkResponse.answer;
uint scaledBandPrice = _bandResponse.value;
// Get the relative price difference between the oracles. Use the lower price as the denominator, i.e. the reference for the calculation.uint minPrice = (scaledBandPrice < scaledChainlinkPrice) ? scaledBandPrice : scaledChainlinkPrice;
uint maxPrice = (scaledBandPrice >= scaledChainlinkPrice) ? scaledBandPrice : scaledChainlinkPrice;
uint percentPriceDifference = maxPrice.sub(minPrice).mul(DECIMAL_PRECISION).div(minPrice);
/*
* Return true if the relative price difference is <= 3%: if so, we assume both oracles are probably reporting
* the honest market price, as it is unlikely that both have been broken/hacked and are still in-sync.
*/return percentPriceDifference <= MAX_PRICE_DIFFERENCE_BETWEEN_ORACLES;
}
function_scaleChainlinkPriceByDigits(uint _price, uint _answerDigits) internalpurereturns (uint) {
/*
* Convert the price returned by the Chainlink oracle to an 18-digit decimal for use by Liquity.
* At date of Liquity launch, Chainlink uses an 8-digit price, but we also handle the possibility of
* future changes.
*
*/uint price;
if (_answerDigits >= TARGET_DIGITS) {
// Scale the returned price value down to Liquity's target precision
price = _price.div(10** (_answerDigits - TARGET_DIGITS));
}
elseif (_answerDigits < TARGET_DIGITS) {
// Scale the returned price value up to Liquity's target precision
price = _price.mul(10** (TARGET_DIGITS - _answerDigits));
}
return price;
}
// --- Oracle response wrapper functions ---function_getCurrentBandResponse() internalviewreturns (BandResponse memory bandResponse) {
try bandOracle.getReferenceData(bandBase, bandQuote) returns
(
uint256 value,
uint256 lastUpdatedBase,
uint256 lastUpdatedQuote
)
{
// If call to Band succeeds, return the response and success = true
bandResponse.value= value;
bandResponse.timestamp = lastUpdatedBase < lastUpdatedQuote ? lastUpdatedBase : lastUpdatedQuote;
bandResponse.success =true;
return (bandResponse);
}catch {
// If call to Band reverts, return a zero response with success = falsereturn (bandResponse);
}
}
function_getCurrentChainlinkResponse() internalviewreturns (ChainlinkResponse memory chainlinkResponse) {
// First, try to get current decimal precision:try chainlinkOracle.decimals() returns (uint8 decimals) {
// If call to Chainlink succeeds, record the current decimal precision
chainlinkResponse.decimals = decimals;
} catch {
// If call to Chainlink aggregator reverts, return a zero response with success = falsereturn chainlinkResponse;
}
// Secondly, try to get latest price data:try chainlinkOracle.latestRoundData() returns
(
uint80 roundId,
int256 answer,
uint256/* startedAt */,
uint256 timestamp,
uint80/* answeredInRound */
)
{
// If call to Chainlink succeeds, return the response and success = true
chainlinkResponse.roundId = roundId;
chainlinkResponse.answer = _scaleChainlinkPriceByDigits(uint256(answer), chainlinkResponse.decimals);
chainlinkResponse.timestamp = timestamp;
chainlinkResponse.success =true;
return chainlinkResponse;
} catch {
// If call to Chainlink aggregator reverts, return a zero response with success = falsereturn chainlinkResponse;
}
}
function_getPrevChainlinkResponse(uint80 _currentRoundId, uint8 _currentDecimals) internalviewreturns (ChainlinkResponse memory prevChainlinkResponse) {
/*
* NOTE: Chainlink only offers a current decimals() value - there is no way to obtain the decimal precision used in a
* previous round. We assume the decimals used in the previous round are the same as the current round.
*/// Try to get the price data from the previous round:try chainlinkOracle.getRoundData(_currentRoundId -1) returns
(
uint80 roundId,
int256 answer,
uint256/* startedAt */,
uint256 timestamp,
uint80/* answeredInRound */
)
{
// If call to Chainlink succeeds, return the response and success = true
prevChainlinkResponse.roundId = roundId;
prevChainlinkResponse.answer = _scaleChainlinkPriceByDigits(uint256(answer), _currentDecimals);
prevChainlinkResponse.timestamp = timestamp;
prevChainlinkResponse.decimals = _currentDecimals;
prevChainlinkResponse.success =true;
return prevChainlinkResponse;
} catch {
// If call to Chainlink aggregator reverts, return a zero response with success = falsereturn prevChainlinkResponse;
}
}
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @title Proxy
* @dev Implements delegation of calls to other contracts, with proper
* forwarding of return values and bubbling of failures.
* It defines a fallback function that delegates all calls to the address
* returned by the abstract _implementation() internal function.
*/abstractcontractProxy{
/**
* @dev Fallback function.
* Implemented entirely in `_fallback`.
*/fallback() externalpayable{
_fallback();
}
/**
* @return The Address of the implementation.
*/function_implementation() internalviewvirtualreturns (address);
/**
* @dev Delegates execution to an implementation contract.
* This is a low level function that doesn't return to its internal call site.
* It will return to the external caller whatever the implementation returns.
* @param implementation Address to delegate.
*/function_delegate(address implementation) internal{
//solium-disable-next-lineassembly {
// Copy msg.data. We take full control of memory in this inline assembly// block because it will not return to Solidity code. We overwrite the// Solidity scratch pad at memory position 0.calldatacopy(0, 0, calldatasize())
// Call the implementation.// out and outsize are 0 because we don't know the size yet.let result :=delegatecall(gas(), implementation, 0, calldatasize(), 0, 0)
// Copy the returned data.returndatacopy(0, 0, returndatasize())
switch result
// delegatecall returns 0 on error.case0 {
revert(0, returndatasize())
}
default {
return(0, returndatasize())
}
}
}
/**
* @dev Function that is run as the first thing in the fallback function.
* Can be redefined in derived contracts to add functionality.
* Redefinitions must call super._willFallback().
*/function_willFallback() internalvirtual{}
/**
* @dev fallback implementation.
* Extracted to enable manual triggering.
*/function_fallback() internal{
_willFallback();
_delegate(_implementation());
}
}
Contract Source Code
File 75 of 91: ReserveConfiguration.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
*/libraryReserveConfiguration{
uint256constant LTV_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000; // prettier-ignoreuint256constant LIQUIDATION_THRESHOLD_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000FFFF; // prettier-ignoreuint256constant LIQUIDATION_BONUS_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000FFFFFFFF; // prettier-ignoreuint256constant DECIMALS_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF00FFFFFFFFFFFF; // prettier-ignoreuint256constant ACTIVE_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFF; // prettier-ignoreuint256constant FROZEN_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFDFFFFFFFFFFFFFF; // prettier-ignoreuint256constant BORROWING_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFBFFFFFFFFFFFFFF; // prettier-ignoreuint256constant STABLE_BORROWING_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFFFFF; // prettier-ignoreuint256constant RESERVE_FACTOR_MASK =0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000FFFFFFFFFFFFFFFF; // prettier-ignore/// @dev For the LTV, the start bit is 0 (up to 15), hence no bitshifting is neededuint256constant LIQUIDATION_THRESHOLD_START_BIT_POSITION =16;
uint256constant LIQUIDATION_BONUS_START_BIT_POSITION =32;
uint256constant RESERVE_DECIMALS_START_BIT_POSITION =48;
uint256constant IS_ACTIVE_START_BIT_POSITION =56;
uint256constant IS_FROZEN_START_BIT_POSITION =57;
uint256constant BORROWING_ENABLED_START_BIT_POSITION =58;
uint256constant STABLE_BORROWING_ENABLED_START_BIT_POSITION =59;
uint256constant RESERVE_FACTOR_START_BIT_POSITION =64;
uint256constant MAX_VALID_LTV =65535;
uint256constant MAX_VALID_LIQUIDATION_THRESHOLD =65535;
uint256constant MAX_VALID_LIQUIDATION_BONUS =65535;
uint256constant MAX_VALID_DECIMALS =255;
uint256constant MAX_VALID_RESERVE_FACTOR =65535;
/**
* @dev Sets the Loan to Value of the reserve
* @param self The reserve configuration
* @param ltv the new ltv
**/functionsetLtv(DataTypes.ReserveConfigurationMap memoryself, uint256 ltv) internalpure{
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
**/functiongetLtv(DataTypes.ReserveConfigurationMap storageself) internalviewreturns (uint256) {
returnself.data &~LTV_MASK;
}
/**
* @dev Sets the liquidation threshold of the reserve
* @param self The reserve configuration
* @param threshold The new liquidation threshold
**/functionsetLiquidationThreshold(DataTypes.ReserveConfigurationMap memoryself, uint256 threshold)
internalpure{
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
**/functiongetLiquidationThreshold(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functionsetLiquidationBonus(DataTypes.ReserveConfigurationMap memoryself, uint256 bonus)
internalpure{
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
**/functiongetLiquidationBonus(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functionsetDecimals(DataTypes.ReserveConfigurationMap memoryself, uint256 decimals)
internalpure{
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
**/functiongetDecimals(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functionsetActive(DataTypes.ReserveConfigurationMap memoryself, bool active) internalpure{
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
**/functiongetActive(DataTypes.ReserveConfigurationMap storageself) internalviewreturns (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
**/functionsetFrozen(DataTypes.ReserveConfigurationMap memoryself, bool frozen) internalpure{
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
**/functiongetFrozen(DataTypes.ReserveConfigurationMap storageself) internalviewreturns (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
**/functionsetBorrowingEnabled(DataTypes.ReserveConfigurationMap memoryself, bool enabled)
internalpure{
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
**/functiongetBorrowingEnabled(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functionsetStableRateBorrowingEnabled(
DataTypes.ReserveConfigurationMap memoryself,
bool enabled
) internalpure{
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
**/functiongetStableRateBorrowingEnabled(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functionsetReserveFactor(DataTypes.ReserveConfigurationMap memoryself, uint256 reserveFactor)
internalpure{
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
**/functiongetReserveFactor(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functiongetFlags(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functiongetParams(DataTypes.ReserveConfigurationMap storageself)
internalviewreturns (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
**/functiongetParamsMemory(DataTypes.ReserveConfigurationMap memoryself)
internalpurereturns (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
**/functiongetFlagsMemory(DataTypes.ReserveConfigurationMap memoryself)
internalpurereturns (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
);
}
}
Contract Source Code
File 76 of 91: ReserveLogic.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
*/libraryReserveLogic{
usingSafeMathforuint256;
usingWadRayMathforuint256;
usingPercentageMathforuint256;
usingSafeERC20forIERC20;
/**
* @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
**/eventReserveDataUpdated(addressindexed asset,
uint256 liquidityRate,
uint256 stableBorrowRate,
uint256 variableBorrowRate,
uint256 liquidityIndex,
uint256 variableBorrowIndex
);
usingReserveLogicforDataTypes.ReserveData;
usingReserveConfigurationforDataTypes.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
**/functiongetNormalizedIncome(DataTypes.ReserveData storage reserve)
internalviewreturns (uint256)
{
uint40 timestamp = reserve.lastUpdateTimestamp;
//solium-disable-next-lineif (timestamp ==uint40(block.timestamp)) {
//if the index was updated in the same block, no need to perform any calculationreturn 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
**/functiongetNormalizedDebt(DataTypes.ReserveData storage reserve)
internalviewreturns (uint256)
{
uint40 timestamp = reserve.lastUpdateTimestamp;
//solium-disable-next-lineif (timestamp ==uint40(block.timestamp)) {
//if the index was updated in the same block, no need to perform any calculationreturn 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
**/functionupdateState(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
**/functioncumulateToLiquidityIndex(
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
**/functioninit(
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;
}
structUpdateInterestRatesLocalVars {
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)
**/functionupdateInterestRates(
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
);
}
structMintToTreasuryLocalVars {
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
) internalreturns (uint256, uint256) {
uint256 currentLiquidityRate = reserve.currentLiquidityRate;
uint256 newLiquidityIndex = liquidityIndex;
uint256 newVariableBorrowIndex = variableBorrowIndex;
//only cumulating if there is any income being producedif (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 accumulatingif (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);
}
}
Contract Source Code
File 77 of 91: SafeERC20.sol
// SPDX-License-Identifier: MITpragmasolidity 0.7.6;import {IERC20} from'./IERC20.sol';
import {SafeMath} from'./SafeMath.sol';
import {Address} from'./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.
*/librarySafeERC20{
usingSafeMathforuint256;
usingAddressforaddress;
functionsafeTransfer(
IERC20 token,
address to,
uint256 value
) internal{
callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
functionsafeTransferFrom(
IERC20 token,
addressfrom,
address to,
uint256 value
) internal{
callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
functionsafeApprove(
IERC20 token,
address spender,
uint256 value
) internal{
require(
(value ==0) || (token.allowance(address(this), spender) ==0),
'SafeERC20: approve from non-zero to non-zero allowance'
);
callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
functioncallOptionalReturn(IERC20 token, bytesmemory data) private{
require(address(token).isContract(), 'SafeERC20: call to non-contract');
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytesmemory returndata) =address(token).call(data);
require(success, 'SafeERC20: low-level call failed');
if (returndata.length>0) {
// Return data is optional// solhint-disable-next-line max-line-lengthrequire(abi.decode(returndata, (bool)), 'SafeERC20: ERC20 operation did not succeed');
}
}
}
Contract Source Code
File 78 of 91: SafeMath.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;/**
* @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.
*/librarySafeMath{
/**
* @dev Returns the addition of two unsigned integers, reverting on
* overflow.
*
* Counterpart to Solidity's `+` operator.
*
* Requirements:
* - Addition cannot overflow.
*/functionadd(uint256 a, uint256 b) internalpurereturns (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.
*/functionsub(uint256 a, uint256 b) internalpurereturns (uint256) {
return sub(a, b, 'SafeMath: subtraction overflow');
}
/**
* @dev Returns the subtraction of two unsigned integers, reverting with custom message on
* overflow (when the result is negative).
*
* Counterpart to Solidity's `-` operator.
*
* Requirements:
* - Subtraction cannot overflow.
*/functionsub(uint256 a,
uint256 b,
stringmemory errorMessage
) internalpurereturns (uint256) {
require(b <= a, errorMessage);
uint256 c = a - b;
return c;
}
/**
* @dev Returns the multiplication of two unsigned integers, reverting on
* overflow.
*
* Counterpart to Solidity's `*` operator.
*
* Requirements:
* - Multiplication cannot overflow.
*/functionmul(uint256 a, uint256 b) internalpurereturns (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/522if (a ==0) {
return0;
}
uint256 c = a * b;
require(c / a == b, 'SafeMath: multiplication overflow');
return c;
}
/**
* @dev Returns the integer division of two unsigned integers. Reverts on
* division by zero. The result is rounded towards zero.
*
* Counterpart to Solidity's `/` operator. Note: this function uses a
* `revert` opcode (which leaves remaining gas untouched) while Solidity
* uses an invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*/functiondiv(uint256 a, uint256 b) internalpurereturns (uint256) {
return div(a, b, 'SafeMath: division by zero');
}
/**
* @dev Returns the integer division of two unsigned integers. Reverts with custom message on
* division by zero. The result is rounded towards zero.
*
* Counterpart to Solidity's `/` operator. Note: this function uses a
* `revert` opcode (which leaves remaining gas untouched) while Solidity
* uses an invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*/functiondiv(uint256 a,
uint256 b,
stringmemory errorMessage
) internalpurereturns (uint256) {
// Solidity only automatically asserts when dividing by 0require(b >0, errorMessage);
uint256 c = a / b;
// assert(a == b * c + a % b); // There is no case in which this doesn't holdreturn c;
}
/**
* @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
* Reverts when dividing by zero.
*
* Counterpart to Solidity's `%` operator. This function uses a `revert`
* opcode (which leaves remaining gas untouched) while Solidity uses an
* invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*/functionmod(uint256 a, uint256 b) internalpurereturns (uint256) {
return mod(a, b, 'SafeMath: modulo by zero');
}
/**
* @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
* Reverts with custom message when dividing by zero.
*
* Counterpart to Solidity's `%` operator. This function uses a `revert`
* opcode (which leaves remaining gas untouched) while Solidity uses an
* invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*/functionmod(uint256 a,
uint256 b,
stringmemory errorMessage
) internalpurereturns (uint256) {
require(b !=0, errorMessage);
return a % b;
}
}
Contract Source Code
File 79 of 91: StableAndVariableTokensHelper.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;import {StableDebtToken} from'../protocol/tokenization/StableDebtToken.sol';
import {VariableDebtToken} from'../protocol/tokenization/VariableDebtToken.sol';
import {LendingRateOracle} from'../mocks/oracle/LendingRateOracle.sol';
import {Ownable} from'../dependencies/openzeppelin/contracts/Ownable.sol';
contractStableAndVariableTokensHelperisOwnable{
addresspayableprivate pool;
addressprivate addressesProvider;
eventdeployedContracts(address stableToken, address variableToken);
constructor(addresspayable _pool, address _addressesProvider) {
pool = _pool;
addressesProvider = _addressesProvider;
}
functioninitDeployment(address[] calldata tokens, string[] calldata symbols) externalonlyOwner{
require(tokens.length== symbols.length, 'Arrays not same length');
require(pool !=address(0), 'Pool can not be zero address');
for (uint256 i =0; i < tokens.length; i++) {
emit deployedContracts(address(new StableDebtToken()), address(new VariableDebtToken()));
}
}
functionsetOracleBorrowRates(address[] calldata assets,
uint256[] calldata rates,
address oracle
) externalonlyOwner{
require(assets.length== rates.length, 'Arrays not same length');
for (uint256 i =0; i < assets.length; i++) {
// LendingRateOracle owner must be this contract
LendingRateOracle(oracle).setMarketBorrowRate(assets[i], rates[i]);
}
}
functionsetOracleOwnership(address oracle, address admin) externalonlyOwner{
require(admin !=address(0), 'owner can not be zero');
require(LendingRateOracle(oracle).owner() ==address(this), 'helper is not owner');
LendingRateOracle(oracle).transferOwnership(admin);
}
}
Contract Source Code
File 80 of 91: StableDebtToken.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {DebtTokenBase} from'./base/DebtTokenBase.sol';
import {MathUtils} from'../libraries/math/MathUtils.sol';
import {WadRayMath} from'../libraries/math/WadRayMath.sol';
import {IStableDebtToken} from'../../interfaces/IStableDebtToken.sol';
import {ILendingPool} from'../../interfaces/ILendingPool.sol';
import {IAaveIncentivesController} from'../../interfaces/IAaveIncentivesController.sol';
import {Errors} from'../libraries/helpers/Errors.sol';
import {SafeMath} from'../../dependencies/openzeppelin/contracts/SafeMath.sol';
/**
* @title StableDebtToken
* @notice Implements a stable debt token to track the borrowing positions of users
* at stable rate mode
* @author Aave
**/contractStableDebtTokenisIStableDebtToken, DebtTokenBase{
usingWadRayMathforuint256;
usingSafeMathforuint256;
uint256publicconstant DEBT_TOKEN_REVISION =0x1;
uint256internal _avgStableRate;
mapping(address=>uint40) internal _timestamps;
mapping(address=>uint256) internal _usersStableRate;
uint40internal _totalSupplyTimestamp;
IAaveIncentivesController internal _incentivesController;
/**
* @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 incentivesController The smart contract managing potential incentives distribution
* @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
*/functioninitialize(
ILendingPool pool,
address underlyingAsset,
IAaveIncentivesController incentivesController,
uint8 debtTokenDecimals,
stringmemory debtTokenName,
stringmemory debtTokenSymbol,
bytescalldata params
) publicoverrideinitializer{
_setName(debtTokenName);
_setSymbol(debtTokenSymbol);
_setDecimals(debtTokenDecimals);
_pool = pool;
_underlyingAsset = underlyingAsset;
_incentivesController = incentivesController;
emit Initialized(
underlyingAsset,
address(pool),
address(incentivesController),
debtTokenDecimals,
debtTokenName,
debtTokenSymbol,
params
);
}
/**
* @dev Gets the revision of the stable debt token implementation
* @return The debt token implementation revision
**/functiongetRevision() internalpurevirtualoverridereturns (uint256) {
return DEBT_TOKEN_REVISION;
}
/**
* @dev Returns the average stable rate across all the stable rate debt
* @return the average stable rate
**/functiongetAverageStableRate() externalviewvirtualoverridereturns (uint256) {
return _avgStableRate;
}
/**
* @dev Returns the timestamp of the last user action
* @return The last update timestamp
**/functiongetUserLastUpdated(address user) externalviewvirtualoverridereturns (uint40) {
return _timestamps[user];
}
/**
* @dev Returns the stable rate of the user
* @param user The address of the user
* @return The stable rate of user
**/functiongetUserStableRate(address user) externalviewvirtualoverridereturns (uint256) {
return _usersStableRate[user];
}
/**
* @dev Calculates the current user debt balance
* @return The accumulated debt of the user
**/functionbalanceOf(address account) publicviewvirtualoverridereturns (uint256) {
uint256 accountBalance =super.balanceOf(account);
uint256 stableRate = _usersStableRate[account];
if (accountBalance ==0) {
return0;
}
uint256 cumulatedInterest =
MathUtils.calculateCompoundedInterest(stableRate, _timestamps[account]);
return accountBalance.rayMul(cumulatedInterest);
}
structMintLocalVars {
uint256 previousSupply;
uint256 nextSupply;
uint256 amountInRay;
uint256 newStableRate;
uint256 currentAvgStableRate;
}
/**
* @dev Mints debt token to the `onBehalfOf` address.
* - Only callable by the LendingPool
* - 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
**/functionmint(address user,
address onBehalfOf,
uint256 amount,
uint256 rate
) externaloverrideonlyLendingPoolreturns (bool) {
MintLocalVars memory vars;
if (user != onBehalfOf) {
_decreaseBorrowAllowance(onBehalfOf, user, amount);
}
(, uint256 currentBalance, uint256 balanceIncrease) = _calculateBalanceIncrease(onBehalfOf);
vars.previousSupply = totalSupply();
vars.currentAvgStableRate = _avgStableRate;
vars.nextSupply = _totalSupply = vars.previousSupply.add(amount);
vars.amountInRay = amount.wadToRay();
vars.newStableRate = _usersStableRate[onBehalfOf]
.rayMul(currentBalance.wadToRay())
.add(vars.amountInRay.rayMul(rate))
.rayDiv(currentBalance.add(amount).wadToRay());
require(vars.newStableRate <=type(uint128).max, Errors.SDT_STABLE_DEBT_OVERFLOW);
_usersStableRate[onBehalfOf] = vars.newStableRate;
//solium-disable-next-line
_totalSupplyTimestamp = _timestamps[onBehalfOf] =uint40(block.timestamp);
// Calculates the updated average stable rate
vars.currentAvgStableRate = _avgStableRate = vars
.currentAvgStableRate
.rayMul(vars.previousSupply.wadToRay())
.add(rate.rayMul(vars.amountInRay))
.rayDiv(vars.nextSupply.wadToRay());
_mint(onBehalfOf, amount.add(balanceIncrease), vars.previousSupply);
emit Transfer(address(0), onBehalfOf, amount);
emit Mint(
user,
onBehalfOf,
amount,
currentBalance,
balanceIncrease,
vars.newStableRate,
vars.currentAvgStableRate,
vars.nextSupply
);
return currentBalance ==0;
}
/**
* @dev Burns debt of `user`
* @param user The address of the user getting his debt burned
* @param amount The amount of debt tokens getting burned
**/functionburn(address user, uint256 amount) externaloverrideonlyLendingPool{
(, uint256 currentBalance, uint256 balanceIncrease) = _calculateBalanceIncrease(user);
uint256 previousSupply = totalSupply();
uint256 newAvgStableRate =0;
uint256 nextSupply =0;
uint256 userStableRate = _usersStableRate[user];
// Since the total supply and each single user debt accrue separately,// there might be accumulation errors so that the last borrower repaying// mght actually try to repay more than the available debt supply.// In this case we simply set the total supply and the avg stable rate to 0if (previousSupply <= amount) {
_avgStableRate =0;
_totalSupply =0;
} else {
nextSupply = _totalSupply = previousSupply.sub(amount);
uint256 firstTerm = _avgStableRate.rayMul(previousSupply.wadToRay());
uint256 secondTerm = userStableRate.rayMul(amount.wadToRay());
// For the same reason described above, when the last user is repaying it might// happen that user rate * user balance > avg rate * total supply. In that case,// we simply set the avg rate to 0if (secondTerm >= firstTerm) {
newAvgStableRate = _avgStableRate = _totalSupply =0;
} else {
newAvgStableRate = _avgStableRate = firstTerm.sub(secondTerm).rayDiv(nextSupply.wadToRay());
}
}
if (amount == currentBalance) {
_usersStableRate[user] =0;
_timestamps[user] =0;
} else {
//solium-disable-next-line
_timestamps[user] =uint40(block.timestamp);
}
//solium-disable-next-line
_totalSupplyTimestamp =uint40(block.timestamp);
if (balanceIncrease > amount) {
uint256 amountToMint = balanceIncrease.sub(amount);
_mint(user, amountToMint, previousSupply);
emit Mint(
user,
user,
amountToMint,
currentBalance,
balanceIncrease,
userStableRate,
newAvgStableRate,
nextSupply
);
} else {
uint256 amountToBurn = amount.sub(balanceIncrease);
_burn(user, amountToBurn, previousSupply);
emit Burn(user, amountToBurn, currentBalance, balanceIncrease, newAvgStableRate, nextSupply);
}
emit Transfer(user, address(0), amount);
}
/**
* @dev Calculates the increase in balance since the last user interaction
* @param user The address of the user for which the interest is being accumulated
* @return The previous principal balance, the new principal balance and the balance increase
**/function_calculateBalanceIncrease(address user)
internalviewreturns (uint256,
uint256,
uint256)
{
uint256 previousPrincipalBalance =super.balanceOf(user);
if (previousPrincipalBalance ==0) {
return (0, 0, 0);
}
// Calculation of the accrued interest since the last accumulationuint256 balanceIncrease = balanceOf(user).sub(previousPrincipalBalance);
return (
previousPrincipalBalance,
previousPrincipalBalance.add(balanceIncrease),
balanceIncrease
);
}
/**
* @dev Returns the principal and total supply, the average borrow rate and the last supply update timestamp
**/functiongetSupplyData()
publicviewoverridereturns (uint256,
uint256,
uint256,
uint40)
{
uint256 avgRate = _avgStableRate;
return (super.totalSupply(), _calcTotalSupply(avgRate), avgRate, _totalSupplyTimestamp);
}
/**
* @dev Returns the the total supply and the average stable rate
**/functiongetTotalSupplyAndAvgRate() publicviewoverridereturns (uint256, uint256) {
uint256 avgRate = _avgStableRate;
return (_calcTotalSupply(avgRate), avgRate);
}
/**
* @dev Returns the total supply
**/functiontotalSupply() publicviewoverridereturns (uint256) {
return _calcTotalSupply(_avgStableRate);
}
/**
* @dev Returns the timestamp at which the total supply was updated
**/functiongetTotalSupplyLastUpdated() publicviewoverridereturns (uint40) {
return _totalSupplyTimestamp;
}
/**
* @dev Returns the principal debt balance of the user from
* @param user The user's address
* @return The debt balance of the user since the last burn/mint action
**/functionprincipalBalanceOf(address user) externalviewvirtualoverridereturns (uint256) {
returnsuper.balanceOf(user);
}
/**
* @dev Returns the address of the underlying asset of this aToken (E.g. WETH for aWETH)
**/functionUNDERLYING_ASSET_ADDRESS() publicviewreturns (address) {
return _underlyingAsset;
}
/**
* @dev Returns the address of the lending pool where this aToken is used
**/functionPOOL() publicviewreturns (ILendingPool) {
return _pool;
}
/**
* @dev Returns the address of the incentives controller contract
**/functiongetIncentivesController() externalviewoverridereturns (IAaveIncentivesController) {
return _getIncentivesController();
}
/**
* @dev For internal usage in the logic of the parent contracts
**/function_getIncentivesController() internalviewoverridereturns (IAaveIncentivesController) {
return _incentivesController;
}
/**
* @dev For internal usage in the logic of the parent contracts
**/function_getUnderlyingAssetAddress() internalviewoverridereturns (address) {
return _underlyingAsset;
}
/**
* @dev For internal usage in the logic of the parent contracts
**/function_getLendingPool() internalviewoverridereturns (ILendingPool) {
return _pool;
}
/**
* @dev Calculates the total supply
* @param avgRate The average rate at which the total supply increases
* @return The debt balance of the user since the last burn/mint action
**/function_calcTotalSupply(uint256 avgRate) internalviewvirtualreturns (uint256) {
uint256 principalSupply =super.totalSupply();
if (principalSupply ==0) {
return0;
}
uint256 cumulatedInterest =
MathUtils.calculateCompoundedInterest(avgRate, _totalSupplyTimestamp);
return principalSupply.rayMul(cumulatedInterest);
}
/**
* @dev Mints stable debt tokens to an user
* @param account The account receiving the debt tokens
* @param amount The amount being minted
* @param oldTotalSupply the total supply before the minting event
**/function_mint(address account,
uint256 amount,
uint256 oldTotalSupply
) internal{
uint256 oldAccountBalance = _balances[account];
_balances[account] = oldAccountBalance.add(amount);
if (address(_incentivesController) !=address(0)) {
_incentivesController.handleAction(account, oldAccountBalance, oldTotalSupply);
}
}
/**
* @dev Burns stable debt tokens of an user
* @param account The user getting his debt burned
* @param amount The amount being burned
* @param oldTotalSupply The total supply before the burning event
**/function_burn(address account,
uint256 amount,
uint256 oldTotalSupply
) internal{
uint256 oldAccountBalance = _balances[account];
_balances[account] = oldAccountBalance.sub(amount, Errors.SDT_BURN_EXCEEDS_BALANCE);
if (address(_incentivesController) !=address(0)) {
_incentivesController.handleAction(account, oldAccountBalance, oldTotalSupply);
}
}
}
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import'./BaseUpgradeabilityProxy.sol';
/**
* @title UpgradeabilityProxy
* @dev Extends BaseUpgradeabilityProxy with a constructor for initializing
* implementation and init data.
*/contractUpgradeabilityProxyisBaseUpgradeabilityProxy{
/**
* @dev Contract constructor.
* @param _logic Address of the initial implementation.
* @param _data Data to send as msg.data to the implementation to initialize the proxied contract.
* It should include the signature and the parameters of the function to be called, as described in
* https://solidity.readthedocs.io/en/v0.4.24/abi-spec.html#function-selector-and-argument-encoding.
* This parameter is optional, if no data is given the initialization call to proxied contract will be skipped.
*/constructor(address _logic, bytesmemory _data) payable{
assert(IMPLEMENTATION_SLOT ==bytes32(uint256(keccak256('eip1967.proxy.implementation')) -1));
_setImplementation(_logic);
if (_data.length>0) {
(bool success, ) = _logic.delegatecall(_data);
require(success);
}
}
}
Contract Source Code
File 85 of 91: UserConfiguration.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
*/libraryUserConfiguration{
uint256internalconstant 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
**/functionsetBorrowing(
DataTypes.UserConfigurationMap storageself,
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
**/functionsetUsingAsCollateral(
DataTypes.UserConfigurationMap storageself,
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
**/functionisUsingAsCollateralOrBorrowing(
DataTypes.UserConfigurationMap memoryself,
uint256 reserveIndex
) internalpurereturns (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
**/functionisBorrowing(DataTypes.UserConfigurationMap memoryself, uint256 reserveIndex)
internalpurereturns (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
**/functionisUsingAsCollateral(DataTypes.UserConfigurationMap memoryself, uint256 reserveIndex)
internalpurereturns (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
**/functionisBorrowingAny(DataTypes.UserConfigurationMap memoryself) internalpurereturns (bool) {
returnself.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
**/functionisEmpty(DataTypes.UserConfigurationMap memoryself) internalpurereturns (bool) {
returnself.data ==0;
}
}
Contract Source Code
File 86 of 91: ValidationLogic.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;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 {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
*/libraryValidationLogic{
usingReserveLogicforDataTypes.ReserveData;
usingSafeMathforuint256;
usingWadRayMathforuint256;
usingPercentageMathforuint256;
usingSafeERC20forIERC20;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
usingUserConfigurationforDataTypes.UserConfigurationMap;
uint256publicconstant REBALANCE_UP_LIQUIDITY_RATE_THRESHOLD =4000;
uint256publicconstant 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 amount The amount to be deposited
*/functionvalidateDeposit(DataTypes.ReserveData storage reserve, uint256 amount) externalview{
(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);
}
/**
* @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 userConfig The user configuration
* @param reserves The addresses of the reserves
* @param reservesCount The number of reserves
* @param oracle The price oracle
*/functionvalidateWithdraw(address reserveAddress,
uint256 amount,
uint256 userBalance,
mapping(address => DataTypes.ReserveData) storage reservesData,
DataTypes.UserConfigurationMap storage userConfig,
mapping(uint256 => address) storage reserves,
uint256 reservesCount,
address oracle
) externalview{
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);
require(
GenericLogic.balanceDecreaseAllowed(
reserveAddress,
msg.sender,
amount,
reservesData,
userConfig,
reserves,
reservesCount,
oracle
),
Errors.VL_TRANSFER_NOT_ALLOWED
);
}
structValidateBorrowLocalVars {
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
*/functionvalidateBorrow(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
) externalview{
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 moderequire(
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 percentagerequire(
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 reserverequire(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 liquidityuint256 maxLoanSizeStable = vars.availableLiquidity.percentMul(maxStableLoanPercent);
require(amount <= maxLoanSizeStable, Errors.VL_AMOUNT_BIGGER_THAN_MAX_LOAN_SIZE_STABLE);
}
}
/**
* @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
*/functionvalidateRepay(
DataTypes.ReserveData storage reserve,
uint256 amountSent,
DataTypes.InterestRateMode rateMode,
address onBehalfOf,
uint256 stableDebt,
uint256 variableDebt
) externalview{
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
*/functionvalidateSwapRateMode(
DataTypes.ReserveData storage reserve,
DataTypes.UserConfigurationMap storage userConfig,
uint256 stableDebt,
uint256 variableDebt,
DataTypes.InterestRateMode currentRateMode
) externalview{
(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);
} elseif (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
*/functionvalidateRebalanceStableBorrowRate(
DataTypes.ReserveData storage reserve,
address reserveAddress,
IERC20 stableDebtToken,
IERC20 variableDebtToken,
address aTokenAddress
) externalview{
(bool isActive, , , ) = reserve.configuration.getFlags();
require(isActive, Errors.VL_NO_ACTIVE_RESERVE);
//if the usage ratio is below 95%, no rebalances are neededuint256 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
*/functionvalidateSetUseReserveAsCollateral(
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
) externalview{
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
**/functionvalidateFlashloan(address[] memory assets, uint256[] memory amounts) internalpure{
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
**/functionvalidateLiquidationCall(
DataTypes.ReserveData storage collateralReserve,
DataTypes.ReserveData storage principalReserve,
DataTypes.UserConfigurationMap storage userConfig,
uint256 userHealthFactor,
uint256 userStableDebt,
uint256 userVariableDebt
) internalviewreturns (uint256, stringmemory) {
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 liquidatedif (!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
*/functionvalidateTransfer(addressfrom,
mapping(address => DataTypes.ReserveData) storage reservesData,
DataTypes.UserConfigurationMap storage userConfig,
mapping(uint256 => address) storage reserves,
uint256 reservesCount,
address oracle
) internalview{
(, , , , uint256 healthFactor) =
GenericLogic.calculateUserAccountData(
from,
reservesData,
userConfig,
reserves,
reservesCount,
oracle
);
require(
healthFactor >= GenericLogic.HEALTH_FACTOR_LIQUIDATION_THRESHOLD,
Errors.VL_TRANSFER_NOT_ALLOWED
);
}
}
Contract Source Code
File 87 of 91: VariableDebtToken.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;import {IVariableDebtToken} from'../../interfaces/IVariableDebtToken.sol';
import {WadRayMath} from'../libraries/math/WadRayMath.sol';
import {Errors} from'../libraries/helpers/Errors.sol';
import {DebtTokenBase} from'./base/DebtTokenBase.sol';
import {ILendingPool} from'../../interfaces/ILendingPool.sol';
import {IAaveIncentivesController} from'../../interfaces/IAaveIncentivesController.sol';
import {SafeMath} from'../../dependencies/openzeppelin/contracts/SafeMath.sol';
/**
* @title VariableDebtToken
* @notice Implements a variable debt token to track the borrowing positions of users
* at variable rate mode
* @author Aave
**/contractVariableDebtTokenisDebtTokenBase, IVariableDebtToken{
usingWadRayMathforuint256;
usingSafeMathforuint256;
uint256publicconstant DEBT_TOKEN_REVISION =0x1;
IAaveIncentivesController internal _incentivesController;
/**
* @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 incentivesController The smart contract managing potential incentives distribution
* @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
*/functioninitialize(
ILendingPool pool,
address underlyingAsset,
IAaveIncentivesController incentivesController,
uint8 debtTokenDecimals,
stringmemory debtTokenName,
stringmemory debtTokenSymbol,
bytescalldata params
) publicoverrideinitializer{
_setName(debtTokenName);
_setSymbol(debtTokenSymbol);
_setDecimals(debtTokenDecimals);
_pool = pool;
_underlyingAsset = underlyingAsset;
_incentivesController = incentivesController;
emit Initialized(
underlyingAsset,
address(pool),
address(incentivesController),
debtTokenDecimals,
debtTokenName,
debtTokenSymbol,
params
);
}
/**
* @dev Gets the revision of the stable debt token implementation
* @return The debt token implementation revision
**/functiongetRevision() internalpurevirtualoverridereturns (uint256) {
return DEBT_TOKEN_REVISION;
}
/**
* @dev Calculates the accumulated debt balance of the user
* @return The debt balance of the user
**/functionbalanceOf(address user) publicviewvirtualoverridereturns (uint256) {
uint256 scaledBalance =super.balanceOf(user);
if (scaledBalance ==0) {
return0;
}
return scaledBalance.rayMul(_pool.getReserveNormalizedVariableDebt(_underlyingAsset));
}
/**
* @dev Mints debt token to the `onBehalfOf` address
* - Only callable by the LendingPool
* @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
**/functionmint(address user,
address onBehalfOf,
uint256 amount,
uint256 index
) externaloverrideonlyLendingPoolreturns (bool) {
if (user != onBehalfOf) {
_decreaseBorrowAllowance(onBehalfOf, user, amount);
}
uint256 previousBalance =super.balanceOf(onBehalfOf);
uint256 amountScaled = amount.rayDiv(index);
require(amountScaled !=0, Errors.CT_INVALID_MINT_AMOUNT);
_mint(onBehalfOf, amountScaled);
emit Transfer(address(0), onBehalfOf, amount);
emit Mint(user, onBehalfOf, amount, index);
return previousBalance ==0;
}
/**
* @dev Burns user variable debt
* - Only callable by the LendingPool
* @param user The user whose debt is getting burned
* @param amount The amount getting burned
* @param index The variable debt index of the reserve
**/functionburn(address user,
uint256 amount,
uint256 index
) externaloverrideonlyLendingPool{
uint256 amountScaled = amount.rayDiv(index);
require(amountScaled !=0, Errors.CT_INVALID_BURN_AMOUNT);
_burn(user, amountScaled);
emit Transfer(user, address(0), amount);
emit Burn(user, amount, index);
}
/**
* @dev Returns the principal debt balance of the user from
* @return The debt balance of the user since the last burn/mint action
**/functionscaledBalanceOf(address user) publicviewvirtualoverridereturns (uint256) {
returnsuper.balanceOf(user);
}
/**
* @dev Returns the total supply of the variable debt token. Represents the total debt accrued by the users
* @return The total supply
**/functiontotalSupply() publicviewvirtualoverridereturns (uint256) {
returnsuper.totalSupply().rayMul(_pool.getReserveNormalizedVariableDebt(_underlyingAsset));
}
/**
* @dev Returns the scaled total supply of the variable debt token. Represents sum(debt/index)
* @return the scaled total supply
**/functionscaledTotalSupply() publicviewvirtualoverridereturns (uint256) {
returnsuper.totalSupply();
}
/**
* @dev Returns the principal balance of the user and principal total supply.
* @param user The address of the user
* @return The principal balance of the user
* @return The principal total supply
**/functiongetScaledUserBalanceAndSupply(address user)
externalviewoverridereturns (uint256, uint256)
{
return (super.balanceOf(user), super.totalSupply());
}
/**
* @dev Returns the address of the underlying asset of this aToken (E.g. WETH for aWETH)
**/functionUNDERLYING_ASSET_ADDRESS() publicviewreturns (address) {
return _underlyingAsset;
}
/**
* @dev Returns the address of the incentives controller contract
**/functiongetIncentivesController() externalviewoverridereturns (IAaveIncentivesController) {
return _getIncentivesController();
}
/**
* @dev Returns the address of the lending pool where this aToken is used
**/functionPOOL() publicviewreturns (ILendingPool) {
return _pool;
}
function_getIncentivesController() internalviewoverridereturns (IAaveIncentivesController) {
return _incentivesController;
}
function_getUnderlyingAssetAddress() internalviewoverridereturns (address) {
return _underlyingAsset;
}
function_getLendingPool() internalviewoverridereturns (ILendingPool) {
return _pool;
}
}
Contract Source Code
File 88 of 91: VersionedInitializable.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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
*/abstractcontractVersionedInitializable{
/**
* @dev Indicates that the contract has been initialized.
*/uint256private lastInitializedRevision =0;
/**
* @dev Indicates that the contract is in the process of being initialized.
*/boolprivate initializing;
/**
* @dev Modifier to use in the initializer function of a contract.
*/modifierinitializer() {
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.
**/functiongetRevision() internalpurevirtualreturns (uint256);
/**
* @dev Returns true if and only if the function is running in the constructor
**/functionisConstructor() privateviewreturns (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-lineassembly {
cs :=extcodesize(address())
}
return cs ==0;
}
// Reserved storage space to allow for layout changes in the future.uint256[50] private ______gap;
}
Contract Source Code
File 89 of 91: WETHGateway.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;import {Ownable} from'../dependencies/openzeppelin/contracts/Ownable.sol';
import {IERC20} from'../dependencies/openzeppelin/contracts/IERC20.sol';
import {IWETH} from'./interfaces/IWETH.sol';
import {IWETHGateway} from'./interfaces/IWETHGateway.sol';
import {ILendingPool} from'../interfaces/ILendingPool.sol';
import {IAToken} from'../interfaces/IAToken.sol';
import {ReserveConfiguration} from'../protocol/libraries/configuration/ReserveConfiguration.sol';
import {UserConfiguration} from'../protocol/libraries/configuration/UserConfiguration.sol';
import {Helpers} from'../protocol/libraries/helpers/Helpers.sol';
import {DataTypes} from'../protocol/libraries/types/DataTypes.sol';
contractWETHGatewayisIWETHGateway, Ownable{
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
usingUserConfigurationforDataTypes.UserConfigurationMap;
IWETH internalimmutable WETH;
/**
* @dev Sets the WETH address and the LendingPoolAddressesProvider address. Infinite approves lending pool.
* @param weth Address of the Wrapped Ether contract
**/constructor(address weth) {
WETH = IWETH(weth);
}
functionauthorizeLendingPool(address lendingPool) externalonlyOwner{
WETH.approve(lendingPool, uint256(-1));
}
/**
* @dev deposits WETH into the reserve, using native ETH. A corresponding amount of the overlying asset (aTokens)
* is minted.
* @param lendingPool address of the targeted underlying lending pool
* @param onBehalfOf address of the user who will receive the aTokens representing the deposit
* @param referralCode integrators are assigned a referral code and can potentially receive rewards.
**/functiondepositETH(address lendingPool,
address onBehalfOf,
uint16 referralCode
) externalpayableoverride{
WETH.deposit{value: msg.value}();
ILendingPool(lendingPool).deposit(address(WETH), msg.value, onBehalfOf, referralCode);
}
/**
* @dev withdraws the WETH _reserves of msg.sender.
* @param lendingPool address of the targeted underlying lending pool
* @param amount amount of aWETH to withdraw and receive native ETH
* @param to address of the user who will receive native ETH
*/functionwithdrawETH(address lendingPool,
uint256 amount,
address to
) externaloverride{
IAToken aWETH = IAToken(ILendingPool(lendingPool).getReserveData(address(WETH)).aTokenAddress);
uint256 userBalance = aWETH.balanceOf(msg.sender);
uint256 amountToWithdraw = amount;
// if amount is equal to uint(-1), the user wants to redeem everythingif (amount ==type(uint256).max) {
amountToWithdraw = userBalance;
}
aWETH.transferFrom(msg.sender, address(this), amountToWithdraw);
ILendingPool(lendingPool).withdraw(address(WETH), amountToWithdraw, address(this));
WETH.withdraw(amountToWithdraw);
_safeTransferETH(to, amountToWithdraw);
}
/**
* @dev repays a borrow on the WETH reserve, for the specified amount (or for the whole amount, if uint256(-1) is specified).
* @param lendingPool address of the targeted underlying lending pool
* @param amount the amount to repay, or uint256(-1) if the user wants to repay everything
* @param rateMode the rate mode to repay
* @param onBehalfOf the address for which msg.sender is repaying
*/functionrepayETH(address lendingPool,
uint256 amount,
uint256 rateMode,
address onBehalfOf
) externalpayableoverride{
(uint256 stableDebt, uint256 variableDebt) =
Helpers.getUserCurrentDebtMemory(
onBehalfOf,
ILendingPool(lendingPool).getReserveData(address(WETH))
);
uint256 paybackAmount =
DataTypes.InterestRateMode(rateMode) == DataTypes.InterestRateMode.STABLE
? stableDebt
: variableDebt;
if (amount < paybackAmount) {
paybackAmount = amount;
}
require(msg.value>= paybackAmount, 'msg.value is less than repayment amount');
WETH.deposit{value: paybackAmount}();
ILendingPool(lendingPool).repay(address(WETH), msg.value, rateMode, onBehalfOf);
// refund remaining dust ethif (msg.value> paybackAmount) _safeTransferETH(msg.sender, msg.value- paybackAmount);
}
/**
* @dev borrow WETH, unwraps to ETH and send both the ETH and DebtTokens to msg.sender, via `approveDelegation` and onBehalf argument in `LendingPool.borrow`.
* @param lendingPool address of the targeted underlying lending pool
* @param amount the amount of ETH to borrow
* @param interesRateMode the interest rate mode
* @param referralCode integrators are assigned a referral code and can potentially receive rewards
*/functionborrowETH(address lendingPool,
uint256 amount,
uint256 interesRateMode,
uint16 referralCode
) externaloverride{
ILendingPool(lendingPool).borrow(
address(WETH),
amount,
interesRateMode,
referralCode,
msg.sender
);
WETH.withdraw(amount);
_safeTransferETH(msg.sender, amount);
}
/**
* @dev transfer ETH to an address, revert if it fails.
* @param to recipient of the transfer
* @param value the amount to send
*/function_safeTransferETH(address to, uint256 value) internal{
(bool success, ) = to.call{value: value}(newbytes(0));
require(success, 'ETH_TRANSFER_FAILED');
}
/**
* @dev transfer ERC20 from the utility contract, for ERC20 recovery in case of stuck tokens due
* direct transfers to the contract address.
* @param token token to transfer
* @param to recipient of the transfer
* @param amount amount to send
*/functionemergencyTokenTransfer(address token,
address to,
uint256 amount
) externalonlyOwner{
IERC20(token).transfer(to, amount);
}
/**
* @dev transfer native Ether from the utility contract, for native Ether recovery in case of stuck Ether
* due selfdestructs or transfer ether to pre-computated contract address before deployment.
* @param to recipient of the transfer
* @param amount amount to send
*/functionemergencyEtherTransfer(address to, uint256 amount) externalonlyOwner{
_safeTransferETH(to, amount);
}
/**
* @dev Get WETH address used by WETHGateway
*/functiongetWETHAddress() externalviewreturns (address) {
returnaddress(WETH);
}
/**
* @dev Only WETH contract is allowed to transfer ETH here. Prevent other addresses to send Ether to this contract.
*/receive() externalpayable{
require(msg.sender==address(WETH), 'Receive not allowed');
}
/**
* @dev Revert fallback calls
*/fallback() externalpayable{
revert('Fallback not allowed');
}
}
Contract Source Code
File 90 of 91: WadRayMath.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 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)
**/libraryWadRayMath{
uint256internalconstant WAD =1e18;
uint256internalconstant halfWAD = WAD /2;
uint256internalconstant RAY =1e27;
uint256internalconstant halfRAY = RAY /2;
uint256internalconstant WAD_RAY_RATIO =1e9;
/**
* @return One ray, 1e27
**/functionray() internalpurereturns (uint256) {
return RAY;
}
/**
* @return One wad, 1e18
**/functionwad() internalpurereturns (uint256) {
return WAD;
}
/**
* @return Half ray, 1e27/2
**/functionhalfRay() internalpurereturns (uint256) {
return halfRAY;
}
/**
* @return Half ray, 1e18/2
**/functionhalfWad() internalpurereturns (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
**/functionwadMul(uint256 a, uint256 b) internalpurereturns (uint256) {
if (a ==0|| b ==0) {
return0;
}
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
**/functionwadDiv(uint256 a, uint256 b) internalpurereturns (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
**/functionrayMul(uint256 a, uint256 b) internalpurereturns (uint256) {
if (a ==0|| b ==0) {
return0;
}
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
**/functionrayDiv(uint256 a, uint256 b) internalpurereturns (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
**/functionrayToWad(uint256 a) internalpurereturns (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
**/functionwadToRay(uint256 a) internalpurereturns (uint256) {
uint256 result = a * WAD_RAY_RATIO;
require(result / WAD_RAY_RATIO == a, Errors.MATH_MULTIPLICATION_OVERFLOW);
return result;
}
}
Contract Source Code
File 91 of 91: WalletBalanceProvider.sol
// SPDX-License-Identifier: agpl-3.0pragmasolidity 0.7.6;pragmaexperimentalABIEncoderV2;import {Address} from'../dependencies/openzeppelin/contracts/Address.sol';
import {IERC20} from'../dependencies/openzeppelin/contracts/IERC20.sol';
import {ILendingPoolAddressesProvider} from'../interfaces/ILendingPoolAddressesProvider.sol';
import {ILendingPool} from'../interfaces/ILendingPool.sol';
import {SafeERC20} from'../dependencies/openzeppelin/contracts/SafeERC20.sol';
import {ReserveConfiguration} from'../protocol/libraries/configuration/ReserveConfiguration.sol';
import {DataTypes} from'../protocol/libraries/types/DataTypes.sol';
/**
* @title WalletBalanceProvider contract
* @author Aave, influenced by https://github.com/wbobeirne/eth-balance-checker/blob/master/contracts/BalanceChecker.sol
* @notice Implements a logic of getting multiple tokens balance for one user address
* @dev NOTE: THIS CONTRACT IS NOT USED WITHIN THE AAVE PROTOCOL. It's an accessory contract used to reduce the number of calls
* towards the blockchain from the Aave backend.
**/contractWalletBalanceProvider{
usingAddressforaddresspayable;
usingAddressforaddress;
usingSafeERC20forIERC20;
usingReserveConfigurationforDataTypes.ReserveConfigurationMap;
addressconstant MOCK_ETH_ADDRESS =0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
/**
@dev Fallback function, don't accept any ETH
**/receive() externalpayable{
//only contracts can send ETH to the corerequire(msg.sender.isContract(), '22');
}
/**
@dev Check the token balance of a wallet in a token contract
Returns the balance of the token for user. Avoids possible errors:
- return 0 on non-contract address
**/functionbalanceOf(address user, address token) publicviewreturns (uint256) {
if (token == MOCK_ETH_ADDRESS) {
return user.balance; // ETH balance// check if token is actually a contract
} elseif (token.isContract()) {
return IERC20(token).balanceOf(user);
}
revert('INVALID_TOKEN');
}
/**
* @notice Fetches, for a list of _users and _tokens (ETH included with mock address), the balances
* @param users The list of users
* @param tokens The list of tokens
* @return And array with the concatenation of, for each user, his/her balances
**/functionbatchBalanceOf(address[] calldata users, address[] calldata tokens)
externalviewreturns (uint256[] memory)
{
uint256[] memory balances =newuint256[](users.length* tokens.length);
for (uint256 i =0; i < users.length; i++) {
for (uint256 j =0; j < tokens.length; j++) {
balances[i * tokens.length+ j] = balanceOf(users[i], tokens[j]);
}
}
return balances;
}
/**
@dev provides balances of user wallet for all reserves available on the pool
*/functiongetUserWalletBalances(address provider, address user)
externalviewreturns (address[] memory, uint256[] memory)
{
ILendingPool pool = ILendingPool(ILendingPoolAddressesProvider(provider).getLendingPool());
address[] memory reserves = pool.getReservesList();
address[] memory reservesWithEth =newaddress[](reserves.length+1);
for (uint256 i =0; i < reserves.length; i++) {
reservesWithEth[i] = reserves[i];
}
reservesWithEth[reserves.length] = MOCK_ETH_ADDRESS;
uint256[] memory balances =newuint256[](reservesWithEth.length);
for (uint256 j =0; j < reserves.length; j++) {
DataTypes.ReserveConfigurationMap memory configuration =
pool.getConfiguration(reservesWithEth[j]);
(bool isActive, , , ) = configuration.getFlagsMemory();
if (!isActive) {
balances[j] =0;
continue;
}
balances[j] = balanceOf(user, reservesWithEth[j]);
}
balances[reserves.length] = balanceOf(user, MOCK_ETH_ADDRESS);
return (reservesWithEth, balances);
}
}