| 
									
										
										
										
											2023-01-27 13:37:20 -08:00
										 |  |  | // Copyright (c) Tailscale Inc & AUTHORS | 
					
						
							|  |  |  | // SPDX-License-Identifier: BSD-3-Clause | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | package hashx | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"crypto/sha256" | 
					
						
							|  |  |  | 	"encoding/binary" | 
					
						
							|  |  |  | 	"hash" | 
					
						
							|  |  |  | 	"math/rand" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qt "github.com/frankban/quicktest" | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 	"tailscale.com/util/must" | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // naiveHash is an obviously correct implementation of Hash. | 
					
						
							|  |  |  | type naiveHash struct { | 
					
						
							|  |  |  | 	hash.Hash | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 	scratch [256]byte | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func newNaive() *naiveHash               { return &naiveHash{Hash: sha256.New()} } | 
					
						
							|  |  |  | func (h *naiveHash) HashUint8(n uint8)   { h.Write(append(h.scratch[:0], n)) } | 
					
						
							|  |  |  | func (h *naiveHash) HashUint16(n uint16) { h.Write(binary.LittleEndian.AppendUint16(h.scratch[:0], n)) } | 
					
						
							|  |  |  | func (h *naiveHash) HashUint32(n uint32) { h.Write(binary.LittleEndian.AppendUint32(h.scratch[:0], n)) } | 
					
						
							|  |  |  | func (h *naiveHash) HashUint64(n uint64) { h.Write(binary.LittleEndian.AppendUint64(h.scratch[:0], n)) } | 
					
						
							|  |  |  | func (h *naiveHash) HashBytes(b []byte)  { h.Write(b) } | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | func (h *naiveHash) HashString(s string) { h.Write(append(h.scratch[:0], s...)) } | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | var bytes = func() (out []byte) { | 
					
						
							|  |  |  | 	out = make([]byte, 130) | 
					
						
							|  |  |  | 	for i := range out { | 
					
						
							|  |  |  | 		out[i] = byte(i) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return out | 
					
						
							|  |  |  | }() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type hasher interface { | 
					
						
							|  |  |  | 	HashUint8(uint8) | 
					
						
							|  |  |  | 	HashUint16(uint16) | 
					
						
							|  |  |  | 	HashUint32(uint32) | 
					
						
							|  |  |  | 	HashUint64(uint64) | 
					
						
							|  |  |  | 	HashBytes([]byte) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 	HashString(string) | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func hashSuite(h hasher) { | 
					
						
							| 
									
										
										
										
											2024-04-16 13:15:13 -07:00
										 |  |  | 	for i := range 10 { | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 		for j := 0; j < 10; j++ { | 
					
						
							|  |  |  | 			h.HashUint8(0x01) | 
					
						
							|  |  |  | 			h.HashUint8(0x23) | 
					
						
							|  |  |  | 			h.HashUint32(0x456789ab) | 
					
						
							|  |  |  | 			h.HashUint8(0xcd) | 
					
						
							|  |  |  | 			h.HashUint8(0xef) | 
					
						
							|  |  |  | 			h.HashUint16(0x0123) | 
					
						
							|  |  |  | 			h.HashUint32(0x456789ab) | 
					
						
							|  |  |  | 			h.HashUint16(0xcdef) | 
					
						
							|  |  |  | 			h.HashUint8(0x01) | 
					
						
							|  |  |  | 			h.HashUint64(0x23456789abcdef01) | 
					
						
							|  |  |  | 			h.HashUint16(0x2345) | 
					
						
							|  |  |  | 			h.HashUint8(0x67) | 
					
						
							|  |  |  | 			h.HashUint16(0x89ab) | 
					
						
							|  |  |  | 			h.HashUint8(0xcd) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		b := bytes[:(i+1)*13] | 
					
						
							|  |  |  | 		if i%2 == 0 { | 
					
						
							|  |  |  | 			h.HashBytes(b) | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			h.HashString(string(b)) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-10 17:31:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | func Test(t *testing.T) { | 
					
						
							|  |  |  | 	c := qt.New(t) | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 	h1 := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 	h2 := newNaive() | 
					
						
							|  |  |  | 	hashSuite(h1) | 
					
						
							|  |  |  | 	hashSuite(h2) | 
					
						
							|  |  |  | 	c.Assert(h1.Sum(nil), qt.DeepEquals, h2.Sum(nil)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | func TestAllocations(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2022-08-10 17:31:44 -07:00
										 |  |  | 	c := qt.New(t) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 	c.Run("Sum", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			var a [sha256.Size]byte | 
					
						
							|  |  |  | 			h.Sum(a[:0]) | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	c.Run("HashUint8", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			h.HashUint8(0x01) | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	c.Run("HashUint16", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			h.HashUint16(0x0123) | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	c.Run("HashUint32", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			h.HashUint32(0x01234567) | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	c.Run("HashUint64", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			h.HashUint64(0x0123456789abcdef) | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	c.Run("HashBytes", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			h.HashBytes(bytes) | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	c.Run("HashString", func(c *qt.C) { | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-11 17:44:09 -07:00
										 |  |  | 		c.Assert(testing.AllocsPerRun(100, func() { | 
					
						
							|  |  |  | 			h.HashString("abcdefghijklmnopqrstuvwxyz") | 
					
						
							|  |  |  | 		}), qt.Equals, 0.0) | 
					
						
							| 
									
										
										
										
											2022-08-10 17:31:44 -07:00
										 |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | func Fuzz(f *testing.F) { | 
					
						
							|  |  |  | 	f.Fuzz(func(t *testing.T, seed int64) { | 
					
						
							|  |  |  | 		c := qt.New(t) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		execute := func(h hasher, r *rand.Rand) { | 
					
						
							| 
									
										
										
										
											2024-04-16 13:15:13 -07:00
										 |  |  | 			for range r.Intn(256) { | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 				switch r.Intn(5) { | 
					
						
							|  |  |  | 				case 0: | 
					
						
							|  |  |  | 					n := uint8(r.Uint64()) | 
					
						
							|  |  |  | 					h.HashUint8(n) | 
					
						
							|  |  |  | 				case 1: | 
					
						
							|  |  |  | 					n := uint16(r.Uint64()) | 
					
						
							|  |  |  | 					h.HashUint16(n) | 
					
						
							|  |  |  | 				case 2: | 
					
						
							|  |  |  | 					n := uint32(r.Uint64()) | 
					
						
							|  |  |  | 					h.HashUint32(n) | 
					
						
							|  |  |  | 				case 3: | 
					
						
							|  |  |  | 					n := uint64(r.Uint64()) | 
					
						
							|  |  |  | 					h.HashUint64(n) | 
					
						
							|  |  |  | 				case 4: | 
					
						
							|  |  |  | 					b := make([]byte, r.Intn(256)) | 
					
						
							|  |  |  | 					r.Read(b) | 
					
						
							|  |  |  | 					h.HashBytes(b) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		r1 := rand.New(rand.NewSource(seed)) | 
					
						
							|  |  |  | 		r2 := rand.New(rand.NewSource(seed)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h1 := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 		h2 := newNaive() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		execute(h1, r1) | 
					
						
							|  |  |  | 		execute(h2, r2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		c.Assert(h1.Sum(nil), qt.DeepEquals, h2.Sum(nil)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		execute(h1, r1) | 
					
						
							|  |  |  | 		execute(h2, r2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		c.Assert(h1.Sum(nil), qt.DeepEquals, h2.Sum(nil)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		h1.Reset() | 
					
						
							|  |  |  | 		h2.Reset() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		execute(h1, r1) | 
					
						
							|  |  |  | 		execute(h2, r2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		c.Assert(h1.Sum(nil), qt.DeepEquals, h2.Sum(nil)) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func Benchmark(b *testing.B) { | 
					
						
							|  |  |  | 	var sum [sha256.Size]byte | 
					
						
							|  |  |  | 	b.Run("Hash", func(b *testing.B) { | 
					
						
							|  |  |  | 		b.ReportAllocs() | 
					
						
							| 
									
										
										
										
											2022-08-16 09:49:48 -07:00
										 |  |  | 		h := must.Get(New512(sha256.New())) | 
					
						
							| 
									
										
										
										
											2024-04-16 13:15:13 -07:00
										 |  |  | 		for range b.N { | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 			h.Reset() | 
					
						
							|  |  |  | 			hashSuite(h) | 
					
						
							|  |  |  | 			h.Sum(sum[:0]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	b.Run("Naive", func(b *testing.B) { | 
					
						
							|  |  |  | 		b.ReportAllocs() | 
					
						
							|  |  |  | 		h := newNaive() | 
					
						
							| 
									
										
										
										
											2024-04-16 13:15:13 -07:00
										 |  |  | 		for range b.N { | 
					
						
							| 
									
										
										
										
											2022-08-10 15:49:36 -07:00
										 |  |  | 			h.Reset() | 
					
						
							|  |  |  | 			hashSuite(h) | 
					
						
							|  |  |  | 			h.Sum(sum[:0]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } |