1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889 |
- package bitutil
- import "fmt"
- const (
- c1 uint32 = 0xcc9e2d51
- c2 uint32 = 0x1b873593
- )
- func MurMurHashData(data []byte, offset int, size int, seed int) (uint32, error) {
-
- if offset < 0 || size < 0 {
- return 0, fmt.Errorf("Invalid data boundaries; offset: %v; size: %v",
- offset, size)
- }
- h1 := uint32(seed)
- end := offset + size
- end -= end % 4
-
- if len(data) <= end {
- return 0, fmt.Errorf("Data out of bounds; set boundary: %v; data length: %v",
- end, len(data))
- }
- for i := offset; i < end; i += 4 {
- var k1 = uint32(data[i])
- k1 |= uint32(data[i+1]) << 8
- k1 |= uint32(data[i+2]) << 16
- k1 |= uint32(data[i+3]) << 24
- k1 *= c1
- k1 = (k1 << 15) | (k1 >> 17)
- k1 *= c2
- h1 ^= k1
- h1 = (h1 << 13) | (h1 >> 19)
- h1 = h1*5 + 0xe6546b64
- }
-
- var k1 uint32
- switch size & 3 {
- case 3:
- k1 = uint32(data[end+2]) << 16
- fallthrough
- case 2:
- k1 |= uint32(data[end+1]) << 8
- fallthrough
- case 1:
- k1 |= uint32(data[end])
- k1 *= c1
- k1 = (k1 << 15) | (k1 >> 17)
- k1 *= c2
- h1 ^= k1
- }
- h1 ^= uint32(size)
- h1 ^= h1 >> 16
- h1 *= 0x85ebca6b
- h1 ^= h1 >> 13
- h1 *= 0xc2b2ae35
- h1 ^= h1 >> 16
- return h1, nil
- }
|