Skip to content

Commit

Permalink
feat: do not callback hook if caller is hook
Browse files Browse the repository at this point in the history
  • Loading branch information
ChefMist committed Apr 29, 2024
1 parent 2eb67b3 commit 8624ae1
Showing 55 changed files with 1,190 additions and 96 deletions.
2 changes: 1 addition & 1 deletion .forge-snapshots/BinHookTest#testBurnSucceedsWithHook.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
143699
142950
Original file line number Diff line number Diff line change
@@ -1 +1 @@
135043
134297
Original file line number Diff line number Diff line change
@@ -1 +1 @@
108914
108533
2 changes: 1 addition & 1 deletion .forge-snapshots/BinHookTest#testMintSucceedsWithHook.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
301091
300345
2 changes: 1 addition & 1 deletion .forge-snapshots/BinHookTest#testSwapSucceedsWithHook.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
139136
138390
Original file line number Diff line number Diff line change
@@ -1 +1 @@
90479
93346
Original file line number Diff line number Diff line change
@@ -1 +1 @@
65672
68539
Original file line number Diff line number Diff line change
@@ -1 +1 @@
149227
152077
2 changes: 1 addition & 1 deletion .forge-snapshots/BinPoolManagerTest#testGasBurnOneBin.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
66851
69718
2 changes: 1 addition & 1 deletion .forge-snapshots/BinPoolManagerTest#testGasDonate.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
52444
55313
Original file line number Diff line number Diff line change
@@ -1 +1 @@
968444
971345
Original file line number Diff line number Diff line change
@@ -1 +1 @@
119958
122859
Original file line number Diff line number Diff line change
@@ -1 +1 @@
337634
340528
Original file line number Diff line number Diff line change
@@ -1 +1 @@
54695
57589
Original file line number Diff line number Diff line change
@@ -1 +1 @@
89312
92098
Original file line number Diff line number Diff line change
@@ -1 +1 @@
91297
94083
Original file line number Diff line number Diff line change
@@ -1 +1 @@
70741
73587
Original file line number Diff line number Diff line change
@@ -1 +1 @@
319341
322235
2 changes: 1 addition & 1 deletion .forge-snapshots/BinPoolManagerTest#testNoOpGas_Burn.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
41503
44517
Original file line number Diff line number Diff line change
@@ -1 +1 @@
19392
20863
Original file line number Diff line number Diff line change
@@ -1 +1 @@
37117
42510
2 changes: 1 addition & 1 deletion .forge-snapshots/BinPoolManagerTest#testNoOpGas_Mint.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
39136
42175
2 changes: 1 addition & 1 deletion .forge-snapshots/BinPoolManagerTest#testNoOpGas_Swap.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
22507
25539
Original file line number Diff line number Diff line change
@@ -1 +1 @@
8148
8153
Original file line number Diff line number Diff line change
@@ -1 +1 @@
348848
351744
Original file line number Diff line number Diff line change
@@ -1 +1 @@
59396
62292
Original file line number Diff line number Diff line change
@@ -1 +1 @@
242441
245337
2 changes: 1 addition & 1 deletion .forge-snapshots/CLPoolManagerTest#donateBothTokens.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
82514
85388
2 changes: 1 addition & 1 deletion .forge-snapshots/CLPoolManagerTest#gasDonateOneToken.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
52476
55350
Original file line number Diff line number Diff line change
@@ -1 +1 @@
36547
41952
Original file line number Diff line number Diff line change
@@ -1 +1 @@
42374
45262
Original file line number Diff line number Diff line change
@@ -1 +1 @@
54672
57550
Original file line number Diff line number Diff line change
@@ -1 +1 @@
100748
103626
Original file line number Diff line number Diff line change
@@ -1 +1 @@
25040458
25043336
2 changes: 1 addition & 1 deletion .forge-snapshots/CLPoolManagerTest#swap_simple.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
35872
38750
Original file line number Diff line number Diff line change
@@ -1 +1 @@
101510
104388
2 changes: 1 addition & 1 deletion .forge-snapshots/CLPoolManagerTest#swap_withHooks.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
41522
44489
2 changes: 1 addition & 1 deletion .forge-snapshots/CLPoolManagerTest#swap_withNative.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
35875
38753
Original file line number Diff line number Diff line change
@@ -1 +1 @@
19218
22231
Original file line number Diff line number Diff line change
@@ -1 +1 @@
37497
42902
Original file line number Diff line number Diff line change
@@ -1 +1 @@
29405
32427
2 changes: 1 addition & 1 deletion .forge-snapshots/CLPoolManagerTest#testNoOp_gas_Swap.snap
Original file line number Diff line number Diff line change
@@ -1 +1 @@
21667
24680
12 changes: 10 additions & 2 deletions src/libraries/Hooks.sol
Original file line number Diff line number Diff line change
@@ -45,14 +45,22 @@ library Hooks {
}
}

