2020-06-30 03:36:45 +01:00
|
|
|
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2021-07-21 17:23:04 +01:00
|
|
|
// Package deephash hashes a Go value recursively, in a predictable order,
|
|
|
|
// without looping. The hash is only valid within the lifetime of a program.
|
|
|
|
// Users should not store the hash on disk or send it over the network.
|
|
|
|
// The hash is sufficiently strong and unique such that
|
|
|
|
// Hash(x) == Hash(y) is an appropriate replacement for x == y.
|
2021-07-03 05:30:29 +01:00
|
|
|
//
|
util/deephash: remove unnecessary formatting for structs and slices (#2571)
The index for every struct field or slice element and
the number of fields for the struct is unncessary.
The hashing of Go values is unambiguous because every type (except maps)
encodes in a parsable manner. So long as we know the type information,
we could theoretically decode every value (except for maps).
At a high level:
* numbers are encoded as fixed-width records according to precision.
* strings (and AppendTo output) are encoded with a fixed-width length,
followed by the contents of the buffer.
* slices are prefixed by a fixed-width length, followed by the encoding
of each value. So long as we know the type of each element, we could
theoretically decode each element.
* arrays are encoded just like slices, but elide the length
since it is determined from the Go type.
* maps are encoded first with a byte indicating whether it is a cycle.
If a cycle, it is followed by a fixed-width index for the pointer,
otherwise followed by the SHA-256 hash of its contents. The encoding of maps
is not decodeable, but a SHA-256 hash is sufficient to avoid ambiguities.
* interfaces are encoded first with a byte indicating whether it is nil.
If not nil, it is followed by a fixed-width index for the type,
and then the encoding for the underlying value. Having the type be encoded
first ensures that the value could theoretically be decoded next.
* pointers are encoded first with a byte indicating whether it is
1) nil, 2) a cycle, or 3) newly seen. If a cycle, it is followed by
a fixed-width index for the pointer. If newly seen, it is followed by
the encoding for the pointed-at value.
Removing unnecessary details speeds up hashing:
name old time/op new time/op delta
Hash-8 76.0µs ± 1% 55.8µs ± 2% -26.62% (p=0.000 n=10+10)
HashMapAcyclic-8 61.9µs ± 0% 62.0µs ± 0% ~ (p=0.666 n=9+9)
TailcfgNode-8 10.2µs ± 1% 7.5µs ± 1% -26.90% (p=0.000 n=10+9)
HashArray-8 1.07µs ± 1% 0.70µs ± 1% -34.67% (p=0.000 n=10+9)
Signed-off-by: Joe Tsai <joetsai@digital-static.net>
2021-08-04 04:35:57 +01:00
|
|
|
// The definition of equality is identical to reflect.DeepEqual except:
|
|
|
|
// * Floating-point values are compared based on the raw bits,
|
|
|
|
// which means that NaNs (with the same bit pattern) are treated as equal.
|
|
|
|
// * Types which implement interface { AppendTo([]byte) []byte } use
|
|
|
|
// the AppendTo method to produce a textual representation of the value.
|
|
|
|
// Thus, two values are equal if AppendTo produces the same bytes.
|
|
|
|
//
|
|
|
|
// WARNING: This package, like most of the tailscale.com Go module,
|
|
|
|
// should be considered Tailscale-internal; we make no API promises.
|
2021-05-11 20:09:25 +01:00
|
|
|
package deephash
|
2020-06-28 18:58:21 +01:00
|
|
|
|
|
|
|
import (
|
2021-05-10 21:29:56 +01:00
|
|
|
"bufio"
|
2020-06-28 18:58:21 +01:00
|
|
|
"crypto/sha256"
|
2021-07-06 06:13:33 +01:00
|
|
|
"encoding/binary"
|
2021-05-19 19:51:21 +01:00
|
|
|
"encoding/hex"
|
2020-06-28 18:58:21 +01:00
|
|
|
"fmt"
|
2021-05-11 21:17:12 +01:00
|
|
|
"hash"
|
2021-07-06 05:28:54 +01:00
|
|
|
"math"
|
2020-06-28 18:58:21 +01:00
|
|
|
"reflect"
|
2021-05-11 21:17:12 +01:00
|
|
|
"sync"
|
2021-07-21 17:23:04 +01:00
|
|
|
"time"
|
2021-07-22 23:22:48 +01:00
|
|
|
"unsafe"
|
2020-06-28 18:58:21 +01:00
|
|
|
)
|
|
|
|
|
util/deephash: remove unnecessary formatting for structs and slices (#2571)
The index for every struct field or slice element and
the number of fields for the struct is unncessary.
The hashing of Go values is unambiguous because every type (except maps)
encodes in a parsable manner. So long as we know the type information,
we could theoretically decode every value (except for maps).
At a high level:
* numbers are encoded as fixed-width records according to precision.
* strings (and AppendTo output) are encoded with a fixed-width length,
followed by the contents of the buffer.
* slices are prefixed by a fixed-width length, followed by the encoding
of each value. So long as we know the type of each element, we could
theoretically decode each element.
* arrays are encoded just like slices, but elide the length
since it is determined from the Go type.
* maps are encoded first with a byte indicating whether it is a cycle.
If a cycle, it is followed by a fixed-width index for the pointer,
otherwise followed by the SHA-256 hash of its contents. The encoding of maps
is not decodeable, but a SHA-256 hash is sufficient to avoid ambiguities.
* interfaces are encoded first with a byte indicating whether it is nil.
If not nil, it is followed by a fixed-width index for the type,
and then the encoding for the underlying value. Having the type be encoded
first ensures that the value could theoretically be decoded next.
* pointers are encoded first with a byte indicating whether it is
1) nil, 2) a cycle, or 3) newly seen. If a cycle, it is followed by
a fixed-width index for the pointer. If newly seen, it is followed by
the encoding for the pointed-at value.
Removing unnecessary details speeds up hashing:
name old time/op new time/op delta
Hash-8 76.0µs ± 1% 55.8µs ± 2% -26.62% (p=0.000 n=10+10)
HashMapAcyclic-8 61.9µs ± 0% 62.0µs ± 0% ~ (p=0.666 n=9+9)
TailcfgNode-8 10.2µs ± 1% 7.5µs ± 1% -26.90% (p=0.000 n=10+9)
HashArray-8 1.07µs ± 1% 0.70µs ± 1% -34.67% (p=0.000 n=10+9)
Signed-off-by: Joe Tsai <joetsai@digital-static.net>
2021-08-04 04:35:57 +01:00
|
|
|
// There is much overlap between the theory of serialization and hashing.
|
2021-08-24 15:36:48 +01:00
|
|
|
// A hash (useful for determining equality) can be produced by printing a value
|
util/deephash: remove unnecessary formatting for structs and slices (#2571)
The index for every struct field or slice element and
the number of fields for the struct is unncessary.
The hashing of Go values is unambiguous because every type (except maps)
encodes in a parsable manner. So long as we know the type information,
we could theoretically decode every value (except for maps).
At a high level:
* numbers are encoded as fixed-width records according to precision.
* strings (and AppendTo output) are encoded with a fixed-width length,
followed by the contents of the buffer.
* slices are prefixed by a fixed-width length, followed by the encoding
of each value. So long as we know the type of each element, we could
theoretically decode each element.
* arrays are encoded just like slices, but elide the length
since it is determined from the Go type.
* maps are encoded first with a byte indicating whether it is a cycle.
If a cycle, it is followed by a fixed-width index for the pointer,
otherwise followed by the SHA-256 hash of its contents. The encoding of maps
is not decodeable, but a SHA-256 hash is sufficient to avoid ambiguities.
* interfaces are encoded first with a byte indicating whether it is nil.
If not nil, it is followed by a fixed-width index for the type,
and then the encoding for the underlying value. Having the type be encoded
first ensures that the value could theoretically be decoded next.
* pointers are encoded first with a byte indicating whether it is
1) nil, 2) a cycle, or 3) newly seen. If a cycle, it is followed by
a fixed-width index for the pointer. If newly seen, it is followed by
the encoding for the pointed-at value.
Removing unnecessary details speeds up hashing:
name old time/op new time/op delta
Hash-8 76.0µs ± 1% 55.8µs ± 2% -26.62% (p=0.000 n=10+10)
HashMapAcyclic-8 61.9µs ± 0% 62.0µs ± 0% ~ (p=0.666 n=9+9)
TailcfgNode-8 10.2µs ± 1% 7.5µs ± 1% -26.90% (p=0.000 n=10+9)
HashArray-8 1.07µs ± 1% 0.70µs ± 1% -34.67% (p=0.000 n=10+9)
Signed-off-by: Joe Tsai <joetsai@digital-static.net>
2021-08-04 04:35:57 +01:00
|
|
|
// and hashing the output. The format must:
|
|
|
|
// * be deterministic such that the same value hashes to the same output, and
|
|
|
|
// * be parsable such that the same value can be reproduced by the output.
|
|
|
|
//
|
|
|
|
// The logic below hashes a value by printing it to a hash.Hash.
|
|
|
|
// To be parsable, it assumes that we know the Go type of each value:
|
|
|
|
// * scalar types (e.g., bool or int32) are printed as fixed-width fields.
|
|
|
|
// * list types (e.g., strings, slices, and AppendTo buffers) are prefixed
|
|
|
|
// by a fixed-width length field, followed by the contents of the list.
|
|
|
|
// * slices, arrays, and structs print each element/field consecutively.
|
|
|
|
// * interfaces print with a 1-byte prefix indicating whether it is nil.
|
|
|
|
// If non-nil, it is followed by a fixed-width field of the type index,
|
|
|
|
// followed by the format of the underlying value.
|
|
|
|
// * pointers print with a 1-byte prefix indicating whether the pointer is
|
|
|
|
// 1) nil, 2) previously seen, or 3) newly seen. Previously seen pointers are
|
|
|
|
// followed by a fixed-width field with the index of the previous pointer.
|
|
|
|
// Newly seen pointers are followed by the format of the underlying value.
|
|
|
|
// * maps print with a 1-byte prefix indicating whether the map pointer is
|
|
|
|
// 1) nil, 2) previously seen, or 3) newly seen. Previously seen pointers
|
|
|
|
// are followed by a fixed-width field of the index of the previous pointer.
|
|
|
|
// Newly seen maps are printed as a fixed-width field with the XOR of the
|
|
|
|
// hash of every map entry. With a sufficiently strong hash, this value is
|
|
|
|
// theoretically "parsable" by looking up the hash in a magical map that
|
|
|
|
// returns the set of entries for that given hash.
|
|
|
|
|
2021-07-06 06:13:33 +01:00
|
|
|
const scratchSize = 128
|
|
|
|
|
2021-07-05 05:25:15 +01:00
|
|
|
// hasher is reusable state for hashing a value.
|
|
|
|
// Get one via hasherPool.
|
|
|
|
type hasher struct {
|
2021-07-22 23:22:48 +01:00
|
|
|
h hash.Hash
|
|
|
|
bw *bufio.Writer
|
|
|
|
scratch [scratchSize]byte
|
|
|
|
visitStack visitStack
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
|
|
|
|
2021-08-03 05:29:14 +01:00
|
|
|
func (h *hasher) reset() {
|
|
|
|
if h.h == nil {
|
|
|
|
h.h = sha256.New()
|
|
|
|
}
|
|
|
|
if h.bw == nil {
|
|
|
|
h.bw = bufio.NewWriterSize(h.h, h.h.BlockSize())
|
|
|
|
}
|
|
|
|
h.bw.Flush()
|
|
|
|
h.h.Reset()
|
2021-07-07 06:37:32 +01:00
|
|
|
}
|
|
|
|
|
2021-07-20 06:49:51 +01:00
|
|
|
// Sum is an opaque checksum type that is comparable.
|
|
|
|
type Sum struct {
|
|
|
|
sum [sha256.Size]byte
|
|
|
|
}
|
|
|
|
|
2021-08-03 05:29:14 +01:00
|
|
|
func (s1 *Sum) xor(s2 Sum) {
|
|
|
|
for i := 0; i < sha256.Size; i++ {
|
|
|
|
s1.sum[i] ^= s2.sum[i]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-20 06:49:51 +01:00
|
|
|
func (s Sum) String() string {
|
|
|
|
return hex.EncodeToString(s.sum[:])
|
|
|
|
}
|
|
|
|
|
2021-07-21 17:23:04 +01:00
|
|
|
var (
|
2022-06-16 21:21:32 +01:00
|
|
|
seedOnce sync.Once
|
|
|
|
seed uint64
|
2021-07-21 17:23:04 +01:00
|
|
|
)
|
|
|
|
|
2022-06-16 21:21:32 +01:00
|
|
|
func initSeed() {
|
|
|
|
seed = uint64(time.Now().UnixNano())
|
|
|
|
}
|
|
|
|
|
2021-08-03 05:29:14 +01:00
|
|
|
func (h *hasher) sum() (s Sum) {
|
2021-07-05 05:25:15 +01:00
|
|
|
h.bw.Flush()
|
2021-07-06 06:13:33 +01:00
|
|
|
// Sum into scratch & copy out, as hash.Hash is an interface
|
|
|
|
// so the slice necessarily escapes, and there's no sha256
|
|
|
|
// concrete type exported and we don't want the 'hash' result
|
|
|
|
// parameter to escape to the heap:
|
2021-08-03 05:29:14 +01:00
|
|
|
copy(s.sum[:], h.h.Sum(h.scratch[:0]))
|
|
|
|
return s
|
2021-07-05 05:25:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var hasherPool = &sync.Pool{
|
2022-03-16 23:27:57 +00:00
|
|
|
New: func() any { return new(hasher) },
|
2021-07-05 05:25:15 +01:00
|
|
|
}
|
|
|
|
|
2021-07-20 06:49:51 +01:00
|
|
|
// Hash returns the hash of v.
|
2022-03-16 23:27:57 +00:00
|
|
|
func Hash(v any) (s Sum) {
|
2021-07-07 06:37:32 +01:00
|
|
|
h := hasherPool.Get().(*hasher)
|
|
|
|
defer hasherPool.Put(h)
|
2021-08-03 05:29:14 +01:00
|
|
|
h.reset()
|
2022-06-16 21:21:32 +01:00
|
|
|
seedOnce.Do(initSeed)
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint64(seed)
|
2022-06-15 06:49:11 +01:00
|
|
|
h.hashValue(reflect.ValueOf(v), false)
|
2021-08-03 05:29:14 +01:00
|
|
|
return h.sum()
|
2021-07-05 05:25:15 +01:00
|
|
|
}
|
|
|
|
|
2021-07-20 06:49:51 +01:00
|
|
|
// Update sets last to the hash of v and reports whether its value changed.
|
2022-03-16 23:27:57 +00:00
|
|
|
func Update(last *Sum, v ...any) (changed bool) {
|
2021-07-05 05:25:15 +01:00
|
|
|
sum := Hash(v)
|
2021-07-20 06:49:51 +01:00
|
|
|
if sum == *last {
|
2021-07-05 05:25:15 +01:00
|
|
|
// unchanged.
|
|
|
|
return false
|
|
|
|
}
|
2021-07-20 06:49:51 +01:00
|
|
|
*last = sum
|
2021-07-05 05:25:15 +01:00
|
|
|
return true
|
2020-07-29 02:47:23 +01:00
|
|
|
}
|
|
|
|
|
2021-05-24 23:13:18 +01:00
|
|
|
var appenderToType = reflect.TypeOf((*appenderTo)(nil)).Elem()
|
2021-05-10 22:15:31 +01:00
|
|
|
|
2021-05-24 22:31:24 +01:00
|
|
|
type appenderTo interface {
|
|
|
|
AppendTo([]byte) []byte
|
|
|
|
}
|
|
|
|
|
2021-08-03 05:44:13 +01:00
|
|
|
func (h *hasher) hashUint8(i uint8) {
|
|
|
|
h.bw.WriteByte(i)
|
2021-07-06 06:13:33 +01:00
|
|
|
}
|
2021-08-03 05:44:13 +01:00
|
|
|
func (h *hasher) hashUint16(i uint16) {
|
|
|
|
binary.LittleEndian.PutUint16(h.scratch[:2], i)
|
|
|
|
h.bw.Write(h.scratch[:2])
|
|
|
|
}
|
|
|
|
func (h *hasher) hashUint32(i uint32) {
|
|
|
|
binary.LittleEndian.PutUint32(h.scratch[:4], i)
|
|
|
|
h.bw.Write(h.scratch[:4])
|
|
|
|
}
|
|
|
|
func (h *hasher) hashUint64(i uint64) {
|
|
|
|
binary.LittleEndian.PutUint64(h.scratch[:8], i)
|
2021-07-06 06:13:33 +01:00
|
|
|
h.bw.Write(h.scratch[:8])
|
|
|
|
}
|
|
|
|
|
2021-07-07 19:58:02 +01:00
|
|
|
var uint8Type = reflect.TypeOf(byte(0))
|
|
|
|
|
2022-06-15 06:49:11 +01:00
|
|
|
// typeInfo describes properties of a type.
|
|
|
|
type typeInfo struct {
|
|
|
|
rtype reflect.Type
|
|
|
|
isRecursive bool
|
|
|
|
|
|
|
|
// elemTypeInfo is the element type's typeInfo.
|
|
|
|
// It's set when rtype is of Kind Ptr, Slice, Array, Map.
|
|
|
|
elemTypeInfo *typeInfo
|
|
|
|
|
|
|
|
// keyTypeInfo is the map key type's typeInfo.
|
|
|
|
// It's set when rtype is of Kind Map.
|
|
|
|
keyTypeInfo *typeInfo
|
|
|
|
}
|
|
|
|
|
|
|
|
var typeInfoMap sync.Map // map[reflect.Type]*typeInfo
|
|
|
|
var typeInfoMapPopulate sync.Mutex // just for adding to typeInfoMap
|
|
|
|
|
|
|
|
func getTypeInfo(t reflect.Type) *typeInfo {
|
|
|
|
if f, ok := typeInfoMap.Load(t); ok {
|
|
|
|
return f.(*typeInfo)
|
|
|
|
}
|
|
|
|
typeInfoMapPopulate.Lock()
|
|
|
|
defer typeInfoMapPopulate.Unlock()
|
|
|
|
newTypes := map[reflect.Type]*typeInfo{}
|
|
|
|
ti := getTypeInfoLocked(t, newTypes)
|
|
|
|
for t, ti := range newTypes {
|
|
|
|
typeInfoMap.Store(t, ti)
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
2022-06-15 06:49:11 +01:00
|
|
|
return ti
|
|
|
|
}
|
2021-05-10 22:15:31 +01:00
|
|
|
|
2022-06-15 06:49:11 +01:00
|
|
|
func getTypeInfoLocked(t reflect.Type, incomplete map[reflect.Type]*typeInfo) *typeInfo {
|
|
|
|
if v, ok := typeInfoMap.Load(t); ok {
|
|
|
|
return v.(*typeInfo)
|
|
|
|
}
|
|
|
|
if ti, ok := incomplete[t]; ok {
|
|
|
|
return ti
|
|
|
|
}
|
|
|
|
ti := &typeInfo{
|
|
|
|
rtype: t,
|
|
|
|
isRecursive: typeIsRecursive(t),
|
|
|
|
}
|
|
|
|
incomplete[t] = ti
|
2021-07-07 06:37:32 +01:00
|
|
|
|
2022-06-15 06:49:11 +01:00
|
|
|
switch t.Kind() {
|
|
|
|
case reflect.Map:
|
|
|
|
ti.keyTypeInfo = getTypeInfoLocked(t.Key(), incomplete)
|
|
|
|
fallthrough
|
|
|
|
case reflect.Ptr, reflect.Slice, reflect.Array:
|
|
|
|
ti.elemTypeInfo = getTypeInfoLocked(t.Elem(), incomplete)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ti
|
|
|
|
}
|
|
|
|
|
|
|
|
// typeIsRecursive reports whether t has a path back to itself.
|
|
|
|
//
|
|
|
|
// For interfaces, it currently always reports true.
|
|
|
|
func typeIsRecursive(t reflect.Type) bool {
|
|
|
|
inStack := map[reflect.Type]bool{}
|
|
|
|
|
|
|
|
var stack []reflect.Type
|
|
|
|
|
|
|
|
var visitType func(t reflect.Type) (isRecursiveSoFar bool)
|
|
|
|
visitType = func(t reflect.Type) (isRecursiveSoFar bool) {
|
|
|
|
switch t.Kind() {
|
|
|
|
case reflect.Bool,
|
|
|
|
reflect.Int,
|
|
|
|
reflect.Int8,
|
|
|
|
reflect.Int16,
|
|
|
|
reflect.Int32,
|
|
|
|
reflect.Int64,
|
|
|
|
reflect.Uint,
|
|
|
|
reflect.Uint8,
|
|
|
|
reflect.Uint16,
|
|
|
|
reflect.Uint32,
|
|
|
|
reflect.Uint64,
|
|
|
|
reflect.Uintptr,
|
|
|
|
reflect.Float32,
|
|
|
|
reflect.Float64,
|
|
|
|
reflect.Complex64,
|
|
|
|
reflect.Complex128,
|
|
|
|
reflect.String,
|
|
|
|
reflect.UnsafePointer,
|
|
|
|
reflect.Func:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if t.Size() == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if inStack[t] {
|
|
|
|
return true
|
2021-05-24 22:31:24 +01:00
|
|
|
}
|
2022-06-15 06:49:11 +01:00
|
|
|
stack = append(stack, t)
|
|
|
|
inStack[t] = true
|
|
|
|
defer func() {
|
|
|
|
delete(inStack, t)
|
|
|
|
stack = stack[:len(stack)-1]
|
|
|
|
}()
|
|
|
|
|
|
|
|
switch t.Kind() {
|
|
|
|
default:
|
|
|
|
panic("unhandled kind " + t.Kind().String())
|
|
|
|
case reflect.Interface:
|
|
|
|
// Assume the worst for now. TODO(bradfitz): in some cases
|
|
|
|
// we should be able to prove that it's not recursive. Not worth
|
|
|
|
// it for now.
|
|
|
|
return true
|
|
|
|
case reflect.Array, reflect.Chan, reflect.Pointer, reflect.Slice:
|
|
|
|
return visitType(t.Elem())
|
|
|
|
case reflect.Map:
|
|
|
|
if visitType(t.Key()) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if visitType(t.Elem()) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
case reflect.Struct:
|
|
|
|
if t.String() == "intern.Value" {
|
|
|
|
// Otherwise its interface{} makes this return true.
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for i, numField := 0, t.NumField(); i < numField; i++ {
|
|
|
|
if visitType(t.Field(i).Type) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return false
|
2021-05-10 22:15:31 +01:00
|
|
|
}
|
2022-06-15 06:49:11 +01:00
|
|
|
return visitType(t)
|
|
|
|
}
|
2021-05-10 22:15:31 +01:00
|
|
|
|
2022-06-15 06:49:11 +01:00
|
|
|
func (h *hasher) hashValue(v reflect.Value, forceCycleChecking bool) {
|
|
|
|
if !v.IsValid() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ti := getTypeInfo(v.Type())
|
|
|
|
h.hashValueWithType(v, ti, forceCycleChecking)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *hasher) hashValueWithType(v reflect.Value, ti *typeInfo, forceCycleChecking bool) {
|
|
|
|
w := h.bw
|
|
|
|
doCheckCycles := forceCycleChecking || ti.isRecursive
|
2021-07-22 23:22:48 +01:00
|
|
|
|
2021-05-10 22:15:31 +01:00
|
|
|
// Generic handling.
|
2020-06-28 18:58:21 +01:00
|
|
|
switch v.Kind() {
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unhandled kind %v for type %v", v.Kind(), v.Type()))
|
|
|
|
case reflect.Ptr:
|
2021-07-22 23:22:48 +01:00
|
|
|
if v.IsNil() {
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint8(0) // indicates nil
|
2021-07-22 23:22:48 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-06-15 06:49:11 +01:00
|
|
|
if doCheckCycles {
|
|
|
|
ptr := pointerOf(v)
|
|
|
|
if idx, ok := h.visitStack.seen(ptr); ok {
|
|
|
|
h.hashUint8(2) // indicates cycle
|
|
|
|
h.hashUint64(uint64(idx))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
h.visitStack.push(ptr)
|
|
|
|
defer h.visitStack.pop(ptr)
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
2021-07-22 23:22:48 +01:00
|
|
|
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint8(1) // indicates visiting a pointer
|
2022-06-15 06:49:11 +01:00
|
|
|
h.hashValueWithType(v.Elem(), ti.elemTypeInfo, doCheckCycles)
|
2020-06-28 18:58:21 +01:00
|
|
|
case reflect.Struct:
|
|
|
|
for i, n := 0, v.NumField(); i < n; i++ {
|
2022-06-15 06:49:11 +01:00
|
|
|
h.hashValue(v.Field(i), doCheckCycles)
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
|
|
|
case reflect.Slice, reflect.Array:
|
2021-07-06 06:13:33 +01:00
|
|
|
vLen := v.Len()
|
|
|
|
if v.Kind() == reflect.Slice {
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint64(uint64(vLen))
|
2021-07-06 06:13:33 +01:00
|
|
|
}
|
2021-07-07 19:58:02 +01:00
|
|
|
if v.Type().Elem() == uint8Type && v.CanInterface() {
|
2021-07-06 06:13:33 +01:00
|
|
|
if vLen > 0 && vLen <= scratchSize {
|
|
|
|
// If it fits in scratch, avoid the Interface allocation.
|
|
|
|
// It seems tempting to do this for all sizes, doing
|
|
|
|
// scratchSize bytes at a time, but reflect.Slice seems
|
|
|
|
// to allocate, so it's not a win.
|
|
|
|
n := reflect.Copy(reflect.ValueOf(&h.scratch).Elem(), v)
|
|
|
|
w.Write(h.scratch[:n])
|
2021-07-22 23:22:48 +01:00
|
|
|
return
|
2021-07-06 06:13:33 +01:00
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "%s", v.Interface())
|
2021-07-22 23:22:48 +01:00
|
|
|
return
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
2021-07-06 06:13:33 +01:00
|
|
|
for i := 0; i < vLen; i++ {
|
2021-07-22 23:22:48 +01:00
|
|
|
// TODO(dsnet): Perform cycle detection for slices,
|
|
|
|
// which is functionally a list of pointers.
|
|
|
|
// See https://github.com/google/go-cmp/blob/402949e8139bb890c71a707b6faf6dd05c92f4e5/cmp/compare.go#L438-L450
|
2022-06-15 06:49:11 +01:00
|
|
|
h.hashValueWithType(v.Index(i), ti.elemTypeInfo, doCheckCycles)
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
|
|
|
case reflect.Interface:
|
2021-07-21 18:26:04 +01:00
|
|
|
if v.IsNil() {
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint8(0) // indicates nil
|
2021-07-22 23:22:48 +01:00
|
|
|
return
|
2021-07-21 18:26:04 +01:00
|
|
|
}
|
|
|
|
v = v.Elem()
|
|
|
|
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint8(1) // indicates visiting interface value
|
2021-07-21 18:26:04 +01:00
|
|
|
h.hashType(v.Type())
|
2022-06-15 06:49:11 +01:00
|
|
|
h.hashValue(v, doCheckCycles)
|
2020-06-28 18:58:21 +01:00
|
|
|
case reflect.Map:
|
2021-07-22 23:22:48 +01:00
|
|
|
// Check for cycle.
|
2022-06-15 06:49:11 +01:00
|
|
|
if doCheckCycles {
|
|
|
|
ptr := pointerOf(v)
|
|
|
|
if idx, ok := h.visitStack.seen(ptr); ok {
|
|
|
|
h.hashUint8(2) // indicates cycle
|
|
|
|
h.hashUint64(uint64(idx))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
h.visitStack.push(ptr)
|
|
|
|
defer h.visitStack.pop(ptr)
|
2021-07-07 05:41:18 +01:00
|
|
|
}
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint8(1) // indicates visiting a map
|
2022-06-15 06:49:11 +01:00
|
|
|
h.hashMap(v, ti, doCheckCycles)
|
2020-06-28 18:58:21 +01:00
|
|
|
case reflect.String:
|
2021-08-03 05:44:13 +01:00
|
|
|
s := v.String()
|
|
|
|
h.hashUint64(uint64(len(s)))
|
|
|
|
w.WriteString(s)
|
2020-06-28 18:58:21 +01:00
|
|
|
case reflect.Bool:
|
2021-08-03 05:44:13 +01:00
|
|
|
if v.Bool() {
|
|
|
|
h.hashUint8(1)
|
|
|
|
} else {
|
|
|
|
h.hashUint8(0)
|
|
|
|
}
|
|
|
|
case reflect.Int8:
|
|
|
|
h.hashUint8(uint8(v.Int()))
|
|
|
|
case reflect.Int16:
|
|
|
|
h.hashUint16(uint16(v.Int()))
|
|
|
|
case reflect.Int32:
|
|
|
|
h.hashUint32(uint32(v.Int()))
|
|
|
|
case reflect.Int64, reflect.Int:
|
|
|
|
h.hashUint64(uint64(v.Int()))
|
|
|
|
case reflect.Uint8:
|
|
|
|
h.hashUint8(uint8(v.Uint()))
|
|
|
|
case reflect.Uint16:
|
|
|
|
h.hashUint16(uint16(v.Uint()))
|
|
|
|
case reflect.Uint32:
|
|
|
|
h.hashUint32(uint32(v.Uint()))
|
|
|
|
case reflect.Uint64, reflect.Uint, reflect.Uintptr:
|
|
|
|
h.hashUint64(uint64(v.Uint()))
|
|
|
|
case reflect.Float32:
|
|
|
|
h.hashUint32(math.Float32bits(float32(v.Float())))
|
|
|
|
case reflect.Float64:
|
|
|
|
h.hashUint64(math.Float64bits(float64(v.Float())))
|
|
|
|
case reflect.Complex64:
|
|
|
|
h.hashUint32(math.Float32bits(real(complex64(v.Complex()))))
|
|
|
|
h.hashUint32(math.Float32bits(imag(complex64(v.Complex()))))
|
|
|
|
case reflect.Complex128:
|
|
|
|
h.hashUint64(math.Float64bits(real(complex128(v.Complex()))))
|
|
|
|
h.hashUint64(math.Float64bits(imag(complex128(v.Complex()))))
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
2021-05-11 21:17:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type mapHasher struct {
|
2022-06-16 21:21:32 +01:00
|
|
|
h hasher
|
|
|
|
valKey, valElem valueCache // re-usable values for map iteration
|
|
|
|
iter reflect.MapIter // re-usable map iterator
|
2021-05-11 21:17:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var mapHasherPool = &sync.Pool{
|
2022-03-16 23:27:57 +00:00
|
|
|
New: func() any { return new(mapHasher) },
|
2021-05-11 21:17:12 +01:00
|
|
|
}
|
|
|
|
|
2021-05-18 17:20:52 +01:00
|
|
|
type valueCache map[reflect.Type]reflect.Value
|
|
|
|
|
2021-08-03 05:29:14 +01:00
|
|
|
func (c *valueCache) get(t reflect.Type) reflect.Value {
|
|
|
|
v, ok := (*c)[t]
|
2021-05-18 17:20:52 +01:00
|
|
|
if !ok {
|
|
|
|
v = reflect.New(t).Elem()
|
2021-08-03 05:29:14 +01:00
|
|
|
if *c == nil {
|
|
|
|
*c = make(valueCache)
|
|
|
|
}
|
|
|
|
(*c)[t] = v
|
2021-05-18 17:20:52 +01:00
|
|
|
}
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2021-07-22 23:22:48 +01:00
|
|
|
// hashMap hashes a map in a sort-free manner.
|
|
|
|
// It relies on a map being a functionally an unordered set of KV entries.
|
|
|
|
// So long as we hash each KV entry together, we can XOR all
|
|
|
|
// of the individual hashes to produce a unique hash for the entire map.
|
2022-06-15 06:49:11 +01:00
|
|
|
func (h *hasher) hashMap(v reflect.Value, ti *typeInfo, checkCycles bool) {
|
2021-05-11 21:17:12 +01:00
|
|
|
mh := mapHasherPool.Get().(*mapHasher)
|
|
|
|
defer mapHasherPool.Put(mh)
|
2022-03-15 22:44:28 +00:00
|
|
|
|
|
|
|
iter := &mh.iter
|
|
|
|
iter.Reset(v)
|
|
|
|
defer iter.Reset(reflect.Value{}) // avoid pinning v from mh.iter when we return
|
2021-07-07 06:37:32 +01:00
|
|
|
|
2021-08-03 05:29:14 +01:00
|
|
|
var sum Sum
|
2022-06-16 21:21:32 +01:00
|
|
|
if v.IsNil() {
|
|
|
|
sum.sum[0] = 1 // something non-zero
|
|
|
|
}
|
|
|
|
|
|
|
|
k := mh.valKey.get(v.Type().Key())
|
|
|
|
e := mh.valElem.get(v.Type().Elem())
|
2021-08-03 05:29:14 +01:00
|
|
|
mh.h.visitStack = h.visitStack // always use the parent's visit stack to avoid cycles
|
2021-05-11 21:17:12 +01:00
|
|
|
for iter.Next() {
|
2022-03-15 22:44:28 +00:00
|
|
|
k.SetIterKey(iter)
|
|
|
|
e.SetIterValue(iter)
|
2021-08-03 05:29:14 +01:00
|
|
|
mh.h.reset()
|
2022-06-15 06:49:11 +01:00
|
|
|
mh.h.hashValueWithType(k, ti.keyTypeInfo, checkCycles)
|
|
|
|
mh.h.hashValueWithType(e, ti.elemTypeInfo, checkCycles)
|
2021-08-03 05:29:14 +01:00
|
|
|
sum.xor(mh.h.sum())
|
2021-05-11 21:17:12 +01:00
|
|
|
}
|
2021-08-03 05:29:14 +01:00
|
|
|
h.bw.Write(append(h.scratch[:0], sum.sum[:]...)) // append into scratch to avoid heap allocation
|
2021-05-11 21:17:12 +01:00
|
|
|
}
|
|
|
|
|
2021-07-22 23:22:48 +01:00
|
|
|
// visitStack is a stack of pointers visited.
|
|
|
|
// Pointers are pushed onto the stack when visited, and popped when leaving.
|
|
|
|
// The integer value is the depth at which the pointer was visited.
|
|
|
|
// The length of this stack should be zero after every hashing operation.
|
|
|
|
type visitStack map[pointer]int
|
|
|
|
|
|
|
|
func (v visitStack) seen(p pointer) (int, bool) {
|
|
|
|
idx, ok := v[p]
|
|
|
|
return idx, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *visitStack) push(p pointer) {
|
|
|
|
if *v == nil {
|
|
|
|
*v = make(map[pointer]int)
|
2021-05-11 21:17:12 +01:00
|
|
|
}
|
2021-07-22 23:22:48 +01:00
|
|
|
(*v)[p] = len(*v)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v visitStack) pop(p pointer) {
|
|
|
|
delete(v, p)
|
|
|
|
}
|
|
|
|
|
|
|
|
// pointer is a thin wrapper over unsafe.Pointer.
|
|
|
|
// We only rely on comparability of pointers; we cannot rely on uintptr since
|
|
|
|
// that would break if Go ever switched to a moving GC.
|
|
|
|
type pointer struct{ p unsafe.Pointer }
|
|
|
|
|
|
|
|
func pointerOf(v reflect.Value) pointer {
|
|
|
|
return pointer{unsafe.Pointer(v.Pointer())}
|
2020-06-28 18:58:21 +01:00
|
|
|
}
|
2021-07-21 18:26:04 +01:00
|
|
|
|
|
|
|
// hashType hashes a reflect.Type.
|
|
|
|
// The hash is only consistent within the lifetime of a program.
|
|
|
|
func (h *hasher) hashType(t reflect.Type) {
|
|
|
|
// This approach relies on reflect.Type always being backed by a unique
|
|
|
|
// *reflect.rtype pointer. A safer approach is to use a global sync.Map
|
|
|
|
// that maps reflect.Type to some arbitrary and unique index.
|
|
|
|
// While safer, it requires global state with memory that can never be GC'd.
|
|
|
|
rtypeAddr := reflect.ValueOf(t).Pointer() // address of *reflect.rtype
|
2021-08-03 05:44:13 +01:00
|
|
|
h.hashUint64(uint64(rtypeAddr))
|
2021-07-21 18:26:04 +01:00
|
|
|
}
|