From c484d715e92eed6c5d56fdfe6c9269ab83c024a3 Mon Sep 17 00:00:00 2001 From: 0age <0age@protonmail.com> Date: Tue, 5 Mar 2024 20:00:02 -0500 Subject: [PATCH] skip execution filtering but do not transfer when amount is zero --- src/core/lib/FulfillmentApplier.sol | 9 -- src/core/lib/OrderCombiner.sol | 156 +++++++++------------------- 2 files changed, 50 insertions(+), 115 deletions(-) diff --git a/src/core/lib/FulfillmentApplier.sol b/src/core/lib/FulfillmentApplier.sol index 405e230..a13dab8 100644 --- a/src/core/lib/FulfillmentApplier.sol +++ b/src/core/lib/FulfillmentApplier.sol @@ -283,15 +283,6 @@ contract FulfillmentApplier is FulfillmentApplicationErrors { // Set fulfiller conduit key as the conduit key on execution. execution.conduitKey = fulfillerConduitKey; } - - // Set the offerer and recipient to null address and the item type - // to a non-native item type if the execution amount is zero. This - // will cause the execution item to be skipped. - if (item.amount == 0) { - execution.offerer = address(0); - item.recipient = payable(0); - item.itemType = ItemType.ERC20; - } } } diff --git a/src/core/lib/OrderCombiner.sol b/src/core/lib/OrderCombiner.sol index d6dbece..9f7ee27 100644 --- a/src/core/lib/OrderCombiner.sol +++ b/src/core/lib/OrderCombiner.sol @@ -682,67 +682,30 @@ contract OrderCombiner is OrderFulfiller, FulfillmentApplier { // Skip overflow checks as all for loops are indexed starting at zero. unchecked { - // Track number of filtered executions. - uint256 totalFilteredExecutions = 0; - // Iterate over each offer fulfillment. - for (uint256 i = 0; i < totalOfferFulfillments;) { - // Derive aggregated execution corresponding with fulfillment. - Execution memory execution = _aggregateAvailable( + for (uint256 i = 0; i < totalOfferFulfillments; ++i) { + // Derive aggregated execution corresponding with fulfillment + // and assign it to the executions array. + executions[i] = _aggregateAvailable( advancedOrders, Side.OFFER, offerFulfillments[i], fulfillerConduitKey, recipient ); - - // If the execution is filterable... - if (_isFilterableExecution(execution)) { - // Increment total filtered executions. - ++totalFilteredExecutions; - } else { - // Otherwise, assign the execution to the executions array. - executions[i - totalFilteredExecutions] = execution; - } - - // Increment iterator. - ++i; } // Iterate over each consideration fulfillment. - for (uint256 i = 0; i < totalConsiderationFulfillments;) { - // Derive aggregated execution corresponding with fulfillment. - Execution memory execution = _aggregateAvailable( + for (uint256 i = 0; i < totalConsiderationFulfillments; ++i) { + // Derive aggregated execution corresponding with fulfillment + // and assign it to the executions array. + executions[i + totalOfferFulfillments] = _aggregateAvailable( advancedOrders, Side.CONSIDERATION, considerationFulfillments[i], fulfillerConduitKey, address(0) // unused ); - - // If the execution is filterable... - if (_isFilterableExecution(execution)) { - // Increment total filtered executions. - ++totalFilteredExecutions; - } else { - // Otherwise, assign the execution to the executions array. - executions[i + totalOfferFulfillments - - totalFilteredExecutions] = execution; - } - - // Increment iterator. - ++i; - } - - // If some number of executions have been filtered... - if (totalFilteredExecutions != 0) { - // reduce the total length of the executions array. - assembly { - mstore( - executions, - sub(mload(executions), totalFilteredExecutions) - ) - } } } @@ -815,46 +778,49 @@ contract OrderCombiner is OrderFulfiller, FulfillmentApplier { _getReadExecutionByOffset()(executions, i) ); - // Retrieve the associated received item. + // Retrieve the associated received item and amount. ReceivedItem memory item = execution.item; + uint256 amount = item.amount; + + // Transfer the item specified by the execution as long as the + // execution is not a zero-amount execution (which can occur if + // the corresponding fulfillment contained only items on orders + // that are unavailable or are out of range of the respective + // item array). + if (amount != 0) { + // Utilize assembly to check for native token balance. + assembly { + // Ensure a sufficient native balance if relevant. + if and( + iszero(mload(item)), // itemType == ItemType.NATIVE + // item.amount > address(this).balance + gt(amount, selfbalance()) + ) { + // Store left-padded selector with push4, + // mem[28:32] = selector + mstore( + 0, + InsufficientNativeTokensSupplied_error_selector + ) - // If execution transfers native tokens, check available value. - assembly { - // Ensure sufficient available balance for native transfers. - if and( - iszero(mload(item)), // itemType == ItemType.NATIVE - // item.amount > address(this).balance - gt( - mload( - add( - item, - ReceivedItem_amount_offset - ) - ), - selfbalance() - ) - ) { - // Store left-padded selector with push4, - // mem[28:32] = selector - mstore( - 0, - InsufficientNativeTokensSupplied_error_selector - ) - - // revert(abi.encodeWithSignature( - // "InsufficientNativeTokensSupplied()" - // )) - revert( - Error_selector_offset, - InsufficientNativeTokensSupplied_error_length - ) + // revert(abi.encodeWithSignature( + // "InsufficientNativeTokensSupplied()" + // )) + revert( + Error_selector_offset, + InsufficientNativeTokensSupplied_error_length + ) + } } - } - // Transfer the item specified by the execution. - _transfer( - item, execution.offerer, execution.conduitKey, accumulator - ); + // Transfer the item specified by the execution. + _transfer( + item, + execution.offerer, + execution.conduitKey, + accumulator + ); + } } } @@ -1159,41 +1125,19 @@ contract OrderCombiner is OrderFulfiller, FulfillmentApplier { // Skip overflow checks as all for loops are indexed starting at zero. unchecked { - // Track number of filtered executions. - uint256 totalFilteredExecutions = 0; - // Iterate over each fulfillment. for (uint256 i = 0; i < totalFulfillments; ++i) { /// Retrieve the fulfillment in question. Fulfillment memory fulfillment = fulfillments[i]; - // Derive the execution corresponding with the fulfillment. - Execution memory execution = _applyFulfillment( + // Derive the execution corresponding with the fulfillment and + // assign it to the executions array. + executions[i] = _applyFulfillment( advancedOrders, fulfillment.offerComponents, fulfillment.considerationComponents, i ); - - // If the execution is filterable... - if (_isFilterableExecution(execution)) { - // Increment total filtered executions. - ++totalFilteredExecutions; - } else { - // Otherwise, assign the execution to the executions array. - executions[i - totalFilteredExecutions] = execution; - } - } - - // If some number of executions have been filtered... - if (totalFilteredExecutions != 0) { - // reduce the total length of the executions array. - assembly { - mstore( - executions, - sub(mload(executions), totalFilteredExecutions) - ) - } } }