2017-04-05 14:34:41 +00:00
|
|
|
// Package rollinghash/buzhash implements buzhash as described by
|
|
|
|
// https://en.wikipedia.org/wiki/Rolling_hash#Cyclic_polynomial
|
|
|
|
|
|
|
|
package buzhash64
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
import (
|
|
|
|
"math/rand"
|
|
|
|
|
|
|
|
"github.com/chmduquesne/rollinghash"
|
|
|
|
)
|
|
|
|
|
|
|
|
var defaultHashes [256]uint64
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
defaultHashes = GenerateHashes(1)
|
2017-04-05 14:34:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// The size of the checksum.
|
|
|
|
const Size = 8
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// Buzhash64 is a digest which satisfies the rollinghash.Hash64 interface.
|
|
|
|
// It implements the cyclic polynomial algorithm
|
|
|
|
// https://en.wikipedia.org/wiki/Rolling_hash#Cyclic_polynomial
|
|
|
|
type Buzhash64 struct {
|
2017-04-05 14:34:41 +00:00
|
|
|
sum uint64
|
|
|
|
nRotate uint
|
|
|
|
nRotateComplement uint // redundant, but pre-computed to spare an operation
|
|
|
|
|
|
|
|
// window is treated like a circular buffer, where the oldest element
|
|
|
|
// is indicated by d.oldest
|
|
|
|
window []byte
|
|
|
|
oldest int
|
|
|
|
bytehash [256]uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset resets the Hash to its initial state.
|
2017-12-29 11:38:00 +00:00
|
|
|
func (d *Buzhash64) Reset() {
|
2017-04-05 14:34:41 +00:00
|
|
|
d.window = d.window[:0]
|
|
|
|
d.oldest = 0
|
|
|
|
d.sum = 0
|
|
|
|
}
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// GenerateHashes generates a list of hashes to use with buzhash
|
|
|
|
func GenerateHashes(seed int64) (res [256]uint64) {
|
|
|
|
random := rand.New(rand.NewSource(seed))
|
|
|
|
used := make(map[uint64]bool)
|
|
|
|
for i, _ := range res {
|
|
|
|
x := uint64(random.Int63())
|
|
|
|
for used[x] {
|
|
|
|
x = uint64(random.Int63())
|
|
|
|
}
|
|
|
|
used[x] = true
|
|
|
|
res[i] = x
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
// New returns a buzhash based on a list of hashes provided by a call to
|
|
|
|
// GenerateHashes, seeded with the default value 1.
|
|
|
|
func New() *Buzhash64 {
|
|
|
|
return NewFromUint64Array(defaultHashes)
|
2017-04-05 14:34:41 +00:00
|
|
|
}
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// NewFromUint64Array returns a buzhash based on the provided table uint64 values.
|
|
|
|
func NewFromUint64Array(b [256]uint64) *Buzhash64 {
|
|
|
|
return &Buzhash64{
|
2017-04-05 14:34:41 +00:00
|
|
|
sum: 0,
|
2017-12-29 11:38:00 +00:00
|
|
|
window: make([]byte, 1, rollinghash.DefaultWindowCap),
|
2017-04-05 14:34:41 +00:00
|
|
|
oldest: 0,
|
|
|
|
bytehash: b,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// Size is 8 bytes
|
|
|
|
func (d *Buzhash64) Size() int { return Size }
|
2017-04-05 14:34:41 +00:00
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// BlockSize is 1 byte
|
|
|
|
func (d *Buzhash64) BlockSize() int { return 1 }
|
2017-04-05 14:34:41 +00:00
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// Write (re)initializes the rolling window with the input byte slice and
|
|
|
|
// adds its data to the digest.
|
|
|
|
func (d *Buzhash64) Write(data []byte) (int, error) {
|
2017-04-05 14:34:41 +00:00
|
|
|
// Copy the window, avoiding allocations where possible
|
2017-12-29 11:38:00 +00:00
|
|
|
l := len(data)
|
|
|
|
if l == 0 {
|
|
|
|
l = 1
|
|
|
|
}
|
|
|
|
if len(d.window) != l {
|
|
|
|
if cap(d.window) >= l {
|
|
|
|
d.window = d.window[:l]
|
2017-04-05 14:34:41 +00:00
|
|
|
} else {
|
2017-12-29 11:38:00 +00:00
|
|
|
d.window = make([]byte, l)
|
2017-04-05 14:34:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
copy(d.window, data)
|
|
|
|
|
|
|
|
for _, c := range d.window {
|
|
|
|
d.sum = d.sum<<1 | d.sum>>63
|
|
|
|
d.sum ^= d.bytehash[int(c)]
|
|
|
|
}
|
|
|
|
d.nRotate = uint(len(d.window)) % 64
|
|
|
|
d.nRotateComplement = 64 - d.nRotate
|
|
|
|
return len(d.window), nil
|
|
|
|
}
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// Sum64 returns the hash as a uint64
|
|
|
|
func (d *Buzhash64) Sum64() uint64 {
|
2017-04-05 14:34:41 +00:00
|
|
|
return d.sum
|
|
|
|
}
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// Sum returns the hash as a byte slice
|
|
|
|
func (d *Buzhash64) Sum(b []byte) []byte {
|
2017-04-05 14:34:41 +00:00
|
|
|
v := d.Sum64()
|
|
|
|
return append(b, byte(v>>56), byte(v>>48), byte(v>>40), byte(v>>32), byte(v>>24), byte(v>>16), byte(v>>8), byte(v))
|
|
|
|
}
|
|
|
|
|
2017-12-29 11:38:00 +00:00
|
|
|
// Roll updates the checksum of the window from the entering byte. You
|
|
|
|
// MUST initialize a window with Write() before calling this method.
|
|
|
|
func (d *Buzhash64) Roll(c byte) {
|
2017-04-05 14:34:41 +00:00
|
|
|
// extract the entering/leaving bytes and update the circular buffer.
|
|
|
|
hn := d.bytehash[int(c)]
|
|
|
|
h0 := d.bytehash[int(d.window[d.oldest])]
|
|
|
|
|
|
|
|
d.window[d.oldest] = c
|
|
|
|
l := len(d.window)
|
|
|
|
d.oldest += 1
|
|
|
|
if d.oldest >= l {
|
|
|
|
d.oldest = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
d.sum = (d.sum<<1 | d.sum>>63) ^ (h0<<d.nRotate | h0>>d.nRotateComplement) ^ hn
|
|
|
|
}
|