Skip to content

code-423n4/2024-04-panoptic

Repository files navigation

Panoptic audit details

  • Total Prize Pool: $120,000 in USDC

    • HM awards: $100,320 in USDC
    • QA awards: $4,180 in USDC
    • Judge awards: $9,000 USDC
    • Lookout awards: $6,000 USDC
    • Scout awards: $500 in USDC
  • Join C4 Discord to register

  • Submit findings using the C4 form

  • Read our guidelines for more details

  • Starts April 1, 2024 20:00 UTC

  • Ends April 22, 2024 20:00 UTC

Automated Findings / Publicly Known Issues

The 4naly3er report can be found here.

Note for C4 wardens: Anything included in this Automated Findings / Publicly Known Issues section is considered a publicly known issue and is ineligible for awards.

  • Transfers of ERC1155 SFPM tokens are very limited by design. It is expected that certain accounts will be unable to transfer or receive tokens. Some tokens may not be transferable at all.
  • Construction helper functions (prefixed with add) in the TokenId library and other types do not perform extensive input validation. Passing invalid or nonsensical inputs into these functions or attempting to overwrite already filled slots may yield unexpected or invalid results. This is by design, so it is expected that users of these functions will validate the inputs beforehand.
  • Very large quantities of tokens are not supported. It should be assumed that for any given pool, the cumulative amount of tokens that enter the system (associated with that pool, through adding liquidity, collection, etc.) will not exceed 2^127 - 1. Note that this is only a per-pool assumption, so if it is broken on one pool it should not affect the operation of any other pools, only the pool in question.
  • If one token on a pool is broken/does not meet listed criteria/is malicious there are no guarantees as to the security of the other token in that pool, as long as other pools with two legitimate and compliant tokens are not affected.
  • Price/oracle manipulation that is not atomic or requires attackers to hold a price across more than one block (i.e., to manipulate a Uniswap observation, you need to set the manipulated price at the end of one block, and then keep it there until the next block) is not in scope
  • Attacks that stem from the TWAP being extremely stale compared to the market price within its period (currently 10 minutes)
  • As a general rule, only price manipulation issues that can be triggered by manipulating the price atomically from a normal pool/oracle state are valid
  • The constants VEGOID, TWAP_WINDOW, SLOW_ORACLE_UNISWAP_MODE, MEDIAN_PERIOD, FAST_ORACLE_CARDINALITY, SLOW_ORACLE_CARDINALITY, SLOW_ORACLE_PERIOD, FAST_ORACLE_PERIOD, MAX_TWAP_DELTA_LIQUIDATION, MAX_SLOW_FAST_DELTA, MAX_SPREAD, MAX_POSITIONS, BP_DECREASE_BUFFER, and NO_BUFFER are all parameters and subject to change, but within reasonable levels. Cardinalities should be odd, and we're aware that the oracle/pool deployment will fail if the period is too large for the pool. It's important that the SLOW_ORACLE_UNISWAP_MODE flag works as intended, but otherwise not looking for anything other than a super glaring issue with the implementation of those parameters. Playing with the parameters to find an extreme example that technically won't work isn't going to be very useful to us.
  • Given a small enough pool and low seller diversity, premium manipulation by swapping back and forth in Uniswap is a known risk. As long as it's not possible to do it between two of your own accounts profitably and doesn't cause protocol loss, that's acceptable
  • Front-running via insufficient slippage specification is not in scope
  • It's known that liquidators sometimes have a limited capacity to force liquidations to execute at a less favorable price and extract some additional profit from that. This is acceptable even if it causes some amount of unnecessary protocol loss.
  • It's possible to leverage the rounding direction to artificially inflate the total gross premium and significantly decrease the rate of premium option sellers earn/are able to withdraw (but not the premium buyers pay) in the future (only significant for very-low-decimal pools, since this must be done one token at a time).
  • It's also possible for options buyers to avoid paying premium by calling settleLongPremium if the amount of premium owed is sufficiently small.
  • Premium accumulation can become permanently capped if the accumulator exceeds the maximum value; this can happen if a low amount of liquidity earns a large amount of (token) fees
  • The liquidator may not be able to execute a liquidation if MAX_POSITIONS is too high for the deployed chain due to an insufficient gas limit. This parameter is not final and will be adjusted by deployed chain such that the most expensive liquidation is well within a safe margin of the gas limit.
  • It's expected that liquidators may have to sell options, perform force exercises, and deposit collateral to perform some liquidations. In some situations, the liquidation may not be profitable.
  • In some situations (stale TWAP tick), force exercised users will be worse off than if they had burnt their position.
  • For the purposes of this competition, assume the constructor arguments to the CollateralTracker are: 10, 2_000, 1_000, -128, 5_000, 9_000, 20_000
  • Depending on the token, the amount of funds required for the initial factory deployment may be high or unrealistic

Overview

Panoptic is a permissionless options trading protocol. It enables the trading of perpetual options on top of any Uniswap V3 pool

The Panoptic protocol is noncustodial, has no counterparty risk, offers instantaneous settlement, and is designed to remain fully collateralized at all times.

Core Contracts

SemiFungiblePositionManager

A gas-efficient alternative to Uniswap’s NonFungiblePositionManager that manages complex, multi-leg Uniswap positions encoded in ERC1155 tokenIds, performs swaps allowing users to mint positions with only one type of token, and, most crucially, supports the minting of both typical LP positions where liquidity is added to Uniswap and “long” positions where Uniswap liquidity is burnt. While the SFPM is enshrined as a core component of the protocol and we consider it to be the “engine” of Panoptic, it is also a public good that we hope savvy Uniswap V3 LPs will grow to find an essential tool and upgrade for managing their liquidity.

CollateralTracker

An ERC4626 vault where token liquidity from passive Panoptic Liquidity Providers (PLPs) and collateral for option positions are deposited. CollateralTrackers are also responsible for paying out commission fees and options premia, handling payments of intrinsic value for options and distributing P&L, calculating liquidation bonuses, and determining costs for forcefully exercising another user’s options. However, by far the most important functionality of the CollateralTracker is to calculate the collateral requirement for every account and position. Each time positions are minted or burned in Panoptic, the CollateralTracker updates the collateral balances and provides information on the collateral requirement, ensuring that the protocol remains solvent and we retain the ability to liquidate distressed positions when needed.

PanopticPool

The Panoptic Pool exposes the core functionality of the protocol. If the SFPM is the “engine” of Panoptic, the Panoptic Pool is the “conductor”. All interactions with the protocol, be it minting or burning positions, liquidating or force exercising distressed accounts, or just checking position balances and accumulating premiums, originate in this contract. It is responsible for orchestrating the required calls to the SFPM to actually create option positions in Uniswap, tracking user balances of and accumulating the premia on those positions, and calling the CollateralTracker with the data it needs to settle position changes.

Architecture & Actors

Each instance of the Panoptic protocol on a Uniswap pool contains:

  • One PanopticPool that orchestrates all interactions in the protocol
  • Two CollateralTrackers, one for each constituent token0/token1 in the Uniswap pool
  • A canonical SFPM - the SFPM manages liquidity across every Panoptic Pool

There are five primary roles assumed by actors in this Panoptic Ecosystem:

Panoptic Liquidity Providers (PLPs)

Users who deposit tokens into one or both CollateralTracker vaults. The liquidity deposited by these users is borrowed by option sellers to create their positions - their liquidity is what enables undercollateralized positions. In return, they receive commission fees on both the notional and intrinsic values of option positions when they are minted. Note that options buyers and sellers are PLPs too - they must deposit collateral to open their positions. We consider users who deposit collateral but do not trade on Panoptic to be “passive” PLPs

Option Sellers

These users deposit liquidity into the Uniswap pool through Panoptic, making it available for options buyers to remove. This role is similar to providing liquidity directly to Uniswap V3, but offers numerous benefits including advanced tools to manage risky, complex positions and a multiplier on the fees/premia generated by their liquidity when it is removed by option buyers. Sold option positions on Panoptic have similar payoffs to traditional options.

Option Buyers

These users remove liquidity added by option sellers from the Uniswap Pool and move the tokens back into Panoptic. The premia they pay to sellers for the privilege is equivalent to the fees that would have been generated by the removed liquidity, plus a spread multiplier based on the portion of available liquidity in their Uniswap liquidity chunk that has been removed or utilized.

Liquidators

These users are responsible for liquidating distressed accounts that no longer meet the collateral requirements needed to maintain their positions. They provide the tokens necessary to close all positions in the distressed account and receive a bonus from the remaining collateral. Sometimes, they may also need to buy or sell options to allow lower liquidity positions to be exercised

Force Exercisors

These are usually options sellers. They provide the required tokens and forcefully exercise long positions (from option buyers) in out-of-range strikes that are no longer generating premia, so the liquidity from those positions is added back to Uniswap and the sellers can exercise their positions (which involves burning that liquidity). They pay a fee to the exercised user for the inconvenience.

Flow

All protocol users first onboard by depositing tokens into one or both CollateralTracker vaults and being issued shares (becoming PLPs in the process). Panoptic’s CollateralTracker supports the full ERC4626 interface, making deposits and withdrawals a simple and standardized process. Passive PLPs stop here.

Once they have deposited, there are many options for the other actors in the protocol. Buyers and sellers can call :

  • mintOptions - create an option position with up to four distinct legs with a specially encoded - positionID/tokenID, each of which is its own short (sold/added) or long (bought/removed) liquidity chunk
  • burnOptions - burn or exercise a position created through mintOptions
  • settleLongPremium - Force a solvent option buyer to pay any premium owed to sellers. Options sellers will want to call this if a large portion of the total premium owed in their chunk is pending/has not been settled.
  • pokeMedian - Takes a new internal median observation and inserts it into the internal median ring buffer if enough time has passed since the last observation. Otherwise, the buffer is poked during mintOptions/burnOptions

Meanwhile, force exercisers and liquidators can perform their respective roles with the forceExercise and liquidateAccount functions.

Repository Structure

contracts/
├── CollateralTracker"ERC4626 vault where token liquidity from Panoptic Liquidity Providers (PLPs) and collateral for option positions are deposited and collateral requirements are computed"
├── PanopticFactory"Handles deployment of new Panoptic instances on top of Uniswap pools, initial liquidity deployments, and NFT rewards for deployers"
├── PanopticPool"Coordinates all options trading activity - minting, burning, force exercises, liquidations"
├── SemiFungiblePositionManager"The 'engine' of Panoptic - manages all Uniswap V3 positions in the protocol as well as being a more advanced, gas-efficient alternative to NFPM for Uniswap LPs"
├── tokens
│   ├── ERC1155Minimal"A minimalist implementation of the ERC1155 token standard without metadata"
│   ├── ERC20Minimal"A minimalist implementation of the ERC20 token standard without metadata"
│   └── interfaces
        ├── IDonorNFT"An interface the PanopticFactory can use to issue reward NFTs"
│       └── IERC20Partial"An incomplete ERC20 interface containing functions used in Panoptic with some return values omitted to support noncompliant tokens such as USDT"
├── types
│   ├── LeftRight"Implementation for a set of custom data types that can hold two 128-bit numbers"
│   ├── LiquidityChunk"Implementation for a custom data type that can represent a liquidity chunk of a given size in Uniswap - containing a tickLower, tickUpper, and liquidity"
│   └── TokenId"Implementation for the custom data type used in the SFPM and Panoptic to encode position data in 256-bit ERC1155 tokenIds - holds a pool identifier and up to four full position legs"
├── libraries
│   ├── CallbackLib"Library for verifying and decoding Uniswap callbacks"
│   ├── Constants"Library of Constants used in Panoptic"
│   ├── Errors"Contains all custom errors used in Panoptic's core contracts"
│   ├── FeesCalc"Utility to calculate up-to-date swap fees for liquidity chunks"
│   ├── InteractionHelper"Helpers to perform bytecode-size-heavy interactions with external contracts like batch approvals and metadata queries"
│   ├── Math"Library of generic math functions like abs(), mulDiv, etc"
│   ├── PanopticMath"Library containing advanced Panoptic/Uniswap-specific functionality such as our TWAP, price conversions, and position sizing math"
│   └── SafeTransferLib"Safe ERC20 transfer library that gracefully handles missing return values"
└── multicall
    └── Multicall"Adds a function to inheriting contracts that allows for multiple calls to be executed in a single transaction"

