2015-07-07 02:54:22 +02:00
// Copyright 2014 The go-ethereum Authors
2015-07-22 18:48:40 +02:00
// This file is part of the go-ethereum library.
2015-07-07 02:54:22 +02:00
//
2015-07-23 18:35:11 +02:00
// The go-ethereum library is free software: you can redistribute it and/or modify
2015-07-07 02:54:22 +02:00
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
2015-07-22 18:48:40 +02:00
// The go-ethereum library is distributed in the hope that it will be useful,
2015-07-07 02:54:22 +02:00
// but WITHOUT ANY WARRANTY; without even the implied warranty of
2015-07-22 18:48:40 +02:00
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2015-07-07 02:54:22 +02:00
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
2015-07-22 18:48:40 +02:00
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
2015-07-07 02:54:22 +02:00
2015-07-07 05:08:16 +02:00
// Package core implements the Ethereum consensus protocol.
2014-12-04 10:28:02 +01:00
package core
2014-02-14 23:56:09 +01:00
import (
2015-07-10 14:29:40 +02:00
"errors"
2014-09-24 11:39:17 +02:00
"fmt"
2015-03-18 13:36:48 +01:00
"io"
2014-07-30 00:31:15 +02:00
"math/big"
2022-09-19 10:04:16 +02:00
"runtime"
2019-11-21 15:35:22 +01:00
"sort"
2014-12-18 13:12:54 +01:00
"sync"
2015-06-12 16:45:53 +02:00
"sync/atomic"
2015-04-04 16:35:23 +02:00
"time"
2014-07-30 00:31:15 +02:00
2023-12-25 11:07:31 +08:00
mapset "github.com/deckarep/golang-set/v2"
2023-08-23 17:46:08 +08:00
exlru "github.com/hashicorp/golang-lru"
2022-07-05 11:14:21 +08:00
"golang.org/x/crypto/sha3"
2015-03-16 11:27:38 +01:00
"github.com/ethereum/go-ethereum/common"
2022-11-14 15:41:56 +01:00
"github.com/ethereum/go-ethereum/common/lru"
2017-01-10 22:55:54 +01:00
"github.com/ethereum/go-ethereum/common/mclock"
2018-09-03 23:33:21 +08:00
"github.com/ethereum/go-ethereum/common/prque"
2017-04-05 01:16:29 +03:00
"github.com/ethereum/go-ethereum/consensus"
2023-07-27 13:11:09 +02:00
"github.com/ethereum/go-ethereum/consensus/misc/eip4844"
2023-02-03 15:53:36 +08:00
"github.com/ethereum/go-ethereum/core/monitor"
2018-05-07 14:35:06 +03:00
"github.com/ethereum/go-ethereum/core/rawdb"
2015-03-23 22:59:19 +01:00
"github.com/ethereum/go-ethereum/core/state"
2019-08-06 13:40:28 +03:00
"github.com/ethereum/go-ethereum/core/state/snapshot"
2024-01-04 18:26:36 +08:00
"github.com/ethereum/go-ethereum/core/systemcontracts"
2015-03-16 23:48:18 +01:00
"github.com/ethereum/go-ethereum/core/types"
2015-10-19 16:08:17 +02:00
"github.com/ethereum/go-ethereum/core/vm"
2015-09-14 09:35:57 +02:00
"github.com/ethereum/go-ethereum/ethdb"
2014-12-03 14:05:19 +01:00
"github.com/ethereum/go-ethereum/event"
2021-10-07 15:47:50 +02:00
"github.com/ethereum/go-ethereum/internal/syncx"
2022-09-19 10:04:16 +02:00
"github.com/ethereum/go-ethereum/internal/version"
2017-02-22 14:10:07 +02:00
"github.com/ethereum/go-ethereum/log"
2015-06-29 16:11:01 +03:00
"github.com/ethereum/go-ethereum/metrics"
2016-10-20 13:36:29 +02:00
"github.com/ethereum/go-ethereum/params"
2022-07-05 11:14:21 +08:00
"github.com/ethereum/go-ethereum/rlp"
2015-10-05 19:37:56 +03:00
"github.com/ethereum/go-ethereum/trie"
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
"github.com/ethereum/go-ethereum/trie/triedb/hashdb"
"github.com/ethereum/go-ethereum/trie/triedb/pathdb"
2023-06-20 05:58:47 -04:00
"golang.org/x/exp/slices"
2014-02-14 23:56:09 +01:00
)
2015-03-03 15:44:41 +07:00
var (
2023-12-25 11:07:31 +08:00
badBlockRecords = mapset . NewSet [ common . Hash ] ( )
badBlockRecordslimit = 1000
badBlockGauge = metrics . NewRegisteredGauge ( "chain/insert/badBlock" , nil )
2019-06-10 14:21:02 +03:00
headBlockGauge = metrics . NewRegisteredGauge ( "chain/head/block" , nil )
headHeaderGauge = metrics . NewRegisteredGauge ( "chain/head/header" , nil )
headFastBlockGauge = metrics . NewRegisteredGauge ( "chain/head/receipt" , nil )
2023-04-10 18:36:45 +08:00
justifiedBlockGauge = metrics . NewRegisteredGauge ( "chain/head/justified" , nil )
finalizedBlockGauge = metrics . NewRegisteredGauge ( "chain/head/finalized" , nil )
2019-03-25 10:01:18 +02:00
accountReadTimer = metrics . NewRegisteredTimer ( "chain/account/reads" , nil )
accountHashTimer = metrics . NewRegisteredTimer ( "chain/account/hashes" , nil )
accountUpdateTimer = metrics . NewRegisteredTimer ( "chain/account/updates" , nil )
accountCommitTimer = metrics . NewRegisteredTimer ( "chain/account/commits" , nil )
storageReadTimer = metrics . NewRegisteredTimer ( "chain/storage/reads" , nil )
storageHashTimer = metrics . NewRegisteredTimer ( "chain/storage/hashes" , nil )
storageUpdateTimer = metrics . NewRegisteredTimer ( "chain/storage/updates" , nil )
storageCommitTimer = metrics . NewRegisteredTimer ( "chain/storage/commits" , nil )
2019-11-26 09:48:29 +02:00
snapshotAccountReadTimer = metrics . NewRegisteredTimer ( "chain/snapshot/account/reads" , nil )
snapshotStorageReadTimer = metrics . NewRegisteredTimer ( "chain/snapshot/storage/reads" , nil )
2019-08-06 13:40:28 +03:00
snapshotCommitTimer = metrics . NewRegisteredTimer ( "chain/snapshot/commits" , nil )
cmd, core, eth, les, light: track deleted nodes (#25757)
* cmd, core, eth, les, light: track deleted nodes
* trie: add docs
* trie: address comments
* cmd, core, eth, les, light, trie: trie id
* trie: add tests
* trie, core: updates
* trie: fix imports
* trie: add utility print-method for nodeset
* trie: import err
* trie: fix go vet warnings
Co-authored-by: Martin Holst Swende <martin@swende.se>
2022-09-27 16:01:02 +08:00
triedbCommitTimer = metrics . NewRegisteredTimer ( "chain/triedb/commits" , nil )
2018-11-28 09:29:05 +01:00
blockInsertTimer = metrics . NewRegisteredTimer ( "chain/inserts" , nil )
blockValidationTimer = metrics . NewRegisteredTimer ( "chain/validation" , nil )
blockExecutionTimer = metrics . NewRegisteredTimer ( "chain/execution" , nil )
blockWriteTimer = metrics . NewRegisteredTimer ( "chain/write" , nil )
2020-08-20 09:49:35 +02:00
2022-10-24 16:13:55 +03:00
blockReorgMeter = metrics . NewRegisteredMeter ( "chain/reorg/executes" , nil )
blockReorgAddMeter = metrics . NewRegisteredMeter ( "chain/reorg/add" , nil )
blockReorgDropMeter = metrics . NewRegisteredMeter ( "chain/reorg/drop" , nil )
2015-07-10 14:29:40 +02:00
2022-07-05 11:14:21 +08:00
errStateRootVerificationFailed = errors . New ( "state root verification failed" )
2023-08-23 17:46:08 +08:00
errInsertionInterrupted = errors . New ( "insertion is interrupted" )
2022-07-05 11:14:21 +08:00
errChainStopped = errors . New ( "blockchain is stopped" )
2023-08-23 17:46:08 +08:00
errInvalidOldChain = errors . New ( "invalid old chain" )
errInvalidNewChain = errors . New ( "invalid new chain" )
2015-03-03 15:44:41 +07:00
)
2014-06-23 12:54:10 +01:00
2015-04-20 20:37:40 +02:00
const (
2023-09-07 16:39:29 +08:00
bodyCacheLimit = 256
blockCacheLimit = 256
diffLayerCacheLimit = 1024
receiptsCacheLimit = 10000
txLookupCacheLimit = 1024
maxBadBlockLimit = 16
maxFutureBlocks = 256
maxTimeFutureBlocks = 30
TriesInMemory = 128
maxBeyondBlocks = 2048
prefetchTxNumber = 100
2022-07-05 11:14:21 +08:00
diffLayerFreezerRecheckInterval = 3 * time . Second
2023-09-07 16:39:29 +08:00
maxDiffForkDist = 11 // Maximum allowed backward distance from the chain head
2022-07-05 11:14:21 +08:00
rewindBadBlockInterval = 1 * time . Second
2017-05-25 17:21:20 +03:00
// BlockChainVersion ensures that an incompatible database forces a resync from scratch.
2019-02-21 21:14:35 +08:00
//
2019-03-27 09:11:24 -07:00
// Changelog:
//
// - Version 4
// The following incompatible database changes were added:
// * the `BlockNumber`, `TxHash`, `TxIndex`, `BlockHash` and `Index` fields of log are deleted
// * the `Bloom` field of receipt is deleted
// * the `BlockIndex` and `TxIndex` fields of txlookup are deleted
// - Version 5
// The following incompatible database changes were added:
// * the `TxHash`, `GasCost`, and `ContractAddress` fields are no longer stored for a receipt
// * the `TxHash`, `GasCost`, and `ContractAddress` fields are computed by looking up the
// receipts' corresponding block
2019-04-25 07:24:56 -07:00
// - Version 6
// The following incompatible database changes were added:
// * Transaction lookup information stores the corresponding block number instead of block hash
core, cmd, vendor: fixes and database inspection tool (#15)
* core, eth: some fixes for freezer
* vendor, core/rawdb, cmd/geth: add db inspector
* core, cmd/utils: check ancient store path forceily
* cmd/geth, common, core/rawdb: a few fixes
* cmd/geth: support windows file rename and fix rename error
* core: support ancient plugin
* core, cmd: streaming file copy
* cmd, consensus, core, tests: keep genesis in leveldb
* core: write txlookup during ancient init
* core: bump database version
2019-05-14 22:07:44 +08:00
// - Version 7
// The following incompatible database changes were added:
// * Use freezer as the ancient database to maintain all ancient data
2020-08-21 20:10:40 +08:00
// - Version 8
// The following incompatible database changes were added:
// * New scheme for contract code in order to separate the codes and trie nodes
BlockChainVersion uint64 = 8
2015-04-20 20:37:40 +02:00
)
2015-01-02 12:07:54 +01:00
2022-10-28 16:23:49 +08:00
// CacheConfig contains the configuration values for the trie database
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// and state snapshot these are resident in a blockchain.
2018-02-05 18:40:32 +02:00
type CacheConfig struct {
2019-04-01 11:52:11 +03:00
TrieCleanLimit int // Memory allowance (MB) to use for caching trie nodes in memory
TrieCleanNoPrefetch bool // Whether to disable heuristic state prefetching for followup blocks
TrieDirtyLimit int // Memory limit (MB) at which to start flushing dirty trie nodes to disk
TrieDirtyDisabled bool // Whether to disable trie write caching and GC altogether (archive node)
TrieTimeLimit time . Duration // Time limit after which to flush the current in-memory trie to disk
2019-11-26 09:48:29 +02:00
SnapshotLimit int // Memory allowance (MB) to use for caching snapshot entries in memory
2020-11-18 17:51:33 +08:00
Preimages bool // Whether to store preimage of trie key to the disk
2022-07-05 11:14:21 +08:00
TriesInMemory uint64 // How many tries keeps in memory
NoTries bool // Insecure settings. Do not have any tries in databases if enabled.
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
StateHistory uint64 // Number of blocks from head whose state histories are reserved.
StateScheme string // Scheme used to store ethereum states and merkle tree nodes on top
2023-11-06 14:11:17 +08:00
PathSyncFlush bool // Whether sync flush the trienodebuffer of pathdb to disk.
2020-03-03 09:10:23 +02:00
2022-09-24 02:20:36 +08:00
SnapshotNoBuild bool // Whether the background generation is allowed
SnapshotWait bool // Wait for snapshot construction on startup. TODO(karalabe): This is a dirty hack for testing, nuke it
2018-02-05 18:40:32 +02:00
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// triedbConfig derives the configures for trie database.
func ( c * CacheConfig ) triedbConfig ( ) * trie . Config {
2023-09-24 12:21:00 +08:00
config := & trie . Config {
Cache : c . TrieCleanLimit ,
Preimages : c . Preimages ,
NoTries : c . NoTries ,
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if c . StateScheme == rawdb . HashScheme {
config . HashDB = & hashdb . Config {
CleanCacheSize : c . TrieCleanLimit * 1024 * 1024 ,
}
}
if c . StateScheme == rawdb . PathScheme {
config . PathDB = & pathdb . Config {
2023-11-06 14:11:17 +08:00
SyncFlush : c . PathSyncFlush ,
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
StateHistory : c . StateHistory ,
CleanCacheSize : c . TrieCleanLimit * 1024 * 1024 ,
DirtyCacheSize : c . TrieDirtyLimit * 1024 * 1024 ,
}
}
return config
}
2020-08-20 13:01:24 +03:00
// defaultCacheConfig are the default caching values if none are specified by the
// user (also used during testing).
var defaultCacheConfig = & CacheConfig {
TrieCleanLimit : 256 ,
TrieDirtyLimit : 256 ,
TrieTimeLimit : 5 * time . Minute ,
SnapshotLimit : 256 ,
2022-07-05 11:14:21 +08:00
TriesInMemory : 128 ,
2020-08-20 13:01:24 +03:00
SnapshotWait : true ,
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
StateScheme : rawdb . HashScheme ,
}
// DefaultCacheConfigWithScheme returns a deep copied default cache config with
// a provided trie node scheme.
func DefaultCacheConfigWithScheme ( scheme string ) * CacheConfig {
config := * defaultCacheConfig
config . StateScheme = scheme
return & config
2020-08-20 13:01:24 +03:00
}
2022-07-05 11:14:21 +08:00
type BlockChainOption func ( * BlockChain ) ( * BlockChain , error )
2015-10-19 16:08:17 +02:00
// BlockChain represents the canonical chain given a database with a genesis
// block. The Blockchain manages chain imports, reverts, chain reorganisations.
//
// Importing blocks in to the block chain happens according to the set of rules
// defined by the two stage Validator. Processing of blocks is done using the
// Processor which processes the included transaction. The validation of the state
// is done in the second part of the Validator. Failing results in aborting of
// the import.
//
// The BlockChain also helps in returning blocks from **any** chain included
// in the database as well as blocks that represents the canonical chain. It's
// important to note that GetBlock can return any block and does not need to be
// included in the canonical one where as GetBlockByNumber always represents the
// canonical chain.
2015-08-31 17:09:50 +02:00
type BlockChain struct {
2018-02-05 18:40:32 +02:00
chainConfig * params . ChainConfig // Chain & network configuration
cacheConfig * CacheConfig // Cache configuration for pruning
2023-02-09 13:03:54 +02:00
db ethdb . Database // Low level persistent database to store final content in
snaps * snapshot . Tree // Snapshot tree for fast trie leaf access
triegc * prque . Prque [ int64 , common . Hash ] // Priority queue mapping block numbers to tries to gc
gcproc time . Duration // Accumulates canonical block processing for trie dumping
2023-08-23 17:46:08 +08:00
commitLock sync . Mutex // CommitLock is used to protect above field from being modified concurrently
2023-02-09 13:03:54 +02:00
lastWrite uint64 // Last block when the state was flushed
2023-03-30 18:53:32 +08:00
flushInterval atomic . Int64 // Time interval (processing time) after which to flush a state
2023-02-09 13:03:54 +02:00
triedb * trie . Database // The database handler for maintaining trie nodes.
stateCache state . Database // State database to reuse between imports (contains state cache)
2016-03-01 23:32:43 +01:00
2020-05-11 17:58:43 +02:00
// txLookupLimit is the maximum number of blocks from head whose tx indices
// are reserved:
// * 0: means no limit and regenerate any missing indexes
// * N: means N block limit [HEAD-N+1, HEAD] and delete extra indexes
// * nil: disable tx reindexer/deleter, but still index new blocks
txLookupLimit uint64
2022-07-05 11:14:21 +08:00
triesInMemory uint64
2023-04-10 18:36:45 +08:00
hc * HeaderChain
rmLogsFeed event . Feed
chainFeed event . Feed
chainSideFeed event . Feed
chainHeadFeed event . Feed
chainBlockFeed event . Feed
logsFeed event . Feed
blockProcFeed event . Feed
finalizedHeaderFeed event . Feed
scope event . SubscriptionScope
genesisBlock * types . Block
2015-12-16 04:26:23 +01:00
2021-10-07 15:47:50 +02:00
// This mutex synchronizes chain write operations.
// Readers don't need to take it, they can just read the database.
chainmu * syncx . ClosableMutex
2015-04-28 17:48:46 +02:00
2023-09-07 16:39:29 +08:00
highestVerifiedHeader atomic . Pointer [ types . Header ]
2023-08-23 17:46:08 +08:00
currentBlock atomic . Pointer [ types . Header ] // Current head of the chain
currentSnapBlock atomic . Pointer [ types . Header ] // Current head of snap-sync
2014-02-14 23:56:09 +01:00
2022-11-14 15:41:56 +01:00
bodyCache * lru . Cache [ common . Hash , * types . Body ]
bodyRLPCache * lru . Cache [ common . Hash , rlp . RawValue ]
receiptsCache * lru . Cache [ common . Hash , [ ] * types . Receipt ]
blockCache * lru . Cache [ common . Hash , * types . Block ]
txLookupCache * lru . Cache [ common . Hash , * rawdb . LegacyTxLookupEntry ]
2014-02-14 23:56:09 +01:00
2022-11-14 15:41:56 +01:00
// future blocks are blocks added for later processing
futureBlocks * lru . Cache [ common . Hash , * types . Block ]
2023-08-23 17:46:08 +08:00
// Cache for the blocks that failed to pass MPT root verification
badBlockCache * lru . Cache [ common . Hash , time . Time ]
2022-07-05 11:14:21 +08:00
// trusted diff layers
2023-08-23 17:46:08 +08:00
diffLayerCache * exlru . Cache // Cache for the diffLayers
diffLayerChanCache * exlru . Cache // Cache for the difflayer channel
diffQueue * prque . Prque [ int64 , * types . DiffLayer ] // A Priority queue to store recent diff layer
2022-07-05 11:14:21 +08:00
diffQueueBuffer chan * types . DiffLayer
diffLayerFreezerBlockLimit uint64
2021-10-07 15:47:50 +02:00
wg sync . WaitGroup //
quit chan struct { } // shutdown signal, closed in Stop.
2023-03-30 18:53:32 +08:00
stopping atomic . Bool // false if chain is running, true when stopped
procInterrupt atomic . Bool // interrupt signaler for block processing
2015-05-17 01:42:30 +02:00
2019-03-25 12:41:50 +02:00
engine consensus . Engine
2021-01-08 15:01:49 +02:00
prefetcher Prefetcher
2022-07-05 11:14:21 +08:00
validator Validator // Block and state validator interface
2021-01-08 15:01:49 +02:00
processor Processor // Block transaction processor interface
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
forker * ForkChoice
2019-03-25 12:41:50 +02:00
vmConfig vm . Config
2022-07-05 11:14:21 +08:00
pipeCommit bool
2023-02-03 15:53:36 +08:00
// monitor
doubleSignMonitor * monitor . DoubleSignMonitor
2014-12-18 13:12:54 +01:00
}
2014-02-14 23:56:09 +01:00
2015-10-19 16:08:17 +02:00
// NewBlockChain returns a fully initialised block chain using information
2022-07-05 11:14:21 +08:00
// available in the database. It initialises the default Ethereum Validator and
// Processor.
2023-08-23 17:46:08 +08:00
func NewBlockChain ( db ethdb . Database , cacheConfig * CacheConfig , genesis * Genesis , overrides * ChainOverrides , engine consensus . Engine ,
2022-07-05 11:14:21 +08:00
vmConfig vm . Config , shouldPreserve func ( block * types . Header ) bool , txLookupLimit * uint64 ,
options ... BlockChainOption ) ( * BlockChain , error ) {
2018-02-05 18:40:32 +02:00
if cacheConfig == nil {
2020-08-20 13:01:24 +03:00
cacheConfig = defaultCacheConfig
2018-02-05 18:40:32 +02:00
}
2022-07-05 11:14:21 +08:00
if cacheConfig . TriesInMemory != 128 {
log . Warn ( "TriesInMemory isn't the default value(128), you need specify exact same TriesInMemory when prune data" ,
"triesInMemory" , cacheConfig . TriesInMemory )
}
2023-08-23 17:46:08 +08:00
diffLayerCache , _ := exlru . New ( diffLayerCacheLimit )
diffLayerChanCache , _ := exlru . New ( diffLayerCacheLimit )
2022-11-28 21:31:28 +08:00
// Open trie database with provided config
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
triedb := trie . NewDatabase ( db , cacheConfig . triedbConfig ( ) )
2022-08-31 00:22:28 +08:00
// Setup the genesis block, commit the provided genesis specification
// to database if the genesis block is not present yet, or load the
// stored one from database.
2022-11-28 21:31:28 +08:00
chainConfig , genesisHash , genesisErr := SetupGenesisBlockWithOverride ( db , triedb , genesis , overrides )
2022-08-31 00:22:28 +08:00
if _ , ok := genesisErr . ( * params . ConfigCompatError ) ; genesisErr != nil && ! ok {
return nil , genesisErr
}
2024-01-04 18:26:36 +08:00
systemcontracts . GenesisHash = genesisHash
2023-09-21 12:02:59 +08:00
log . Info ( "Initialised chain configuration" , "config" , chainConfig )
// Description of chainConfig is empty now
/ *
log . Info ( "" )
log . Info ( strings . Repeat ( "-" , 153 ) )
for _ , line := range strings . Split ( chainConfig . Description ( ) , "\n" ) {
log . Info ( line )
}
log . Info ( strings . Repeat ( "-" , 153 ) )
log . Info ( "" )
* /
2015-08-31 20:21:02 +03:00
2015-08-31 17:09:50 +02:00
bc := & BlockChain {
2023-09-07 16:39:29 +08:00
chainConfig : chainConfig ,
cacheConfig : cacheConfig ,
db : db ,
triedb : triedb ,
triegc : prque . New [ int64 , common . Hash ] ( nil ) ,
quit : make ( chan struct { } ) ,
triesInMemory : cacheConfig . TriesInMemory ,
chainmu : syncx . NewClosableMutex ( ) ,
bodyCache : lru . NewCache [ common . Hash , * types . Body ] ( bodyCacheLimit ) ,
bodyRLPCache : lru . NewCache [ common . Hash , rlp . RawValue ] ( bodyCacheLimit ) ,
receiptsCache : lru . NewCache [ common . Hash , [ ] * types . Receipt ] ( receiptsCacheLimit ) ,
blockCache : lru . NewCache [ common . Hash , * types . Block ] ( blockCacheLimit ) ,
txLookupCache : lru . NewCache [ common . Hash , * rawdb . LegacyTxLookupEntry ] ( txLookupCacheLimit ) ,
futureBlocks : lru . NewCache [ common . Hash , * types . Block ] ( maxFutureBlocks ) ,
badBlockCache : lru . NewCache [ common . Hash , time . Time ] ( maxBadBlockLimit ) ,
diffLayerCache : diffLayerCache ,
diffLayerChanCache : diffLayerChanCache ,
engine : engine ,
vmConfig : vmConfig ,
diffQueue : prque . New [ int64 , * types . DiffLayer ] ( nil ) ,
diffQueueBuffer : make ( chan * types . DiffLayer ) ,
2022-07-05 11:14:21 +08:00
}
2023-03-30 18:53:32 +08:00
bc . flushInterval . Store ( int64 ( cacheConfig . TrieTimeLimit ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
bc . forker = NewForkChoice ( bc , shouldPreserve )
2023-09-07 16:39:29 +08:00
bc . stateCache = state . NewDatabaseWithNodeDB ( bc . db , bc . triedb )
2019-03-25 12:41:50 +02:00
bc . validator = NewBlockValidator ( chainConfig , bc , engine )
2023-09-07 16:39:29 +08:00
bc . prefetcher = NewStatePrefetcher ( chainConfig , bc , engine )
2019-03-25 12:41:50 +02:00
bc . processor = NewStateProcessor ( chainConfig , bc , engine )
2015-12-16 04:26:23 +01:00
var err error
2020-05-26 21:37:37 +02:00
bc . hc , err = NewHeaderChain ( db , chainConfig , engine , bc . insertStopped )
2015-10-13 12:04:25 +03:00
if err != nil {
return nil , err
}
2015-07-10 14:29:40 +02:00
bc . genesisBlock = bc . GetBlockByNumber ( 0 )
if bc . genesisBlock == nil {
2016-03-01 23:32:43 +01:00
return nil , ErrNoGenesis
2015-07-10 14:29:40 +02:00
}
2019-09-26 11:10:35 +02:00
2023-08-23 17:46:08 +08:00
bc . highestVerifiedHeader . Store ( nil )
2023-03-02 08:29:15 +02:00
bc . currentBlock . Store ( nil )
bc . currentSnapBlock . Store ( nil )
2020-05-11 17:58:43 +02:00
2022-09-29 15:50:24 +08:00
// If Geth is initialized with an external ancient store, re-initialize the
// missing chain indexes and chain flags. This procedure can survive crash
// and can be resumed in next restart since chain flags are updated in last step.
core, cmd, vendor: fixes and database inspection tool (#15)
* core, eth: some fixes for freezer
* vendor, core/rawdb, cmd/geth: add db inspector
* core, cmd/utils: check ancient store path forceily
* cmd/geth, common, core/rawdb: a few fixes
* cmd/geth: support windows file rename and fix rename error
* core: support ancient plugin
* core, cmd: streaming file copy
* cmd, consensus, core, tests: keep genesis in leveldb
* core: write txlookup during ancient init
* core: bump database version
2019-05-14 22:07:44 +08:00
if bc . empty ( ) {
2019-05-27 15:48:30 +03:00
rawdb . InitDatabaseFromFreezer ( bc . db )
core, cmd, vendor: fixes and database inspection tool (#15)
* core, eth: some fixes for freezer
* vendor, core/rawdb, cmd/geth: add db inspector
* core, cmd/utils: check ancient store path forceily
* cmd/geth, common, core/rawdb: a few fixes
* cmd/geth: support windows file rename and fix rename error
* core: support ancient plugin
* core, cmd: streaming file copy
* cmd, consensus, core, tests: keep genesis in leveldb
* core: write txlookup during ancient init
* core: bump database version
2019-05-14 22:07:44 +08:00
}
2022-09-29 15:50:24 +08:00
// Load blockchain states from disk
2015-09-21 15:36:29 +03:00
if err := bc . loadLastState ( ) ; err != nil {
2015-07-10 14:29:40 +02:00
return nil , err
2015-06-08 12:12:13 +02:00
}
2020-08-20 13:01:24 +03:00
// Make sure the state associated with the block is available
head := bc . CurrentBlock ( )
2023-09-28 10:07:21 +08:00
if ! bc . stateCache . NoTries ( ) && ! bc . HasState ( head . Root ) {
2020-10-30 03:01:58 +08:00
// Head state is missing, before the state recovery, find out the
// disk layer point of snapshot(if it's enabled). Make sure the
// rewound point is lower than disk layer.
var diskRoot common . Hash
if bc . cacheConfig . SnapshotLimit > 0 {
diskRoot = rawdb . ReadSnapshotRoot ( bc . db )
2023-11-06 14:11:17 +08:00
}
if bc . triedb . Scheme ( ) == rawdb . PathScheme {
recoverable , _ := bc . triedb . Recoverable ( diskRoot )
if ! bc . HasState ( diskRoot ) && ! recoverable {
diskRoot = bc . triedb . Head ( )
}
2020-10-30 03:01:58 +08:00
}
if diskRoot != ( common . Hash { } ) {
2023-11-06 14:11:17 +08:00
log . Warn ( "Head state missing, repairing" , "number" , head . Number , "hash" , head . Hash ( ) , "diskRoot" , diskRoot )
2020-10-30 03:01:58 +08:00
2023-03-02 08:29:15 +02:00
snapDisk , err := bc . setHeadBeyondRoot ( head . Number . Uint64 ( ) , 0 , diskRoot , true )
2020-10-30 03:01:58 +08:00
if err != nil {
return nil , err
}
// Chain rewound, persist old snapshot number to indicate recovery procedure
if snapDisk != 0 {
rawdb . WriteSnapshotRecoveryNumber ( bc . db , snapDisk )
}
} else {
2023-03-02 08:29:15 +02:00
log . Warn ( "Head state missing, repairing" , "number" , head . Number , "hash" , head . Hash ( ) )
if _ , err := bc . setHeadBeyondRoot ( head . Number . Uint64 ( ) , 0 , common . Hash { } , true ) ; err != nil {
2020-10-30 03:01:58 +08:00
return nil , err
}
2020-08-20 13:01:24 +03:00
}
}
// Ensure that a previous crash in SetHead doesn't leave extra ancients
2022-07-05 11:14:21 +08:00
if frozen , err := bc . db . ItemAmountInAncient ( ) ; err == nil && frozen > 0 {
frozen , err = bc . db . Ancients ( )
if err != nil {
return nil , err
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
var (
needRewind bool
low uint64
)
// The head full block may be rolled back to a very low height due to
// blockchain repair. If the head full block is even lower than the ancient
// chain, truncate the ancient store.
fullBlock := bc . CurrentBlock ( )
2023-03-02 08:29:15 +02:00
if fullBlock != nil && fullBlock . Hash ( ) != bc . genesisBlock . Hash ( ) && fullBlock . Number . Uint64 ( ) < frozen - 1 {
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
needRewind = true
2023-03-02 08:29:15 +02:00
low = fullBlock . Number . Uint64 ( )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2023-07-14 13:06:51 +01:00
// In snap sync, it may happen that ancient data has been written to the
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
// ancient store, but the LastFastBlock has not been updated, truncate the
// extra data here.
2023-03-02 08:29:15 +02:00
snapBlock := bc . CurrentSnapBlock ( )
if snapBlock != nil && snapBlock . Number . Uint64 ( ) < frozen - 1 {
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
needRewind = true
2023-03-02 08:29:15 +02:00
if snapBlock . Number . Uint64 ( ) < low || low == 0 {
low = snapBlock . Number . Uint64 ( )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
}
if needRewind {
2020-08-20 13:01:24 +03:00
log . Error ( "Truncating ancient chain" , "from" , bc . CurrentHeader ( ) . Number . Uint64 ( ) , "to" , low )
if err := bc . SetHead ( low ) ; err != nil {
return nil , err
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
}
}
2020-08-20 13:01:24 +03:00
// The first thing the node will do is reconstruct the verification data for
// the head block (ethash cache or clique voting snapshot). Might as well do
// it in advance.
2023-05-03 12:58:39 +03:00
bc . engine . VerifyHeader ( bc , bc . CurrentHeader ( ) )
2020-08-20 13:01:24 +03:00
2015-04-20 12:58:17 +02:00
// Check the current state of the block hashes and make sure that we do not have any of the bad blocks in our chain
2017-01-06 15:52:03 +01:00
for hash := range BadHashes {
2016-04-05 15:22:04 +02:00
if header := bc . GetHeaderByHash ( hash ) ; header != nil {
2016-11-28 10:37:42 +01:00
// get the canonical block corresponding to the offending header's number
headerByNumber := bc . GetHeaderByNumber ( header . Number . Uint64 ( ) )
// make sure the headerByNumber (if present) is in our current canonical chain
if headerByNumber != nil && headerByNumber . Hash ( ) == header . Hash ( ) {
2017-02-28 13:35:17 +02:00
log . Error ( "Found bad hash, rewinding chain" , "number" , header . Number , "hash" , header . ParentHash )
2020-08-20 13:01:24 +03:00
if err := bc . SetHead ( header . Number . Uint64 ( ) - 1 ) ; err != nil {
return nil , err
}
2017-02-28 13:35:17 +02:00
log . Error ( "Chain rewind was successful, resuming normal operation" )
2016-11-28 10:37:42 +01:00
}
2015-04-20 12:58:17 +02:00
}
}
2021-10-07 15:47:50 +02:00
2019-08-06 13:40:28 +03:00
// Load any existing snapshot, regenerating it if loading failed
2020-01-19 20:57:56 +01:00
if bc . cacheConfig . SnapshotLimit > 0 {
2020-10-30 03:01:58 +08:00
// If the chain was rewound past the snapshot persistent layer (causing
// a recovery block number to be persisted to disk), check if we're still
// in recovery mode and in that case, don't invalidate the snapshot on a
// head mismatch.
var recover bool
head := bc . CurrentBlock ( )
2023-03-02 08:29:15 +02:00
if layer := rawdb . ReadSnapshotRecoveryNumber ( bc . db ) ; layer != nil && * layer >= head . Number . Uint64 ( ) {
log . Warn ( "Enabling snapshot recovery" , "chainhead" , head . Number , "diskbase" , * layer )
2020-10-30 03:01:58 +08:00
recover = true
}
2022-09-24 02:20:36 +08:00
snapconfig := snapshot . Config {
CacheSize : bc . cacheConfig . SnapshotLimit ,
Recovery : recover ,
NoBuild : bc . cacheConfig . SnapshotNoBuild ,
AsyncBuild : ! bc . cacheConfig . SnapshotWait ,
}
2023-08-23 17:46:08 +08:00
bc . snaps , _ = snapshot . New ( snapconfig , bc . db , bc . triedb , head . Root , int ( bc . cacheConfig . TriesInMemory ) , bc . stateCache . NoTries ( ) )
2022-07-05 11:14:21 +08:00
}
// do options before start any routine
for _ , option := range options {
bc , err = option ( bc )
if err != nil {
return nil , err
}
2020-01-19 20:57:56 +01:00
}
2021-10-07 15:47:50 +02:00
// Start future block processor.
bc . wg . Add ( 1 )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
go bc . updateFutureBlocks ( )
2021-10-07 15:47:50 +02:00
2022-07-05 11:14:21 +08:00
// Need persist and prune diff layer
if bc . db . DiffStore ( ) != nil {
bc . wg . Add ( 1 )
go bc . trustedDiffLayerLoop ( )
}
if bc . pipeCommit {
// check current block and rewind invalid one
bc . wg . Add ( 1 )
go bc . rewindInvalidHeaderBlockLoop ( )
}
2023-02-03 15:53:36 +08:00
if bc . doubleSignMonitor != nil {
bc . wg . Add ( 1 )
go bc . startDoubleSignMonitor ( )
}
2022-08-31 00:22:28 +08:00
// Rewind the chain in case of an incompatible config upgrade.
if compat , ok := genesisErr . ( * params . ConfigCompatError ) ; ok {
log . Warn ( "Rewinding chain to upgrade configuration" , "err" , compat )
2022-12-15 09:40:33 +02:00
if compat . RewindToTime > 0 {
2022-12-16 14:06:22 +02:00
bc . SetHeadWithTimestamp ( compat . RewindToTime )
2022-12-15 09:40:33 +02:00
} else {
bc . SetHead ( compat . RewindToBlock )
}
2022-08-31 00:22:28 +08:00
rawdb . WriteChainConfig ( db , genesisHash , chainConfig )
}
2022-09-29 15:50:24 +08:00
// Start tx indexer/unindexer if required.
if txLookupLimit != nil {
bc . txLookupLimit = * txLookupLimit
bc . wg . Add ( 1 )
go bc . maintainTxIndex ( )
}
2015-06-08 12:12:13 +02:00
return bc , nil
2015-12-16 04:26:23 +01:00
}
core, internal, eth, miner, les: Take VM config from BlockChain (#17955)
Until this commit, when sending an RPC request that called `NewEVM`, a blank `vm.Config`
would be taken so as to set some options, based on the default configuration. If some extra
configuration switches were passed to the blockchain, those would be ignored.
This PR adds a function to get the config from the blockchain, and this is what is now used
for RPC calls.
Some subsequent changes need to be made, see https://github.com/ethereum/go-ethereum/pull/17955#pullrequestreview-182237244
for the details of the discussion.
2018-12-06 14:34:49 +01:00
// GetVMConfig returns the block chain VM config.
func ( bc * BlockChain ) GetVMConfig ( ) * vm . Config {
return & bc . vmConfig
}
2023-11-14 16:40:25 +08:00
func ( bc * BlockChain ) cacheReceipts ( hash common . Hash , receipts types . Receipts , block * types . Block ) {
2022-07-05 11:14:21 +08:00
// TODO, This is a hot fix for the block hash of logs is `0x0000000000000000000000000000000000000000000000000000000000000000` for system tx
2023-06-30 12:06:56 +08:00
// Please check details in https://github.com/bnb-chain/bsc/issues/443
2022-07-05 11:14:21 +08:00
// This is a temporary fix, the official fix should be a hard fork.
const possibleSystemReceipts = 3 // One slash tx, two reward distribute txs.
numOfReceipts := len ( receipts )
for i := numOfReceipts - 1 ; i >= 0 && i >= numOfReceipts - possibleSystemReceipts ; i -- {
for j := 0 ; j < len ( receipts [ i ] . Logs ) ; j ++ {
receipts [ i ] . Logs [ j ] . BlockHash = hash
}
}
2023-11-14 16:40:25 +08:00
txs := block . Transactions ( )
if len ( txs ) != len ( receipts ) {
log . Warn ( "transaction and receipt count mismatch" )
return
}
for i , receipt := range receipts {
receipt . EffectiveGasPrice = txs [ i ] . EffectiveGasTipValue ( block . BaseFee ( ) ) // basefee is supposed to be nil or zero
}
2021-03-25 17:00:24 +08:00
bc . receiptsCache . Add ( hash , receipts )
}
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) cacheDiffLayer ( diffLayer * types . DiffLayer , diffLayerCh chan struct { } ) {
// The difflayer in the system is stored by the map structure,
// so it will be out of order.
// It must be sorted first and then cached,
// otherwise the DiffHash calculated by different nodes will be inconsistent
sort . SliceStable ( diffLayer . Codes , func ( i , j int ) bool {
return diffLayer . Codes [ i ] . Hash . Hex ( ) < diffLayer . Codes [ j ] . Hash . Hex ( )
} )
sort . SliceStable ( diffLayer . Destructs , func ( i , j int ) bool {
return diffLayer . Destructs [ i ] . Hex ( ) < ( diffLayer . Destructs [ j ] . Hex ( ) )
} )
sort . SliceStable ( diffLayer . Accounts , func ( i , j int ) bool {
return diffLayer . Accounts [ i ] . Account . Hex ( ) < diffLayer . Accounts [ j ] . Account . Hex ( )
} )
sort . SliceStable ( diffLayer . Storages , func ( i , j int ) bool {
return diffLayer . Storages [ i ] . Account . Hex ( ) < diffLayer . Storages [ j ] . Account . Hex ( )
} )
for index := range diffLayer . Storages {
// Sort keys and vals by key.
sort . Sort ( & diffLayer . Storages [ index ] )
}
if bc . diffLayerCache . Len ( ) >= diffLayerCacheLimit {
bc . diffLayerCache . RemoveOldest ( )
}
bc . diffLayerCache . Add ( diffLayer . BlockHash , diffLayer )
close ( diffLayerCh )
if bc . db . DiffStore ( ) != nil {
// push to priority queue before persisting
bc . diffQueueBuffer <- diffLayer
}
}
func ( bc * BlockChain ) cacheBlock ( hash common . Hash , block * types . Block ) {
bc . blockCache . Add ( hash , block )
}
core, cmd, vendor: fixes and database inspection tool (#15)
* core, eth: some fixes for freezer
* vendor, core/rawdb, cmd/geth: add db inspector
* core, cmd/utils: check ancient store path forceily
* cmd/geth, common, core/rawdb: a few fixes
* cmd/geth: support windows file rename and fix rename error
* core: support ancient plugin
* core, cmd: streaming file copy
* cmd, consensus, core, tests: keep genesis in leveldb
* core: write txlookup during ancient init
* core: bump database version
2019-05-14 22:07:44 +08:00
// empty returns an indicator whether the blockchain is empty.
// Note, it's a special case that we connect a non-empty ancient
// database with an empty node, so that we can plugin the ancient
// into node seamlessly.
func ( bc * BlockChain ) empty ( ) bool {
genesis := bc . genesisBlock . Hash ( )
for _ , hash := range [ ] common . Hash { rawdb . ReadHeadBlockHash ( bc . db ) , rawdb . ReadHeadHeaderHash ( bc . db ) , rawdb . ReadHeadFastBlockHash ( bc . db ) } {
if hash != genesis {
return false
}
}
return true
}
2023-04-10 18:36:45 +08:00
// GetJustifiedNumber returns the highest justified blockNumber on the branch including and before `header`.
func ( bc * BlockChain ) GetJustifiedNumber ( header * types . Header ) uint64 {
if p , ok := bc . engine . ( consensus . PoSA ) ; ok {
2024-01-03 15:58:06 +08:00
justifiedBlockNumber , _ , err := p . GetJustifiedNumberAndHash ( bc , [ ] * types . Header { header } )
2023-04-10 18:36:45 +08:00
if err == nil {
return justifiedBlockNumber
}
}
// return 0 when err!=nil
// so the input `header` will at a disadvantage during reorg
return 0
}
// getFinalizedNumber returns the highest finalized number before the specific block.
func ( bc * BlockChain ) getFinalizedNumber ( header * types . Header ) uint64 {
if p , ok := bc . engine . ( consensus . PoSA ) ; ok {
if finalizedHeader := p . GetFinalizedHeader ( bc , header ) ; finalizedHeader != nil {
return finalizedHeader . Number . Uint64 ( )
}
}
return 0
}
2015-09-21 15:36:29 +03:00
// loadLastState loads the last known chain state from the database. This method
// assumes that the chain manager mutex is held.
2017-05-11 09:55:48 +08:00
func ( bc * BlockChain ) loadLastState ( ) error {
2015-09-21 15:36:29 +03:00
// Restore the last known head block
2018-05-07 14:35:06 +03:00
head := rawdb . ReadHeadBlockHash ( bc . db )
2015-09-21 15:36:29 +03:00
if head == ( common . Hash { } ) {
// Corrupt or empty database, init from scratch
2017-03-22 02:37:24 +02:00
log . Warn ( "Empty database, resetting chain" )
2017-05-11 09:55:48 +08:00
return bc . Reset ( )
2017-03-22 02:37:24 +02:00
}
// Make sure the entire head block is available
2023-03-02 08:29:15 +02:00
headBlock := bc . GetBlockByHash ( head )
if headBlock == nil {
2017-03-22 02:37:24 +02:00
// Corrupt or empty database, init from scratch
log . Warn ( "Head block missing, resetting chain" , "hash" , head )
2017-05-11 09:55:48 +08:00
return bc . Reset ( )
2017-03-22 02:37:24 +02:00
}
2022-07-05 11:14:21 +08:00
2017-03-22 02:37:24 +02:00
// Everything seems to be fine, set as the head block
2023-03-02 08:29:15 +02:00
bc . currentBlock . Store ( headBlock . Header ( ) )
headBlockGauge . Update ( int64 ( headBlock . NumberU64 ( ) ) )
2023-08-23 17:46:08 +08:00
justifiedBlockGauge . Update ( int64 ( bc . GetJustifiedNumber ( headBlock . Header ( ) ) ) )
finalizedBlockGauge . Update ( int64 ( bc . getFinalizedNumber ( headBlock . Header ( ) ) ) )
2017-03-22 02:37:24 +02:00
2015-09-21 15:36:29 +03:00
// Restore the last known head header
2023-03-02 08:29:15 +02:00
headHeader := headBlock . Header ( )
2018-05-07 14:35:06 +03:00
if head := rawdb . ReadHeadHeaderHash ( bc . db ) ; head != ( common . Hash { } ) {
2017-05-11 09:55:48 +08:00
if header := bc . GetHeaderByHash ( head ) ; header != nil {
2023-03-02 08:29:15 +02:00
headHeader = header
2015-09-21 15:36:29 +03:00
}
}
2023-03-02 08:29:15 +02:00
bc . hc . SetCurrentHeader ( headHeader )
2017-03-22 02:37:24 +02:00
2023-07-14 13:06:51 +01:00
// Restore the last known head snap block
2023-03-02 08:29:15 +02:00
bc . currentSnapBlock . Store ( headBlock . Header ( ) )
headFastBlockGauge . Update ( int64 ( headBlock . NumberU64 ( ) ) )
2019-06-10 14:21:02 +03:00
2018-05-07 14:35:06 +03:00
if head := rawdb . ReadHeadFastBlockHash ( bc . db ) ; head != ( common . Hash { } ) {
2017-05-11 09:55:48 +08:00
if block := bc . GetBlockByHash ( head ) ; block != nil {
2023-03-02 08:29:15 +02:00
bc . currentSnapBlock . Store ( block . Header ( ) )
2019-06-10 14:21:02 +03:00
headFastBlockGauge . Update ( int64 ( block . NumberU64 ( ) ) )
2015-09-30 19:23:31 +03:00
}
}
2022-05-18 16:30:42 +02:00
2016-09-22 21:04:58 +02:00
// Issue a status log for the user
2023-03-02 08:29:15 +02:00
var (
2023-08-23 17:46:08 +08:00
currentSnapBlock = bc . CurrentSnapBlock ( )
2018-02-26 10:53:10 +01:00
2023-03-02 08:29:15 +02:00
headerTd = bc . GetTd ( headHeader . Hash ( ) , headHeader . Number . Uint64 ( ) )
blockTd = bc . GetTd ( headBlock . Hash ( ) , headBlock . NumberU64 ( ) )
)
if headHeader . Hash ( ) != headBlock . Hash ( ) {
2023-11-06 14:11:17 +08:00
log . Info ( "Loaded most recent local header" , "number" , headHeader . Number , "hash" , headHeader . Hash ( ) , "hash" , headHeader . Root , "td" , headerTd , "age" , common . PrettyAge ( time . Unix ( int64 ( headHeader . Time ) , 0 ) ) )
2023-03-02 08:29:15 +02:00
}
2023-11-06 14:11:17 +08:00
log . Info ( "Loaded most recent local block" , "number" , headBlock . Number ( ) , "hash" , headBlock . Hash ( ) , "root" , headBlock . Root ( ) , "td" , blockTd , "age" , common . PrettyAge ( time . Unix ( int64 ( headBlock . Time ( ) ) , 0 ) ) )
2023-03-02 08:29:15 +02:00
if headBlock . Hash ( ) != currentSnapBlock . Hash ( ) {
2023-07-14 13:06:51 +01:00
snapTd := bc . GetTd ( currentSnapBlock . Hash ( ) , currentSnapBlock . Number . Uint64 ( ) )
2023-11-06 14:11:17 +08:00
log . Info ( "Loaded most recent local snap block" , "number" , currentSnapBlock . Number , "hash" , currentSnapBlock . Hash ( ) , "root" , currentSnapBlock . Root , "td" , snapTd , "age" , common . PrettyAge ( time . Unix ( int64 ( currentSnapBlock . Time ) , 0 ) ) )
2023-03-02 08:29:15 +02:00
}
2023-08-23 17:46:08 +08:00
if posa , ok := bc . engine . ( consensus . PoSA ) ; ok {
if currentFinalizedHeader := posa . GetFinalizedHeader ( bc , headHeader ) ; currentFinalizedHeader != nil {
if currentFinalizedBlock := bc . GetBlockByHash ( currentFinalizedHeader . Hash ( ) ) ; currentFinalizedBlock != nil {
finalTd := bc . GetTd ( currentFinalizedBlock . Hash ( ) , currentFinalizedBlock . NumberU64 ( ) )
2023-11-06 14:11:17 +08:00
log . Info ( "Loaded most recent local finalized block" , "number" , currentFinalizedBlock . Number ( ) , "hash" , currentFinalizedBlock . Hash ( ) , "root" , currentFinalizedBlock . Root ( ) , "td" , finalTd , "age" , common . PrettyAge ( time . Unix ( int64 ( currentFinalizedBlock . Time ( ) ) , 0 ) ) )
2023-08-23 17:46:08 +08:00
}
}
2022-05-18 16:30:42 +02:00
}
2015-09-21 15:36:29 +03:00
2020-08-20 13:01:24 +03:00
if pivot := rawdb . ReadLastPivotNumber ( bc . db ) ; pivot != nil {
2023-07-14 13:06:51 +01:00
log . Info ( "Loaded last snap-sync pivot marker" , "number" , * pivot )
2020-08-20 13:01:24 +03:00
}
2015-09-21 15:36:29 +03:00
return nil
}
2020-08-20 13:01:24 +03:00
// SetHead rewinds the local chain to a new head. Depending on whether the node
2023-07-14 13:06:51 +01:00
// was snap synced or full synced and in which state, the method will try to
2020-08-20 13:01:24 +03:00
// delete minimal data from disk whilst retaining chain consistency.
2017-03-22 02:37:24 +02:00
func ( bc * BlockChain ) SetHead ( head uint64 ) error {
2022-12-16 14:06:22 +02:00
if _ , err := bc . setHeadBeyondRoot ( head , 0 , common . Hash { } , false ) ; err != nil {
return err
2022-07-05 11:14:21 +08:00
}
2022-12-16 14:06:22 +02:00
// Send chain head event to update the transaction pool
2023-03-02 08:29:15 +02:00
header := bc . CurrentBlock ( )
block := bc . GetBlock ( header . Hash ( ) , header . Number . Uint64 ( ) )
if block == nil {
// This should never happen. In practice, previsouly currentBlock
// contained the entire block whereas now only a "marker", so there
// is an ever so slight chance for a race we should handle.
log . Error ( "Current block not found in database" , "block" , header . Number , "hash" , header . Hash ( ) )
return fmt . Errorf ( "current block missing: #%d [%x..]" , header . Number , header . Hash ( ) . Bytes ( ) [ : 4 ] )
}
bc . chainHeadFeed . Send ( ChainHeadEvent { Block : block } )
2022-12-16 14:06:22 +02:00
return nil
}
// SetHeadWithTimestamp rewinds the local chain to a new head that has at max
2023-07-14 13:06:51 +01:00
// the given timestamp. Depending on whether the node was snap synced or full
2022-12-16 14:06:22 +02:00
// synced and in which state, the method will try to delete minimal data from
// disk whilst retaining chain consistency.
func ( bc * BlockChain ) SetHeadWithTimestamp ( timestamp uint64 ) error {
if _ , err := bc . setHeadBeyondRoot ( 0 , timestamp , common . Hash { } , false ) ; err != nil {
2022-12-29 13:24:34 +01:00
return err
}
// Send chain head event to update the transaction pool
2023-03-02 08:29:15 +02:00
header := bc . CurrentBlock ( )
block := bc . GetBlock ( header . Hash ( ) , header . Number . Uint64 ( ) )
if block == nil {
// This should never happen. In practice, previsouly currentBlock
// contained the entire block whereas now only a "marker", so there
// is an ever so slight chance for a race we should handle.
log . Error ( "Current block not found in database" , "block" , header . Number , "hash" , header . Hash ( ) )
return fmt . Errorf ( "current block missing: #%d [%x..]" , header . Number , header . Hash ( ) . Bytes ( ) [ : 4 ] )
}
bc . chainHeadFeed . Send ( ChainHeadEvent { Block : block } )
2022-12-29 13:24:34 +01:00
return nil
2020-10-30 03:01:58 +08:00
}
2017-03-22 02:37:24 +02:00
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) tryRewindBadBlocks ( ) {
2021-10-07 15:47:50 +02:00
if ! bc . chainmu . TryLock ( ) {
2022-07-05 11:14:21 +08:00
return
2021-10-07 15:47:50 +02:00
}
2019-01-11 15:27:47 +02:00
defer bc . chainmu . Unlock ( )
2022-07-05 11:14:21 +08:00
block := bc . CurrentBlock ( )
snaps := bc . snaps
// Verified and Result is false
2023-08-23 17:46:08 +08:00
if snaps != nil && snaps . Snapshot ( block . Root ) != nil &&
snaps . Snapshot ( block . Root ) . Verified ( ) && ! snaps . Snapshot ( block . Root ) . WaitAndGetVerifyRes ( ) {
2022-07-05 11:14:21 +08:00
// Rewind by one block
2023-08-23 17:46:08 +08:00
log . Warn ( "current block verified failed, rewind to its parent" , "height" , block . Number . Uint64 ( ) , "hash" , block . Hash ( ) )
2022-07-05 11:14:21 +08:00
bc . futureBlocks . Remove ( block . Hash ( ) )
bc . badBlockCache . Add ( block . Hash ( ) , time . Now ( ) )
bc . diffLayerCache . Remove ( block . Hash ( ) )
2023-08-23 17:46:08 +08:00
bc . reportBlock ( bc . GetBlockByHash ( block . Hash ( ) ) , nil , errStateRootVerificationFailed )
bc . setHeadBeyondRoot ( block . Number . Uint64 ( ) - 1 , 0 , common . Hash { } , false )
2022-07-05 11:14:21 +08:00
}
}
2015-04-20 12:29:02 +02:00
2021-11-22 11:11:59 +02:00
// setHeadBeyondRoot rewinds the local chain to a new head with the extra condition
2020-10-30 03:01:58 +08:00
// that the rewind must pass the specified state root. This method is meant to be
2021-04-30 06:50:02 -04:00
// used when rewinding with snapshots enabled to ensure that we go back further than
2023-07-14 13:06:51 +01:00
// persistent disk layer. Depending on whether the node was snap synced or full, and
2020-10-30 03:01:58 +08:00
// in which state, the method will try to delete minimal data from disk whilst
// retaining chain consistency.
//
2022-12-16 14:06:22 +02:00
// The method also works in timestamp mode if `head == 0` but `time != 0`. In that
// case blocks are rolled back until the new head becomes older or equal to the
// requested time. If both `head` and `time` is 0, the chain is rewound to genesis.
//
2020-10-30 03:01:58 +08:00
// The method returns the block number where the requested root cap was found.
2022-12-16 14:06:22 +02:00
func ( bc * BlockChain ) setHeadBeyondRoot ( head uint64 , time uint64 , root common . Hash , repair bool ) ( uint64 , error ) {
2021-10-07 15:47:50 +02:00
if ! bc . chainmu . TryLock ( ) {
return 0 , errChainStopped
}
2019-01-11 15:27:47 +02:00
defer bc . chainmu . Unlock ( )
2015-04-20 12:29:02 +02:00
2020-10-30 03:01:58 +08:00
// Track the block number of the requested root hash
var rootNumber uint64 // (no root == always 0)
2020-08-20 13:01:24 +03:00
// Retrieve the last pivot block to short circuit rollbacks beyond it and the
// current freezer limit to start nuking id underflown
pivot := rawdb . ReadLastPivotNumber ( bc . db )
frozen , _ := bc . db . Ancients ( )
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// resetState resets the persistent state to genesis if it's not available.
resetState := func ( ) {
2023-11-06 14:11:17 +08:00
log . Info ( "Reset to block with genesis state" , "number" , bc . genesisBlock . NumberU64 ( ) , "hash" , bc . genesisBlock . Hash ( ) )
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// Short circuit if the genesis state is already present.
if bc . HasState ( bc . genesisBlock . Root ( ) ) {
return
}
// Reset the state database to empty for committing genesis state.
// Note, it should only happen in path-based scheme and Reset function
// is also only call-able in this mode.
if bc . triedb . Scheme ( ) == rawdb . PathScheme {
if err := bc . triedb . Reset ( types . EmptyRootHash ) ; err != nil {
log . Crit ( "Failed to clean state" , "err" , err ) // Shouldn't happen
}
}
// Write genesis state into database.
if err := CommitGenesisState ( bc . db , bc . triedb , bc . genesisBlock . Hash ( ) ) ; err != nil {
log . Crit ( "Failed to commit genesis state" , "err" , err )
}
}
2022-12-16 14:06:22 +02:00
updateFn := func ( db ethdb . KeyValueWriter , header * types . Header ) ( * types . Header , bool ) {
2021-11-22 11:11:59 +02:00
// Rewind the blockchain, ensuring we don't end up with a stateless head
2020-08-20 13:01:24 +03:00
// block. Note, depth equality is permitted to allow using SetHead as a
// chain reparation mechanism without deleting any data!
2023-03-02 08:29:15 +02:00
if currentBlock := bc . CurrentBlock ( ) ; currentBlock != nil && header . Number . Uint64 ( ) <= currentBlock . Number . Uint64 ( ) {
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
newHeadBlock := bc . GetBlock ( header . Hash ( ) , header . Number . Uint64 ( ) )
if newHeadBlock == nil {
2020-08-20 13:01:24 +03:00
log . Error ( "Gap in the chain, rewinding to genesis" , "number" , header . Number , "hash" , header . Hash ( ) )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
newHeadBlock = bc . genesisBlock
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
resetState ( )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
} else {
2020-10-30 03:01:58 +08:00
// Block exists, keep rewinding until we find one with state,
// keeping rewinding until we exceed the optional threshold
// root hash
beyondRoot := ( root == common . Hash { } ) // Flag whether we're beyond the requested root (no root, always true)
2023-11-06 14:11:17 +08:00
2020-08-20 13:01:24 +03:00
for {
2020-10-30 03:01:58 +08:00
// If a root threshold was requested but not yet crossed, check
if root != ( common . Hash { } ) && ! beyondRoot && newHeadBlock . Root ( ) == root {
beyondRoot , rootNumber = true , newHeadBlock . NumberU64 ( )
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if ! bc . HasState ( newHeadBlock . Root ( ) ) && ! bc . stateRecoverable ( newHeadBlock . Root ( ) ) {
2020-08-20 13:01:24 +03:00
log . Trace ( "Block state missing, rewinding further" , "number" , newHeadBlock . NumberU64 ( ) , "hash" , newHeadBlock . Hash ( ) )
if pivot == nil || newHeadBlock . NumberU64 ( ) > * pivot {
2021-01-25 21:29:45 +08:00
parent := bc . GetBlock ( newHeadBlock . ParentHash ( ) , newHeadBlock . NumberU64 ( ) - 1 )
if parent != nil {
newHeadBlock = parent
continue
}
log . Error ( "Missing block in the middle, aiming genesis" , "number" , newHeadBlock . NumberU64 ( ) - 1 , "hash" , newHeadBlock . ParentHash ( ) )
newHeadBlock = bc . genesisBlock
2020-08-20 13:01:24 +03:00
} else {
2023-11-06 14:11:17 +08:00
log . Info ( "Rewind passed pivot, aiming genesis" , "number" , newHeadBlock . NumberU64 ( ) , "hash" , newHeadBlock . Hash ( ) , "pivot" , * pivot )
2020-08-20 13:01:24 +03:00
newHeadBlock = bc . genesisBlock
}
}
2023-11-06 14:11:17 +08:00
if beyondRoot || newHeadBlock . NumberU64 ( ) == 0 {
2022-03-22 17:53:22 +08:00
if newHeadBlock . NumberU64 ( ) == 0 {
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
resetState ( )
} else if ! bc . HasState ( newHeadBlock . Root ( ) ) {
// Rewind to a block with recoverable state. If the state is
// missing, run the state recovery here.
if err := bc . triedb . Recover ( newHeadBlock . Root ( ) ) ; err != nil {
log . Crit ( "Failed to rollback state" , "err" , err ) // Shouldn't happen
2022-03-22 17:53:22 +08:00
}
}
2023-09-24 12:21:00 +08:00
log . Info ( "Rewound to block with state" , "number" , newHeadBlock . NumberU64 ( ) , "hash" , newHeadBlock . Hash ( ) )
2020-10-30 03:01:58 +08:00
break
}
log . Debug ( "Skipping block with threshold state" , "number" , newHeadBlock . NumberU64 ( ) , "hash" , newHeadBlock . Hash ( ) , "root" , newHeadBlock . Root ( ) )
newHeadBlock = bc . GetBlock ( newHeadBlock . ParentHash ( ) , newHeadBlock . NumberU64 ( ) - 1 ) // Keep rewinding
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
}
rawdb . WriteHeadBlockHash ( db , newHeadBlock . Hash ( ) )
2020-01-17 18:49:32 +08:00
// Degrade the chain markers if they are explicitly reverted.
// In theory we should update all in-memory markers in the
// last step, however the direction of SetHead is from high
2022-01-06 22:02:23 +08:00
// to low, so it's safe to update in-memory markers directly.
2023-03-02 08:29:15 +02:00
bc . currentBlock . Store ( newHeadBlock . Header ( ) )
2019-06-10 14:21:02 +03:00
headBlockGauge . Update ( int64 ( newHeadBlock . NumberU64 ( ) ) )
2023-04-10 18:36:45 +08:00
justifiedBlockGauge . Update ( int64 ( bc . GetJustifiedNumber ( newHeadBlock . Header ( ) ) ) )
finalizedBlockGauge . Update ( int64 ( bc . getFinalizedNumber ( newHeadBlock . Header ( ) ) ) )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2023-07-14 13:06:51 +01:00
// Rewind the snap block in a simpleton way to the target head
2023-03-02 08:29:15 +02:00
if currentSnapBlock := bc . CurrentSnapBlock ( ) ; currentSnapBlock != nil && header . Number . Uint64 ( ) < currentSnapBlock . Number . Uint64 ( ) {
newHeadSnapBlock := bc . GetBlock ( header . Hash ( ) , header . Number . Uint64 ( ) )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
// If either blocks reached nil, reset to the genesis state
2023-03-02 08:29:15 +02:00
if newHeadSnapBlock == nil {
newHeadSnapBlock = bc . genesisBlock
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2023-03-02 08:29:15 +02:00
rawdb . WriteHeadFastBlockHash ( db , newHeadSnapBlock . Hash ( ) )
2020-01-17 18:49:32 +08:00
// Degrade the chain markers if they are explicitly reverted.
// In theory we should update all in-memory markers in the
// last step, however the direction of SetHead is from high
// to low, so it's safe the update in-memory markers directly.
2023-03-02 08:29:15 +02:00
bc . currentSnapBlock . Store ( newHeadSnapBlock . Header ( ) )
headFastBlockGauge . Update ( int64 ( newHeadSnapBlock . NumberU64 ( ) ) )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2022-12-16 14:06:22 +02:00
var (
2023-03-02 08:29:15 +02:00
headHeader = bc . CurrentBlock ( )
2022-12-16 14:06:22 +02:00
headNumber = headHeader . Number . Uint64 ( )
)
2020-08-20 13:01:24 +03:00
// If setHead underflown the freezer threshold and the block processing
// intent afterwards is full block importing, delete the chain segment
// between the stateful-block and the sethead target.
var wipe bool
2022-12-16 14:06:22 +02:00
if headNumber + 1 < frozen {
wipe = pivot == nil || headNumber >= * pivot
2020-08-20 13:01:24 +03:00
}
2022-12-16 14:06:22 +02:00
return headHeader , wipe // Only force wipe if full synced
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2017-03-22 02:37:24 +02:00
// Rewind the header chain, deleting all block bodies until then
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
delFn := func ( db ethdb . KeyValueWriter , hash common . Hash , num uint64 ) {
// Ignore the error here since light client won't hit this path
frozen , _ := bc . db . Ancients ( )
if num + 1 <= frozen {
// Truncate all relative data(header, total difficulty, body, receipt
// and canonical hash) from ancient store.
2023-08-01 20:17:32 +08:00
if _ , err := bc . db . TruncateHead ( num ) ; err != nil {
core, cmd, vendor: fixes and database inspection tool (#15)
* core, eth: some fixes for freezer
* vendor, core/rawdb, cmd/geth: add db inspector
* core, cmd/utils: check ancient store path forceily
* cmd/geth, common, core/rawdb: a few fixes
* cmd/geth: support windows file rename and fix rename error
* core: support ancient plugin
* core, cmd: streaming file copy
* cmd, consensus, core, tests: keep genesis in leveldb
* core: write txlookup during ancient init
* core: bump database version
2019-05-14 22:07:44 +08:00
log . Crit ( "Failed to truncate ancient data" , "number" , num , "err" , err )
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
// Remove the hash <-> number mapping from the active store.
rawdb . DeleteHeaderNumber ( db , hash )
} else {
// Remove relative body and receipts from the active store.
// The header, total difficulty and canonical hash will be
// removed in the hc.SetHead function.
rawdb . DeleteBody ( db , hash , num )
rawdb . DeleteReceipts ( db , hash , num )
}
// Todo(rjl493456442) txlookup, bloombits, etc
2015-09-30 19:23:31 +03:00
}
2020-08-20 13:01:24 +03:00
// If SetHead was only called as a chain reparation method, try to skip
// touching the header chain altogether, unless the freezer is broken
2021-11-22 11:11:59 +02:00
if repair {
2023-03-02 08:29:15 +02:00
if target , force := updateFn ( bc . db , bc . CurrentBlock ( ) ) ; force {
2022-12-16 14:06:22 +02:00
bc . hc . SetHead ( target . Number . Uint64 ( ) , updateFn , delFn )
2020-08-20 13:01:24 +03:00
}
} else {
// Rewind the chain to the requested head and keep going backwards until a
2023-07-14 13:06:51 +01:00
// block with a state is found or snap sync pivot is passed
2022-12-16 14:06:22 +02:00
if time > 0 {
log . Warn ( "Rewinding blockchain to timestamp" , "target" , time )
bc . hc . SetHeadWithTimestamp ( time , updateFn , delFn )
} else {
log . Warn ( "Rewinding blockchain to block" , "target" , head )
bc . hc . SetHead ( head , updateFn , delFn )
}
2020-08-20 13:01:24 +03:00
}
2015-09-21 15:36:29 +03:00
// Clear out any stale content from the caches
2015-08-31 20:21:02 +03:00
bc . bodyCache . Purge ( )
bc . bodyRLPCache . Purge ( )
2018-09-29 13:53:31 -07:00
bc . receiptsCache . Purge ( )
2015-08-31 20:21:02 +03:00
bc . blockCache . Purge ( )
2019-08-21 17:29:34 +08:00
bc . txLookupCache . Purge ( )
2015-08-31 20:21:02 +03:00
bc . futureBlocks . Purge ( )
2015-04-20 12:29:02 +02:00
2020-10-30 03:01:58 +08:00
return rootNumber , bc . loadLastState ( )
2015-04-20 12:29:02 +02:00
}
2021-11-26 13:26:03 +02:00
// SnapSyncCommitHead sets the current head block to the one defined by the hash
2015-10-05 19:37:56 +03:00
// irrelevant what the chain contents were prior.
2021-11-26 13:26:03 +02:00
func ( bc * BlockChain ) SnapSyncCommitHead ( hash common . Hash ) error {
2015-10-13 12:04:25 +03:00
// Make sure that both the block as well at its state trie exists
2017-05-11 09:55:48 +08:00
block := bc . GetBlockByHash ( hash )
2015-10-05 19:37:56 +03:00
if block == nil {
2021-04-15 20:35:00 +03:00
return fmt . Errorf ( "non existent block [%x..]" , hash [ : 4 ] )
2015-10-05 19:37:56 +03:00
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// Reset the trie database with the fresh snap synced state.
cmd, core, eth, les, light: track deleted nodes (#25757)
* cmd, core, eth, les, light: track deleted nodes
* trie: add docs
* trie: address comments
* cmd, core, eth, les, light, trie: trie id
* trie: add tests
* trie, core: updates
* trie: fix imports
* trie: add utility print-method for nodeset
* trie: import err
* trie: fix go vet warnings
Co-authored-by: Martin Holst Swende <martin@swende.se>
2022-09-27 16:01:02 +08:00
root := block . Root ( )
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if bc . triedb . Scheme ( ) == rawdb . PathScheme {
if err := bc . triedb . Reset ( root ) ; err != nil {
return err
}
}
cmd, core, eth, les, light: track deleted nodes (#25757)
* cmd, core, eth, les, light: track deleted nodes
* trie: add docs
* trie: address comments
* cmd, core, eth, les, light, trie: trie id
* trie: add tests
* trie, core: updates
* trie: fix imports
* trie: add utility print-method for nodeset
* trie: import err
* trie: fix go vet warnings
Co-authored-by: Martin Holst Swende <martin@swende.se>
2022-09-27 16:01:02 +08:00
if ! bc . HasState ( root ) {
return fmt . Errorf ( "non existent state [%x..]" , root [ : 4 ] )
2015-10-05 19:37:56 +03:00
}
2021-10-07 15:47:50 +02:00
// If all checks out, manually set the head block.
if ! bc . chainmu . TryLock ( ) {
return errChainStopped
}
2023-03-02 08:29:15 +02:00
bc . currentBlock . Store ( block . Header ( ) )
2019-06-10 14:21:02 +03:00
headBlockGauge . Update ( int64 ( block . NumberU64 ( ) ) )
2023-04-10 18:36:45 +08:00
justifiedBlockGauge . Update ( int64 ( bc . GetJustifiedNumber ( block . Header ( ) ) ) )
finalizedBlockGauge . Update ( int64 ( bc . getFinalizedNumber ( block . Header ( ) ) ) )
2019-01-11 15:27:47 +02:00
bc . chainmu . Unlock ( )
2015-10-05 19:37:56 +03:00
2021-04-29 17:33:45 +03:00
// Destroy any existing state snapshot and regenerate it in the background,
// also resuming the normal maintenance of any previously paused snapshot.
2020-01-19 20:57:56 +01:00
if bc . snaps != nil {
cmd, core, eth, les, light: track deleted nodes (#25757)
* cmd, core, eth, les, light: track deleted nodes
* trie: add docs
* trie: address comments
* cmd, core, eth, les, light, trie: trie id
* trie: add tests
* trie, core: updates
* trie: fix imports
* trie: add utility print-method for nodeset
* trie: import err
* trie: fix go vet warnings
Co-authored-by: Martin Holst Swende <martin@swende.se>
2022-09-27 16:01:02 +08:00
bc . snaps . Rebuild ( root )
2020-01-19 20:57:56 +01:00
}
2017-02-28 13:35:17 +02:00
log . Info ( "Committed new head block" , "number" , block . Number ( ) , "hash" , hash )
2015-10-05 19:37:56 +03:00
return nil
}
2015-08-31 20:21:02 +03:00
// Reset purges the entire blockchain, restoring it to its genesis state.
2017-03-22 02:37:24 +02:00
func ( bc * BlockChain ) Reset ( ) error {
return bc . ResetWithGenesisBlock ( bc . genesisBlock )
2015-03-13 18:29:42 +01:00
}
2015-08-31 20:21:02 +03:00
// ResetWithGenesisBlock purges the entire blockchain, restoring it to the
// specified genesis state.
2017-03-22 02:37:24 +02:00
func ( bc * BlockChain ) ResetWithGenesisBlock ( genesis * types . Block ) error {
2015-09-30 19:23:31 +03:00
// Dump the entire block chain and purge the caches
2017-03-22 02:37:24 +02:00
if err := bc . SetHead ( 0 ) ; err != nil {
return err
}
2021-10-07 15:47:50 +02:00
if ! bc . chainmu . TryLock ( ) {
return errChainStopped
}
2019-01-11 15:27:47 +02:00
defer bc . chainmu . Unlock ( )
2015-03-03 18:41:51 +01:00
2015-10-13 12:04:25 +03:00
// Prepare the genesis block and reinitialise the chain
2020-01-17 18:49:32 +08:00
batch := bc . db . NewBatch ( )
rawdb . WriteTd ( batch , genesis . Hash ( ) , genesis . NumberU64 ( ) , genesis . Difficulty ( ) )
rawdb . WriteBlock ( batch , genesis )
if err := batch . Write ( ) ; err != nil {
log . Crit ( "Failed to write genesis block" , "err" , err )
2015-09-07 20:43:01 +03:00
}
2020-01-17 18:49:32 +08:00
bc . writeHeadBlock ( genesis )
2018-05-07 14:35:06 +03:00
2020-01-17 18:49:32 +08:00
// Last update all in-memory chain markers
2015-09-22 23:55:31 +02:00
bc . genesisBlock = genesis
2023-03-02 08:29:15 +02:00
bc . currentBlock . Store ( bc . genesisBlock . Header ( ) )
2019-06-10 14:21:02 +03:00
headBlockGauge . Update ( int64 ( bc . genesisBlock . NumberU64 ( ) ) )
2023-04-10 18:36:45 +08:00
justifiedBlockGauge . Update ( int64 ( bc . genesisBlock . NumberU64 ( ) ) )
finalizedBlockGauge . Update ( int64 ( bc . genesisBlock . NumberU64 ( ) ) )
2015-12-16 04:26:23 +01:00
bc . hc . SetGenesis ( bc . genesisBlock . Header ( ) )
bc . hc . SetCurrentHeader ( bc . genesisBlock . Header ( ) )
2023-03-02 08:29:15 +02:00
bc . currentSnapBlock . Store ( bc . genesisBlock . Header ( ) )
2019-06-10 14:21:02 +03:00
headFastBlockGauge . Update ( int64 ( bc . genesisBlock . NumberU64 ( ) ) )
2017-03-22 02:37:24 +02:00
return nil
2015-03-03 18:41:51 +01:00
}
2015-03-18 13:36:48 +01:00
// Export writes the active chain to the given writer.
2017-05-11 09:55:48 +08:00
func ( bc * BlockChain ) Export ( w io . Writer ) error {
2023-03-02 08:29:15 +02:00
return bc . ExportN ( w , uint64 ( 0 ) , bc . CurrentBlock ( ) . Number . Uint64 ( ) )
2015-06-05 23:01:54 -04:00
}
// ExportN writes a subset of the active chain to the given writer.
2017-05-11 09:55:48 +08:00
func ( bc * BlockChain ) ExportN ( w io . Writer , first uint64 , last uint64 ) error {
2015-06-05 23:01:54 -04:00
if first > last {
return fmt . Errorf ( "export failed: first (%d) is greater than last (%d)" , first , last )
}
2017-02-28 13:35:17 +02:00
log . Info ( "Exporting batch of blocks" , "count" , last - first + 1 )
2015-04-13 10:13:52 +02:00
2022-07-14 14:55:54 +02:00
var (
parentHash common . Hash
start = time . Now ( )
reported = time . Now ( )
)
2015-06-05 23:01:54 -04:00
for nr := first ; nr <= last ; nr ++ {
2017-05-11 09:55:48 +08:00
block := bc . GetBlockByNumber ( nr )
2015-04-20 16:02:50 +02:00
if block == nil {
return fmt . Errorf ( "export failed on #%d: not found" , nr )
}
2022-07-14 14:55:54 +02:00
if nr > first && block . ParentHash ( ) != parentHash {
2023-05-25 20:24:09 +08:00
return errors . New ( "export failed: chain reorg during export" )
2022-07-14 14:55:54 +02:00
}
parentHash = block . Hash ( )
2015-04-20 16:02:50 +02:00
if err := block . EncodeRLP ( w ) ; err != nil {
2015-03-18 13:36:48 +01:00
return err
}
2018-07-26 23:26:24 +12:00
if time . Since ( reported ) >= statsReportLimit {
log . Info ( "Exporting blocks" , "exported" , block . NumberU64 ( ) - first , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) )
reported = time . Now ( )
}
2014-12-17 12:57:35 +01:00
}
2015-03-18 13:36:48 +01:00
return nil
2014-12-17 12:57:35 +01:00
}
2020-01-17 18:49:32 +08:00
// writeHeadBlock injects a new head block into the current block chain. This method
2015-09-21 15:36:29 +03:00
// assumes that the block is indeed a true head. It will also reset the head
2023-07-14 13:06:51 +01:00
// header and the head snap sync block to this very same block if they are older
2015-10-28 17:40:24 +02:00
// or if they are on a different side chain.
2015-09-21 15:36:29 +03:00
//
// Note, this function assumes that the `mu` mutex is held!
2020-01-17 18:49:32 +08:00
func ( bc * BlockChain ) writeHeadBlock ( block * types . Block ) {
2015-09-07 20:43:01 +03:00
// Add the block to the canonical chain number scheme and mark as the head
2020-01-17 18:49:32 +08:00
batch := bc . db . NewBatch ( )
2021-11-26 13:26:03 +02:00
rawdb . WriteHeadHeaderHash ( batch , block . Hash ( ) )
rawdb . WriteHeadFastBlockHash ( batch , block . Hash ( ) )
2020-01-17 18:49:32 +08:00
rawdb . WriteCanonicalHash ( batch , block . Hash ( ) , block . NumberU64 ( ) )
2020-09-15 10:37:01 +02:00
rawdb . WriteTxLookupEntriesByBlock ( batch , block )
2020-01-17 18:49:32 +08:00
rawdb . WriteHeadBlockHash ( batch , block . Hash ( ) )
2015-10-28 17:40:24 +02:00
2020-01-17 18:49:32 +08:00
// Flush the whole batch into the disk, exit the node if failed
if err := batch . Write ( ) ; err != nil {
log . Crit ( "Failed to update chain indexes and markers" , "err" , err )
}
// Update all in-memory chain markers in the last step
2021-11-26 13:26:03 +02:00
bc . hc . SetCurrentHeader ( block . Header ( ) )
2018-02-05 18:40:32 +02:00
2023-03-02 08:29:15 +02:00
bc . currentSnapBlock . Store ( block . Header ( ) )
2021-11-26 13:26:03 +02:00
headFastBlockGauge . Update ( int64 ( block . NumberU64 ( ) ) )
2020-08-21 20:10:40 +08:00
2023-03-02 08:29:15 +02:00
bc . currentBlock . Store ( block . Header ( ) )
2020-01-17 18:49:32 +08:00
headBlockGauge . Update ( int64 ( block . NumberU64 ( ) ) )
2023-04-10 18:36:45 +08:00
justifiedBlockGauge . Update ( int64 ( bc . GetJustifiedNumber ( block . Header ( ) ) ) )
finalizedBlockGauge . Update ( int64 ( bc . getFinalizedNumber ( block . Header ( ) ) ) )
2020-08-21 20:10:40 +08:00
}
2023-04-17 08:02:31 -07:00
// stopWithoutSaving stops the blockchain service. If any imports are currently in progress
2022-10-06 13:39:20 +02:00
// it will abort them using the procInterrupt. This method stops all running
// goroutines, but does not do all the post-stop work of persisting data.
// OBS! It is generally recommended to use the Stop method!
// This method has been exposed to allow tests to stop the blockchain while simulating
// a crash.
func ( bc * BlockChain ) stopWithoutSaving ( ) {
2023-03-30 18:53:32 +08:00
if ! bc . stopping . CompareAndSwap ( false , true ) {
2015-08-01 12:32:28 +02:00
return
}
2021-10-07 15:47:50 +02:00
// Unsubscribe all subscriptions registered from blockchain.
2017-08-18 18:58:36 +08:00
bc . scope . Close ( )
2021-10-07 15:47:50 +02:00
// Signal shutdown to all goroutines.
2015-03-06 15:50:44 +01:00
close ( bc . quit )
2020-05-26 21:37:37 +02:00
bc . StopInsert ( )
2021-10-07 15:47:50 +02:00
// Now wait for all chain modifications to end and persistent goroutines to exit.
//
// Note: Close waits for the mutex to become available, i.e. any running chain
// modification will have exited when Close returns. Since we also called StopInsert,
// the mutex should become available quickly. It cannot be taken again after Close has
// returned.
bc . chainmu . Close ( )
2015-04-30 17:50:23 +02:00
bc . wg . Wait ( )
2022-10-06 13:39:20 +02:00
}
// Stop stops the blockchain service. If any imports are currently in progress
// it will abort them using the procInterrupt.
func ( bc * BlockChain ) Stop ( ) {
bc . stopWithoutSaving ( )
2018-02-05 18:40:32 +02:00
2019-08-06 13:40:28 +03:00
// Ensure that the entirety of the state snapshot is journalled to disk.
2020-01-19 20:57:56 +01:00
var snapBase common . Hash
if bc . snaps != nil {
var err error
2023-03-02 08:29:15 +02:00
if snapBase , err = bc . snaps . Journal ( bc . CurrentBlock ( ) . Root ) ; err != nil {
2021-04-20 13:27:46 +08:00
log . Error ( "Failed to journal state snapshot" , "err" , err )
2020-01-19 20:57:56 +01:00
}
2019-08-06 13:40:28 +03:00
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if bc . triedb . Scheme ( ) == rawdb . PathScheme {
// Ensure that the in-memory trie nodes are journaled to disk properly.
if err := bc . triedb . Journal ( bc . CurrentBlock ( ) . Root ) ; err != nil {
log . Info ( "Failed to journal in-memory trie nodes" , "err" , err )
}
} else {
// Ensure the state of a recent block is also stored to disk before exiting.
// We're writing three different states to catch different restart scenarios:
// - HEAD: So we don't need to reprocess any blocks in the general case
// - HEAD-1: So we don't do large reorgs if our HEAD becomes an uncle
// - HEAD-127: So we have a hard limit on the number of blocks reexecuted
if ! bc . cacheConfig . TrieDirtyDisabled {
triedb := bc . triedb
2023-10-16 11:48:59 +08:00
var once sync . Once
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
for _ , offset := range [ ] uint64 { 0 , 1 , TriesInMemory - 1 } {
if number := bc . CurrentBlock ( ) . Number . Uint64 ( ) ; number > offset {
recent := bc . GetBlockByNumber ( number - offset )
log . Info ( "Writing cached state to disk" , "block" , recent . Number ( ) , "hash" , recent . Hash ( ) , "root" , recent . Root ( ) )
if err := triedb . Commit ( recent . Root ( ) , true ) ; err != nil {
log . Error ( "Failed to commit recent state trie" , "err" , err )
2023-10-12 22:17:11 +08:00
} else {
rawdb . WriteSafePointBlockNumber ( bc . db , recent . NumberU64 ( ) )
once . Do ( func ( ) {
2023-10-16 11:48:59 +08:00
rawdb . WriteHeadBlockHash ( bc . db , recent . Hash ( ) )
2023-10-12 22:17:11 +08:00
} )
2023-10-16 11:48:59 +08:00
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
}
}
if snapBase != ( common . Hash { } ) {
log . Info ( "Writing snapshot state to disk" , "root" , snapBase )
if err := triedb . Commit ( snapBase , true ) ; err != nil {
log . Error ( "Failed to commit recent state trie" , "err" , err )
} else {
rawdb . WriteSafePointBlockNumber ( bc . db , bc . CurrentBlock ( ) . Number . Uint64 ( ) )
}
}
2018-02-23 14:02:33 +02:00
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if snapBase != ( common . Hash { } ) {
log . Info ( "Writing snapshot state to disk" , "root" , snapBase )
if err := bc . triedb . Commit ( snapBase , true ) ; err != nil {
2018-02-23 14:02:33 +02:00
log . Error ( "Failed to commit recent state trie" , "err" , err )
2022-07-05 11:14:21 +08:00
} else {
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
rawdb . WriteSafePointBlockNumber ( bc . db , bc . CurrentBlock ( ) . Number . Uint64 ( ) )
2018-02-23 14:02:33 +02:00
}
2018-02-05 18:40:32 +02:00
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
for ! bc . triegc . Empty ( ) {
2023-10-12 22:17:11 +08:00
triedb . Dereference ( bc . triegc . PopItem ( ) )
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
}
2023-11-06 14:11:17 +08:00
if _ , size , _ , _ := triedb . Size ( ) ; size != 0 {
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
log . Error ( "Dangling trie nodes after full cleanup" )
2019-11-26 09:48:29 +02:00
}
2018-02-05 18:40:32 +02:00
}
}
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// Close the trie database, release all the held resources as the last step.
if err := bc . triedb . Close ( ) ; err != nil {
log . Error ( "Failed to close trie database" , "err" , err )
2020-07-28 21:30:31 +08:00
}
2020-03-27 14:03:20 +01:00
log . Info ( "Blockchain stopped" )
2015-03-06 15:50:44 +01:00
}
2020-05-26 21:37:37 +02:00
// StopInsert interrupts all insertion methods, causing them to return
// errInsertionInterrupted as soon as possible. Insertion is permanently disabled after
// calling this method.
func ( bc * BlockChain ) StopInsert ( ) {
2023-03-30 18:53:32 +08:00
bc . procInterrupt . Store ( true )
2020-05-26 21:37:37 +02:00
}
// insertStopped returns true after StopInsert has been called.
func ( bc * BlockChain ) insertStopped ( ) bool {
2023-03-30 18:53:32 +08:00
return bc . procInterrupt . Load ( )
2020-05-26 21:37:37 +02:00
}
2017-05-11 09:55:48 +08:00
func ( bc * BlockChain ) procFutureBlocks ( ) {
blocks := make ( [ ] * types . Block , 0 , bc . futureBlocks . Len ( ) )
for _ , hash := range bc . futureBlocks . Keys ( ) {
if block , exist := bc . futureBlocks . Peek ( hash ) ; exist {
2022-11-14 15:41:56 +01:00
blocks = append ( blocks , block )
2016-03-08 15:55:27 +01:00
}
2015-06-29 22:42:13 +02:00
}
2015-05-29 18:55:42 +02:00
if len ( blocks ) > 0 {
2023-08-12 01:04:12 +03:00
slices . SortFunc ( blocks , func ( a , b * types . Block ) int {
return a . Number ( ) . Cmp ( b . Number ( ) )
2019-11-21 15:35:22 +01:00
} )
2016-12-13 16:14:33 +02:00
// Insert one by one as chain insertion needs contiguous ancestry between blocks
for i := range blocks {
2017-05-11 09:55:48 +08:00
bc . InsertChain ( blocks [ i : i + 1 ] )
2016-12-13 16:14:33 +02:00
}
2015-05-29 18:55:42 +02:00
}
2015-04-04 16:35:23 +02:00
}
2017-05-11 09:55:48 +08:00
// WriteStatus status of write
2015-12-16 04:26:23 +01:00
type WriteStatus byte
2015-06-29 12:12:30 +02:00
const (
2015-12-16 04:26:23 +01:00
NonStatTy WriteStatus = iota
2015-07-03 11:24:42 +02:00
CanonStatTy
SideStatTy
2015-06-29 12:12:30 +02:00
)
2015-09-30 19:23:31 +03:00
// InsertReceiptChain attempts to complete an already existing header chain with
// transaction and receipt data.
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
func ( bc * BlockChain ) InsertReceiptChain ( blockChain types . Blocks , receiptChain [ ] types . Receipts , ancientLimit uint64 ) ( int , error ) {
2019-07-03 16:19:15 +08:00
// We don't require the chainMu here since we want to maximize the
// concurrency of header insertion and receipt insertion.
2017-09-09 18:03:07 +02:00
bc . wg . Add ( 1 )
defer bc . wg . Done ( )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
var (
ancientBlocks , liveBlocks types . Blocks
ancientReceipts , liveReceipts [ ] types . Receipts
)
2016-12-13 16:14:33 +02:00
// Do a sanity check that the provided chain is actually ordered and linked
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
for i := 0 ; i < len ( blockChain ) ; i ++ {
if i != 0 {
if blockChain [ i ] . NumberU64 ( ) != blockChain [ i - 1 ] . NumberU64 ( ) + 1 || blockChain [ i ] . ParentHash ( ) != blockChain [ i - 1 ] . Hash ( ) {
log . Error ( "Non contiguous receipt insert" , "number" , blockChain [ i ] . Number ( ) , "hash" , blockChain [ i ] . Hash ( ) , "parent" , blockChain [ i ] . ParentHash ( ) ,
"prevnumber" , blockChain [ i - 1 ] . Number ( ) , "prevhash" , blockChain [ i - 1 ] . Hash ( ) )
2021-04-15 20:35:00 +03:00
return 0 , fmt . Errorf ( "non contiguous insert: item %d is #%d [%x..], item %d is #%d [%x..] (parent [%x..])" , i - 1 , blockChain [ i - 1 ] . NumberU64 ( ) ,
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
blockChain [ i - 1 ] . Hash ( ) . Bytes ( ) [ : 4 ] , i , blockChain [ i ] . NumberU64 ( ) , blockChain [ i ] . Hash ( ) . Bytes ( ) [ : 4 ] , blockChain [ i ] . ParentHash ( ) . Bytes ( ) [ : 4 ] )
}
}
if blockChain [ i ] . NumberU64 ( ) <= ancientLimit {
ancientBlocks , ancientReceipts = append ( ancientBlocks , blockChain [ i ] ) , append ( ancientReceipts , receiptChain [ i ] )
} else {
liveBlocks , liveReceipts = append ( liveBlocks , blockChain [ i ] ) , append ( liveReceipts , receiptChain [ i ] )
2016-12-13 16:14:33 +02:00
}
}
2015-09-30 19:23:31 +03:00
2017-09-09 18:03:07 +02:00
var (
stats = struct { processed , ignored int32 } { }
start = time . Now ( )
2021-09-07 12:31:17 +02:00
size = int64 ( 0 )
2017-09-09 18:03:07 +02:00
)
2021-09-07 12:31:17 +02:00
2023-07-14 13:06:51 +01:00
// updateHead updates the head snap sync block if the inserted blocks are better
2020-05-25 16:21:28 +08:00
// and returns an indicator whether the inserted blocks are canonical.
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
updateHead := func ( head * types . Block ) bool {
2021-10-07 15:47:50 +02:00
if ! bc . chainmu . TryLock ( ) {
return false
}
2021-09-07 12:31:17 +02:00
defer bc . chainmu . Unlock ( )
2019-07-03 16:19:15 +08:00
// Rewind may have occurred, skip in that case.
if bc . CurrentHeader ( ) . Number . Cmp ( head . Number ( ) ) >= 0 {
2023-08-23 17:46:08 +08:00
reorg , err := bc . forker . ReorgNeededWithFastFinality ( bc . CurrentSnapBlock ( ) , head . Header ( ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if err != nil {
log . Warn ( "Reorg failed" , "err" , err )
return false
} else if ! reorg {
return false
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
rawdb . WriteHeadFastBlockHash ( bc . db , head . Hash ( ) )
2023-03-02 08:29:15 +02:00
bc . currentSnapBlock . Store ( head . Header ( ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
headFastBlockGauge . Update ( int64 ( head . NumberU64 ( ) ) )
return true
2017-09-09 18:03:07 +02:00
}
2019-07-03 16:19:15 +08:00
return false
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
// writeAncient writes blockchain and corresponding receipt chain into ancient store.
//
// this function only accepts canonical chain data. All side chain will be reverted
// eventually.
writeAncient := func ( blockChain types . Blocks , receiptChain [ ] types . Receipts ) ( int , error ) {
2021-09-07 12:31:17 +02:00
first := blockChain [ 0 ]
last := blockChain [ len ( blockChain ) - 1 ]
// Ensure genesis is in ancients.
if first . NumberU64 ( ) == 1 {
if frozen , _ := bc . db . Ancients ( ) ; frozen == 0 {
b := bc . genesisBlock
td := bc . genesisBlock . Difficulty ( )
writeSize , err := rawdb . WriteAncientBlocks ( bc . db , [ ] * types . Block { b } , [ ] types . Receipts { nil } , td )
size += writeSize
if err != nil {
log . Error ( "Error writing genesis to ancients" , "err" , err )
return 0 , err
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2021-09-07 12:31:17 +02:00
log . Info ( "Wrote genesis to ancients" )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2021-09-07 12:31:17 +02:00
}
// Before writing the blocks to the ancients, we need to ensure that
// they correspond to the what the headerchain 'expects'.
// We only check the last block/header, since it's a contiguous chain.
if ! bc . HasHeader ( last . Hash ( ) , last . NumberU64 ( ) ) {
return 0 , fmt . Errorf ( "containing header #%d [%x..] unknown" , last . Number ( ) , last . Hash ( ) . Bytes ( ) [ : 4 ] )
}
// Write all chain data to ancients.
td := bc . GetTd ( first . Hash ( ) , first . NumberU64 ( ) )
writeSize , err := rawdb . WriteAncientBlocks ( bc . db , blockChain , receiptChain , td )
size += writeSize
if err != nil {
log . Error ( "Error importing chain data to ancients" , "err" , err )
return 0 , err
}
// Write tx indices if any condition is satisfied:
// * If user requires to reserve all tx indices(txlookuplimit=0)
// * If all ancient tx indices are required to be reserved(txlookuplimit is even higher than ancientlimit)
// * If block number is large enough to be regarded as a recent block
// It means blocks below the ancientLimit-txlookupLimit won't be indexed.
//
// But if the `TxIndexTail` is not nil, e.g. Geth is initialized with
// an external ancient database, during the setup, blockchain will start
// a background routine to re-indexed all indices in [ancients - txlookupLimit, ancients)
// range. In this case, all tx indices of newly imported blocks should be
// generated.
var batch = bc . db . NewBatch ( )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
for i , block := range blockChain {
2020-05-11 17:58:43 +02:00
if bc . txLookupLimit == 0 || ancientLimit <= bc . txLookupLimit || block . NumberU64 ( ) >= ancientLimit - bc . txLookupLimit {
2020-09-15 10:37:01 +02:00
rawdb . WriteTxLookupEntriesByBlock ( batch , block )
2020-05-11 17:58:43 +02:00
} else if rawdb . ReadTxIndexTail ( bc . db ) != nil {
2020-09-15 10:37:01 +02:00
rawdb . WriteTxLookupEntriesByBlock ( batch , block )
2020-05-11 17:58:43 +02:00
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
stats . processed ++
2021-09-07 12:31:17 +02:00
2022-01-05 15:00:03 +08:00
if batch . ValueSize ( ) > ethdb . IdealBatchSize || i == len ( blockChain ) - 1 {
size += int64 ( batch . ValueSize ( ) )
if err = batch . Write ( ) ; err != nil {
2023-03-02 08:29:15 +02:00
snapBlock := bc . CurrentSnapBlock ( ) . Number . Uint64 ( )
2023-08-01 20:17:32 +08:00
if _ , err := bc . db . TruncateHead ( snapBlock + 1 ) ; err != nil {
2022-01-05 15:00:03 +08:00
log . Error ( "Can't truncate ancient store after failed insert" , "err" , err )
}
return 0 , err
}
batch . Reset ( )
2021-09-07 12:31:17 +02:00
}
2017-09-09 18:03:07 +02:00
}
2015-10-07 12:14:30 +03:00
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
// Sync the ancient store explicitly to ensure all data has been flushed to disk.
if err := bc . db . Sync ( ) ; err != nil {
return 0 , err
}
2023-07-14 13:06:51 +01:00
// Update the current snap block because all block data is now present in DB.
2023-03-02 08:29:15 +02:00
previousSnapBlock := bc . CurrentSnapBlock ( ) . Number . Uint64 ( )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
if ! updateHead ( blockChain [ len ( blockChain ) - 1 ] ) {
2021-09-07 12:31:17 +02:00
// We end up here if the header chain has reorg'ed, and the blocks/receipts
// don't match the canonical chain.
2023-08-01 20:17:32 +08:00
if _ , err := bc . db . TruncateHead ( previousSnapBlock + 1 ) ; err != nil {
2021-09-07 12:31:17 +02:00
log . Error ( "Can't truncate ancient store after failed insert" , "err" , err )
2019-05-27 17:05:45 +08:00
}
2021-09-07 12:31:17 +02:00
return 0 , errSideChainReceipts
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
core, cmd, vendor: fixes and database inspection tool (#15)
* core, eth: some fixes for freezer
* vendor, core/rawdb, cmd/geth: add db inspector
* core, cmd/utils: check ancient store path forceily
* cmd/geth, common, core/rawdb: a few fixes
* cmd/geth: support windows file rename and fix rename error
* core: support ancient plugin
* core, cmd: streaming file copy
* cmd, consensus, core, tests: keep genesis in leveldb
* core: write txlookup during ancient init
* core: bump database version
2019-05-14 22:07:44 +08:00
2021-09-07 12:31:17 +02:00
// Delete block data from the main database.
batch . Reset ( )
canonHashes := make ( map [ common . Hash ] struct { } )
for _ , block := range blockChain {
canonHashes [ block . Hash ( ) ] = struct { } { }
if block . NumberU64 ( ) == 0 {
continue
2019-07-08 10:24:16 +02:00
}
2021-09-07 12:31:17 +02:00
rawdb . DeleteCanonicalHash ( batch , block . NumberU64 ( ) )
rawdb . DeleteBlockWithoutNumber ( batch , block . Hash ( ) , block . NumberU64 ( ) )
2019-07-08 10:24:16 +02:00
}
2021-09-07 12:31:17 +02:00
// Delete side chain hash-to-number mappings.
for _ , nh := range rawdb . ReadAllHashesInRange ( bc . db , first . NumberU64 ( ) , last . NumberU64 ( ) ) {
if _ , canon := canonHashes [ nh . Hash ] ; ! canon {
rawdb . DeleteHeader ( batch , nh . Hash , nh . Number )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
}
if err := batch . Write ( ) ; err != nil {
return 0 , err
}
return 0 , nil
}
2021-09-07 12:31:17 +02:00
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
// writeLive writes blockchain and corresponding receipt chain into active store.
writeLive := func ( blockChain types . Blocks , receiptChain [ ] types . Receipts ) ( int , error ) {
2020-08-22 18:12:04 +02:00
skipPresenceCheck := false
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
batch := bc . db . NewBatch ( )
for i , block := range blockChain {
// Short circuit insertion if shutting down or processing failed
2020-05-26 21:37:37 +02:00
if bc . insertStopped ( ) {
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
return 0 , errInsertionInterrupted
}
// Short circuit if the owner header is unknown
if ! bc . HasHeader ( block . Hash ( ) , block . NumberU64 ( ) ) {
2021-04-15 20:35:00 +03:00
return i , fmt . Errorf ( "containing header #%d [%x..] unknown" , block . Number ( ) , block . Hash ( ) . Bytes ( ) [ : 4 ] )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2020-08-22 18:12:04 +02:00
if ! skipPresenceCheck {
// Ignore if the entire data is already known
if bc . HasBlock ( block . Hash ( ) , block . NumberU64 ( ) ) {
stats . ignored ++
continue
} else {
// If block N is not present, neither are the later blocks.
// This should be true, but if we are mistaken, the shortcut
// here will only cause overwriting of some existing data
skipPresenceCheck = true
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
// Write all the data out into the database
rawdb . WriteBody ( batch , block . Hash ( ) , block . NumberU64 ( ) , block . Body ( ) )
rawdb . WriteReceipts ( batch , block . Hash ( ) , block . NumberU64 ( ) , receiptChain [ i ] )
2020-09-15 10:37:01 +02:00
rawdb . WriteTxLookupEntriesByBlock ( batch , block ) // Always write tx indices for live blocks, we assume they are needed
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
2020-01-17 18:49:32 +08:00
// Write everything belongs to the blocks into the database. So that
// we can ensure all components of body is completed(body, receipts,
// tx indexes)
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
if batch . ValueSize ( ) >= ethdb . IdealBatchSize {
if err := batch . Write ( ) ; err != nil {
return 0 , err
}
2021-09-07 12:31:17 +02:00
size += int64 ( batch . ValueSize ( ) )
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
batch . Reset ( )
}
2020-01-17 18:49:32 +08:00
stats . processed ++
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
}
2020-01-17 18:49:32 +08:00
// Write everything belongs to the blocks into the database. So that
// we can ensure all components of body is completed(body, receipts,
// tx indexes)
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
if batch . ValueSize ( ) > 0 {
2021-09-07 12:31:17 +02:00
size += int64 ( batch . ValueSize ( ) )
2017-09-09 18:03:07 +02:00
if err := batch . Write ( ) ; err != nil {
return 0 , err
2015-10-07 12:14:30 +03:00
}
2015-09-30 19:23:31 +03:00
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
updateHead ( blockChain [ len ( blockChain ) - 1 ] )
return 0 , nil
2015-10-07 12:14:30 +03:00
}
2021-09-07 12:31:17 +02:00
2020-05-11 17:58:43 +02:00
// Write downloaded chain data and corresponding receipt chain data
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
if len ( ancientBlocks ) > 0 {
if n , err := writeAncient ( ancientBlocks , ancientReceipts ) ; err != nil {
if err == errInsertionInterrupted {
return 0 , nil
}
return n , err
2015-09-30 19:23:31 +03:00
}
}
2020-05-11 17:58:43 +02:00
// Write the tx index tail (block number from where we index) before write any live blocks
if len ( liveBlocks ) > 0 && liveBlocks [ 0 ] . NumberU64 ( ) == ancientLimit + 1 {
// The tx index tail can only be one of the following two options:
// * 0: all ancient blocks have been indexed
// * ancient-limit: the indices of blocks before ancient-limit are ignored
if tail := rawdb . ReadTxIndexTail ( bc . db ) ; tail == nil {
if bc . txLookupLimit == 0 || ancientLimit <= bc . txLookupLimit {
rawdb . WriteTxIndexTail ( bc . db , 0 )
} else {
rawdb . WriteTxIndexTail ( bc . db , ancientLimit - bc . txLookupLimit )
}
}
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
if len ( liveBlocks ) > 0 {
if n , err := writeLive ( liveBlocks , liveReceipts ) ; err != nil {
if err == errInsertionInterrupted {
return 0 , nil
}
return n , err
2015-10-07 12:14:30 +03:00
}
}
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
head := blockChain [ len ( blockChain ) - 1 ]
2018-09-20 11:41:59 +03:00
context := [ ] interface { } {
"count" , stats . processed , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) ,
2019-04-02 22:28:48 +02:00
"number" , head . Number ( ) , "hash" , head . Hash ( ) , "age" , common . PrettyAge ( time . Unix ( int64 ( head . Time ( ) ) , 0 ) ) ,
all: integrate the freezer with fast sync
* all: freezer style syncing
core, eth, les, light: clean up freezer relative APIs
core, eth, les, trie, ethdb, light: clean a bit
core, eth, les, light: add unit tests
core, light: rewrite setHead function
core, eth: fix downloader unit tests
core: add receipt chain insertion test
core: use constant instead of hardcoding table name
core: fix rollback
core: fix setHead
core/rawdb: remove canonical block first and then iterate side chain
core/rawdb, ethdb: add hasAncient interface
eth/downloader: calculate ancient limit via cht first
core, eth, ethdb: lots of fixes
* eth/downloader: print ancient disable log only for fast sync
2019-04-25 22:59:48 +08:00
"size" , common . StorageSize ( size ) ,
2018-09-20 11:41:59 +03:00
}
if stats . ignored > 0 {
context = append ( context , [ ] interface { } { "ignored" , stats . ignored } ... )
}
2023-02-21 12:17:34 +02:00
log . Debug ( "Imported new block receipts" , context ... )
2018-09-20 11:41:59 +03:00
2015-09-30 19:23:31 +03:00
return 0 , nil
}
2019-05-08 19:30:36 +08:00
// writeBlockWithoutState writes only the block and its metadata to the database,
2018-02-05 18:40:32 +02:00
// but does not write any state. This is used to construct competing side forks
// up to the point where they exceed the canonical total difficulty.
2019-05-08 19:30:36 +08:00
func ( bc * BlockChain ) writeBlockWithoutState ( block * types . Block , td * big . Int ) ( err error ) {
2021-10-07 15:47:50 +02:00
if bc . insertStopped ( ) {
return errInsertionInterrupted
}
2018-02-05 18:40:32 +02:00
2020-01-17 18:49:32 +08:00
batch := bc . db . NewBatch ( )
rawdb . WriteTd ( batch , block . Hash ( ) , block . NumberU64 ( ) , td )
rawdb . WriteBlock ( batch , block )
if err := batch . Write ( ) ; err != nil {
log . Crit ( "Failed to write block into disk" , "err" , err )
2018-02-05 18:40:32 +02:00
}
return nil
}
2019-05-08 19:30:36 +08:00
// writeKnownBlock updates the head block flag with a known block
// and introduces chain reorg if necessary.
func ( bc * BlockChain ) writeKnownBlock ( block * types . Block ) error {
current := bc . CurrentBlock ( )
if block . ParentHash ( ) != current . Hash ( ) {
if err := bc . reorg ( current , block ) ; err != nil {
return err
}
}
2020-01-17 18:49:32 +08:00
bc . writeHeadBlock ( block )
2019-05-08 19:30:36 +08:00
return nil
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// writeBlockWithState writes block, metadata and corresponding state data to the
// database.
2022-08-04 16:03:20 +08:00
func ( bc * BlockChain ) writeBlockWithState ( block * types . Block , receipts [ ] * types . Receipt , state * state . StateDB ) error {
2015-09-07 20:43:01 +03:00
// Calculate the total difficulty of the block
2017-05-11 09:55:48 +08:00
ptd := bc . GetTd ( block . ParentHash ( ) , block . NumberU64 ( ) - 1 )
2015-09-07 20:43:01 +03:00
if ptd == nil {
2022-08-31 13:30:25 +08:00
state . StopPrefetcher ( )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
return consensus . ErrUnknownAncestor
2015-09-07 20:43:01 +03:00
}
2015-09-29 14:24:28 +03:00
// Make sure no inconsistent state is leaked during insertion
2016-07-08 16:59:19 +03:00
externTd := new ( big . Int ) . Add ( block . Difficulty ( ) , ptd )
2020-01-17 18:49:32 +08:00
// Irrelevant of the canonical status, write the block itself to the database.
//
// Note all the components of block(td, hash->number map, header, body, receipts)
// should be written atomically. BlockBatch is used for containing all components.
2022-07-05 11:14:21 +08:00
wg := sync . WaitGroup { }
wg . Add ( 1 )
go func ( ) {
blockBatch := bc . db . NewBatch ( )
rawdb . WriteTd ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , externTd )
rawdb . WriteBlock ( blockBatch , block )
rawdb . WriteReceipts ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , receipts )
rawdb . WritePreimages ( blockBatch , state . Preimages ( ) )
if err := blockBatch . Write ( ) ; err != nil {
log . Crit ( "Failed to write block into disk" , "err" , err )
}
wg . Done ( )
} ( )
2018-02-05 18:40:32 +02:00
2022-07-05 11:14:21 +08:00
tryCommitTrieDB := func ( ) error {
bc . commitLock . Lock ( )
defer bc . commitLock . Unlock ( )
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
// If node is running in path mode, skip explicit gc operation
// which is unnecessary in this mode.
if bc . triedb . Scheme ( ) == rawdb . PathScheme {
return nil
}
2022-07-05 11:14:21 +08:00
triedb := bc . stateCache . TrieDB ( )
// If we're running an archive node, always flush
if bc . cacheConfig . TrieDirtyDisabled {
2023-08-23 17:46:08 +08:00
err := triedb . Commit ( block . Root ( ) , false )
2022-07-05 11:14:21 +08:00
if err != nil {
return err
2018-06-04 10:47:43 +03:00
}
2022-07-05 11:14:21 +08:00
} else {
// Full but not archive node, do proper garbage collection
triedb . Reference ( block . Root ( ) , common . Hash { } ) // metadata reference to keep trie alive
bc . triegc . Push ( block . Root ( ) , - int64 ( block . NumberU64 ( ) ) )
if current := block . NumberU64 ( ) ; current > bc . triesInMemory {
// If we exceeded our memory allowance, flush matured singleton nodes to disk
var (
2023-11-06 14:11:17 +08:00
_ , nodes , _ , imgs = triedb . Size ( )
limit = common . StorageSize ( bc . cacheConfig . TrieDirtyLimit ) * 1024 * 1024
2022-07-05 11:14:21 +08:00
)
if nodes > limit || imgs > 4 * 1024 * 1024 {
triedb . Cap ( limit - ethdb . IdealBatchSize )
2021-03-19 13:42:26 +08:00
}
2022-07-05 11:14:21 +08:00
// Find the next state trie we need to commit
chosen := current - bc . triesInMemory
2023-08-23 17:46:08 +08:00
flushInterval := time . Duration ( bc . flushInterval . Load ( ) )
2022-07-05 11:14:21 +08:00
// If we exceeded out time allowance, flush an entire trie to disk
2023-08-23 17:46:08 +08:00
if bc . gcproc > flushInterval {
2022-07-05 11:14:21 +08:00
canWrite := true
if posa , ok := bc . engine . ( consensus . PoSA ) ; ok {
if ! posa . EnoughDistance ( bc , block . Header ( ) ) {
canWrite = false
}
}
if canWrite {
// If the header is missing (canonical chain behind), we're reorging a low
// diff sidechain. Suspend committing until this operation is completed.
header := bc . GetHeaderByNumber ( chosen )
if header == nil {
log . Warn ( "Reorg in progress, trie commit postponed" , "number" , chosen )
} else {
// If we're exceeding limits but haven't reached a large enough memory gap,
// warn the user that the system is becoming unstable.
2023-08-23 17:46:08 +08:00
if chosen < bc . lastWrite + bc . triesInMemory && bc . gcproc >= 2 * flushInterval {
log . Info ( "State in memory for too long, committing" , "time" , bc . gcproc , "allowance" , flushInterval , "optimum" , float64 ( chosen - bc . lastWrite ) / float64 ( bc . triesInMemory ) )
2022-07-05 11:14:21 +08:00
}
// Flush an entire trie and restart the counters
2023-08-23 17:46:08 +08:00
triedb . Commit ( header . Root , true )
2022-07-05 11:14:21 +08:00
rawdb . WriteSafePointBlockNumber ( bc . db , chosen )
2023-08-23 17:46:08 +08:00
bc . lastWrite = chosen
2022-07-05 11:14:21 +08:00
bc . gcproc = 0
2021-03-19 13:42:26 +08:00
}
2019-02-08 11:11:31 +02:00
}
2018-02-05 18:40:32 +02:00
}
2022-07-05 11:14:21 +08:00
// Garbage collect anything below our required write retention
2023-04-16 15:36:52 +08:00
wg2 := sync . WaitGroup { }
2022-07-05 11:14:21 +08:00
for ! bc . triegc . Empty ( ) {
root , number := bc . triegc . Pop ( )
if uint64 ( - number ) > chosen {
bc . triegc . Push ( root , number )
break
}
2023-04-16 15:36:52 +08:00
wg2 . Add ( 1 )
2023-04-03 16:11:09 +02:00
go func ( ) {
2023-08-23 17:46:08 +08:00
triedb . Dereference ( root )
2023-04-16 15:36:52 +08:00
wg2 . Done ( )
2023-04-03 16:11:09 +02:00
} ( )
2018-02-05 18:40:32 +02:00
}
2023-04-16 15:36:52 +08:00
wg2 . Wait ( )
2018-02-05 18:40:32 +02:00
}
}
2022-07-05 11:14:21 +08:00
return nil
}
// Commit all cached state changes into underlying memory database.
2023-08-23 17:46:08 +08:00
_ , diffLayer , err := state . Commit ( block . NumberU64 ( ) , bc . tryRewindBadBlocks , tryCommitTrieDB )
2022-07-05 11:14:21 +08:00
if err != nil {
return err
}
// Ensure no empty block body
if diffLayer != nil && block . Header ( ) . TxHash != types . EmptyRootHash {
// Filling necessary field
diffLayer . Receipts = receipts
diffLayer . BlockHash = block . Hash ( )
diffLayer . Number = block . NumberU64 ( )
diffLayerCh := make ( chan struct { } )
if bc . diffLayerChanCache . Len ( ) >= diffLayerCacheLimit {
bc . diffLayerChanCache . RemoveOldest ( )
}
bc . diffLayerChanCache . Add ( diffLayer . BlockHash , diffLayerCh )
go bc . cacheDiffLayer ( diffLayer , diffLayerCh )
2018-02-05 18:40:32 +02:00
}
2022-07-05 11:14:21 +08:00
wg . Wait ( )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
return nil
}
2022-05-05 00:22:00 -07:00
// WriteBlockAndSetHead writes the given block and all associated state to the database,
// and applies the block as the new chain head.
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
func ( bc * BlockChain ) WriteBlockAndSetHead ( block * types . Block , receipts [ ] * types . Receipt , logs [ ] * types . Log , state * state . StateDB , emitHeadEvent bool ) ( status WriteStatus , err error ) {
if ! bc . chainmu . TryLock ( ) {
return NonStatTy , errChainStopped
}
defer bc . chainmu . Unlock ( )
return bc . writeBlockAndSetHead ( block , receipts , logs , state , emitHeadEvent )
}
2022-05-05 00:22:00 -07:00
// writeBlockAndSetHead is the internal implementation of WriteBlockAndSetHead.
// This function expects the chain mutex to be held.
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
func ( bc * BlockChain ) writeBlockAndSetHead ( block * types . Block , receipts [ ] * types . Receipt , logs [ ] * types . Log , state * state . StateDB , emitHeadEvent bool ) ( status WriteStatus , err error ) {
2022-08-04 16:03:20 +08:00
if err := bc . writeBlockWithState ( block , receipts , state ) ; err != nil {
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
return NonStatTy , err
}
currentBlock := bc . CurrentBlock ( )
2023-08-23 17:46:08 +08:00
reorg , err := bc . forker . ReorgNeededWithFastFinality ( currentBlock , block . Header ( ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if err != nil {
return NonStatTy , err
2017-11-13 17:07:05 +02:00
}
if reorg {
2016-03-15 11:55:39 -07:00
// Reorganise the chain if the parent is not the head block
2018-02-26 10:53:10 +01:00
if block . ParentHash ( ) != currentBlock . Hash ( ) {
if err := bc . reorg ( currentBlock , block ) ; err != nil {
2015-07-03 11:24:42 +02:00
return NonStatTy , err
2015-06-29 12:12:30 +02:00
}
}
2015-09-29 14:24:28 +03:00
status = CanonStatTy
2015-06-29 12:12:30 +02:00
} else {
2015-07-03 11:24:42 +02:00
status = SideStatTy
2015-06-29 12:12:30 +02:00
}
2017-09-09 18:03:07 +02:00
// Set new head.
if status == CanonStatTy {
2020-01-17 18:49:32 +08:00
bc . writeHeadBlock ( block )
2017-09-09 18:03:07 +02:00
}
2017-05-11 09:55:48 +08:00
bc . futureBlocks . Remove ( block . Hash ( ) )
2019-11-29 21:22:08 +08:00
if status == CanonStatTy {
bc . chainFeed . Send ( ChainEvent { Block : block , Hash : block . Hash ( ) , Logs : logs } )
if len ( logs ) > 0 {
bc . logsFeed . Send ( logs )
}
2022-10-28 16:23:49 +08:00
// In theory, we should fire a ChainHeadEvent when we inject
2019-11-29 21:22:08 +08:00
// a canonical block, but sometimes we can insert a batch of
2022-08-19 01:00:21 -05:00
// canonical blocks. Avoid firing too many ChainHeadEvents,
2019-11-29 21:22:08 +08:00
// we will fire an accumulated ChainHeadEvent and disable fire
// event here.
if emitHeadEvent {
bc . chainHeadFeed . Send ( ChainHeadEvent { Block : block } )
2023-04-10 18:36:45 +08:00
if posa , ok := bc . Engine ( ) . ( consensus . PoSA ) ; ok {
if finalizedHeader := posa . GetFinalizedHeader ( bc , block . Header ( ) ) ; finalizedHeader != nil {
bc . finalizedHeaderFeed . Send ( FinalizedHeaderEvent { finalizedHeader } )
}
}
2019-11-29 21:22:08 +08:00
}
} else {
bc . chainSideFeed . Send ( ChainSideEvent { Block : block } )
}
2017-09-09 18:03:07 +02:00
return status , nil
2015-06-29 12:12:30 +02:00
}
2018-11-20 14:15:26 +02:00
// addFutureBlock checks if the block is within the max allowed window to get
// accepted for future processing, and returns an error if the block is too far
// ahead and was not added.
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
//
// TODO after the transition, the future block shouldn't be kept. Because
// it's not checked in the Geth side anymore.
2018-10-20 10:43:59 +02:00
func ( bc * BlockChain ) addFutureBlock ( block * types . Block ) error {
2019-04-02 22:28:48 +02:00
max := uint64 ( time . Now ( ) . Unix ( ) + maxTimeFutureBlocks )
if block . Time ( ) > max {
2018-11-20 14:15:26 +02:00
return fmt . Errorf ( "future block timestamp %v > allowed %v" , block . Time ( ) , max )
2018-10-20 10:43:59 +02:00
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if block . Difficulty ( ) . Cmp ( common . Big0 ) == 0 {
// Never add PoS blocks into the future queue
return nil
}
2018-10-20 10:43:59 +02:00
bc . futureBlocks . Add ( block . Hash ( ) , block )
return nil
}
2017-09-11 13:13:05 +03:00
// InsertChain attempts to insert the given batch of blocks in to the canonical
// chain or, otherwise, create a fork. If an error is returned it will return
// the index number of the failing block as well an error describing what went
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// wrong. After insertion is done, all accumulated events will be fired.
2017-05-11 09:55:48 +08:00
func ( bc * BlockChain ) InsertChain ( chain types . Blocks ) ( int , error ) {
2018-06-04 14:09:16 +03:00
// Sanity check that we have something meaningful to import
if len ( chain ) == 0 {
2018-11-20 14:15:26 +02:00
return 0 , nil
2018-06-04 14:09:16 +03:00
}
2019-02-26 12:32:48 +01:00
bc . blockProcFeed . Send ( true )
defer bc . blockProcFeed . Send ( false )
2021-10-07 15:47:50 +02:00
// Do a sanity check that the provided chain is actually ordered and linked.
2016-12-13 16:14:33 +02:00
for i := 1 ; i < len ( chain ) ; i ++ {
2021-10-07 15:47:50 +02:00
block , prev := chain [ i ] , chain [ i - 1 ]
2019-02-21 11:36:49 +01:00
if block . NumberU64 ( ) != prev . NumberU64 ( ) + 1 || block . ParentHash ( ) != prev . Hash ( ) {
2021-10-07 15:47:50 +02:00
log . Error ( "Non contiguous block insert" ,
"number" , block . Number ( ) ,
"hash" , block . Hash ( ) ,
"parent" , block . ParentHash ( ) ,
"prevnumber" , prev . Number ( ) ,
"prevhash" , prev . Hash ( ) ,
)
2021-04-15 20:35:00 +03:00
return 0 , fmt . Errorf ( "non contiguous insert: item %d is #%d [%x..], item %d is #%d [%x..] (parent [%x..])" , i - 1 , prev . NumberU64 ( ) ,
2019-02-21 11:36:49 +01:00
prev . Hash ( ) . Bytes ( ) [ : 4 ] , i , block . NumberU64 ( ) , block . Hash ( ) . Bytes ( ) [ : 4 ] , block . ParentHash ( ) . Bytes ( ) [ : 4 ] )
2016-12-13 16:14:33 +02:00
}
}
// Pre-checks passed, start the full block imports
2021-10-07 15:47:50 +02:00
if ! bc . chainmu . TryLock ( ) {
return 0 , errChainStopped
}
defer bc . chainmu . Unlock ( )
2023-05-03 12:58:39 +03:00
return bc . insertChain ( chain , true )
2021-04-16 21:29:22 +02:00
}
2019-04-02 21:01:02 +11:00
// insertChain is the internal implementation of InsertChain, which assumes that
2018-11-20 14:15:26 +02:00
// 1) chains are contiguous, and 2) The chain mutex is held.
//
// This method is split out so that import batches that require re-injecting
// historical blocks can do so without releasing the lock, which could lead to
// racey behaviour. If a sidechain import is in progress, and the historic state
// is imported, but then new canon-head is added before the actual sidechain
// completes, then the historic state could be pruned again
2023-05-03 12:58:39 +03:00
func ( bc * BlockChain ) insertChain ( chain types . Blocks , setHead bool ) ( int , error ) {
2021-10-07 15:47:50 +02:00
// If the chain is terminating, don't even bother starting up.
if bc . insertStopped ( ) {
2019-11-29 21:22:08 +08:00
return 0 , nil
2018-11-20 14:15:26 +02:00
}
2021-10-07 15:47:50 +02:00
2018-10-20 10:43:59 +02:00
// Start a parallel signature recovery (signer will fluke on fork transition, minimal perf loss)
2023-08-23 17:46:08 +08:00
signer := types . MakeSigner ( bc . chainConfig , chain [ 0 ] . Number ( ) , chain [ 0 ] . Time ( ) )
go SenderCacher . RecoverFromBlocks ( signer , chain )
2015-05-17 00:55:02 +02:00
2015-04-04 23:04:19 +02:00
var (
2019-11-29 21:22:08 +08:00
stats = insertStats { startTime : mclock . Now ( ) }
lastCanon * types . Block
2015-04-04 23:04:19 +02:00
)
2019-11-29 21:22:08 +08:00
// Fire a single chain head event if we've progressed the chain
defer func ( ) {
if lastCanon != nil && bc . CurrentBlock ( ) . Hash ( ) == lastCanon . Hash ( ) {
bc . chainHeadFeed . Send ( ChainHeadEvent { lastCanon } )
2023-04-10 18:36:45 +08:00
if posa , ok := bc . Engine ( ) . ( consensus . PoSA ) ; ok {
if finalizedHeader := posa . GetFinalizedHeader ( bc , lastCanon . Header ( ) ) ; finalizedHeader != nil {
bc . finalizedHeaderFeed . Send ( FinalizedHeaderEvent { finalizedHeader } )
}
}
2019-11-29 21:22:08 +08:00
}
} ( )
2017-04-05 01:16:29 +03:00
// Start the parallel header verifier
headers := make ( [ ] * types . Header , len ( chain ) )
for i , block := range chain {
headers [ i ] = block . Header ( )
}
2023-05-03 12:58:39 +03:00
abort , results := bc . engine . VerifyHeaders ( bc , headers )
2017-04-05 01:16:29 +03:00
defer close ( abort )
2015-06-19 16:21:20 +02:00
2018-11-20 14:15:26 +02:00
// Peek the error for the first block to decide the directing import logic
2019-03-25 12:41:50 +02:00
it := newInsertIterator ( chain , results , bc . validator )
2018-11-20 14:15:26 +02:00
block , err := it . next ( )
2019-02-21 11:36:49 +01:00
2021-11-01 21:09:36 +08:00
// Left-trim all the known blocks that don't need to build snapshot
if bc . skipBlock ( err , it ) {
2019-02-21 11:36:49 +01:00
// First block (and state) is known
// 1. We did a roll-back, and should now do a re-import
// 2. The block is stored as a sidechain, and is lying about it's stateroot, and passes a stateroot
2021-10-07 15:47:50 +02:00
// from the canonical chain, which has not been verified.
// Skip all known blocks that are behind us.
2019-05-08 19:30:36 +08:00
var (
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
reorg bool
current = bc . CurrentBlock ( )
2019-05-08 19:30:36 +08:00
)
2021-11-01 21:09:36 +08:00
for block != nil && bc . skipBlock ( err , it ) {
2023-08-23 17:46:08 +08:00
reorg , err = bc . forker . ReorgNeededWithFastFinality ( current , block . Header ( ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if err != nil {
return it . index , err
}
if reorg {
// Switch to import mode if the forker says the reorg is necessary
// and also the block is not on the canonical chain.
// In eth2 the forker always returns true for reorg decision (blindly trusting
// the external consensus engine), but in order to prevent the unnecessary
// reorgs when importing known blocks, the special case is handled here.
2023-03-02 08:29:15 +02:00
if block . NumberU64 ( ) > current . Number . Uint64 ( ) || bc . GetCanonicalHash ( block . NumberU64 ( ) ) != block . Hash ( ) {
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
break
}
2019-05-08 19:30:36 +08:00
}
2019-05-10 17:04:10 +03:00
log . Debug ( "Ignoring already known block" , "number" , block . Number ( ) , "hash" , block . Hash ( ) )
2019-02-21 11:36:49 +01:00
stats . ignored ++
2019-05-10 17:04:10 +03:00
2019-02-21 11:36:49 +01:00
block , err = it . next ( )
}
2019-05-08 19:30:36 +08:00
// The remaining blocks are still known blocks, the only scenario here is:
2023-07-14 13:06:51 +01:00
// During the snap sync, the pivot point is already submitted but rollback
2019-05-08 19:30:36 +08:00
// happens. Then node resets the head full block to a lower height via `rollback`
// and leaves a few known blocks in the database.
//
2023-07-14 13:06:51 +01:00
// When node runs a snap sync again, it can re-import a batch of known blocks via
2019-05-08 19:30:36 +08:00
// `insertChain` while a part of them have higher total difficulty than current
// head full block(new pivot point).
2021-11-01 21:09:36 +08:00
for block != nil && bc . skipBlock ( err , it ) {
2019-05-10 17:04:10 +03:00
log . Debug ( "Writing previously known block" , "number" , block . Number ( ) , "hash" , block . Hash ( ) )
2019-05-08 19:30:36 +08:00
if err := bc . writeKnownBlock ( block ) ; err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2019-05-08 19:30:36 +08:00
}
lastCanon = block
block , err = it . next ( )
}
2019-02-21 11:36:49 +01:00
// Falls through to the block import
}
2018-11-20 14:15:26 +02:00
switch {
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// First block is pruned
2020-08-05 09:52:54 +02:00
case errors . Is ( err , consensus . ErrPrunedAncestor ) :
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if setHead {
// First block is pruned, insert as sidechain and reorg only if TD grows enough
log . Debug ( "Pruned ancestor, inserting as sidechain" , "number" , block . Number ( ) , "hash" , block . Hash ( ) )
return bc . insertSideChain ( block , it )
} else {
// We're post-merge and the parent is pruned, try to recover the parent state
log . Debug ( "Pruned ancestor" , "number" , block . Number ( ) , "hash" , block . Hash ( ) )
2022-05-17 11:32:55 +02:00
_ , err := bc . recoverAncestors ( block )
return it . index , err
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
2018-11-20 14:15:26 +02:00
// First block is future, shove it (and all children) to the future queue (unknown ancestor)
2020-08-05 09:52:54 +02:00
case errors . Is ( err , consensus . ErrFutureBlock ) || ( errors . Is ( err , consensus . ErrUnknownAncestor ) && bc . futureBlocks . Contains ( it . first ( ) . ParentHash ( ) ) ) :
for block != nil && ( it . index == 0 || errors . Is ( err , consensus . ErrUnknownAncestor ) ) {
2019-05-10 17:04:10 +03:00
log . Debug ( "Future block, postponing import" , "number" , block . Number ( ) , "hash" , block . Hash ( ) )
2018-11-20 14:15:26 +02:00
if err := bc . addFutureBlock ( block ) ; err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2018-10-20 10:43:59 +02:00
}
2018-11-20 14:15:26 +02:00
block , err = it . next ( )
}
stats . queued += it . processed ( )
stats . ignored += it . remaining ( )
// If there are any still remaining, mark as ignored
2019-11-29 21:22:08 +08:00
return it . index , err
2018-11-20 14:15:26 +02:00
2021-11-01 21:09:36 +08:00
// Some other error(except ErrKnownBlock) occurred, abort.
// ErrKnownBlock is allowed here since some known blocks
// still need re-execution to generate snapshots that are missing
case err != nil && ! errors . Is ( err , ErrKnownBlock ) :
2019-09-26 16:57:51 +08:00
bc . futureBlocks . Remove ( block . Hash ( ) )
2018-11-20 14:15:26 +02:00
stats . ignored += len ( it . chain )
bc . reportBlock ( block , nil , err )
2019-11-29 21:22:08 +08:00
return it . index , err
2018-10-20 10:43:59 +02:00
}
2021-02-12 12:45:34 +02:00
2021-11-01 21:09:36 +08:00
for ; block != nil && err == nil || errors . Is ( err , ErrKnownBlock ) ; block , err = it . next ( ) {
2017-04-05 01:16:29 +03:00
// If the chain is terminating, stop processing blocks
2020-05-26 21:37:37 +02:00
if bc . insertStopped ( ) {
log . Debug ( "Abort during block processing" )
2015-06-12 16:45:53 +02:00
break
}
2017-04-05 01:16:29 +03:00
// If the header is a banned one, straight out abort
2015-06-12 16:45:53 +02:00
if BadHashes [ block . Hash ( ) ] {
2021-07-29 10:17:40 +02:00
bc . reportBlock ( block , nil , ErrBannedHash )
return it . index , ErrBannedHash
2015-06-12 16:45:53 +02:00
}
2019-05-10 17:04:10 +03:00
// If the block is known (in the middle of the chain), it's a special case for
// Clique blocks where they can share state among each other, so importing an
// older block might complete the state of the subsequent one. In this case,
// just skip the block (we already validated it once fully (and crashed), since
2021-11-01 21:09:36 +08:00
// its header and body was already in the database). But if the corresponding
// snapshot layer is missing, forcibly rerun the execution to build it.
if bc . skipBlock ( err , it ) {
2019-05-10 17:04:10 +03:00
logger := log . Debug
if bc . chainConfig . Clique == nil {
logger = log . Warn
}
logger ( "Inserted known block" , "number" , block . Number ( ) , "hash" , block . Hash ( ) ,
"uncles" , len ( block . Uncles ( ) ) , "txs" , len ( block . Transactions ( ) ) , "gas" , block . GasUsed ( ) ,
"root" , block . Root ( ) )
2020-05-13 16:33:48 +08:00
// Special case. Commit the empty receipt slice if we meet the known
// block in the middle. It can only happen in the clique chain. Whenever
// we insert blocks via `insertSideChain`, we only commit `td`, `header`
// and `body` if it's non-existent. Since we don't have receipts without
2022-08-19 01:00:21 -05:00
// reexecution, so nothing to commit. But if the sidechain will be adopted
2020-05-13 16:33:48 +08:00
// as the canonical chain eventually, it needs to be reexecuted for missing
// state, but if it's this special case here(skip reexecution) we will lose
// the empty receipt entry.
if len ( block . Transactions ( ) ) == 0 {
rawdb . WriteReceipts ( bc . db , block . Hash ( ) , block . NumberU64 ( ) , nil )
} else {
log . Error ( "Please file an issue, skip known block execution without receipt" ,
"hash" , block . Hash ( ) , "number" , block . NumberU64 ( ) )
}
2019-05-10 17:04:10 +03:00
if err := bc . writeKnownBlock ( block ) ; err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2019-05-10 17:04:10 +03:00
}
stats . processed ++
// We can assume that logs are empty here, since the only way for consecutive
// Clique blocks to have the same state is if there are no transactions.
lastCanon = block
continue
}
2021-10-07 15:47:50 +02:00
2018-11-20 14:15:26 +02:00
// Retrieve the parent block and it's state to execute on top
start := time . Now ( )
parent := it . previous ( )
2018-10-20 10:43:59 +02:00
if parent == nil {
2019-03-13 12:31:35 +02:00
parent = bc . GetHeader ( block . ParentHash ( ) , block . NumberU64 ( ) - 1 )
2016-05-19 13:24:14 +03:00
}
2022-07-05 11:14:21 +08:00
statedb , err := state . NewWithSharedPool ( parent . Root , bc . stateCache , bc . snaps )
2015-10-19 16:08:17 +02:00
if err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2015-06-12 16:45:53 +02:00
}
2022-07-05 11:14:21 +08:00
bc . updateHighestVerifiedHeader ( block . Header ( ) )
2021-10-07 15:47:50 +02:00
2021-01-08 15:01:49 +02:00
// Enable prefetching to pull in trie node paths while processing transactions
statedb . StartPrefetcher ( "chain" )
2022-07-05 11:14:21 +08:00
interruptCh := make ( chan struct { } )
// For diff sync, it may fallback to full sync, so we still do prefetch
if len ( block . Transactions ( ) ) >= prefetchTxNumber {
// do Prefetch in a separate goroutine to avoid blocking the critical path
[Feature]: Improve trie prefetch (#952)
* trie prefetcher for From/To address in advance
We found that trie prefetch could be not fast enough, especially trie prefetch of
the outer big state trie tree.
Instead of do trie prefetch until a transaction is finalized, we could do trie prefetch
in advance. Try to prefetch the trie node of the From/To accounts, since their root hash
are most likely to be changed.
* Parallel TriePrefetch for large trie update.
Currently, we create a subfetch for each account address to do trie prefetch. If the address
has very large state change, trie prefetch could be not fast enough, e.g. a contract modified
lots of KV pair or a large number of account's root hash is changed in a block.
With this commit, there will be children subfetcher created to do trie prefetch in parallell if
the parent subfetch's workload exceed the threshold.
* some improvemnts of parallel trie prefetch implementation
1.childrenLock is removed, since it is not necessary
APIs of triePrefetcher is not thread safe, they should be used sequentially.
A prefetch will be interrupted by trie() or clos(), so we only need mark it as
interrupted and check before call scheduleParallel to avoid the concurrent access to paraChildren
2.rename subfetcher.children to subfetcher.paraChildren
3.use subfetcher.pendingSize to replace totalSize & processedIndex
4.randomly select the start child to avoid always feed the first one
5.increase threshold and capacity to avoid create too many child routine
* fix review comments
** nil check refine
** create a separate routine for From/To prefetch, avoid blocking the cirtical path
* remove the interrupt member
* not create a signer for each transaction
* some changes to triePrefetcher
** remove the abortLoop, move the subfetcher abort operation into mainLoop
since we want to make subfetcher's create & schedule & abort within a loop to
avoid concurrent access locks.
** no wait subfetcher's term signal in abort()
it could speed up the close by closing subfetcher concurrently.
we send stop signnal to all subfetchers in burst and wait their term signal later.
* some coding improve for subfetcher.scheduleParallel
* fix a UT crash of s.prefetcher == nil
* update parallel trie prefetcher configuration
tested with different combination of parallelTriePrefetchThreshold & parallelTriePrefetchCapacity,
found the most efficient configure could be:
parallelTriePrefetchThreshold = 10
parallelTriePrefetchCapacity = 20
* fix review comments: code refine
2022-07-15 19:17:08 +08:00
// 1.do state prefetch for snapshot cache
2022-07-26 22:23:55 +08:00
throwaway := statedb . CopyDoPrefetch ( )
2022-07-05 11:14:21 +08:00
go bc . prefetcher . Prefetch ( block , throwaway , & bc . vmConfig , interruptCh )
[Feature]: Improve trie prefetch (#952)
* trie prefetcher for From/To address in advance
We found that trie prefetch could be not fast enough, especially trie prefetch of
the outer big state trie tree.
Instead of do trie prefetch until a transaction is finalized, we could do trie prefetch
in advance. Try to prefetch the trie node of the From/To accounts, since their root hash
are most likely to be changed.
* Parallel TriePrefetch for large trie update.
Currently, we create a subfetch for each account address to do trie prefetch. If the address
has very large state change, trie prefetch could be not fast enough, e.g. a contract modified
lots of KV pair or a large number of account's root hash is changed in a block.
With this commit, there will be children subfetcher created to do trie prefetch in parallell if
the parent subfetch's workload exceed the threshold.
* some improvemnts of parallel trie prefetch implementation
1.childrenLock is removed, since it is not necessary
APIs of triePrefetcher is not thread safe, they should be used sequentially.
A prefetch will be interrupted by trie() or clos(), so we only need mark it as
interrupted and check before call scheduleParallel to avoid the concurrent access to paraChildren
2.rename subfetcher.children to subfetcher.paraChildren
3.use subfetcher.pendingSize to replace totalSize & processedIndex
4.randomly select the start child to avoid always feed the first one
5.increase threshold and capacity to avoid create too many child routine
* fix review comments
** nil check refine
** create a separate routine for From/To prefetch, avoid blocking the cirtical path
* remove the interrupt member
* not create a signer for each transaction
* some changes to triePrefetcher
** remove the abortLoop, move the subfetcher abort operation into mainLoop
since we want to make subfetcher's create & schedule & abort within a loop to
avoid concurrent access locks.
** no wait subfetcher's term signal in abort()
it could speed up the close by closing subfetcher concurrently.
we send stop signnal to all subfetchers in burst and wait their term signal later.
* some coding improve for subfetcher.scheduleParallel
* fix a UT crash of s.prefetcher == nil
* update parallel trie prefetcher configuration
tested with different combination of parallelTriePrefetchThreshold & parallelTriePrefetchCapacity,
found the most efficient configure could be:
parallelTriePrefetchThreshold = 10
parallelTriePrefetchCapacity = 20
* fix review comments: code refine
2022-07-15 19:17:08 +08:00
// 2.do trie prefetch for MPT trie node cache
// it is for the big state trie tree, prefetch based on transaction's From/To address.
// trie prefetcher is thread safe now, ok to prefetch in a separate routine
2022-07-26 22:23:55 +08:00
go throwaway . TriePrefetchInAdvance ( block , signer )
2022-07-05 11:14:21 +08:00
}
[Feature]: Improve trie prefetch (#952)
* trie prefetcher for From/To address in advance
We found that trie prefetch could be not fast enough, especially trie prefetch of
the outer big state trie tree.
Instead of do trie prefetch until a transaction is finalized, we could do trie prefetch
in advance. Try to prefetch the trie node of the From/To accounts, since their root hash
are most likely to be changed.
* Parallel TriePrefetch for large trie update.
Currently, we create a subfetch for each account address to do trie prefetch. If the address
has very large state change, trie prefetch could be not fast enough, e.g. a contract modified
lots of KV pair or a large number of account's root hash is changed in a block.
With this commit, there will be children subfetcher created to do trie prefetch in parallell if
the parent subfetch's workload exceed the threshold.
* some improvemnts of parallel trie prefetch implementation
1.childrenLock is removed, since it is not necessary
APIs of triePrefetcher is not thread safe, they should be used sequentially.
A prefetch will be interrupted by trie() or clos(), so we only need mark it as
interrupted and check before call scheduleParallel to avoid the concurrent access to paraChildren
2.rename subfetcher.children to subfetcher.paraChildren
3.use subfetcher.pendingSize to replace totalSize & processedIndex
4.randomly select the start child to avoid always feed the first one
5.increase threshold and capacity to avoid create too many child routine
* fix review comments
** nil check refine
** create a separate routine for From/To prefetch, avoid blocking the cirtical path
* remove the interrupt member
* not create a signer for each transaction
* some changes to triePrefetcher
** remove the abortLoop, move the subfetcher abort operation into mainLoop
since we want to make subfetcher's create & schedule & abort within a loop to
avoid concurrent access locks.
** no wait subfetcher's term signal in abort()
it could speed up the close by closing subfetcher concurrently.
we send stop signnal to all subfetchers in burst and wait their term signal later.
* some coding improve for subfetcher.scheduleParallel
* fix a UT crash of s.prefetcher == nil
* update parallel trie prefetcher configuration
tested with different combination of parallelTriePrefetchThreshold & parallelTriePrefetchCapacity,
found the most efficient configure could be:
parallelTriePrefetchThreshold = 10
parallelTriePrefetchCapacity = 20
* fix review comments: code refine
2022-07-15 19:17:08 +08:00
2022-07-05 11:14:21 +08:00
//Process block using the parent state as reference point
if bc . pipeCommit {
statedb . EnablePipeCommit ( )
}
statedb . SetExpectedStateRoot ( block . Root ( ) )
2023-03-17 03:34:25 +08:00
pstart := time . Now ( )
2022-07-05 11:14:21 +08:00
statedb , receipts , logs , usedGas , err := bc . processor . Process ( block , statedb , bc . vmConfig )
close ( interruptCh ) // state prefetch can be stopped
2015-10-19 16:08:17 +02:00
if err != nil {
2017-05-11 09:55:48 +08:00
bc . reportBlock ( block , receipts , err )
2022-08-31 13:30:25 +08:00
statedb . StopPrefetcher ( )
2019-11-29 21:22:08 +08:00
return it . index , err
2015-10-19 16:08:17 +02:00
}
2023-03-17 03:34:25 +08:00
ptime := time . Since ( pstart )
2019-03-27 13:02:04 +02:00
2015-10-19 16:08:17 +02:00
// Validate the state using the default validator
2023-03-17 03:34:25 +08:00
vstart := time . Now ( )
2023-09-07 16:39:29 +08:00
if err := bc . validator . ValidateState ( block , statedb , receipts , usedGas ) ; err != nil {
log . Error ( "validate state failed" , "error" , err )
bc . reportBlock ( block , receipts , err )
statedb . StopPrefetcher ( )
return it . index , err
2015-10-19 16:08:17 +02:00
}
2023-03-17 03:34:25 +08:00
vtime := time . Since ( vstart )
proctime := time . Since ( start ) // processing + validation
2022-07-05 11:14:21 +08:00
bc . cacheBlock ( block . Hash ( ) , block )
2023-03-17 03:34:25 +08:00
// Update the metrics touched during block processing and validation
accountReadTimer . Update ( statedb . AccountReads ) // Account reads are complete(in processing)
storageReadTimer . Update ( statedb . StorageReads ) // Storage reads are complete(in processing)
snapshotAccountReadTimer . Update ( statedb . SnapshotAccountReads ) // Account reads are complete(in processing)
snapshotStorageReadTimer . Update ( statedb . SnapshotStorageReads ) // Storage reads are complete(in processing)
accountUpdateTimer . Update ( statedb . AccountUpdates ) // Account updates are complete(in validation)
storageUpdateTimer . Update ( statedb . StorageUpdates ) // Storage updates are complete(in validation)
accountHashTimer . Update ( statedb . AccountHashes ) // Account hashes are complete(in validation)
storageHashTimer . Update ( statedb . StorageHashes ) // Storage hashes are complete(in validation)
triehash := statedb . AccountHashes + statedb . StorageHashes // The time spent on tries hashing
trieUpdate := statedb . AccountUpdates + statedb . StorageUpdates // The time spent on tries update
trieRead := statedb . SnapshotAccountReads + statedb . AccountReads // The time spent on account read
trieRead += statedb . SnapshotStorageReads + statedb . StorageReads // The time spent on storage read
blockExecutionTimer . Update ( ptime - trieRead ) // The time spent on EVM processing
blockValidationTimer . Update ( vtime - ( triehash + trieUpdate ) ) // The time spent on block validation
2019-03-27 13:02:04 +02:00
2017-09-09 18:03:07 +02:00
// Write the block to the chain and get the status.
2023-03-17 03:34:25 +08:00
var (
wstart = time . Now ( )
status WriteStatus
)
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if ! setHead {
// Don't set the head, only insert the block
2022-08-04 16:03:20 +08:00
err = bc . writeBlockWithState ( block , receipts , statedb )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
} else {
status , err = bc . writeBlockAndSetHead ( block , receipts , logs , statedb , false )
}
2015-06-29 12:12:30 +02:00
if err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2015-06-29 12:12:30 +02:00
}
2023-11-14 16:40:25 +08:00
bc . cacheReceipts ( block . Hash ( ) , receipts , block )
2019-03-27 13:02:04 +02:00
// Update the metrics touched during block commit
2019-08-06 13:40:28 +03:00
accountCommitTimer . Update ( statedb . AccountCommits ) // Account commits are complete, we can mark them
storageCommitTimer . Update ( statedb . StorageCommits ) // Storage commits are complete, we can mark them
snapshotCommitTimer . Update ( statedb . SnapshotCommits ) // Snapshot commits are complete, we can mark them
2023-03-17 03:34:25 +08:00
triedbCommitTimer . Update ( statedb . TrieDBCommits ) // Trie database commits are complete, we can mark them
2019-03-25 10:01:18 +02:00
2023-03-17 03:34:25 +08:00
blockWriteTimer . Update ( time . Since ( wstart ) - statedb . AccountCommits - statedb . StorageCommits - statedb . SnapshotCommits - statedb . TrieDBCommits )
2018-11-28 09:29:05 +01:00
blockInsertTimer . UpdateSince ( start )
2019-03-25 10:01:18 +02:00
2022-03-11 14:14:45 +02:00
// Report the import stats before returning the various results
stats . processed ++
stats . usedGas += usedGas
2023-11-06 14:11:17 +08:00
trieDiffNodes , trieBufNodes , trieImmutableBufNodes , _ := bc . triedb . Size ( )
stats . report ( chain , it . index , trieDiffNodes , trieBufNodes , trieImmutableBufNodes , setHead )
2022-03-11 14:14:45 +02:00
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if ! setHead {
2022-12-07 17:36:54 +01:00
// After merge we expect few side chains. Simply count
// all blocks the CL gives us for GC processing time
bc . gcproc += proctime
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
2022-03-11 14:14:45 +02:00
return it . index , nil // Direct block insertion of a single block
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
2015-06-29 12:12:30 +02:00
switch status {
2015-07-03 11:24:42 +02:00
case CanonStatTy :
2018-10-20 10:43:59 +02:00
log . Debug ( "Inserted new block" , "number" , block . Number ( ) , "hash" , block . Hash ( ) ,
"uncles" , len ( block . Uncles ( ) ) , "txs" , len ( block . Transactions ( ) ) , "gas" , block . GasUsed ( ) ,
2018-11-20 14:15:26 +02:00
"elapsed" , common . PrettyDuration ( time . Since ( start ) ) ,
"root" , block . Root ( ) )
2017-09-11 13:13:05 +03:00
lastCanon = block
2018-02-05 18:40:32 +02:00
// Only count canonical blocks for GC processing time
bc . gcproc += proctime
2015-07-03 11:24:42 +02:00
case SideStatTy :
2018-10-20 10:43:59 +02:00
log . Debug ( "Inserted forked block" , "number" , block . Number ( ) , "hash" , block . Hash ( ) ,
2018-11-20 14:15:26 +02:00
"diff" , block . Difficulty ( ) , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) ,
2018-10-20 10:43:59 +02:00
"txs" , len ( block . Transactions ( ) ) , "gas" , block . GasUsed ( ) , "uncles" , len ( block . Uncles ( ) ) ,
2018-11-20 14:15:26 +02:00
"root" , block . Root ( ) )
2019-05-10 17:04:10 +03:00
default :
// This in theory is impossible, but lets be nice to our future selves and leave
// a log, instead of trying to track down blocks imports that don't emit logs.
log . Warn ( "Inserted block with unknown status" , "number" , block . Number ( ) , "hash" , block . Hash ( ) ,
"diff" , block . Difficulty ( ) , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) ,
"txs" , len ( block . Transactions ( ) ) , "gas" , block . GasUsed ( ) , "uncles" , len ( block . Uncles ( ) ) ,
"root" , block . Root ( ) )
2015-06-12 13:36:38 +02:00
}
2022-07-05 11:14:21 +08:00
bc . chainBlockFeed . Send ( ChainHeadEvent { block } )
2014-11-04 12:46:33 +01:00
}
2021-10-07 15:47:50 +02:00
2018-11-20 14:15:26 +02:00
// Any blocks remaining here? The only ones we care about are the future ones
2020-08-05 09:52:54 +02:00
if block != nil && errors . Is ( err , consensus . ErrFutureBlock ) {
2018-11-20 14:15:26 +02:00
if err := bc . addFutureBlock ( block ) ; err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2018-10-20 10:43:59 +02:00
}
2018-11-20 14:15:26 +02:00
block , err = it . next ( )
2020-08-05 09:52:54 +02:00
for ; block != nil && errors . Is ( err , consensus . ErrUnknownAncestor ) ; block , err = it . next ( ) {
2018-11-20 14:15:26 +02:00
if err := bc . addFutureBlock ( block ) ; err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2018-10-20 10:43:59 +02:00
}
stats . queued ++
}
2014-11-04 12:46:33 +01:00
}
2018-11-20 14:15:26 +02:00
stats . ignored += it . remaining ( )
2019-11-29 21:22:08 +08:00
return it . index , err
2014-11-17 12:12:55 +01:00
}
2015-01-02 12:07:54 +01:00
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) updateHighestVerifiedHeader ( header * types . Header ) {
if header == nil || header . Number == nil {
return
}
2023-09-07 16:39:29 +08:00
currentHeader := bc . highestVerifiedHeader . Load ( )
2022-07-05 11:14:21 +08:00
if currentHeader == nil {
bc . highestVerifiedHeader . Store ( types . CopyHeader ( header ) )
return
}
newParentTD := bc . GetTd ( header . ParentHash , header . Number . Uint64 ( ) - 1 )
if newParentTD == nil {
newParentTD = big . NewInt ( 0 )
}
oldParentTD := bc . GetTd ( currentHeader . ParentHash , currentHeader . Number . Uint64 ( ) - 1 )
if oldParentTD == nil {
oldParentTD = big . NewInt ( 0 )
}
newTD := big . NewInt ( 0 ) . Add ( newParentTD , header . Difficulty )
oldTD := big . NewInt ( 0 ) . Add ( oldParentTD , currentHeader . Difficulty )
if newTD . Cmp ( oldTD ) > 0 {
bc . highestVerifiedHeader . Store ( types . CopyHeader ( header ) )
return
}
}
func ( bc * BlockChain ) GetHighestVerifiedHeader ( ) * types . Header {
2023-09-07 16:39:29 +08:00
return bc . highestVerifiedHeader . Load ( )
2022-07-05 11:14:21 +08:00
}
2019-05-08 19:30:36 +08:00
// insertSideChain is called when an import batch hits upon a pruned ancestor
2018-11-20 14:15:26 +02:00
// error, which happens when a sidechain with a sufficiently old fork-block is
// found.
//
// The method writes all (header-and-body-valid) blocks to disk, then tries to
// switch over to the new chain if the TD exceeded the current chain.
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// insertSideChain is only used pre-merge.
2019-11-29 21:22:08 +08:00
func ( bc * BlockChain ) insertSideChain ( block * types . Block , it * insertIterator ) ( int , error ) {
2016-10-18 11:18:07 +03:00
var (
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
externTd * big . Int
lastBlock = block
current = bc . CurrentBlock ( )
2016-10-18 11:18:07 +03:00
)
2018-11-20 14:15:26 +02:00
// The first sidechain block error is already verified to be ErrPrunedAncestor.
// Since we don't import them here, we expect ErrUnknownAncestor for the remaining
// ones. Any other errors means that the block is invalid, and should not be written
// to disk.
2019-02-04 13:30:19 +01:00
err := consensus . ErrPrunedAncestor
2020-08-05 09:52:54 +02:00
for ; block != nil && errors . Is ( err , consensus . ErrPrunedAncestor ) ; block , err = it . next ( ) {
2018-11-20 14:15:26 +02:00
// Check the canonical state root for that number
2023-03-02 08:29:15 +02:00
if number := block . NumberU64 ( ) ; current . Number . Uint64 ( ) >= number {
2019-02-21 11:36:49 +01:00
canonical := bc . GetBlockByNumber ( number )
if canonical != nil && canonical . Hash ( ) == block . Hash ( ) {
// Not a sidechain block, this is a re-import of a canon block which has it's state pruned
2019-08-21 09:17:19 +02:00
// Collect the TD of the block. Since we know it's a canon one,
// we can get it directly, and not (like further below) use
// the parent and then add the block on top
externTd = bc . GetTd ( block . Hash ( ) , block . NumberU64 ( ) )
2019-02-21 11:36:49 +01:00
continue
}
if canonical != nil && canonical . Root ( ) == block . Root ( ) {
2018-11-20 14:15:26 +02:00
// This is most likely a shadow-state attack. When a fork is imported into the
// database, and it eventually reaches a block height which is not pruned, we
// just found that the state already exist! This means that the sidechain block
2018-10-20 10:43:59 +02:00
// refers to a state which already exists in our canon chain.
2018-11-20 14:15:26 +02:00
//
// If left unchecked, we would now proceed importing the blocks, without actually
// having verified the state of the previous blocks.
log . Warn ( "Sidechain ghost-state attack detected" , "number" , block . NumberU64 ( ) , "sideroot" , block . Root ( ) , "canonroot" , canonical . Root ( ) )
2018-10-20 10:43:59 +02:00
// If someone legitimately side-mines blocks, they would still be imported as usual. However,
// we cannot risk writing unverified blocks to disk when they obviously target the pruning
// mechanism.
2019-11-29 21:22:08 +08:00
return it . index , errors . New ( "sidechain ghost-state attack" )
2018-10-20 10:43:59 +02:00
}
2016-10-18 11:18:07 +03:00
}
2018-10-20 10:43:59 +02:00
if externTd == nil {
externTd = bc . GetTd ( block . ParentHash ( ) , block . NumberU64 ( ) - 1 )
2018-09-20 11:41:59 +03:00
}
2018-10-20 10:43:59 +02:00
externTd = new ( big . Int ) . Add ( externTd , block . Difficulty ( ) )
2018-09-20 11:41:59 +03:00
2018-10-20 10:43:59 +02:00
if ! bc . HasBlock ( block . Hash ( ) , block . NumberU64 ( ) ) {
2018-11-20 14:15:26 +02:00
start := time . Now ( )
2019-05-08 19:30:36 +08:00
if err := bc . writeBlockWithoutState ( block , externTd ) ; err != nil {
2019-11-29 21:22:08 +08:00
return it . index , err
2018-10-20 10:43:59 +02:00
}
2019-02-08 11:11:31 +02:00
log . Debug ( "Injected sidechain block" , "number" , block . Number ( ) , "hash" , block . Hash ( ) ,
2018-11-20 14:15:26 +02:00
"diff" , block . Difficulty ( ) , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) ,
"txs" , len ( block . Transactions ( ) ) , "gas" , block . GasUsed ( ) , "uncles" , len ( block . Uncles ( ) ) ,
"root" , block . Root ( ) )
2017-02-28 13:35:17 +02:00
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
lastBlock = block
2018-10-20 10:43:59 +02:00
}
2018-11-20 14:15:26 +02:00
// At this point, we've written all sidechain blocks to database. Loop ended
// either on some other error or all were processed. If there was some other
// error, we can ignore the rest of those blocks.
2018-10-20 10:43:59 +02:00
//
// If the externTd was larger than our local TD, we now need to reimport the previous
// blocks to regenerate the required state
2023-08-23 17:46:08 +08:00
reorg , err := bc . forker . ReorgNeededWithFastFinality ( current , lastBlock . Header ( ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if err != nil {
return it . index , err
}
if ! reorg {
2023-03-02 08:29:15 +02:00
localTd := bc . GetTd ( current . Hash ( ) , current . Number . Uint64 ( ) )
2019-03-13 12:31:35 +02:00
log . Info ( "Sidechain written to disk" , "start" , it . first ( ) . NumberU64 ( ) , "end" , it . previous ( ) . Number , "sidetd" , externTd , "localtd" , localTd )
2019-11-29 21:22:08 +08:00
return it . index , err
2018-10-20 10:43:59 +02:00
}
2018-11-20 14:15:26 +02:00
// Gather all the sidechain hashes (full blocks may be memory heavy)
var (
hashes [ ] common . Hash
numbers [ ] uint64
)
2019-03-13 12:31:35 +02:00
parent := it . previous ( )
2018-11-20 14:15:26 +02:00
for parent != nil && ! bc . HasState ( parent . Root ) {
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if bc . stateRecoverable ( parent . Root ) {
if err := bc . triedb . Recover ( parent . Root ) ; err != nil {
return 0 , err
}
break
}
2018-11-20 14:15:26 +02:00
hashes = append ( hashes , parent . Hash ( ) )
numbers = append ( numbers , parent . Number . Uint64 ( ) )
2016-10-18 11:18:07 +03:00
2018-11-20 14:15:26 +02:00
parent = bc . GetHeader ( parent . ParentHash , parent . Number . Uint64 ( ) - 1 )
2016-10-07 14:25:01 +02:00
}
2018-11-20 14:15:26 +02:00
if parent == nil {
2019-11-29 21:22:08 +08:00
return it . index , errors . New ( "missing parent" )
2018-10-20 10:43:59 +02:00
}
// Import all the pruned blocks to make the state available
2016-10-18 11:18:07 +03:00
var (
2018-11-20 14:15:26 +02:00
blocks [ ] * types . Block
2022-10-26 15:23:07 +03:00
memory uint64
2016-10-18 11:18:07 +03:00
)
2018-11-20 14:15:26 +02:00
for i := len ( hashes ) - 1 ; i >= 0 ; i -- {
// Append the next block to our batch
block := bc . GetBlock ( hashes [ i ] , numbers [ i ] )
2022-07-05 11:14:21 +08:00
if block == nil {
log . Crit ( "Importing heavy sidechain block is nil" , "hash" , hashes [ i ] , "number" , numbers [ i ] )
}
2018-09-20 11:41:59 +03:00
2018-11-20 14:15:26 +02:00
blocks = append ( blocks , block )
memory += block . Size ( )
// If memory use grew too large, import and continue. Sadly we need to discard
// all raised events and logs from notifications since we're too heavy on the
// memory here.
if len ( blocks ) >= 2048 || memory > 64 * 1024 * 1024 {
log . Info ( "Importing heavy sidechain segment" , "blocks" , len ( blocks ) , "start" , blocks [ 0 ] . NumberU64 ( ) , "end" , block . NumberU64 ( ) )
2023-05-03 12:58:39 +03:00
if _ , err := bc . insertChain ( blocks , true ) ; err != nil {
2019-11-29 21:22:08 +08:00
return 0 , err
2018-11-20 14:15:26 +02:00
}
blocks , memory = blocks [ : 0 ] , 0
2016-10-07 14:25:01 +02:00
2018-11-20 14:15:26 +02:00
// If the chain is terminating, stop processing blocks
2020-05-26 21:37:37 +02:00
if bc . insertStopped ( ) {
log . Debug ( "Abort during blocks processing" )
2019-11-29 21:22:08 +08:00
return 0 , nil
2018-11-20 14:15:26 +02:00
}
}
2016-10-07 14:25:01 +02:00
}
2018-11-20 14:15:26 +02:00
if len ( blocks ) > 0 {
log . Info ( "Importing sidechain segment" , "start" , blocks [ 0 ] . NumberU64 ( ) , "end" , blocks [ len ( blocks ) - 1 ] . NumberU64 ( ) )
2023-05-03 12:58:39 +03:00
return bc . insertChain ( blocks , true )
2016-10-07 14:25:01 +02:00
}
2019-11-29 21:22:08 +08:00
return 0 , nil
2016-10-07 14:25:01 +02:00
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// recoverAncestors finds the closest ancestor with available state and re-execute
// all the ancestor blocks since that.
// recoverAncestors is only used post-merge.
2022-05-17 11:32:55 +02:00
// We return the hash of the latest block that we could correctly validate.
func ( bc * BlockChain ) recoverAncestors ( block * types . Block ) ( common . Hash , error ) {
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// Gather all the sidechain hashes (full blocks may be memory heavy)
var (
hashes [ ] common . Hash
numbers [ ] uint64
parent = block
)
for parent != nil && ! bc . HasState ( parent . Root ( ) ) {
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
if bc . stateRecoverable ( parent . Root ( ) ) {
if err := bc . triedb . Recover ( parent . Root ( ) ) ; err != nil {
return common . Hash { } , err
}
break
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
hashes = append ( hashes , parent . Hash ( ) )
numbers = append ( numbers , parent . NumberU64 ( ) )
parent = bc . GetBlock ( parent . ParentHash ( ) , parent . NumberU64 ( ) - 1 )
// If the chain is terminating, stop iteration
if bc . insertStopped ( ) {
log . Debug ( "Abort during blocks iteration" )
2022-05-17 11:32:55 +02:00
return common . Hash { } , errInsertionInterrupted
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
}
if parent == nil {
2022-05-17 11:32:55 +02:00
return common . Hash { } , errors . New ( "missing parent" )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
// Import all the pruned blocks to make the state available
for i := len ( hashes ) - 1 ; i >= 0 ; i -- {
// If the chain is terminating, stop processing blocks
if bc . insertStopped ( ) {
log . Debug ( "Abort during blocks processing" )
2022-05-17 11:32:55 +02:00
return common . Hash { } , errInsertionInterrupted
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
var b * types . Block
if i == 0 {
b = block
} else {
b = bc . GetBlock ( hashes [ i ] , numbers [ i ] )
}
2023-05-03 12:58:39 +03:00
if _ , err := bc . insertChain ( types . Blocks { b } , false ) ; err != nil {
2022-05-17 11:32:55 +02:00
return b . ParentHash ( ) , err
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
}
2022-05-17 11:32:55 +02:00
return block . Hash ( ) , nil
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
// collectLogs collects the logs that were generated or removed during
2022-12-09 16:14:33 +01:00
// the processing of a block. These logs are later announced as deleted or reborn.
func ( bc * BlockChain ) collectLogs ( b * types . Block , removed bool ) [ ] * types . Log {
2023-07-27 16:53:28 +03:00
var blobGasPrice * big . Int
excessBlobGas := b . ExcessBlobGas ( )
if excessBlobGas != nil {
blobGasPrice = eip4844 . CalcBlobFee ( * excessBlobGas )
2023-07-27 13:11:09 +02:00
}
2022-12-09 16:14:33 +01:00
receipts := rawdb . ReadRawReceipts ( bc . db , b . Hash ( ) , b . NumberU64 ( ) )
2023-07-27 16:53:28 +03:00
if err := receipts . DeriveFields ( bc . chainConfig , b . Hash ( ) , b . NumberU64 ( ) , b . Time ( ) , b . BaseFee ( ) , blobGasPrice , b . Transactions ( ) ) ; err != nil {
2023-06-27 04:29:19 -04:00
log . Error ( "Failed to derive block receipts fields" , "hash" , b . Hash ( ) , "number" , b . NumberU64 ( ) , "err" , err )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
var logs [ ] * types . Log
for _ , receipt := range receipts {
for _ , log := range receipt . Logs {
if removed {
2023-06-15 07:52:06 -04:00
log . Removed = true
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
2023-06-15 07:52:06 -04:00
logs = append ( logs , log )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
}
return logs
}
2019-04-04 14:39:11 +03:00
// reorg takes two blocks, an old chain and a new chain and will reconstruct the
// blocks and inserts them to be part of the new canonical chain and accumulates
// potential missing transactions and post an event about them.
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// Note the new head block won't be processed here, callers need to handle it
// externally.
2023-03-02 08:29:15 +02:00
func ( bc * BlockChain ) reorg ( oldHead * types . Header , newHead * types . Block ) error {
2015-05-13 22:27:18 +02:00
var (
2016-12-04 19:07:24 +01:00
newChain types . Blocks
oldChain types . Blocks
commonBlock * types . Block
2019-04-04 14:39:11 +03:00
2022-05-30 08:42:06 +02:00
deletedTxs [ ] common . Hash
addedTxs [ ] common . Hash
2015-05-13 22:27:18 +02:00
)
2023-03-02 08:29:15 +02:00
oldBlock := bc . GetBlock ( oldHead . Hash ( ) , oldHead . Number . Uint64 ( ) )
if oldBlock == nil {
return errors . New ( "current head block missing" )
}
newBlock := newHead
2019-04-04 14:39:11 +03:00
// Reduce the longer chain to the same number as the shorter one
2015-05-15 15:30:34 +02:00
if oldBlock . NumberU64 ( ) > newBlock . NumberU64 ( ) {
2019-04-04 14:39:11 +03:00
// Old chain is longer, gather all transactions and logs as deleted ones
2017-05-11 09:55:48 +08:00
for ; oldBlock != nil && oldBlock . NumberU64 ( ) != newBlock . NumberU64 ( ) ; oldBlock = bc . GetBlock ( oldBlock . ParentHash ( ) , oldBlock . NumberU64 ( ) - 1 ) {
2016-03-07 18:11:52 +01:00
oldChain = append ( oldChain , oldBlock )
2022-05-30 08:42:06 +02:00
for _ , tx := range oldBlock . Transactions ( ) {
deletedTxs = append ( deletedTxs , tx . Hash ( ) )
}
2015-05-15 15:30:34 +02:00
}
} else {
2019-04-04 14:39:11 +03:00
// New chain is longer, stash all blocks away for subsequent insertion
2017-05-11 09:55:48 +08:00
for ; newBlock != nil && newBlock . NumberU64 ( ) != oldBlock . NumberU64 ( ) ; newBlock = bc . GetBlock ( newBlock . ParentHash ( ) , newBlock . NumberU64 ( ) - 1 ) {
2015-05-15 15:30:34 +02:00
newChain = append ( newChain , newBlock )
}
2015-04-29 12:43:24 +02:00
}
2015-05-28 18:18:23 +02:00
if oldBlock == nil {
2023-05-25 20:24:09 +08:00
return errInvalidOldChain
2015-05-28 18:18:23 +02:00
}
if newBlock == nil {
2023-05-25 20:24:09 +08:00
return errInvalidNewChain
2015-05-28 18:18:23 +02:00
}
2019-04-04 14:39:11 +03:00
// Both sides of the reorg are at the same number, reduce both until the common
// ancestor is found
2015-04-14 00:18:38 +02:00
for {
2019-04-04 14:39:11 +03:00
// If the common ancestor was found, bail out
2015-04-14 00:18:38 +02:00
if oldBlock . Hash ( ) == newBlock . Hash ( ) {
2015-05-13 22:27:18 +02:00
commonBlock = oldBlock
2015-04-14 00:18:38 +02:00
break
}
2019-04-04 14:39:11 +03:00
// Remove an old block as well as stash away a new block
2016-03-07 18:11:52 +01:00
oldChain = append ( oldChain , oldBlock )
2022-05-30 08:42:06 +02:00
for _ , tx := range oldBlock . Transactions ( ) {
deletedTxs = append ( deletedTxs , tx . Hash ( ) )
}
2019-04-04 14:39:11 +03:00
newChain = append ( newChain , newBlock )
// Step back with both chains
oldBlock = bc . GetBlock ( oldBlock . ParentHash ( ) , oldBlock . NumberU64 ( ) - 1 )
2015-05-28 15:35:50 +02:00
if oldBlock == nil {
2023-05-25 20:24:09 +08:00
return errInvalidOldChain
2015-05-28 15:35:50 +02:00
}
2019-04-04 14:39:11 +03:00
newBlock = bc . GetBlock ( newBlock . ParentHash ( ) , newBlock . NumberU64 ( ) - 1 )
2015-05-28 15:35:50 +02:00
if newBlock == nil {
2023-05-25 20:24:09 +08:00
return errInvalidNewChain
2015-05-28 15:35:50 +02:00
}
2015-04-14 00:18:38 +02:00
}
2022-05-30 08:42:06 +02:00
2017-02-22 14:10:07 +02:00
// Ensure the user sees large reorgs
2017-03-03 09:54:13 +02:00
if len ( oldChain ) > 0 && len ( newChain ) > 0 {
2019-08-22 18:28:23 +09:00
logFn := log . Info
msg := "Chain reorg detected"
2017-03-03 09:54:13 +02:00
if len ( oldChain ) > 63 {
2019-08-22 18:28:23 +09:00
msg = "Large chain reorg detected"
2017-03-03 09:54:13 +02:00
logFn = log . Warn
}
2019-08-22 18:28:23 +09:00
logFn ( msg , "number" , commonBlock . Number ( ) , "hash" , commonBlock . Hash ( ) ,
2017-03-03 09:54:13 +02:00
"drop" , len ( oldChain ) , "dropfrom" , oldChain [ 0 ] . Hash ( ) , "add" , len ( newChain ) , "addfrom" , newChain [ 0 ] . Hash ( ) )
2019-08-22 18:28:23 +09:00
blockReorgAddMeter . Mark ( int64 ( len ( newChain ) ) )
blockReorgDropMeter . Mark ( int64 ( len ( oldChain ) ) )
2020-08-20 09:49:35 +02:00
blockReorgMeter . Mark ( 1 )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
} else if len ( newChain ) > 0 {
// Special case happens in the post merge stage that current head is
// the ancestor of new head while these two blocks are not consecutive
2022-05-30 08:42:06 +02:00
log . Info ( "Extend chain" , "add" , len ( newChain ) , "number" , newChain [ 0 ] . Number ( ) , "hash" , newChain [ 0 ] . Hash ( ) )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
blockReorgAddMeter . Mark ( int64 ( len ( newChain ) ) )
2017-03-03 09:54:13 +02:00
} else {
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// len(newChain) == 0 && len(oldChain) > 0
// rewind the canonical chain to a lower point.
log . Error ( "Impossible reorg, please file an issue" , "oldnum" , oldBlock . Number ( ) , "oldhash" , oldBlock . Hash ( ) , "oldblocks" , len ( oldChain ) , "newnum" , newBlock . Number ( ) , "newhash" , newBlock . Hash ( ) , "newblocks" , len ( newChain ) )
2017-02-22 14:10:07 +02:00
}
2019-05-08 19:30:36 +08:00
// Insert the new chain(except the head block(reverse order)),
// taking care of the proper incremental order.
for i := len ( newChain ) - 1 ; i >= 1 ; i -- {
2019-04-04 14:39:11 +03:00
// Insert the block in the canonical way, re-writing history
2020-01-17 18:49:32 +08:00
bc . writeHeadBlock ( newChain [ i ] )
2019-04-04 14:39:11 +03:00
2020-01-17 18:49:32 +08:00
// Collect the new added transactions.
2022-05-30 08:42:06 +02:00
for _ , tx := range newChain [ i ] . Transactions ( ) {
addedTxs = append ( addedTxs , tx . Hash ( ) )
}
2015-08-17 14:01:41 +02:00
}
2022-05-30 08:42:06 +02:00
2020-01-17 18:49:32 +08:00
// Delete useless indexes right now which includes the non-canonical
// transaction indexes, canonical chain indexes which above the head.
indexesBatch := bc . db . NewBatch ( )
2022-05-30 08:42:06 +02:00
for _ , tx := range types . HashDifference ( deletedTxs , addedTxs ) {
rawdb . DeleteTxLookupEntry ( indexesBatch , tx )
2015-04-14 00:18:38 +02:00
}
2022-06-01 17:03:24 +08:00
// Delete all hash markers that are not part of the new canonical chain.
// Because the reorg function does not handle new chain head, all hash
// markers greater than or equal to new chain head should be deleted.
number := commonBlock . NumberU64 ( )
if len ( newChain ) > 1 {
number = newChain [ 1 ] . NumberU64 ( )
}
2019-05-07 14:26:00 +02:00
for i := number + 1 ; ; i ++ {
hash := rawdb . ReadCanonicalHash ( bc . db , i )
if hash == ( common . Hash { } ) {
break
}
2020-01-17 18:49:32 +08:00
rawdb . DeleteCanonicalHash ( indexesBatch , i )
}
if err := indexesBatch . Write ( ) ; err != nil {
log . Crit ( "Failed to delete useless indexes" , "err" , err )
2019-05-07 14:26:00 +02:00
}
2022-05-30 08:42:06 +02:00
2022-09-09 15:25:55 +02:00
// Send out events for logs from the old canon chain, and 'reborn'
// logs from the new canon chain. The number of logs can be very
// high, so the events are sent in batches of size around 512.
// Deleted logs + blocks:
var deletedLogs [ ] * types . Log
for i := len ( oldChain ) - 1 ; i >= 0 ; i -- {
// Also send event for blocks removed from the canon chain.
bc . chainSideFeed . Send ( ChainSideEvent { Block : oldChain [ i ] } )
// Collect deleted logs for notification
2022-12-09 16:14:33 +01:00
if logs := bc . collectLogs ( oldChain [ i ] , true ) ; len ( logs ) > 0 {
2022-09-09 15:25:55 +02:00
deletedLogs = append ( deletedLogs , logs ... )
}
if len ( deletedLogs ) > 512 {
bc . rmLogsFeed . Send ( RemovedLogsEvent { deletedLogs } )
deletedLogs = nil
2022-05-30 08:42:06 +02:00
}
}
2019-11-29 21:22:08 +08:00
if len ( deletedLogs ) > 0 {
2022-09-09 15:25:55 +02:00
bc . rmLogsFeed . Send ( RemovedLogsEvent { deletedLogs } )
2019-11-29 21:22:08 +08:00
}
2022-09-09 15:25:55 +02:00
// New logs:
var rebirthLogs [ ] * types . Log
for i := len ( newChain ) - 1 ; i >= 1 ; i -- {
2022-12-09 16:14:33 +01:00
if logs := bc . collectLogs ( newChain [ i ] , false ) ; len ( logs ) > 0 {
2022-09-09 15:25:55 +02:00
rebirthLogs = append ( rebirthLogs , logs ... )
2015-10-12 15:04:38 +03:00
}
2022-09-09 15:25:55 +02:00
if len ( rebirthLogs ) > 512 {
bc . logsFeed . Send ( rebirthLogs )
rebirthLogs = nil
2015-10-12 15:04:38 +03:00
}
}
2022-09-09 15:25:55 +02:00
if len ( rebirthLogs ) > 0 {
bc . logsFeed . Send ( rebirthLogs )
2015-10-12 15:04:38 +03:00
}
2019-11-29 21:22:08 +08:00
return nil
2015-10-12 15:04:38 +03:00
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// InsertBlockWithoutSetHead executes the block, runs the necessary verification
// upon it and then persist the block and the associate state into the database.
// The key difference between the InsertChain is it won't do the canonical chain
2022-05-05 15:36:26 +08:00
// updating. It relies on the additional SetCanonical call to finalize the entire
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// procedure.
func ( bc * BlockChain ) InsertBlockWithoutSetHead ( block * types . Block ) error {
if ! bc . chainmu . TryLock ( ) {
return errChainStopped
}
defer bc . chainmu . Unlock ( )
2023-05-03 12:58:39 +03:00
_ , err := bc . insertChain ( types . Blocks { block } , false )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
return err
}
2022-05-05 15:36:26 +08:00
// SetCanonical rewinds the chain to set the new head block as the specified
// block. It's possible that the state of the new head is missing, and it will
// be recovered in this function as well.
2022-05-17 11:32:55 +02:00
func ( bc * BlockChain ) SetCanonical ( head * types . Block ) ( common . Hash , error ) {
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if ! bc . chainmu . TryLock ( ) {
2022-05-17 11:32:55 +02:00
return common . Hash { } , errChainStopped
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
defer bc . chainmu . Unlock ( )
2021-10-07 15:47:50 +02:00
2022-05-05 15:36:26 +08:00
// Re-execute the reorged chain in case the head state is missing.
if ! bc . HasState ( head . Root ( ) ) {
2022-05-17 11:32:55 +02:00
if latestValidHash , err := bc . recoverAncestors ( head ) ; err != nil {
return latestValidHash , err
2022-05-05 15:36:26 +08:00
}
log . Info ( "Recovered head state" , "number" , head . Number ( ) , "hash" , head . Hash ( ) )
}
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// Run the reorg if necessary and set the given block as new head.
2022-03-11 14:14:45 +02:00
start := time . Now ( )
if head . ParentHash ( ) != bc . CurrentBlock ( ) . Hash ( ) {
if err := bc . reorg ( bc . CurrentBlock ( ) , head ) ; err != nil {
2022-05-17 11:32:55 +02:00
return common . Hash { } , err
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
}
2022-03-11 14:14:45 +02:00
bc . writeHeadBlock ( head )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
// Emit events
2022-12-09 16:14:33 +01:00
logs := bc . collectLogs ( head , false )
2022-03-11 14:14:45 +02:00
bc . chainFeed . Send ( ChainEvent { Block : head , Hash : head . Hash ( ) , Logs : logs } )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
if len ( logs ) > 0 {
bc . logsFeed . Send ( logs )
}
2022-03-11 14:14:45 +02:00
bc . chainHeadFeed . Send ( ChainHeadEvent { Block : head } )
context := [ ] interface { } {
"number" , head . Number ( ) ,
"hash" , head . Hash ( ) ,
"root" , head . Root ( ) ,
"elapsed" , time . Since ( start ) ,
}
if timestamp := time . Unix ( int64 ( head . Time ( ) ) , 0 ) ; time . Since ( timestamp ) > time . Minute {
context = append ( context , [ ] interface { } { "age" , common . PrettyAge ( timestamp ) } ... )
}
log . Info ( "Chain head was updated" , context ... )
2022-05-17 11:32:55 +02:00
return head . Hash ( ) , nil
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
}
func ( bc * BlockChain ) updateFutureBlocks ( ) {
2018-01-02 12:50:46 +01:00
futureTimer := time . NewTicker ( 5 * time . Second )
defer futureTimer . Stop ( )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
defer bc . wg . Done ( )
2015-03-06 15:50:44 +01:00
for {
select {
2018-01-02 12:50:46 +01:00
case <- futureTimer . C :
2017-05-11 09:55:48 +08:00
bc . procFutureBlocks ( )
case <- bc . quit :
2015-10-12 15:04:38 +03:00
return
2015-03-06 15:50:44 +01:00
}
}
}
2015-05-17 01:42:30 +02:00
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) rewindInvalidHeaderBlockLoop ( ) {
recheck := time . NewTicker ( rewindBadBlockInterval )
defer func ( ) {
recheck . Stop ( )
bc . wg . Done ( )
} ( )
for {
select {
case <- recheck . C :
bc . tryRewindBadBlocks ( )
case <- bc . quit :
return
}
}
}
func ( bc * BlockChain ) trustedDiffLayerLoop ( ) {
recheck := time . NewTicker ( diffLayerFreezerRecheckInterval )
defer func ( ) {
recheck . Stop ( )
bc . wg . Done ( )
} ( )
for {
select {
case diff := <- bc . diffQueueBuffer :
bc . diffQueue . Push ( diff , - ( int64 ( diff . Number ) ) )
case <- bc . quit :
// Persist all diffLayers when shutdown, it will introduce redundant storage, but it is acceptable.
// If the client been ungracefully shutdown, it will missing all cached diff layers, it is acceptable as well.
var batch ethdb . Batch
for ! bc . diffQueue . Empty ( ) {
2023-08-23 17:46:08 +08:00
diffLayer , _ := bc . diffQueue . Pop ( )
2022-07-05 11:14:21 +08:00
if batch == nil {
batch = bc . db . DiffStore ( ) . NewBatch ( )
}
rawdb . WriteDiffLayer ( batch , diffLayer . BlockHash , diffLayer )
if batch . ValueSize ( ) > ethdb . IdealBatchSize {
if err := batch . Write ( ) ; err != nil {
log . Error ( "Failed to write diff layer" , "err" , err )
return
}
batch . Reset ( )
}
}
if batch != nil {
// flush data
if err := batch . Write ( ) ; err != nil {
log . Error ( "Failed to write diff layer" , "err" , err )
return
}
batch . Reset ( )
}
return
case <- recheck . C :
2023-08-23 17:46:08 +08:00
currentHeight := bc . CurrentBlock ( ) . Number . Uint64 ( )
2022-07-05 11:14:21 +08:00
var batch ethdb . Batch
for ! bc . diffQueue . Empty ( ) {
2023-08-23 17:46:08 +08:00
diffLayer , prio := bc . diffQueue . Pop ( )
2022-07-05 11:14:21 +08:00
// if the block not old enough
if int64 ( currentHeight ) + prio < int64 ( bc . triesInMemory ) {
bc . diffQueue . Push ( diffLayer , prio )
break
}
canonicalHash := bc . GetCanonicalHash ( uint64 ( - prio ) )
// on the canonical chain
if canonicalHash == diffLayer . BlockHash {
if batch == nil {
batch = bc . db . DiffStore ( ) . NewBatch ( )
}
rawdb . WriteDiffLayer ( batch , diffLayer . BlockHash , diffLayer )
staleHash := bc . GetCanonicalHash ( uint64 ( - prio ) - bc . diffLayerFreezerBlockLimit )
rawdb . DeleteDiffLayer ( batch , staleHash )
}
if batch != nil && batch . ValueSize ( ) > ethdb . IdealBatchSize {
if err := batch . Write ( ) ; err != nil {
panic ( fmt . Sprintf ( "Failed to write diff layer, error %v" , err ) )
}
batch . Reset ( )
}
}
if batch != nil {
if err := batch . Write ( ) ; err != nil {
panic ( fmt . Sprintf ( "Failed to write diff layer, error %v" , err ) )
}
batch . Reset ( )
}
}
}
}
2023-02-03 15:53:36 +08:00
func ( bc * BlockChain ) startDoubleSignMonitor ( ) {
eventChan := make ( chan ChainHeadEvent , monitor . MaxCacheHeader )
sub := bc . SubscribeChainHeadEvent ( eventChan )
defer func ( ) {
sub . Unsubscribe ( )
close ( eventChan )
bc . wg . Done ( )
} ( )
for {
select {
case event := <- eventChan :
if bc . doubleSignMonitor != nil {
bc . doubleSignMonitor . Verify ( event . Block . Header ( ) )
}
case <- bc . quit :
return
}
}
}
2021-11-01 21:09:36 +08:00
// skipBlock returns 'true', if the block being imported can be skipped over, meaning
// that the block does not need to be processed but can be considered already fully 'done'.
func ( bc * BlockChain ) skipBlock ( err error , it * insertIterator ) bool {
// We can only ever bypass processing if the only error returned by the validator
// is ErrKnownBlock, which means all checks passed, but we already have the block
// and state.
if ! errors . Is ( err , ErrKnownBlock ) {
return false
}
// If we're not using snapshots, we can skip this, since we have both block
// and (trie-) state
if bc . snaps == nil {
return true
}
var (
header = it . current ( ) // header can't be nil
parentRoot common . Hash
)
// If we also have the snapshot-state, we can skip the processing.
if bc . snaps . Snapshot ( header . Root ) != nil {
return true
}
// In this case, we have the trie-state but not snapshot-state. If the parent
// snapshot-state exists, we need to process this in order to not get a gap
// in the snapshot layers.
// Resolve parent block
if parent := it . previous ( ) ; parent != nil {
parentRoot = parent . Root
} else if parent = bc . GetHeaderByHash ( header . ParentHash ) ; parent != nil {
parentRoot = parent . Root
}
if parentRoot == ( common . Hash { } ) {
return false // Theoretically impossible case
}
// Parent is also missing snapshot: we can skip this. Otherwise process.
if bc . snaps . Snapshot ( parentRoot ) == nil {
return true
}
return false
}
2022-09-29 15:50:24 +08:00
// indexBlocks reindexes or unindexes transactions depending on user configuration
func ( bc * BlockChain ) indexBlocks ( tail * uint64 , head uint64 , done chan struct { } ) {
defer func ( ) { close ( done ) } ( )
// The tail flag is not existent, it means the node is just initialized
// and all blocks(may from ancient store) are not indexed yet.
if tail == nil {
from := uint64 ( 0 )
if bc . txLookupLimit != 0 && head >= bc . txLookupLimit {
from = head - bc . txLookupLimit + 1
}
rawdb . IndexTransactions ( bc . db , from , head + 1 , bc . quit )
return
}
// The tail flag is existent, but the whole chain is required to be indexed.
if bc . txLookupLimit == 0 || head < bc . txLookupLimit {
if * tail > 0 {
// It can happen when chain is rewound to a historical point which
// is even lower than the indexes tail, recap the indexing target
// to new head to avoid reading non-existent block bodies.
end := * tail
if end > head + 1 {
end = head + 1
}
rawdb . IndexTransactions ( bc . db , 0 , end , bc . quit )
}
return
}
// Update the transaction index to the new chain state
if head - bc . txLookupLimit + 1 < * tail {
// Reindex a part of missing indices and rewind index tail to HEAD-limit
rawdb . IndexTransactions ( bc . db , head - bc . txLookupLimit + 1 , * tail , bc . quit )
} else {
// Unindex a part of stale indices and forward index tail to HEAD-limit
rawdb . UnindexTransactions ( bc . db , * tail , head - bc . txLookupLimit + 1 , bc . quit )
}
}
2020-05-11 17:58:43 +02:00
// maintainTxIndex is responsible for the construction and deletion of the
// transaction index.
//
// User can use flag `txlookuplimit` to specify a "recentness" block, below
// which ancient tx indices get deleted. If `txlookuplimit` is 0, it means
// all tx indices will be reserved.
//
2022-09-29 15:50:24 +08:00
// The user can adjust the txlookuplimit value for each launch after sync,
// Geth will automatically construct the missing indices or delete the extra
// indices.
func ( bc * BlockChain ) maintainTxIndex ( ) {
2020-10-20 22:34:50 +08:00
defer bc . wg . Done ( )
2022-09-29 15:50:24 +08:00
// Listening to chain events and manipulate the transaction indexes.
2020-05-11 17:58:43 +02:00
var (
done chan struct { } // Non-nil if background unindexing or reindexing routine is active.
headCh = make ( chan ChainHeadEvent , 1 ) // Buffered to avoid locking up the event feed
)
sub := bc . SubscribeChainHeadEvent ( headCh )
if sub == nil {
return
}
defer sub . Unsubscribe ( )
all: activate pbss as experimental feature from eth (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-11 03:21:36 +08:00
log . Info ( "Initialized transaction indexer" , "limit" , bc . TxLookupLimit ( ) )
2017-02-13 21:44:06 +01:00
2020-05-11 17:58:43 +02:00
for {
select {
case head := <- headCh :
if done == nil {
done = make ( chan struct { } )
2022-09-29 15:50:24 +08:00
go bc . indexBlocks ( rawdb . ReadTxIndexTail ( bc . db ) , head . Block . NumberU64 ( ) , done )
2020-05-11 17:58:43 +02:00
}
case <- done :
done = nil
case <- bc . quit :
2020-10-20 22:34:50 +08:00
if done != nil {
log . Info ( "Waiting background transaction indexer to exit" )
<- done
}
2020-05-11 17:58:43 +02:00
return
}
}
2017-02-13 21:44:06 +01:00
}
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) isCachedBadBlock ( block * types . Block ) bool {
if timeAt , exist := bc . badBlockCache . Get ( block . Hash ( ) ) ; exist {
2023-08-23 17:46:08 +08:00
if time . Since ( timeAt ) >= badBlockCacheExpire {
2022-07-05 11:14:21 +08:00
bc . badBlockCache . Remove ( block . Hash ( ) )
return false
}
return true
}
return false
}
2016-05-24 18:49:54 +02:00
// reportBlock logs a bad block error.
2016-11-23 13:32:25 +01:00
func ( bc * BlockChain ) reportBlock ( block * types . Block , receipts types . Receipts , err error ) {
2021-01-10 19:54:15 +08:00
rawdb . WriteBadBlock ( bc . db , block )
2022-09-17 21:27:10 +02:00
log . Error ( summarizeBadBlock ( block , receipts , bc . Config ( ) , err ) )
2015-05-29 18:07:23 +02:00
}
2017-02-28 13:35:17 +02:00
2022-09-19 10:04:16 +02:00
// summarizeBadBlock returns a string summarizing the bad block and other
// relevant information.
func summarizeBadBlock ( block * types . Block , receipts [ ] * types . Receipt , config * params . ChainConfig , err error ) string {
2017-02-28 13:35:17 +02:00
var receiptString string
2018-11-22 10:00:16 +01:00
for i , receipt := range receipts {
2022-09-19 10:04:16 +02:00
receiptString += fmt . Sprintf ( "\n %d: cumulative: %v gas: %v contract: %v status: %v tx: %v logs: %v bloom: %x state: %x" ,
2018-11-22 10:00:16 +01:00
i , receipt . CumulativeGasUsed , receipt . GasUsed , receipt . ContractAddress . Hex ( ) ,
receipt . Status , receipt . TxHash . Hex ( ) , receipt . Logs , receipt . Bloom , receipt . PostState )
2017-02-28 13:35:17 +02:00
}
2022-09-19 10:04:16 +02:00
version , vcs := version . Info ( )
platform := fmt . Sprintf ( "%s %s %s %s" , version , runtime . Version ( ) , runtime . GOARCH , runtime . GOOS )
if vcs != "" {
vcs = fmt . Sprintf ( "\nVCS: %s" , vcs )
}
2023-12-25 11:07:31 +08:00
if badBlockRecords . Cardinality ( ) < badBlockRecordslimit {
badBlockRecords . Add ( block . Hash ( ) )
badBlockGauge . Update ( int64 ( badBlockRecords . Cardinality ( ) ) )
}
2022-09-19 10:04:16 +02:00
return fmt . Sprintf ( `
2016-11-23 13:32:25 +01:00
# # # # # # # # # # BAD BLOCK # # # # # # # # #
2022-09-19 10:04:16 +02:00
Block : % v ( % # x )
2023-12-25 11:07:31 +08:00
Miner : % v
2016-11-23 13:32:25 +01:00
Error : % v
2022-09-19 10:04:16 +02:00
Platform : % v % v
Chain config : % # v
Receipts : % v
2016-11-23 13:32:25 +01:00
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
2023-12-25 11:07:31 +08:00
` , block . Number ( ) , block . Hash ( ) , block . Coinbase ( ) , err , platform , vcs , config , receiptString )
2015-05-29 18:07:23 +02:00
}
2015-12-16 04:26:23 +01:00
// InsertHeaderChain attempts to insert the given header chain in to the local
// chain, possibly creating a reorg. If an error is returned, it will return the
// index number of the failing header as well an error describing what went wrong.
2023-05-03 12:58:39 +03:00
func ( bc * BlockChain ) InsertHeaderChain ( chain [ ] * types . Header ) ( int , error ) {
2022-03-23 01:58:05 +08:00
if len ( chain ) == 0 {
return 0 , nil
}
2017-03-22 20:44:22 +01:00
start := time . Now ( )
2023-05-03 12:58:39 +03:00
if i , err := bc . hc . ValidateHeaderChain ( chain ) ; err != nil {
2017-03-22 20:44:22 +01:00
return i , err
}
2021-10-07 15:47:50 +02:00
if ! bc . chainmu . TryLock ( ) {
return 0 , errChainStopped
}
2017-05-11 09:55:48 +08:00
defer bc . chainmu . Unlock ( )
all: core rework for the merge transition (#23761)
* all: work for eth1/2 transtition
* consensus/beacon, eth: change beacon difficulty to 0
* eth: updates
* all: add terminalBlockDifficulty config, fix rebasing issues
* eth: implemented merge interop spec
* internal/ethapi: update to v1.0.0.alpha.2
This commit updates the code to the new spec, moving payloadId into
it's own object. It also fixes an issue with finalizing an empty blockhash.
It also properly sets the basefee
* all: sync polishes, other fixes + refactors
* core, eth: correct semantics for LeavePoW, EnterPoS
* core: fixed rebasing artifacts
* core: light: performance improvements
* core: use keyed field (f)
* core: eth: fix compilation issues + tests
* eth/catalyst: dbetter error codes
* all: move Merger to consensus/, remove reliance on it in bc
* all: renamed EnterPoS and LeavePoW to ReachTDD and FinalizePoS
* core: make mergelogs a function
* core: use InsertChain instead of InsertBlock
* les: drop merger from lightchain object
* consensus: add merger
* core: recoverAncestors in catalyst mode
* core: fix nitpick
* all: removed merger from beacon, use TTD, nitpicks
* consensus: eth: add docstring, removed unnecessary code duplication
* consensus/beacon: better comment
* all: easy to fix nitpicks by karalabe
* consensus/beacon: verify known headers to be sure
* core: comments
* core: eth: don't drop peers who advertise blocks, nitpicks
* core: never add beacon blocks to the future queue
* core: fixed nitpicks
* consensus/beacon: simplify IsTTDReached check
* consensus/beacon: correct IsTTDReached check
Co-authored-by: rjl493456442 <garyrong0905@gmail.com>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2021-11-26 12:23:02 +01:00
_ , err := bc . hc . InsertHeaderChain ( chain , start , bc . forker )
2020-12-09 11:13:02 +01:00
return 0 , err
2015-12-16 04:26:23 +01:00
}
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) TriesInMemory ( ) uint64 { return bc . triesInMemory }
func EnablePipelineCommit ( bc * BlockChain ) ( * BlockChain , error ) {
2023-09-07 16:39:29 +08:00
bc . pipeCommit = false
2022-07-05 11:14:21 +08:00
return bc , nil
}
func EnablePersistDiff ( limit uint64 ) BlockChainOption {
return func ( chain * BlockChain ) ( * BlockChain , error ) {
chain . diffLayerFreezerBlockLimit = limit
return chain , nil
}
}
func EnableBlockValidator ( chainConfig * params . ChainConfig , engine consensus . Engine , mode VerifyMode , peers verifyPeers ) BlockChainOption {
return func ( bc * BlockChain ) ( * BlockChain , error ) {
if mode . NeedRemoteVerify ( ) {
vm , err := NewVerifyManager ( bc , peers , mode == InsecureVerify )
if err != nil {
return nil , err
}
go vm . mainLoop ( )
bc . validator = NewBlockValidator ( chainConfig , bc , engine , EnableRemoteVerifyManager ( vm ) )
}
return bc , nil
}
}
2023-02-03 15:53:36 +08:00
func EnableDoubleSignChecker ( bc * BlockChain ) ( * BlockChain , error ) {
bc . doubleSignMonitor = monitor . NewDoubleSignMonitor ( )
return bc , nil
}
2022-07-05 11:14:21 +08:00
func ( bc * BlockChain ) GetVerifyResult ( blockNumber uint64 , blockHash common . Hash , diffHash common . Hash ) * VerifyResult {
var res VerifyResult
res . BlockNumber = blockNumber
res . BlockHash = blockHash
if blockNumber > bc . CurrentHeader ( ) . Number . Uint64 ( ) + maxDiffForkDist {
res . Status = types . StatusBlockTooNew
return & res
} else if blockNumber > bc . CurrentHeader ( ) . Number . Uint64 ( ) {
res . Status = types . StatusBlockNewer
return & res
}
header := bc . GetHeaderByHash ( blockHash )
if header == nil {
if blockNumber > bc . CurrentHeader ( ) . Number . Uint64 ( ) - maxDiffForkDist {
res . Status = types . StatusPossibleFork
return & res
}
res . Status = types . StatusImpossibleFork
return & res
}
diff := bc . GetTrustedDiffLayer ( blockHash )
if diff != nil {
if diff . DiffHash . Load ( ) == nil {
hash , err := CalculateDiffHash ( diff )
if err != nil {
res . Status = types . StatusUnexpectedError
return & res
}
diff . DiffHash . Store ( hash )
}
if diffHash != diff . DiffHash . Load ( ) . ( common . Hash ) {
res . Status = types . StatusDiffHashMismatch
return & res
}
res . Status = types . StatusFullVerified
res . Root = header . Root
return & res
}
res . Status = types . StatusPartiallyVerified
res . Root = header . Root
return & res
}
func ( bc * BlockChain ) GetTrustedDiffLayer ( blockHash common . Hash ) * types . DiffLayer {
var diff * types . DiffLayer
if cached , ok := bc . diffLayerCache . Get ( blockHash ) ; ok {
diff = cached . ( * types . DiffLayer )
return diff
}
diffStore := bc . db . DiffStore ( )
if diffStore != nil {
diff = rawdb . ReadDiffLayer ( diffStore , blockHash )
}
return diff
}
func CalculateDiffHash ( d * types . DiffLayer ) ( common . Hash , error ) {
if d == nil {
return common . Hash { } , fmt . Errorf ( "nil diff layer" )
}
diff := & types . ExtDiffLayer {
BlockHash : d . BlockHash ,
Receipts : make ( [ ] * types . ReceiptForStorage , 0 ) ,
Number : d . Number ,
Codes : d . Codes ,
Destructs : d . Destructs ,
Accounts : d . Accounts ,
Storages : d . Storages ,
}
for index , account := range diff . Accounts {
2023-08-23 17:46:08 +08:00
full , err := types . FullAccount ( account . Blob )
2022-07-05 11:14:21 +08:00
if err != nil {
return common . Hash { } , fmt . Errorf ( "decode full account error: %v" , err )
}
// set account root to empty root
2023-08-23 17:46:08 +08:00
full . Root = types . EmptyRootHash
diff . Accounts [ index ] . Blob = types . SlimAccountRLP ( * full )
2022-07-05 11:14:21 +08:00
}
rawData , err := rlp . EncodeToBytes ( diff )
if err != nil {
return common . Hash { } , fmt . Errorf ( "encode new diff error: %v" , err )
}
hasher := sha3 . NewLegacyKeccak256 ( )
_ , err = hasher . Write ( rawData )
if err != nil {
return common . Hash { } , fmt . Errorf ( "hasher write error: %v" , err )
}
var hash common . Hash
hasher . Sum ( hash [ : 0 ] )
return hash , nil
}
2023-08-23 17:46:08 +08:00
2022-07-05 09:02:49 +02:00
// SetBlockValidatorAndProcessorForTesting sets the current validator and processor.
2022-07-04 11:24:06 +02:00
// This method can be used to force an invalid blockchain to be verified for tests.
// This method is unsafe and should only be used before block import starts.
2022-07-05 09:02:49 +02:00
func ( bc * BlockChain ) SetBlockValidatorAndProcessorForTesting ( v Validator , p Processor ) {
2022-07-04 11:24:06 +02:00
bc . validator = v
2022-07-05 09:02:49 +02:00
bc . processor = p
2022-07-04 11:24:06 +02:00
}
core,eth: add `debug_setTrieFlushInterval` to change trie flush frequency (#24785)
This PR makes it possible to modify the flush interval time via RPC. On one extreme, `0s`, it would act as an archive node. If set to `1h`, means that after one hour of effective block processing time, the trie would be flushed. If one block takes 200ms, this means that a flush would occur every `5*3600=18000` blocks -- however, if the memory size of the cached states grows too large, it will flush sooner.
Essentially, this makes it possible to configure the node to be more or less "archive:ish", and without restarting the node while reconfiguring it.
2022-12-09 13:40:17 +01:00
// SetTrieFlushInterval configures how often in-memory tries are persisted to disk.
// The interval is in terms of block processing time, not wall clock.
// It is thread-safe and can be called repeatedly without side effects.
func ( bc * BlockChain ) SetTrieFlushInterval ( interval time . Duration ) {
2023-03-30 18:53:32 +08:00
bc . flushInterval . Store ( int64 ( interval ) )
core,eth: add `debug_setTrieFlushInterval` to change trie flush frequency (#24785)
This PR makes it possible to modify the flush interval time via RPC. On one extreme, `0s`, it would act as an archive node. If set to `1h`, means that after one hour of effective block processing time, the trie would be flushed. If one block takes 200ms, this means that a flush would occur every `5*3600=18000` blocks -- however, if the memory size of the cached states grows too large, it will flush sooner.
Essentially, this makes it possible to configure the node to be more or less "archive:ish", and without restarting the node while reconfiguring it.
2022-12-09 13:40:17 +01:00
}
2023-06-06 20:41:44 +08:00
// GetTrieFlushInterval gets the in-memroy tries flush interval
func ( bc * BlockChain ) GetTrieFlushInterval ( ) time . Duration {
return time . Duration ( bc . flushInterval . Load ( ) )
}