-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathuint128.go
173 lines (144 loc) · 3.97 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
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
// Copyright 2018 Weborama. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package uint128 // import "github.com/weborama/uint128"
//go:generate go run make_tables.go
import (
"bytes"
"encoding/binary"
"encoding/hex"
"fmt"
)
const (
numBits = 128
numHalfBits = 64
)
// Uint128 defines an unsigned integer of 128 bits.
type Uint128 struct {
H, L uint64
}
// Zero is a 0 valued Uint128.
func Zero() Uint128 {
return Uint128{H: 0, L: 0}
}
// MaxUint128 returns the maximum value of an Uint128.
func MaxUint128() Uint128 {
return Uint128{0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF}
}
// Cmp compares two Uint128 and returns one of the following values:
//
// -1 if x < y
// 0 if x == y
// +1 if x > y
func (x Uint128) Cmp(y Uint128) int {
return Cmp(x, y)
}
// IsZero returns true if x is zero.
func (x Uint128) IsZero() bool {
return IsZero(x)
}
// ShiftLeft shifts x to the left by the provided number of bits.
func (x Uint128) ShiftLeft(bits uint) Uint128 {
return ShiftLeft(x, bits)
}
// ShiftRight shifts x to the right by the provided number of bits.
func (x Uint128) ShiftRight(bits uint) Uint128 {
return ShiftRight(x, bits)
}
// And returns the logical AND of x and y.
func (x Uint128) And(y Uint128) Uint128 {
return And(x, y)
}
// AndNot returns the logical AND NOT of x and y.
func (x Uint128) AndNot(y Uint128) Uint128 {
return AndNot(x, y)
}
// Not returns the logical AND of x and y.
func (x Uint128) Not() Uint128 {
return Not(x)
}
// Xor returns the logical XOR of x and y.
func (x Uint128) Xor(y Uint128) Uint128 {
return Xor(x, y)
}
// Or returns the logical OR of x and y.
func (x Uint128) Or(y Uint128) Uint128 {
return Or(x, y)
}
// Add adds x and y.
func (x Uint128) Add(y Uint128) Uint128 {
return Add(x, y)
}
// Add128 returns the sum with carry of x, y and carry: sum = x + y + carry.
// The carry input must be 0 or 1; otherwise the behavior is undefined.
// The carryOut output is guaranteed to be 0 or 1.
func (x Uint128) Add128(y, carry Uint128) (sum, carryOut Uint128) {
sum, carryOut = Add128(x, y, carry)
return
}
// Incr increments x by one.
func (x Uint128) Incr() Uint128 {
return Incr(x)
}
// Sub subtracts x and y.
func (x Uint128) Sub(y Uint128) Uint128 {
return Sub(x, y)
}
// Decr decrements x by one.
func (x Uint128) Decr() Uint128 {
return Decr(x)
}
// NewFromString creates a new Uint128 from its hexadecimal string representation.
// XXX: Do a proper job of it.
func NewFromString(str string) (x Uint128, err error) {
x = Uint128{0, 0}
// nolint: gomnd // Number of characters in a hexadecimal representation of an uint128
if len(str) > 32 {
return x, fmt.Errorf("s:%s length greater than 32", str) // nolint: goerr113
}
b, err := hex.DecodeString(fmt.Sprintf("%032s", str))
if err != nil {
return x, fmt.Errorf("hex.DecodeString(): %w", err)
}
rdr := bytes.NewReader(b)
err = binary.Read(rdr, binary.BigEndian, &x)
return
}
// HexString returns a Hexadecimal string representation of an Uint128.
// XXX: Do a proper job of it.
func (x Uint128) HexString() string {
if x.H == 0 {
return fmt.Sprintf("%x", x.L)
}
return fmt.Sprintf("%x%016x", x.H, x.L)
}
func (x Uint128) String() string {
return fmt.Sprintf("0x%032x", x.HexString())
}
// Format is a custom formatter for Uint128
// XXX: Do a proper job of it.
func (x Uint128) Format(fmtState fmt.State, c rune) {
switch c {
case 'v':
switch {
case fmtState.Flag('+'):
fmt.Fprintf(fmtState, "(%+v, %+v)", x.H, x.L)
case fmtState.Flag('#'):
fmt.Fprintf(fmtState, "(%#v, %#v)", x.H, x.L)
default:
fmt.Fprintf(fmtState, "(%v, %v)", x.H, x.L)
}
case 'T':
fmt.Fprintf(fmtState, "%T", x)
case 'b':
fmt.Fprintf(fmtState, "%0b%064b", x.H, x.L)
// case 'd':
// fmt.Fprintf(fmtState, "%0d%064d", x.H, x.L)
// case 'o':
// fmt.Fprintf(fmtState, "%0o%064o", x.H, x.L)
case 'x':
fmt.Fprintf(fmtState, "0x%032x", x.HexString())
case 'X':
fmt.Fprintf(fmtState, "0x%032X", x.HexString())
}
}