Links

Further Reading

Panoptic has been presented at conferences and was conceived with the first Panoptic's Genesis blog post in mid-summer 2021:


Scope

Files in scope

File Logic Contracts Interfaces SLOC
contracts/CollateralTracker.sol 1 **** 792
contracts/PanopticFactory.sol 1 **** 249
contracts/PanopticPool.sol 1 **** 1162
contracts/SemiFungiblePositionManager.sol 1 **** 724
contracts/libraries/CallbackLib.sol 1 **** 22
contracts/libraries/Constants.sol 1 **** 9
contracts/libraries/Errors.sol 1 **** 35
contracts/libraries/FeesCalc.sol 1 **** 86
contracts/libraries/InteractionHelper.sol 1 **** 72
contracts/libraries/Math.sol 1 **** 417
contracts/libraries/PanopticMath.sol 1 **** 573
contracts/libraries/SafeTransferLib.sol 1 **** 33
contracts/multicall/Multicall.sol 1 **** 18
contracts/tokens/ERC1155Minimal.sol 1 **** 115
contracts/tokens/ERC20Minimal.sol 1 **** 52
contracts/tokens/interfaces/IDonorNFT.sol **** 1 4
contracts/tokens/interfaces/IERC20Partial.sol **** 1 6
contracts/types/LeftRight.sol 1 **** 156
contracts/types/LiquidityChunk.sol 1 **** 91
contracts/types/TokenId.sol 1 **** 305
Totals 18 1 4921

Files out of scope

Contract
deploy/DeployProtocol.s.sol
periphery/PanopticHelper.sol
scripts/DeployTestPool.s.sol
scripts/tokens/ERC20S.sol
test/foundry/core/CollateralTracker.t.sol
test/foundry/core/Misc.t.sol
test/foundry/core/PanopticFactory.t.sol
test/foundry/core/PanopticPool.t.sol
test/foundry/core/SemiFungiblePositionManager.t.sol
test/foundry/libraries/CallbackLib.t.sol
test/foundry/libraries/FeesCalc.t.sol
test/foundry/libraries/Math.t.sol
test/foundry/libraries/PanopticMath.t.sol
test/foundry/libraries/PositionAmountsTest.sol
test/foundry/libraries/SafeTransferLib.t.sol
test/foundry/libraries/harnesses/CallbackLibHarness.sol
test/foundry/libraries/harnesses/FeesCalcHarness.sol
test/foundry/libraries/harnesses/MathHarness.sol
test/foundry/libraries/harnesses/PanopticMathHarness.sol
test/foundry/periphery/PanopticHelper.t.sol
test/foundry/testUtils/PositionUtils.sol
test/foundry/testUtils/PriceMocks.sol
test/foundry/testUtils/ReentrancyMocks.sol
test/foundry/tokens/ERC1155Minimal.t.sol
test/foundry/types/LeftRight.t.sol
test/foundry/types/LiquidityChunk.t.sol
test/foundry/types/TokenId.t.sol
test/foundry/types/harnesses/LeftRightHarness.sol
test/foundry/types/harnesses/LiquidityChunkHarness.sol
test/foundry/types/harnesses/TokenIdHarness.sol

Scoping Q & A

General questions

Question Answer
Test coverage -
ERC20 used by the protocol any
ERC721 used by the protocol any
ERC777 used by the protocol no
ERC1155 used by the protocol SFPM and factory ERC1155 tokens
Chains the protocol will be deployed on Ethereum, Arbitrum, Avax, Base, BSC, Optimism ,Polygon

