Line data Source code
1 : /* Simple multiprecision - small MP library for testing
2 : * Copyright (C) 2022-2024 Frediano Ziglio
3 : *
4 : * This library is free software; you can redistribute it and/or
5 : * modify it under the terms of the GNU Library General Public
6 : * License as published by the Free Software Foundation; either
7 : * version 2 of the License, or (at your option) any later version.
8 : *
9 : * This library is distributed in the hope that it will be useful,
10 : * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 : * Library General Public License for more details.
13 : *
14 : * You should have received a copy of the GNU Library General Public
15 : * License along with this library; if not, write to the
16 : * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 : * Boston, MA 02111-1307, USA.
18 : */
19 :
20 : #undef NDEBUG
21 : #include <config.h>
22 :
23 : #include <assert.h>
24 : #include <string.h>
25 :
26 : #include <freetds/utils/smp.h>
27 : #include <freetds/sysdep_private.h>
28 :
29 : #define SMP_NUM_COMPONENTS \
30 : (sizeof(((smp*)0)->comp) / sizeof(((smp*)0)->comp[0]))
31 :
32 : const smp smp_zero = {
33 : { 0, 0, 0, 0, 0, 0, 0, 0 },
34 : };
35 :
36 : const smp smp_one = {
37 : { 1, 0, 0, 0, 0, 0, 0, 0 },
38 : };
39 :
40 : smp
41 6614872 : smp_add(smp a, smp b)
42 : {
43 : int i;
44 6614872 : uint32_t carry = 0;
45 : smp res;
46 :
47 59533848 : for (i = 0; i < SMP_NUM_COMPONENTS; ++i) {
48 52918976 : uint32_t sum = carry + a.comp[i] + b.comp[i];
49 52918976 : res.comp[i] = (uint16_t) (sum & 0xffffu);
50 52918976 : carry = sum >> 16;
51 : }
52 6614872 : assert(smp_is_negative(a) != smp_is_negative(b) || smp_is_negative(a) == smp_is_negative(res));
53 6614872 : return res;
54 : }
55 :
56 : smp
57 879208 : smp_not(smp a)
58 : {
59 : int i;
60 : smp res;
61 7912872 : for (i = 0; i < SMP_NUM_COMPONENTS; ++i)
62 7033664 : res.comp[i] = a.comp[i] ^ 0xffffu;
63 879208 : return res;
64 : }
65 :
66 : smp
67 879208 : smp_negate(smp a)
68 : {
69 879208 : return smp_add(smp_not(a), smp_one);
70 : }
71 :
72 : smp
73 1605392 : smp_from_int(int64_t n)
74 : {
75 : int i;
76 : uint64_t un;
77 : smp res;
78 :
79 1605392 : if (n >= 0) {
80 1274968 : un = (uint64_t) n;
81 : } else {
82 330424 : un = (uint64_t) -n;
83 : }
84 12843136 : for (i = 0; i < SMP_NUM_COMPONENTS; ++i) {
85 12843136 : res.comp[i] = (uint16_t) (un & 0xffffu);
86 12843136 : un = un >> 16;
87 : }
88 1605392 : if (n < 0)
89 330424 : return smp_negate(res);
90 1274968 : return res;
91 : }
92 :
93 : bool
94 25902752 : smp_is_negative(smp a)
95 : {
96 25902752 : return (a.comp[SMP_NUM_COMPONENTS-1] >> 15) != 0;
97 : }
98 :
99 : bool
100 7319984 : smp_is_zero(smp a)
101 : {
102 : int i;
103 7319984 : uint16_t or = 0;
104 :
105 65879856 : for (i = 0; i < SMP_NUM_COMPONENTS; ++i)
106 58559872 : or |= a.comp[i];
107 7319984 : return or == 0;
108 : }
109 :
110 : smp
111 432 : smp_sub(smp a, smp b)
112 : {
113 432 : smp c = smp_negate(b);
114 432 : return smp_add(a, c);
115 : }
116 :
117 : double
118 102256 : smp_to_double(smp a)
119 : {
120 : int i;
121 102256 : double n = 0;
122 102256 : double mult = 1.0;
123 102256 : smp b = a;
124 102256 : if (smp_is_negative(a))
125 51024 : b = smp_negate(b);
126 920304 : for (i = 0; i < SMP_NUM_COMPONENTS; ++i) {
127 818048 : n += b.comp[i] * mult;
128 818048 : mult = mult * 65536.0;
129 : }
130 102256 : if (smp_is_negative(a))
131 51024 : return -n;
132 : return n;
133 : }
134 :
135 : int
136 24 : smp_cmp(smp a, smp b)
137 : {
138 24 : smp diff = smp_sub(a, b);
139 24 : if (smp_is_negative(diff))
140 : return -1;
141 16 : if (smp_is_zero(diff))
142 : return 0;
143 8 : return 1;
144 : }
145 :
146 : // divide and return remainder
147 : static uint16_t
148 : div_small(smp *n, uint16_t div)
149 : {
150 : int i;
151 7291736 : uint32_t remainder = 0;
152 72917360 : for (i = SMP_NUM_COMPONENTS; --i >= 0;) {
153 58333888 : uint32_t comp = remainder * 0x10000u + n->comp[i];
154 58333888 : remainder = comp % div;
155 58333888 : n->comp[i] = (uint16_t) (comp / div);
156 : }
157 : return (uint16_t) remainder;
158 : }
159 :
160 : char *
161 795064 : smp_to_string(smp a)
162 : {
163 : char buf[SMP_NUM_COMPONENTS * 16 * 4 / 10 + 2];
164 795064 : char *p = buf+sizeof(buf);
165 795064 : const bool negative = smp_is_negative(a);
166 795064 : smp n = negative ? smp_negate(a) : a;
167 :
168 795064 : *--p = 0;
169 : do
170 7291736 : *--p = div_small(&n, 10) + '0';
171 7291736 : while (!smp_is_zero(n));
172 795064 : if (negative)
173 396504 : *--p = '-';
174 795064 : return strdup(p);
175 : }
176 :
177 : // multiple a number
178 : static void
179 908784 : mul_small(smp *n, uint16_t factor)
180 : {
181 : int i;
182 908784 : uint32_t carry = 0;
183 8179056 : for (i = 0; i < SMP_NUM_COMPONENTS; ++i) {
184 7270272 : uint32_t comp = (uint32_t) n->comp[i] * factor + carry;
185 7270272 : carry = comp >> 16;
186 7270272 : n->comp[i] = (uint16_t) (comp & 0xffffu);
187 : }
188 908784 : assert(carry == 0);
189 908784 : }
190 :
191 : smp
192 98872 : smp_from_string(const char *s)
193 : {
194 98872 : bool negative = false;
195 98872 : smp n = smp_zero;
196 98872 : uint16_t base = 10;
197 :
198 98872 : switch (*s) {
199 49280 : case '-':
200 49280 : negative = true;
201 49280 : case '+':
202 49280 : ++s;
203 : }
204 98872 : if (*s == '0') {
205 280 : base = 8;
206 280 : ++s;
207 280 : if (*s == 'x' || *s == 'X') {
208 48 : ++s;
209 48 : base = 16;
210 : }
211 : }
212 908872 : for (;*s; ++s) {
213 908872 : int digit = 0;
214 908872 : if (*s == '\'')
215 88 : continue;
216 908784 : mul_small(&n, base);
217 908784 : if (*s >= '0' && *s <= '9')
218 908784 : digit = *s - '0';
219 0 : else if (*s >= 'a' && *s <= 'z')
220 0 : digit = *s - 'a' + 10;
221 0 : else if (*s >= 'A' && *s <= 'Z')
222 0 : digit = *s - 'A' + 10;
223 : else
224 : assert(!!"Invalid digit entered");
225 908784 : n = smp_add(n, smp_from_int(digit));
226 : }
227 98872 : if (negative)
228 49280 : return smp_negate(n);
229 49592 : return n;
230 : }
|