xref: /aosp_15_r20/external/skia/modules/bentleyottmann/tests/Int96Test.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker // Copyright 2023 Google LLC
2*c8dee2aaSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
3*c8dee2aaSAndroid Build Coastguard Worker 
4*c8dee2aaSAndroid Build Coastguard Worker #include "modules/bentleyottmann/include/Int96.h"
5*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
6*c8dee2aaSAndroid Build Coastguard Worker 
7*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
8*c8dee2aaSAndroid Build Coastguard Worker #include <limits>
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker using namespace bentleyottmann;
11*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(BO_Int96Basic,reporter)12*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(BO_Int96Basic, reporter) {
13*c8dee2aaSAndroid Build Coastguard Worker     {
14*c8dee2aaSAndroid Build Coastguard Worker         int32_t t = 0;
15*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
16*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == 0 && z.lo == 0);
17*c8dee2aaSAndroid Build Coastguard Worker     }
18*c8dee2aaSAndroid Build Coastguard Worker     {
19*c8dee2aaSAndroid Build Coastguard Worker         int64_t t = 0;
20*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
21*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == 0 && z.lo == 0);
22*c8dee2aaSAndroid Build Coastguard Worker     }
23*c8dee2aaSAndroid Build Coastguard Worker     {
24*c8dee2aaSAndroid Build Coastguard Worker         int32_t t = -1;
25*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
26*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == -1 && z.lo == 0xFFFFFFFF);
27*c8dee2aaSAndroid Build Coastguard Worker     }
28*c8dee2aaSAndroid Build Coastguard Worker     {
29*c8dee2aaSAndroid Build Coastguard Worker         int64_t t = -1;
30*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
31*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == -1 && z.lo == 0xFFFFFFFF);
32*c8dee2aaSAndroid Build Coastguard Worker     }
33*c8dee2aaSAndroid Build Coastguard Worker     {
34*c8dee2aaSAndroid Build Coastguard Worker         int32_t t = 3;
35*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
36*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == 0 && z.lo == 3);
37*c8dee2aaSAndroid Build Coastguard Worker     }
38*c8dee2aaSAndroid Build Coastguard Worker     {
39*c8dee2aaSAndroid Build Coastguard Worker         int64_t t = 3;
40*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
41*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == 0 && z.lo == 3);
42*c8dee2aaSAndroid Build Coastguard Worker     }
43*c8dee2aaSAndroid Build Coastguard Worker     {
44*c8dee2aaSAndroid Build Coastguard Worker         int32_t t = -3;
45*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
46*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == -1 && z.lo == (uint32_t)-3);
47*c8dee2aaSAndroid Build Coastguard Worker     }
48*c8dee2aaSAndroid Build Coastguard Worker     {
49*c8dee2aaSAndroid Build Coastguard Worker         int64_t t = -3;
50*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
51*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == -1 && z.lo == (uint32_t)-3);
52*c8dee2aaSAndroid Build Coastguard Worker     }
53*c8dee2aaSAndroid Build Coastguard Worker 
54*c8dee2aaSAndroid Build Coastguard Worker     {
55*c8dee2aaSAndroid Build Coastguard Worker         int64_t t = 1ll << 32;
56*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
57*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == 1 && z.lo == 0);
58*c8dee2aaSAndroid Build Coastguard Worker     }
59*c8dee2aaSAndroid Build Coastguard Worker     {
60*c8dee2aaSAndroid Build Coastguard Worker         // -2 << 32 -- without the warnings.
61*c8dee2aaSAndroid Build Coastguard Worker         int64_t t = -(2ll << 32);
62*c8dee2aaSAndroid Build Coastguard Worker         Int96 z = Int96::Make(t);
63*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, z.hi == -2 && z.lo == 0);
64*c8dee2aaSAndroid Build Coastguard Worker     }
65*c8dee2aaSAndroid Build Coastguard Worker }
66*c8dee2aaSAndroid Build Coastguard Worker 
67*c8dee2aaSAndroid Build Coastguard Worker [[maybe_unused]]
68*c8dee2aaSAndroid Build Coastguard Worker static int64_t interesting64[] = {-std::numeric_limits<int64_t>::max(),
69*c8dee2aaSAndroid Build Coastguard Worker                                   -std::numeric_limits<int64_t>::max() + 1,
70*c8dee2aaSAndroid Build Coastguard Worker                                   (int64_t) -std::numeric_limits<int32_t>::max() - 1,
71*c8dee2aaSAndroid Build Coastguard Worker                                   (int64_t) -std::numeric_limits<int32_t>::max(),
72*c8dee2aaSAndroid Build Coastguard Worker                                   (int64_t) -std::numeric_limits<int32_t>::max() + 1,
73*c8dee2aaSAndroid Build Coastguard Worker                                   -2,
74*c8dee2aaSAndroid Build Coastguard Worker                                   -1,
75*c8dee2aaSAndroid Build Coastguard Worker                                   0,
76*c8dee2aaSAndroid Build Coastguard Worker                                   1,
77*c8dee2aaSAndroid Build Coastguard Worker                                   2,
78*c8dee2aaSAndroid Build Coastguard Worker                                   (int64_t) std::numeric_limits<int32_t>::max() - 1,
79*c8dee2aaSAndroid Build Coastguard Worker                                   (int64_t) std::numeric_limits<int32_t>::max(),
80*c8dee2aaSAndroid Build Coastguard Worker                                   (int64_t) std::numeric_limits<int32_t>::max() + 1,
81*c8dee2aaSAndroid Build Coastguard Worker                                   std::numeric_limits<int64_t>::max() - 1,
82*c8dee2aaSAndroid Build Coastguard Worker                                   std::numeric_limits<int64_t>::max()};
83*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(BO_Int96Less,reporter)84*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(BO_Int96Less, reporter) {
85*c8dee2aaSAndroid Build Coastguard Worker #if (defined(__clang__) || defined(__GNUC__)) && defined(__SIZEOF_INT128__)
86*c8dee2aaSAndroid Build Coastguard Worker     for (auto a : interesting64) {
87*c8dee2aaSAndroid Build Coastguard Worker         for (auto b : interesting64) {
88*c8dee2aaSAndroid Build Coastguard Worker             __int128 a128 = a,
89*c8dee2aaSAndroid Build Coastguard Worker                     b128 = b;
90*c8dee2aaSAndroid Build Coastguard Worker             bool l128 = a128 < b128,
91*c8dee2aaSAndroid Build Coastguard Worker                  g128 = b128 < a128;
92*c8dee2aaSAndroid Build Coastguard Worker 
93*c8dee2aaSAndroid Build Coastguard Worker             Int96 a96 = Int96::Make(a),
94*c8dee2aaSAndroid Build Coastguard Worker                   b96 = Int96::Make(b);
95*c8dee2aaSAndroid Build Coastguard Worker             bool l96 = a96 < b96,
96*c8dee2aaSAndroid Build Coastguard Worker                  g96 = b96 < a96;
97*c8dee2aaSAndroid Build Coastguard Worker 
98*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, l128 == l96);
99*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, g128 == g96);
100*c8dee2aaSAndroid Build Coastguard Worker 
101*c8dee2aaSAndroid Build Coastguard Worker         }
102*c8dee2aaSAndroid Build Coastguard Worker     }
103*c8dee2aaSAndroid Build Coastguard Worker #endif
104*c8dee2aaSAndroid Build Coastguard Worker }
105*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(BO_Int96Add,reporter)106*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(BO_Int96Add, reporter) {
107*c8dee2aaSAndroid Build Coastguard Worker #if (defined(__clang__) || defined(__GNUC__)) && defined(__SIZEOF_INT128__)
108*c8dee2aaSAndroid Build Coastguard Worker     for (auto a : interesting64) {
109*c8dee2aaSAndroid Build Coastguard Worker         for (auto b : interesting64) {
110*c8dee2aaSAndroid Build Coastguard Worker             __int128 a128 = a,
111*c8dee2aaSAndroid Build Coastguard Worker                      b128 = b,
112*c8dee2aaSAndroid Build Coastguard Worker                      r128 = a128 + b128;
113*c8dee2aaSAndroid Build Coastguard Worker 
114*c8dee2aaSAndroid Build Coastguard Worker             Int96 a96 = Int96::Make(a),
115*c8dee2aaSAndroid Build Coastguard Worker                   b96 = Int96::Make(b),
116*c8dee2aaSAndroid Build Coastguard Worker                   r96 = a96 + b96;
117*c8dee2aaSAndroid Build Coastguard Worker 
118*c8dee2aaSAndroid Build Coastguard Worker             // Explicitly check the low bits.
119*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, r96.lo == (r128 & 0xFFFFFFFF));
120*c8dee2aaSAndroid Build Coastguard Worker 
121*c8dee2aaSAndroid Build Coastguard Worker             // Build a __int128 from an Int96.
122*c8dee2aaSAndroid Build Coastguard Worker             __int128 hi128 = r96.hi,
123*c8dee2aaSAndroid Build Coastguard Worker                      lo128 = r96.lo,
124*c8dee2aaSAndroid Build Coastguard Worker                      all128 = hi128 * 0x1'0000'0000 + lo128;
125*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, r128 == all128);
126*c8dee2aaSAndroid Build Coastguard Worker         }
127*c8dee2aaSAndroid Build Coastguard Worker     }
128*c8dee2aaSAndroid Build Coastguard Worker #endif
129*c8dee2aaSAndroid Build Coastguard Worker }
130*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(BO_Int96Mult,reporter)131*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(BO_Int96Mult, reporter) {
132*c8dee2aaSAndroid Build Coastguard Worker #if (defined(__clang__) || defined(__GNUC__)) && defined(__SIZEOF_INT128__)
133*c8dee2aaSAndroid Build Coastguard Worker     int32_t interesting32[] = {-std::numeric_limits<int32_t>::max(),
134*c8dee2aaSAndroid Build Coastguard Worker                                -std::numeric_limits<int32_t>::max() + 1,
135*c8dee2aaSAndroid Build Coastguard Worker                                -2,
136*c8dee2aaSAndroid Build Coastguard Worker                                -1,
137*c8dee2aaSAndroid Build Coastguard Worker                                0,
138*c8dee2aaSAndroid Build Coastguard Worker                                1,
139*c8dee2aaSAndroid Build Coastguard Worker                                2,
140*c8dee2aaSAndroid Build Coastguard Worker                                std::numeric_limits<int32_t>::max() - 1,
141*c8dee2aaSAndroid Build Coastguard Worker                                std::numeric_limits<int32_t>::max()};
142*c8dee2aaSAndroid Build Coastguard Worker 
143*c8dee2aaSAndroid Build Coastguard Worker     for (auto i64 : interesting64) {
144*c8dee2aaSAndroid Build Coastguard Worker         for (auto i32 : interesting32) {
145*c8dee2aaSAndroid Build Coastguard Worker             __int128 a128 = i64,
146*c8dee2aaSAndroid Build Coastguard Worker                      b128 = i32,
147*c8dee2aaSAndroid Build Coastguard Worker                      r128 = a128 * b128;
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker             Int96 r96 = multiply(i64, i32);
150*c8dee2aaSAndroid Build Coastguard Worker 
151*c8dee2aaSAndroid Build Coastguard Worker             // Explicitly check the low bits.
152*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, r96.lo == (r128 & 0xFFFFFFFF));
153*c8dee2aaSAndroid Build Coastguard Worker 
154*c8dee2aaSAndroid Build Coastguard Worker             // Build a __int128 from an Int96.
155*c8dee2aaSAndroid Build Coastguard Worker             __int128 hi128 = r96.hi,
156*c8dee2aaSAndroid Build Coastguard Worker                      lo128 = r96.lo,
157*c8dee2aaSAndroid Build Coastguard Worker                      all128 = hi128 * 0x1'0000'0000 + lo128;
158*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, r128 == all128);
159*c8dee2aaSAndroid Build Coastguard Worker         }
160*c8dee2aaSAndroid Build Coastguard Worker     }
161*c8dee2aaSAndroid Build Coastguard Worker #endif
162*c8dee2aaSAndroid Build Coastguard Worker }
163