-
Notifications
You must be signed in to change notification settings - Fork 25
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
20 changed files
with
836 additions
and
100 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
// SPDX-License-Identifier: GPL-3.0 | ||
pragma solidity ^0.8.25; | ||
|
||
/// @title Sparse Calldata Segment Library | ||
/// @notice Library for working with sparsely-packed calldata segments, identified with an index. | ||
/// @dev The first byte of each segment is the index of the segment. | ||
/// To prevent accidental stack-to-deep errors, the body and index of the segment are extracted separately, rather | ||
/// than inline as part of the tuple returned by `getNextSegment`. | ||
library SparseCalldataSegmentLib { | ||
/// @notice Splits out a segment of calldata, sparsely-packed. | ||
/// The expected format is: | ||
/// [uint32(len(segment0)), segment0, uint32(len(segment1)), segment1, ... uint32(len(segmentN)), segmentN] | ||
/// @param source The calldata to extract the segment from. | ||
/// @return segment The extracted segment. Using the above example, this would be segment0. | ||
/// @return remainder The remaining calldata. Using the above example, | ||
/// this would start at uint32(len(segment1)) and continue to the end at segmentN. | ||
function getNextSegment(bytes calldata source) | ||
internal | ||
pure | ||
returns (bytes calldata segment, bytes calldata remainder) | ||
{ | ||
// The first 4 bytes hold the length of the segment, excluding the index. | ||
uint32 length = uint32(bytes4(source[:4])); | ||
|
||
// The offset of the remainder of the calldata. | ||
uint256 remainderOffset = 4 + length; | ||
|
||
// The segment is the next `length` + 1 bytes, to account for the index. | ||
// By convention, the first byte of each segment is the index of the segment. | ||
segment = source[4:remainderOffset]; | ||
|
||
// The remainder is the rest of the calldata. | ||
remainder = source[remainderOffset:]; | ||
} | ||
|
||
/// @notice Extracts the index from a segment. | ||
/// @dev The first byte of the segment is the index. | ||
/// @param segment The segment to extract the index from | ||
/// @return The index of the segment | ||
function getIndex(bytes calldata segment) internal pure returns (uint8) { | ||
return uint8(segment[0]); | ||
} | ||
|
||
/// @notice Extracts the body from a segment. | ||
/// @dev The body is the segment without the index. | ||
/// @param segment The segment to extract the body from | ||
/// @return The body of the segment. | ||
function getBody(bytes calldata segment) internal pure returns (bytes calldata) { | ||
return segment[1:]; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.