forked from TrueCloudLab/tzhash
d891a9c591
Provide default implementations in gf127 package and all optimizations in subpackages. This way it will be easier to use from a client.
92 lines
2 KiB
Go
92 lines
2 KiB
Go
package tz
|
|
|
|
import (
|
|
"github.com/nspcc-dev/tzhash/gf127"
|
|
)
|
|
|
|
type digestp struct {
|
|
x [4]gf127.GF127
|
|
}
|
|
|
|
// New returns a new hash.Hash computing the Tillich-Zémor checksum.
|
|
func newPure() *digestp {
|
|
d := new(digestp)
|
|
d.Reset()
|
|
return d
|
|
}
|
|
|
|
func (d *digestp) Sum(in []byte) []byte {
|
|
// Make a copy of d so that caller can keep writing and summing.
|
|
d0 := *d
|
|
h := d0.checkSum()
|
|
return append(in, h[:]...)
|
|
}
|
|
|
|
func (d *digestp) checkSum() [hashSize]byte {
|
|
return d.byteArray()
|
|
}
|
|
|
|
func (d *digestp) byteArray() (b [hashSize]byte) {
|
|
for i := 0; i < 4; i++ {
|
|
t := d.x[i].ByteArray()
|
|
copy(b[i*16:], t[:])
|
|
}
|
|
return
|
|
}
|
|
|
|
func (d *digestp) Reset() {
|
|
d.x[0] = gf127.GF127{1, 0}
|
|
d.x[1] = gf127.GF127{0, 0}
|
|
d.x[2] = gf127.GF127{0, 0}
|
|
d.x[3] = gf127.GF127{1, 0}
|
|
}
|
|
|
|
func (d *digestp) Write(data []byte) (n int, err error) {
|
|
n = len(data)
|
|
tmp := new(gf127.GF127)
|
|
for _, b := range data {
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x80 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x40 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x20 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x10 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x08 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x04 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x02 != 0, tmp)
|
|
mulBitRightPure(&d.x[0], &d.x[1], &d.x[2], &d.x[3], b&0x01 != 0, tmp)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (d *digestp) Size() int {
|
|
return hashSize
|
|
}
|
|
|
|
func (d *digestp) BlockSize() int {
|
|
return hashBlockSize
|
|
}
|
|
|
|
func mulBitRightPure(c00, c01, c10, c11 *gf127.GF127, bit bool, tmp *gf127.GF127) {
|
|
if bit {
|
|
*tmp = *c00
|
|
gf127.Mul10(c00, c00)
|
|
gf127.Add(c00, c01, c00)
|
|
gf127.Mul11(tmp, tmp)
|
|
gf127.Add(c01, tmp, c01)
|
|
|
|
*tmp = *c10
|
|
gf127.Mul10(c10, c10)
|
|
gf127.Add(c10, c11, c10)
|
|
gf127.Mul11(tmp, tmp)
|
|
gf127.Add(c11, tmp, c11)
|
|
} else {
|
|
*tmp = *c00
|
|
gf127.Mul10(c00, c00)
|
|
gf127.Add(c00, c01, c00)
|
|
*c01 = *tmp
|
|
|
|
*tmp = *c10
|
|
gf127.Mul10(c10, c10)
|
|
gf127.Add(c10, c11, c10)
|
|
*c11 = *tmp
|
|
}
|
|
}
|