xref: /openbmc/qemu/tests/unit/test-div128.c (revision 02346297)
1*02346297SLuis Pires /*
2*02346297SLuis Pires  * Test 128-bit division functions
3*02346297SLuis Pires  *
4*02346297SLuis Pires  * Copyright (c) 2021 Instituto de Pesquisas Eldorado (eldorado.org.br)
5*02346297SLuis Pires  *
6*02346297SLuis Pires  * This library is free software; you can redistribute it and/or
7*02346297SLuis Pires  * modify it under the terms of the GNU Lesser General Public
8*02346297SLuis Pires  * License as published by the Free Software Foundation; either
9*02346297SLuis Pires  * version 2.1 of the License, or (at your option) any later version.
10*02346297SLuis Pires  *
11*02346297SLuis Pires  * This library is distributed in the hope that it will be useful,
12*02346297SLuis Pires  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*02346297SLuis Pires  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14*02346297SLuis Pires  * Lesser General Public License for more details.
15*02346297SLuis Pires  *
16*02346297SLuis Pires  * You should have received a copy of the GNU Lesser General Public
17*02346297SLuis Pires  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18*02346297SLuis Pires  */
19*02346297SLuis Pires 
20*02346297SLuis Pires #include "qemu/osdep.h"
21*02346297SLuis Pires #include "qemu/host-utils.h"
22*02346297SLuis Pires 
23*02346297SLuis Pires typedef struct {
24*02346297SLuis Pires     uint64_t high;
25*02346297SLuis Pires     uint64_t low;
26*02346297SLuis Pires     uint64_t rhigh;
27*02346297SLuis Pires     uint64_t rlow;
28*02346297SLuis Pires     uint64_t divisor;
29*02346297SLuis Pires     uint64_t remainder;
30*02346297SLuis Pires } test_data_unsigned;
31*02346297SLuis Pires 
32*02346297SLuis Pires typedef struct {
33*02346297SLuis Pires     int64_t high;
34*02346297SLuis Pires     uint64_t low;
35*02346297SLuis Pires     int64_t rhigh;
36*02346297SLuis Pires     uint64_t rlow;
37*02346297SLuis Pires     int64_t divisor;
38*02346297SLuis Pires     int64_t remainder;
39*02346297SLuis Pires } test_data_signed;
40*02346297SLuis Pires 
41*02346297SLuis Pires static const test_data_unsigned test_table_unsigned[] = {
42*02346297SLuis Pires     /* Dividend fits in 64 bits */
43*02346297SLuis Pires     { 0x0000000000000000ULL, 0x0000000000000000ULL,
44*02346297SLuis Pires       0x0000000000000000ULL, 0x0000000000000000ULL,
45*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL},
46*02346297SLuis Pires     { 0x0000000000000000ULL, 0x0000000000000001ULL,
47*02346297SLuis Pires       0x0000000000000000ULL, 0x0000000000000001ULL,
48*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL},
49*02346297SLuis Pires     { 0x0000000000000000ULL, 0x0000000000000003ULL,
50*02346297SLuis Pires       0x0000000000000000ULL, 0x0000000000000001ULL,
51*02346297SLuis Pires       0x0000000000000002ULL, 0x0000000000000001ULL},
52*02346297SLuis Pires     { 0x0000000000000000ULL, 0x8000000000000000ULL,
53*02346297SLuis Pires       0x0000000000000000ULL, 0x8000000000000000ULL,
54*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL},
55*02346297SLuis Pires     { 0x0000000000000000ULL, 0xa000000000000000ULL,
56*02346297SLuis Pires       0x0000000000000000ULL, 0x0000000000000002ULL,
57*02346297SLuis Pires       0x4000000000000000ULL, 0x2000000000000000ULL},
58*02346297SLuis Pires     { 0x0000000000000000ULL, 0x8000000000000000ULL,
59*02346297SLuis Pires       0x0000000000000000ULL, 0x0000000000000001ULL,
60*02346297SLuis Pires       0x8000000000000000ULL, 0x0000000000000000ULL},
61*02346297SLuis Pires 
62*02346297SLuis Pires     /* Dividend > 64 bits, with MSB 0 */
63*02346297SLuis Pires     { 0x123456789abcdefeULL, 0xefedcba987654321ULL,
64*02346297SLuis Pires       0x123456789abcdefeULL, 0xefedcba987654321ULL,
65*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL},
66*02346297SLuis Pires     { 0x123456789abcdefeULL, 0xefedcba987654321ULL,
67*02346297SLuis Pires       0x0000000000000001ULL, 0x000000000000000dULL,
68*02346297SLuis Pires       0x123456789abcdefeULL, 0x03456789abcdf03bULL},
69*02346297SLuis Pires     { 0x123456789abcdefeULL, 0xefedcba987654321ULL,
70*02346297SLuis Pires       0x0123456789abcdefULL, 0xeefedcba98765432ULL,
71*02346297SLuis Pires       0x0000000000000010ULL, 0x0000000000000001ULL},
72*02346297SLuis Pires 
73*02346297SLuis Pires     /* Dividend > 64 bits, with MSB 1 */
74*02346297SLuis Pires     { 0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
75*02346297SLuis Pires       0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
76*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL},
77*02346297SLuis Pires     { 0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
78*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL,
79*02346297SLuis Pires       0xfeeddccbbaa99887ULL, 0x766554433221100fULL},
80*02346297SLuis Pires     { 0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
81*02346297SLuis Pires       0x0feeddccbbaa9988ULL, 0x7766554433221100ULL,
82*02346297SLuis Pires       0x0000000000000010ULL, 0x000000000000000fULL},
83*02346297SLuis Pires     { 0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
84*02346297SLuis Pires       0x000000000000000eULL, 0x00f0f0f0f0f0f35aULL,
85*02346297SLuis Pires       0x123456789abcdefeULL, 0x0f8922bc55ef90c3ULL},
86*02346297SLuis Pires 
87*02346297SLuis Pires     /**
88*02346297SLuis Pires      * Divisor == 64 bits, with MSB 1
89*02346297SLuis Pires      * and high 64 bits of dividend >= divisor
90*02346297SLuis Pires      * (for testing normalization)
91*02346297SLuis Pires      */
92*02346297SLuis Pires     { 0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
93*02346297SLuis Pires       0x0000000000000001ULL, 0x0000000000000000ULL,
94*02346297SLuis Pires       0xfeeddccbbaa99887ULL, 0x766554433221100fULL},
95*02346297SLuis Pires     { 0xfeeddccbbaa99887ULL, 0x766554433221100fULL,
96*02346297SLuis Pires       0x0000000000000001ULL, 0xfddbb9977553310aULL,
97*02346297SLuis Pires       0x8000000000000001ULL, 0x78899aabbccddf05ULL},
98*02346297SLuis Pires 
99*02346297SLuis Pires     /* Dividend > 64 bits, divisor almost as big */
100*02346297SLuis Pires     { 0x0000000000000001ULL, 0x23456789abcdef01ULL,
101*02346297SLuis Pires       0x0000000000000000ULL, 0x000000000000000fULL,
102*02346297SLuis Pires       0x123456789abcdefeULL, 0x123456789abcde1fULL},
103*02346297SLuis Pires };
104*02346297SLuis Pires 
105*02346297SLuis Pires static const test_data_signed test_table_signed[] = {
106*02346297SLuis Pires     /* Positive dividend, positive/negative divisors */
107*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
108*02346297SLuis Pires       0x0000000000000000LL, 0x0000000000bc614eULL,
109*02346297SLuis Pires       0x0000000000000001LL, 0x0000000000000000LL},
110*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
111*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
112*02346297SLuis Pires       0xffffffffffffffffLL, 0x0000000000000000LL},
113*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
114*02346297SLuis Pires       0x0000000000000000LL, 0x00000000005e30a7ULL,
115*02346297SLuis Pires       0x0000000000000002LL, 0x0000000000000000LL},
116*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
117*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffffa1cf59ULL,
118*02346297SLuis Pires       0xfffffffffffffffeLL, 0x0000000000000000LL},
119*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
120*02346297SLuis Pires       0x0000000000000000LL, 0x0000000000178c29ULL,
121*02346297SLuis Pires       0x0000000000000008LL, 0x0000000000000006LL},
122*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
123*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffffe873d7ULL,
124*02346297SLuis Pires       0xfffffffffffffff8LL, 0x0000000000000006LL},
125*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
126*02346297SLuis Pires       0x0000000000000000LL, 0x000000000000550dULL,
127*02346297SLuis Pires       0x0000000000000237LL, 0x0000000000000183LL},
128*02346297SLuis Pires     { 0x0000000000000000LL, 0x0000000000bc614eULL,
129*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffffffaaf3ULL,
130*02346297SLuis Pires       0xfffffffffffffdc9LL, 0x0000000000000183LL},
131*02346297SLuis Pires 
132*02346297SLuis Pires     /* Negative dividend, positive/negative divisors */
133*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
134*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
135*02346297SLuis Pires       0x0000000000000001LL, 0x0000000000000000LL},
136*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
137*02346297SLuis Pires       0x0000000000000000LL, 0x0000000000bc614eULL,
138*02346297SLuis Pires       0xffffffffffffffffLL, 0x0000000000000000LL},
139*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
140*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffffa1cf59ULL,
141*02346297SLuis Pires       0x0000000000000002LL, 0x0000000000000000LL},
142*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
143*02346297SLuis Pires       0x0000000000000000LL, 0x00000000005e30a7ULL,
144*02346297SLuis Pires       0xfffffffffffffffeLL, 0x0000000000000000LL},
145*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
146*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffffe873d7ULL,
147*02346297SLuis Pires       0x0000000000000008LL, 0xfffffffffffffffaLL},
148*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
149*02346297SLuis Pires       0x0000000000000000LL, 0x0000000000178c29ULL,
150*02346297SLuis Pires       0xfffffffffffffff8LL, 0xfffffffffffffffaLL},
151*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
152*02346297SLuis Pires       0xffffffffffffffffLL, 0xffffffffffffaaf3ULL,
153*02346297SLuis Pires       0x0000000000000237LL, 0xfffffffffffffe7dLL},
154*02346297SLuis Pires     { 0xffffffffffffffffLL, 0xffffffffff439eb2ULL,
155*02346297SLuis Pires       0x0000000000000000LL, 0x000000000000550dULL,
156*02346297SLuis Pires       0xfffffffffffffdc9LL, 0xfffffffffffffe7dLL},
157*02346297SLuis Pires };
158*02346297SLuis Pires 
test_divu128(void)159*02346297SLuis Pires static void test_divu128(void)
160*02346297SLuis Pires {
161*02346297SLuis Pires     int i;
162*02346297SLuis Pires     uint64_t rem;
163*02346297SLuis Pires     test_data_unsigned tmp;
164*02346297SLuis Pires 
165*02346297SLuis Pires     for (i = 0; i < ARRAY_SIZE(test_table_unsigned); ++i) {
166*02346297SLuis Pires         tmp = test_table_unsigned[i];
167*02346297SLuis Pires 
168*02346297SLuis Pires         rem = divu128(&tmp.low, &tmp.high, tmp.divisor);
169*02346297SLuis Pires         g_assert_cmpuint(tmp.low, ==, tmp.rlow);
170*02346297SLuis Pires         g_assert_cmpuint(tmp.high, ==, tmp.rhigh);
171*02346297SLuis Pires         g_assert_cmpuint(rem, ==, tmp.remainder);
172*02346297SLuis Pires     }
173*02346297SLuis Pires }
174*02346297SLuis Pires 
test_divs128(void)175*02346297SLuis Pires static void test_divs128(void)
176*02346297SLuis Pires {
177*02346297SLuis Pires     int i;
178*02346297SLuis Pires     int64_t rem;
179*02346297SLuis Pires     test_data_signed tmp;
180*02346297SLuis Pires 
181*02346297SLuis Pires     for (i = 0; i < ARRAY_SIZE(test_table_signed); ++i) {
182*02346297SLuis Pires         tmp = test_table_signed[i];
183*02346297SLuis Pires 
184*02346297SLuis Pires         rem = divs128(&tmp.low, &tmp.high, tmp.divisor);
185*02346297SLuis Pires         g_assert_cmpuint(tmp.low, ==, tmp.rlow);
186*02346297SLuis Pires         g_assert_cmpuint(tmp.high, ==, tmp.rhigh);
187*02346297SLuis Pires         g_assert_cmpuint(rem, ==, tmp.remainder);
188*02346297SLuis Pires     }
189*02346297SLuis Pires }
190*02346297SLuis Pires 
main(int argc,char ** argv)191*02346297SLuis Pires int main(int argc, char **argv)
192*02346297SLuis Pires {
193*02346297SLuis Pires     g_test_init(&argc, &argv, NULL);
194*02346297SLuis Pires     g_test_add_func("/host-utils/test_divu128", test_divu128);
195*02346297SLuis Pires     g_test_add_func("/host-utils/test_divs128", test_divs128);
196*02346297SLuis Pires     return g_test_run();
197*02346297SLuis Pires }
198