function shouldCall(bytes32 parameters, uint8 offset) internal pure returns (bool) {
/// @return true if parameter has offset enabled
function hasOffsetEnabled(bytes32 parameters, uint8 offset) internal pure returns (bool) {
return parameters.decodeBool(offset);
}

/// @notice checks if hook should be called -- based on 2 factors:
/// 1. whether pool.parameters has the callback offset registered
/// 2. whether msg.sender is the hook itself
function shouldCall(bytes32 parameters, uint8 offset, IHooks hook) internal view returns (bool) {
return hasOffsetEnabled(parameters, offset) && address(hook) != msg.sender;
}

/// @dev Verify hook return value matches no-op when these 2 conditions are met
/// 1) Hook have permission for no-op
/// 2) Return value is no-op selector
function isValidNoOpCall(bytes32 parameters, uint8 noOpOffset, bytes4 selector) internal pure returns (bool) {
return shouldCall(parameters, noOpOffset) && selector == NO_OP_SELECTOR;
return hasOffsetEnabled(parameters, noOpOffset) && selector == NO_OP_SELECTOR;
}
}
30 changes: 15 additions & 15 deletions src/pool-bin/BinPoolManager.sol
Original file line number Diff line number Diff line change
@@ -106,7 +106,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
uint24 swapFee = key.fee.getInitialSwapFee();
if (swapFee.isSwapFeeTooLarge(SwapFeeLibrary.TEN_PERCENT_FEE)) revert FeeTooLarge();

