2019-05-27 13:51:49 +00:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"hash"
|
2022-12-04 10:23:44 +00:00
|
|
|
"sync"
|
2021-04-19 21:58:05 +00:00
|
|
|
|
|
|
|
"golang.org/x/crypto/sha3"
|
2019-05-27 13:51:49 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// keccakState wraps sha3.state. In addition to the usual hash methods, it also supports
|
|
|
|
// Read to get a variable amount of data from the hash state. Read is faster than Sum
|
|
|
|
// because it doesn't copy the internal state, but also modifies the internal state.
|
|
|
|
type keccakState interface {
|
|
|
|
hash.Hash
|
|
|
|
Read([]byte) (int, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
type Hasher struct {
|
|
|
|
Sha keccakState
|
|
|
|
}
|
|
|
|
|
2022-12-04 10:23:44 +00:00
|
|
|
var hashersPool = sync.Pool{
|
|
|
|
New: func() any {
|
|
|
|
return &Hasher{Sha: sha3.NewLegacyKeccak256().(keccakState)}
|
|
|
|
},
|
|
|
|
}
|
2019-05-27 13:51:49 +00:00
|
|
|
|
|
|
|
func NewHasher() *Hasher {
|
2022-12-04 10:23:44 +00:00
|
|
|
h := hashersPool.Get().(*Hasher)
|
|
|
|
h.Sha.Reset()
|
2019-05-27 13:51:49 +00:00
|
|
|
return h
|
|
|
|
}
|
2022-12-04 10:23:44 +00:00
|
|
|
func ReturnHasherToPool(h *Hasher) { hashersPool.Put(h) }
|
2019-05-27 13:51:49 +00:00
|
|
|
|
|
|
|
func HashData(data []byte) (Hash, error) {
|
|
|
|
h := NewHasher()
|
|
|
|
defer ReturnHasherToPool(h)
|
|
|
|
|
|
|
|
_, err := h.Sha.Write(data)
|
|
|
|
if err != nil {
|
|
|
|
return Hash{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var buf Hash
|
|
|
|
_, err = h.Sha.Read(buf[:])
|
|
|
|
if err != nil {
|
|
|
|
return Hash{}, err
|
|
|
|
}
|
|
|
|
return buf, nil
|
|
|
|
}
|