This EIP proposes net gas metering changes for SSTORE
opcode, enabling
new usages for contract storage, and reducing excessive gas costs
where it doesn't match how most implementation works.
This acts as an alternative for EIP-1087, where it tries to be friendlier to implementations that use different optimization strategies for storage change caches.
This EIP proposes a way for gas metering on SSTORE (as an alternative for EIP-1087 and EIP-1153), using information that is more universally available to most implementations, and require as little change in implementation structures as possible.
Usages that benefits from this EIP's gas reduction scheme includes:
Definitions of terms are as below:
Replace SSTORE
opcode gas cost calculation (including refunds) with
the following logic:
Refund counter works as before -- it is limited to half of the gas consumed. On a transaction level, refund counter will never go below zero. However, there are some important notes depending on the implementation details:
The new gas cost scheme for SSTORE
is divided into three different
types:
We can see that the above three types cover all possible variations of original value, current value, and new value.
No-op is a trivial operation. Below we only consider cases for Fresh and Dirty.
All initial (not-No-op) SSTORE
on a particular storage slot starts
with Fresh. After that, it will become Dirty if the value has
been changed. When going from Fresh to Dirty, we charge the
gas cost the same as current scheme. A Dirty storage slot can be
reset back to Fresh via a SSTORE
opcode. This will trigger a
refund.
When a storage slot remains at Dirty, we charge 200 gas. In this
case, we would also need to keep track of R_SCLEAR
refunds -- if we
already issued the refund but it no longer applies (current value is
0), then removes this refund from the refund counter. If we didn't
issue the refund but it applies now (new value is 0), then adds this
refund to the refund counter. It is not possible where a refund is not
issued but we remove the refund in the above case, because all storage
slot starts with Fresh state.
Below is a graph (by @Arachnid) showing possible state transition of gas costs. We ignore No-op state because that is trivial:
Below is table version of the above diagram. Vertical shows the new value being set, and horizontal shows the state of original value and current value.
When original value is 0:
A (current=orig=0 ) |
B (current!=orig ) |
|
---|---|---|
~0 | B; 20k gas | B; 200 gas |
0 | A; 200 gas | A; 200 gas, 19.8k refund |
When original value is not 0:
X (current=orig!=0 ) |
Y (current!=orig ) |
Z (current=0 ) |
|
---|---|---|---|
orig |
X; 200 gas | X; 200 gas, 4.8k refund | X; 200 gas, -10.2k refund |
~orig, ~0 |
Y; 5k gas | Y; 200 gas | Y; 200 gas, -15k refund |
0 | Z; 5k gas, 15k refund | Z; 200 gas, 15k refund | Z; 200 gas |
This EIP mostly achieves what a transient storage tries to do (EIP-1087 and EIP-1153), but without the complexity of introducing the concept of "dirty maps", or an extra storage struct.
Regarding SSTORE
gas cost and refunds, see Appendix for proofs of
properties that this EIP satisfies.
Examine examples provided in EIP-1087's Motivation:
20000 + 200 - 19800 = 400
gas.20000 + 5 * 200 = 21000
gas.5000 * 3 + 200 - 4800 = 10400
gas.This EIP requires a hard fork to implement. No gas cost increase is anticipated, and many contracts will see gas reduction.
Below we provide 17 test cases. 15 of them covering consecutive two
SSTORE
operations are based on work by
@chfast. Two additional
case with three SSTORE
operations is used to test the case when a
slot is reset and then set again.
Code | Used Gas | Refund | Original | 1st | 2nd | 3rd |
---|---|---|---|---|---|---|
0x60006000556000600055 |
412 | 0 | 0 | 0 | 0 | |
0x60006000556001600055 |
20212 | 0 | 0 | 0 | 1 | |
0x60016000556000600055 |
20212 | 19800 | 0 | 1 | 0 | |
0x60016000556002600055 |
20212 | 0 | 0 | 1 | 2 | |
0x60016000556001600055 |
20212 | 0 | 0 | 1 | 1 | |
0x60006000556000600055 |
5212 | 15000 | 1 | 0 | 0 | |
0x60006000556001600055 |
5212 | 4800 | 1 | 0 | 1 | |
0x60006000556002600055 |
5212 | 0 | 1 | 0 | 2 | |
0x60026000556000600055 |
5212 | 15000 | 1 | 2 | 0 | |
0x60026000556003600055 |
5212 | 0 | 1 | 2 | 3 | |
0x60026000556001600055 |
5212 | 4800 | 1 | 2 | 1 | |
0x60026000556002600055 |
5212 | 0 | 1 | 2 | 2 | |
0x60016000556000600055 |
5212 | 15000 | 1 | 1 | 0 | |
0x60016000556002600055 |
5212 | 0 | 1 | 1 | 2 | |
0x60016000556001600055 |
412 | 0 | 1 | 1 | 1 | |
0x600160005560006000556001600055 |
40218 | 19800 | 0 | 1 | 0 | 1 |
0x600060005560016000556000600055 |
10218 | 19800 | 1 | 0 | 1 | 0 |
Because the storage slot's original value is defined as the value when a reversion happens on the current transaction, it's easy to see that call frames won't interfere SSTORE gas calculation. So although the below proof is discussed without call frames, it applies to all situations with call frames. We will discuss the case separately for original value being zero and not zero, and use induction to prove some properties of SSTORE gas cost.
Final value is the value of a particular storage slot at the end of
a transaction. Absolute gas used is the absolute value of gas used
minus refund. We use N
to represent the total number of SSTORE
operations on a storage slot. For states discussed below, refer to
State Transition in Explanation section.
When original value is 0, we want to prove that:
200 *
N
gases, because no disk write is needed.20000 + 200 * (N-1)
gas, because it requires writing this
slot to disk.We always start at state A. The first SSTORE can:
200 * N == 200 * 1
.20000 + 200 * (N-1) == 20000 + 200 * 0
.200 * (N-1)
. The current
gas cost is 200 + 200 * (N-1)
. It satisfy Case I.200 * (N-1)
. The current
gas cost is 20000 + 200 * (N-1)
. It satisfy Case II.20000 + 200 * (N-2)
. The
current gas cost is 200 + 20000 + 200 * (N-2)
. It satisfy
Case II.20000 + 200 * (N-2)
. The
current gas cost is 200 - 19800 + 20000 + 200 * (N-2)
. It satisfy
Case I.When original value is not 0, we want to prove that:
200 * N
gases, because no disk write is needed.5000 - 15000 + 200 * (N-1)
gas. Note that 15000
is the
refund in actual definition.5000 + 200 * (N-1)
gas.We always start at state X. The first SSTORE can:
200 * N == 200 * 1
.5000 + 200 * (N-1) == 5000 + 200 * 0
.5000 - 15000
where 15000
is the refund. We satisfy Case II because 5000 - 15000 + 200 *
(N-1) == 5000 - 15000 + 200 * 0
.200 * (N-1)
. The current gas
cost is 200 + 200 * (N-1)
. It satisfy Case I.200 * (N-1)
. The current gas
cost is 5000 + 200 * (N-1)
. It satisfy Case III.200 * (N-1)
. The current
absolute gas cost is 5000 - 15000 + 200 * (N-1)
. It satisfy Case
II.5000 + 200 * (N-2)
. The
absolute current gas cost is 200 - 4800 + 5000 + 200 * (N-2)
. It
satisfy Case I.5000 + 200 * (N-2)
. The
current gas cost is 200 + 5000 + 200 * (N-2)
. It satisfy Case
III.5000 + 200 * (N-2)
. The
current absolute gas cost is 200 - 15000 + 5000 + 200 * (N-2)
. It
satisfy Case II.5000 - 15000 + 200 *
(N-2)
. The current absolute gas cost is 200 + 10200 + 5000 -
15000 + 200 * (N-2)
. It satisfy Case I.5000 - 15000 + 200 *
(N-2)
. The current absolute gas cost is 200 + 15000 + 5000 -
15000 + 200 * (N-2)
. It satisfy Case III.5000 - 15000 + 200 *
(N-2)
. The current absolute gas cost is 200 + 5000 - 15000 + 200 *
(N-2)
. It satisfy Case II.Copyright and related rights waived via CC0.