if (key.parameters.shouldCall(HOOKS_BEFORE_INITIALIZE_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_BEFORE_INITIALIZE_OFFSET, key.hooks)) {
if (hooks.beforeInitialize(msg.sender, key, activeId, hookData) != IBinHooks.beforeInitialize.selector) {
revert Hooks.InvalidHookResponse();
}
@@ -120,7 +120,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
/// @notice Make sure the first event is noted, so that later events from afterHook won't get mixed up with this one
emit Initialize(id, key.currency0, key.currency1, key.fee, binStep, hooks);

if (key.parameters.shouldCall(HOOKS_AFTER_INITIALIZE_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_AFTER_INITIALIZE_OFFSET, key.hooks)) {
if (hooks.afterInitialize(msg.sender, key, activeId, hookData) != IBinHooks.afterInitialize.selector) {
revert Hooks.InvalidHookResponse();
}
@@ -139,7 +139,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
_checkPoolInitialized(id);

IBinHooks hooks = IBinHooks(address(key.hooks));
if (key.parameters.shouldCall(HOOKS_BEFORE_SWAP_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_BEFORE_SWAP_OFFSET, key.hooks)) {
bytes4 selector = hooks.beforeSwap(msg.sender, key, swapForY, amountIn, hookData);
if (key.parameters.isValidNoOpCall(HOOKS_NO_OP_OFFSET, selector)) {
// Sentinel return value used to signify that a NoOp occurred.
@@ -170,7 +170,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
emit Swap(id, msg.sender, delta.amount0(), delta.amount1(), activeId, swapFee, feeForProtocol);
}

if (key.parameters.shouldCall(HOOKS_AFTER_SWAP_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_AFTER_SWAP_OFFSET, key.hooks)) {
if (hooks.afterSwap(msg.sender, key, swapForY, amountIn, delta, hookData) != IBinHooks.afterSwap.selector) {
revert Hooks.InvalidHookResponse();
}
@@ -241,7 +241,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
_checkPoolInitialized(id);

IBinHooks hooks = IBinHooks(address(key.hooks));
if (key.parameters.shouldCall(HOOKS_BEFORE_MINT_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_BEFORE_MINT_OFFSET, key.hooks)) {
bytes4 selector = hooks.beforeMint(msg.sender, key, params, hookData);
if (key.parameters.isValidNoOpCall(HOOKS_NO_OP_OFFSET, selector)) {
// Sentinel return value used to signify that a NoOp occurred.
@@ -274,7 +274,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
/// @notice Make sure the first event is noted, so that later events from afterHook won't get mixed up with this one
emit Mint(id, msg.sender, mintArray.ids, mintArray.amounts, compositionFee, feeForProtocol);

if (key.parameters.shouldCall(HOOKS_AFTER_MINT_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_AFTER_MINT_OFFSET, key.hooks)) {
if (hooks.afterMint(msg.sender, key, params, delta, hookData) != IBinHooks.afterMint.selector) {
revert Hooks.InvalidHookResponse();
}
@@ -292,7 +292,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
_checkPoolInitialized(id);

IBinHooks hooks = IBinHooks(address(key.hooks));
if (key.parameters.shouldCall(HOOKS_BEFORE_BURN_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_BEFORE_BURN_OFFSET, key.hooks)) {
bytes4 selector = hooks.beforeBurn(msg.sender, key, params, hookData);
if (key.parameters.isValidNoOpCall(HOOKS_NO_OP_OFFSET, selector)) {
// Sentinel return value used to signify that a NoOp occurred.
@@ -312,7 +312,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
/// @notice Make sure the first event is noted, so that later events from afterHook won't get mixed up with this one
emit Burn(id, msg.sender, binIds, amountRemoved);

if (key.parameters.shouldCall(HOOKS_AFTER_BURN_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_AFTER_BURN_OFFSET, key.hooks)) {
if (hooks.afterBurn(msg.sender, key, params, delta, hookData) != IBinHooks.afterBurn.selector) {
revert Hooks.InvalidHookResponse();
}
@@ -330,7 +330,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
_checkPoolInitialized(id);

IBinHooks hooks = IBinHooks(address(key.hooks));
if (key.parameters.shouldCall(HOOKS_BEFORE_DONATE_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_BEFORE_DONATE_OFFSET, key.hooks)) {
bytes4 selector = hooks.beforeDonate(msg.sender, key, amount0, amount1, hookData);
if (key.parameters.isValidNoOpCall(HOOKS_NO_OP_OFFSET, selector)) {
// Sentinel return value used to signify that a NoOp occurred.
@@ -347,7 +347,7 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {
/// @notice Make sure the first event is noted, so that later events from afterHook won't get mixed up with this one
emit Donate(id, msg.sender, delta.amount0(), delta.amount1(), binId);

if (key.parameters.shouldCall(HOOKS_AFTER_DONATE_OFFSET)) {
if (key.parameters.shouldCall(HOOKS_AFTER_DONATE_OFFSET, key.hooks)) {
if (hooks.afterDonate(msg.sender, key, amount0, amount1, hookData) != IBinHooks.afterDonate.selector) {
revert Hooks.InvalidHookResponse();
}
@@ -386,12 +386,12 @@ contract BinPoolManager is IBinPoolManager, Fees, Extsload {

function _validateHookNoOp(PoolKey memory key) internal pure {
// if no-op is active for hook, there must be a before* hook active too
if (key.parameters.shouldCall(HOOKS_NO_OP_OFFSET)) {
if (key.parameters.hasOffsetEnabled(HOOKS_NO_OP_OFFSET)) {
if (
!key.parameters.shouldCall(HOOKS_BEFORE_MINT_OFFSET)
&& !key.parameters.shouldCall(HOOKS_BEFORE_BURN_OFFSET)
&& !key.parameters.shouldCall(HOOKS_BEFORE_SWAP_OFFSET)
&& !key.parameters.shouldCall(HOOKS_BEFORE_DONATE_OFFSET)
!key.parameters.hasOffsetEnabled(HOOKS_BEFORE_MINT_OFFSET)
&& !key.parameters.hasOffsetEnabled(HOOKS_BEFORE_BURN_OFFSET)
&& !key.parameters.hasOffsetEnabled(HOOKS_BEFORE_SWAP_OFFSET)
&& !key.parameters.hasOffsetEnabled(HOOKS_BEFORE_DONATE_OFFSET)
) {
revert Hooks.NoOpHookMissingBeforeCall();
}
Loading

0 comments on commit 8624ae1

Please sign in to comment.