-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy path02-13_overflow_test.go
140 lines (122 loc) · 3.6 KB
/
02-13_overflow_test.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
package hd_test
import (
"fmt"
"math"
"testing"
hd "github.com/nikolaydubina/go-hackers-delight"
)
func ExampleIsAddOverflow() {
fmt.Println(hd.IsAddOverflow(int32(math.MaxInt32), 1)>>31, hd.IsAddOverflow(int32(10), 1)>>31)
// Output: -1 0
}
func ExampleIsSubOverflow() {
fmt.Println(hd.IsSubOverflow2(int32(math.MinInt32), 1)>>31, hd.IsSubOverflow2(int32(-10), 1)>>31)
// Output: -1 0
}
func ExampleIsAddOverflowUnsigned() {
fmt.Println(hd.IsAddOverflowUnsigned(uint32(math.MaxUint32), 1)>>31, hd.IsAddOverflowUnsigned(uint32(10), 1)>>31)
// Output: 1 0
}
func ExampleIsSubOverflowUnsigned() {
fmt.Println(hd.IsSubOverflowUnsigned(uint32(0), 1)>>31, hd.IsSubOverflowUnsigned(uint32(10), 1)>>31)
// Output: 1 0
}
func ExampleIsAddOverflowUnsigned4() {
var x uint32 = 1
fmt.Println(hd.IsAddOverflowUnsigned4(math.MaxUint32, 1, math.MaxUint32+x), hd.IsAddOverflowUnsigned4(10, 1, 10+x))
// Output: true false
}
func ExampleIsSubOverflowUnsigned4() {
var x uint32 = 1
fmt.Println(hd.IsSubOverflowUnsigned4(0, 1, 0-x), hd.IsSubOverflowUnsigned4(10, 1, 10-x))
// Output: true false
}
func FuzzOverflowInt32(f *testing.F) {
for _, x := range fuzzInt32 {
for _, y := range fuzzInt32 {
f.Add(x, y)
}
}
f.Fuzz(func(t *testing.T, x, y int32) {
a, b := float64(x), float64(y)
sum := a + b
sub := a - b
mul := a * b
div := a / b
sumOverflow := sum > math.MaxInt32 || sum < math.MinInt32
subOverflow := sub > math.MaxInt32 || sub < math.MinInt32
mulOverflow := mul > math.MaxInt32 || mul < math.MinInt32
divOverflow := div > math.MaxInt32 || div < math.MinInt32
v := []struct {
exp bool
got int32
}{
{sumOverflow, hd.IsAddOverflow(x, y)},
{sumOverflow, hd.IsAddOverflow2(x, y)},
{subOverflow, hd.IsSubOverflow(x, y)},
{subOverflow, hd.IsSubOverflow2(x, y)},
}
for i, q := range v {
if hd.IsMostSignificantSet(q.got) != q.exp {
t.Error(i, x, y)
}
}
if mulOverflow != hd.IsMulOverflow(x, y) {
t.Error("mul", x, y)
}
// in Go 0/0 is panic at runtime, therefore overflow value is not determined for this case
if x > 0 && divOverflow != hd.IsDivOverflow(x, y) {
t.Error("div", x, y)
}
})
}
func FuzzOverflowUint32(f *testing.F) {
for _, x := range fuzzUint32 {
for _, y := range fuzzUint32 {
f.Add(x, y)
}
}
f.Add(uint32(0x0000FFFF), uint32(0x0001FFFF)) // 4 * 4 + 3 * 4 + 3 = 31 bits leading zeros n + m
f.Fuzz(func(t *testing.T, x, y uint32) {
a, b := uint64(x), uint64(y)
sum := a + b
sub := a - b
mul := a * b
div := uint64(0)
if b > 0 {
div = a / b
}
sumOverflow := sum > math.MaxUint32
subOverflow := sub > math.MaxUint32
mulOverflow := mul > math.MaxUint32
divOverflow := div > math.MaxUint32
v := []struct {
exp bool
got bool
}{
{sumOverflow, hd.IsMostSignificantSet(hd.IsAddOverflowUnsigned(x, y))},
{sumOverflow, hd.IsMostSignificantSet(hd.IsAddOverflowUnsigned2(x, y))},
{sumOverflow, hd.IsAddOverflowUnsigned3(x, y)},
{sumOverflow, hd.IsAddOverflowUnsigned4(x, y, x+y)},
{subOverflow, hd.IsMostSignificantSet(hd.IsSubOverflowUnsigned(x, y))},
{subOverflow, hd.IsMostSignificantSet(hd.IsSubOverflowUnsigned2(x, y))},
{subOverflow, hd.IsSubOverflowUnsigned3(x, y)},
{subOverflow, hd.IsSubOverflowUnsigned4(x, y, x-y)},
{mulOverflow, hd.IsMulOverflowUnsigned(x, y)},
{mulOverflow, hd.IsMulOverflowUnsignedNLZ(x, y)},
}
for i, q := range v {
if q.got != q.exp {
t.Error(i, x, y)
}
}
t.Run("IsDivOverflowUnsigned", func(t *testing.T) {
if y == 0 {
t.Skip("cannot divide by 0")
}
if divOverflow != hd.IsDivOverflowUnsigned(x, y) {
t.Error("div", x, y)
}
})
})
}