Upgrade to prereleases of Math::BigInt 1.70 and
[p5sagit/p5-mst-13.2.git] / lib / Math / BigInt / t / bigintc.t
CommitLineData
0716bf9b 1#!/usr/bin/perl -w
2
3use strict;
4use Test;
5
6BEGIN
7 {
8 $| = 1;
027dc388 9 chdir 't' if -d 't';
56d9de68 10 unshift @INC, '../lib'; # for running manually
9b924220 11 plan tests => 308;
61f5c3f5 12 }
13
14use Math::BigInt::Calc;
15
091c87b1 16my ($BASE_LEN, $AND_BITS, $XOR_BITS, $OR_BITS, $BASE_LEN_SMALL, $MAX_VAL) =
17 Math::BigInt::Calc->_base_len();
18
19print "# BASE_LEN = $BASE_LEN\n";
20print "# MAX_VAL = $MAX_VAL\n";
21print "# AND_BITS = $AND_BITS\n";
22print "# XOR_BITS = $XOR_BITS\n";
23print "# IOR_BITS = $OR_BITS\n";
24
56d9de68 25# testing of Math::BigInt::Calc
0716bf9b 26
56d9de68 27my $C = 'Math::BigInt::Calc'; # pass classname to sub's
574bacfe 28
0716bf9b 29# _new and _str
9b924220 30my $x = $C->_new("123"); my $y = $C->_new("321");
31ok (ref($x),'ARRAY'); ok ($C->_str($x),123); ok ($C->_str($y),321);
0716bf9b 32
61f5c3f5 33###############################################################################
0716bf9b 34# _add, _sub, _mul, _div
9b924220 35ok ($C->_str($C->_add($x,$y)),444);
36ok ($C->_str($C->_sub($x,$y)),123);
37ok ($C->_str($C->_mul($x,$y)),39483);
38ok ($C->_str($C->_div($x,$y)),123);
0716bf9b 39
61f5c3f5 40###############################################################################
41# check that mul/div doesn't change $y
42# and returns the same reference, not something new
9b924220 43ok ($C->_str($C->_mul($x,$y)),39483);
44ok ($C->_str($x),39483); ok ($C->_str($y),321);
61f5c3f5 45
9b924220 46ok ($C->_str($C->_div($x,$y)),123);
47ok ($C->_str($x),123); ok ($C->_str($y),321);
61f5c3f5 48
9b924220 49$x = $C->_new("39483");
61f5c3f5 50my ($x1,$r1) = $C->_div($x,$y);
51ok ("$x1","$x");
52$C->_inc($x1);
53ok ("$x1","$x");
9b924220 54ok ($C->_str($r1),'0');
61f5c3f5 55
9b924220 56$x = $C->_new("39483"); # reset
61f5c3f5 57
58###############################################################################
9b924220 59my $z = $C->_new("2");
60ok ($C->_str($C->_add($x,$z)),39485);
bd05a461 61my ($re,$rr) = $C->_div($x,$y);
0716bf9b 62
9b924220 63ok ($C->_str($re),123); ok ($C->_str($rr),2);
0716bf9b 64
65# is_zero, _is_one, _one, _zero
56d9de68 66ok ($C->_is_zero($x)||0,0);
67ok ($C->_is_one($x)||0,0);
0716bf9b 68
9b924220 69ok ($C->_str($C->_zero()),"0");
70ok ($C->_str($C->_one()),"1");
56d9de68 71
9b924220 72# _two() and _ten()
73ok ($C->_str($C->_two()),"2");
74ok ($C->_str($C->_ten()),"10");
75ok ($C->_is_ten($C->_two()),0);
76ok ($C->_is_two($C->_two()),1);
77ok ($C->_is_ten($C->_ten()),1);
78ok ($C->_is_two($C->_ten()),0);
56d9de68 79
80ok ($C->_is_one($C->_one()),1);
9b924220 81ok ($C->_is_one($C->_two()),0);
82ok ($C->_is_one($C->_ten()),0);
56d9de68 83
84ok ($C->_is_one($C->_zero()) || 0,0);
85
86ok ($C->_is_zero($C->_zero()),1);
87
88ok ($C->_is_zero($C->_one()) || 0,0);
574bacfe 89
90# is_odd, is_even
56d9de68 91ok ($C->_is_odd($C->_one()),1); ok ($C->_is_odd($C->_zero())||0,0);
92ok ($C->_is_even($C->_one()) || 0,0); ok ($C->_is_even($C->_zero()),1);
93
94# _len
9b924220 95$x = $C->_new("1"); ok ($C->_len($x),1);
96$x = $C->_new("12"); ok ($C->_len($x),2);
97$x = $C->_new("123"); ok ($C->_len($x),3);
98$x = $C->_new("1234"); ok ($C->_len($x),4);
99$x = $C->_new("12345"); ok ($C->_len($x),5);
100$x = $C->_new("123456"); ok ($C->_len($x),6);
101$x = $C->_new("1234567"); ok ($C->_len($x),7);
102$x = $C->_new("12345678"); ok ($C->_len($x),8);
103$x = $C->_new("123456789"); ok ($C->_len($x),9);
104
105$x = $C->_new("8"); ok ($C->_len($x),1);
106$x = $C->_new("21"); ok ($C->_len($x),2);
107$x = $C->_new("321"); ok ($C->_len($x),3);
108$x = $C->_new("4321"); ok ($C->_len($x),4);
109$x = $C->_new("54321"); ok ($C->_len($x),5);
110$x = $C->_new("654321"); ok ($C->_len($x),6);
111$x = $C->_new("7654321"); ok ($C->_len($x),7);
112$x = $C->_new("87654321"); ok ($C->_len($x),8);
113$x = $C->_new("987654321"); ok ($C->_len($x),9);
56d9de68 114
115for (my $i = 1; $i < 9; $i++)
116 {
117 my $a = "$i" . '0' x ($i-1);
9b924220 118 $x = $C->_new($a);
56d9de68 119 print "# Tried len '$a'\n" unless ok ($C->_len($x),$i);
120 }
0716bf9b 121
574bacfe 122# _digit
9b924220 123$x = $C->_new("123456789");
bd05a461 124ok ($C->_digit($x,0),9);
125ok ($C->_digit($x,1),8);
126ok ($C->_digit($x,2),7);
127ok ($C->_digit($x,-1),1);
128ok ($C->_digit($x,-2),2);
129ok ($C->_digit($x,-3),3);
574bacfe 130
131# _copy
56d9de68 132foreach (qw/ 1 12 123 1234 12345 123456 1234567 12345678 123456789/)
133 {
9b924220 134 $x = $C->_new("$_");
135 ok ($C->_str($C->_copy($x)),"$_");
136 ok ($C->_str($x),"$_"); # did _copy destroy original x?
56d9de68 137 }
574bacfe 138
139# _zeros
9b924220 140$x = $C->_new("1256000000"); ok ($C->_zeros($x),6);
141$x = $C->_new("152"); ok ($C->_zeros($x),0);
142$x = $C->_new("123000"); ok ($C->_zeros($x),3);
143$x = $C->_new("0"); ok ($C->_zeros($x),0);
574bacfe 144
145# _lsft, _rsft
9b924220 146$x = $C->_new("10"); $y = $C->_new("3");
147ok ($C->_str($C->_lsft($x,$y,10)),10000);
148$x = $C->_new("20"); $y = $C->_new("3");
149ok ($C->_str($C->_lsft($x,$y,10)),20000);
61f5c3f5 150
9b924220 151$x = $C->_new("128"); $y = $C->_new("4");
152ok ($C->_str($C->_lsft($x,$y,2)), 128 << 4);
61f5c3f5 153
9b924220 154$x = $C->_new("1000"); $y = $C->_new("3");
155ok ($C->_str($C->_rsft($x,$y,10)),1);
156$x = $C->_new("20000"); $y = $C->_new("3");
157ok ($C->_str($C->_rsft($x,$y,10)),20);
158$x = $C->_new("256"); $y = $C->_new("4");
159ok ($C->_str($C->_rsft($x,$y,2)),256 >> 4);
0716bf9b 160
9b924220 161$x = $C->_new("6411906467305339182857313397200584952398");
162$y = $C->_new("45");
163ok ($C->_str($C->_rsft($x,$y,10)),0);
56d9de68 164
574bacfe 165# _acmp
9b924220 166$x = $C->_new("123456789");
167$y = $C->_new("987654321");
bd05a461 168ok ($C->_acmp($x,$y),-1);
169ok ($C->_acmp($y,$x),1);
170ok ($C->_acmp($x,$x),0);
171ok ($C->_acmp($y,$y),0);
9b924220 172$x = $C->_new("12");
173$y = $C->_new("12");
091c87b1 174ok ($C->_acmp($x,$y),0);
9b924220 175$x = $C->_new("21");
091c87b1 176ok ($C->_acmp($x,$y),1);
177ok ($C->_acmp($y,$x),-1);
9b924220 178$x = $C->_new("123456789");
179$y = $C->_new("1987654321");
091c87b1 180ok ($C->_acmp($x,$y),-1);
181ok ($C->_acmp($y,$x),+1);
574bacfe 182
9b924220 183$x = $C->_new("1234567890123456789");
184$y = $C->_new("987654321012345678");
56d9de68 185ok ($C->_acmp($x,$y),1);
186ok ($C->_acmp($y,$x),-1);
187ok ($C->_acmp($x,$x),0);
188ok ($C->_acmp($y,$y),0);
189
9b924220 190$x = $C->_new("1234");
191$y = $C->_new("987654321012345678");
56d9de68 192ok ($C->_acmp($x,$y),-1);
193ok ($C->_acmp($y,$x),1);
194ok ($C->_acmp($x,$x),0);
195ok ($C->_acmp($y,$y),0);
196
197# _modinv
9b924220 198$x = $C->_new("8");
199$y = $C->_new("5033");
56d9de68 200my ($xmod,$sign) = $C->_modinv($x,$y);
9b924220 201ok ($C->_str($xmod),'629'); # -629 % 5033 == 4404
56d9de68 202ok ($sign, '-');
203
574bacfe 204# _div
9b924220 205$x = $C->_new("3333"); $y = $C->_new("1111");
206ok ($C->_str(scalar $C->_div($x,$y)),3);
207$x = $C->_new("33333"); $y = $C->_new("1111"); ($x,$y) = $C->_div($x,$y);
208ok ($C->_str($x),30); ok ($C->_str($y),3);
209$x = $C->_new("123"); $y = $C->_new("1111");
210($x,$y) = $C->_div($x,$y); ok ($C->_str($x),0); ok ($C->_str($y),123);
574bacfe 211
212# _num
56d9de68 213foreach (qw/1 12 123 1234 12345 1234567 12345678 123456789 1234567890/)
214 {
9b924220 215 $x = $C->_new("$_");
216 ok (ref($x)||'','ARRAY'); ok ($C->_str($x),"$_");
56d9de68 217 $x = $C->_num($x); ok (ref($x)||'',''); ok ($x,$_);
218 }
574bacfe 219
b3abae2a 220# _sqrt
9b924220 221$x = $C->_new("144"); ok ($C->_str($C->_sqrt($x)),'12');
222$x = $C->_new("144000000000000"); ok ($C->_str($C->_sqrt($x)),'12000000');
990fb837 223
224# _root
9b924220 225$x = $C->_new("81"); my $n = $C->_new("3"); # 4*4*4 = 64, 5*5*5 = 125
226ok ($C->_str($C->_root($x,$n)),'4'); # 4.xx => 4.0
227$x = $C->_new("81"); $n = $C->_new("4"); # 3*3*3*3 == 81
228ok ($C->_str($C->_root($x,$n)),'3');
b3abae2a 229
091c87b1 230# _pow (and _root)
9b924220 231$x = $C->_new("0"); $n = $C->_new("3"); # 0 ** y => 0
232ok ($C->_str($C->_pow($x,$n)), 0);
233$x = $C->_new("3"); $n = $C->_new("0"); # x ** 0 => 1
234ok ($C->_str($C->_pow($x,$n)), 1);
235$x = $C->_new("1"); $n = $C->_new("3"); # 1 ** y => 1
236ok ($C->_str($C->_pow($x,$n)), 1);
237$x = $C->_new("5"); $n = $C->_new("1"); # x ** 1 => x
238ok ($C->_str($C->_pow($x,$n)), 5);
b282a552 239
9b924220 240$x = $C->_new("81"); $n = $C->_new("3"); # 81 ** 3 == 531441
241ok ($C->_str($C->_pow($x,$n)),81 ** 3);
091c87b1 242
9b924220 243ok ($C->_str($C->_root($x,$n)),81);
091c87b1 244
9b924220 245$x = $C->_new("81");
246ok ($C->_str($C->_pow($x,$n)),81 ** 3);
247ok ($C->_str($C->_pow($x,$n)),'150094635296999121'); # 531441 ** 3 ==
091c87b1 248
9b924220 249ok ($C->_str($C->_root($x,$n)),'531441');
250ok ($C->_str($C->_root($x,$n)),'81');
091c87b1 251
9b924220 252$x = $C->_new("81"); $n = $C->_new("14");
253ok ($C->_str($C->_pow($x,$n)),'523347633027360537213511521');
254ok ($C->_str($C->_root($x,$n)),'81');
091c87b1 255
9b924220 256$x = $C->_new("523347633027360537213511520");
257ok ($C->_str($C->_root($x,$n)),'80');
091c87b1 258
9b924220 259$x = $C->_new("523347633027360537213511522");
260ok ($C->_str($C->_root($x,$n)),'81');
091c87b1 261
262my $res = [ qw/ 9 31 99 316 999 3162 9999/ ];
263
264# 99 ** 2 = 9801, 999 ** 2 = 998001 etc
265for my $i (2 .. 9)
266 {
9b924220 267 $x = '9' x $i; $x = $C->_new($x);
268 $n = $C->_new("2");
091c87b1 269 my $rc = '9' x ($i-1). '8' . '0' x ($i-1) . '1';
270 print "# _pow( ", '9' x $i, ", 2) \n" unless
9b924220 271 ok ($C->_str($C->_pow($x,$n)),$rc);
091c87b1 272
273 if ($i <= 7)
274 {
9b924220 275 $x = '9' x $i; $x = $C->_new($x);
276 $n = '9' x $i; $n = $C->_new($n);
091c87b1 277 print "# _root( ", '9' x $i, ", ", 9 x $i, ") \n" unless
9b924220 278 ok ($C->_str($C->_root($x,$n)),'1');
091c87b1 279
9b924220 280 $x = '9' x $i; $x = $C->_new($x);
281 $n = $C->_new("2");
091c87b1 282 print "# _root( ", '9' x $i, ", ", 9 x $i, ") \n" unless
9b924220 283 ok ($C->_str($C->_root($x,$n)), $res->[$i-2]);
091c87b1 284 }
285 }
286
287##############################################################################
b3abae2a 288# _fac
9b924220 289$x = $C->_new("0"); ok ($C->_str($C->_fac($x)),'1');
290$x = $C->_new("1"); ok ($C->_str($C->_fac($x)),'1');
291$x = $C->_new("2"); ok ($C->_str($C->_fac($x)),'2');
292$x = $C->_new("3"); ok ($C->_str($C->_fac($x)),'6');
293$x = $C->_new("4"); ok ($C->_str($C->_fac($x)),'24');
294$x = $C->_new("5"); ok ($C->_str($C->_fac($x)),'120');
295$x = $C->_new("10"); ok ($C->_str($C->_fac($x)),'3628800');
296$x = $C->_new("11"); ok ($C->_str($C->_fac($x)),'39916800');
297$x = $C->_new("12"); ok ($C->_str($C->_fac($x)),'479001600');
298$x = $C->_new("13"); ok ($C->_str($C->_fac($x)),'6227020800');
091c87b1 299
300# test that _fac modifes $x in place for small arguments
9b924220 301$x = $C->_new("3"); $C->_fac($x); ok ($C->_str($x),'6');
302$x = $C->_new("13"); $C->_fac($x); ok ($C->_str($x),'6227020800');
56d9de68 303
304##############################################################################
305# _inc and _dec
306foreach (qw/1 11 121 1231 12341 1234561 12345671 123456781 1234567891/)
307 {
9b924220 308 $x = $C->_new("$_"); $C->_inc($x);
309 print "# \$x = ",$C->_str($x),"\n"
310 unless ok ($C->_str($x),substr($_,0,length($_)-1) . '2');
311 $C->_dec($x); ok ($C->_str($x),$_);
56d9de68 312 }
313foreach (qw/19 119 1219 12319 1234519 12345619 123456719 1234567819/)
314 {
9b924220 315 $x = $C->_new("$_"); $C->_inc($x);
316 print "# \$x = ",$C->_str($x),"\n"
317 unless ok ($C->_str($x),substr($_,0,length($_)-2) . '20');
318 $C->_dec($x); ok ($C->_str($x),$_);
56d9de68 319 }
320foreach (qw/999 9999 99999 9999999 99999999 999999999 9999999999 99999999999/)
321 {
9b924220 322 $x = $C->_new("$_"); $C->_inc($x);
323 print "# \$x = ",$C->_str($x),"\n"
324 unless ok ($C->_str($x), '1' . '0' x (length($_)));
325 $C->_dec($x); ok ($C->_str($x),$_);
56d9de68 326 }
b3abae2a 327
9b924220 328$x = $C->_new("1000"); $C->_inc($x); ok ($C->_str($x),'1001');
329$C->_dec($x); ok ($C->_str($x),'1000');
e745a66c 330
56d9de68 331my $BL;
332{
333 no strict 'refs';
334 $BL = &{"$C"."::_base_len"}();
335}
336
e745a66c 337$x = '1' . '0' x $BL;
338$z = '1' . '0' x ($BL-1); $z .= '1';
9b924220 339$x = $C->_new($x); $C->_inc($x); ok ($C->_str($x),$z);
e745a66c 340
341$x = '1' . '0' x $BL; $z = '9' x $BL;
9b924220 342$x = $C->_new($x); $C->_dec($x); ok ($C->_str($x),$z);
e745a66c 343
574bacfe 344# should not happen:
9b924220 345# $x = $C->_new("-2"); $y = $C->_new("4"); ok ($C->_acmp($x,$y),-1);
574bacfe 346
56d9de68 347###############################################################################
394e6ffb 348# _mod
9b924220 349$x = $C->_new("1000"); $y = $C->_new("3");
350ok ($C->_str(scalar $C->_mod($x,$y)),1);
351$x = $C->_new("1000"); $y = $C->_new("2");
352ok ($C->_str(scalar $C->_mod($x,$y)),0);
394e6ffb 353
354# _and, _or, _xor
9b924220 355$x = $C->_new("5"); $y = $C->_new("2");
356ok ($C->_str(scalar $C->_xor($x,$y)),7);
357$x = $C->_new("5"); $y = $C->_new("2");
358ok ($C->_str(scalar $C->_or($x,$y)),7);
359$x = $C->_new("5"); $y = $C->_new("3");
360ok ($C->_str(scalar $C->_and($x,$y)),1);
394e6ffb 361
362# _from_hex, _from_bin
9b924220 363ok ($C->_str( $C->_from_hex("0xFf")),255);
364ok ($C->_str( $C->_from_bin("0b10101011")),160+11);
394e6ffb 365
61f5c3f5 366# _as_hex, _as_bin
9b924220 367ok ($C->_str( $C->_from_hex( $C->_as_hex( $C->_new("128")))), 128);
368ok ($C->_str( $C->_from_bin( $C->_as_bin( $C->_new("128")))), 128);
369ok ($C->_str( $C->_from_hex( $C->_as_hex( $C->_new("0")))), 0);
370ok ($C->_str( $C->_from_bin( $C->_as_bin( $C->_new("0")))), 0);
371ok ($C->_as_hex( $C->_new("0")), '0x0');
372ok ($C->_as_bin( $C->_new("0")), '0b0');
373ok ($C->_as_hex( $C->_new("12")), '0xc');
374ok ($C->_as_bin( $C->_new("12")), '0b1100');
61f5c3f5 375
574bacfe 376# _check
9b924220 377$x = $C->_new("123456789");
bd05a461 378ok ($C->_check($x),0);
379ok ($C->_check(123),'123 is not a reference');
0716bf9b 380
61f5c3f5 381###############################################################################
56d9de68 382# __strip_zeros
383
384{
385 no strict 'refs';
386 # correct empty arrays
387 $x = &{$C."::__strip_zeros"}([]); ok (@$x,1); ok ($x->[0],0);
388 # don't strip single elements
389 $x = &{$C."::__strip_zeros"}([0]); ok (@$x,1); ok ($x->[0],0);
390 $x = &{$C."::__strip_zeros"}([1]); ok (@$x,1); ok ($x->[0],1);
391 # don't strip non-zero elements
392 $x = &{$C."::__strip_zeros"}([0,1]);
393 ok (@$x,2); ok ($x->[0],0); ok ($x->[1],1);
394 $x = &{$C."::__strip_zeros"}([0,1,2]);
395 ok (@$x,3); ok ($x->[0],0); ok ($x->[1],1); ok ($x->[2],2);
396
397 # but strip leading zeros
398 $x = &{$C."::__strip_zeros"}([0,1,2,0]);
399 ok (@$x,3); ok ($x->[0],0); ok ($x->[1],1); ok ($x->[2],2);
400
401 $x = &{$C."::__strip_zeros"}([0,1,2,0,0]);
402 ok (@$x,3); ok ($x->[0],0); ok ($x->[1],1); ok ($x->[2],2);
403
404 $x = &{$C."::__strip_zeros"}([0,1,2,0,0,0]);
405 ok (@$x,3); ok ($x->[0],0); ok ($x->[1],1); ok ($x->[2],2);
406
407 # collapse multiple zeros
408 $x = &{$C."::__strip_zeros"}([0,0,0,0]);
409 ok (@$x,1); ok ($x->[0],0);
410}
61f5c3f5 411
0716bf9b 412# done
413
4141;
574bacfe 415