Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat/enable disable flashloan config #710

Merged
merged 28 commits into from
Nov 15, 2022
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
bb62572
feat: enable and disable flashloans
Aug 29, 2022
d7aa26a
fix: remove gitignore update
Aug 29, 2022
348ce20
feat: bump to beta version
Aug 29, 2022
8b9221b
feat: remove borrow enabled requirement
Aug 29, 2022
8d12d79
feat: updates and tests
Aug 29, 2022
a5ce86a
fix: remove unrelated change
Aug 29, 2022
4c2cda0
fix: remove formatting conflicts
Aug 29, 2022
2710b9c
Update contracts/protocol/pool/PoolConfigurator.sol
Oct 18, 2022
c3c6e75
Update contracts/protocol/libraries/configuration/ReserveConfiguratio…
Oct 18, 2022
6f5d692
Update contracts/protocol/libraries/configuration/ReserveConfiguratio…
Oct 18, 2022
4fb58e8
Update contracts/protocol/libraries/configuration/ReserveConfiguratio…
Oct 18, 2022
a5d188c
Update contracts/protocol/libraries/configuration/ReserveConfiguratio…
Oct 18, 2022
c646e35
Update contracts/misc/AaveProtocolDataProvider.sol
Oct 18, 2022
7d3bedd
Update contracts/interfaces/IPoolConfigurator.sol
Oct 18, 2022
c17c5a8
Update contracts/interfaces/IPoolConfigurator.sol comments
Oct 18, 2022
2f2e18e
Update contracts/interfaces/IPoolConfigurator.sol comments
Oct 18, 2022
9d84549
fix: update comment for setReserveFlashLoaning
Oct 18, 2022
284b492
fix: check revert msg and event emission
Oct 18, 2022
8888093
feat: add additional flashloan scenario
Oct 18, 2022
748818f
feat: switch bit used for flashloan enabled
Oct 25, 2022
7bd04e7
Update test-suites/pool-flashloan.spec.ts - move await
Oct 25, 2022
49d0f4e
feat: add unit test for reserve configuration
Oct 25, 2022
516e0e8
fix: streamline test
Oct 25, 2022
81b9cb9
Merge branch 'feat/enable-disable-flashloan-config' of github.com:aav…
Oct 25, 2022
d87efa6
Merge branch 'feat/3.0.1' into feat/enable-disable-flashloan-config
Oct 28, 2022
1820962
Merge branch 'feat/3.0.1' into feat/enable-disable-flashloan-config
Nov 10, 2022
078fa28
fix: update deploy and periphery dependencies
Nov 15, 2022
bf652c2
fix: add validation to simpleFlashLoan
Nov 15, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions contracts/deployments/ReservesSetupHelper.sol
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ contract ReservesSetupHelper is Ownable {
uint256 supplyCap;
bool stableBorrowingEnabled;
bool borrowingEnabled;
bool flashLoanEnabled;
}

/**
Expand Down Expand Up @@ -50,6 +51,7 @@ contract ReservesSetupHelper is Ownable {
inputParams[i].stableBorrowingEnabled
);
}
configurator.setReserveFlashLoaning(inputParams[i].asset, inputParams[i].flashLoanEnabled);
configurator.setSupplyCap(inputParams[i].asset, inputParams[i].supplyCap);
configurator.setReserveFactor(inputParams[i].asset, inputParams[i].reserveFactor);
}
Expand Down
15 changes: 15 additions & 0 deletions contracts/interfaces/IPoolConfigurator.sol
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,13 @@ interface IPoolConfigurator {
**/
event ReserveBorrowing(address indexed asset, bool enabled);

/**
* @dev Emitted when flashloans are enabled or disabled on a reserve.
* @param asset The address of the underlying asset of the reserve
* @param enabled True if flashloans are enabled, False if flashloans are disabled
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
*/
event ReserveFlashLoaning(address indexed asset, bool enabled);

/**
* @dev Emitted when the collateralization risk parameters for the specified asset are updated.
* @param asset The address of the underlying asset of the reserve
Expand Down Expand Up @@ -310,6 +317,14 @@ interface IPoolConfigurator {
**/
function setReserveStableRateBorrowing(address asset, bool enabled) external;

/**
* @notice Configures flashloans on a reserve
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
* @dev Can only be enabled (set to true) if borrowing is enabled
miguelmtzinf marked this conversation as resolved.
Show resolved Hide resolved
* @param asset The address of the underlying asset of the reserve
* @param enabled True if flashloans need to be enabled, false to disable falshloans
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
*/
function setReserveFlashLoaning(address asset, bool enabled) external;

/**
* @notice Activate or deactivate a reserve
* @param asset The address of the underlying asset of the reserve
Expand Down
12 changes: 12 additions & 0 deletions contracts/misc/AaveProtocolDataProvider.sol
Original file line number Diff line number Diff line change
Expand Up @@ -374,4 +374,16 @@ contract AaveProtocolDataProvider is IPoolDataProvider {

return (reserve.interestRateStrategyAddress);
}

/**
* @notice Returns whether the reserve has FlashLoans enabled or disabled
* @param asset The address of the underlying asset of the reserve
* @return enabled True if FlashLoans are enabled, False if disabled
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
* */
function getFlashLoanEnabled(address asset) external view returns (bool) {
DataTypes.ReserveConfigurationMap memory configuration = IPool(ADDRESSES_PROVIDER.getPool())
.getConfiguration(asset);

return configuration.getFlashLoanEnabled();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ library ReserveConfiguration {
uint256 internal constant EMODE_CATEGORY_MASK = 0xFFFFFFFFFFFFFFFFFFFF00FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF; // prettier-ignore
uint256 internal constant UNBACKED_MINT_CAP_MASK = 0xFFFFFFFFFFF000000000FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF; // prettier-ignore
uint256 internal constant DEBT_CEILING_MASK = 0xF0000000000FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF; // prettier-ignore
uint256 internal constant FLASHLOAN_ENABLED_MASK = 0xEFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF; // prettier-ignore

/// @dev For the LTV, the start bit is 0 (up to 15), hence no bitshifting is needed
uint256 internal constant LIQUIDATION_THRESHOLD_START_BIT_POSITION = 16;
Expand All @@ -49,6 +50,7 @@ library ReserveConfiguration {
uint256 internal constant EMODE_CATEGORY_START_BIT_POSITION = 168;
uint256 internal constant UNBACKED_MINT_CAP_START_BIT_POSITION = 176;
uint256 internal constant DEBT_CEILING_START_BIT_POSITION = 212;
uint256 internal constant FLASHLOAN_ENABLED_START_BIT_POSITION = 252;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unless it was previously used for other stuff, ewe can use the reserved 63th bit

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure if this bit is reserved for something or has been used previously. @The-3D any insight?


uint256 internal constant MAX_VALID_LTV = 65535;
uint256 internal constant MAX_VALID_LIQUIDATION_THRESHOLD = 65535;
Expand Down Expand Up @@ -547,6 +549,32 @@ library ReserveConfiguration {
return (self.data & ~EMODE_CATEGORY_MASK) >> EMODE_CATEGORY_START_BIT_POSITION;
}

/**
* @dev Set whether or not FlashLoaning this asset is enabled
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
* @param self The reserve configuration
* @param flashLoanEnabled boolean to indicated if Flashloans should be enabled (1) or disabled (0)
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
*/
function setFlashLoanEnabled(DataTypes.ReserveConfigurationMap memory self, bool flashLoanEnabled)
miguelmtzinf marked this conversation as resolved.
Show resolved Hide resolved
internal
pure
{
self.data =
(self.data & FLASHLOAN_ENABLED_MASK) |
(uint256(flashLoanEnabled ? 1 : 0) << FLASHLOAN_ENABLED_START_BIT_POSITION);
}

/**
* @dev Get the flashLoanEnabledSetting
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
* @param self The reserve configuration
*/
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
function getFlashLoanEnabled(DataTypes.ReserveConfigurationMap memory self)
internal
pure
returns (bool)
{
return (self.data & ~FLASHLOAN_ENABLED_MASK) != 0;
}

/**
* @notice Gets the configuration flags of the reserve
* @param self The reserve configuration
Expand Down
1 change: 1 addition & 0 deletions contracts/protocol/libraries/helpers/Errors.sol
Original file line number Diff line number Diff line change
Expand Up @@ -97,4 +97,5 @@ library Errors {
string public constant STABLE_BORROWING_ENABLED = '88'; // 'Stable borrowing is enabled'
string public constant SILOED_BORROWING_VIOLATION = '89'; // 'User is trying to borrow multiple assets including a siloed one'
string public constant RESERVE_DEBT_NOT_ZERO = '90'; // the total debt of the reserve needs to be 0
string public constant FLASHLOAN_DISABLED = '91'; // FlashLoaning for this asset is disabled
}
1 change: 1 addition & 0 deletions contracts/protocol/libraries/logic/ValidationLogic.sol
Original file line number Diff line number Diff line change
Expand Up @@ -472,6 +472,7 @@ library ValidationLogic {
.configuration;
require(!configuration.getPaused(), Errors.RESERVE_PAUSED);
require(configuration.getActive(), Errors.RESERVE_INACTIVE);
require(configuration.getFlashLoanEnabled(), Errors.FLASHLOAN_DISABLED);
Copy link
Collaborator

@kyzia551 kyzia551 Oct 27, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the same require missing in the validateFlashloanSimple in some lines above. Because of that looks like a good idea to reuse simple ie:

   ...
    require(assets.length == amounts.length, Errors.INCONSISTENT_FLASHLOAN_PARAMS);
    for (uint256 i = 0; i < assets.length; i++) {
      validateFlashloanSimple(reservesData[assets[i]].configuration);
    }

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good catch. Let's add tests and evaluate this change!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed - will update

}
}

Expand Down
15 changes: 14 additions & 1 deletion contracts/protocol/pool/PoolConfigurator.sol
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ contract PoolConfigurator is VersionedInitializable, IPoolConfigurator {
uint256 public constant CONFIGURATOR_REVISION = 0x1;

/// @inheritdoc VersionedInitializable
function getRevision() internal pure virtual override returns (uint256) {
function getRevision() internal virtual override pure returns (uint256) {
stevenvaleri marked this conversation as resolved.
Show resolved Hide resolved
return CONFIGURATOR_REVISION;
}

Expand Down Expand Up @@ -191,6 +191,19 @@ contract PoolConfigurator is VersionedInitializable, IPoolConfigurator {
emit ReserveStableRateBorrowing(asset, enabled);
}

/// @inheritdoc IPoolConfigurator
function setReserveFlashLoaning(address asset, bool enabled)
external
override
onlyRiskOrPoolAdmins
{
DataTypes.ReserveConfigurationMap memory currentConfig = _pool.getConfiguration(asset);

currentConfig.setFlashLoanEnabled(enabled);
_pool.setConfiguration(asset, currentConfig);
emit ReserveFlashLoaning(asset, enabled);
}

/// @inheritdoc IPoolConfigurator
function setReserveActive(address asset, bool active) external override onlyPoolAdmin {
if (!active) _checkNoSuppliers(asset);
Expand Down
21 changes: 12 additions & 9 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 2 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@aave/core-v3",
"version": "1.16.2",
"version": "1.16.2-beta.1",
"description": "Aave Protocol V3 core smart contracts",
"files": [
"contracts",
Expand Down Expand Up @@ -30,7 +30,7 @@
"prepublish": "npm run compile"
},
"devDependencies": {
"@aave/deploy-v3": "1.27.0",
"@aave/deploy-v3": "1.24.0-beta.2",
"@aave/periphery-v3": "1.18.0",
"@nomiclabs/hardhat-ethers": "npm:hardhat-deploy-ethers@^0.3.0-beta.13",
"@tenderly/hardhat-tenderly": "1.1.0-beta.5",
Expand Down
25 changes: 25 additions & 0 deletions test-suites/configurator.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1075,4 +1075,29 @@ makeSuite('PoolConfigurator', (testEnv: TestEnv) => {
const { helpersContract } = testEnv;
expect(await helpersContract.getDebtCeilingDecimals()).to.be.eq(2);
});

it('Check that the reserves have flashloans enabled', async () => {
const { weth, aave, usdc, dai, helpersContract } = testEnv;

const wethFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(weth.address);
expect(wethFlashLoanEnabled).to.be.equal(true);
miguelmtzinf marked this conversation as resolved.
Show resolved Hide resolved

const aaveFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(aave.address);
expect(aaveFlashLoanEnabled).to.be.equal(true);

const usdcFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(usdc.address);
expect(usdcFlashLoanEnabled).to.be.equal(true);

const daiFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(dai.address);
expect(daiFlashLoanEnabled).to.be.equal(true);
});

it('Disable weth flashloans', async () => {
const { weth, configurator, helpersContract } = testEnv;

expect(await configurator.setReserveFlashLoaning(weth.address, false));

const wethFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(weth.address);
expect(wethFlashLoanEnabled).to.be.equal(false);
});
});
35 changes: 35 additions & 0 deletions test-suites/pool-flashloan.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import { MAX_UINT_AMOUNT } from '../helpers/constants';
import { convertToCurrencyDecimals } from '../helpers/contracts-helpers';
import { MockFlashLoanReceiver } from '../types/MockFlashLoanReceiver';
import { ProtocolErrors } from '../helpers/types';

import {
getMockFlashLoanReceiver,
getStableDebtToken,
Expand Down Expand Up @@ -194,6 +195,7 @@ makeSuite('Pool: FlashLoan', (testEnv: TestEnv) => {

expect(totalLiquidityBefore.add(totalFees)).to.be.closeTo(totalLiquidityAfter, 2);
});

it('Takes an ETH flashloan with mode = 0 as big as the available liquidity', async () => {
const { pool, helpersContract, weth, aWETH, deployer } = testEnv;

Expand Down Expand Up @@ -253,6 +255,39 @@ makeSuite('Pool: FlashLoan', (testEnv: TestEnv) => {
reservesAfter.sub(feesToProtocol).mul(liquidityIndexBefore).div(currentLiquidityIndex)
).to.be.closeTo(reservesBefore, 2);
});

it('Disable ETH flashloan and takes an ETH flashloan (revert expected)', async () => {
const { pool, configurator, helpersContract, weth, deployer } = testEnv;

expect(await configurator.setReserveFlashLoaning(weth.address, false));

let wethFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(weth.address);
expect(wethFlashLoanEnabled).to.be.equal(false);

let reserveData = await helpersContract.getReserveData(weth.address);

const totalLiquidityBefore = reserveData.totalAToken;

const flashBorrowedAmount = totalLiquidityBefore;

await expect(
pool.flashLoan(
_mockFlashLoanReceiver.address,
[weth.address],
[flashBorrowedAmount],
[0],
_mockFlashLoanReceiver.address,
'0x10',
'0'
)
).to.be.reverted;
miguelmtzinf marked this conversation as resolved.
Show resolved Hide resolved

expect(await configurator.setReserveFlashLoaning(weth.address, true));
miguelmtzinf marked this conversation as resolved.
Show resolved Hide resolved

wethFlashLoanEnabled = await helpersContract.getFlashLoanEnabled(weth.address);
expect(wethFlashLoanEnabled).to.be.equal(true);
});

miguelmtzinf marked this conversation as resolved.
Show resolved Hide resolved
it('Takes WETH flashloan, does not return the funds with mode = 0 (revert expected)', async () => {
const { pool, weth, users } = testEnv;
const caller = users[1];
Expand Down