2020-08-20 12:32:20 +00:00
|
|
|
// SPDX-License-Identifier: agpl-3.0
|
2020-11-20 10:45:20 +00:00
|
|
|
pragma solidity 0.6.12;
|
2020-06-30 12:09:28 +00:00
|
|
|
|
2020-11-26 09:21:18 +00:00
|
|
|
import {IVariableDebtToken} from '../../interfaces/IVariableDebtToken.sol';
|
2020-08-20 07:51:21 +00:00
|
|
|
import {WadRayMath} from '../libraries/math/WadRayMath.sol';
|
2020-09-30 15:40:47 +00:00
|
|
|
import {Errors} from '../libraries/helpers/Errors.sol';
|
2020-11-26 09:21:18 +00:00
|
|
|
import {DebtTokenBase} from './base/DebtTokenBase.sol';
|
2021-01-28 10:05:19 +00:00
|
|
|
import {ILendingPool} from '../../interfaces/ILendingPool.sol';
|
|
|
|
import {IAaveIncentivesController} from '../../interfaces/IAaveIncentivesController.sol';
|
2020-06-30 12:09:28 +00:00
|
|
|
|
2020-07-09 09:59:49 +00:00
|
|
|
/**
|
2020-11-26 09:21:18 +00:00
|
|
|
* @title VariableDebtToken
|
|
|
|
* @notice Implements a variable debt token to track the borrowing positions of users
|
|
|
|
* at variable rate mode
|
2020-07-13 08:54:08 +00:00
|
|
|
* @author Aave
|
|
|
|
**/
|
2020-06-30 12:09:28 +00:00
|
|
|
contract VariableDebtToken is DebtTokenBase, IVariableDebtToken {
|
|
|
|
using WadRayMath for uint256;
|
|
|
|
|
2020-08-17 19:28:50 +00:00
|
|
|
uint256 public constant DEBT_TOKEN_REVISION = 0x1;
|
|
|
|
|
2021-01-28 10:05:19 +00:00
|
|
|
ILendingPool internal _pool;
|
|
|
|
address internal _underlyingAsset;
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
function initialize(
|
|
|
|
ILendingPool pool,
|
2020-08-21 10:38:08 +00:00
|
|
|
address underlyingAsset,
|
2021-01-28 10:05:19 +00:00
|
|
|
IAaveIncentivesController incentivesController,
|
|
|
|
uint8 debtTokenDecimals,
|
|
|
|
string memory debtTokenName,
|
|
|
|
string memory debtTokenSymbol
|
|
|
|
) public override initializer {
|
|
|
|
_setName(debtTokenName);
|
|
|
|
_setSymbol(debtTokenSymbol);
|
|
|
|
_setDecimals(debtTokenDecimals);
|
|
|
|
|
|
|
|
_pool = pool;
|
|
|
|
_underlyingAsset = underlyingAsset;
|
|
|
|
_incentivesController = incentivesController;
|
|
|
|
}
|
2020-08-11 10:26:15 +00:00
|
|
|
|
2020-08-17 19:28:50 +00:00
|
|
|
/**
|
2020-11-26 09:21:18 +00:00
|
|
|
* @dev Gets the revision of the stable debt token implementation
|
|
|
|
* @return The debt token implementation revision
|
2020-08-17 19:28:50 +00:00
|
|
|
**/
|
2020-11-23 10:28:57 +00:00
|
|
|
function getRevision() internal pure virtual override returns (uint256) {
|
2020-08-17 19:28:50 +00:00
|
|
|
return DEBT_TOKEN_REVISION;
|
|
|
|
}
|
|
|
|
|
2020-07-09 14:52:57 +00:00
|
|
|
/**
|
2020-11-26 09:21:18 +00:00
|
|
|
* @dev Calculates the accumulated debt balance of the user
|
|
|
|
* @return The debt balance of the user
|
2020-07-13 08:54:08 +00:00
|
|
|
**/
|
2020-11-23 10:28:57 +00:00
|
|
|
function balanceOf(address user) public view virtual override returns (uint256) {
|
2020-09-14 13:09:16 +00:00
|
|
|
uint256 scaledBalance = super.balanceOf(user);
|
2020-09-21 14:11:14 +00:00
|
|
|
|
2020-09-10 09:25:45 +00:00
|
|
|
if (scaledBalance == 0) {
|
2020-06-30 12:09:28 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-01-28 10:05:19 +00:00
|
|
|
return scaledBalance.rayMul(_pool.getReserveNormalizedVariableDebt(_underlyingAsset));
|
2020-06-30 12:09:28 +00:00
|
|
|
}
|
2020-07-03 21:20:02 +00:00
|
|
|
|
2020-07-09 09:59:49 +00:00
|
|
|
/**
|
2020-11-26 09:21:18 +00:00
|
|
|
* @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
|
2020-07-13 08:54:08 +00:00
|
|
|
**/
|
2020-09-21 14:11:14 +00:00
|
|
|
function mint(
|
|
|
|
address user,
|
2020-11-03 18:47:57 +00:00
|
|
|
address onBehalfOf,
|
2020-09-21 14:11:14 +00:00
|
|
|
uint256 amount,
|
|
|
|
uint256 index
|
2020-10-15 12:13:46 +00:00
|
|
|
) external override onlyLendingPool returns (bool) {
|
2020-11-03 18:47:57 +00:00
|
|
|
if (user != onBehalfOf) {
|
|
|
|
_decreaseBorrowAllowance(onBehalfOf, user, amount);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint256 previousBalance = super.balanceOf(onBehalfOf);
|
2020-09-30 15:40:47 +00:00
|
|
|
uint256 amountScaled = amount.rayDiv(index);
|
2020-11-12 12:54:23 +00:00
|
|
|
require(amountScaled != 0, Errors.CT_INVALID_MINT_AMOUNT);
|
2020-09-30 15:40:47 +00:00
|
|
|
|
2020-11-03 18:47:57 +00:00
|
|
|
_mint(onBehalfOf, amountScaled);
|
2020-07-03 21:20:02 +00:00
|
|
|
|
2020-11-03 18:47:57 +00:00
|
|
|
emit Transfer(address(0), onBehalfOf, amount);
|
|
|
|
emit Mint(user, onBehalfOf, amount, index);
|
2020-10-15 12:13:46 +00:00
|
|
|
|
|
|
|
return previousBalance == 0;
|
2020-06-30 12:09:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-11-26 09:21:18 +00:00
|
|
|
* @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
|
2020-07-13 08:54:08 +00:00
|
|
|
**/
|
2020-09-21 14:11:14 +00:00
|
|
|
function burn(
|
|
|
|
address user,
|
|
|
|
uint256 amount,
|
|
|
|
uint256 index
|
|
|
|
) external override onlyLendingPool {
|
2020-09-30 15:40:47 +00:00
|
|
|
uint256 amountScaled = amount.rayDiv(index);
|
2020-11-12 12:54:23 +00:00
|
|
|
require(amountScaled != 0, Errors.CT_INVALID_BURN_AMOUNT);
|
2020-09-30 15:40:47 +00:00
|
|
|
|
|
|
|
_burn(user, amountScaled);
|
2020-07-03 21:20:02 +00:00
|
|
|
|
2020-09-15 13:20:32 +00:00
|
|
|
emit Transfer(user, address(0), amount);
|
2020-09-21 15:41:38 +00:00
|
|
|
emit Burn(user, amount, index);
|
2020-06-30 12:09:28 +00:00
|
|
|
}
|
2020-09-10 10:51:52 +00:00
|
|
|
|
2020-09-10 11:05:02 +00:00
|
|
|
/**
|
2020-09-21 14:11:14 +00:00
|
|
|
* @dev Returns the principal debt balance of the user from
|
|
|
|
* @return The debt balance of the user since the last burn/mint action
|
|
|
|
**/
|
2020-11-23 10:28:57 +00:00
|
|
|
function scaledBalanceOf(address user) public view virtual override returns (uint256) {
|
2020-09-10 11:05:02 +00:00
|
|
|
return super.balanceOf(user);
|
|
|
|
}
|
|
|
|
|
2020-09-14 08:43:30 +00:00
|
|
|
/**
|
2020-09-21 14:11:14 +00:00
|
|
|
* @dev Returns the total supply of the variable debt token. Represents the total debt accrued by the users
|
2020-11-26 09:21:18 +00:00
|
|
|
* @return The total supply
|
2020-09-21 14:11:14 +00:00
|
|
|
**/
|
2020-11-23 10:28:57 +00:00
|
|
|
function totalSupply() public view virtual override returns (uint256) {
|
2021-01-28 10:05:19 +00:00
|
|
|
return super.totalSupply().rayMul(_pool.getReserveNormalizedVariableDebt(_underlyingAsset));
|
2020-09-10 11:05:02 +00:00
|
|
|
}
|
2020-09-14 08:43:30 +00:00
|
|
|
|
|
|
|
/**
|
2020-11-23 17:52:52 +00:00
|
|
|
* @dev Returns the scaled total supply of the variable debt token. Represents sum(debt/index)
|
2020-09-21 14:11:14 +00:00
|
|
|
* @return the scaled total supply
|
|
|
|
**/
|
2020-11-23 10:28:57 +00:00
|
|
|
function scaledTotalSupply() public view virtual override returns (uint256) {
|
2020-09-14 08:43:30 +00:00
|
|
|
return super.totalSupply();
|
|
|
|
}
|
2020-09-21 15:41:38 +00:00
|
|
|
|
2020-10-08 13:41:48 +00:00
|
|
|
/**
|
2020-11-26 13:58:42 +00:00
|
|
|
* @dev Returns the principal balance of the user and principal total supply.
|
2020-11-26 09:21:18 +00:00
|
|
|
* @param user The address of the user
|
|
|
|
* @return The principal balance of the user
|
|
|
|
* @return The principal total supply
|
2020-09-21 15:41:38 +00:00
|
|
|
**/
|
2020-10-08 13:41:48 +00:00
|
|
|
function getScaledUserBalanceAndSupply(address user)
|
|
|
|
external
|
|
|
|
view
|
2020-11-23 10:28:57 +00:00
|
|
|
override
|
2020-10-08 13:41:48 +00:00
|
|
|
returns (uint256, uint256)
|
|
|
|
{
|
2020-09-21 15:41:38 +00:00
|
|
|
return (super.balanceOf(user), super.totalSupply());
|
2020-06-30 12:09:28 +00:00
|
|
|
}
|
2021-01-28 10:05:19 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev Returns the address of the underlying asset of this aToken (E.g. WETH for aWETH)
|
|
|
|
**/
|
|
|
|
function UNDERLYING_ASSET_ADDRESS() public view returns (address) {
|
|
|
|
return _underlyingAsset;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev Returns the address of the incentives controller contract
|
|
|
|
**/
|
|
|
|
function getIncentivesController() external view override returns (IAaveIncentivesController) {
|
|
|
|
return _getIncentivesController();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev Returns the address of the lending pool where this aToken is used
|
|
|
|
**/
|
|
|
|
function POOL() public view returns (ILendingPool) {
|
|
|
|
return _pool;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _getIncentivesController() internal view override returns (IAaveIncentivesController) {
|
|
|
|
return _incentivesController;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _getUnderlyingAssetAddress() internal view override returns (address) {
|
|
|
|
return _underlyingAsset;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _getLendingPool() internal view override returns (ILendingPool) {
|
|
|
|
return _pool;
|
|
|
|
}
|
2020-06-30 12:09:28 +00:00
|
|
|
}
|