The protocol implicitly assumes a uniform decimal precision when calculating mintAmount during deposits. However, ERC20 tokens are not required to use 18 decimals and may use lower precision values such as 6 or 8 decimals.
Because the deposit logic does not normalize token amounts to a common precision before share calculation, the computed mintAmount becomes inaccurate when non-18-decimal tokens are deposited. As a result, users receive fewer shares than they should for the same economic value.
Since redemption logic depends on the number of shares held, this discrepancy directly reduces the amount of assets users can withdraw. The protocol remains solvent on paper, but individual users experience irreversible losses, making this a critical accounting flaw.
Likelihood:
High.
This issue will occur whenever the protocol supports or integrates ERC20 tokens that do not use 18 decimals (e.g., USDC with 6 decimals). Such tokens are extremely common in production DeFi systems, making this scenario highly realistic rather than theoretical.
Impact:
High — Permanent Loss of User Funds.
Users depositing tokens with fewer decimals than expected will receive an incorrect number of shares. Since redemption calculations rely on these shares, affected users will permanently lose a portion of their deposited assets. This loss compounds over time and cannot be recovered through normal protocol operations.
Please add this test on test/unit/TestThunderLoan.t.sol
Consider to add a logic to handle different decimals on any other erc20
## Description Within the functions `ThunderLoan::getCalculatedFee()` and `ThunderLoanUpgraded::getCalculatedFee()`, an issue arises with the calculated fee value when dealing with non-standard ERC20 tokens. Specifically, the calculated value for non-standard tokens appears significantly lower compared to that of standard ERC20 tokens. ## Vulnerability Details //ThunderLoan.sol ```solidity function getCalculatedFee(IERC20 token, uint256 amount) public view returns (uint256 fee) { //slither-disable-next-line divide-before-multiply @> uint256 valueOfBorrowedToken = (amount * getPriceInWeth(address(token))) / s_feePrecision; @> //slither-disable-next-line divide-before-multiply fee = (valueOfBorrowedToken * s_flashLoanFee) / s_feePrecision; } ``` ```solidity //ThunderLoanUpgraded.sol function getCalculatedFee(IERC20 token, uint256 amount) public view returns (uint256 fee) { //slither-disable-next-line divide-before-multiply @> uint256 valueOfBorrowedToken = (amount * getPriceInWeth(address(token))) / FEE_PRECISION; //slither-disable-next-line divide-before-multiply @> fee = (valueOfBorrowedToken * s_flashLoanFee) / FEE_PRECISION; } ``` ## Impact Let's say: - user_1 asks a flashloan for 1 ETH. - user_2 asks a flashloan for 2000 USDT. ```solidity function getCalculatedFee(IERC20 token, uint256 amount) public view returns (uint256 fee) { //1 ETH = 1e18 WEI //2000 USDT = 2 * 1e9 WEI uint256 valueOfBorrowedToken = (amount * getPriceInWeth(address(token))) / s_feePrecision; // valueOfBorrowedToken ETH = 1e18 * 1e18 / 1e18 WEI // valueOfBorrowedToken USDT= 2 * 1e9 * 1e18 / 1e18 WEI fee = (valueOfBorrowedToken * s_flashLoanFee) / s_feePrecision; //fee ETH = 1e18 * 3e15 / 1e18 = 3e15 WEI = 0,003 ETH //fee USDT: 2 * 1e9 * 3e15 / 1e18 = 6e6 WEI = 0,000000000006 ETH } ``` The fee for the user_2 are much lower then user_1 despite they asks a flashloan for the same value (hypotesis 1 ETH = 2000 USDT). ## Recommendations Adjust the precision accordinly with the allowed tokens considering that the non standard ERC20 haven't 18 decimals.
The contest is live. Earn rewards by submitting a finding.
Submissions are being reviewed by our AI judge. Results will be available in a few minutes.
View all submissionsThe contest is complete and the rewards are being distributed.