mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-10 21:11:20 +00:00
75ca6b8c76
* Initial work on integration tests * Delete subtree * Squashed 'interfaces/' content from commit 41a082ba4 git-subtree-dir: interfaces git-subtree-split: 41a082ba4bde38647325eb0416cb1da1b4ca2b12 * Add consensus interfaces * More stuff * comments * Fix compile * Squashed 'interfaces/' changes from 41a082ba4..1b13a42a7 1b13a42a7 Add chainspec to consensus interface git-subtree-dir: interfaces git-subtree-split: 1b13a42a7803f5464722867a71065c27a7ebe8c3 * Squashed 'interfaces/' changes from 1b13a42a7..93a072c4c 93a072c4c Add missing import git-subtree-dir: interfaces git-subtree-split: 93a072c4c099d169322a3a53b95e40203276820b * New consensus interfaces * More on clique * Fix tests * Squashed 'interfaces/' changes from 93a072c4c..62f4ec4b2 62f4ec4b2 Add test service for consensus engine git-subtree-dir: interfaces git-subtree-split: 62f4ec4b263107635ffa3aabd5d634af22e813c6 * Squashed 'interfaces/' changes from 62f4ec4b2..061a63543 061a63543 Fix git-subtree-dir: interfaces git-subtree-split: 061a63543babdeb51ab7e3a96dec56b2485d4389 * Configuring clique engine with toml specs - start * More toml parsing * Constructed rinkeby genesis * Simplify VerifyHeaders functions * Fix lint * Remove concurrent verification tests * Fix lint Co-authored-by: Alex Sharp <alexsharp@Alexs-MacBook-Pro.local> Co-authored-by: Alexey Sharp <alexeysharp@Alexeys-iMac.local>
265 lines
8.9 KiB
Go
265 lines
8.9 KiB
Go
package clique
|
|
|
|
import (
|
|
"bytes"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
"github.com/ledgerwatch/turbo-geth/consensus"
|
|
"github.com/ledgerwatch/turbo-geth/consensus/misc"
|
|
"github.com/ledgerwatch/turbo-geth/core/types"
|
|
"github.com/ledgerwatch/turbo-geth/log"
|
|
"github.com/ledgerwatch/turbo-geth/params"
|
|
)
|
|
|
|
// verifyHeader checks whether a header conforms to the consensus rules.The
|
|
// caller may optionally pass in a batch of parents (ascending order) to avoid
|
|
// looking those up from the database. This is useful for concurrently verifying
|
|
// a batch of new headers.
|
|
func (c *Clique) verifyHeader(chain consensus.ChainHeaderReader, header *types.Header, parents []*types.Header) error {
|
|
if header.Number == nil {
|
|
return errUnknownBlock
|
|
}
|
|
number := header.Number.Uint64()
|
|
|
|
now := time.Now()
|
|
nowUnix := now.Unix()
|
|
|
|
// Don't waste time checking blocks from the future
|
|
if header.Time > uint64(nowUnix) {
|
|
return consensus.ErrFutureBlock
|
|
}
|
|
|
|
// Checkpoint blocks need to enforce zero beneficiary
|
|
checkpoint := (number % c.config.Epoch) == 0
|
|
if checkpoint && header.Coinbase != (common.Address{}) {
|
|
return errInvalidCheckpointBeneficiary
|
|
}
|
|
|
|
// Nonces must be 0x00..0 or 0xff..f, zeroes enforced on checkpoints
|
|
if !bytes.Equal(header.Nonce[:], nonceAuthVote) && !bytes.Equal(header.Nonce[:], nonceDropVote) {
|
|
return errInvalidVote
|
|
}
|
|
|
|
if checkpoint && !bytes.Equal(header.Nonce[:], nonceDropVote) {
|
|
return errInvalidCheckpointVote
|
|
}
|
|
|
|
// Check that the extra-data contains both the vanity and signature
|
|
if len(header.Extra) < ExtraVanity {
|
|
return errMissingVanity
|
|
}
|
|
if len(header.Extra) < ExtraVanity+ExtraSeal {
|
|
return errMissingSignature
|
|
}
|
|
// Ensure that the extra-data contains a signer list on checkpoint, but none otherwise
|
|
signersBytes := len(header.Extra) - ExtraVanity - ExtraSeal
|
|
if !checkpoint && signersBytes != 0 {
|
|
return errExtraSigners
|
|
}
|
|
if checkpoint && signersBytes%common.AddressLength != 0 {
|
|
return errInvalidCheckpointSigners
|
|
}
|
|
// Ensure that the mix digest is zero as we don't have fork protection currently
|
|
if header.MixDigest != (common.Hash{}) {
|
|
return errInvalidMixDigest
|
|
}
|
|
// Ensure that the block doesn't contain any uncles which are meaningless in PoA
|
|
if header.UncleHash != uncleHash {
|
|
return errInvalidUncleHash
|
|
}
|
|
// Ensure that the block's difficulty is meaningful (may not be correct at this point)
|
|
if number > 0 {
|
|
if header.Difficulty == nil || (header.Difficulty.Cmp(diffInTurn) != 0 && header.Difficulty.Cmp(diffNoTurn) != 0) {
|
|
return errInvalidDifficulty
|
|
}
|
|
}
|
|
|
|
// If all checks passed, validate any special fields for hard forks
|
|
if err := misc.VerifyForkHashes(c.chainConfig, header, false); err != nil {
|
|
return err
|
|
}
|
|
|
|
// All basic checks passed, verify cascading fields
|
|
return c.verifyCascadingFields(chain, header, parents)
|
|
}
|
|
|
|
// verifyCascadingFields verifies all the header fields that are not standalone,
|
|
// rather depend on a batch of previous headers. The caller may optionally pass
|
|
// in a batch of parents (ascending order) to avoid looking those up from the
|
|
// database. This is useful for concurrently verifying a batch of new headers.
|
|
func (c *Clique) verifyCascadingFields(chain consensus.ChainHeaderReader, header *types.Header, parents []*types.Header) error {
|
|
// The genesis block is the always valid dead-end
|
|
number := header.Number.Uint64()
|
|
if number == 0 {
|
|
return nil
|
|
}
|
|
|
|
// Retrieve the snapshot needed to verify this header and cache it
|
|
var parent *types.Header
|
|
if len(parents) > 0 {
|
|
parent = parents[len(parents)-1]
|
|
} else {
|
|
parent = chain.GetHeader(header.ParentHash, number-1)
|
|
}
|
|
if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash {
|
|
return consensus.ErrUnknownAncestor
|
|
}
|
|
// Verify the header's EIP-1559 attributes.
|
|
if chain.Config().IsAleut(header.Number.Uint64()) {
|
|
if err := misc.VerifyEip1559Header(parent, header, chain.Config().IsAleut(parent.Number.Uint64())); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if parent.Time+c.config.Period > header.Time {
|
|
return errInvalidTimestamp
|
|
}
|
|
|
|
// Retrieve the snapshot needed to verify this header and cache it
|
|
snap, err := c.snapshot(chain, number-1, header.ParentHash, parents)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// If the block is a checkpoint block, verify the signer list
|
|
if number%c.config.Epoch == 0 {
|
|
signers := make([]byte, len(snap.Signers)*common.AddressLength)
|
|
for i, signer := range snap.signers() {
|
|
copy(signers[i*common.AddressLength:], signer[:])
|
|
}
|
|
|
|
extraSuffix := len(header.Extra) - ExtraSeal
|
|
if !bytes.Equal(header.Extra[ExtraVanity:extraSuffix], signers) {
|
|
return errMismatchingCheckpointSigners
|
|
}
|
|
}
|
|
|
|
// All basic checks passed, verify the seal and return
|
|
return c.verifySeal(chain, header, snap)
|
|
}
|
|
|
|
func (c *Clique) snapshot(chain consensus.ChainHeaderReader, number uint64, hash common.Hash, parents []*types.Header) (*Snapshot, error) {
|
|
// Search for a snapshot in memory or on disk for checkpoints
|
|
var (
|
|
headers []*types.Header
|
|
snap *Snapshot
|
|
)
|
|
for snap == nil {
|
|
// If an in-memory snapshot was found, use that
|
|
if s, ok := c.recents.Get(hash); ok {
|
|
snap = s.(*Snapshot)
|
|
break
|
|
}
|
|
// If an on-disk checkpoint snapshot can be found, use that
|
|
if number%c.snapshotConfig.CheckpointInterval == 0 {
|
|
if s, err := loadSnapshot(c.config, c.db, number, hash); err == nil {
|
|
log.Trace("Loaded voting snapshot from disk", "number", number, "hash", hash)
|
|
snap = s
|
|
break
|
|
}
|
|
}
|
|
// If we're at the genesis, snapshot the initial state. Alternatively if we're
|
|
// at a checkpoint block without a parent (light client CHT), or we have piled
|
|
// up more headers than allowed to be reorged (chain reinit from a freezer),
|
|
// consider the checkpoint trusted and snapshot it.
|
|
if number == 0 || (number%c.config.Epoch == 0 && (len(headers) > params.FullImmutabilityThreshold || chain.GetHeaderByNumber(number-1) == nil)) {
|
|
checkpoint := chain.GetHeaderByNumber(number)
|
|
if checkpoint != nil {
|
|
hash := checkpoint.Hash()
|
|
|
|
signers := make([]common.Address, (len(checkpoint.Extra)-ExtraVanity-ExtraSeal)/common.AddressLength)
|
|
for i := 0; i < len(signers); i++ {
|
|
copy(signers[i][:], checkpoint.Extra[ExtraVanity+i*common.AddressLength:])
|
|
}
|
|
snap = newSnapshot(c.config, number, hash, signers)
|
|
if err := snap.store(c.db); err != nil {
|
|
return nil, err
|
|
}
|
|
log.Info("Stored checkpoint snapshot to disk", "number", number, "hash", hash)
|
|
break
|
|
}
|
|
}
|
|
// No snapshot for this header, gather the header and move backward
|
|
var header *types.Header
|
|
if len(parents) > 0 {
|
|
// If we have explicit parents, pick from there (enforced)
|
|
header = parents[len(parents)-1]
|
|
if header.Hash() != hash || header.Number.Uint64() != number {
|
|
return nil, consensus.ErrUnknownAncestor
|
|
}
|
|
parents = parents[:len(parents)-1]
|
|
} else {
|
|
// No explicit parents (or no more left), reach out to the database
|
|
header = chain.GetHeader(hash, number)
|
|
if header == nil {
|
|
return nil, consensus.ErrUnknownAncestor
|
|
}
|
|
}
|
|
headers = append(headers, header)
|
|
number, hash = number-1, header.ParentHash
|
|
}
|
|
// Previous snapshot found, apply any pending headers on top of it
|
|
for i := 0; i < len(headers)/2; i++ {
|
|
headers[i], headers[len(headers)-1-i] = headers[len(headers)-1-i], headers[i]
|
|
}
|
|
snap, err := snap.apply(c.signatures, headers...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
c.recents.Add(snap.Hash, snap)
|
|
|
|
// If we've generated a new checkpoint snapshot, save to disk
|
|
if snap.Number%c.snapshotConfig.CheckpointInterval == 0 && len(headers) > 0 {
|
|
if err = snap.store(c.db); err != nil {
|
|
return nil, err
|
|
}
|
|
log.Trace("Stored voting snapshot to disk", "number", snap.Number, "hash", snap.Hash)
|
|
}
|
|
return snap, err
|
|
}
|
|
|
|
// verifySeal checks whether the signature contained in the header satisfies the
|
|
// consensus protocol requirements. The method accepts an optional list of parent
|
|
// headers that aren't yet part of the local blockchain to generate the snapshots
|
|
// from.
|
|
func (c *Clique) verifySeal(chain consensus.ChainHeaderReader, header *types.Header, snap *Snapshot) error {
|
|
// Verifying the genesis block is not supported
|
|
number := header.Number.Uint64()
|
|
if number == 0 {
|
|
return errUnknownBlock
|
|
}
|
|
|
|
// Resolve the authorization key and check against signers
|
|
signer, err := ecrecover(header, c.signatures)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, ok := snap.Signers[signer]; !ok {
|
|
return errUnauthorizedSigner
|
|
}
|
|
|
|
for seen, recent := range snap.Recents {
|
|
if recent == signer {
|
|
// Signer is among RecentsRLP, only fail if the current block doesn't shift it out
|
|
if limit := uint64(len(snap.Signers)/2 + 1); seen > number-limit {
|
|
return errRecentlySigned
|
|
}
|
|
}
|
|
}
|
|
|
|
// Ensure that the difficulty corresponds to the turn-ness of the signer
|
|
if !c.fakeDiff {
|
|
inturn := snap.inturn(header.Number.Uint64(), signer)
|
|
if inturn && header.Difficulty.Cmp(diffInTurn) != 0 {
|
|
return errWrongDifficulty
|
|
}
|
|
if !inturn && header.Difficulty.Cmp(diffNoTurn) != 0 {
|
|
return errWrongDifficulty
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|