mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 09:21:11 +00:00
59909a7efe
This PR adds support to store the transaction dependency (generated by the block producer) in the block header for bor. This transaction dependency will then be used by the parallel processor ([Block-STM](https://github.com/ledgerwatch/erigon/pull/7812/)). I have created another [PR](https://github.com/ledgerwatch/erigon-lib/pull/1064) in the erigon-lib repo which adds the `IsParallelUniverse()` function.
680 lines
24 KiB
Go
680 lines
24 KiB
Go
/*
|
|
Copyright 2021 Erigon contributors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package chain
|
|
|
|
import (
|
|
"fmt"
|
|
"math/big"
|
|
"sort"
|
|
"strconv"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/common/fixedgas"
|
|
)
|
|
|
|
// Config is the core config which determines the blockchain settings.
|
|
//
|
|
// Config is stored in the database on a per block basis. This means
|
|
// that any network, identified by its genesis block, can have its own
|
|
// set of configuration options.
|
|
type Config struct {
|
|
ChainName string
|
|
ChainID *big.Int `json:"chainId"` // chainId identifies the current chain and is used for replay protection
|
|
|
|
Consensus ConsensusName `json:"consensus,omitempty"` // aura, ethash or clique
|
|
|
|
// *Block fields activate the corresponding hard fork at a certain block number,
|
|
// while *Time fields do so based on the block's time stamp.
|
|
// nil means that the hard-fork is not scheduled,
|
|
// while 0 means that it's already activated from genesis.
|
|
|
|
// ETH mainnet upgrades
|
|
// See https://github.com/ethereum/execution-specs/blob/master/network-upgrades/mainnet-upgrades
|
|
HomesteadBlock *big.Int `json:"homesteadBlock,omitempty"`
|
|
DAOForkBlock *big.Int `json:"daoForkBlock,omitempty"`
|
|
TangerineWhistleBlock *big.Int `json:"eip150Block,omitempty"`
|
|
SpuriousDragonBlock *big.Int `json:"eip155Block,omitempty"`
|
|
ByzantiumBlock *big.Int `json:"byzantiumBlock,omitempty"`
|
|
ConstantinopleBlock *big.Int `json:"constantinopleBlock,omitempty"`
|
|
PetersburgBlock *big.Int `json:"petersburgBlock,omitempty"`
|
|
IstanbulBlock *big.Int `json:"istanbulBlock,omitempty"`
|
|
MuirGlacierBlock *big.Int `json:"muirGlacierBlock,omitempty"`
|
|
BerlinBlock *big.Int `json:"berlinBlock,omitempty"`
|
|
LondonBlock *big.Int `json:"londonBlock,omitempty"`
|
|
ArrowGlacierBlock *big.Int `json:"arrowGlacierBlock,omitempty"`
|
|
GrayGlacierBlock *big.Int `json:"grayGlacierBlock,omitempty"`
|
|
|
|
// EIP-3675: Upgrade consensus to Proof-of-Stake (a.k.a. "Paris", "The Merge")
|
|
TerminalTotalDifficulty *big.Int `json:"terminalTotalDifficulty,omitempty"` // The merge happens when terminal total difficulty is reached
|
|
TerminalTotalDifficultyPassed bool `json:"terminalTotalDifficultyPassed,omitempty"` // Disable PoW sync for networks that have already passed through the Merge
|
|
MergeNetsplitBlock *big.Int `json:"mergeNetsplitBlock,omitempty"` // Virtual fork after The Merge to use as a network splitter; see FORK_NEXT_VALUE in EIP-3675
|
|
|
|
// Mainnet fork scheduling switched from block numbers to timestamps after The Merge
|
|
ShanghaiTime *big.Int `json:"shanghaiTime,omitempty"`
|
|
CancunTime *big.Int `json:"cancunTime,omitempty"`
|
|
PragueTime *big.Int `json:"pragueTime,omitempty"`
|
|
|
|
// Optional EIP-4844 parameters
|
|
MinBlobGasPrice *uint64 `json:"minBlobGasPrice,omitempty"`
|
|
MaxBlobGasPerBlock *uint64 `json:"maxBlobGasPerBlock,omitempty"`
|
|
TargetBlobGasPerBlock *uint64 `json:"targetBlobGasPerBlock,omitempty"`
|
|
BlobGasPriceUpdateFraction *uint64 `json:"blobGasPriceUpdateFraction,omitempty"`
|
|
|
|
// (Optional) governance contract where EIP-1559 fees will be sent to that otherwise would be burnt since the London fork
|
|
BurntContract map[string]common.Address `json:"burntContract,omitempty"`
|
|
|
|
// Various consensus engines
|
|
Ethash *EthashConfig `json:"ethash,omitempty"`
|
|
Clique *CliqueConfig `json:"clique,omitempty"`
|
|
Aura *AuRaConfig `json:"aura,omitempty"`
|
|
Bor *BorConfig `json:"bor,omitempty"`
|
|
}
|
|
|
|
func (c *Config) String() string {
|
|
engine := c.getEngine()
|
|
|
|
return fmt.Sprintf("{ChainID: %v, Homestead: %v, DAO: %v, Tangerine Whistle: %v, Spurious Dragon: %v, Byzantium: %v, Constantinople: %v, Petersburg: %v, Istanbul: %v, Muir Glacier: %v, Berlin: %v, London: %v, Arrow Glacier: %v, Gray Glacier: %v, Terminal Total Difficulty: %v, Merge Netsplit: %v, Shanghai: %v, Cancun: %v, Prague: %v, Engine: %v}",
|
|
c.ChainID,
|
|
c.HomesteadBlock,
|
|
c.DAOForkBlock,
|
|
c.TangerineWhistleBlock,
|
|
c.SpuriousDragonBlock,
|
|
c.ByzantiumBlock,
|
|
c.ConstantinopleBlock,
|
|
c.PetersburgBlock,
|
|
c.IstanbulBlock,
|
|
c.MuirGlacierBlock,
|
|
c.BerlinBlock,
|
|
c.LondonBlock,
|
|
c.ArrowGlacierBlock,
|
|
c.GrayGlacierBlock,
|
|
c.TerminalTotalDifficulty,
|
|
c.MergeNetsplitBlock,
|
|
c.ShanghaiTime,
|
|
c.CancunTime,
|
|
c.PragueTime,
|
|
engine,
|
|
)
|
|
}
|
|
|
|
func (c *Config) getEngine() string {
|
|
switch {
|
|
case c.Ethash != nil:
|
|
return c.Ethash.String()
|
|
case c.Clique != nil:
|
|
return c.Clique.String()
|
|
case c.Bor != nil:
|
|
return c.Bor.String()
|
|
case c.Aura != nil:
|
|
return c.Aura.String()
|
|
default:
|
|
return "unknown"
|
|
}
|
|
}
|
|
|
|
// IsHomestead returns whether num is either equal to the homestead block or greater.
|
|
func (c *Config) IsHomestead(num uint64) bool {
|
|
return isForked(c.HomesteadBlock, num)
|
|
}
|
|
|
|
// IsDAOFork returns whether num is either equal to the DAO fork block or greater.
|
|
func (c *Config) IsDAOFork(num uint64) bool {
|
|
return isForked(c.DAOForkBlock, num)
|
|
}
|
|
|
|
// IsTangerineWhistle returns whether num is either equal to the Tangerine Whistle (EIP150) fork block or greater.
|
|
func (c *Config) IsTangerineWhistle(num uint64) bool {
|
|
return isForked(c.TangerineWhistleBlock, num)
|
|
}
|
|
|
|
// IsSpuriousDragon returns whether num is either equal to the Spurious Dragon fork block or greater.
|
|
func (c *Config) IsSpuriousDragon(num uint64) bool {
|
|
return isForked(c.SpuriousDragonBlock, num)
|
|
}
|
|
|
|
// IsByzantium returns whether num is either equal to the Byzantium fork block or greater.
|
|
func (c *Config) IsByzantium(num uint64) bool {
|
|
return isForked(c.ByzantiumBlock, num)
|
|
}
|
|
|
|
// IsConstantinople returns whether num is either equal to the Constantinople fork block or greater.
|
|
func (c *Config) IsConstantinople(num uint64) bool {
|
|
return isForked(c.ConstantinopleBlock, num)
|
|
}
|
|
|
|
// IsMuirGlacier returns whether num is either equal to the Muir Glacier (EIP-2384) fork block or greater.
|
|
func (c *Config) IsMuirGlacier(num uint64) bool {
|
|
return isForked(c.MuirGlacierBlock, num)
|
|
}
|
|
|
|
// IsPetersburg returns whether num is either
|
|
// - equal to or greater than the PetersburgBlock fork block,
|
|
// - OR is nil, and Constantinople is active
|
|
func (c *Config) IsPetersburg(num uint64) bool {
|
|
return isForked(c.PetersburgBlock, num) || c.PetersburgBlock == nil && isForked(c.ConstantinopleBlock, num)
|
|
}
|
|
|
|
// IsIstanbul returns whether num is either equal to the Istanbul fork block or greater.
|
|
func (c *Config) IsIstanbul(num uint64) bool {
|
|
return isForked(c.IstanbulBlock, num)
|
|
}
|
|
|
|
// IsBerlin returns whether num is either equal to the Berlin fork block or greater.
|
|
func (c *Config) IsBerlin(num uint64) bool {
|
|
return isForked(c.BerlinBlock, num)
|
|
}
|
|
|
|
// IsLondon returns whether num is either equal to the London fork block or greater.
|
|
func (c *Config) IsLondon(num uint64) bool {
|
|
return isForked(c.LondonBlock, num)
|
|
}
|
|
|
|
// IsArrowGlacier returns whether num is either equal to the Arrow Glacier (EIP-4345) fork block or greater.
|
|
func (c *Config) IsArrowGlacier(num uint64) bool {
|
|
return isForked(c.ArrowGlacierBlock, num)
|
|
}
|
|
|
|
// IsGrayGlacier returns whether num is either equal to the Gray Glacier (EIP-5133) fork block or greater.
|
|
func (c *Config) IsGrayGlacier(num uint64) bool {
|
|
return isForked(c.GrayGlacierBlock, num)
|
|
}
|
|
|
|
// IsShanghai returns whether time is either equal to the Shanghai fork time or greater.
|
|
func (c *Config) IsShanghai(time uint64) bool {
|
|
return isForked(c.ShanghaiTime, time)
|
|
}
|
|
|
|
// IsAgra returns whether num is either equal to the Agra fork block or greater.
|
|
// The Agra hard fork is based on the Shanghai hard fork, but it doesn't include withdrawals.
|
|
// Also Agra is activated based on the block number rather than the timestamp.
|
|
// Refer to https://forum.polygon.technology/t/pip-28-agra-hardfork
|
|
func (c *Config) IsAgra(num uint64) bool {
|
|
if c == nil || c.Bor == nil {
|
|
return false
|
|
}
|
|
return isForked(c.Bor.AgraBlock, num)
|
|
}
|
|
|
|
// IsCancun returns whether time is either equal to the Cancun fork time or greater.
|
|
func (c *Config) IsCancun(time uint64) bool {
|
|
return isForked(c.CancunTime, time)
|
|
}
|
|
|
|
// IsPrague returns whether time is either equal to the Prague fork time or greater.
|
|
func (c *Config) IsPrague(time uint64) bool {
|
|
return isForked(c.PragueTime, time)
|
|
}
|
|
|
|
func (c *Config) GetBurntContract(num uint64) *common.Address {
|
|
if len(c.BurntContract) == 0 {
|
|
return nil
|
|
}
|
|
addr := borKeyValueConfigHelper(c.BurntContract, num)
|
|
return &addr
|
|
}
|
|
|
|
func (c *Config) GetMinBlobGasPrice() uint64 {
|
|
if c.MinBlobGasPrice != nil {
|
|
return *c.MinBlobGasPrice
|
|
}
|
|
return 1 // MIN_BLOB_GASPRICE (EIP-4844)
|
|
}
|
|
|
|
func (c *Config) GetMaxBlobGasPerBlock() uint64 {
|
|
if c.MaxBlobGasPerBlock != nil {
|
|
return *c.MaxBlobGasPerBlock
|
|
}
|
|
return 786432 // MAX_BLOB_GAS_PER_BLOCK (EIP-4844)
|
|
}
|
|
|
|
func (c *Config) GetTargetBlobGasPerBlock() uint64 {
|
|
if c.TargetBlobGasPerBlock != nil {
|
|
return *c.TargetBlobGasPerBlock
|
|
}
|
|
return 393216 // TARGET_BLOB_GAS_PER_BLOCK (EIP-4844)
|
|
}
|
|
|
|
func (c *Config) GetBlobGasPriceUpdateFraction() uint64 {
|
|
if c.BlobGasPriceUpdateFraction != nil {
|
|
return *c.BlobGasPriceUpdateFraction
|
|
}
|
|
return 3338477 // BLOB_GASPRICE_UPDATE_FRACTION (EIP-4844)
|
|
}
|
|
|
|
func (c *Config) GetMaxBlobsPerBlock() uint64 {
|
|
return c.GetMaxBlobGasPerBlock() / fixedgas.BlobGasPerBlob
|
|
}
|
|
|
|
// CheckCompatible checks whether scheduled fork transitions have been imported
|
|
// with a mismatching chain configuration.
|
|
func (c *Config) CheckCompatible(newcfg *Config, height uint64) *ConfigCompatError {
|
|
bhead := height
|
|
|
|
// Iterate checkCompatible to find the lowest conflict.
|
|
var lasterr *ConfigCompatError
|
|
for {
|
|
err := c.checkCompatible(newcfg, bhead)
|
|
if err == nil || (lasterr != nil && err.RewindTo == lasterr.RewindTo) {
|
|
break
|
|
}
|
|
lasterr = err
|
|
bhead = err.RewindTo
|
|
}
|
|
return lasterr
|
|
}
|
|
|
|
type forkBlockNumber struct {
|
|
name string
|
|
blockNumber *big.Int
|
|
optional bool // if true, the fork may be nil and next fork is still allowed
|
|
}
|
|
|
|
func (c *Config) forkBlockNumbers() []forkBlockNumber {
|
|
return []forkBlockNumber{
|
|
{name: "homesteadBlock", blockNumber: c.HomesteadBlock},
|
|
{name: "daoForkBlock", blockNumber: c.DAOForkBlock, optional: true},
|
|
{name: "eip150Block", blockNumber: c.TangerineWhistleBlock},
|
|
{name: "eip155Block", blockNumber: c.SpuriousDragonBlock},
|
|
{name: "byzantiumBlock", blockNumber: c.ByzantiumBlock},
|
|
{name: "constantinopleBlock", blockNumber: c.ConstantinopleBlock},
|
|
{name: "petersburgBlock", blockNumber: c.PetersburgBlock},
|
|
{name: "istanbulBlock", blockNumber: c.IstanbulBlock},
|
|
{name: "muirGlacierBlock", blockNumber: c.MuirGlacierBlock, optional: true},
|
|
{name: "berlinBlock", blockNumber: c.BerlinBlock},
|
|
{name: "londonBlock", blockNumber: c.LondonBlock},
|
|
{name: "arrowGlacierBlock", blockNumber: c.ArrowGlacierBlock, optional: true},
|
|
{name: "grayGlacierBlock", blockNumber: c.GrayGlacierBlock, optional: true},
|
|
{name: "mergeNetsplitBlock", blockNumber: c.MergeNetsplitBlock, optional: true},
|
|
}
|
|
}
|
|
|
|
// CheckConfigForkOrder checks that we don't "skip" any forks
|
|
func (c *Config) CheckConfigForkOrder() error {
|
|
if c != nil && c.ChainID != nil && c.ChainID.Uint64() == 77 {
|
|
return nil
|
|
}
|
|
|
|
var lastFork forkBlockNumber
|
|
|
|
for _, fork := range c.forkBlockNumbers() {
|
|
if lastFork.name != "" {
|
|
// Next one must be higher number
|
|
if lastFork.blockNumber == nil && fork.blockNumber != nil {
|
|
return fmt.Errorf("unsupported fork ordering: %v not enabled, but %v enabled at %v",
|
|
lastFork.name, fork.name, fork.blockNumber)
|
|
}
|
|
if lastFork.blockNumber != nil && fork.blockNumber != nil {
|
|
if lastFork.blockNumber.Cmp(fork.blockNumber) > 0 {
|
|
return fmt.Errorf("unsupported fork ordering: %v enabled at %v, but %v enabled at %v",
|
|
lastFork.name, lastFork.blockNumber, fork.name, fork.blockNumber)
|
|
}
|
|
}
|
|
// If it was optional and not set, then ignore it
|
|
}
|
|
if !fork.optional || fork.blockNumber != nil {
|
|
lastFork = fork
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *Config) checkCompatible(newcfg *Config, head uint64) *ConfigCompatError {
|
|
// returns true if a fork scheduled at s1 cannot be rescheduled to block s2 because head is already past the fork.
|
|
incompatible := func(s1, s2 *big.Int, head uint64) bool {
|
|
return (isForked(s1, head) || isForked(s2, head)) && !numEqual(s1, s2)
|
|
}
|
|
|
|
// Ethereum mainnet forks
|
|
if incompatible(c.HomesteadBlock, newcfg.HomesteadBlock, head) {
|
|
return newCompatError("Homestead fork block", c.HomesteadBlock, newcfg.HomesteadBlock)
|
|
}
|
|
if incompatible(c.DAOForkBlock, newcfg.DAOForkBlock, head) {
|
|
return newCompatError("DAO fork block", c.DAOForkBlock, newcfg.DAOForkBlock)
|
|
}
|
|
if incompatible(c.TangerineWhistleBlock, newcfg.TangerineWhistleBlock, head) {
|
|
return newCompatError("Tangerine Whistle fork block", c.TangerineWhistleBlock, newcfg.TangerineWhistleBlock)
|
|
}
|
|
if incompatible(c.SpuriousDragonBlock, newcfg.SpuriousDragonBlock, head) {
|
|
return newCompatError("Spurious Dragon fork block", c.SpuriousDragonBlock, newcfg.SpuriousDragonBlock)
|
|
}
|
|
if c.IsSpuriousDragon(head) && !numEqual(c.ChainID, newcfg.ChainID) {
|
|
return newCompatError("EIP155 chain ID", c.SpuriousDragonBlock, newcfg.SpuriousDragonBlock)
|
|
}
|
|
if incompatible(c.ByzantiumBlock, newcfg.ByzantiumBlock, head) {
|
|
return newCompatError("Byzantium fork block", c.ByzantiumBlock, newcfg.ByzantiumBlock)
|
|
}
|
|
if incompatible(c.ConstantinopleBlock, newcfg.ConstantinopleBlock, head) {
|
|
return newCompatError("Constantinople fork block", c.ConstantinopleBlock, newcfg.ConstantinopleBlock)
|
|
}
|
|
if incompatible(c.PetersburgBlock, newcfg.PetersburgBlock, head) {
|
|
// the only case where we allow Petersburg to be set in the past is if it is equal to Constantinople
|
|
// mainly to satisfy fork ordering requirements which state that Petersburg fork be set if Constantinople fork is set
|
|
if incompatible(c.ConstantinopleBlock, newcfg.PetersburgBlock, head) {
|
|
return newCompatError("Petersburg fork block", c.PetersburgBlock, newcfg.PetersburgBlock)
|
|
}
|
|
}
|
|
if incompatible(c.IstanbulBlock, newcfg.IstanbulBlock, head) {
|
|
return newCompatError("Istanbul fork block", c.IstanbulBlock, newcfg.IstanbulBlock)
|
|
}
|
|
if incompatible(c.MuirGlacierBlock, newcfg.MuirGlacierBlock, head) {
|
|
return newCompatError("Muir Glacier fork block", c.MuirGlacierBlock, newcfg.MuirGlacierBlock)
|
|
}
|
|
if incompatible(c.BerlinBlock, newcfg.BerlinBlock, head) {
|
|
return newCompatError("Berlin fork block", c.BerlinBlock, newcfg.BerlinBlock)
|
|
}
|
|
if incompatible(c.LondonBlock, newcfg.LondonBlock, head) {
|
|
return newCompatError("London fork block", c.LondonBlock, newcfg.LondonBlock)
|
|
}
|
|
if incompatible(c.ArrowGlacierBlock, newcfg.ArrowGlacierBlock, head) {
|
|
return newCompatError("Arrow Glacier fork block", c.ArrowGlacierBlock, newcfg.ArrowGlacierBlock)
|
|
}
|
|
if incompatible(c.GrayGlacierBlock, newcfg.GrayGlacierBlock, head) {
|
|
return newCompatError("Gray Glacier fork block", c.GrayGlacierBlock, newcfg.GrayGlacierBlock)
|
|
}
|
|
if incompatible(c.MergeNetsplitBlock, newcfg.MergeNetsplitBlock, head) {
|
|
return newCompatError("Merge netsplit block", c.MergeNetsplitBlock, newcfg.MergeNetsplitBlock)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func numEqual(x, y *big.Int) bool {
|
|
if x == nil {
|
|
return y == nil
|
|
}
|
|
if y == nil {
|
|
return x == nil
|
|
}
|
|
return x.Cmp(y) == 0
|
|
}
|
|
|
|
// ConfigCompatError is raised if the locally-stored blockchain is initialised with a
|
|
// ChainConfig that would alter the past.
|
|
type ConfigCompatError struct {
|
|
What string
|
|
// block numbers of the stored and new configurations
|
|
StoredConfig, NewConfig *big.Int
|
|
// the block number to which the local chain must be rewound to correct the error
|
|
RewindTo uint64
|
|
}
|
|
|
|
func newCompatError(what string, storedblock, newblock *big.Int) *ConfigCompatError {
|
|
var rew *big.Int
|
|
switch {
|
|
case storedblock == nil:
|
|
rew = newblock
|
|
case newblock == nil || storedblock.Cmp(newblock) < 0:
|
|
rew = storedblock
|
|
default:
|
|
rew = newblock
|
|
}
|
|
err := &ConfigCompatError{what, storedblock, newblock, 0}
|
|
if rew != nil && rew.Sign() > 0 {
|
|
err.RewindTo = rew.Uint64() - 1
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (err *ConfigCompatError) Error() string {
|
|
return fmt.Sprintf("mismatching %s in database (have %d, want %d, rewindto %d)", err.What, err.StoredConfig, err.NewConfig, err.RewindTo)
|
|
}
|
|
|
|
// EthashConfig is the consensus engine configs for proof-of-work based sealing.
|
|
type EthashConfig struct{}
|
|
|
|
// String implements the stringer interface, returning the consensus engine details.
|
|
func (c *EthashConfig) String() string {
|
|
return "ethash"
|
|
}
|
|
|
|
// CliqueConfig is the consensus engine configs for proof-of-authority based sealing.
|
|
type CliqueConfig struct {
|
|
Period uint64 `json:"period"` // Number of seconds between blocks to enforce
|
|
Epoch uint64 `json:"epoch"` // Epoch length to reset votes and checkpoint
|
|
}
|
|
|
|
// String implements the stringer interface, returning the consensus engine details.
|
|
func (c *CliqueConfig) String() string {
|
|
return "clique"
|
|
}
|
|
|
|
// BorConfig is the consensus engine configs for Matic bor based sealing.
|
|
type BorConfig struct {
|
|
Period map[string]uint64 `json:"period"` // Number of seconds between blocks to enforce
|
|
ProducerDelay map[string]uint64 `json:"producerDelay"` // Number of seconds delay between two producer interval
|
|
Sprint map[string]uint64 `json:"sprint"` // Epoch length to proposer
|
|
BackupMultiplier map[string]uint64 `json:"backupMultiplier"` // Backup multiplier to determine the wiggle time
|
|
ValidatorContract string `json:"validatorContract"` // Validator set contract
|
|
StateReceiverContract string `json:"stateReceiverContract"` // State receiver contract
|
|
|
|
OverrideStateSyncRecords map[string]int `json:"overrideStateSyncRecords"` // override state records count
|
|
BlockAlloc map[string]interface{} `json:"blockAlloc"`
|
|
|
|
JaipurBlock *big.Int `json:"jaipurBlock"` // Jaipur switch block (nil = no fork, 0 = already on jaipur)
|
|
DelhiBlock *big.Int `json:"delhiBlock"` // Delhi switch block (nil = no fork, 0 = already on delhi)
|
|
IndoreBlock *big.Int `json:"indoreBlock"` // Indore switch block (nil = no fork, 0 = already on indore)
|
|
AgraBlock *big.Int `json:"agraBlock"` // Agra switch block (nil = no fork, 0 = already in agra)
|
|
StateSyncConfirmationDelay map[string]uint64 `json:"stateSyncConfirmationDelay"` // StateSync Confirmation Delay, in seconds, to calculate `to`
|
|
|
|
ParallelUniverseBlock *big.Int `json:"parallelUniverseBlock"` // TODO: update all occurrence, change name and finalize number (hardfork for block-stm related changes)
|
|
|
|
sprints sprints
|
|
}
|
|
|
|
// String implements the stringer interface, returning the consensus engine details.
|
|
func (b *BorConfig) String() string {
|
|
return "bor"
|
|
}
|
|
|
|
func (c *BorConfig) CalculateProducerDelay(number uint64) uint64 {
|
|
return borKeyValueConfigHelper(c.ProducerDelay, number)
|
|
}
|
|
|
|
func (c *BorConfig) CalculateSprint(number uint64) uint64 {
|
|
if c.sprints == nil {
|
|
c.sprints = asSprints(c.Sprint)
|
|
}
|
|
|
|
for i := 0; i < len(c.sprints)-1; i++ {
|
|
if number >= c.sprints[i].from && number < c.sprints[i+1].from {
|
|
return c.sprints[i].size
|
|
}
|
|
}
|
|
|
|
return c.sprints[len(c.sprints)-1].size
|
|
}
|
|
|
|
func (c *BorConfig) CalculateSprintCount(from, to uint64) int {
|
|
switch {
|
|
case from > to:
|
|
return 0
|
|
case from < to:
|
|
to--
|
|
}
|
|
|
|
if c.sprints == nil {
|
|
c.sprints = asSprints(c.Sprint)
|
|
}
|
|
|
|
count := uint64(0)
|
|
startCalc := from
|
|
|
|
zeroth := func(boundary uint64, size uint64) uint64 {
|
|
if boundary%size == 0 {
|
|
return 1
|
|
}
|
|
|
|
return 0
|
|
}
|
|
|
|
for i := 0; i < len(c.sprints)-1; i++ {
|
|
if startCalc >= c.sprints[i].from && startCalc < c.sprints[i+1].from {
|
|
if to >= c.sprints[i].from && to < c.sprints[i+1].from {
|
|
if startCalc == to {
|
|
return int(count + zeroth(startCalc, c.sprints[i].size))
|
|
}
|
|
return int(count + zeroth(startCalc, c.sprints[i].size) + (to-startCalc)/c.sprints[i].size)
|
|
} else {
|
|
endCalc := c.sprints[i+1].from - 1
|
|
count += zeroth(startCalc, c.sprints[i].size) + (endCalc-startCalc)/c.sprints[i].size
|
|
startCalc = endCalc + 1
|
|
}
|
|
}
|
|
}
|
|
|
|
if startCalc == to {
|
|
return int(count + zeroth(startCalc, c.sprints[len(c.sprints)-1].size))
|
|
}
|
|
|
|
return int(count + zeroth(startCalc, c.sprints[len(c.sprints)-1].size) + (to-startCalc)/c.sprints[len(c.sprints)-1].size)
|
|
}
|
|
|
|
func (c *BorConfig) CalculateBackupMultiplier(number uint64) uint64 {
|
|
return borKeyValueConfigHelper(c.BackupMultiplier, number)
|
|
}
|
|
|
|
func (c *BorConfig) CalculatePeriod(number uint64) uint64 {
|
|
return borKeyValueConfigHelper(c.Period, number)
|
|
}
|
|
|
|
func (c *BorConfig) IsJaipur(number uint64) bool {
|
|
return isForked(c.JaipurBlock, number)
|
|
}
|
|
|
|
func (c *BorConfig) IsDelhi(number uint64) bool {
|
|
return isForked(c.DelhiBlock, number)
|
|
}
|
|
|
|
func (c *BorConfig) IsIndore(number uint64) bool {
|
|
return isForked(c.IndoreBlock, number)
|
|
}
|
|
|
|
// TODO: modify this function once the block number is finalized
|
|
func (c *BorConfig) IsParallelUniverse(number uint64) bool {
|
|
if c.ParallelUniverseBlock != nil {
|
|
if c.ParallelUniverseBlock.Cmp(big.NewInt(0)) == 0 {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return isForked(c.ParallelUniverseBlock, number)
|
|
}
|
|
|
|
func (c *BorConfig) CalculateStateSyncDelay(number uint64) uint64 {
|
|
return borKeyValueConfigHelper(c.StateSyncConfirmationDelay, number)
|
|
}
|
|
|
|
func borKeyValueConfigHelper[T uint64 | common.Address](field map[string]T, number uint64) T {
|
|
fieldUint := make(map[uint64]T)
|
|
for k, v := range field {
|
|
keyUint, err := strconv.ParseUint(k, 10, 64)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
fieldUint[keyUint] = v
|
|
}
|
|
|
|
keys := common.SortedKeys(fieldUint)
|
|
|
|
for i := 0; i < len(keys)-1; i++ {
|
|
if number >= keys[i] && number < keys[i+1] {
|
|
return fieldUint[keys[i]]
|
|
}
|
|
}
|
|
|
|
return fieldUint[keys[len(keys)-1]]
|
|
}
|
|
|
|
type sprint struct {
|
|
from, size uint64
|
|
}
|
|
|
|
type sprints []sprint
|
|
|
|
func (s sprints) Len() int {
|
|
return len(s)
|
|
}
|
|
|
|
func (s sprints) Swap(i, j int) {
|
|
s[i], s[j] = s[j], s[i]
|
|
}
|
|
|
|
func (s sprints) Less(i, j int) bool {
|
|
return s[i].from < s[j].from
|
|
}
|
|
|
|
func asSprints(configSprints map[string]uint64) sprints {
|
|
sprints := make(sprints, len(configSprints))
|
|
|
|
i := 0
|
|
for key, value := range configSprints {
|
|
sprints[i].from, _ = strconv.ParseUint(key, 10, 64)
|
|
sprints[i].size = value
|
|
i++
|
|
}
|
|
|
|
sort.Sort(sprints)
|
|
|
|
return sprints
|
|
}
|
|
|
|
// Rules is syntactic sugar over Config. It can be used for functions
|
|
// that do not have or require information about the block.
|
|
//
|
|
// Rules is a one time interface meaning that it shouldn't be used in between transition
|
|
// phases.
|
|
type Rules struct {
|
|
ChainID *big.Int
|
|
IsHomestead, IsTangerineWhistle, IsSpuriousDragon bool
|
|
IsByzantium, IsConstantinople, IsPetersburg, IsIstanbul bool
|
|
IsBerlin, IsLondon, IsShanghai, IsCancun, IsPrague bool
|
|
IsAura bool
|
|
}
|
|
|
|
// Rules ensures c's ChainID is not nil and returns a new Rules instance
|
|
func (c *Config) Rules(num uint64, time uint64) *Rules {
|
|
chainID := c.ChainID
|
|
if chainID == nil {
|
|
chainID = new(big.Int)
|
|
}
|
|
|
|
return &Rules{
|
|
ChainID: new(big.Int).Set(chainID),
|
|
IsHomestead: c.IsHomestead(num),
|
|
IsTangerineWhistle: c.IsTangerineWhistle(num),
|
|
IsSpuriousDragon: c.IsSpuriousDragon(num),
|
|
IsByzantium: c.IsByzantium(num),
|
|
IsConstantinople: c.IsConstantinople(num),
|
|
IsPetersburg: c.IsPetersburg(num),
|
|
IsIstanbul: c.IsIstanbul(num),
|
|
IsBerlin: c.IsBerlin(num),
|
|
IsLondon: c.IsLondon(num),
|
|
IsShanghai: c.IsShanghai(time) || c.IsAgra(num),
|
|
IsCancun: c.IsCancun(time),
|
|
IsPrague: c.IsPrague(time),
|
|
IsAura: c.Aura != nil,
|
|
}
|
|
}
|
|
|
|
// isForked returns whether a fork scheduled at block s is active at the given head block.
|
|
func isForked(s *big.Int, head uint64) bool {
|
|
if s == nil {
|
|
return false
|
|
}
|
|
return s.Uint64() <= head
|
|
}
|