Skip to content
This repository was archived by the owner on Dec 10, 2025. It is now read-only.
Open
Changes from all commits
Commits
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
68 changes: 28 additions & 40 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,65 +1,53 @@
# Aave Token design
# Aave Token Design

AAVE is an ERC-20 compatible token. It implements governance-inspired features, and will allow Aave to bootstrap the rewards program for safety and ecosystem growth.
The following document explains the main features of AAVE, it’s monetary policy, and the redemption process from LEND.
AAVE is an ERC-20 compatible token. It implements governance-inspired features and enables Aave to bootstrap the rewards program for safety and ecosystem growth. This document explains the main features of AAVE, its monetary policy, and the redemption process from LEND.

## Roles

The initial AAVE token implementation does not have any admin roles configured. The contract will be proxied using the Openzeppelin implementation of the EIP-1967 Transparent Proxy pattern. The proxy has an Admin role, and the Admin of the proxy contract will be set upon deployment to the Aave governance contracts.
The initial AAVE token implementation has no admin roles configured. The contract is proxied using OpenZeppelin’s EIP-1967 Transparent Proxy pattern. The proxy has an Admin role, and the Admin of the proxy contract will be set upon deployment to the Aave governance contracts.

## ERC-20

The AAVE token implements the standard methods of the ERC-20 interface. A balance snapshot feature has been added to keep track of the balances of the users at specific block heights. This will help with the Aave governance integration of AAVE.
AAVE also integrates the EIP 2612 `permit` function, that will allow gasless transaction and one tx approval/transfer.
The AAVE token implements the standard ERC-20 interface. A balance snapshot feature keeps track of user balances at specific block heights to support Aave governance. AAVE also integrates the EIP-2612 `permit` function, allowing gasless approvals and one-tx approval/transfer.

# LendToAaveMigrator

Smart contract for LEND token holders to execute the migration to the AAVE token, using part of the initial emission of AAVE for it.
Smart contract for LEND holders to migrate to AAVE using part of the initial AAVE emission.

The contract is covered by a proxy, whose owner will be the AAVE governance. Once the governance passes the corresponding proposal, the proxy will be connected to the implementation and LEND holders will be able to call the `migrateFromLend()` function, which, after LEND approval, will pull LEND from the holder wallet and transfer back an equivalent AAVE amount defined by the `LEND_AAVE_RATIO` constant.
The contract is behind a proxy owned by Aave governance. After governance approves the proposal, the proxy connects to the implementation and LEND holders can call `migrateFromLend()`, which (after LEND approval) pulls LEND from the holder and returns an equivalent AAVE amount defined by `LEND_AAVE_RATIO`.

One tradeOff of `migrateFromLend()` is that, as the AAVE total supply will be lower than LEND, the `LEND_AAVE_RATIO` will be always > 1, causing a loss of precision for amounts of LEND that are not multiple of `LEND_AAVE_RATIO`. E.g. a person sending 1.000000000000000022 LEND, with a `LEND_AAVE_RATIO` == 100, will receive 0.01 AAVE, losing the value of the last 22 small units of LEND.
Taking into account the current value of LEND and the future value of AAVE, a lack of precision for less than LEND_AAVE_RATIO small units represents a value several orders of magnitude smaller than 0.01\$. We evaluated some potential solutions for this, specifically:
### Trade-off

1. Rounding half up the amount of AAVE returned from the migration. This opens up to potential attacks where users might purposely migrate less than LEND_AAVE_RATIO to obtain more AAVE as a result of the round up.
2. Returning back the excess LEND: this would leave LEND in circulation forever, which is not the expected end result of the migration.
3. Require the users to migrate only amounts that are multiple of LEND_AAVE_RATIO: This presents considerable UX friction.
Because AAVE total supply < LEND, `LEND_AAVE_RATIO` > 1, which can cause precision loss for LEND amounts not divisible by the ratio. Example: with ratio 100, sending `1.000000000000000022 LEND` yields `0.01 AAVE`, losing the final tiny units. Alternatives considered:
1. Round half up → exploitable via dust.
2. Return excess LEND → leaves LEND in circulation.
3. Require multiples of ratio → poor UX.
The implemented approach is preferred.

