1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2017 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker
5*3ac0a46fSAndroid Build Coastguard Worker // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6*3ac0a46fSAndroid Build Coastguard Worker
7*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fgas/crt/cfgas_decimal.h"
8*3ac0a46fSAndroid Build Coastguard Worker
9*3ac0a46fSAndroid Build Coastguard Worker #include <math.h>
10*3ac0a46fSAndroid Build Coastguard Worker
11*3ac0a46fSAndroid Build Coastguard Worker #include <algorithm>
12*3ac0a46fSAndroid Build Coastguard Worker #include <limits>
13*3ac0a46fSAndroid Build Coastguard Worker #include <utility>
14*3ac0a46fSAndroid Build Coastguard Worker
15*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_extension.h"
16*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/check.h"
17*3ac0a46fSAndroid Build Coastguard Worker
18*3ac0a46fSAndroid Build Coastguard Worker #define FXMATH_DECIMAL_SCALELIMIT 0x1c
19*3ac0a46fSAndroid Build Coastguard Worker #define FXMATH_DECIMAL_RSHIFT32BIT(x) ((x) >> 0x10 >> 0x10)
20*3ac0a46fSAndroid Build Coastguard Worker #define FXMATH_DECIMAL_LSHIFT32BIT(x) ((x) << 0x10 << 0x10)
21*3ac0a46fSAndroid Build Coastguard Worker
22*3ac0a46fSAndroid Build Coastguard Worker namespace {
23*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_div10(uint64_t & phi,uint64_t & pmid,uint64_t & plo)24*3ac0a46fSAndroid Build Coastguard Worker inline uint8_t decimal_helper_div10(uint64_t& phi,
25*3ac0a46fSAndroid Build Coastguard Worker uint64_t& pmid,
26*3ac0a46fSAndroid Build Coastguard Worker uint64_t& plo) {
27*3ac0a46fSAndroid Build Coastguard Worker uint8_t retVal;
28*3ac0a46fSAndroid Build Coastguard Worker pmid += FXMATH_DECIMAL_LSHIFT32BIT(phi % 0xA);
29*3ac0a46fSAndroid Build Coastguard Worker phi /= 0xA;
30*3ac0a46fSAndroid Build Coastguard Worker plo += FXMATH_DECIMAL_LSHIFT32BIT(pmid % 0xA);
31*3ac0a46fSAndroid Build Coastguard Worker pmid /= 0xA;
32*3ac0a46fSAndroid Build Coastguard Worker retVal = plo % 0xA;
33*3ac0a46fSAndroid Build Coastguard Worker plo /= 0xA;
34*3ac0a46fSAndroid Build Coastguard Worker return retVal;
35*3ac0a46fSAndroid Build Coastguard Worker }
36*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_div10_any(uint64_t nums[],uint8_t numcount)37*3ac0a46fSAndroid Build Coastguard Worker inline uint8_t decimal_helper_div10_any(uint64_t nums[], uint8_t numcount) {
38*3ac0a46fSAndroid Build Coastguard Worker uint8_t retVal = 0;
39*3ac0a46fSAndroid Build Coastguard Worker for (int i = numcount - 1; i > 0; i--) {
40*3ac0a46fSAndroid Build Coastguard Worker nums[i - 1] += FXMATH_DECIMAL_LSHIFT32BIT(nums[i] % 0xA);
41*3ac0a46fSAndroid Build Coastguard Worker nums[i] /= 0xA;
42*3ac0a46fSAndroid Build Coastguard Worker }
43*3ac0a46fSAndroid Build Coastguard Worker if (numcount) {
44*3ac0a46fSAndroid Build Coastguard Worker retVal = nums[0] % 0xA;
45*3ac0a46fSAndroid Build Coastguard Worker nums[0] /= 0xA;
46*3ac0a46fSAndroid Build Coastguard Worker }
47*3ac0a46fSAndroid Build Coastguard Worker return retVal;
48*3ac0a46fSAndroid Build Coastguard Worker }
49*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_mul10(uint64_t & phi,uint64_t & pmid,uint64_t & plo)50*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_mul10(uint64_t& phi, uint64_t& pmid, uint64_t& plo) {
51*3ac0a46fSAndroid Build Coastguard Worker plo *= 0xA;
52*3ac0a46fSAndroid Build Coastguard Worker pmid = pmid * 0xA + FXMATH_DECIMAL_RSHIFT32BIT(plo);
53*3ac0a46fSAndroid Build Coastguard Worker plo = (uint32_t)plo;
54*3ac0a46fSAndroid Build Coastguard Worker phi = phi * 0xA + FXMATH_DECIMAL_RSHIFT32BIT(pmid);
55*3ac0a46fSAndroid Build Coastguard Worker pmid = (uint32_t)pmid;
56*3ac0a46fSAndroid Build Coastguard Worker }
57*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_mul10_any(uint64_t nums[],uint8_t numcount)58*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_mul10_any(uint64_t nums[], uint8_t numcount) {
59*3ac0a46fSAndroid Build Coastguard Worker nums[0] *= 0xA;
60*3ac0a46fSAndroid Build Coastguard Worker for (int i = 1; i < numcount; i++) {
61*3ac0a46fSAndroid Build Coastguard Worker nums[i] = nums[i] * 0xA + FXMATH_DECIMAL_RSHIFT32BIT(nums[i - 1]);
62*3ac0a46fSAndroid Build Coastguard Worker nums[i - 1] = (uint32_t)nums[i - 1];
63*3ac0a46fSAndroid Build Coastguard Worker }
64*3ac0a46fSAndroid Build Coastguard Worker }
65*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_normalize(uint64_t & phi,uint64_t & pmid,uint64_t & plo)66*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_normalize(uint64_t& phi,
67*3ac0a46fSAndroid Build Coastguard Worker uint64_t& pmid,
68*3ac0a46fSAndroid Build Coastguard Worker uint64_t& plo) {
69*3ac0a46fSAndroid Build Coastguard Worker phi += FXMATH_DECIMAL_RSHIFT32BIT(pmid);
70*3ac0a46fSAndroid Build Coastguard Worker pmid = (uint32_t)pmid;
71*3ac0a46fSAndroid Build Coastguard Worker pmid += FXMATH_DECIMAL_RSHIFT32BIT(plo);
72*3ac0a46fSAndroid Build Coastguard Worker plo = (uint32_t)plo;
73*3ac0a46fSAndroid Build Coastguard Worker phi += FXMATH_DECIMAL_RSHIFT32BIT(pmid);
74*3ac0a46fSAndroid Build Coastguard Worker pmid = (uint32_t)pmid;
75*3ac0a46fSAndroid Build Coastguard Worker }
76*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_normalize_any(uint64_t nums[],uint8_t len)77*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_normalize_any(uint64_t nums[], uint8_t len) {
78*3ac0a46fSAndroid Build Coastguard Worker for (int i = len - 2; i > 0; i--) {
79*3ac0a46fSAndroid Build Coastguard Worker nums[i + 1] += FXMATH_DECIMAL_RSHIFT32BIT(nums[i]);
80*3ac0a46fSAndroid Build Coastguard Worker nums[i] = (uint32_t)nums[i];
81*3ac0a46fSAndroid Build Coastguard Worker }
82*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < len - 1; i++) {
83*3ac0a46fSAndroid Build Coastguard Worker nums[i + 1] += FXMATH_DECIMAL_RSHIFT32BIT(nums[i]);
84*3ac0a46fSAndroid Build Coastguard Worker nums[i] = (uint32_t)nums[i];
85*3ac0a46fSAndroid Build Coastguard Worker }
86*3ac0a46fSAndroid Build Coastguard Worker }
87*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_raw_compare_any(uint64_t a[],uint8_t al,uint64_t b[],uint8_t bl)88*3ac0a46fSAndroid Build Coastguard Worker inline int8_t decimal_helper_raw_compare_any(uint64_t a[],
89*3ac0a46fSAndroid Build Coastguard Worker uint8_t al,
90*3ac0a46fSAndroid Build Coastguard Worker uint64_t b[],
91*3ac0a46fSAndroid Build Coastguard Worker uint8_t bl) {
92*3ac0a46fSAndroid Build Coastguard Worker int8_t retVal = 0;
93*3ac0a46fSAndroid Build Coastguard Worker for (int i = std::max(al - 1, bl - 1); i >= 0; i--) {
94*3ac0a46fSAndroid Build Coastguard Worker uint64_t l = (i >= al ? 0 : a[i]), r = (i >= bl ? 0 : b[i]);
95*3ac0a46fSAndroid Build Coastguard Worker retVal += (l > r ? 1 : (l < r ? -1 : 0));
96*3ac0a46fSAndroid Build Coastguard Worker if (retVal)
97*3ac0a46fSAndroid Build Coastguard Worker return retVal;
98*3ac0a46fSAndroid Build Coastguard Worker }
99*3ac0a46fSAndroid Build Coastguard Worker return retVal;
100*3ac0a46fSAndroid Build Coastguard Worker }
101*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_dec_any(uint64_t a[],uint8_t al)102*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_dec_any(uint64_t a[], uint8_t al) {
103*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < al; i++) {
104*3ac0a46fSAndroid Build Coastguard Worker if (a[i]--)
105*3ac0a46fSAndroid Build Coastguard Worker return;
106*3ac0a46fSAndroid Build Coastguard Worker }
107*3ac0a46fSAndroid Build Coastguard Worker }
108*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_inc_any(uint64_t a[],uint8_t al)109*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_inc_any(uint64_t a[], uint8_t al) {
110*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < al; i++) {
111*3ac0a46fSAndroid Build Coastguard Worker a[i]++;
112*3ac0a46fSAndroid Build Coastguard Worker if ((uint32_t)a[i] == a[i])
113*3ac0a46fSAndroid Build Coastguard Worker return;
114*3ac0a46fSAndroid Build Coastguard Worker a[i] = 0;
115*3ac0a46fSAndroid Build Coastguard Worker }
116*3ac0a46fSAndroid Build Coastguard Worker }
117*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_raw_mul(uint64_t a[],uint8_t al,uint64_t b[],uint8_t bl,uint64_t c[],uint8_t cl)118*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_raw_mul(uint64_t a[],
119*3ac0a46fSAndroid Build Coastguard Worker uint8_t al,
120*3ac0a46fSAndroid Build Coastguard Worker uint64_t b[],
121*3ac0a46fSAndroid Build Coastguard Worker uint8_t bl,
122*3ac0a46fSAndroid Build Coastguard Worker uint64_t c[],
123*3ac0a46fSAndroid Build Coastguard Worker uint8_t cl) {
124*3ac0a46fSAndroid Build Coastguard Worker DCHECK(al + bl <= cl);
125*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < cl; i++)
126*3ac0a46fSAndroid Build Coastguard Worker c[i] = 0;
127*3ac0a46fSAndroid Build Coastguard Worker
128*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < al; i++) {
129*3ac0a46fSAndroid Build Coastguard Worker for (int j = 0; j < bl; j++) {
130*3ac0a46fSAndroid Build Coastguard Worker uint64_t m = (uint64_t)a[i] * b[j];
131*3ac0a46fSAndroid Build Coastguard Worker c[i + j] += (uint32_t)m;
132*3ac0a46fSAndroid Build Coastguard Worker c[i + j + 1] += FXMATH_DECIMAL_RSHIFT32BIT(m);
133*3ac0a46fSAndroid Build Coastguard Worker }
134*3ac0a46fSAndroid Build Coastguard Worker }
135*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < cl - 1; i++) {
136*3ac0a46fSAndroid Build Coastguard Worker c[i + 1] += FXMATH_DECIMAL_RSHIFT32BIT(c[i]);
137*3ac0a46fSAndroid Build Coastguard Worker c[i] = (uint32_t)c[i];
138*3ac0a46fSAndroid Build Coastguard Worker }
139*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < cl; i++)
140*3ac0a46fSAndroid Build Coastguard Worker c[i] = (uint32_t)c[i];
141*3ac0a46fSAndroid Build Coastguard Worker }
142*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_raw_div(uint64_t a[],uint8_t al,uint64_t b[],uint8_t bl,uint64_t c[],uint8_t cl)143*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_raw_div(uint64_t a[],
144*3ac0a46fSAndroid Build Coastguard Worker uint8_t al,
145*3ac0a46fSAndroid Build Coastguard Worker uint64_t b[],
146*3ac0a46fSAndroid Build Coastguard Worker uint8_t bl,
147*3ac0a46fSAndroid Build Coastguard Worker uint64_t c[],
148*3ac0a46fSAndroid Build Coastguard Worker uint8_t cl) {
149*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < cl; i++)
150*3ac0a46fSAndroid Build Coastguard Worker c[i] = 0;
151*3ac0a46fSAndroid Build Coastguard Worker
152*3ac0a46fSAndroid Build Coastguard Worker uint64_t left[16] = {0};
153*3ac0a46fSAndroid Build Coastguard Worker uint64_t right[16] = {0};
154*3ac0a46fSAndroid Build Coastguard Worker left[0] = 0;
155*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < al; i++)
156*3ac0a46fSAndroid Build Coastguard Worker right[i] = a[i];
157*3ac0a46fSAndroid Build Coastguard Worker
158*3ac0a46fSAndroid Build Coastguard Worker uint64_t tmp[16];
159*3ac0a46fSAndroid Build Coastguard Worker while (decimal_helper_raw_compare_any(left, al, right, al) <= 0) {
160*3ac0a46fSAndroid Build Coastguard Worker uint64_t cur[16];
161*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < al; i++)
162*3ac0a46fSAndroid Build Coastguard Worker cur[i] = left[i] + right[i];
163*3ac0a46fSAndroid Build Coastguard Worker
164*3ac0a46fSAndroid Build Coastguard Worker for (int i = al - 1; i >= 0; i--) {
165*3ac0a46fSAndroid Build Coastguard Worker if (i)
166*3ac0a46fSAndroid Build Coastguard Worker cur[i - 1] += FXMATH_DECIMAL_LSHIFT32BIT(cur[i] % 2);
167*3ac0a46fSAndroid Build Coastguard Worker cur[i] /= 2;
168*3ac0a46fSAndroid Build Coastguard Worker }
169*3ac0a46fSAndroid Build Coastguard Worker
170*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_raw_mul(cur, al, b, bl, tmp, 16);
171*3ac0a46fSAndroid Build Coastguard Worker switch (decimal_helper_raw_compare_any(tmp, 16, a, al)) {
172*3ac0a46fSAndroid Build Coastguard Worker case -1:
173*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < 16; i++)
174*3ac0a46fSAndroid Build Coastguard Worker left[i] = cur[i];
175*3ac0a46fSAndroid Build Coastguard Worker
176*3ac0a46fSAndroid Build Coastguard Worker left[0]++;
177*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_normalize_any(left, al);
178*3ac0a46fSAndroid Build Coastguard Worker break;
179*3ac0a46fSAndroid Build Coastguard Worker case 1:
180*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < 16; i++)
181*3ac0a46fSAndroid Build Coastguard Worker right[i] = cur[i];
182*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_dec_any(right, al);
183*3ac0a46fSAndroid Build Coastguard Worker break;
184*3ac0a46fSAndroid Build Coastguard Worker case 0:
185*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < std::min(al, cl); i++)
186*3ac0a46fSAndroid Build Coastguard Worker c[i] = cur[i];
187*3ac0a46fSAndroid Build Coastguard Worker return;
188*3ac0a46fSAndroid Build Coastguard Worker }
189*3ac0a46fSAndroid Build Coastguard Worker }
190*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < std::min(al, cl); i++)
191*3ac0a46fSAndroid Build Coastguard Worker c[i] = left[i];
192*3ac0a46fSAndroid Build Coastguard Worker }
193*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_outofrange(uint64_t a[],uint8_t al,uint8_t goal)194*3ac0a46fSAndroid Build Coastguard Worker inline bool decimal_helper_outofrange(uint64_t a[], uint8_t al, uint8_t goal) {
195*3ac0a46fSAndroid Build Coastguard Worker for (int i = goal; i < al; i++) {
196*3ac0a46fSAndroid Build Coastguard Worker if (a[i])
197*3ac0a46fSAndroid Build Coastguard Worker return true;
198*3ac0a46fSAndroid Build Coastguard Worker }
199*3ac0a46fSAndroid Build Coastguard Worker return false;
200*3ac0a46fSAndroid Build Coastguard Worker }
201*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_shrinkintorange(uint64_t a[],uint8_t al,uint8_t goal,uint8_t & scale)202*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_shrinkintorange(uint64_t a[],
203*3ac0a46fSAndroid Build Coastguard Worker uint8_t al,
204*3ac0a46fSAndroid Build Coastguard Worker uint8_t goal,
205*3ac0a46fSAndroid Build Coastguard Worker uint8_t& scale) {
206*3ac0a46fSAndroid Build Coastguard Worker bool bRoundUp = false;
207*3ac0a46fSAndroid Build Coastguard Worker while (scale != 0 && (scale > FXMATH_DECIMAL_SCALELIMIT ||
208*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_outofrange(a, al, goal))) {
209*3ac0a46fSAndroid Build Coastguard Worker bRoundUp = decimal_helper_div10_any(a, al) >= 5;
210*3ac0a46fSAndroid Build Coastguard Worker scale--;
211*3ac0a46fSAndroid Build Coastguard Worker }
212*3ac0a46fSAndroid Build Coastguard Worker if (bRoundUp) {
213*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_normalize_any(a, goal);
214*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_inc_any(a, goal);
215*3ac0a46fSAndroid Build Coastguard Worker }
216*3ac0a46fSAndroid Build Coastguard Worker }
217*3ac0a46fSAndroid Build Coastguard Worker
decimal_helper_truncate(uint64_t & phi,uint64_t & pmid,uint64_t & plo,uint8_t & scale,uint8_t minscale=0)218*3ac0a46fSAndroid Build Coastguard Worker inline void decimal_helper_truncate(uint64_t& phi,
219*3ac0a46fSAndroid Build Coastguard Worker uint64_t& pmid,
220*3ac0a46fSAndroid Build Coastguard Worker uint64_t& plo,
221*3ac0a46fSAndroid Build Coastguard Worker uint8_t& scale,
222*3ac0a46fSAndroid Build Coastguard Worker uint8_t minscale = 0) {
223*3ac0a46fSAndroid Build Coastguard Worker while (scale > minscale) {
224*3ac0a46fSAndroid Build Coastguard Worker uint64_t thi = phi, tmid = pmid, tlo = plo;
225*3ac0a46fSAndroid Build Coastguard Worker if (decimal_helper_div10(thi, tmid, tlo) != 0)
226*3ac0a46fSAndroid Build Coastguard Worker break;
227*3ac0a46fSAndroid Build Coastguard Worker
228*3ac0a46fSAndroid Build Coastguard Worker phi = thi;
229*3ac0a46fSAndroid Build Coastguard Worker pmid = tmid;
230*3ac0a46fSAndroid Build Coastguard Worker plo = tlo;
231*3ac0a46fSAndroid Build Coastguard Worker scale--;
232*3ac0a46fSAndroid Build Coastguard Worker }
233*3ac0a46fSAndroid Build Coastguard Worker }
234*3ac0a46fSAndroid Build Coastguard Worker
235*3ac0a46fSAndroid Build Coastguard Worker } // namespace
236*3ac0a46fSAndroid Build Coastguard Worker
237*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal() = default;
238*3ac0a46fSAndroid Build Coastguard Worker
CFGAS_Decimal(uint64_t val)239*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal(uint64_t val)
240*3ac0a46fSAndroid Build Coastguard Worker : m_uMid(static_cast<uint32_t>(FXMATH_DECIMAL_RSHIFT32BIT(val))),
241*3ac0a46fSAndroid Build Coastguard Worker m_uLo(static_cast<uint32_t>(val)) {}
242*3ac0a46fSAndroid Build Coastguard Worker
CFGAS_Decimal(uint32_t val)243*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal(uint32_t val)
244*3ac0a46fSAndroid Build Coastguard Worker : m_uLo(static_cast<uint32_t>(val)) {}
245*3ac0a46fSAndroid Build Coastguard Worker
CFGAS_Decimal(uint32_t lo,uint32_t mid,uint32_t hi,bool neg,uint8_t scale)246*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal(uint32_t lo,
247*3ac0a46fSAndroid Build Coastguard Worker uint32_t mid,
248*3ac0a46fSAndroid Build Coastguard Worker uint32_t hi,
249*3ac0a46fSAndroid Build Coastguard Worker bool neg,
250*3ac0a46fSAndroid Build Coastguard Worker uint8_t scale)
251*3ac0a46fSAndroid Build Coastguard Worker : m_uHi(hi),
252*3ac0a46fSAndroid Build Coastguard Worker m_uMid(mid),
253*3ac0a46fSAndroid Build Coastguard Worker m_uLo(lo),
254*3ac0a46fSAndroid Build Coastguard Worker m_bNeg(neg && IsNotZero()),
255*3ac0a46fSAndroid Build Coastguard Worker m_uScale(scale > FXMATH_DECIMAL_SCALELIMIT ? 0 : scale) {}
256*3ac0a46fSAndroid Build Coastguard Worker
CFGAS_Decimal(int32_t val)257*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal(int32_t val) {
258*3ac0a46fSAndroid Build Coastguard Worker if (val >= 0) {
259*3ac0a46fSAndroid Build Coastguard Worker *this = CFGAS_Decimal(static_cast<uint32_t>(val));
260*3ac0a46fSAndroid Build Coastguard Worker } else if (val == std::numeric_limits<int32_t>::min()) {
261*3ac0a46fSAndroid Build Coastguard Worker *this = CFGAS_Decimal(static_cast<uint32_t>(val));
262*3ac0a46fSAndroid Build Coastguard Worker SetNegate();
263*3ac0a46fSAndroid Build Coastguard Worker } else {
264*3ac0a46fSAndroid Build Coastguard Worker *this = CFGAS_Decimal(static_cast<uint32_t>(-val));
265*3ac0a46fSAndroid Build Coastguard Worker SetNegate();
266*3ac0a46fSAndroid Build Coastguard Worker }
267*3ac0a46fSAndroid Build Coastguard Worker }
268*3ac0a46fSAndroid Build Coastguard Worker
CFGAS_Decimal(float val,uint8_t scale)269*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal(float val, uint8_t scale) {
270*3ac0a46fSAndroid Build Coastguard Worker float newval = fabs(val);
271*3ac0a46fSAndroid Build Coastguard Worker float divisor = powf(2.0, 64.0f);
272*3ac0a46fSAndroid Build Coastguard Worker uint64_t bottom64 = static_cast<uint64_t>(fmodf(newval, divisor));
273*3ac0a46fSAndroid Build Coastguard Worker uint64_t top64 = static_cast<uint64_t>(newval / divisor);
274*3ac0a46fSAndroid Build Coastguard Worker uint64_t plo = bottom64 & 0xFFFFFFFF;
275*3ac0a46fSAndroid Build Coastguard Worker uint64_t pmid = bottom64 >> 32;
276*3ac0a46fSAndroid Build Coastguard Worker uint64_t phi = top64 & 0xFFFFFFFF;
277*3ac0a46fSAndroid Build Coastguard Worker
278*3ac0a46fSAndroid Build Coastguard Worker newval = fmodf(newval, 1.0f);
279*3ac0a46fSAndroid Build Coastguard Worker for (uint8_t iter = 0; iter < scale; iter++) {
280*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_mul10(phi, pmid, plo);
281*3ac0a46fSAndroid Build Coastguard Worker newval *= 10;
282*3ac0a46fSAndroid Build Coastguard Worker plo += static_cast<uint64_t>(newval);
283*3ac0a46fSAndroid Build Coastguard Worker newval = fmodf(newval, 1.0f);
284*3ac0a46fSAndroid Build Coastguard Worker }
285*3ac0a46fSAndroid Build Coastguard Worker
286*3ac0a46fSAndroid Build Coastguard Worker plo += FXSYS_roundf(newval);
287*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_normalize(phi, pmid, plo);
288*3ac0a46fSAndroid Build Coastguard Worker m_uHi = static_cast<uint32_t>(phi);
289*3ac0a46fSAndroid Build Coastguard Worker m_uMid = static_cast<uint32_t>(pmid);
290*3ac0a46fSAndroid Build Coastguard Worker m_uLo = static_cast<uint32_t>(plo);
291*3ac0a46fSAndroid Build Coastguard Worker m_bNeg = val < 0 && IsNotZero();
292*3ac0a46fSAndroid Build Coastguard Worker m_uScale = scale;
293*3ac0a46fSAndroid Build Coastguard Worker }
294*3ac0a46fSAndroid Build Coastguard Worker
CFGAS_Decimal(WideStringView strObj)295*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal::CFGAS_Decimal(WideStringView strObj) {
296*3ac0a46fSAndroid Build Coastguard Worker const wchar_t* str = strObj.unterminated_c_str();
297*3ac0a46fSAndroid Build Coastguard Worker const wchar_t* strBound = str + strObj.GetLength();
298*3ac0a46fSAndroid Build Coastguard Worker bool pointmet = false;
299*3ac0a46fSAndroid Build Coastguard Worker bool negmet = false;
300*3ac0a46fSAndroid Build Coastguard Worker uint8_t scale = 0;
301*3ac0a46fSAndroid Build Coastguard Worker while (str != strBound && *str == ' ')
302*3ac0a46fSAndroid Build Coastguard Worker str++;
303*3ac0a46fSAndroid Build Coastguard Worker if (str != strBound && *str == '-') {
304*3ac0a46fSAndroid Build Coastguard Worker negmet = true;
305*3ac0a46fSAndroid Build Coastguard Worker str++;
306*3ac0a46fSAndroid Build Coastguard Worker } else if (str != strBound && *str == '+') {
307*3ac0a46fSAndroid Build Coastguard Worker str++;
308*3ac0a46fSAndroid Build Coastguard Worker }
309*3ac0a46fSAndroid Build Coastguard Worker
310*3ac0a46fSAndroid Build Coastguard Worker while (str != strBound && (FXSYS_IsDecimalDigit(*str) || *str == '.') &&
311*3ac0a46fSAndroid Build Coastguard Worker scale < FXMATH_DECIMAL_SCALELIMIT) {
312*3ac0a46fSAndroid Build Coastguard Worker if (*str == '.') {
313*3ac0a46fSAndroid Build Coastguard Worker if (!pointmet)
314*3ac0a46fSAndroid Build Coastguard Worker pointmet = true;
315*3ac0a46fSAndroid Build Coastguard Worker } else {
316*3ac0a46fSAndroid Build Coastguard Worker m_uHi = m_uHi * 0xA + FXMATH_DECIMAL_RSHIFT32BIT((uint64_t)m_uMid * 0xA);
317*3ac0a46fSAndroid Build Coastguard Worker m_uMid = m_uMid * 0xA + FXMATH_DECIMAL_RSHIFT32BIT((uint64_t)m_uLo * 0xA);
318*3ac0a46fSAndroid Build Coastguard Worker m_uLo = m_uLo * 0xA + (*str - '0');
319*3ac0a46fSAndroid Build Coastguard Worker if (pointmet)
320*3ac0a46fSAndroid Build Coastguard Worker scale++;
321*3ac0a46fSAndroid Build Coastguard Worker }
322*3ac0a46fSAndroid Build Coastguard Worker str++;
323*3ac0a46fSAndroid Build Coastguard Worker }
324*3ac0a46fSAndroid Build Coastguard Worker m_bNeg = negmet && IsNotZero();
325*3ac0a46fSAndroid Build Coastguard Worker m_uScale = scale;
326*3ac0a46fSAndroid Build Coastguard Worker }
327*3ac0a46fSAndroid Build Coastguard Worker
ToWideString() const328*3ac0a46fSAndroid Build Coastguard Worker WideString CFGAS_Decimal::ToWideString() const {
329*3ac0a46fSAndroid Build Coastguard Worker WideString retString;
330*3ac0a46fSAndroid Build Coastguard Worker WideString tmpbuf;
331*3ac0a46fSAndroid Build Coastguard Worker uint64_t phi = m_uHi;
332*3ac0a46fSAndroid Build Coastguard Worker uint64_t pmid = m_uMid;
333*3ac0a46fSAndroid Build Coastguard Worker uint64_t plo = m_uLo;
334*3ac0a46fSAndroid Build Coastguard Worker while (phi || pmid || plo)
335*3ac0a46fSAndroid Build Coastguard Worker tmpbuf += decimal_helper_div10(phi, pmid, plo) + '0';
336*3ac0a46fSAndroid Build Coastguard Worker
337*3ac0a46fSAndroid Build Coastguard Worker uint8_t outputlen = (uint8_t)tmpbuf.GetLength();
338*3ac0a46fSAndroid Build Coastguard Worker uint8_t scale = m_uScale;
339*3ac0a46fSAndroid Build Coastguard Worker while (scale >= outputlen) {
340*3ac0a46fSAndroid Build Coastguard Worker tmpbuf += '0';
341*3ac0a46fSAndroid Build Coastguard Worker outputlen++;
342*3ac0a46fSAndroid Build Coastguard Worker }
343*3ac0a46fSAndroid Build Coastguard Worker if (m_bNeg && IsNotZero())
344*3ac0a46fSAndroid Build Coastguard Worker retString += '-';
345*3ac0a46fSAndroid Build Coastguard Worker
346*3ac0a46fSAndroid Build Coastguard Worker for (uint8_t idx = 0; idx < outputlen; idx++) {
347*3ac0a46fSAndroid Build Coastguard Worker if (idx == (outputlen - scale) && scale != 0)
348*3ac0a46fSAndroid Build Coastguard Worker retString += '.';
349*3ac0a46fSAndroid Build Coastguard Worker retString += tmpbuf[outputlen - 1 - idx];
350*3ac0a46fSAndroid Build Coastguard Worker }
351*3ac0a46fSAndroid Build Coastguard Worker return retString;
352*3ac0a46fSAndroid Build Coastguard Worker }
353*3ac0a46fSAndroid Build Coastguard Worker
ToFloat() const354*3ac0a46fSAndroid Build Coastguard Worker float CFGAS_Decimal::ToFloat() const {
355*3ac0a46fSAndroid Build Coastguard Worker return static_cast<float>(ToDouble());
356*3ac0a46fSAndroid Build Coastguard Worker }
357*3ac0a46fSAndroid Build Coastguard Worker
ToDouble() const358*3ac0a46fSAndroid Build Coastguard Worker double CFGAS_Decimal::ToDouble() const {
359*3ac0a46fSAndroid Build Coastguard Worker double pow = (double)(1 << 16) * (1 << 16);
360*3ac0a46fSAndroid Build Coastguard Worker double base = static_cast<double>(m_uHi) * pow * pow +
361*3ac0a46fSAndroid Build Coastguard Worker static_cast<double>(m_uMid) * pow + static_cast<double>(m_uLo);
362*3ac0a46fSAndroid Build Coastguard Worker return (m_bNeg ? -1 : 1) * base * powf(10.0f, -m_uScale);
363*3ac0a46fSAndroid Build Coastguard Worker }
364*3ac0a46fSAndroid Build Coastguard Worker
SetScale(uint8_t newscale)365*3ac0a46fSAndroid Build Coastguard Worker void CFGAS_Decimal::SetScale(uint8_t newscale) {
366*3ac0a46fSAndroid Build Coastguard Worker uint8_t oldscale = m_uScale;
367*3ac0a46fSAndroid Build Coastguard Worker if (oldscale == newscale)
368*3ac0a46fSAndroid Build Coastguard Worker return;
369*3ac0a46fSAndroid Build Coastguard Worker
370*3ac0a46fSAndroid Build Coastguard Worker uint64_t phi = m_uHi;
371*3ac0a46fSAndroid Build Coastguard Worker uint64_t pmid = m_uMid;
372*3ac0a46fSAndroid Build Coastguard Worker uint64_t plo = m_uLo;
373*3ac0a46fSAndroid Build Coastguard Worker if (newscale > oldscale) {
374*3ac0a46fSAndroid Build Coastguard Worker for (uint8_t iter = 0; iter < newscale - oldscale; iter++)
375*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_mul10(phi, pmid, plo);
376*3ac0a46fSAndroid Build Coastguard Worker
377*3ac0a46fSAndroid Build Coastguard Worker m_uHi = static_cast<uint32_t>(phi);
378*3ac0a46fSAndroid Build Coastguard Worker m_uMid = static_cast<uint32_t>(pmid);
379*3ac0a46fSAndroid Build Coastguard Worker m_uLo = static_cast<uint32_t>(plo);
380*3ac0a46fSAndroid Build Coastguard Worker m_bNeg = m_bNeg && IsNotZero();
381*3ac0a46fSAndroid Build Coastguard Worker m_uScale = newscale;
382*3ac0a46fSAndroid Build Coastguard Worker } else {
383*3ac0a46fSAndroid Build Coastguard Worker uint64_t point5_hi = 0;
384*3ac0a46fSAndroid Build Coastguard Worker uint64_t point5_mid = 0;
385*3ac0a46fSAndroid Build Coastguard Worker uint64_t point5_lo = 5;
386*3ac0a46fSAndroid Build Coastguard Worker for (uint8_t iter = 0; iter < oldscale - newscale - 1; iter++)
387*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_mul10(point5_hi, point5_mid, point5_lo);
388*3ac0a46fSAndroid Build Coastguard Worker
389*3ac0a46fSAndroid Build Coastguard Worker phi += point5_hi;
390*3ac0a46fSAndroid Build Coastguard Worker pmid += point5_mid;
391*3ac0a46fSAndroid Build Coastguard Worker plo += point5_lo;
392*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_normalize(phi, pmid, plo);
393*3ac0a46fSAndroid Build Coastguard Worker for (uint8_t iter = 0; iter < oldscale - newscale; iter++)
394*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_div10(phi, pmid, plo);
395*3ac0a46fSAndroid Build Coastguard Worker }
396*3ac0a46fSAndroid Build Coastguard Worker m_uHi = static_cast<uint32_t>(phi);
397*3ac0a46fSAndroid Build Coastguard Worker m_uMid = static_cast<uint32_t>(pmid);
398*3ac0a46fSAndroid Build Coastguard Worker m_uLo = static_cast<uint32_t>(plo);
399*3ac0a46fSAndroid Build Coastguard Worker m_bNeg = m_bNeg && IsNotZero();
400*3ac0a46fSAndroid Build Coastguard Worker m_uScale = newscale;
401*3ac0a46fSAndroid Build Coastguard Worker }
402*3ac0a46fSAndroid Build Coastguard Worker
SetNegate()403*3ac0a46fSAndroid Build Coastguard Worker void CFGAS_Decimal::SetNegate() {
404*3ac0a46fSAndroid Build Coastguard Worker if (IsNotZero())
405*3ac0a46fSAndroid Build Coastguard Worker m_bNeg = !m_bNeg;
406*3ac0a46fSAndroid Build Coastguard Worker }
407*3ac0a46fSAndroid Build Coastguard Worker
operator *(const CFGAS_Decimal & val) const408*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal CFGAS_Decimal::operator*(const CFGAS_Decimal& val) const {
409*3ac0a46fSAndroid Build Coastguard Worker uint64_t a[3] = {m_uLo, m_uMid, m_uHi},
410*3ac0a46fSAndroid Build Coastguard Worker b[3] = {val.m_uLo, val.m_uMid, val.m_uHi};
411*3ac0a46fSAndroid Build Coastguard Worker uint64_t c[6];
412*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_raw_mul(a, 3, b, 3, c, 6);
413*3ac0a46fSAndroid Build Coastguard Worker bool neg = m_bNeg ^ val.m_bNeg;
414*3ac0a46fSAndroid Build Coastguard Worker uint8_t scale = m_uScale + val.m_uScale;
415*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_shrinkintorange(c, 6, 3, scale);
416*3ac0a46fSAndroid Build Coastguard Worker return CFGAS_Decimal(static_cast<uint32_t>(c[0]), static_cast<uint32_t>(c[1]),
417*3ac0a46fSAndroid Build Coastguard Worker static_cast<uint32_t>(c[2]), neg, scale);
418*3ac0a46fSAndroid Build Coastguard Worker }
419*3ac0a46fSAndroid Build Coastguard Worker
operator /(const CFGAS_Decimal & val) const420*3ac0a46fSAndroid Build Coastguard Worker CFGAS_Decimal CFGAS_Decimal::operator/(const CFGAS_Decimal& val) const {
421*3ac0a46fSAndroid Build Coastguard Worker if (!val.IsNotZero())
422*3ac0a46fSAndroid Build Coastguard Worker return CFGAS_Decimal();
423*3ac0a46fSAndroid Build Coastguard Worker
424*3ac0a46fSAndroid Build Coastguard Worker bool neg = m_bNeg ^ val.m_bNeg;
425*3ac0a46fSAndroid Build Coastguard Worker uint64_t a[7] = {m_uLo, m_uMid, m_uHi},
426*3ac0a46fSAndroid Build Coastguard Worker b[3] = {val.m_uLo, val.m_uMid, val.m_uHi}, c[7] = {0};
427*3ac0a46fSAndroid Build Coastguard Worker uint8_t scale = 0;
428*3ac0a46fSAndroid Build Coastguard Worker if (m_uScale < val.m_uScale) {
429*3ac0a46fSAndroid Build Coastguard Worker for (int i = val.m_uScale - m_uScale; i > 0; i--)
430*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_mul10_any(a, 7);
431*3ac0a46fSAndroid Build Coastguard Worker } else {
432*3ac0a46fSAndroid Build Coastguard Worker scale = m_uScale - val.m_uScale;
433*3ac0a46fSAndroid Build Coastguard Worker }
434*3ac0a46fSAndroid Build Coastguard Worker
435*3ac0a46fSAndroid Build Coastguard Worker uint8_t minscale = scale;
436*3ac0a46fSAndroid Build Coastguard Worker if (!IsNotZero())
437*3ac0a46fSAndroid Build Coastguard Worker return CFGAS_Decimal(0, 0, 0, false, minscale);
438*3ac0a46fSAndroid Build Coastguard Worker
439*3ac0a46fSAndroid Build Coastguard Worker while (!a[6]) {
440*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_mul10_any(a, 7);
441*3ac0a46fSAndroid Build Coastguard Worker scale++;
442*3ac0a46fSAndroid Build Coastguard Worker }
443*3ac0a46fSAndroid Build Coastguard Worker
444*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_div10_any(a, 7);
445*3ac0a46fSAndroid Build Coastguard Worker scale--;
446*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_raw_div(a, 6, b, 3, c, 7);
447*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_shrinkintorange(c, 6, 3, scale);
448*3ac0a46fSAndroid Build Coastguard Worker decimal_helper_truncate(c[2], c[1], c[0], scale, minscale);
449*3ac0a46fSAndroid Build Coastguard Worker return CFGAS_Decimal(static_cast<uint32_t>(c[0]), static_cast<uint32_t>(c[1]),
450*3ac0a46fSAndroid Build Coastguard Worker static_cast<uint32_t>(c[2]), neg, scale);
451*3ac0a46fSAndroid Build Coastguard Worker }
452