ERC20 token behaviors in scope

Question Answer
Missing return values ✅ Yes
Fee on transfer ❌ No
Balance changes outside of transfers ❌ No
Upgradeability ❌ No
Flash minting ✅ Yes
Pausability ❌ No
Approval race protections ✅ Yes
Revert on approval to zero address ✅ Yes
Revert on zero value approvals ✅ Yes
Revert on zero value transfers ✅ Yes
Revert on transfer to the zero address ✅ Yes
Revert on large approvals and/or transfers ❌ No
Doesn't revert on failure ❌ No
Multiple token addresses ❌ No
Low decimals ( < 6) ✅ Yes
High decimals ( > 18) ✅ Yes
Blocklists ❌ No

External integrations (e.g., Uniswap) behavior in scope:

Question Answer
Enabling/disabling fees (e.g. Blur disables/enables fees) No
Pausability (e.g. Uniswap pool gets paused) No
Upgradeability (e.g. Uniswap gets upgraded) No

EIP compliance checklist

N/A

Additional context

Main invariants

SFPM:

  • Users of the SFPM can only remove liquidity (via isLong==1 or burning positions) that they have previously added under the same (tickLower, tickUpper, tokenType) key. They cannot remove liquidity owned by other users.
  • Fees collected from Uniswap during any given operation should not exceed the amount of fees earned by the liquidity owned by the user performing the operation.
  • Fees paid to a given user should not exceed the amount of fees earned by the liquidity owned by that user. Panoptic:
  • Commission paid by an account should be greater than or equal to 1 (of either) token
  • Option sellers must pay back the exact amounts of tokens that were borrowed (the "shortAmounts") when their positions are burned
  • Option buyers must add back the same amount of liquidity that was borrowed when their positions are burned
  • Users should not be allowed to mint/burn options or pay premium if their end state is insolvent (at the fast oracle tick, or both ticks if the fast and slow oracle ticks are further away than the threshold)
  • Users should not be allowed to withdraw collateral if they have open positions
  • Users should not be allowed to mint more positions than the limit
  • Option sellers should not receive premium that has not settled (by collection from Uniswap or payments from long option holders) yet
  • Option sellers in a given chunk should not receive premium, as a fraction of the premium owed to them , that is greater than the ratio of total owed premium to total settled premium in that chunk
  • Liquidations must only occur if the liquidated account is insolvent at the TWAP tick and the TWAP tick is not above the threshold for distance to the current tick
  • Note that some executable liquidations may not be profitable due to the required actions to execute the liquidation (depositing collateral, force exercises) or the price the token conversions occur at
  • Note that it is acceptable for an account to cause protocol loss (shares minted to liquidator) during a liquidation even if they have a residual token balance in their account, this may happen if the value of tokens remaining correspond to less than 1 of the other token
  • If, at the final state after a liquidation, any premium paid to sellers DURING the liquidation has not been revoked, there must not be any protocol loss (shares minted to liquidator). In other words, this means that premium cannot be paid over the protocol loss threshold.
  • The value of the bonus paid to the liquidator must not exceed the liquidated user's collateral balance at the beginning of the call

Attack ideas (where to focus for bugs)

The factory contract and usage of libraries by external integrators is relatively unimportant -- wardens should focus their efforts on the security of the SFPM (previously audited by C4, code has changed since that audit however), PanopticPool, and CollateralTracker

All trusted roles in the protocol

N/A There is a factory owner but they do not have any permissions over the contracts in scope.

Running tests

⚠️Note: You will need to provide your own Ethereum Mainnet eth_rpc_url (Works best with a local archives node) in the Foundry.toml.

git clone --recurse-submodules https://github.com/code-423n4/2024-04-panoptic.git
cd 2024-04-panoptic
export FOUNDRY_PROFILE=ci_test  # (tests WILL fail without this because of a Foundry bug)
forge build
forge test

To run gas benchmarks:

forge test --gas-report

Miscellaneous

Employees of Panoptic and employees' family members are ineligible to participate in this audit.

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published