-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathBytesLib.sol
179 lines (152 loc) · 7.48 KB
/
BytesLib.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
// SPDX-License-Identifier: Unlicense
/*
* @title Solidity Bytes Arrays Utils
* @author Gonçalo Sá <goncalo.sa@consensys.net>
*
* @dev Bytes tightly packed arrays utility library for ethereum contracts written in Solidity.
* This is a reduced version of the library.
*/
pragma solidity >=0.8.0 <0.9.0;
library BytesLib {
uint256 private constant freeMemoryPtr = 0x40;
uint256 private constant maskModulo32 = 0x1f;
/**
* Size of word read by `mload` instruction.
*/
uint256 private constant memoryWord = 32;
uint256 internal constant uint8Size = 1;
uint256 internal constant uint16Size = 2;
uint256 internal constant uint32Size = 4;
uint256 internal constant uint64Size = 8;
uint256 internal constant uint128Size = 16;
uint256 internal constant uint256Size = 32;
uint256 internal constant addressSize = 20;
/**
* Bits in 12 bytes.
*/
uint256 private constant bytes12Bits = 96;
function slice(bytes memory buffer, uint256 startIndex, uint256 length) internal pure returns (bytes memory) {
unchecked {
require(length + 31 >= length, "slice_overflow");
}
require(buffer.length >= startIndex + length, "slice_outOfBounds");
bytes memory tempBytes;
assembly ("memory-safe") {
// Get a location of some free memory and store it in tempBytes as
// Solidity does for memory variables.
tempBytes := mload(freeMemoryPtr)
switch iszero(length)
case 0 {
// The first word of the slice result is potentially a partial
// word read from the original array. To read it, we calculate
// the length of that partial word and start copying that many
// bytes into the array. The first word we copy will start with
// data we don't care about, but the last `lengthmod` bytes will
// land at the beginning of the contents of the new array. When
// we're done copying, we overwrite the full first word with
// the actual length of the slice.
let lengthmod := and(length, maskModulo32)
// The multiplication in the next line is necessary
// because when slicing multiples of 32 bytes (lengthmod == 0)
// the following copy loop was copying the origin's length
// and then ending prematurely not copying everything it should.
let startOffset := add(lengthmod, mul(memoryWord, iszero(lengthmod)))
let dst := add(tempBytes, startOffset)
let end := add(dst, length)
for { let src := add(add(buffer, startOffset), startIndex) } lt(dst, end) {
dst := add(dst, memoryWord)
src := add(src, memoryWord)
} { mstore(dst, mload(src)) }
// Update free-memory pointer
// allocating the array padded to 32 bytes like the compiler does now
// Note that negating bitwise the `maskModulo32` produces a mask that aligns addressing to 32 bytes.
mstore(freeMemoryPtr, and(add(dst, maskModulo32), not(maskModulo32)))
}
//if we want a zero-length slice let's just return a zero-length array
default { mstore(freeMemoryPtr, add(tempBytes, memoryWord)) }
// Store the length of the buffer
// We need to do it even if the length is zero because Solidity does not garbage collect
mstore(tempBytes, length)
}
return tempBytes;
}
function toAddress(bytes memory buffer, uint256 startIndex) internal pure returns (address) {
require(buffer.length >= startIndex + addressSize, "toAddress_outOfBounds");
address tempAddress;
assembly ("memory-safe") {
// We want to shift into the lower 12 bytes and leave the upper 12 bytes clear.
tempAddress := shr(bytes12Bits, mload(add(add(buffer, memoryWord), startIndex)))
}
return tempAddress;
}
function toUint8(bytes memory buffer, uint256 startIndex) internal pure returns (uint8) {
require(buffer.length > startIndex, "toUint8_outOfBounds");
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
uint256 startOffset = startIndex + uint8Size;
uint8 tempUint;
assembly ("memory-safe") {
tempUint := mload(add(buffer, startOffset))
}
return tempUint;
}
function toUint16(bytes memory buffer, uint256 startIndex) internal pure returns (uint16) {
uint256 endIndex = startIndex + uint16Size;
require(buffer.length >= endIndex, "toUint16_outOfBounds");
uint16 tempUint;
assembly ("memory-safe") {
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
tempUint := mload(add(buffer, endIndex))
}
return tempUint;
}
function toUint32(bytes memory buffer, uint256 startIndex) internal pure returns (uint32) {
uint256 endIndex = startIndex + uint32Size;
require(buffer.length >= endIndex, "toUint32_outOfBounds");
uint32 tempUint;
assembly ("memory-safe") {
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
tempUint := mload(add(buffer, endIndex))
}
return tempUint;
}
function toUint64(bytes memory buffer, uint256 startIndex) internal pure returns (uint64) {
uint256 endIndex = startIndex + uint64Size;
require(buffer.length >= endIndex, "toUint64_outOfBounds");
uint64 tempUint;
assembly ("memory-safe") {
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
tempUint := mload(add(buffer, endIndex))
}
return tempUint;
}
function toUint128(bytes memory buffer, uint256 startIndex) internal pure returns (uint128) {
uint256 endIndex = startIndex + uint128Size;
require(buffer.length >= endIndex, "toUint128_outOfBounds");
uint128 tempUint;
assembly ("memory-safe") {
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
tempUint := mload(add(buffer, endIndex))
}
return tempUint;
}
function toUint256(bytes memory buffer, uint256 startIndex) internal pure returns (uint256) {
uint256 endIndex = startIndex + uint256Size;
require(buffer.length >= endIndex, "toUint256_outOfBounds");
uint256 tempUint;
assembly ("memory-safe") {
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
tempUint := mload(add(buffer, endIndex))
}
return tempUint;
}
function toBytes32(bytes memory buffer, uint256 startIndex) internal pure returns (bytes32) {
uint256 endIndex = startIndex + uint256Size;
require(buffer.length >= endIndex, "toBytes32_outOfBounds");
bytes32 tempBytes32;
assembly ("memory-safe") {
// Note that `endIndex == startOffset` for a given buffer due to the 32 bytes at the start that store the length.
tempBytes32 := mload(add(buffer, endIndex))
}
return tempBytes32;
}
}