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"
2024-02-13 21:49:53 +08:00
"github.com/ethereum/go-ethereum/triedb"
"github.com/ethereum/go-ethereum/triedb/hashdb"
"github.com/ethereum/go-ethereum/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 )
2024-04-30 17:58:04 +08:00
blockInsertMgaspsGauge = metrics . NewRegisteredGauge ( "chain/insert/mgasps" , nil )
2023-08-31 19:37:17 +02:00
chainInfoGauge = metrics . NewRegisteredGaugeInfo ( "chain/info" , 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
2024-03-22 22:37:47 +08:00
sidecarsCacheLimit = 1024
2023-09-07 16:39:29 +08:00
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.
2024-04-15 10:47:54 +08:00
JournalFilePath string
2024-04-19 16:23:28 +08:00
JournalFile bool
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.
2024-02-13 21:49:53 +08:00
func ( c * CacheConfig ) triedbConfig ( ) * triedb . Config {
2024-03-08 15:36:25 +08:00
config := & triedb . Config {
2023-09-24 12:21:00 +08:00
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 {
2024-04-15 10:47:54 +08:00
SyncFlush : c . PathSyncFlush ,
StateHistory : c . StateHistory ,
CleanCacheSize : c . TrieCleanLimit * 1024 * 1024 ,
DirtyCacheSize : c . TrieDirtyLimit * 1024 * 1024 ,
JournalFilePath : c . JournalFilePath ,
2024-04-19 16:23:28 +08:00
JournalFile : c . JournalFile ,
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
}
}
2024-08-09 17:29:47 +08:00
// TODO:: support other versa db config items, currently use the default config
if c . StateScheme == rawdb . VersionScheme {
config . IsVersion = 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
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 )
2024-01-23 04:05:18 +08:00
// txLookup is wrapper over transaction lookup along with the corresponding
// transaction object.
2024-02-02 15:43:33 +08:00
2024-01-23 04:05:18 +08:00
type txLookup struct {
lookup * rawdb . LegacyTxLookupEntry
transaction * types . Transaction
}
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
2024-02-13 21:49:53 +08:00
triedb * triedb . Database // The database handler for maintaining trie nodes.
2023-02-09 13:03:54 +02:00
stateCache state . Database // State database to reuse between imports (contains state cache)
2022-07-05 11:14:21 +08:00
triesInMemory uint64
2024-02-02 15:43:33 +08:00
txIndexer * txIndexer // Transaction indexer, might be nil if not enabled
2022-07-05 11:14:21 +08:00
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
2024-04-18 15:12:05 +08:00
currentFinalBlock atomic . Pointer [ types . Header ] // Latest (consensus) finalized block
2024-03-22 22:37:47 +08:00
chasingHead atomic . Pointer [ types . Header ]
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 ]
2024-01-23 04:05:18 +08:00
txLookupCache * lru . Cache [ common . Hash , txLookup ]
2024-03-22 22:37:47 +08:00
sidecarsCache * lru . Cache [ common . Hash , types . BlobSidecars ]
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
2024-01-23 04:05:18 +08:00
wg sync . WaitGroup
2024-06-18 15:20:23 +08:00
dbWg sync . WaitGroup
2024-01-23 04:05:18 +08:00
quit chan struct { } // shutdown signal, closed in Stop.
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
}
2024-03-21 11:37:47 +08:00
if cacheConfig . StateScheme == rawdb . HashScheme && cacheConfig . TriesInMemory != 128 {
log . Warn ( "TriesInMemory isn't the default value (128), you need specify the same TriesInMemory when pruning data" ,
"triesInMemory" , cacheConfig . TriesInMemory , "scheme" , cacheConfig . StateScheme )
2022-07-05 11:14:21 +08:00
}
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
2024-02-13 21:49:53 +08:00
triedb := triedb . NewDatabase ( db , cacheConfig . triedbConfig ( ) )
all: activate pbss as experimental feature (#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
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 ) ,
2024-03-22 22:37:47 +08:00
sidecarsCache : lru . NewCache [ common . Hash , types . BlobSidecars ] ( sidecarsCacheLimit ) ,
2023-09-07 16:39:29 +08:00
blockCache : lru . NewCache [ common . Hash , * types . Block ] ( blockCacheLimit ) ,
2024-02-02 15:43:33 +08:00
txLookupCache : lru . NewCache [ common . Hash , txLookup ] ( txLookupCacheLimit ) ,
2023-09-07 16:39:29 +08:00
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 )
2024-08-09 17:29:47 +08:00
bc . stateCache = state . NewDatabaseWithNodeDB ( bc . db , bc . triedb , true )
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 )
2024-03-22 22:37:47 +08:00
bc . chasingHead . Store ( nil )
2020-05-11 17:58:43 +02:00
2023-08-31 19:37:17 +02:00
// Update chain info data metrics
chainInfoGauge . Update ( metrics . GaugeInfoValue { "chain_id" : bc . chainConfig . ChainID . String ( ) } )
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
}
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 15:00:53 +08:00
// Make sure the state associated with the block is available, or log out
// if there is no available state, waiting for state sync.
2020-08-20 13:01:24 +03:00
head := bc . CurrentBlock ( )
2024-04-18 13:45:02 +08:00
if ! bc . HasState ( head . Root ) {
2023-09-17 22:35:09 +08:00
if head . Number . Uint64 ( ) == 0 {
// The genesis state is missing, which is only possible in the path-based
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 15:00:53 +08:00
// scheme. This situation occurs when the initial state sync is not finished
2023-10-23 09:23:41 -04:00
// yet, or the chain head is rewound below the pivot point. In both scenarios,
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 15:00:53 +08:00
// there is no possible recovery approach except for rerunning a snap sync.
// Do nothing here until the state syncer picks it up.
log . Info ( "Genesis state is missing, wait state sync" )
2020-10-30 03:01:58 +08:00
} else {
2023-09-17 22:35:09 +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 )
}
2024-04-18 13:45:02 +08:00
if bc . triedb . Scheme ( ) == rawdb . PathScheme && ! bc . NoTries ( ) {
2024-02-02 15:43:33 +08:00
recoverable , _ := bc . triedb . Recoverable ( diskRoot )
if ! bc . HasState ( diskRoot ) && ! recoverable {
diskRoot = bc . triedb . Head ( )
}
2023-11-06 14:11:17 +08:00
}
2023-09-17 22:35:09 +08:00
if diskRoot != ( common . Hash { } ) {
2024-02-02 15:43:33 +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-09-17 22:35:09 +08:00
snapDisk , err := bc . setHeadBeyondRoot ( head . Number . Uint64 ( ) , 0 , diskRoot , true )
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 {
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 {
return nil , err
}
2020-10-30 03:01:58 +08:00
}
2020-08-20 13:01:24 +03:00
}
}
// Ensure that a previous crash in SetHead doesn't leave extra ancients
2024-07-16 22:37:03 +08:00
if frozen , err := bc . db . BlockStore ( ) . ItemAmountInAncient ( ) ; err == nil && frozen > 0 {
frozen , err = bc . db . BlockStore ( ) . Ancients ( )
2022-07-05 11:14:21 +08:00
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 ,
}
2024-02-02 15:43:33 +08:00
bc . snaps , _ = snapshot . New ( snapconfig , bc . db , bc . triedb , head . Root , int ( bc . cacheConfig . TriesInMemory ) , bc . 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 )
}
2024-01-24 04:00:50 +08:00
// Start tx indexer if it's enabled.
2022-09-29 15:50:24 +08:00
if txLookupLimit != nil {
2024-01-24 04:00:50 +08:00
bc . txIndexer = newTxIndexer ( * txLookupLimit , bc )
2022-09-29 15:50:24 +08:00
}
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
}
2024-02-02 15:43:33 +08:00
func ( bc * BlockChain ) NoTries ( ) bool {
return bc . stateCache . NoTries ( )
}
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
}
2024-02-21 19:19:20 +08:00
blockBaseFee := block . BaseFee ( )
if blockBaseFee == nil {
blockBaseFee = big . NewInt ( 0 )
}
2023-11-14 16:40:25 +08:00
for i , receipt := range receipts {
2024-02-21 19:19:20 +08:00
receipt . EffectiveGasPrice = big . NewInt ( 0 ) . Add ( blockBaseFee , txs [ i ] . EffectiveGasTipValue ( blockBaseFee ) )
if receipt . Logs == nil {
receipt . Logs = [ ] * types . Log { }
}
2023-11-14 16:40:25 +08:00
}
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
}
}
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 ( )
2024-07-16 22:37:03 +08:00
for _ , hash := range [ ] common . Hash { rawdb . ReadHeadBlockHash ( bc . db ) , rawdb . ReadHeadHeaderHash ( bc . db ) , rawdb . ReadHeadFastBlockHash ( 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
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
2024-07-16 22:37:03 +08: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 ( )
2024-07-16 22:37:03 +08: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
2024-07-16 22:37:03 +08: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 {
2023-09-28 18:15:50 +08:00
// This should never happen. In practice, previously currentBlock
2023-03-02 08:29:15 +02:00
// 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 {
2023-09-28 18:15:50 +08:00
// This should never happen. In practice, previously currentBlock
2023-03-02 08:29:15 +02:00
// 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
2024-04-01 11:41:09 +08:00
// rewindHashHead implements the logic of rewindHead in the context of hash scheme.
func ( bc * BlockChain ) rewindHashHead ( head * types . Header , root common . Hash ) ( * types . Header , uint64 ) {
var (
limit uint64 // The oldest block that will be searched for this rewinding
beyondRoot = root == common . Hash { } // Flag whether we're beyond the requested root (no root, always true)
pivot = rawdb . ReadLastPivotNumber ( bc . db ) // Associated block number of pivot point state
rootNumber uint64 // Associated block number of requested root
start = time . Now ( ) // Timestamp the rewinding is restarted
logged = time . Now ( ) // Timestamp last progress log was printed
)
// The oldest block to be searched is determined by the pivot block or a constant
// searching threshold. The rationale behind this is as follows:
//
// - Snap sync is selected if the pivot block is available. The earliest available
// state is the pivot block itself, so there is no sense in going further back.
//
// - Full sync is selected if the pivot block does not exist. The hash database
// periodically flushes the state to disk, and the used searching threshold is
// considered sufficient to find a persistent state, even for the testnet. It
// might be not enough for a chain that is nearly empty. In the worst case,
// the entire chain is reset to genesis, and snap sync is re-enabled on top,
// which is still acceptable.
if pivot != nil {
limit = * pivot
} else if head . Number . Uint64 ( ) > params . FullImmutabilityThreshold {
limit = head . Number . Uint64 ( ) - params . FullImmutabilityThreshold
}
for {
logger := log . Trace
if time . Since ( logged ) > time . Second * 8 {
logged = time . Now ( )
logger = log . Info
}
logger ( "Block state missing, rewinding further" , "number" , head . Number , "hash" , head . Hash ( ) , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) )
// If a root threshold was requested but not yet crossed, check
if ! beyondRoot && head . Root == root {
beyondRoot , rootNumber = true , head . Number . Uint64 ( )
}
// If search limit is reached, return the genesis block as the
// new chain head.
if head . Number . Uint64 ( ) < limit {
log . Info ( "Rewinding limit reached, resetting to genesis" , "number" , head . Number , "hash" , head . Hash ( ) , "limit" , limit )
return bc . genesisBlock . Header ( ) , rootNumber
}
// If the associated state is not reachable, continue searching
// backwards until an available state is found.
if ! bc . HasState ( head . Root ) {
// If the chain is gapped in the middle, return the genesis
// block as the new chain head.
parent := bc . GetHeader ( head . ParentHash , head . Number . Uint64 ( ) - 1 )
if parent == nil {
log . Error ( "Missing block in the middle, resetting to genesis" , "number" , head . Number . Uint64 ( ) - 1 , "hash" , head . ParentHash )
return bc . genesisBlock . Header ( ) , rootNumber
}
head = parent
// If the genesis block is reached, stop searching.
if head . Number . Uint64 ( ) == 0 {
log . Info ( "Genesis block reached" , "number" , head . Number , "hash" , head . Hash ( ) )
return head , rootNumber
}
continue // keep rewinding
}
// Once the available state is found, ensure that the requested root
// has already been crossed. If not, continue rewinding.
if beyondRoot || head . Number . Uint64 ( ) == 0 {
log . Info ( "Rewound to block with state" , "number" , head . Number , "hash" , head . Hash ( ) )
return head , rootNumber
}
log . Debug ( "Skipping block with threshold state" , "number" , head . Number , "hash" , head . Hash ( ) , "root" , head . Root )
head = bc . GetHeader ( head . ParentHash , head . Number . Uint64 ( ) - 1 ) // Keep rewinding
}
}
// rewindPathHead implements the logic of rewindHead in the context of path scheme.
func ( bc * BlockChain ) rewindPathHead ( head * types . Header , root common . Hash ) ( * types . Header , uint64 ) {
var (
pivot = rawdb . ReadLastPivotNumber ( bc . db ) // Associated block number of pivot block
rootNumber uint64 // Associated block number of requested root
// BeyondRoot represents whether the requested root is already
// crossed. The flag value is set to true if the root is empty.
beyondRoot = root == common . Hash { }
// noState represents if the target state requested for search
// is unavailable and impossible to be recovered.
noState = ! bc . HasState ( root ) && ! bc . stateRecoverable ( root )
start = time . Now ( ) // Timestamp the rewinding is restarted
logged = time . Now ( ) // Timestamp last progress log was printed
)
// Rewind the head block tag until an available state is found.
for {
logger := log . Trace
if time . Since ( logged ) > time . Second * 8 {
logged = time . Now ( )
logger = log . Info
}
logger ( "Block state missing, rewinding further" , "number" , head . Number , "hash" , head . Hash ( ) , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) )
// If a root threshold was requested but not yet crossed, check
if ! beyondRoot && head . Root == root {
beyondRoot , rootNumber = true , head . Number . Uint64 ( )
}
// If the root threshold hasn't been crossed but the available
// state is reached, quickly determine if the target state is
// possible to be reached or not.
if ! beyondRoot && noState && bc . HasState ( head . Root ) {
beyondRoot = true
log . Info ( "Disable the search for unattainable state" , "root" , root )
}
// Check if the associated state is available or recoverable if
// the requested root has already been crossed.
if beyondRoot && ( bc . HasState ( head . Root ) || bc . stateRecoverable ( head . Root ) ) {
break
}
// If pivot block is reached, return the genesis block as the
// new chain head. Theoretically there must be a persistent
// state before or at the pivot block, prevent endless rewinding
// towards the genesis just in case.
if pivot != nil && * pivot >= head . Number . Uint64 ( ) {
log . Info ( "Pivot block reached, resetting to genesis" , "number" , head . Number , "hash" , head . Hash ( ) )
return bc . genesisBlock . Header ( ) , rootNumber
}
// If the chain is gapped in the middle, return the genesis
// block as the new chain head
parent := bc . GetHeader ( head . ParentHash , head . Number . Uint64 ( ) - 1 ) // Keep rewinding
if parent == nil {
log . Error ( "Missing block in the middle, resetting to genesis" , "number" , head . Number . Uint64 ( ) - 1 , "hash" , head . ParentHash )
return bc . genesisBlock . Header ( ) , rootNumber
}
head = parent
// If the genesis block is reached, stop searching.
if head . Number . Uint64 ( ) == 0 {
log . Info ( "Genesis block reached" , "number" , head . Number , "hash" , head . Hash ( ) )
return head , rootNumber
}
}
// Recover if the target state if it's not available yet.
if ! bc . HasState ( head . Root ) {
if err := bc . triedb . Recover ( head . Root ) ; err != nil {
log . Crit ( "Failed to rollback state" , "err" , err )
}
}
log . Info ( "Rewound to block with state" , "number" , head . Number , "hash" , head . Hash ( ) )
return head , rootNumber
}
// rewindHead searches the available states in the database and returns the associated
// block as the new head block.
//
// If the given root is not empty, then the rewind should attempt to pass the specified
// state root and return the associated block number as well. If the root, typically
// representing the state corresponding to snapshot disk layer, is deemed impassable,
// then block number zero is returned, indicating that snapshot recovery is disabled
// and the whole snapshot should be auto-generated in case of head mismatch.
func ( bc * BlockChain ) rewindHead ( head * types . Header , root common . Hash ) ( * types . Header , uint64 ) {
2024-04-18 13:45:02 +08:00
if bc . triedb . Scheme ( ) == rawdb . PathScheme && ! bc . NoTries ( ) {
2024-04-01 11:41:09 +08:00
return bc . rewindPathHead ( head , root )
}
return bc . rewindHashHead ( head , root )
}
2024-04-18 15:12:05 +08:00
// SetFinalized sets the finalized block.
// This function differs slightly from Ethereum; we fine-tune it through the outer-layer setting finalizedBlockGauge.
func ( bc * BlockChain ) SetFinalized ( header * types . Header ) {
bc . currentFinalBlock . Store ( header )
if header != nil {
rawdb . WriteFinalizedBlockHash ( bc . db . BlockStore ( ) , header . Hash ( ) )
} else {
rawdb . WriteFinalizedBlockHash ( bc . db . BlockStore ( ) , common . Hash { } )
}
}
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
2024-04-01 11:41:09 +08:00
var (
// Track the block number of the requested root hash
rootNumber uint64 // (no root == always 0)
2020-08-20 13:01:24 +03:00
2024-04-01 11:41:09 +08:00
// Retrieve the last pivot block to short circuit rollbacks beyond it
// and the current freezer limit to start nuking it's underflown.
pivot = rawdb . ReadLastPivotNumber ( bc . db )
)
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 ( ) {
2024-04-18 13:45:02 +08:00
// load bc.snaps for the judge `HasState`
if bc . NoTries ( ) {
if bc . cacheConfig . SnapshotLimit > 0 {
snapconfig := snapshot . Config {
CacheSize : bc . cacheConfig . SnapshotLimit ,
NoBuild : bc . cacheConfig . SnapshotNoBuild ,
AsyncBuild : ! bc . cacheConfig . SnapshotWait ,
}
bc . snaps , _ = snapshot . New ( snapconfig , bc . db , bc . triedb , header . Root , int ( bc . cacheConfig . TriesInMemory ) , bc . NoTries ( ) )
}
defer func ( ) { bc . snaps = nil } ( )
}
2024-04-01 11:41:09 +08:00
var newHeadBlock * types . Header
newHeadBlock , rootNumber = bc . rewindHead ( header , root )
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.
2024-04-01 11:41:09 +08:00
bc . currentBlock . Store ( newHeadBlock )
headBlockGauge . Update ( int64 ( newHeadBlock . Number . Uint64 ( ) ) )
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 15:00:53 +08:00
// The head state is missing, which is only possible in the path-based
// scheme. This situation occurs when the chain head is rewound below
// the pivot point. In this scenario, there is no possible recovery
// approach except for rerunning a snap sync. Do nothing here until the
// state syncer picks it up.
2024-04-01 11:41:09 +08:00
if ! bc . HasState ( newHeadBlock . Root ) {
if newHeadBlock . Number . Uint64 ( ) != 0 {
log . Crit ( "Chain is stateless at a non-genesis block" )
}
log . Info ( "Chain is stateless, wait state sync" , "number" , newHeadBlock . Number , "hash" , newHeadBlock . Hash ( ) )
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 15:00:53 +08: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
}
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
2024-07-16 22:37:03 +08:00
frozen , _ := bc . db . BlockStore ( ) . Ancients ( )
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
2024-07-16 22:37:03 +08:00
frozen , _ := bc . db . BlockStore ( ) . 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
if num + 1 <= frozen {
// Truncate all relative data(header, total difficulty, body, receipt
// and canonical hash) from ancient store.
2024-07-16 22:37:03 +08:00
if _ , err := bc . db . BlockStore ( ) . 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 )
2024-03-28 16:35:39 +08:00
rawdb . DeleteBlobSidecars ( db , hash , num )
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 . 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 {
2024-06-18 15:20:23 +08:00
if target , force := updateFn ( bc . db . BlockStore ( ) , 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 ( )
2024-03-22 22:37:47 +08:00
bc . sidecarsCache . 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
2024-04-18 15:12:05 +08:00
if finalized := bc . CurrentFinalBlock ( ) ; finalized != nil && head < finalized . Number . Uint64 ( ) {
log . Error ( "SetHead invalidated finalized block" )
bc . SetFinalized ( nil )
}
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 {
2024-08-09 17:29:47 +08:00
// TODO:: temporarily not support for snapsync
if bc . triedb . Scheme ( ) == rawdb . VersionScheme {
panic ( "version db not support snap sync" )
}
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 {
core, accounts, eth, trie: handle genesis state missing (#28171)
* core, accounts, eth, trie: handle genesis state missing
* core, eth, trie: polish
* core: manage txpool subscription in mainpool
* eth/backend: fix test
* cmd, eth: fix test
* core/rawdb, trie/triedb/pathdb: address comments
* eth, trie: address comments
* eth: inline the function
* eth: use synced flag
* core/txpool: revert changes in txpool
* core, eth, trie: rename functions
2023-09-28 15:00:53 +08:00
if err := bc . triedb . Enable ( root ) ; err != nil {
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
return err
}
}
2024-03-07 10:00:01 +08:00
if ! bc . NoTries ( ) && ! bc . HasState ( root ) {
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
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
}
2024-03-22 22:37:47 +08:00
// UpdateChasingHead update remote best chain head, used by DA check now.
func ( bc * BlockChain ) UpdateChasingHead ( head * types . Header ) {
bc . chasingHead . Store ( head )
}
// ChasingHead return the best chain head of peers.
func ( bc * BlockChain ) ChasingHead ( ) * types . Header {
return bc . chasingHead . Load ( )
}
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
2024-04-18 15:12:05 +08:00
blockBatch := bc . db . BlockStore ( ) . NewBatch ( )
rawdb . WriteTd ( blockBatch , genesis . Hash ( ) , genesis . NumberU64 ( ) , genesis . Difficulty ( ) )
rawdb . WriteBlock ( blockBatch , genesis )
if err := blockBatch . Write ( ) ; err != nil {
2020-01-17 18:49:32 +08:00
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 ) {
2024-06-18 15:20:23 +08:00
bc . dbWg . Add ( 2 )
defer bc . dbWg . Wait ( )
go func ( ) {
defer bc . dbWg . Done ( )
// Add the block to the canonical chain number scheme and mark as the head
blockBatch := bc . db . BlockStore ( ) . NewBatch ( )
rawdb . WriteCanonicalHash ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) )
rawdb . WriteHeadHeaderHash ( blockBatch , block . Hash ( ) )
rawdb . WriteHeadBlockHash ( blockBatch , block . Hash ( ) )
rawdb . WriteHeadFastBlockHash ( blockBatch , block . Hash ( ) )
// Flush the whole batch into the disk, exit the node if failed
if err := blockBatch . Write ( ) ; err != nil {
log . Crit ( "Failed to update chain indexes and markers in block db" , "err" , err )
}
} ( )
go func ( ) {
defer bc . dbWg . Done ( )
2024-04-18 15:12:05 +08:00
2024-06-18 15:20:23 +08:00
batch := bc . db . NewBatch ( )
rawdb . WriteTxLookupEntriesByBlock ( batch , block )
// 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 in chain db" , "err" , err )
}
} ( )
2015-10-28 17:40:24 +02:00
2020-01-17 18:49:32 +08:00
// 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
}
2024-01-24 04:00:50 +08:00
// Signal shutdown tx indexer.
if bc . txIndexer != nil {
bc . txIndexer . close ( )
}
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
2023-09-28 18:15:50 +08:00
// Ensure that the entirety of the state snapshot is journaled 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
}
2023-10-20 13:35:49 +02:00
bc . snaps . Release ( )
2019-08-06 13:40:28 +03:00
}
2024-08-09 17:29:47 +08:00
if bc . triedb . Scheme ( ) != rawdb . VersionScheme {
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
var once sync . Once
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 )
} else {
rawdb . WriteSafePointBlockNumber ( bc . db , recent . NumberU64 ( ) )
once . Do ( func ( ) {
rawdb . WriteHeadBlockHash ( bc . db . BlockStore ( ) , recent . Hash ( ) )
} )
}
}
}
if snapBase != ( common . Hash { } ) {
log . Info ( "Writing snapshot state to disk" , "root" , snapBase )
if err := triedb . Commit ( snapBase , true ) ; err != nil {
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 ( "Failed to commit recent state trie" , "err" , err )
2023-10-12 22:17:11 +08:00
} else {
2024-08-09 17:29:47 +08:00
rawdb . WriteSafePointBlockNumber ( bc . db , bc . CurrentBlock ( ) . Number . Uint64 ( ) )
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
}
2024-08-09 17:29:47 +08:00
for ! bc . triegc . Empty ( ) {
triedb . Dereference ( bc . triegc . PopItem ( ) )
}
if _ , size , _ , _ := triedb . Size ( ) ; size != 0 {
log . Error ( "Dangling trie nodes after full cleanup" )
2018-02-23 14:02:33 +02:00
}
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
core/types: support for optional blob sidecar in BlobTx (#27841)
This PR removes the newly added txpool.Transaction wrapper type, and instead adds a way
of keeping the blob sidecar within types.Transaction. It's better this way because most
code in go-ethereum does not care about blob transactions, and probably never will. This
will start mattering especially on the client side of RPC, where all APIs are based on
types.Transaction. Users need to be able to use the same signing flows they already
have.
However, since blobs are only allowed in some places but not others, we will now need to
add checks to avoid creating invalid blocks. I'm still trying to figure out the best place
to do some of these. The way I have it currently is as follows:
- In block validation (import), txs are verified not to have a blob sidecar.
- In miner, we strip off the sidecar when committing the transaction into the block.
- In TxPool validation, txs must have a sidecar to be added into the blobpool.
- Note there is a special case here: when transactions are re-added because of a chain
reorg, we cannot use the transactions gathered from the old chain blocks as-is,
because they will be missing their blobs. This was previously handled by storing the
blobs into the 'blobpool limbo'. The code has now changed to store the full
transaction in the limbo instead, but it might be confusing for code readers why we're
not simply adding the types.Transaction we already have.
Code changes summary:
- txpool.Transaction removed and all uses replaced by types.Transaction again
- blobpool now stores types.Transaction instead of defining its own blobTx format for storage
- the blobpool limbo now stores types.Transaction instead of storing only the blobs
- checks to validate the presence/absence of the blob sidecar added in certain critical places
2023-08-14 10:13:34 +02:00
for i , block := range blockChain {
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 i != 0 {
core/types: support for optional blob sidecar in BlobTx (#27841)
This PR removes the newly added txpool.Transaction wrapper type, and instead adds a way
of keeping the blob sidecar within types.Transaction. It's better this way because most
code in go-ethereum does not care about blob transactions, and probably never will. This
will start mattering especially on the client side of RPC, where all APIs are based on
types.Transaction. Users need to be able to use the same signing flows they already
have.
However, since blobs are only allowed in some places but not others, we will now need to
add checks to avoid creating invalid blocks. I'm still trying to figure out the best place
to do some of these. The way I have it currently is as follows:
- In block validation (import), txs are verified not to have a blob sidecar.
- In miner, we strip off the sidecar when committing the transaction into the block.
- In TxPool validation, txs must have a sidecar to be added into the blobpool.
- Note there is a special case here: when transactions are re-added because of a chain
reorg, we cannot use the transactions gathered from the old chain blocks as-is,
because they will be missing their blobs. This was previously handled by storing the
blobs into the 'blobpool limbo'. The code has now changed to store the full
transaction in the limbo instead, but it might be confusing for code readers why we're
not simply adding the types.Transaction we already have.
Code changes summary:
- txpool.Transaction removed and all uses replaced by types.Transaction again
- blobpool now stores types.Transaction instead of defining its own blobTx format for storage
- the blobpool limbo now stores types.Transaction instead of storing only the blobs
- checks to validate the presence/absence of the blob sidecar added in certain critical places
2023-08-14 10:13:34 +02:00
prev := blockChain [ i - 1 ]
if block . NumberU64 ( ) != prev . NumberU64 ( ) + 1 || block . ParentHash ( ) != prev . Hash ( ) {
log . Error ( "Non contiguous receipt insert" ,
"number" , block . Number ( ) , "hash" , block . Hash ( ) , "parent" , block . ParentHash ( ) ,
"prevnumber" , prev . Number ( ) , "prevhash" , prev . Hash ( ) )
return 0 , fmt . Errorf ( "non contiguous insert: item %d is #%d [%x..], item %d is #%d [%x..] (parent [%x..])" ,
i - 1 , prev . NumberU64 ( ) , prev . Hash ( ) . Bytes ( ) [ : 4 ] ,
i , block . NumberU64 ( ) , block . Hash ( ) . Bytes ( ) [ : 4 ] , block . ParentHash ( ) . 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
}
}
core/types: support for optional blob sidecar in BlobTx (#27841)
This PR removes the newly added txpool.Transaction wrapper type, and instead adds a way
of keeping the blob sidecar within types.Transaction. It's better this way because most
code in go-ethereum does not care about blob transactions, and probably never will. This
will start mattering especially on the client side of RPC, where all APIs are based on
types.Transaction. Users need to be able to use the same signing flows they already
have.
However, since blobs are only allowed in some places but not others, we will now need to
add checks to avoid creating invalid blocks. I'm still trying to figure out the best place
to do some of these. The way I have it currently is as follows:
- In block validation (import), txs are verified not to have a blob sidecar.
- In miner, we strip off the sidecar when committing the transaction into the block.
- In TxPool validation, txs must have a sidecar to be added into the blobpool.
- Note there is a special case here: when transactions are re-added because of a chain
reorg, we cannot use the transactions gathered from the old chain blocks as-is,
because they will be missing their blobs. This was previously handled by storing the
blobs into the 'blobpool limbo'. The code has now changed to store the full
transaction in the limbo instead, but it might be confusing for code readers why we're
not simply adding the types.Transaction we already have.
Code changes summary:
- txpool.Transaction removed and all uses replaced by types.Transaction again
- blobpool now stores types.Transaction instead of defining its own blobTx format for storage
- the blobpool limbo now stores types.Transaction instead of storing only the blobs
- checks to validate the presence/absence of the blob sidecar added in certain critical places
2023-08-14 10:13:34 +02:00
if block . NumberU64 ( ) <= ancientLimit {
ancientBlocks , ancientReceipts = append ( ancientBlocks , block ) , append ( ancientReceipts , receiptChain [ i ] )
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 {
core/types: support for optional blob sidecar in BlobTx (#27841)
This PR removes the newly added txpool.Transaction wrapper type, and instead adds a way
of keeping the blob sidecar within types.Transaction. It's better this way because most
code in go-ethereum does not care about blob transactions, and probably never will. This
will start mattering especially on the client side of RPC, where all APIs are based on
types.Transaction. Users need to be able to use the same signing flows they already
have.
However, since blobs are only allowed in some places but not others, we will now need to
add checks to avoid creating invalid blocks. I'm still trying to figure out the best place
to do some of these. The way I have it currently is as follows:
- In block validation (import), txs are verified not to have a blob sidecar.
- In miner, we strip off the sidecar when committing the transaction into the block.
- In TxPool validation, txs must have a sidecar to be added into the blobpool.
- Note there is a special case here: when transactions are re-added because of a chain
reorg, we cannot use the transactions gathered from the old chain blocks as-is,
because they will be missing their blobs. This was previously handled by storing the
blobs into the 'blobpool limbo'. The code has now changed to store the full
transaction in the limbo instead, but it might be confusing for code readers why we're
not simply adding the types.Transaction we already have.
Code changes summary:
- txpool.Transaction removed and all uses replaced by types.Transaction again
- blobpool now stores types.Transaction instead of defining its own blobTx format for storage
- the blobpool limbo now stores types.Transaction instead of storing only the blobs
- checks to validate the presence/absence of the blob sidecar added in certain critical places
2023-08-14 10:13:34 +02:00
liveBlocks , liveReceipts = append ( liveBlocks , block ) , append ( liveReceipts , receiptChain [ i ] )
}
// Here we also validate that blob transactions in the block do not contain a sidecar.
// While the sidecar does not affect the block hash / tx hash, sending blobs within a block is not allowed.
for txIndex , tx := range block . Transactions ( ) {
if tx . Type ( ) == types . BlobTxType && tx . BlobTxSidecar ( ) != nil {
return 0 , fmt . Errorf ( "block #%d contains unexpected blob sidecar in tx at index %d" , block . NumberU64 ( ) , txIndex )
}
2016-12-13 16:14:33 +02:00
}
}
2015-09-30 19:23:31 +03:00
2024-03-22 22:37:47 +08:00
// check DA after cancun
lastBlk := blockChain [ len ( blockChain ) - 1 ]
if bc . chainConfig . Parlia != nil && bc . chainConfig . IsCancun ( lastBlk . Number ( ) , lastBlk . Time ( ) ) {
if _ , err := CheckDataAvailableInBatch ( bc , blockChain ) ; err != nil {
2024-03-28 16:35:39 +08:00
log . Debug ( "CheckDataAvailableInBatch" , "err" , err )
2024-03-22 22:37:47 +08:00
return 0 , err
}
}
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
}
2024-06-18 15:20:23 +08:00
rawdb . WriteHeadFastBlockHash ( bc . db . BlockStore ( ) , 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 {
2024-07-16 22:37:03 +08:00
if frozen , _ := bc . db . BlockStore ( ) . Ancients ( ) ; frozen == 0 {
2021-09-07 12:31:17 +02:00
td := bc . genesisBlock . Difficulty ( )
2024-07-16 22:37:03 +08:00
writeSize , err := rawdb . WriteAncientBlocks ( bc . db . BlockStore ( ) , [ ] * types . Block { bc . genesisBlock } , [ ] types . Receipts { nil } , td )
2021-09-07 12:31:17 +02:00
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
}
2024-01-23 04:05:18 +08:00
size += writeSize
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 ( ) )
2024-07-16 22:37:03 +08:00
writeSize , err := rawdb . WriteAncientBlocksWithBlobs ( bc . db . BlockStore ( ) , blockChain , receiptChain , td )
2021-09-07 12:31:17 +02:00
if err != nil {
log . Error ( "Error importing chain data to ancients" , "err" , err )
return 0 , err
}
2024-01-23 04:05:18 +08:00
size += writeSize
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.
2024-07-16 22:37:03 +08:00
if err := bc . db . BlockStore ( ) . Sync ( ) ; err != nil {
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 , 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.
2024-07-16 22:37:03 +08:00
if _ , err := bc . db . BlockStore ( ) . 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.
2024-01-23 04:05:18 +08:00
var (
canonHashes = make ( map [ common . Hash ] struct { } )
2024-04-18 15:12:05 +08:00
blockBatch = bc . db . BlockStore ( ) . NewBatch ( )
2024-01-23 04:05:18 +08:00
)
2021-09-07 12:31:17 +02:00
for _ , block := range blockChain {
canonHashes [ block . Hash ( ) ] = struct { } { }
if block . NumberU64 ( ) == 0 {
continue
2019-07-08 10:24:16 +02:00
}
2024-04-18 15:12:05 +08:00
rawdb . DeleteCanonicalHash ( blockBatch , block . NumberU64 ( ) )
rawdb . DeleteBlockWithoutNumber ( blockBatch , 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.
2024-07-16 22:37:03 +08:00
for _ , nh := range rawdb . ReadAllHashesInRange ( bc . db . BlockStore ( ) , first . NumberU64 ( ) , last . NumberU64 ( ) ) {
2021-09-07 12:31:17 +02:00
if _ , canon := canonHashes [ nh . Hash ] ; ! canon {
2024-04-18 15:12:05 +08:00
rawdb . DeleteHeader ( blockBatch , 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
}
}
2024-04-18 15:12:05 +08:00
if err := blockBatch . Write ( ) ; err != nil {
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 , err
}
2024-01-23 04:05:18 +08:00
stats . processed += int32 ( len ( blockChain ) )
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 , 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 ) {
2024-01-23 04:05:18 +08:00
var (
skipPresenceCheck = false
batch = bc . db . NewBatch ( )
2024-04-18 15:12:05 +08:00
blockBatch = bc . db . BlockStore ( ) . NewBatch ( )
2024-01-23 04:05:18 +08: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
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
2024-04-18 15:12:05 +08:00
rawdb . WriteBody ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , block . Body ( ) )
rawdb . WriteReceipts ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , receiptChain [ i ] )
2024-03-22 22:37:47 +08:00
if bc . chainConfig . IsCancun ( block . Number ( ) , block . Time ( ) ) {
2024-04-18 15:12:05 +08:00
rawdb . WriteBlobSidecars ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , block . Sidecars ( ) )
2024-03-22 22:37:47 +08: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
2020-01-17 18:49:32 +08:00
// Write everything belongs to the blocks into the database. So that
2024-01-23 04:05:18 +08:00
// we can ensure all components of body is completed(body, receipts)
// except transaction indexes(will be created once sync is finished).
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 ( )
}
2024-04-18 15:12:05 +08:00
if blockBatch . ValueSize ( ) >= ethdb . IdealBatchSize {
if err := blockBatch . Write ( ) ; err != nil {
return 0 , err
}
size += int64 ( blockBatch . ValueSize ( ) )
blockBatch . 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
}
2024-04-18 15:12:05 +08:00
if blockBatch . ValueSize ( ) > 0 {
size += int64 ( blockBatch . ValueSize ( ) )
if err := blockBatch . Write ( ) ; err != nil {
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
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
}
}
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
}
}
2024-01-23 04:05:18 +08:00
var (
head = blockChain [ len ( blockChain ) - 1 ]
context = [ ] interface { } {
"count" , stats . processed , "elapsed" , common . PrettyDuration ( time . Since ( start ) ) ,
"number" , head . Number ( ) , "hash" , head . Hash ( ) , "age" , common . PrettyAge ( time . Unix ( int64 ( head . Time ( ) ) , 0 ) ) ,
"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
}
2024-04-18 15:12:05 +08:00
blockBatch := bc . db . BlockStore ( ) . NewBatch ( )
rawdb . WriteTd ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , td )
rawdb . WriteBlock ( blockBatch , block )
2024-03-22 22:37:47 +08:00
// if cancun is enabled, here need to write sidecars too
if bc . chainConfig . IsCancun ( block . Number ( ) , block . Time ( ) ) {
2024-04-18 15:12:05 +08:00
rawdb . WriteBlobSidecars ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , block . Sidecars ( ) )
2024-03-22 22:37:47 +08:00
}
2024-04-18 15:12:05 +08:00
if err := blockBatch . Write ( ) ; err != nil {
2020-01-17 18:49:32 +08:00
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 ( ) {
2024-04-18 15:12:05 +08:00
blockBatch := bc . db . BlockStore ( ) . NewBatch ( )
2022-07-05 11:14:21 +08:00
rawdb . WriteTd ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , externTd )
rawdb . WriteBlock ( blockBatch , block )
rawdb . WriteReceipts ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , receipts )
2024-03-22 22:37:47 +08:00
// if cancun is enabled, here need to write sidecars too
if bc . chainConfig . IsCancun ( block . Number ( ) , block . Time ( ) ) {
rawdb . WriteBlobSidecars ( blockBatch , block . Hash ( ) , block . NumberU64 ( ) , block . Sidecars ( ) )
}
2024-06-18 15:20:23 +08:00
if bc . db . StateStore ( ) != nil {
rawdb . WritePreimages ( bc . db . StateStore ( ) , state . Preimages ( ) )
} else {
rawdb . WritePreimages ( blockBatch , state . Preimages ( ) )
}
2022-07-05 11:14:21 +08:00
if err := blockBatch . Write ( ) ; err != nil {
log . Crit ( "Failed to write block into disk" , "err" , err )
}
2024-07-22 10:16:01 +08:00
bc . hc . tdCache . Add ( block . Hash ( ) , externTd )
bc . blockCache . Add ( block . Hash ( ) , block )
bc . receiptsCache . Add ( block . Hash ( ) , receipts )
if bc . chainConfig . IsCancun ( block . Number ( ) , block . Time ( ) ) {
bc . sidecarsCache . Add ( block . Hash ( ) , block . Sidecars ( ) )
}
2022-07-05 11:14:21 +08:00
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.
2024-08-09 17:29:47 +08:00
if bc . triedb . Scheme ( ) != rawdb . HashScheme {
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
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 {
2024-02-02 15:43:33 +08:00
return triedb . Commit ( block . Root ( ) , false )
}
// 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 ( ) ) )
2023-08-23 17:46:08 +08:00
2024-02-02 15:43:33 +08:00
// Flush limits are not considered for the first TriesInMemory blocks.
current := block . NumberU64 ( )
if current <= TriesInMemory {
return nil
}
// If we exceeded our memory allowance, flush matured singleton nodes to disk
var (
_ , nodes , _ , imgs = triedb . Size ( )
limit = common . StorageSize ( bc . cacheConfig . TrieDirtyLimit ) * 1024 * 1024
)
if nodes > limit || imgs > 4 * 1024 * 1024 {
triedb . Cap ( limit - ethdb . IdealBatchSize )
}
// Find the next state trie we need to commit
chosen := current - bc . triesInMemory
flushInterval := time . Duration ( bc . flushInterval . Load ( ) )
// If we exceeded out time allowance, flush an entire trie to disk
if bc . gcproc > flushInterval {
canWrite := true
if posa , ok := bc . engine . ( consensus . PoSA ) ; ok {
if ! posa . EnoughDistance ( bc , block . Header ( ) ) {
canWrite = false
2018-02-05 18:40:32 +02:00
}
2024-02-02 15:43:33 +08:00
}
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.
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
}
2024-02-02 15:43:33 +08:00
// Flush an entire trie and restart the counters
triedb . Commit ( header . Root , true )
rawdb . WriteSafePointBlockNumber ( bc . db , chosen )
bc . lastWrite = chosen
bc . gcproc = 0
2018-02-05 18:40:32 +02:00
}
}
}
2024-02-02 15:43:33 +08:00
// Garbage collect anything below our required write retention
wg2 := sync . WaitGroup { }
for ! bc . triegc . Empty ( ) {
root , number := bc . triegc . Pop ( )
if uint64 ( - number ) > chosen {
bc . triegc . Push ( root , number )
break
}
wg2 . Add ( 1 )
go func ( ) {
triedb . Dereference ( root )
wg2 . Done ( )
} ( )
}
wg2 . Wait ( )
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.
2024-07-22 18:23:10 +08:00
func ( bc * BlockChain ) WriteBlockAndSetHead ( block * types . Block , receipts [ ] * types . Receipt , logs [ ] * types . Log , state * state . StateDB , emitHeadEvent bool ) ( status WriteStatus , err 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 ( ) {
return NonStatTy , errChainStopped
}
defer bc . chainmu . Unlock ( )
2024-07-22 18:23:10 +08:00
return bc . writeBlockAndSetHead ( block , receipts , logs , state , emitHeadEvent )
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-05 00:22:00 -07:00
// writeBlockAndSetHead is the internal implementation of WriteBlockAndSetHead.
// This function expects the chain mutex to be held.
2024-07-22 18:23:10 +08:00
func ( bc * BlockChain ) writeBlockAndSetHead ( block * types . Block , receipts [ ] * types . Receipt , logs [ ] * types . Log , state * state . StateDB , emitHeadEvent bool ) ( status WriteStatus , err error ) {
if err := bc . writeBlockWithState ( block , receipts , state ) ; err != nil {
return NonStatTy , 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
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.
2024-04-18 15:12:05 +08:00
var finalizedHeader * types . Header
if posa , ok := bc . Engine ( ) . ( consensus . PoSA ) ; ok {
if finalizedHeader = posa . GetFinalizedHeader ( bc , block . Header ( ) ) ; finalizedHeader != nil {
bc . SetFinalized ( finalizedHeader )
}
}
2019-11-29 21:22:08 +08:00
if emitHeadEvent {
bc . chainHeadFeed . Send ( ChainHeadEvent { Block : block } )
2024-04-18 15:12:05 +08:00
if finalizedHeader != nil {
bc . finalizedHeaderFeed . Send ( FinalizedHeaderEvent { finalizedHeader } )
2023-04-10 18:36:45 +08:00
}
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
}
} ( )
2024-03-22 22:37:47 +08:00
// check block data available first
if bc . chainConfig . Parlia != nil {
if index , err := CheckDataAvailableInBatch ( bc , chain ) ; err != nil {
return index , err
}
}
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 {
2024-04-18 15:12:05 +08:00
rawdb . WriteReceipts ( bc . db . BlockStore ( ) , block . Hash ( ) , block . NumberU64 ( ) , nil )
2020-05-13 16:33:48 +08:00
} 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
}
2024-03-22 22:37:47 +08:00
2024-08-09 17:29:47 +08:00
if bc . stateCache . Scheme ( ) != rawdb . VersionScheme {
if block . NumberU64 ( ) == 2000001 {
log . Crit ( "exit.... path mode, 200w blocks" )
}
}
bc . stateCache . SetVersion ( int64 ( block . NumberU64 ( ) ) )
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 {
2024-08-09 17:29:47 +08:00
bc . stateCache . Release ( )
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
2024-08-09 17:29:47 +08:00
//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
2024-08-09 17:29:47 +08:00
//if len(block.Transactions()) >= prefetchTxNumber {
// // do Prefetch in a separate goroutine to avoid blocking the critical path
//
// // 1.do state prefetch for snapshot cache
// throwaway := statedb.CopyDoPrefetch()
// go bc.prefetcher.Prefetch(block, throwaway, &bc.vmConfig, interruptCh)
//
// // 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
// go throwaway.TriePrefetchInAdvance(block, signer)
//}
[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
2024-03-19 15:18:58 +08:00
// Process block using the parent state as reference point
2022-07-05 11:14:21 +08:00
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 {
2024-08-09 17:29:47 +08:00
bc . stateCache . Release ( )
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 {
2024-08-09 17:29:47 +08:00
bc . stateCache . Release ( )
2023-09-07 16:39:29 +08:00
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
// 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 {
2024-07-22 18:23:10 +08:00
status , err = bc . writeBlockAndSetHead ( block , receipts , logs , statedb , 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
}
2015-06-29 12:12:30 +02:00
if err != nil {
2024-08-09 17:29:47 +08:00
bc . stateCache . Release ( )
2019-11-29 21:22:08 +08:00
return it . index , err
2015-06-29 12:12:30 +02:00
}
2024-08-09 17:29:47 +08:00
bc . stateCache . Release ( )
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-08-23 14:08:39 +03:00
var snapDiffItems , snapBufItems common . StorageSize
if bc . snaps != nil {
2024-02-02 15:43:33 +08:00
snapDiffItems , snapBufItems , _ = bc . snaps . Size ( )
2023-08-23 14:08:39 +03:00
}
2023-11-06 14:11:17 +08:00
trieDiffNodes , trieBufNodes , trieImmutableBufNodes , _ := bc . triedb . Size ( )
2024-04-01 19:26:50 +08:00
stats . report ( chain , it . index , snapDiffItems , snapBufItems , trieDiffNodes , trieBufNodes , trieImmutableBufNodes , status == CanonStatTy )
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
}
2024-07-11 15:01:01 +08:00
currentBlock := bc . CurrentBlock ( )
reorg , err := bc . forker . ReorgNeededWithFastFinality ( currentBlock , header )
if err == nil && reorg {
2022-07-05 11:14:21 +08:00
bc . highestVerifiedHeader . Store ( types . CopyHeader ( header ) )
2024-07-11 15:01:01 +08:00
log . Trace ( "updateHighestVerifiedHeader" , "number" , header . Number . Uint64 ( ) , "hash" , header . Hash ( ) )
2022-07-05 11:14:21 +08:00
}
}
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 ] )
}
2024-03-22 22:37:47 +08:00
if bc . chainConfig . IsCancun ( block . Number ( ) , block . Time ( ) ) {
block = block . WithSidecars ( bc . GetSidecarsByHash ( hashes [ i ] ) )
}
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 ] )
}
2024-03-22 22:37:47 +08:00
if bc . chainConfig . IsCancun ( b . Number ( ) , b . Time ( ) ) {
b = b . WithSidecars ( bc . GetSidecarsByHash ( b . Hash ( ) ) )
}
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
}
2024-01-30 09:34:14 +08:00
// Reset the tx lookup cache in case to clear stale txlookups.
// This is done before writing any new chain data to avoid the
// weird scenario that canonical chain is changed while the
// stale lookups are still cached.
bc . txLookupCache . Purge ( )
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.
2024-01-30 09:34:14 +08:00
var (
indexesBatch = bc . db . NewBatch ( )
diffs = types . HashDifference ( deletedTxs , addedTxs )
2024-04-18 15:12:05 +08:00
blockBatch = bc . db . BlockStore ( ) . NewBatch ( )
2024-01-30 09:34:14 +08:00
)
for _ , tx := range diffs {
2022-05-30 08:42:06 +02:00
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
}
2024-04-18 15:12:05 +08:00
rawdb . DeleteCanonicalHash ( blockBatch , i )
2020-01-17 18:49:32 +08:00
}
if err := indexesBatch . Write ( ) ; err != nil {
log . Crit ( "Failed to delete useless indexes" , "err" , err )
2019-05-07 14:26:00 +02:00
}
2024-04-18 15:12:05 +08:00
if err := blockBatch . Write ( ) ; err != nil {
log . Crit ( "Failed to delete useless indexes use block batch" , "err" , err )
}
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-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.
2024-03-22 22:37:47 +08:00
// bad block need not save receipts & sidecars.
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 {
2024-03-11 14:52:33 +08:00
return common . Hash { } , errors . New ( "nil diff layer" )
2022-07-05 11:14:21 +08:00
}
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
2024-02-16 19:05:33 +01:00
// GetTrieFlushInterval gets the in-memory tries flushAlloc interval
2023-06-06 20:41:44 +08:00
func ( bc * BlockChain ) GetTrieFlushInterval ( ) time . Duration {
return time . Duration ( bc . flushInterval . Load ( ) )
}