mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-22 04:24:05 +00:00
tree_ssz: naive implementation of treeHash
as seen in https://github.com/ethereum/eth2.0-specs/pull/120
This commit is contained in:
parent
998771e7e8
commit
786337e930
112
beacon_chain/tree_hash.nim
Normal file
112
beacon_chain/tree_hash.nim
Normal file
@ -0,0 +1,112 @@
|
||||
import nimcrypto, eth_common, endians, sequtils, algorithm
|
||||
|
||||
# Sample treehash implementation based on:
|
||||
# https://github.com/ethereum/eth2.0-specs/pull/120
|
||||
# Probably wrong - the spec is pretty bare-bones and no test vectors yet
|
||||
|
||||
const CHUNK_SIZE = 128
|
||||
|
||||
# XXX varargs openarray, anyone?
|
||||
func hash(a: openArray[byte]): array[32, byte] =
|
||||
var h: blake2_512
|
||||
h.init()
|
||||
h.update(a)
|
||||
var tmp = h.finish().data
|
||||
copyMem(result.addr, tmp.addr, 32)
|
||||
|
||||
func hash(a, b: openArray[byte]): array[32, byte] =
|
||||
var h: blake2_512
|
||||
h.init()
|
||||
h.update(a)
|
||||
h.update(b)
|
||||
var tmp = h.finish().data
|
||||
copyMem(result.addr, tmp.addr, 32)
|
||||
|
||||
func nextPowerOf2(v: uint32): uint32 =
|
||||
result = v - 1
|
||||
result = result or (result shr 1)
|
||||
result = result or (result shr 2)
|
||||
result = result or (result shr 4)
|
||||
result = result or (result shr 8)
|
||||
result = result or (result shr 16)
|
||||
inc result
|
||||
|
||||
func roundUpTo(v, to: int): int =
|
||||
## Round up to an even boundary of `to`
|
||||
((v + to - 1) div to) * to
|
||||
|
||||
# Concatenate a list of homogeneous objects into data and pad it
|
||||
proc listToGlob(lst: seq[seq[byte]]): seq[byte] =
|
||||
for x in lst:
|
||||
var y = x
|
||||
y.setLen(nextPowerOf2(len(x).uint32))
|
||||
result.add(y)
|
||||
|
||||
# Pad to chunksize
|
||||
result.setLen(result.len().roundUpTo(CHUNK_SIZE))
|
||||
|
||||
# XXX: er, how is this _actually_ done?
|
||||
func empty(T: typedesc): T = discard
|
||||
const emptyChunk = @(empty(array[CHUNK_SIZE, byte]))
|
||||
|
||||
proc merkleHash(lst: seq[seq[byte]]): array[32, byte] =
|
||||
## Merkle tree hash of a list of items
|
||||
# XXX: seq-of-seq looks weird...
|
||||
|
||||
# Turn list into padded data
|
||||
var data = listToGlob(lst)
|
||||
|
||||
# Store length of list (to compensate for non-bijectiveness of padding)
|
||||
var dataLen: array[32, byte]
|
||||
var lstLen = uint64(len(lst))
|
||||
bigEndian64(dataLen[32-8].addr, lstLen.addr)
|
||||
|
||||
# Divide into chunks
|
||||
var chunkz: seq[seq[byte]]
|
||||
for i in countup(0, data.len - 1, CHUNK_SIZE):
|
||||
chunkz.add data[i..<i + CHUNK_SIZE]
|
||||
|
||||
while chunkz.len() > 1:
|
||||
if chunkz.len() mod 2 == 1:
|
||||
chunkz.add emptyChunk
|
||||
for i in 0..<(chunkz.len div 2):
|
||||
# As tradition dictates - one feature, at least one nim bug:
|
||||
# https://github.com/nim-lang/Nim/issues/9684
|
||||
let tmp = @(hash(chunkz[i * 2], chunkz[i * 2 + 1]))
|
||||
chunkz[i] = tmp
|
||||
|
||||
chunkz.setLen(chunkz.len div 2)
|
||||
|
||||
result = hash(chunkz[0], dataLen)
|
||||
|
||||
proc treeHash*(x: SomeInteger): seq[byte] =
|
||||
var v: array[x.sizeof, byte]
|
||||
copyMem(v.addr, x.unsafeAddr, x.sizeof)
|
||||
|
||||
var res: array[x.sizeof, byte]
|
||||
when x.sizeof == 8: bigEndian64(res.addr, v.addr)
|
||||
elif x.sizeof == 4: bigEndian32(res.addr, v.addr)
|
||||
elif x.sizeof == 2: bigEndian16(res.addr, v.addr)
|
||||
elif x.sizeof == 1: res = v
|
||||
else: {.fatal: "boink: " & $x.sizeof .}
|
||||
result = @res
|
||||
|
||||
proc treeHash*(x: EthAddress): seq[byte] = @x
|
||||
proc treeHash*(x: MDigest): seq[byte] = @(x.data)
|
||||
proc treeHash*(x: seq[byte]): seq[byte] = @(hash(x)) # XXX: hash96 also!
|
||||
proc treeHash*[T: seq](x: T): seq[byte] =
|
||||
var tmp: seq[seq[byte]]
|
||||
for v in x:
|
||||
tmp.add treeHash(v)
|
||||
result = merkleHash(tmp)
|
||||
|
||||
proc treeHash*[T](x: T): seq[byte] =
|
||||
# XXX: could probaby compile-time-macro-sort fields...
|
||||
var fields: seq[tuple[name: string, value: seq[byte]]]
|
||||
for name, field in x.fieldPairs:
|
||||
fields.add (name, treeHash(field))
|
||||
|
||||
var tmp: seq[byte]
|
||||
for name, value in fields.sortedByIt(it.name):
|
||||
tmp.add value.value
|
||||
result = @(hash(tmp))
|
Loading…
x
Reference in New Issue
Block a user