consensus/ethash: implement faster difficulty calculators (#21976)
This PR adds re-written difficulty calculators, which are based on uint256. It also adds a fuzzer + oss-fuzz integration for the new fuzzer. It does differential fuzzing between the new and old calculators. Note: this PR does not actually enable the new calculators.
This commit is contained in:
parent
88c696240d
commit
efe6dd2904
@ -485,6 +485,11 @@ func calcDifficultyFrontier(time uint64, parent *types.Header) *big.Int {
|
|||||||
return diff
|
return diff
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Exported for fuzzing
|
||||||
|
var FrontierDifficultyCalulator = calcDifficultyFrontier
|
||||||
|
var HomesteadDifficultyCalulator = calcDifficultyHomestead
|
||||||
|
var DynamicDifficultyCalculator = makeDifficultyCalculator
|
||||||
|
|
||||||
// VerifySeal implements consensus.Engine, checking whether the given block satisfies
|
// VerifySeal implements consensus.Engine, checking whether the given block satisfies
|
||||||
// the PoW difficulty requirements.
|
// the PoW difficulty requirements.
|
||||||
func (ethash *Ethash) VerifySeal(chain consensus.ChainHeaderReader, header *types.Header) error {
|
func (ethash *Ethash) VerifySeal(chain consensus.ChainHeaderReader, header *types.Header) error {
|
||||||
|
@ -17,12 +17,15 @@
|
|||||||
package ethash
|
package ethash
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/binary"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
"math/rand"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/math"
|
"github.com/ethereum/go-ethereum/common/math"
|
||||||
"github.com/ethereum/go-ethereum/core/types"
|
"github.com/ethereum/go-ethereum/core/types"
|
||||||
"github.com/ethereum/go-ethereum/params"
|
"github.com/ethereum/go-ethereum/params"
|
||||||
@ -84,3 +87,102 @@ func TestCalcDifficulty(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func randSlice(min, max uint32) []byte {
|
||||||
|
var b = make([]byte, 4)
|
||||||
|
rand.Read(b)
|
||||||
|
a := binary.LittleEndian.Uint32(b)
|
||||||
|
size := min + a%(max-min)
|
||||||
|
out := make([]byte, size)
|
||||||
|
rand.Read(out)
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDifficultyCalculators(t *testing.T) {
|
||||||
|
rand.Seed(2)
|
||||||
|
for i := 0; i < 5000; i++ {
|
||||||
|
// 1 to 300 seconds diff
|
||||||
|
var timeDelta = uint64(1 + rand.Uint32()%3000)
|
||||||
|
diffBig := big.NewInt(0).SetBytes(randSlice(2, 10))
|
||||||
|
if diffBig.Cmp(params.MinimumDifficulty) < 0 {
|
||||||
|
diffBig.Set(params.MinimumDifficulty)
|
||||||
|
}
|
||||||
|
//rand.Read(difficulty)
|
||||||
|
header := &types.Header{
|
||||||
|
Difficulty: diffBig,
|
||||||
|
Number: new(big.Int).SetUint64(rand.Uint64() % 50_000_000),
|
||||||
|
Time: rand.Uint64() - timeDelta,
|
||||||
|
}
|
||||||
|
if rand.Uint32()&1 == 0 {
|
||||||
|
header.UncleHash = types.EmptyUncleHash
|
||||||
|
}
|
||||||
|
bombDelay := new(big.Int).SetUint64(rand.Uint64() % 50_000_000)
|
||||||
|
for i, pair := range []struct {
|
||||||
|
bigFn func(time uint64, parent *types.Header) *big.Int
|
||||||
|
u256Fn func(time uint64, parent *types.Header) *big.Int
|
||||||
|
}{
|
||||||
|
{FrontierDifficultyCalulator, CalcDifficultyFrontierU256},
|
||||||
|
{HomesteadDifficultyCalulator, CalcDifficultyHomesteadU256},
|
||||||
|
{DynamicDifficultyCalculator(bombDelay), MakeDifficultyCalculatorU256(bombDelay)},
|
||||||
|
} {
|
||||||
|
time := header.Time + timeDelta
|
||||||
|
want := pair.bigFn(time, header)
|
||||||
|
have := pair.u256Fn(time, header)
|
||||||
|
if want.BitLen() > 256 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if want.Cmp(have) != 0 {
|
||||||
|
t.Fatalf("pair %d: want %x have %x\nparent.Number: %x\np.Time: %x\nc.Time: %x\nBombdelay: %v\n", i, want, have,
|
||||||
|
header.Number, header.Time, time, bombDelay)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkDifficultyCalculator(b *testing.B) {
|
||||||
|
x1 := makeDifficultyCalculator(big.NewInt(1000000))
|
||||||
|
x2 := MakeDifficultyCalculatorU256(big.NewInt(1000000))
|
||||||
|
h := &types.Header{
|
||||||
|
ParentHash: common.Hash{},
|
||||||
|
UncleHash: types.EmptyUncleHash,
|
||||||
|
Difficulty: big.NewInt(0xffffff),
|
||||||
|
Number: big.NewInt(500000),
|
||||||
|
Time: 1000000,
|
||||||
|
}
|
||||||
|
b.Run("big-frontier", func(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
calcDifficultyFrontier(1000014, h)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("u256-frontier", func(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
CalcDifficultyFrontierU256(1000014, h)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("big-homestead", func(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
calcDifficultyHomestead(1000014, h)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("u256-homestead", func(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
CalcDifficultyHomesteadU256(1000014, h)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("big-generic", func(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
x1(1000014, h)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("u256-generic", func(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
x2(1000014, h)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
193
consensus/ethash/difficulty.go
Normal file
193
consensus/ethash/difficulty.go
Normal file
@ -0,0 +1,193 @@
|
|||||||
|
// Copyright 2020 The go-ethereum Authors
|
||||||
|
// This file is part of the go-ethereum library.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||||
|
// 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.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU Lesser General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU Lesser General Public License
|
||||||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
package ethash
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/core/types"
|
||||||
|
"github.com/holiman/uint256"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// frontierDurationLimit is for Frontier:
|
||||||
|
// The decision boundary on the blocktime duration used to determine
|
||||||
|
// whether difficulty should go up or down.
|
||||||
|
frontierDurationLimit = 13
|
||||||
|
// minimumDifficulty The minimum that the difficulty may ever be.
|
||||||
|
minimumDifficulty = 131072
|
||||||
|
// expDiffPeriod is the exponential difficulty period
|
||||||
|
expDiffPeriodUint = 100000
|
||||||
|
// difficultyBoundDivisorBitShift is the bound divisor of the difficulty (2048),
|
||||||
|
// This constant is the right-shifts to use for the division.
|
||||||
|
difficultyBoundDivisor = 11
|
||||||
|
)
|
||||||
|
|
||||||
|
// CalcDifficultyFrontierU256 is the difficulty adjustment algorithm. It returns the
|
||||||
|
// difficulty that a new block should have when created at time given the parent
|
||||||
|
// block's time and difficulty. The calculation uses the Frontier rules.
|
||||||
|
func CalcDifficultyFrontierU256(time uint64, parent *types.Header) *big.Int {
|
||||||
|
/*
|
||||||
|
Algorithm
|
||||||
|
block_diff = pdiff + pdiff / 2048 * (1 if time - ptime < 13 else -1) + int(2^((num // 100000) - 2))
|
||||||
|
|
||||||
|
Where:
|
||||||
|
- pdiff = parent.difficulty
|
||||||
|
- ptime = parent.time
|
||||||
|
- time = block.timestamp
|
||||||
|
- num = block.number
|
||||||
|
*/
|
||||||
|
|
||||||
|
pDiff := uint256.NewInt()
|
||||||
|
pDiff.SetFromBig(parent.Difficulty) // pDiff: pdiff
|
||||||
|
adjust := pDiff.Clone()
|
||||||
|
adjust.Rsh(adjust, difficultyBoundDivisor) // adjust: pDiff / 2048
|
||||||
|
|
||||||
|
if time-parent.Time < frontierDurationLimit {
|
||||||
|
pDiff.Add(pDiff, adjust)
|
||||||
|
} else {
|
||||||
|
pDiff.Sub(pDiff, adjust)
|
||||||
|
}
|
||||||
|
if pDiff.LtUint64(minimumDifficulty) {
|
||||||
|
pDiff.SetUint64(minimumDifficulty)
|
||||||
|
}
|
||||||
|
// 'pdiff' now contains:
|
||||||
|
// pdiff + pdiff / 2048 * (1 if time - ptime < 13 else -1)
|
||||||
|
|
||||||
|
if periodCount := (parent.Number.Uint64() + 1) / expDiffPeriodUint; periodCount > 1 {
|
||||||
|
// diff = diff + 2^(periodCount - 2)
|
||||||
|
expDiff := adjust.SetOne()
|
||||||
|
expDiff.Lsh(expDiff, uint(periodCount-2)) // expdiff: 2 ^ (periodCount -2)
|
||||||
|
pDiff.Add(pDiff, expDiff)
|
||||||
|
}
|
||||||
|
return pDiff.ToBig()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CalcDifficultyHomesteadU256 is the difficulty adjustment algorithm. It returns
|
||||||
|
// the difficulty that a new block should have when created at time given the
|
||||||
|
// parent block's time and difficulty. The calculation uses the Homestead rules.
|
||||||
|
func CalcDifficultyHomesteadU256(time uint64, parent *types.Header) *big.Int {
|
||||||
|
/*
|
||||||
|
https://github.com/ethereum/EIPs/blob/master/EIPS/eip-2.md
|
||||||
|
Algorithm:
|
||||||
|
block_diff = pdiff + pdiff / 2048 * max(1 - (time - ptime) / 10, -99) + 2 ^ int((num / 100000) - 2))
|
||||||
|
|
||||||
|
Our modification, to use unsigned ints:
|
||||||
|
block_diff = pdiff - pdiff / 2048 * max((time - ptime) / 10 - 1, 99) + 2 ^ int((num / 100000) - 2))
|
||||||
|
|
||||||
|
Where:
|
||||||
|
- pdiff = parent.difficulty
|
||||||
|
- ptime = parent.time
|
||||||
|
- time = block.timestamp
|
||||||
|
- num = block.number
|
||||||
|
*/
|
||||||
|
|
||||||
|
pDiff := uint256.NewInt()
|
||||||
|
pDiff.SetFromBig(parent.Difficulty) // pDiff: pdiff
|
||||||
|
adjust := pDiff.Clone()
|
||||||
|
adjust.Rsh(adjust, difficultyBoundDivisor) // adjust: pDiff / 2048
|
||||||
|
|
||||||
|
x := (time - parent.Time) / 10 // (time - ptime) / 10)
|
||||||
|
var neg = true
|
||||||
|
if x == 0 {
|
||||||
|
x = 1
|
||||||
|
neg = false
|
||||||
|
} else if x >= 100 {
|
||||||
|
x = 99
|
||||||
|
} else {
|
||||||
|
x = x - 1
|
||||||
|
}
|
||||||
|
z := new(uint256.Int).SetUint64(x)
|
||||||
|
adjust.Mul(adjust, z) // adjust: (pdiff / 2048) * max((time - ptime) / 10 - 1, 99)
|
||||||
|
if neg {
|
||||||
|
pDiff.Sub(pDiff, adjust) // pdiff - pdiff / 2048 * max((time - ptime) / 10 - 1, 99)
|
||||||
|
} else {
|
||||||
|
pDiff.Add(pDiff, adjust) // pdiff + pdiff / 2048 * max((time - ptime) / 10 - 1, 99)
|
||||||
|
}
|
||||||
|
if pDiff.LtUint64(minimumDifficulty) {
|
||||||
|
pDiff.SetUint64(minimumDifficulty)
|
||||||
|
}
|
||||||
|
// for the exponential factor, a.k.a "the bomb"
|
||||||
|
// diff = diff + 2^(periodCount - 2)
|
||||||
|
if periodCount := (1 + parent.Number.Uint64()) / expDiffPeriodUint; periodCount > 1 {
|
||||||
|
expFactor := adjust.Lsh(adjust.SetOne(), uint(periodCount-2))
|
||||||
|
pDiff.Add(pDiff, expFactor)
|
||||||
|
}
|
||||||
|
return pDiff.ToBig()
|
||||||
|
}
|
||||||
|
|
||||||
|
// MakeDifficultyCalculatorU256 creates a difficultyCalculator with the given bomb-delay.
|
||||||
|
// the difficulty is calculated with Byzantium rules, which differs from Homestead in
|
||||||
|
// how uncles affect the calculation
|
||||||
|
func MakeDifficultyCalculatorU256(bombDelay *big.Int) func(time uint64, parent *types.Header) *big.Int {
|
||||||
|
// Note, the calculations below looks at the parent number, which is 1 below
|
||||||
|
// the block number. Thus we remove one from the delay given
|
||||||
|
bombDelayFromParent := bombDelay.Uint64() - 1
|
||||||
|
return func(time uint64, parent *types.Header) *big.Int {
|
||||||
|
/*
|
||||||
|
https://github.com/ethereum/EIPs/issues/100
|
||||||
|
pDiff = parent.difficulty
|
||||||
|
BLOCK_DIFF_FACTOR = 9
|
||||||
|
a = pDiff + (pDiff // BLOCK_DIFF_FACTOR) * adj_factor
|
||||||
|
b = min(parent.difficulty, MIN_DIFF)
|
||||||
|
child_diff = max(a,b )
|
||||||
|
*/
|
||||||
|
x := (time - parent.Time) / 9 // (block_timestamp - parent_timestamp) // 9
|
||||||
|
c := uint64(1) // if parent.unclehash == emptyUncleHashHash
|
||||||
|
if parent.UncleHash != types.EmptyUncleHash {
|
||||||
|
c = 2
|
||||||
|
}
|
||||||
|
xNeg := x >= c
|
||||||
|
if xNeg {
|
||||||
|
// x is now _negative_ adjustment factor
|
||||||
|
x = x - c // - ( (t-p)/p -( 2 or 1) )
|
||||||
|
} else {
|
||||||
|
x = c - x // (2 or 1) - (t-p)/9
|
||||||
|
}
|
||||||
|
if x > 99 {
|
||||||
|
x = 99 // max(x, 99)
|
||||||
|
}
|
||||||
|
// parent_diff + (parent_diff / 2048 * max((2 if len(parent.uncles) else 1) - ((timestamp - parent.timestamp) // 9), -99))
|
||||||
|
y := new(uint256.Int)
|
||||||
|
y.SetFromBig(parent.Difficulty) // y: p_diff
|
||||||
|
pDiff := y.Clone() // pdiff: p_diff
|
||||||
|
z := new(uint256.Int).SetUint64(x) //z : +-adj_factor (either pos or negative)
|
||||||
|
y.Rsh(y, difficultyBoundDivisor) // y: p__diff / 2048
|
||||||
|
z.Mul(y, z) // z: (p_diff / 2048 ) * (+- adj_factor)
|
||||||
|
|
||||||
|
if xNeg {
|
||||||
|
y.Sub(pDiff, z) // y: parent_diff + parent_diff/2048 * adjustment_factor
|
||||||
|
} else {
|
||||||
|
y.Add(pDiff, z) // y: parent_diff + parent_diff/2048 * adjustment_factor
|
||||||
|
}
|
||||||
|
// minimum difficulty can ever be (before exponential factor)
|
||||||
|
if y.LtUint64(minimumDifficulty) {
|
||||||
|
y.SetUint64(minimumDifficulty)
|
||||||
|
}
|
||||||
|
// calculate a fake block number for the ice-age delay
|
||||||
|
// Specification: https://eips.ethereum.org/EIPS/eip-1234
|
||||||
|
var pNum = parent.Number.Uint64()
|
||||||
|
if pNum >= bombDelayFromParent {
|
||||||
|
if fakeBlockNumber := pNum - bombDelayFromParent; fakeBlockNumber >= 2*expDiffPeriodUint {
|
||||||
|
z.SetOne()
|
||||||
|
z.Lsh(z, uint(fakeBlockNumber/expDiffPeriodUint-2))
|
||||||
|
y.Add(z, y)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return y.ToBig()
|
||||||
|
}
|
||||||
|
}
|
@ -57,6 +57,7 @@ compile_fuzzer tests/fuzzers/txfetcher Fuzz fuzzTxfetcher
|
|||||||
compile_fuzzer tests/fuzzers/rlp Fuzz fuzzRlp
|
compile_fuzzer tests/fuzzers/rlp Fuzz fuzzRlp
|
||||||
compile_fuzzer tests/fuzzers/trie Fuzz fuzzTrie
|
compile_fuzzer tests/fuzzers/trie Fuzz fuzzTrie
|
||||||
compile_fuzzer tests/fuzzers/stacktrie Fuzz fuzzStackTrie
|
compile_fuzzer tests/fuzzers/stacktrie Fuzz fuzzStackTrie
|
||||||
|
compile_fuzzer tests/fuzzers/difficulty Fuzz fuzzDifficulty
|
||||||
|
|
||||||
compile_fuzzer tests/fuzzers/bls12381 FuzzG1Add fuzz_g1_add
|
compile_fuzzer tests/fuzzers/bls12381 FuzzG1Add fuzz_g1_add
|
||||||
compile_fuzzer tests/fuzzers/bls12381 FuzzG1Mul fuzz_g1_mul
|
compile_fuzzer tests/fuzzers/bls12381 FuzzG1Mul fuzz_g1_mul
|
||||||
|
23
tests/fuzzers/difficulty/debug/main.go
Normal file
23
tests/fuzzers/difficulty/debug/main.go
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/tests/fuzzers/difficulty"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if len(os.Args) != 2 {
|
||||||
|
fmt.Fprintf(os.Stderr, "Usage: debug <file>")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
crasher := os.Args[1]
|
||||||
|
data, err := ioutil.ReadFile(crasher)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "error loading crasher %v: %v", crasher, err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
difficulty.Fuzz(data)
|
||||||
|
}
|
145
tests/fuzzers/difficulty/difficulty-fuzz.go
Normal file
145
tests/fuzzers/difficulty/difficulty-fuzz.go
Normal file
@ -0,0 +1,145 @@
|
|||||||
|
// Copyright 2020 The go-ethereum Authors
|
||||||
|
// This file is part of the go-ethereum library.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||||
|
// 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.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU Lesser General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU Lesser General Public License
|
||||||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
package difficulty
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/consensus/ethash"
|
||||||
|
"github.com/ethereum/go-ethereum/core/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type fuzzer struct {
|
||||||
|
input io.Reader
|
||||||
|
exhausted bool
|
||||||
|
debugging bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *fuzzer) read(size int) []byte {
|
||||||
|
out := make([]byte, size)
|
||||||
|
if _, err := f.input.Read(out); err != nil {
|
||||||
|
f.exhausted = true
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *fuzzer) readSlice(min, max int) []byte {
|
||||||
|
var a uint16
|
||||||
|
binary.Read(f.input, binary.LittleEndian, &a)
|
||||||
|
size := min + int(a)%(max-min)
|
||||||
|
out := make([]byte, size)
|
||||||
|
if _, err := f.input.Read(out); err != nil {
|
||||||
|
f.exhausted = true
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *fuzzer) readUint64(min, max uint64) uint64 {
|
||||||
|
if min == max {
|
||||||
|
return min
|
||||||
|
}
|
||||||
|
var a uint64
|
||||||
|
if err := binary.Read(f.input, binary.LittleEndian, &a); err != nil {
|
||||||
|
f.exhausted = true
|
||||||
|
}
|
||||||
|
a = min + a%(max-min)
|
||||||
|
return a
|
||||||
|
}
|
||||||
|
func (f *fuzzer) readBool() bool {
|
||||||
|
return f.read(1)[0]&0x1 == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// The function must return
|
||||||
|
// 1 if the fuzzer should increase priority of the
|
||||||
|
// given input during subsequent fuzzing (for example, the input is lexically
|
||||||
|
// correct and was parsed successfully);
|
||||||
|
// -1 if the input must not be added to corpus even if gives new coverage; and
|
||||||
|
// 0 otherwise
|
||||||
|
// other values are reserved for future use.
|
||||||
|
func Fuzz(data []byte) int {
|
||||||
|
f := fuzzer{
|
||||||
|
input: bytes.NewReader(data),
|
||||||
|
exhausted: false,
|
||||||
|
}
|
||||||
|
return f.fuzz()
|
||||||
|
}
|
||||||
|
|
||||||
|
var minDifficulty = big.NewInt(0x2000)
|
||||||
|
|
||||||
|
type calculator func(time uint64, parent *types.Header) *big.Int
|
||||||
|
|
||||||
|
func (f *fuzzer) fuzz() int {
|
||||||
|
// A parent header
|
||||||
|
header := &types.Header{}
|
||||||
|
if f.readBool() {
|
||||||
|
header.UncleHash = types.EmptyUncleHash
|
||||||
|
}
|
||||||
|
// Difficulty can range between 0x2000 (2 bytes) and up to 32 bytes
|
||||||
|
{
|
||||||
|
diff := new(big.Int).SetBytes(f.readSlice(2, 32))
|
||||||
|
if diff.Cmp(minDifficulty) < 0 {
|
||||||
|
diff.Set(minDifficulty)
|
||||||
|
}
|
||||||
|
header.Difficulty = diff
|
||||||
|
}
|
||||||
|
// Number can range between 0 and up to 32 bytes (but not so that the child exceeds it)
|
||||||
|
{
|
||||||
|
// However, if we use astronomic numbers, then the bomb exp karatsuba calculation
|
||||||
|
// in the legacy methods)
|
||||||
|
// times out, so we limit it to fit within reasonable bounds
|
||||||
|
number := new(big.Int).SetBytes(f.readSlice(0, 4)) // 4 bytes: 32 bits: block num max 4 billion
|
||||||
|
header.Number = number
|
||||||
|
}
|
||||||
|
// Both parent and child time must fit within uint64
|
||||||
|
var time uint64
|
||||||
|
{
|
||||||
|
childTime := f.readUint64(1, 0xFFFFFFFFFFFFFFFF)
|
||||||
|
//fmt.Printf("childTime: %x\n",childTime)
|
||||||
|
delta := f.readUint64(1, childTime)
|
||||||
|
//fmt.Printf("delta: %v\n", delta)
|
||||||
|
pTime := childTime - delta
|
||||||
|
header.Time = pTime
|
||||||
|
time = childTime
|
||||||
|
}
|
||||||
|
// Bomb delay will never exceed uint64
|
||||||
|
bombDelay := new(big.Int).SetUint64(f.readUint64(1, 0xFFFFFFFFFFFFFFFe))
|
||||||
|
|
||||||
|
if f.exhausted {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, pair := range []struct {
|
||||||
|
bigFn calculator
|
||||||
|
u256Fn calculator
|
||||||
|
}{
|
||||||
|
{ethash.FrontierDifficultyCalulator, ethash.CalcDifficultyFrontierU256},
|
||||||
|
{ethash.HomesteadDifficultyCalulator, ethash.CalcDifficultyHomesteadU256},
|
||||||
|
{ethash.DynamicDifficultyCalculator(bombDelay), ethash.MakeDifficultyCalculatorU256(bombDelay)},
|
||||||
|
} {
|
||||||
|
want := pair.bigFn(time, header)
|
||||||
|
have := pair.u256Fn(time, header)
|
||||||
|
if want.Cmp(have) != 0 {
|
||||||
|
panic(fmt.Sprintf("pair %d: want %x have %x\nparent.Number: %x\np.Time: %x\nc.Time: %x\nBombdelay: %v\n", i, want, have,
|
||||||
|
header.Number, header.Time, time, bombDelay))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user