Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

beforeSwap returns dynamic fee #648

Merged
merged 26 commits into from
May 17, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
36e7653
allow beforeSwap to return and set the lpFee
saucepoint May 13, 2024
07a9d30
fix tests; default to a sentinel for Hooks.beforeSwap
saucepoint May 13, 2024
28311ce
merge in main; regenerate snapshots
saucepoint May 14, 2024
7406eb9
define constant
saucepoint May 14, 2024
761ccb9
very smol optimization
saucepoint May 14, 2024
f424c1b
Optimise before swap return fee (#650)
hensha256 May 14, 2024
ad6dcf6
Try shared parse lib (#652)
snreynolds May 14, 2024
cd95832
Update src/interfaces/IHooks.sol
saucepoint May 14, 2024
52ef386
misc
saucepoint May 14, 2024
132a877
merge in main; regenerate snapshots
saucepoint May 14, 2024
616bc84
merge in main; regenerate snapshots; fix test
saucepoint May 15, 2024
f508681
review comments; abstract fee checks; add new test
saucepoint May 15, 2024
b5929ae
make ProtocolFeeLibrary similar to LPFeeLibrary
saucepoint May 15, 2024
3a9747c
fix test
saucepoint May 15, 2024
160e4c6
rename how protocol fee is validated
saucepoint May 16, 2024
196af36
fee masking
saucepoint May 16, 2024
525108e
trial new override mechanism
hensha256 May 16, 2024
fdfa579
remove assembly block
hensha256 May 16, 2024
afcda81
cleanup; some natspec
saucepoint May 16, 2024
2ffaa92
revert if override fee exceeds the maximum
saucepoint May 16, 2024
60e1060
review comments; consolidate fee flag
saucepoint May 16, 2024
9499de9
use 2nd bit of uint24 to signal override
saucepoint May 16, 2024
bac221f
naming
saucepoint May 16, 2024
ef84cb0
Update src/libraries/LPFeeLibrary.sol
saucepoint May 17, 2024
35a1402
Update src/libraries/LPFeeLibrary.sol
saucepoint May 17, 2024
abb6988
additional code comment
saucepoint May 17, 2024
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
2 changes: 1 addition & 1 deletion .forge-snapshots/addLiquidity CA fee.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
329540
329350
2 changes: 1 addition & 1 deletion .forge-snapshots/addLiquidity with empty hook.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
284175
284203
2 changes: 1 addition & 1 deletion .forge-snapshots/poolManager bytecode size.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
23817
23822
2 changes: 1 addition & 1 deletion .forge-snapshots/removeLiquidity CA fee.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
185027
184837
2 changes: 1 addition & 1 deletion .forge-snapshots/simple swap with native.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
117381
117523
2 changes: 1 addition & 1 deletion .forge-snapshots/simple swap.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
132620
132762
2 changes: 1 addition & 1 deletion .forge-snapshots/swap CA custom curve + swap noop.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
135892
135819
2 changes: 1 addition & 1 deletion .forge-snapshots/swap CA fee on unspecified.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
184887
184839
Original file line number Diff line number Diff line change
@@ -1 +1 @@
113834
113976
2 changes: 1 addition & 1 deletion .forge-snapshots/swap against liquidity.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
125255
125397
2 changes: 1 addition & 1 deletion .forge-snapshots/swap burn 6909 for input.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
137257
137399
2 changes: 1 addition & 1 deletion .forge-snapshots/swap burn native 6909 for input.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
126357
126499
2 changes: 1 addition & 1 deletion .forge-snapshots/swap mint native output as 6909.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
148435
148577
2 changes: 1 addition & 1 deletion .forge-snapshots/swap mint output as 6909.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
165244
165386
Original file line number Diff line number Diff line change
@@ -1 +1 @@
224777
225026
2 changes: 1 addition & 1 deletion .forge-snapshots/swap with dynamic fee.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
149511
149653
2 changes: 1 addition & 1 deletion .forge-snapshots/swap with hooks.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
125267
125409
2 changes: 1 addition & 1 deletion .forge-snapshots/swap with lp fee and protocol fee.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
181833
181957
1 change: 1 addition & 0 deletions .forge-snapshots/swap with return dynamic fee.snap
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
157369
2 changes: 1 addition & 1 deletion .forge-snapshots/update dynamic fee in before swap.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
160064
160194
32 changes: 19 additions & 13 deletions src/PoolManager.sol
Original file line number Diff line number Diff line change
Expand Up @@ -271,19 +271,25 @@ contract PoolManager is IPoolManager, ProtocolFees, NoDelegateCall, ERC6909Claim
PoolId id = key.toId();
_checkPoolInitialized(id);

(int256 amountToSwap, BeforeSwapDelta beforeSwapDelta) = key.hooks.beforeSwap(key, params, hookData);

// execute swap, account protocol fees, and emit swap event
swapDelta = _swap(
id,
Pool.SwapParams({
tickSpacing: key.tickSpacing,
zeroForOne: params.zeroForOne,
amountSpecified: amountToSwap,
sqrtPriceLimitX96: params.sqrtPriceLimitX96
}),
params.zeroForOne ? key.currency0 : key.currency1 // input token
);
BeforeSwapDelta beforeSwapDelta;
{
hensha256 marked this conversation as resolved.
Show resolved Hide resolved
int256 amountToSwap;
uint24 fee;
(amountToSwap, beforeSwapDelta, fee) = key.hooks.beforeSwap(key, params, hookData);

// execute swap, account protocol fees, and emit swap event
swapDelta = _swap(
id,
Pool.SwapParams({
tickSpacing: key.tickSpacing,
zeroForOne: params.zeroForOne,
amountSpecified: amountToSwap,
sqrtPriceLimitX96: params.sqrtPriceLimitX96,
fee: fee
}),
params.zeroForOne ? key.currency0 : key.currency1 // input token
);
}

BalanceDelta hookDelta;
(swapDelta, hookDelta) = key.hooks.afterSwap(key, params, swapDelta, hookData, beforeSwapDelta);
Expand Down
3 changes: 2 additions & 1 deletion src/interfaces/IHooks.sol
Original file line number Diff line number Diff line change
Expand Up @@ -99,12 +99,13 @@ interface IHooks {
/// @param hookData Arbitrary data handed into the PoolManager by the swapper to be be passed on to the hook
/// @return bytes4 The function selector for the hook
/// @return BeforeSwapDelta The hook's delta in specified and unspecified currencies. Positive: the hook is owed/took currency, negative: the hook owes/sent currency
/// @return uint24 An optional swap fee, only used if the Pool has a dynamic fee and the value is less than of equal to LPFeeLibrary.MAX_LP_FEE
saucepoint marked this conversation as resolved.
Show resolved Hide resolved
function beforeSwap(
address sender,
PoolKey calldata key,
IPoolManager.SwapParams calldata params,
bytes calldata hookData
) external returns (bytes4, BeforeSwapDelta);
) external returns (bytes4, BeforeSwapDelta, uint24);

/// @notice The hook called after a swap
/// @param sender The initial msg.sender for the swap call
Expand Down
34 changes: 15 additions & 19 deletions src/libraries/Hooks.sol
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import {LPFeeLibrary} from "./LPFeeLibrary.sol";
import {BalanceDelta, toBalanceDelta, BalanceDeltaLibrary} from "../types/BalanceDelta.sol";
import {BeforeSwapDelta, BeforeSwapDeltaLibrary} from "../types/BeforeSwapDelta.sol";
import {IPoolManager} from "../interfaces/IPoolManager.sol";
import {ParseBytes} from "../libraries/ParseBytes.sol";

/// @notice V4 decides whether to invoke specific hooks by inspecting the leading bits of the address that
/// the hooks contract is deployed to.
Expand All @@ -18,6 +19,7 @@ library Hooks {
using Hooks for IHooks;
using SafeCast for int256;
using BeforeSwapDeltaLibrary for BeforeSwapDelta;
using ParseBytes for bytes;

uint256 internal constant BEFORE_INITIALIZE_FLAG = 1 << 159;
uint256 internal constant AFTER_INITIALIZE_FLAG = 1 << 158;
Expand Down Expand Up @@ -125,14 +127,8 @@ library Hooks {
(success, result) = address(self).call(data);
if (!success) _revert(result);

bytes4 expectedSelector;
bytes4 selector;
assembly {
expectedSelector := mload(add(data, 0x20))
selector := mload(add(result, 0x20))
}

if (selector != expectedSelector) revert InvalidHookResponse();
// Check expected selector and returned selector match.
if (result.parseSelector() != data.parseSelector()) revert InvalidHookResponse();
hensha256 marked this conversation as resolved.
Show resolved Hide resolved
}

/// @notice performs a hook call using the given calldata on the given hook
Expand All @@ -145,7 +141,7 @@ library Hooks {

// If this hook wasnt meant to return something, default to 0 delta
if (!parseReturn) return 0;
(, delta) = abi.decode(result, (bytes4, int256));
return result.parseReturnDelta();
}

/// @notice modifier to prevent calling a hook if they initiated the action
Expand Down Expand Up @@ -236,22 +232,22 @@ library Hooks {
/// @notice calls beforeSwap hook if permissioned and validates return value
function beforeSwap(IHooks self, PoolKey memory key, IPoolManager.SwapParams memory params, bytes calldata hookData)
internal
returns (int256 amountToSwap, BeforeSwapDelta hookReturn)
returns (int256 amountToSwap, BeforeSwapDelta hookReturn, uint24 lpFee)
{
amountToSwap = params.amountSpecified;
if (msg.sender == address(self)) return (amountToSwap, BeforeSwapDeltaLibrary.ZERO_DELTA);
lpFee = LPFeeLibrary.MAX_UINT24;
if (msg.sender == address(self)) return (amountToSwap, BeforeSwapDeltaLibrary.ZERO_DELTA, lpFee);

if (self.hasPermission(BEFORE_SWAP_FLAG)) {
bool canReturnDelta = self.hasPermission(BEFORE_SWAP_RETURNS_DELTA_FLAG);
hookReturn = BeforeSwapDelta.wrap(
self.callHookWithReturnDelta(
abi.encodeWithSelector(IHooks.beforeSwap.selector, msg.sender, key, params, hookData),
canReturnDelta
)
);
bytes memory result =
callHook(self, abi.encodeWithSelector(IHooks.beforeSwap.selector, msg.sender, key, params, hookData));

if (key.fee.isDynamicFee()) lpFee = result.parseFee();

// skip this logic for the case where the hook return is 0
if (canReturnDelta) {
if (self.hasPermission(BEFORE_SWAP_RETURNS_DELTA_FLAG)) {
hookReturn = BeforeSwapDelta.wrap(result.parseReturnDelta());

// any return in unspecified is passed to the afterSwap hook for handling
int128 hookDeltaSpecified = hookReturn.getSpecifiedDelta();

Expand Down
1 change: 1 addition & 0 deletions src/libraries/LPFeeLibrary.sol
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ library LPFeeLibrary {

// the lp fee is represented in hundredths of a bip, so the max is 100%
uint24 public constant MAX_LP_FEE = 1000000;
uint24 public constant MAX_UINT24 = type(uint24).max;
saucepoint marked this conversation as resolved.
Show resolved Hide resolved

function isDynamicFee(uint24 self) internal pure returns (bool) {
return self & DYNAMIC_FEE_FLAG != 0;
Expand Down
28 changes: 28 additions & 0 deletions src/libraries/ParseBytes.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
// SPDX-License-Identifier: GPL-2.0-or-later
pragma solidity ^0.8.20;

/// @notice Parses bytes returned from hooks and the byte selector used to check return selectors from hooks.
/// @dev parseSelector also is used to parse the expected selector
/// For parsing hook returns, note that all hooks return either bytes4 or (bytes4, 32-byte-delta) or (bytes4, 32-byte-delta, uint24).
library ParseBytes {
function parseSelector(bytes memory result) internal pure returns (bytes4 selector) {
// equivalent: (selector,) = abi.decode(result, (bytes4, int256));
assembly {
selector := mload(add(result, 0x20))
}
}

function parseFee(bytes memory result) internal pure returns (uint24 lpFee) {
// equivalent: (,, lpFee) = abi.decode(result, (bytes4, int256, uint24));
assembly {
lpFee := mload(add(result, 0x60))
}
}

function parseReturnDelta(bytes memory result) internal pure returns (int256 hookReturn) {
// equivalent: (, hookReturnDelta) = abi.decode(result, (bytes4, int256));
assembly {
hookReturn := mload(add(result, 0x40))
}
}
}
5 changes: 3 additions & 2 deletions src/libraries/Pool.sol
Original file line number Diff line number Diff line change
Expand Up @@ -294,6 +294,7 @@ library Pool {
bool zeroForOne;
int256 amountSpecified;
uint160 sqrtPriceLimitX96;
uint24 fee;
}

/// @notice Executes a swap against the state, and returns the amount deltas of the pool
Expand All @@ -317,8 +318,8 @@ library Pool {
state.feeGrowthGlobalX128 = zeroForOne ? self.feeGrowthGlobal0X128 : self.feeGrowthGlobal1X128;
state.liquidity = cache.liquidityStart;

swapFee =
cache.protocolFee == 0 ? slot0Start.lpFee : uint24(cache.protocolFee).calculateSwapFee(slot0Start.lpFee);
uint24 _lpFee = params.fee <= LPFeeLibrary.MAX_LP_FEE ? params.fee : slot0Start.lpFee;
saucepoint marked this conversation as resolved.
Show resolved Hide resolved
swapFee = cache.protocolFee == 0 ? _lpFee : uint24(cache.protocolFee).calculateSwapFee(_lpFee);

bool exactInput = params.amountSpecified < 0;

Expand Down
2 changes: 1 addition & 1 deletion src/test/BaseTestHooks.sol
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@ contract BaseTestHooks is IHooks {
PoolKey calldata, /* key **/
IPoolManager.SwapParams calldata, /* params **/
bytes calldata /* hookData **/
) external virtual returns (bytes4, BeforeSwapDelta) {
) external virtual returns (bytes4, BeforeSwapDelta, uint24) {
revert HookNotImplemented();
}

Expand Down
4 changes: 2 additions & 2 deletions src/test/CustomCurveHook.sol
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ contract CustomCurveHook is BaseTestHooks {
PoolKey calldata key,
IPoolManager.SwapParams calldata params,
bytes calldata /* hookData **/
) external override onlyPoolManager returns (bytes4, BeforeSwapDelta) {
) external override onlyPoolManager returns (bytes4, BeforeSwapDelta, uint24) {
(Currency inputCurrency, Currency outputCurrency, uint256 amount) = _getInputOutputAndAmount(key, params);

// this "custom curve" is a line, 1-1
Expand All @@ -47,7 +47,7 @@ contract CustomCurveHook is BaseTestHooks {

// return -amountSpecified as specified to no-op the concentrated liquidity swap
BeforeSwapDelta hookDelta = toBeforeSwapDelta(int128(-params.amountSpecified), int128(params.amountSpecified));
return (IHooks.beforeSwap.selector, hookDelta);
return (IHooks.beforeSwap.selector, hookDelta, 0);
}

function afterAddLiquidity(
Expand Down
4 changes: 2 additions & 2 deletions src/test/DeltaReturningHook.sol
Original file line number Diff line number Diff line change
Expand Up @@ -51,15 +51,15 @@ contract DeltaReturningHook is BaseTestHooks {
PoolKey calldata key,
IPoolManager.SwapParams calldata params,
bytes calldata /* hookData **/
) external override onlyPoolManager returns (bytes4, BeforeSwapDelta) {
) external override onlyPoolManager returns (bytes4, BeforeSwapDelta, uint24) {
(Currency specifiedCurrency, Currency unspecifiedCurrency) = _sortCurrencies(key, params);

if (deltaSpecified != 0) _settleOrTake(specifiedCurrency, deltaSpecified);
if (deltaUnspecifiedBeforeSwap != 0) _settleOrTake(unspecifiedCurrency, deltaUnspecifiedBeforeSwap);

BeforeSwapDelta beforeSwapDelta = toBeforeSwapDelta(deltaSpecified, deltaUnspecifiedBeforeSwap);

return (IHooks.beforeSwap.selector, beforeSwapDelta);
return (IHooks.beforeSwap.selector, beforeSwapDelta, 0);
}

function afterSwap(
Expand Down
5 changes: 3 additions & 2 deletions src/test/DynamicFeesTestHook.sol
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import {PoolKey} from "../types/PoolKey.sol";
import {IPoolManager} from "../interfaces/IPoolManager.sol";
import {IHooks} from "../interfaces/IHooks.sol";
import {BeforeSwapDelta, BeforeSwapDeltaLibrary} from "../types/BeforeSwapDelta.sol";
import {LPFeeLibrary} from "../libraries/LPFeeLibrary.sol";

contract DynamicFeesTestHook is BaseTestHooks {
uint24 internal fee;
Expand All @@ -31,10 +32,10 @@ contract DynamicFeesTestHook is BaseTestHooks {
function beforeSwap(address, PoolKey calldata key, IPoolManager.SwapParams calldata, bytes calldata)
external
override
returns (bytes4, BeforeSwapDelta)
returns (bytes4, BeforeSwapDelta, uint24)
{
manager.updateDynamicLPFee(key, fee);
return (IHooks.beforeSwap.selector, BeforeSwapDeltaLibrary.ZERO_DELTA);
return (IHooks.beforeSwap.selector, BeforeSwapDeltaLibrary.ZERO_DELTA, LPFeeLibrary.MAX_UINT24);
}

function forcePoolFeeUpdate(PoolKey calldata _key, uint24 _fee) external {
Expand Down
29 changes: 29 additions & 0 deletions src/test/DynamicReturnFeeTestHook.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.24;

import {BaseTestHooks} from "./BaseTestHooks.sol";
import {PoolKey} from "../types/PoolKey.sol";
import {IPoolManager} from "../interfaces/IPoolManager.sol";
import {IHooks} from "../interfaces/IHooks.sol";
import {BeforeSwapDelta, BeforeSwapDeltaLibrary} from "../types/BeforeSwapDelta.sol";

contract DynamicReturnFeeTestHook is BaseTestHooks {
uint24 internal fee;
IPoolManager manager;

function setManager(IPoolManager _manager) external {
manager = _manager;
}

function setFee(uint24 _fee) external {
fee = _fee;
}

function beforeSwap(address, PoolKey calldata, IPoolManager.SwapParams calldata, bytes calldata)
external
override
returns (bytes4, BeforeSwapDelta, uint24)
{
return (IHooks.beforeSwap.selector, BeforeSwapDeltaLibrary.ZERO_DELTA, fee);
}
}
4 changes: 2 additions & 2 deletions src/test/EmptyTestHooks.sol
Original file line number Diff line number Diff line change
Expand Up @@ -92,9 +92,9 @@ contract EmptyTestHooks is IHooks {
external
pure
override
returns (bytes4, BeforeSwapDelta)
returns (bytes4, BeforeSwapDelta, uint24)
{
return (IHooks.beforeSwap.selector, BeforeSwapDeltaLibrary.ZERO_DELTA);
return (IHooks.beforeSwap.selector, BeforeSwapDeltaLibrary.ZERO_DELTA, 0);
}

function afterSwap(address, PoolKey calldata, IPoolManager.SwapParams calldata, BalanceDelta, bytes calldata)
Expand Down
Loading