mirror of
https://github.com/status-im/op-geth.git
synced 2025-01-21 20:19:40 +00:00
2bacf36d80
bmt is a new package that provides hashers for binary merkle tree hashes on size-limited chunks. the main motivation is that using BMT hash as the chunk hash of the swarm hash offers logsize inclusion proofs for arbitrary files on a 32-byte resolution completely viable to use in challenges on the blockchain.
563 lines
15 KiB
Go
563 lines
15 KiB
Go
// Copyright 2017 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 bmt provides a binary merkle tree implementation
|
|
package bmt
|
|
|
|
import (
|
|
"fmt"
|
|
"hash"
|
|
"io"
|
|
"strings"
|
|
"sync"
|
|
"sync/atomic"
|
|
)
|
|
|
|
/*
|
|
Binary Merkle Tree Hash is a hash function over arbitrary datachunks of limited size
|
|
It is defined as the root hash of the binary merkle tree built over fixed size segments
|
|
of the underlying chunk using any base hash function (e.g keccak 256 SHA3)
|
|
|
|
It is used as the chunk hash function in swarm which in turn is the basis for the
|
|
128 branching swarm hash http://swarm-guide.readthedocs.io/en/latest/architecture.html#swarm-hash
|
|
|
|
The BMT is optimal for providing compact inclusion proofs, i.e. prove that a
|
|
segment is a substring of a chunk starting at a particular offset
|
|
The size of the underlying segments is fixed at 32 bytes (called the resolution
|
|
of the BMT hash), the EVM word size to optimize for on-chain BMT verification
|
|
as well as the hash size optimal for inclusion proofs in the merkle tree of the swarm hash.
|
|
|
|
Two implementations are provided:
|
|
|
|
* RefHasher is optimized for code simplicity and meant as a reference implementation
|
|
* Hasher is optimized for speed taking advantage of concurrency with minimalistic
|
|
control structure to coordinate the concurrent routines
|
|
It implements the ChunkHash interface as well as the go standard hash.Hash interface
|
|
|
|
*/
|
|
|
|
const (
|
|
// DefaultSegmentCount is the maximum number of segments of the underlying chunk
|
|
DefaultSegmentCount = 128 // Should be equal to storage.DefaultBranches
|
|
// DefaultPoolSize is the maximum number of bmt trees used by the hashers, i.e,
|
|
// the maximum number of concurrent BMT hashing operations performed by the same hasher
|
|
DefaultPoolSize = 8
|
|
)
|
|
|
|
// BaseHasher is a hash.Hash constructor function used for the base hash of the BMT.
|
|
type BaseHasher func() hash.Hash
|
|
|
|
// Hasher a reusable hasher for fixed maximum size chunks representing a BMT
|
|
// implements the hash.Hash interface
|
|
// reuse pool of Tree-s for amortised memory allocation and resource control
|
|
// supports order-agnostic concurrent segment writes
|
|
// as well as sequential read and write
|
|
// can not be called concurrently on more than one chunk
|
|
// can be further appended after Sum
|
|
// Reset gives back the Tree to the pool and guaranteed to leave
|
|
// the tree and itself in a state reusable for hashing a new chunk
|
|
type Hasher struct {
|
|
pool *TreePool // BMT resource pool
|
|
bmt *Tree // prebuilt BMT resource for flowcontrol and proofs
|
|
blocksize int // segment size (size of hash) also for hash.Hash
|
|
count int // segment count
|
|
size int // for hash.Hash same as hashsize
|
|
cur int // cursor position for righmost currently open chunk
|
|
segment []byte // the rightmost open segment (not complete)
|
|
depth int // index of last level
|
|
result chan []byte // result channel
|
|
hash []byte // to record the result
|
|
max int32 // max segments for SegmentWriter interface
|
|
blockLength []byte // The block length that needes to be added in Sum
|
|
}
|
|
|
|
// New creates a reusable Hasher
|
|
// implements the hash.Hash interface
|
|
// pulls a new Tree from a resource pool for hashing each chunk
|
|
func New(p *TreePool) *Hasher {
|
|
return &Hasher{
|
|
pool: p,
|
|
depth: depth(p.SegmentCount),
|
|
size: p.SegmentSize,
|
|
blocksize: p.SegmentSize,
|
|
count: p.SegmentCount,
|
|
result: make(chan []byte),
|
|
}
|
|
}
|
|
|
|
// Node is a reuseable segment hasher representing a node in a BMT
|
|
// it allows for continued writes after a Sum
|
|
// and is left in completely reusable state after Reset
|
|
type Node struct {
|
|
level, index int // position of node for information/logging only
|
|
initial bool // first and last node
|
|
root bool // whether the node is root to a smaller BMT
|
|
isLeft bool // whether it is left side of the parent double segment
|
|
unbalanced bool // indicates if a node has only the left segment
|
|
parent *Node // BMT connections
|
|
state int32 // atomic increment impl concurrent boolean toggle
|
|
left, right []byte
|
|
}
|
|
|
|
// NewNode constructor for segment hasher nodes in the BMT
|
|
func NewNode(level, index int, parent *Node) *Node {
|
|
return &Node{
|
|
parent: parent,
|
|
level: level,
|
|
index: index,
|
|
initial: index == 0,
|
|
isLeft: index%2 == 0,
|
|
}
|
|
}
|
|
|
|
// TreePool provides a pool of Trees used as resources by Hasher
|
|
// a Tree popped from the pool is guaranteed to have clean state
|
|
// for hashing a new chunk
|
|
// Hasher Reset releases the Tree to the pool
|
|
type TreePool struct {
|
|
lock sync.Mutex
|
|
c chan *Tree
|
|
hasher BaseHasher
|
|
SegmentSize int
|
|
SegmentCount int
|
|
Capacity int
|
|
count int
|
|
}
|
|
|
|
// NewTreePool creates a Tree pool with hasher, segment size, segment count and capacity
|
|
// on GetTree it reuses free Trees or creates a new one if size is not reached
|
|
func NewTreePool(hasher BaseHasher, segmentCount, capacity int) *TreePool {
|
|
return &TreePool{
|
|
c: make(chan *Tree, capacity),
|
|
hasher: hasher,
|
|
SegmentSize: hasher().Size(),
|
|
SegmentCount: segmentCount,
|
|
Capacity: capacity,
|
|
}
|
|
}
|
|
|
|
// Drain drains the pool uptil it has no more than n resources
|
|
func (self *TreePool) Drain(n int) {
|
|
self.lock.Lock()
|
|
defer self.lock.Unlock()
|
|
for len(self.c) > n {
|
|
<-self.c
|
|
self.count--
|
|
}
|
|
}
|
|
|
|
// Reserve is blocking until it returns an available Tree
|
|
// it reuses free Trees or creates a new one if size is not reached
|
|
func (self *TreePool) Reserve() *Tree {
|
|
self.lock.Lock()
|
|
defer self.lock.Unlock()
|
|
var t *Tree
|
|
if self.count == self.Capacity {
|
|
return <-self.c
|
|
}
|
|
select {
|
|
case t = <-self.c:
|
|
default:
|
|
t = NewTree(self.hasher, self.SegmentSize, self.SegmentCount)
|
|
self.count++
|
|
}
|
|
return t
|
|
}
|
|
|
|
// Release gives back a Tree to the pool.
|
|
// This Tree is guaranteed to be in reusable state
|
|
// does not need locking
|
|
func (self *TreePool) Release(t *Tree) {
|
|
self.c <- t // can never fail but...
|
|
}
|
|
|
|
// Tree is a reusable control structure representing a BMT
|
|
// organised in a binary tree
|
|
// Hasher uses a TreePool to pick one for each chunk hash
|
|
// the Tree is 'locked' while not in the pool
|
|
type Tree struct {
|
|
leaves []*Node
|
|
}
|
|
|
|
// Draw draws the BMT (badly)
|
|
func (self *Tree) Draw(hash []byte, d int) string {
|
|
var left, right []string
|
|
var anc []*Node
|
|
for i, n := range self.leaves {
|
|
left = append(left, fmt.Sprintf("%v", hashstr(n.left)))
|
|
if i%2 == 0 {
|
|
anc = append(anc, n.parent)
|
|
}
|
|
right = append(right, fmt.Sprintf("%v", hashstr(n.right)))
|
|
}
|
|
anc = self.leaves
|
|
var hashes [][]string
|
|
for l := 0; len(anc) > 0; l++ {
|
|
var nodes []*Node
|
|
hash := []string{""}
|
|
for i, n := range anc {
|
|
hash = append(hash, fmt.Sprintf("%v|%v", hashstr(n.left), hashstr(n.right)))
|
|
if i%2 == 0 && n.parent != nil {
|
|
nodes = append(nodes, n.parent)
|
|
}
|
|
}
|
|
hash = append(hash, "")
|
|
hashes = append(hashes, hash)
|
|
anc = nodes
|
|
}
|
|
hashes = append(hashes, []string{"", fmt.Sprintf("%v", hashstr(hash)), ""})
|
|
total := 60
|
|
del := " "
|
|
var rows []string
|
|
for i := len(hashes) - 1; i >= 0; i-- {
|
|
var textlen int
|
|
hash := hashes[i]
|
|
for _, s := range hash {
|
|
textlen += len(s)
|
|
}
|
|
if total < textlen {
|
|
total = textlen + len(hash)
|
|
}
|
|
delsize := (total - textlen) / (len(hash) - 1)
|
|
if delsize > len(del) {
|
|
delsize = len(del)
|
|
}
|
|
row := fmt.Sprintf("%v: %v", len(hashes)-i-1, strings.Join(hash, del[:delsize]))
|
|
rows = append(rows, row)
|
|
|
|
}
|
|
rows = append(rows, strings.Join(left, " "))
|
|
rows = append(rows, strings.Join(right, " "))
|
|
return strings.Join(rows, "\n") + "\n"
|
|
}
|
|
|
|
// NewTree initialises the Tree by building up the nodes of a BMT
|
|
// segment size is stipulated to be the size of the hash
|
|
// segmentCount needs to be positive integer and does not need to be
|
|
// a power of two and can even be an odd number
|
|
// segmentSize * segmentCount determines the maximum chunk size
|
|
// hashed using the tree
|
|
func NewTree(hasher BaseHasher, segmentSize, segmentCount int) *Tree {
|
|
n := NewNode(0, 0, nil)
|
|
n.root = true
|
|
prevlevel := []*Node{n}
|
|
// iterate over levels and creates 2^level nodes
|
|
level := 1
|
|
count := 2
|
|
for d := 1; d <= depth(segmentCount); d++ {
|
|
nodes := make([]*Node, count)
|
|
for i := 0; i < len(nodes); i++ {
|
|
var parent *Node
|
|
parent = prevlevel[i/2]
|
|
t := NewNode(level, i, parent)
|
|
nodes[i] = t
|
|
}
|
|
prevlevel = nodes
|
|
level++
|
|
count *= 2
|
|
}
|
|
// the datanode level is the nodes on the last level where
|
|
return &Tree{
|
|
leaves: prevlevel,
|
|
}
|
|
}
|
|
|
|
// methods needed by hash.Hash
|
|
|
|
// Size returns the size
|
|
func (self *Hasher) Size() int {
|
|
return self.size
|
|
}
|
|
|
|
// BlockSize returns the block size
|
|
func (self *Hasher) BlockSize() int {
|
|
return self.blocksize
|
|
}
|
|
|
|
// Sum returns the hash of the buffer
|
|
// hash.Hash interface Sum method appends the byte slice to the underlying
|
|
// data before it calculates and returns the hash of the chunk
|
|
func (self *Hasher) Sum(b []byte) (r []byte) {
|
|
t := self.bmt
|
|
i := self.cur
|
|
n := t.leaves[i]
|
|
j := i
|
|
// must run strictly before all nodes calculate
|
|
// datanodes are guaranteed to have a parent
|
|
if len(self.segment) > self.size && i > 0 && n.parent != nil {
|
|
n = n.parent
|
|
} else {
|
|
i *= 2
|
|
}
|
|
d := self.finalise(n, i)
|
|
self.writeSegment(j, self.segment, d)
|
|
c := <-self.result
|
|
self.releaseTree()
|
|
|
|
// sha3(length + BMT(pure_chunk))
|
|
if self.blockLength == nil {
|
|
return c
|
|
}
|
|
res := self.pool.hasher()
|
|
res.Reset()
|
|
res.Write(self.blockLength)
|
|
res.Write(c)
|
|
return res.Sum(nil)
|
|
}
|
|
|
|
// Hasher implements the SwarmHash interface
|
|
|
|
// Hash waits for the hasher result and returns it
|
|
// caller must call this on a BMT Hasher being written to
|
|
func (self *Hasher) Hash() []byte {
|
|
return <-self.result
|
|
}
|
|
|
|
// Hasher implements the io.Writer interface
|
|
|
|
// Write fills the buffer to hash
|
|
// with every full segment complete launches a hasher go routine
|
|
// that shoots up the BMT
|
|
func (self *Hasher) Write(b []byte) (int, error) {
|
|
l := len(b)
|
|
if l <= 0 {
|
|
return 0, nil
|
|
}
|
|
s := self.segment
|
|
i := self.cur
|
|
count := (self.count + 1) / 2
|
|
need := self.count*self.size - self.cur*2*self.size
|
|
size := self.size
|
|
if need > size {
|
|
size *= 2
|
|
}
|
|
if l < need {
|
|
need = l
|
|
}
|
|
// calculate missing bit to complete current open segment
|
|
rest := size - len(s)
|
|
if need < rest {
|
|
rest = need
|
|
}
|
|
s = append(s, b[:rest]...)
|
|
need -= rest
|
|
// read full segments and the last possibly partial segment
|
|
for need > 0 && i < count-1 {
|
|
// push all finished chunks we read
|
|
self.writeSegment(i, s, self.depth)
|
|
need -= size
|
|
if need < 0 {
|
|
size += need
|
|
}
|
|
s = b[rest : rest+size]
|
|
rest += size
|
|
i++
|
|
}
|
|
self.segment = s
|
|
self.cur = i
|
|
// otherwise, we can assume len(s) == 0, so all buffer is read and chunk is not yet full
|
|
return l, nil
|
|
}
|
|
|
|
// Hasher implements the io.ReaderFrom interface
|
|
|
|
// ReadFrom reads from io.Reader and appends to the data to hash using Write
|
|
// it reads so that chunk to hash is maximum length or reader reaches EOF
|
|
// caller must Reset the hasher prior to call
|
|
func (self *Hasher) ReadFrom(r io.Reader) (m int64, err error) {
|
|
bufsize := self.size*self.count - self.size*self.cur - len(self.segment)
|
|
buf := make([]byte, bufsize)
|
|
var read int
|
|
for {
|
|
var n int
|
|
n, err = r.Read(buf)
|
|
read += n
|
|
if err == io.EOF || read == len(buf) {
|
|
hash := self.Sum(buf[:n])
|
|
if read == len(buf) {
|
|
err = NewEOC(hash)
|
|
}
|
|
break
|
|
}
|
|
if err != nil {
|
|
break
|
|
}
|
|
n, err = self.Write(buf[:n])
|
|
if err != nil {
|
|
break
|
|
}
|
|
}
|
|
return int64(read), err
|
|
}
|
|
|
|
// Reset needs to be called before writing to the hasher
|
|
func (self *Hasher) Reset() {
|
|
self.getTree()
|
|
self.blockLength = nil
|
|
}
|
|
|
|
// Hasher implements the SwarmHash interface
|
|
|
|
// ResetWithLength needs to be called before writing to the hasher
|
|
// the argument is supposed to be the byte slice binary representation of
|
|
// the legth of the data subsumed under the hash
|
|
func (self *Hasher) ResetWithLength(l []byte) {
|
|
self.Reset()
|
|
self.blockLength = l
|
|
|
|
}
|
|
|
|
// Release gives back the Tree to the pool whereby it unlocks
|
|
// it resets tree, segment and index
|
|
func (self *Hasher) releaseTree() {
|
|
if self.bmt != nil {
|
|
n := self.bmt.leaves[self.cur]
|
|
for ; n != nil; n = n.parent {
|
|
n.unbalanced = false
|
|
if n.parent != nil {
|
|
n.root = false
|
|
}
|
|
}
|
|
self.pool.Release(self.bmt)
|
|
self.bmt = nil
|
|
|
|
}
|
|
self.cur = 0
|
|
self.segment = nil
|
|
}
|
|
|
|
func (self *Hasher) writeSegment(i int, s []byte, d int) {
|
|
h := self.pool.hasher()
|
|
n := self.bmt.leaves[i]
|
|
|
|
if len(s) > self.size && n.parent != nil {
|
|
go func() {
|
|
h.Reset()
|
|
h.Write(s)
|
|
s = h.Sum(nil)
|
|
|
|
if n.root {
|
|
self.result <- s
|
|
return
|
|
}
|
|
self.run(n.parent, h, d, n.index, s)
|
|
}()
|
|
return
|
|
}
|
|
go self.run(n, h, d, i*2, s)
|
|
}
|
|
|
|
func (self *Hasher) run(n *Node, h hash.Hash, d int, i int, s []byte) {
|
|
isLeft := i%2 == 0
|
|
for {
|
|
if isLeft {
|
|
n.left = s
|
|
} else {
|
|
n.right = s
|
|
}
|
|
if !n.unbalanced && n.toggle() {
|
|
return
|
|
}
|
|
if !n.unbalanced || !isLeft || i == 0 && d == 0 {
|
|
h.Reset()
|
|
h.Write(n.left)
|
|
h.Write(n.right)
|
|
s = h.Sum(nil)
|
|
|
|
} else {
|
|
s = append(n.left, n.right...)
|
|
}
|
|
|
|
self.hash = s
|
|
if n.root {
|
|
self.result <- s
|
|
return
|
|
}
|
|
|
|
isLeft = n.isLeft
|
|
n = n.parent
|
|
i++
|
|
}
|
|
}
|
|
|
|
// getTree obtains a BMT resource by reserving one from the pool
|
|
func (self *Hasher) getTree() *Tree {
|
|
if self.bmt != nil {
|
|
return self.bmt
|
|
}
|
|
t := self.pool.Reserve()
|
|
self.bmt = t
|
|
return t
|
|
}
|
|
|
|
// atomic bool toggle implementing a concurrent reusable 2-state object
|
|
// atomic addint with %2 implements atomic bool toggle
|
|
// it returns true if the toggler just put it in the active/waiting state
|
|
func (self *Node) toggle() bool {
|
|
return atomic.AddInt32(&self.state, 1)%2 == 1
|
|
}
|
|
|
|
func hashstr(b []byte) string {
|
|
end := len(b)
|
|
if end > 4 {
|
|
end = 4
|
|
}
|
|
return fmt.Sprintf("%x", b[:end])
|
|
}
|
|
|
|
func depth(n int) (d int) {
|
|
for l := (n - 1) / 2; l > 0; l /= 2 {
|
|
d++
|
|
}
|
|
return d
|
|
}
|
|
|
|
// finalise is following the zigzags on the tree belonging
|
|
// to the final datasegment
|
|
func (self *Hasher) finalise(n *Node, i int) (d int) {
|
|
isLeft := i%2 == 0
|
|
for {
|
|
// when the final segment's path is going via left segments
|
|
// the incoming data is pushed to the parent upon pulling the left
|
|
// we do not need toogle the state since this condition is
|
|
// detectable
|
|
n.unbalanced = isLeft
|
|
n.right = nil
|
|
if n.initial {
|
|
n.root = true
|
|
return d
|
|
}
|
|
isLeft = n.isLeft
|
|
n = n.parent
|
|
d++
|
|
}
|
|
}
|
|
|
|
// EOC (end of chunk) implements the error interface
|
|
type EOC struct {
|
|
Hash []byte // read the hash of the chunk off the error
|
|
}
|
|
|
|
// Error returns the error string
|
|
func (self *EOC) Error() string {
|
|
return fmt.Sprintf("hasher limit reached, chunk hash: %x", self.Hash)
|
|
}
|
|
|
|
// NewEOC creates new end of chunk error with the hash
|
|
func NewEOC(hash []byte) *EOC {
|
|
return &EOC{hash}
|
|
}
|