EIP-1283 - Net gas metering for SSTORE without dirty maps

Created 2018-08-01
Status Final
Category Core
Type Standards Track
Authors

Abstract

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.

Motivation

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:

Specification

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:

Explanation

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.

State Transition

Below is a graph (by @Arachnid) showing possible state transition of gas costs. We ignore No-op state because that is trivial:

State Transition

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

Rationale

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:

Backwards Compatibility

This EIP requires a hard fork to implement. No gas cost increase is anticipated, and many contracts will see gas reduction.

Test Cases

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

Appendix: Proof

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.

Original Value Being Zero

When original value is 0, we want to prove that:

Base Case

We always start at state A. The first SSTORE can:

Inductive Step

Original Value Not Being Zero

When original value is not 0, we want to prove that:

Base Case

We always start at state X. The first SSTORE can:

Inductive Step

Copyright

Copyright and related rights waived via CC0.