You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
{{ message }}
This repository has been archived by the owner on Jan 7, 2024. It is now read-only.
panprog - Position closure and position liquidations might always revert in some cases due to allocatedBalances being unsigned, preventing the user from closing its positions or finishing liquidation
#193
Closed
sherlock-admin opened this issue
Jul 3, 2023
· 0 comments
sherlock-admin opened this issue
Jul 3, 2023
· 0 comments
Labels
DuplicateA valid issue that is a duplicate of an issue with `Has Duplicates` labelHighA valid High severity issueRewardA payout will be made for this issue
Position closure and position liquidations might always revert in some cases due to allocatedBalances being unsigned, preventing the user from closing its positions or finishing liquidation
In the other cases formula is similar. This means that in case of high unrealized profit, allocated balance might be valid to be negative. For example, a simple case is if account has opened position with unrealized profit of 1000 and locked balance of 100, allocated balance can theoretically be -900 with account being safe from liquidation: -900 + 1000 = 100 >= lockedBalance(100). But since balance is unsigned, transaction to deallocate such balance will revert. This might be expected behaviour.
However, there are more complex cases possible where unsigned allocated balance can unexpectedly revert different transactions and block users from doing important actions. For example, if user has 2 opposite positions opened of the same quantity (LONG and SHORT) and price has moved significantly, total user upnl = 0 (because opposite positions hedge each other completely). But the user will be unable to close either of these positions, because closing any position will subract high loss from the position from either partyA or partyB, which will make allocated balance negative and revert the transaction:
Before closing position:
Account Value = Position 1 Value (+1000) + Position 2 Value (-1000) + Allocated Balance (100) = +100 (account solvent)
After closing position:
Account Value should be = Position 1 Value (+1000) + Allocated Balance (-900) = +100 (account solvent), but it will revert.
As such, both partyA and partyB will lose funds deposited into the protocol to keep these positions alive. It will also be impossible to finish liquidation of such positions if any account becomes liquidatable, effectively locking the funds for extended time (until price is in the range where positions can be closed without big loss).
Vulnerability Detail
The scenario when user unexpectedly can't close position and funds are locked for extended time:
Open 2 large opposite positions between partyA and partyB such that one of them is in a high loss and the other in the same profit. Use minimal allocated balance both for partyA and partyB (because such opposite positions can't be liquidated by themselves as they perfectly hedge each other).
When price moves significantly, try to close either of these positions, both will revert.
Both partyA and partyB are stuck with the funds in the protocol which they can't take out until price goes back close to initial value.
There might be the other cases with unexpected reverts and users being unable to close their positions.
Impact
PartyA and PartyB funds can be unexpectedly locked in the protocol for extended time. PartyA or PartyB can allocate additional funds to be able to close positions, then deallocate and withdraw, however depending on situation, required funds to close such position might be orders of magnitude higher than what parties have or are willing to risk (due to leverage).
Code Snippet
Add this to any test, for example to ClosePosition.behavior.ts.
it("Unable to close positions due to unsigned allocatedBalances",asyncfunction(){constcontext: RunContext=this.context;this.user_allocated=decimal(82);this.hedger_allocated=decimal(80);this.user=newUser(this.context,this.context.signers.user);awaitthis.user.setup();awaitthis.user.setBalances(this.user_allocated,this.user_allocated,this.user_allocated);this.hedger=newHedger(this.context,this.context.signers.hedger);awaitthis.hedger.setup();awaitthis.hedger.setBalances(this.hedger_allocated,this.hedger_allocated);// open 2 opposite direction positions awaitthis.user.sendQuote(limitQuoteRequestBuilder().quantity(decimal(100)).price(decimal(1)).cva(decimal(10)).lf(decimal(5)).mm(decimal(15)).build());awaitthis.hedger.lockQuote(1,0,decimal(4,17));awaitthis.hedger.openPosition(1,limitOpenRequestBuilder().filledAmount(decimal(100)).openPrice(decimal(1)).price(decimal(1)).build());awaitthis.user.sendQuote(limitQuoteRequestBuilder().positionType(PositionType.SHORT).quantity(decimal(100)).price(decimal(1)).cva(decimal(10)).lf(decimal(5)).mm(decimal(15)).build());awaitthis.hedger.lockQuote(2,0,decimal(4,17));awaitthis.hedger.openPosition(2,limitOpenRequestBuilder().filledAmount(decimal(100)).openPrice(decimal(1)).price(decimal(1)).build());varinfo=awaitthis.user.getBalanceInfo();console.log("partyA allocated: "+info.allocatedBalances/1e18+" locked: "+info.totalLocked/1e18+" pendingLocked: "+info.totalPendingLocked/1e18);varinfo=awaitthis.hedger.getBalanceInfo(this.user.getAddress());console.log("partyB allocated: "+info.allocatedBalances/1e18+" locked: "+info.totalLocked/1e18+" pendingLocked: "+info.totalPendingLocked/1e18);// now the price doubles, with upnl still 0// try to close LONG positionawaitthis.user.requestToClosePosition(1,limitCloseRequestBuilder().quantityToClose(decimal(100)).closePrice(decimal(2)).build(),);awaitexpect(this.hedger.fillCloseRequest(1,limitFillCloseRequestBuilder().filledAmount(decimal(100)).closedPrice(decimal(2)).build(),)).to.be.revertedWith("LibSolvency: PartyB will be liquidatable");console.log("Attempt to close LONG position failed due to partyB being liquidatable");// try to close SHORT positionawaitexpect(this.user.requestToClosePosition(2,limitCloseRequestBuilder().quantityToClose(decimal(100)).closePrice(decimal(2)).build(),)).to.be.revertedWith("LibSolvency: partyA will be liquidatable");console.log("Attempt to close SHORT position failed due to partyA being liquidatable");});
Tool used
Manual Review
Recommendation
Make allocated balances signed ints and change all appropriate parts where allocated balances are changed or verified. This will be more user-friendly (less limiting) and will prevent situations described in this bug report.
Sign up for freeto subscribe to this conversation on GitHub.
Already have an account?
Sign in.
Labels
DuplicateA valid issue that is a duplicate of an issue with `Has Duplicates` labelHighA valid High severity issueRewardA payout will be made for this issue
panprog
high
Position closure and position liquidations might always revert in some cases due to allocatedBalances being unsigned, preventing the user from closing its positions or finishing liquidation
Summary
Allocated balances are stored as unsigned ints.
https://github.com/sherlock-audit/2023-06-symmetrical/blob/main/symmio-core/contracts/storages/AccountStorage.sol#L35
https://github.com/sherlock-audit/2023-06-symmetrical/blob/main/symmio-core/contracts/storages/AccountStorage.sol#L39
Total account value is calculated as
allocated balance + unrealized pnl
, for example for liquidation purposes it's calculated as:https://github.com/sherlock-audit/2023-06-symmetrical/blob/main/symmio-core/contracts/libraries/LibAccount.sol#L83-L85
In the other cases formula is similar. This means that in case of high unrealized profit, allocated balance might be valid to be negative. For example, a simple case is if account has opened position with unrealized profit of 1000 and locked balance of 100, allocated balance can theoretically be
-900
with account being safe from liquidation:-900 + 1000 = 100 >= lockedBalance(100)
. But since balance is unsigned, transaction to deallocate such balance will revert. This might be expected behaviour.However, there are more complex cases possible where unsigned allocated balance can unexpectedly revert different transactions and block users from doing important actions. For example, if user has 2 opposite positions opened of the same quantity (LONG and SHORT) and price has moved significantly, total user
upnl = 0
(because opposite positions hedge each other completely). But the user will be unable to close either of these positions, because closing any position will subract high loss from the position from either partyA or partyB, which will make allocated balance negative and revert the transaction:Before closing position:
Account Value = Position 1 Value (+1000) + Position 2 Value (-1000) + Allocated Balance (100) = +100
(account solvent)After closing position:
Account Value should be = Position 1 Value (+1000) + Allocated Balance (-900) = +100
(account solvent), but it will revert.As such, both partyA and partyB will lose funds deposited into the protocol to keep these positions alive. It will also be impossible to finish liquidation of such positions if any account becomes liquidatable, effectively locking the funds for extended time (until price is in the range where positions can be closed without big loss).
Vulnerability Detail
The scenario when user unexpectedly can't close position and funds are locked for extended time:
There might be the other cases with unexpected reverts and users being unable to close their positions.
Impact
PartyA and PartyB funds can be unexpectedly locked in the protocol for extended time. PartyA or PartyB can allocate additional funds to be able to close positions, then deallocate and withdraw, however depending on situation, required funds to close such position might be orders of magnitude higher than what parties have or are willing to risk (due to leverage).
Code Snippet
Add this to any test, for example to
ClosePosition.behavior.ts
.Tool used
Manual Review
Recommendation
Make allocated balances signed ints and change all appropriate parts where allocated balances are changed or verified. This will be more user-friendly (less limiting) and will prevent situations described in this bug report.
Duplicate of #241
The text was updated successfully, but these errors were encountered: