12
12
You should have received a copy of the GNU General Public License
13
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
18
* @brief SQL standard-compliant decimal number handling
21
* This library implements SQL standard "exact numeric" type
22
* and is not at all generic, but rather intentinally crippled to
23
* follow the standard :)
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
27
17
=======================================================================
18
NOTE: this library implements SQL standard "exact numeric" type
19
and is not at all generic, but rather intentinally crippled to
20
follow the standard :)
21
=======================================================================
28
22
Quoting the standard
29
23
(SQL:2003, Part 2 Foundations, aka ISO/IEC 9075-2:2003)
187
182
@param[in] filler what char to pad with (ZEROFILL et al.)
188
183
@param[out] *str where to store the resulting string
192
187
@retval E_DEC_TRUNCATED
193
188
@retval E_DEC_OVERFLOW
194
189
@retval E_DEC_OOM
197
int class_decimal2string(const type::Decimal *d,
198
uint32_t fixed_dec, String *str)
192
int my_decimal2string(uint32_t mask, const my_decimal *d,
193
uint32_t fixed_prec, uint32_t fixed_dec,
194
char filler, String *str)
200
uint32_t mask= E_DEC_FATAL_ERROR;
203
197
Calculate the size of the string: For DECIMAL(a,b), fixed_prec==a
204
198
holds true iff the type is also ZEROFILL, which in turn implies
207
201
one if the user only wanted decimal places, but we force a leading
208
202
zero on them. Because the type is implicitly UNSIGNED, we do not
209
203
need to reserve a character for the sign. For all other cases,
210
fixed_prec will be 0, and class_decimal_string_length() will be called
204
fixed_prec will be 0, and my_decimal_string_length() will be called
211
205
instead to calculate the required size of the buffer.
214
? (uint32_t)(((0 == fixed_dec) ? 1 : 0) + 1)
215
: (uint32_t)d->string_length());
207
int length= (fixed_prec
208
? (fixed_prec + ((fixed_prec == fixed_dec) ? 1 : 0) + 1)
209
: my_decimal_string_length(d));
217
211
if (str->alloc(length))
218
212
return check_result(mask, E_DEC_OOM);
220
213
result= decimal2string((decimal_t*) d, (char*) str->ptr(),
221
&length, (int)0, fixed_dec,
214
&length, (int)fixed_prec, fixed_dec,
223
216
str->length(length);
224
217
return check_result(mask, result);
229
@brief Convert from decimal to binary representation
231
@param[in] mask error processing mask
232
@param[in] d number for conversion
233
@param[out] bin pointer to buffer where to write result
234
@param[in] prec overall number of decimal digits
235
@param[in] scale number of decimal digits after decimal point
222
Convert from decimal to binary representation
226
mask error processing mask
227
d number for conversion
228
bin pointer to buffer where to write result
229
prec overall number of decimal digits
230
scale number of decimal digits after decimal point
238
233
Before conversion we round number if it need but produce truncation
239
234
error in this case
243
@retval E_DEC_TRUNCATED
244
@retval E_DEC_OVERFLOW
249
int Decimal::val_binary(uint32_t mask, unsigned char *bin, int prec, int scale) const
242
int my_decimal2binary(uint32_t mask, const my_decimal *d, unsigned char *bin, int prec,
251
245
int err1= E_DEC_OK, err2;
252
type::Decimal rounded;
253
class_decimal2decimal(this, &rounded);
247
my_decimal2decimal(d, &rounded);
254
248
rounded.frac= decimal_actual_fraction(&rounded);
255
249
if (scale < rounded.frac)
264
258
return check_result(mask, err2);
271
@brief Convert string for decimal when string can be in some multibyte charset
273
@param mask error processing mask
274
@param from string to process
275
@param length length of given string
276
@param charset charset of given string
280
@retval E_DEC_TRUNCATED
281
@retval E_DEC_OVERFLOW
282
@retval E_DEC_BAD_NUM
263
Convert string for decimal when string can be in some multibyte charset
267
mask error processing mask
268
from string to process
269
length length of given string
270
charset charset of given string
271
decimal_value buffer for result storing
286
int type::Decimal::store(uint32_t mask, const char *from, uint32_t length, const CHARSET_INFO * charset)
281
int str2my_decimal(uint32_t mask, const char *from, uint32_t length,
282
const CHARSET_INFO * charset, my_decimal *decimal_value)
288
284
char *end, *from_end;
291
287
String tmp(buff, sizeof(buff), &my_charset_bin);
292
288
if (charset->mbminlen > 1)
290
uint32_t dummy_errors;
295
291
tmp.copy(from, length, charset, &my_charset_utf8_general_ci, &dummy_errors);
297
293
length= tmp.length();
298
294
charset= &my_charset_bin;
300
296
from_end= end= (char*) from+length;
301
err= string2decimal((char *)from, (decimal_t*) this, &end);
297
err= string2decimal((char *)from, (decimal_t*) decimal_value, &end);
302
298
if (end != from_end && !err)
304
300
/* Give warning if there is something other than end space */
314
check_result_and_overflow(mask, err);
310
check_result_and_overflow(mask, err, decimal_value);
318
void type::Decimal::convert(double &result) const
320
decimal2double(static_cast<const decimal_t*>(this), &result);
323
type::Decimal *date2_class_decimal(type::Time *ltime, type::Decimal *dec)
315
my_decimal *date2my_decimal(DRIZZLE_TIME *ltime, my_decimal *dec)
326
318
date = (ltime->year*100L + ltime->month)*100L + ltime->day;
327
if (ltime->time_type > type::DRIZZLE_TIMESTAMP_DATE)
319
if (ltime->time_type > DRIZZLE_TIMESTAMP_DATE)
328
320
date= ((date*100L + ltime->hour)*100L+ ltime->minute)*100L + ltime->second;
330
if (int2_class_decimal(E_DEC_FATAL_ERROR, date, false, dec))
321
if (int2my_decimal(E_DEC_FATAL_ERROR, date, false, dec))
333
323
if (ltime->second_part)
335
325
dec->buf[(dec->intg-1) / 9 + 1]= ltime->second_part * 1000;
343
void class_decimal_trim(uint32_t *precision, uint32_t *scale)
332
void my_decimal_trim(uint32_t *precision, uint32_t *scale)
345
334
if (!(*precision) && !(*scale))
373
362
#define DIG_MASK 100000000
374
363
#define DIG_BASE 1000000000
375
364
#define DIG_MAX (DIG_BASE-1)
378
inline static T round_up(const T &x)
380
return (x+DIG_PER_DEC1-1)/DIG_PER_DEC1;
365
#define ROUND_UP(X) (((X)+DIG_PER_DEC1-1)/DIG_PER_DEC1)
383
366
static const dec1 powers10[DIG_PER_DEC1+1]={
384
367
1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000};
385
368
static const int dig2bytes[DIG_PER_DEC1+1]={0, 1, 1, 2, 2, 3, 3, 4, 4, 4};
388
371
999900000, 999990000, 999999000,
389
372
999999900, 999999990 };
392
375
#define sanity(d) assert((d)->len > 0)
394
377
#define sanity(d) assert((d)->len >0 && ((d)->buf[0] | \
395
378
(d)->buf[(d)->len-1] | 1))
398
inline static void fix_intg_frac_error(const int len, int &intg1, int &frac1, int &error)
400
if (unlikely(intg1+frac1 > len))
402
if (unlikely(intg1 > len))
406
error=E_DEC_OVERFLOW;
411
error=E_DEC_TRUNCATED;
418
/* assume carry <= 1 */
419
inline static void add(dec1 &to, const dec1 &from1, const dec1& from2, dec1 &carry)
421
dec1 a=from1+from2+carry;
423
if ((carry= (a >= DIG_BASE))) /* no division here! */
428
inline static void add2(dec1 &to, const dec1 &from1, const dec1 &from2, dec1 &carry)
430
dec2 a=dec2(from1)+from2+carry;
431
if ((carry= (a >= DIG_BASE)))
433
if (unlikely(a >= DIG_BASE))
442
inline static void sub(dec1 &to, const dec1 &from1, const dec1 &from2, dec1 &carry)
444
dec1 a=from1-from2-carry;
445
if ((carry= (a < 0)))
451
inline static void sub2(dec1 &to, const dec1 &from1, const dec1 &from2, dec1 &carry)
453
dec1 a=from1-from2-carry;
454
if ((carry= (a < 0)))
381
#define FIX_INTG_FRAC_ERROR(len, intg1, frac1, error) \
384
if (unlikely(intg1+frac1 > (len))) \
386
if (unlikely(intg1 > (len))) \
390
error=E_DEC_OVERFLOW; \
395
error=E_DEC_TRUNCATED; \
402
#define ADD(to, from1, from2, carry) /* assume carry <= 1 */ \
405
dec1 a=(from1)+(from2)+(carry); \
406
assert((carry) <= 1); \
407
if (((carry)= a >= DIG_BASE)) /* no division here! */ \
412
#define ADD2(to, from1, from2, carry) \
415
dec2 a=((dec2)(from1))+(from2)+(carry); \
416
if (((carry)= a >= DIG_BASE)) \
418
if (unlikely(a >= DIG_BASE)) \
426
#define SUB(to, from1, from2, carry) /* to=from1-from2 */ \
429
dec1 a=(from1)-(from2)-(carry); \
430
if (((carry)= a < 0)) \
435
#define SUB2(to, from1, from2, carry) /* to=from1-from2 */ \
438
dec1 a=(from1)-(from2)-(carry); \
439
if (((carry)= a < 0)) \
441
if (unlikely(a < 0)) \
465
@brief Get maximum value for given precision and scale
467
@param precision/scale see decimal_bin_size() below
468
@param to decimal where where the result will be stored
450
Swap the contents of two variables.
452
#define swap_variables(TYPE, a, b) \
463
Get maximum value for given precision and scale
467
precision/scale - see decimal_bin_size() below
468
to - decimal where where the result will be stored
469
469
to->buf and to->len must be set.
523
@brief Count actual length of fraction part (without ending zeroes)
523
Count actual length of fraction part (without ending zeroes)
525
@param from number for processing
526
decimal_actual_fraction()
527
from number for processing
528
530
int decimal_actual_fraction(decimal_t *from)
530
532
int frac= from->frac, i;
531
dec1 *buf0= from->buf + round_up(from->intg) + round_up(frac) - 1;
533
dec1 *buf0= from->buf + ROUND_UP(from->intg) + ROUND_UP(frac) - 1;
552
@brief Convert decimal to its printable string representation
554
Convert decimal to its printable string representation
554
@param from value to convert
555
@param to points to buffer where string representation
557
@param to_len in: size of to buffer
558
out: length of the actually written string
559
@param fixed_precision 0 if representation can be variable length and
558
from - value to convert
559
to - points to buffer where string representation
561
*to_len - in: size of to buffer
562
out: length of the actually written string
563
fixed_precision - 0 if representation can be variable length and
560
564
fixed_decimals will not be checked in this case.
561
565
Put number as with fixed point position with this
562
566
number of digits (sign counted and decimal point is
564
@param fixed_decimals number digits after point.
565
@param filler character to fill gaps in case of fixed_precision > 0
568
fixed_decimals - number digits after point.
569
filler - character to fill gaps in case of fixed_precision > 0
569
@retval E_DEC_TRUNCATED
570
@retval E_DEC_OVERFLOW
572
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW
572
575
int decimal2string(const decimal_t *from, char *to, int *to_len,
573
576
int fixed_precision, int fixed_decimals,
679
@brief Return bounds of decimal digits in the number
682
Return bounds of decimal digits in the number
681
@param from decimal number for processing
682
@param start_result index (from 0 ) of first decimal digits will
683
be written by this address
684
@param end_result index of position just after last decimal digit
686
from - decimal number for processing
687
start_result - index (from 0 ) of first decimal digits will
688
be written by this address
689
end_result - index of position just after last decimal digit
685
690
be written by this address
687
693
static void digits_bounds(decimal_t *from, int *start_result, int *end_result)
689
695
int start, stop, i;
690
696
dec1 *buf_beg= from->buf;
691
dec1 *end= from->buf + round_up(from->intg) + round_up(from->frac);
697
dec1 *end= from->buf + ROUND_UP(from->intg) + ROUND_UP(from->frac);
692
698
dec1 *buf_end= end - 1;
694
700
/* find non-zero digit from number begining */
741
@param Left shift for alignment of data in buffer
743
@param dec pointer to decimal number which have to be shifted
744
@param shift number of decimal digits on which it should be shifted
745
@param beg beginning of decimal digits (see digits_bounds())
746
@param end end of decimal digits (see digits_bounds())
749
Result fitting in the buffer should be garanted.
750
'shift' have to be from 1 to DIG_PER_DEC1-1 (inclusive)
752
@todo Above note is unclear - is 'garanted' a typo for 'guaranteed'
747
Left shift for alignment of data in buffer
751
dec pointer to decimal number which have to be shifted
752
shift number of decimal digits on which it should be shifted
753
beg/end bounds of decimal digits (see digits_bounds())
756
Result fitting in the buffer should be garanted.
757
'shift' have to be from 1 to DIG_PER_DEC1-1 (inclusive)
755
760
static void do_mini_left_shift(decimal_t *dec, int shift, int beg, int last)
757
dec1 *from= dec->buf + round_up(beg + 1) - 1;
758
dec1 *end= dec->buf + round_up(last) - 1;
762
dec1 *from= dec->buf + ROUND_UP(beg + 1) - 1;
763
dec1 *end= dec->buf + ROUND_UP(last) - 1;
759
764
int c_shift= DIG_PER_DEC1 - shift;
760
765
assert(from >= dec->buf);
761
766
assert(end < dec->buf + dec->len);
772
@brief Right shift for alignment of data in buffer
774
@param dec pointer to decimal number which have to be shifted
775
@param shift number of decimal digits on which it should be shifted
776
@param beg beginning of decimal digits (see digits_bounds())
777
@param end end of decimal digits (see digits_bounds())
777
Right shift for alignment of data in buffer
781
dec pointer to decimal number which have to be shifted
782
shift number of decimal digits on which it should be shifted
783
beg/end bounds of decimal digits (see digits_bounds())
780
786
Result fitting in the buffer should be garanted.
781
787
'shift' have to be from 1 to DIG_PER_DEC1-1 (inclusive)
783
790
static void do_mini_right_shift(decimal_t *dec, int shift, int beg, int last)
785
dec1 *from= dec->buf + round_up(last) - 1;
786
dec1 *end= dec->buf + round_up(beg + 1) - 1;
792
dec1 *from= dec->buf + ROUND_UP(last) - 1;
793
dec1 *end= dec->buf + ROUND_UP(beg + 1) - 1;
787
794
int c_shift= DIG_PER_DEC1 - shift;
788
795
assert(from < dec->buf + dec->len);
789
796
assert(end >= dec->buf);
800
@brief Shift of decimal digits in given number (with rounding if it need)
807
Shift of decimal digits in given number (with rounding if it need)
802
@param dec number to be shifted
803
@param shift number of decimal positions
811
dec number to be shifted
812
shift number of decimal positions
804
813
shift > 0 means shift to left shift
805
814
shift < 0 meand right shift
808
816
In fact it is multipling on 10^shift.
819
E_DEC_OVERFLOW operation lead to overflow, number is untoched
820
E_DEC_TRUNCATED number was rounded to fit into buffer
812
@retval E_DEC_OVERFLOW operation lead to overflow, number is untoched
813
@retval E_DEC_TRUNCATED number was rounded to fit into buffer
815
823
static int decimal_shift(decimal_t *dec, int shift)
817
825
/* index of first non zero digit (all indexes from 0) */
956
963
d_shift= (1 - new_front) / DIG_PER_DEC1;
957
to= dec->buf + round_up(end) - 1 + d_shift;
958
barier= dec->buf + round_up(beg + 1) - 1 + d_shift;
964
to= dec->buf + ROUND_UP(end) - 1 + d_shift;
965
barier= dec->buf + ROUND_UP(beg + 1) - 1 + d_shift;
959
966
assert(to < dec->buf + dec->len);
960
967
assert(barier - d_shift >= dec->buf);
961
968
for(; to >= barier; to--)
1004
@brief Convert string to decimal
1011
Convert string to decimal
1006
@param from value to convert. Doesn't have to be \0 terminated!
1007
@param to decimal where where the result will be stored
1015
from - value to convert. Doesn't have to be \0 terminated!
1016
to - decimal where where the result will be stored
1008
1017
to->buf and to->len must be set.
1009
@param end Pointer to pointer to end of string. Will on return be
1018
end - Pointer to pointer to end of string. Will on return be
1010
1019
set to the char after the last used character
1011
@param fixed use to->intg, to->frac as limits for input number
1020
fixed - use to->intg, to->frac as limits for input number
1014
1023
to->intg and to->frac can be modified even when fixed=1
1015
1024
(but only decreased, in this case)
1018
1027
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW/E_DEC_BAD_NUM/E_DEC_OOM
1019
1028
In case of E_DEC_FATAL_ERROR *to is set to decimal zero
1020
1029
(to make error handling easier)
1023
1033
internal_str2dec(char *from, decimal_t *to, char **end, bool fixed)
1322
1336
return E_DEC_OK;
1327
Convert decimal to its binary fixed-length representation (suitable for
1328
comparing with memcmp)
1340
Convert decimal to its binary fixed-length representation
1341
two representations of the same length can be compared with memcmp
1342
with the correct -1/0/+1 result
1346
from - value to convert
1347
to - points to buffer where string representation should be stored
1348
precision/scale - see decimal_bin_size() below
1351
the buffer is assumed to be of the size decimal_bin_size(precision, scale)
1354
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW
1330
1357
for storage decimal numbers are converted to the "binary" format.
1332
1359
This format has the following properties:
1387
1414
And for -1234567890.1234 it would be
1389
1416
7E F2 04 37 2D FB 2D
1392
@param from value to convert
1393
@param to points to buffer where string representation should be stored
1394
@param precision see decimal_bin_size() below
1395
@param frac see decimal_bin_size() below
1398
The buffer is assumed to be of the size decimal_bin_size(precision, scale)
1401
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW
1404
1418
int decimal2bin(const decimal_t *from, unsigned char *to, int precision, int frac)
1523
@brief Restores decimal from its binary fixed-length representation
1525
@param from value to convert
1527
@param precision see decimal_bin_size() below
1528
@param scale see decimal_bin_size() below
1537
Restores decimal from its binary fixed-length representation
1541
from - value to convert
1543
precision/scale - see decimal_bin_size() below
1531
1546
see decimal2bin()
1532
1547
the buffer is assumed to be of the size decimal_bin_size(precision, scale)
1535
1550
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW
1537
1553
int bin2decimal(const unsigned char *from, decimal_t *to, int precision, int scale)
1539
1555
int error=E_DEC_OK, intg=precision-scale,
1652
1670
frac0*sizeof(dec1)+dig2bytes[frac0x];
1656
@brief Rounds the decimal to "scale" digits
1658
@param from - decimal to round,
1659
@param to - result buffer. from==to is allowed
1660
@param scale - to what position to round. can be negative!
1661
@param mode - round to nearest even or truncate
1674
Rounds the decimal to "scale" digits
1678
from - decimal to round,
1679
to - result buffer. from==to is allowed
1680
scale - to what position to round. can be negative!
1681
mode - round to nearest even or truncate
1664
1684
scale can be negative !
1665
1685
one TRUNCATED error (line XXX below) isn't treated very logical :(
1668
1688
E_DEC_OK/E_DEC_TRUNCATED
1671
1692
decimal_round(const decimal_t *from, decimal_t *to, int scale,
1672
1693
decimal_round_mode mode)
1674
int frac0=scale>0 ? round_up(scale) : scale/DIG_PER_DEC1,
1675
frac1=round_up(from->frac), round_digit= 0,
1676
intg0=round_up(from->intg), error=E_DEC_OK, len=to->len,
1677
intg1=round_up(from->intg +
1695
int frac0=scale>0 ? ROUND_UP(scale) : scale/DIG_PER_DEC1,
1696
frac1=ROUND_UP(from->frac), round_digit= 0,
1697
intg0=ROUND_UP(from->intg), error=E_DEC_OK, len=to->len,
1698
intg1=ROUND_UP(from->intg +
1678
1699
(((intg0 + frac0)>0) && (from->buf[0] == DIG_MAX)));
1679
1700
dec1 *buf0=from->buf, *buf1=to->buf, x, y, carry=0;
1863
1884
static int do_add(const decimal_t *from1, const decimal_t *from2, decimal_t *to)
1865
int intg1=round_up(from1->intg), intg2=round_up(from2->intg),
1866
frac1=round_up(from1->frac), frac2=round_up(from2->frac),
1886
int intg1=ROUND_UP(from1->intg), intg2=ROUND_UP(from2->intg),
1887
frac1=ROUND_UP(from1->frac), frac2=ROUND_UP(from2->frac),
1867
1888
frac0=max(frac1, frac2), intg0=max(intg1, intg2), error;
1868
1889
dec1 *buf1, *buf2, *buf0, *stop, *stop2, x, carry;
1879
1900
to->buf[0]=0; /* safety */
1882
fix_intg_frac_error(to->len, intg0, frac0, error);
1903
FIX_INTG_FRAC_ERROR(to->len, intg0, frac0, error);
1883
1904
if (unlikely(error == E_DEC_OVERFLOW))
1885
1906
max_decimal(to->len * DIG_PER_DEC1, 0, to);
1944
1965
if to==0, return -1/0/+1 - the result of the comparison */
1945
1966
static int do_sub(const decimal_t *from1, const decimal_t *from2, decimal_t *to)
1947
int intg1=round_up(from1->intg), intg2=round_up(from2->intg),
1948
frac1=round_up(from1->frac), frac2=round_up(from2->frac);
1968
int intg1=ROUND_UP(from1->intg), intg2=ROUND_UP(from2->intg),
1969
frac1=ROUND_UP(from1->frac), frac2=ROUND_UP(from2->frac);
1949
1970
int frac0=max(frac1, frac2), error;
1950
1971
dec1 *buf1, *buf2, *buf0, *stop1, *stop2, *start1, *start2, carry=0;
2013
2032
/* ensure that always from1 > from2 (and intg1 >= intg2) */
2017
swap(start1, start2);
2035
swap_variables(const decimal_t *,from1, from2);
2036
swap_variables(dec1 *,start1, start2);
2037
swap_variables(int,intg1,intg2);
2038
swap_variables(int,frac1,frac2);
2020
2039
to->sign= 1 - to->sign;
2023
fix_intg_frac_error(to->len, intg1, frac0, error);
2042
FIX_INTG_FRAC_ERROR(to->len, intg1, frac0, error);
2024
2043
buf0=to->buf+intg1+frac0;
2026
2045
to->frac=max(from1->frac, from2->frac);
2055
2074
while (buf2 > stop2)
2057
sub(*--buf0, 0, *--buf2, carry);
2076
SUB(*--buf0, 0, *--buf2, carry);
2061
2080
/* part 2 - cmin(frac) ... intg2 */
2062
2081
while (buf2 > start2)
2064
sub(*--buf0, *--buf1, *--buf2, carry);
2083
SUB(*--buf0, *--buf1, *--buf2, carry);
2067
2086
/* part 3 - intg2 ... intg1 */
2068
2087
while (carry && buf1 > start1)
2070
sub(*--buf0, *--buf1, 0, carry);
2089
SUB(*--buf0, *--buf1, 0, carry);
2073
2092
while (buf1 > start1)
2108
2127
return from1->sign > from2->sign ? -1 : 1;
2111
int decimal_t::isZero() const
2130
int decimal_is_zero(const decimal_t *from)
2114
*end= buf1 +round_up(intg) +round_up(frac);
2132
dec1 *buf1=from->buf,
2133
*end=buf1+ROUND_UP(from->intg)+ROUND_UP(from->frac);
2116
2134
while (buf1 < end)
2128
@brief multiply two decimals
2130
@param[in] from1 First factor
2131
@param[in] from2 Second factor
2132
@param[out] to product
2141
multiply two decimals
2145
from1, from2 - factors
2135
2149
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW;
2138
2152
in this implementation, with sizeof(dec1)=4 we have DIG_PER_DEC1=9,
2139
2153
and 63-digit number will take only 7 dec1 words (basically a 7-digit
2140
2154
"base 999999999" number). Thus there's no need in fast multiplication
2147
2161
int decimal_mul(const decimal_t *from1, const decimal_t *from2, decimal_t *to)
2149
int intg1=round_up(from1->intg), intg2=round_up(from2->intg),
2150
frac1=round_up(from1->frac), frac2=round_up(from2->frac),
2151
intg0=round_up(from1->intg+from2->intg),
2163
int intg1=ROUND_UP(from1->intg), intg2=ROUND_UP(from2->intg),
2164
frac1=ROUND_UP(from1->frac), frac2=ROUND_UP(from2->frac),
2165
intg0=ROUND_UP(from1->intg+from2->intg),
2152
2166
frac0=frac1+frac2, error, i, j, d_to_move;
2153
2167
dec1 *buf1=from1->buf+intg1, *buf2=from2->buf+intg2, *buf0,
2154
2168
*start2, *stop2, *stop1, *start0, carry;
2198
2212
dec2 p= ((dec2)*buf1) * ((dec2)*buf2);
2199
2213
hi=(dec1)(p/DIG_BASE);
2200
2214
lo=(dec1)(p-((dec2)hi)*DIG_BASE);
2201
add2(*buf0, *buf0, lo, carry);
2215
ADD2(*buf0, *buf0, lo, carry);
2206
2220
if (buf0 < to->buf)
2207
2221
return E_DEC_OVERFLOW;
2208
add2(*buf0, *buf0, 0, carry);
2222
ADD2(*buf0, *buf0, 0, carry);
2210
2224
for (buf0--; carry; buf0--)
2212
2226
if (buf0 < to->buf)
2213
2227
return E_DEC_OVERFLOW;
2214
add(*buf0, *buf0, 0, carry);
2228
ADD(*buf0, *buf0, 0, carry);
2254
2268
naive division algorithm (Knuth's Algorithm D in 4.3.1) -
2255
2269
it's ok for short numbers
2256
2270
also we're using alloca() to allocate a temporary buffer
2259
If this library is to be used with huge numbers of thousands of
2272
XXX if this library is to be used with huge numbers of thousands of
2260
2273
digits, fast division must be implemented and alloca should be
2261
2274
changed to malloc (or at least fallback to malloc if alloca() fails)
2262
2275
but then, decimal_mul() should be rewritten too :(
2264
2277
static int do_div_mod(const decimal_t *from1, const decimal_t *from2,
2265
2278
decimal_t *to, decimal_t *mod, int scale_incr)
2267
int frac1=round_up(from1->frac)*DIG_PER_DEC1, prec1=from1->intg+frac1,
2268
frac2=round_up(from2->frac)*DIG_PER_DEC1, prec2=from2->intg+frac2,
2280
int frac1=ROUND_UP(from1->frac)*DIG_PER_DEC1, prec1=from1->intg+frac1,
2281
frac2=ROUND_UP(from2->frac)*DIG_PER_DEC1, prec2=from2->intg+frac2,
2269
2282
error= 0, i, intg0, frac0, len1, len2, dintg, div_mod=(!mod);
2270
2283
dec1 *buf0, *buf1=from1->buf, *buf2=from2->buf, *tmp1,
2271
2284
*start2, *stop2, *stop1, *stop0, norm2, carry, *start1, dcarry;
2334
2347
N2 is in the buf2, has prec2 digits. Scales are frac1 and
2335
2348
frac2 accordingly.
2336
2349
Thus, the result will have
2337
frac = round_up(frac1+frac2+scale_incr)
2350
frac = ROUND_UP(frac1+frac2+scale_incr)
2339
2352
intg = (prec1-frac1) - (prec2-frac2) + 1
2340
2353
prec = intg+frac
2342
frac0=round_up(frac1+frac2+scale_incr);
2343
fix_intg_frac_error(to->len, intg0, frac0, error);
2355
frac0=ROUND_UP(frac1+frac2+scale_incr);
2356
FIX_INTG_FRAC_ERROR(to->len, intg0, frac0, error);
2344
2357
to->sign=from1->sign != from2->sign;
2345
2358
to->intg=intg0*DIG_PER_DEC1;
2346
2359
to->frac=frac0*DIG_PER_DEC1;
2507
@brief division of two decimals
2509
@param[in] from1 dividend
2510
@param[in] from2 divisor
2511
@param[out] to quotient
2520
division of two decimals
2514
2529
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW/E_DEC_DIV_ZERO;
2517
2532
see do_div_mod()
2520
2536
decimal_div(const decimal_t *from1, const decimal_t *from2, decimal_t *to, int scale_incr)
2522
2538
return do_div_mod(from1, from2, to, 0, scale_incr);
2528
the modulus R in R = M mod N
2534
R = M - k*N, where k is integer
2536
thus, there's no requirement for M or N to be integers
2539
@param from1 dividend
2540
@param from2 divisor
2544
2551
E_DEC_OK/E_DEC_TRUNCATED/E_DEC_OVERFLOW/E_DEC_DIV_ZERO;
2547
2554
see do_div_mod()
2557
the modulus R in R = M mod N
2563
R = M - k*N, where k is integer
2565
thus, there's no requirement for M or N to be integers
2550
2568
int decimal_mod(const decimal_t *from1, const decimal_t *from2, decimal_t *to)
2552
2570
return do_div_mod(from1, from2, 0, to, 0);
2555
std::ostream& operator<<(std::ostream& output, const type::Decimal &dec)
2557
drizzled::String str;
2559
class_decimal2string(&dec, 0, &str);
2561
output << "type::Decimal:(";
2562
output << str.c_ptr();
2565
return output; // for multiple << operators.
2568
2573
} /* namespace drizzled */
2579
2584
printf("/* intg=%d, frac=%d, sign=%d, buf[]={", d->intg, d->frac, d->sign);
2580
for (i=0; i < round_up(d->frac)+round_up(d->intg)-1; i++)
2585
for (i=0; i < ROUND_UP(d->frac)+ROUND_UP(d->intg)-1; i++)
2581
2586
printf("%09d, ", d->buf[i]);
2582
2587
printf("%09d} */ ", d->buf[i]);