github.com/yaling888/clash@v1.53.0/common/murmur3/murmur32.go (about)

     1  package murmur3
     2  
     3  // https://github.com/spaolacci/murmur3/blob/master/murmur32.go
     4  
     5  import (
     6  	"hash"
     7  	"math/bits"
     8  	"unsafe"
     9  )
    10  
    11  // Make sure interfaces are correctly implemented.
    12  var (
    13  	_ hash.Hash32 = new(digest32)
    14  	_ bmixer      = new(digest32)
    15  )
    16  
    17  const (
    18  	c1_32 uint32 = 0xcc9e2d51
    19  	c2_32 uint32 = 0x1b873593
    20  )
    21  
    22  // digest32 represents a partial evaluation of a 32 bites hash.
    23  type digest32 struct {
    24  	digest
    25  	h1 uint32 // Unfinalized running hash.
    26  }
    27  
    28  // New32 returns new 32-bit hasher
    29  func New32() hash.Hash32 { return New32WithSeed(0) }
    30  
    31  // New32WithSeed returns new 32-bit hasher set with explicit seed value
    32  func New32WithSeed(seed uint32) hash.Hash32 {
    33  	d := new(digest32)
    34  	d.seed = seed
    35  	d.bmixer = d
    36  	d.Reset()
    37  	return d
    38  }
    39  
    40  func (d *digest32) Size() int { return 4 }
    41  
    42  func (d *digest32) reset() { d.h1 = d.seed }
    43  
    44  func (d *digest32) Sum(b []byte) []byte {
    45  	h := d.Sum32()
    46  	return append(b, byte(h>>24), byte(h>>16), byte(h>>8), byte(h))
    47  }
    48  
    49  // Digest as many blocks as possible.
    50  func (d *digest32) bmix(p []byte) (tail []byte) {
    51  	h1 := d.h1
    52  
    53  	nblocks := len(p) / 4
    54  	for i := 0; i < nblocks; i++ {
    55  		k1 := *(*uint32)(unsafe.Pointer(&p[i*4]))
    56  
    57  		k1 *= c1_32
    58  		k1 = bits.RotateLeft32(k1, 15)
    59  		k1 *= c2_32
    60  
    61  		h1 ^= k1
    62  		h1 = bits.RotateLeft32(h1, 13)
    63  		h1 = h1*4 + h1 + 0xe6546b64
    64  	}
    65  	d.h1 = h1
    66  	return p[nblocks*d.Size():]
    67  }
    68  
    69  func (d *digest32) Sum32() (h1 uint32) {
    70  	h1 = d.h1
    71  
    72  	var k1 uint32
    73  	switch len(d.tail) & 3 {
    74  	case 3:
    75  		k1 ^= uint32(d.tail[2]) << 16
    76  		fallthrough
    77  	case 2:
    78  		k1 ^= uint32(d.tail[1]) << 8
    79  		fallthrough
    80  	case 1:
    81  		k1 ^= uint32(d.tail[0])
    82  		k1 *= c1_32
    83  		k1 = bits.RotateLeft32(k1, 15)
    84  		k1 *= c2_32
    85  		h1 ^= k1
    86  	}
    87  
    88  	h1 ^= uint32(d.clen)
    89  
    90  	h1 ^= h1 >> 16
    91  	h1 *= 0x85ebca6b
    92  	h1 ^= h1 >> 13
    93  	h1 *= 0xc2b2ae35
    94  	h1 ^= h1 >> 16
    95  
    96  	return h1
    97  }
    98  
    99  func Sum32(data []byte) uint32 { return Sum32WithSeed(data, 0) }
   100  
   101  func Sum32WithSeed(data []byte, seed uint32) uint32 {
   102  	h1 := seed
   103  
   104  	nblocks := len(data) / 4
   105  	for i := 0; i < nblocks; i++ {
   106  		k1 := *(*uint32)(unsafe.Pointer(&data[i*4]))
   107  
   108  		k1 *= c1_32
   109  		k1 = bits.RotateLeft32(k1, 15)
   110  		k1 *= c2_32
   111  
   112  		h1 ^= k1
   113  		h1 = bits.RotateLeft32(h1, 13)
   114  		h1 = h1*4 + h1 + 0xe6546b64
   115  	}
   116  
   117  	tail := data[nblocks*4:]
   118  
   119  	var k1 uint32
   120  	switch len(tail) & 3 {
   121  	case 3:
   122  		k1 ^= uint32(tail[2]) << 16
   123  		fallthrough
   124  	case 2:
   125  		k1 ^= uint32(tail[1]) << 8
   126  		fallthrough
   127  	case 1:
   128  		k1 ^= uint32(tail[0])
   129  		k1 *= c1_32
   130  		k1 = bits.RotateLeft32(k1, 15)
   131  		k1 *= c2_32
   132  		h1 ^= k1
   133  	}
   134  
   135  	h1 ^= uint32(len(data))
   136  
   137  	h1 ^= h1 >> 16
   138  	h1 *= 0x85ebca6b
   139  	h1 ^= h1 >> 13
   140  	h1 *= 0xc2b2ae35
   141  	h1 ^= h1 >> 16
   142  
   143  	return h1
   144  }