None of those present a better outcome than the implemented solution.
## Redemption Process

## The Redemption process
Deploy **AaveToken** and **LendToAaveMigrator**, then transfer proxy ownerships to Aave Governance. The Aave team creates an AIP; once approved, the LEND/AAVE redemption and incentives activate, marking AAVE’s initial market emission. LEND supply becomes locked in the new AAVE contract while equivalent AAVE is issued. The AAVE corresponding to LEND burned in the initial phase remains locked in the migrator.

The first step to bootstrap the AAVE emission is to deploy the AAVE token contract and the  LendToAaveMigrator contract. This task will be performed by the Aave team. Upon deployment, the ownership of the Proxy of the AAVE contract and the LendToAaveMigrator will be set to the Aave Governance. To start the LEND redemption process at that point, the Aave team will create an AIP (Aave Improvement Proposal) and submit a proposal to the Aave governance. The proposal will, once approved, activate the LEND/AAVE redemption process and the ecosystem incentives, which will mark the initial emission of AAVE on the market.
The result of the migration procedure will see the supply of LEND being progressively locked within the new AAVE smart contract, while at the same time an equivalent amount of AAVE is being issued.  
The amount of AAVE equivalent to the LEND tokens burned in the initial phase of the AAVE protocol will remain locked in the LendToAaveMigrator contract.
## Technical Implementation

## Technical implementation
### Changes to OpenZeppelin

### Changes to the Openzeppelin original contracts
- In `/contracts/open-zeppelin/ERC20.sol` lines 44–45, `_name` and `_symbol` changed from `private` to `internal`.
- Added `VersionedInitializable` (extends OZ `Initializable`):
1. Replaces `bool initialized` with `uint256 latestInitializedRevision`.
2. `initializer()` reads implementation revision via `getRevision()`.
3. Allows initializing only when the incoming revision is higher.
This enables calling `initialize()` across multiple implementations.

In the context of this implementation, we needed apply the following changes to the OpenZepplin implementation:
### `_beforeTokenTransfer` Hook

- In `/contracts/open-zeppelin/ERC20.sol`, line 44 and 45, `_name` and `_symbol` have been changed from `private` to `internal`
- We extended the original `Initializable` class from the Openzeppelin contracts and created a `VersionedInitializable` contract. The main differences compared to the `Initializable` are:
Overridden to:
1. Snapshot balances on each `mint`, `burn`, `transfer`, `transferFrom` (skip self-transfer).
2. Call Aave governance with the same parameters (assumed trusted; it must handle reentrancy). Skipped for self-transfers.

1. The boolean `initialized` has been replaced with a `uint256 latestInitializedRevision`.
2. The `initializer()` modifier fetch the revision of the implementation using a `getRevision()` function defined in the implementation contract. The `initializer` modifier forces that an implementation
3. with a bigger revision number than the current one is being initialized
## Development Deployment

The change allows us to call `initialize()` on multiple implementations, that was not possible with the original `Initializable` implementation from OZ.

### \_beforeTokenTransfer hook

We override the \_beforeTokenTransfer function on the OZ base ERC20 implementation in order to include the following features:

1. Snapshotting of balances every time an action involved a transfer happens (mint, burn, transfer or transferFrom). If the account does a transfer to itself, no new snapshot is done.
2. Call to the Aave governance contract forwarding the same input parameters of the `_beforeTokenTransfer` hook. Its an assumption that the Aave governance contract is a trustable party, being its responsibility to control all potential reentrancies if calling back the AaveToken. If the account does a transfer to itself, no interaction with the Aave governance contract should happen.

## Development deployment

For development purposes, you can deploy AaveToken and LendToAaveMigrator to a local network via the following command:

```
```bash
npm run dev:deployment
```

Expand Down