Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

all: implement state history v2 #30107

Merged
merged 3 commits into from
Jan 17, 2025

Conversation

rjl493456442
Copy link
Member

@rjl493456442 rjl493456442 commented Jul 2, 2024

This pull request delivers the new version of the state history, where the raw storage key is used instead of the hash.

Before the cancun fork, it's supported by protocol to destruct a specific account and therefore, all the storage slot owned by it should be wiped in the same transition.

Technically, storage wiping should be performed through storage iteration, and only the storage key hash will be available for traversal if the state snapshot is not available. Therefore, the storage key hash is chosen as the identifier in the old version state history.

Fortunately, account self-destruction has been deprecated by the protocol since the Cancun fork, and there are no empty accounts eligible for deletion under EIP-158. Therefore, we can conclude that no storage wiping should occur after the Cancun fork. In this case, it makes no sense to keep using hash.

Besides, another big reason for making this change is the current format state history is unusable if verkle is activated. Verkle tree has a different key derivation scheme (merkle uses keccak256), the preimage of key hash must be provided in order to make verkle rollback functional. This pull request is a prerequisite for landing verkle.

Additionally, the raw storage key is more human-friendly for those who want to manually check the history, even though Solidity already performs some hashing to derive the storage location.


This pull request doesn't bump the database version, as I believe the database should still be compatible if users degrade from the new geth version to old one, the only side effect is the persistent new version state history will be unusable.

@rjl493456442 rjl493456442 force-pushed the state-history-v2 branch 7 times, most recently from c014ee1 to 7def7a7 Compare July 9, 2024 02:50
@rjl493456442 rjl493456442 force-pushed the state-history-v2 branch 2 times, most recently from 7036366 to 22d2359 Compare July 17, 2024 02:42
@holiman
Copy link
Contributor

holiman commented Aug 26, 2024

Fortunately, account self-destruction has been deprecated by the protocol since the Cancun fork, and there are no empty accounts eligible for deletion under EIP-158. Therefore, we can conclude that no storage wiping should occur after the Cancun fork. In this case, it makes no sense to keep using hash.

How does this PR work with chugging through older chain-segments, which potentially contain selfdestructs with storage-traversal needed?

@rjl493456442
Copy link
Member Author

How does this PR work with chugging through older chain-segments, which potentially contain selfdestructs with storage-traversal needed?

The state history of old chain segments (from the cancun fork) will be stored with initial version, the state history of post-cancun fork will be stored in new version.

@@ -1148,6 +1148,9 @@ func (s *StateDB) handleDestruction() (map[common.Hash]*accountDelete, []*trieno
if prev.Root == types.EmptyRootHash {
continue
}
if noStorageWiping {
return nil, nil, fmt.Errorf("unexpected storage wiping, %x", addr)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The case here is that the caller is saying that we're operating in "no selfdestruct-mode", but still tells us to destruct something (which has storage), is that correct?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I feel we might shot ourselves in the foot.

At least what I can imagine is that: for EIP7610, there are 28 accounts which are eligible for contract deployment but has no empty storage.
We plan to remove the leftover storage of them during the verkle transition, or a preceding fork. In this case, we will have storage deletion unfortunately.

func (s *StateDB) Commit(block uint64, deleteEmptyObjects bool) (common.Hash, error) {
ret, err := s.commitAndFlush(block, deleteEmptyObjects)
//
// noStorageWiping is a flag indicating whether storage wiping is permitted.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I find it a bit unintuitive with reverse bools, "no XXX". Couldn't you flip it to wipeStorage bool?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can't we set that flag when we construct the state? We have acces to the ruleset then (block number etfc), don't we? If so a lot of the diffs in this PR goes away (?)

@zsfelfoldi
Copy link
Contributor

zsfelfoldi commented Jan 12, 2025

Note that during the rebase to current master the journal database format has been slightly changed. Previously the state set encoding/decoding happened in loadDiffLayer and diffLayer.journal and the new flag rawStorageKey was encoded between the block id and the node set. Now there is a stateSet and a wrapping StateSetWithOrigin struct that does the encoding itself and this is where rawStorageKey belongs too so I figured it is more logical to encode the flag as a part of this structure. So now it is encoded after the node set, at the beginning of stateSet.

@zsfelfoldi zsfelfoldi merged commit a7f9523 into ethereum:master Jan 17, 2025
3 checks passed
@zsfelfoldi zsfelfoldi added this to the 1.14.13 milestone Jan 17, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants