-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathuint128.go
56 lines (43 loc) · 1.26 KB
/
uint128.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
package extnetip
import "math/bits"
// the internal representation of netip.Addr is mainly an uint128 (at go version 1.18)
type uint128 struct {
hi uint64
lo uint64
}
func (u uint128) and(v uint128) uint128 {
return uint128{u.hi & v.hi, u.lo & v.lo}
}
func (u uint128) or(v uint128) uint128 {
return uint128{u.hi | v.hi, u.lo | v.lo}
}
func (u uint128) xor(v uint128) uint128 {
return uint128{u.hi ^ v.hi, u.lo ^ v.lo}
}
func (u uint128) not() uint128 {
return uint128{^u.hi, ^u.lo}
}
func mask6(n int) uint128 {
return uint128{^(^uint64(0) >> n), ^uint64(0) << (128 - n)}
}
func u64CommonPrefixLen(u, v uint64) int {
return bits.LeadingZeros64(u ^ v)
}
func (u uint128) commonPrefixLen(v uint128) (n int) {
if n = u64CommonPrefixLen(u.hi, v.hi); n == 64 {
n += u64CommonPrefixLen(u.lo, v.lo)
}
return
}
// prefixOK returns the common bits of two uint128 and true if they present exactly a prefix.
func (u uint128) prefixOK(v uint128) (bits int, ok bool) {
bits = u.commonPrefixLen(v)
if bits == 128 {
return bits, true
}
mask := mask6(bits)
// check if mask applied to first and last results in all zeros and all ones
allZero := u.xor(u.and(mask)) == uint128{}
allOnes := v.or(mask) == uint128{^uint64(0), ^uint64(0)}
return bits, allZero && allOnes
}