1256a93a4Safresh1# -*- mode: perl; -*- 25759b3d2Safresh1 35759b3d2Safresh1# Test blog function (and bpow, since it uses blog), as well as bexp(). 45759b3d2Safresh1 55759b3d2Safresh1# It is too slow to be simple included in bigfltpm.inc, where it would get 65759b3d2Safresh1# executed 3 times. One time would be under Math::BigInt::BareCalc, which 7f3efcd01Safresh1# shouldn't make any difference since there is no $LIB->_log() function, and 85759b3d2Safresh1# one time under a subclass, which *should* work. 95759b3d2Safresh1 105759b3d2Safresh1# But it is better to test the numerical functionality, instead of not testing 115759b3d2Safresh1# it at all (which did lead to wrong answers for 0 < $x < 1 in blog() in 125759b3d2Safresh1# versions up to v1.63, and for bsqrt($x) when $x << 1 for instance). 135759b3d2Safresh1 145759b3d2Safresh1use strict; 155759b3d2Safresh1use warnings; 165759b3d2Safresh1 175759b3d2Safresh1use Test::More tests => 73; 185759b3d2Safresh1 195759b3d2Safresh1use Math::BigFloat only => 'FastCalc'; 205759b3d2Safresh1use Math::BigInt; 215759b3d2Safresh1 225759b3d2Safresh1my $class = "Math::BigInt"; 235759b3d2Safresh1 245759b3d2Safresh1############################################################################### 255759b3d2Safresh1# test $n->blog() in Math::BigInt (broken until 1.80) 265759b3d2Safresh1 275759b3d2Safresh1is($class->new(2)->blog(), '0', "$class->new(2)->blog()"); 285759b3d2Safresh1is($class->new(288)->blog(), '5', "$class->new(288)->blog()"); 295759b3d2Safresh1is($class->new(2000)->blog(), '7', "$class->new(2000)->blog()"); 305759b3d2Safresh1 315759b3d2Safresh1############################################################################### 325759b3d2Safresh1# test $n->bexp() in Math::BigInt 335759b3d2Safresh1 345759b3d2Safresh1is($class->new(1)->bexp(), '2', "$class->new(1)->bexp()"); 355759b3d2Safresh1is($class->new(2)->bexp(), '7', "$class->new(2)->bexp()"); 365759b3d2Safresh1is($class->new(3)->bexp(), '20', "$class->new(3)->bexp()"); 375759b3d2Safresh1 385759b3d2Safresh1############################################################################### 395759b3d2Safresh1# Math::BigFloat tests 405759b3d2Safresh1 415759b3d2Safresh1############################################################################### 425759b3d2Safresh1# test $n->blog(undef, N) where N > 67 (broken until 1.82) 435759b3d2Safresh1 445759b3d2Safresh1$class = "Math::BigFloat"; 455759b3d2Safresh1 465759b3d2Safresh1# These tests can take quite a while, but are necessary. Maybe protect them 475759b3d2Safresh1# with some alarm()? 485759b3d2Safresh1 495759b3d2Safresh1# this triggers the calculation and caching of ln(2): 505759b3d2Safresh1is($class->new(5)->blog(undef, 71), 515759b3d2Safresh1 '1.6094379124341003746007593332261876395256013542685177219126478914741790', 525759b3d2Safresh1 "$class->new(5)->blog(undef, 71)"); 535759b3d2Safresh1 545759b3d2Safresh1# if the cache was correct, we should get this result, fast: 555759b3d2Safresh1is($class->new(2)->blog(undef, 71), 565759b3d2Safresh1 '0.69314718055994530941723212145817656807550013436025525412068000949339362', 575759b3d2Safresh1 "$class->new(2)->blog(undef, 71)"); 585759b3d2Safresh1 595759b3d2Safresh1is($class->new(11)->blog(undef, 71), 605759b3d2Safresh1 '2.3978952727983705440619435779651292998217068539374171752185677091305736', 615759b3d2Safresh1 "$class->new(11)->blog(undef, 71)"); 625759b3d2Safresh1 635759b3d2Safresh1is($class->new(21)->blog(undef, 71), 645759b3d2Safresh1 '3.0445224377234229965005979803657054342845752874046106401940844835750742', 655759b3d2Safresh1 "$class->new(21)->blog(undef, 71)"); 665759b3d2Safresh1 675759b3d2Safresh1############################################################################### 685759b3d2Safresh1 695759b3d2Safresh1# These tests are now really fast, since they collapse to blog(10), basically 705759b3d2Safresh1# Don't attempt to run them with older versions. You are warned. 715759b3d2Safresh1 725759b3d2Safresh1# $x < 0 => NaN 735759b3d2Safresh1is($class->new(-2)->blog(), 'NaN', "$class->new(-2)->blog()"); 745759b3d2Safresh1is($class->new(-1)->blog(), 'NaN', "$class->new(-1)->blog()"); 755759b3d2Safresh1is($class->new(-10)->blog(), 'NaN', "$class->new(-10)->blog()"); 765759b3d2Safresh1is($class->new(-2, 2)->blog(), 'NaN', "$class->new(-2, 2)->blog()"); 775759b3d2Safresh1 785759b3d2Safresh1my $ten = $class->new(10)->blog(); 795759b3d2Safresh1 805759b3d2Safresh1# 10 is cached (up to 75 digits) 815759b3d2Safresh1is($class->new(10)->blog(), 825759b3d2Safresh1 '2.302585092994045684017991454684364207601', 835759b3d2Safresh1 qq|$class->new(10)->blog()|); 845759b3d2Safresh1 855759b3d2Safresh1# 0.1 is using the cached value for log(10), too 865759b3d2Safresh1 875759b3d2Safresh1is($class->new("0.1")->blog(), -$ten, 885759b3d2Safresh1 qq|$class->new("0.1")->blog()|); 895759b3d2Safresh1is($class->new("0.01")->blog(), -$ten * 2, 905759b3d2Safresh1 qq|$class->new("0.01")->blog()|); 915759b3d2Safresh1is($class->new("0.001")->blog(), -$ten * 3, 925759b3d2Safresh1 qq|$class->new("0.001")->blog()|); 935759b3d2Safresh1is($class->new("0.0001")->blog(), -$ten * 4, 945759b3d2Safresh1 qq|$class->new("0.0001")->blog()|); 955759b3d2Safresh1 965759b3d2Safresh1# also cached 975759b3d2Safresh1is($class->new(2)->blog(), 985759b3d2Safresh1 '0.6931471805599453094172321214581765680755', 995759b3d2Safresh1 qq|$class->new(2)->blog()|); 1005759b3d2Safresh1is($class->new(4)->blog(), $class->new(2)->blog * 2, 1015759b3d2Safresh1 qq|$class->new(4)->blog()|); 1025759b3d2Safresh1 1035759b3d2Safresh1# These are still slow, so do them only to 10 digits 1045759b3d2Safresh1 1055759b3d2Safresh1is($class->new("0.2")->blog(undef, 10), "-1.609437912", 1065759b3d2Safresh1 qq|$class->new("0.2")->blog(undef, 10)|); 1075759b3d2Safresh1is($class->new("0.3")->blog(undef, 10), "-1.203972804", 1085759b3d2Safresh1 qq|$class->new("0.3")->blog(undef, 10)|); 1095759b3d2Safresh1is($class->new("0.4")->blog(undef, 10), "-0.9162907319", 1105759b3d2Safresh1 qq|$class->new("0.4")->blog(undef, 10)|); 1115759b3d2Safresh1is($class->new("0.5")->blog(undef, 10), "-0.6931471806", 1125759b3d2Safresh1 qq|$class->new("0.5")->blog(undef, 10)|); 1135759b3d2Safresh1is($class->new("0.6")->blog(undef, 10), "-0.5108256238", 1145759b3d2Safresh1 qq|$class->new("0.6")->blog(undef, 10)|); 1155759b3d2Safresh1is($class->new("0.7")->blog(undef, 10), "-0.3566749439", 1165759b3d2Safresh1 qq|$class->new("0.7")->blog(undef, 10)|); 1175759b3d2Safresh1is($class->new("0.8")->blog(undef, 10), "-0.2231435513", 1185759b3d2Safresh1 qq|$class->new("0.8")->blog(undef, 10)|); 1195759b3d2Safresh1is($class->new("0.9")->blog(undef, 10), "-0.1053605157", 1205759b3d2Safresh1 qq|$class->new("0.9")->blog(undef, 10)|); 1215759b3d2Safresh1 1225759b3d2Safresh1is($class->new("9")->blog(undef, 10), "2.197224577", 1235759b3d2Safresh1 qq|$class->new("9")->blog(undef, 10)|); 1245759b3d2Safresh1 1255759b3d2Safresh1is($class->new("10")->blog(10, 10), "1.000000000", 1265759b3d2Safresh1 qq|$class->new("10")->blog(10, 10)|); 1275759b3d2Safresh1is($class->new("20")->blog(20, 10), "1.000000000", 1285759b3d2Safresh1 qq|$class->new("20")->blog(20, 10)|); 1295759b3d2Safresh1is($class->new("100")->blog(100, 10), "1.000000000", 1305759b3d2Safresh1 qq|$class->new("100")->blog(100, 10)|); 1315759b3d2Safresh1 1325759b3d2Safresh1is($class->new("100")->blog(10, 10), "2.000000000", # 10 ** 2 == 100 1335759b3d2Safresh1 qq|$class->new("100")->blog(10, 10)|); 1345759b3d2Safresh1is($class->new("400")->blog(20, 10), "2.000000000", # 20 ** 2 == 400 1355759b3d2Safresh1 qq|$class->new("400")->blog(20, 10)|); 1365759b3d2Safresh1 1375759b3d2Safresh1is($class->new("4")->blog(2, 10), "2.000000000", # 2 ** 2 == 4 1385759b3d2Safresh1 qq|$class->new("4")->blog(2, 10)|); 1395759b3d2Safresh1is($class->new("16")->blog(2, 10), "4.000000000", # 2 ** 4 == 16 1405759b3d2Safresh1 qq|$class->new("16")->blog(2, 10)|); 1415759b3d2Safresh1 1425759b3d2Safresh1is($class->new("1.2")->bpow("0.3", 10), "1.056219968", 1435759b3d2Safresh1 qq|$class->new("1.2")->bpow("0.3", 10)|); 1445759b3d2Safresh1is($class->new("10")->bpow("0.6", 10), "3.981071706", 1455759b3d2Safresh1 qq|$class->new("10")->bpow("0.6", 10)|); 1465759b3d2Safresh1 1475759b3d2Safresh1# blog should handle bigint input 148*f2a19305Safresh1is(Math::BigFloat->blog(Math::BigInt->new(100), 10), 2, "blog(100)"); 1495759b3d2Safresh1 1505759b3d2Safresh1############################################################################### 1515759b3d2Safresh1# some integer results 1525759b3d2Safresh1is($class->new(2)->bpow(32)->blog(2), "32", "2 ** 32"); 1535759b3d2Safresh1is($class->new(3)->bpow(32)->blog(3), "32", "3 ** 32"); 1545759b3d2Safresh1is($class->new(2)->bpow(65)->blog(2), "65", "2 ** 65"); 1555759b3d2Safresh1 1565759b3d2Safresh1my $x = Math::BigInt->new('777') ** 256; 1575759b3d2Safresh1my $base = Math::BigInt->new('12345678901234'); 1585759b3d2Safresh1is($x->copy()->blog($base), 56, 'blog(777**256, 12345678901234)'); 1595759b3d2Safresh1 1605759b3d2Safresh1$x = Math::BigInt->new('777') ** 777; 1615759b3d2Safresh1$base = Math::BigInt->new('777'); 1625759b3d2Safresh1is($x->copy()->blog($base), 777, 'blog(777**777, 777)'); 1635759b3d2Safresh1 1645759b3d2Safresh1############################################################################### 1655759b3d2Safresh1# test for bug in bsqrt() not taking negative _e into account 1665759b3d2Safresh1test_bpow('200', '0.5', 10, '14.14213562'); 1675759b3d2Safresh1test_bpow('20', '0.5', 10, '4.472135955'); 1685759b3d2Safresh1test_bpow('2', '0.5', 10, '1.414213562'); 1695759b3d2Safresh1test_bpow('0.2', '0.5', 10, '0.4472135955'); 1705759b3d2Safresh1test_bpow('0.02', '0.5', 10, '0.1414213562'); 1715759b3d2Safresh1test_bpow('0.49', '0.5', undef, '0.7'); 1725759b3d2Safresh1test_bpow('0.49', '0.5', 10, '0.7000000000'); 1735759b3d2Safresh1test_bpow('0.002', '0.5', 10, '0.04472135955'); 1745759b3d2Safresh1test_bpow('0.0002', '0.5', 10, '0.01414213562'); 1755759b3d2Safresh1test_bpow('0.0049', '0.5', undef, '0.07'); 1765759b3d2Safresh1test_bpow('0.0049', '0.5', 10, '0.07000000000'); 1775759b3d2Safresh1test_bpow('0.000002', '0.5', 10, '0.001414213562'); 1785759b3d2Safresh1test_bpow('0.021', '0.5', 10, '0.1449137675'); 1795759b3d2Safresh1test_bpow('1.2', '0.5', 10, '1.095445115'); 1805759b3d2Safresh1test_bpow('1.23', '0.5', 10, '1.109053651'); 1815759b3d2Safresh1test_bpow('12.3', '0.5', 10, '3.507135583'); 1825759b3d2Safresh1 1835759b3d2Safresh1test_bpow('9.9', '0.5', 10, '3.146426545'); 1845759b3d2Safresh1test_bpow('9.86902225', '0.5', 10, '3.141500000'); 1855759b3d2Safresh1test_bpow('9.86902225', '0.5', undef, '3.1415'); 1865759b3d2Safresh1 1875759b3d2Safresh1############################################################################### 1885759b3d2Safresh1# other tests for bpow() 1895759b3d2Safresh1 1905759b3d2Safresh1test_bpow('0.2', '0.41', 10, '0.5169187652'); 1915759b3d2Safresh1 192*f2a19305Safresh1is($class->new("0.1")->bpow("28.4", 40)->bsstr(), 193*f2a19305Safresh1 '3981071705534972507702523050877520434877e-68', 194*f2a19305Safresh1 qq|$class->new("0.1")->bpow("28.4", 40)->bsstr()|); 1955759b3d2Safresh1 1965759b3d2Safresh1# The following test takes too long. 1975759b3d2Safresh1#is($class->new("2")->bpow("-1034.5", 40)->bsstr(), 1985759b3d2Safresh1# '3841222690408590466868250378242558090957e-351', 1995759b3d2Safresh1# qq|$class->new("2")->bpow("-1034.5", 40)|); 2005759b3d2Safresh1 2015759b3d2Safresh1############################################################################### 2025759b3d2Safresh1# test bexp() with cached results 2035759b3d2Safresh1 2045759b3d2Safresh1is($class->new(1)->bexp(), '2.718281828459045235360287471352662497757', 2055759b3d2Safresh1 'bexp(1)'); 2065759b3d2Safresh1is($class->new(2)->bexp(40), $class->new(1)->bexp(45)->bpow(2, 40), 2075759b3d2Safresh1 'bexp(2)'); 2085759b3d2Safresh1 2095759b3d2Safresh1is($class->new("12.5")->bexp(61), $class->new(1)->bexp(65)->bpow(12.5, 61), 2105759b3d2Safresh1 'bexp(12.5)'); 2115759b3d2Safresh1 2125759b3d2Safresh1############################################################################### 2135759b3d2Safresh1# test bexp() with big values (non-cached) 2145759b3d2Safresh1 2155759b3d2Safresh1is($class->new(1)->bexp(100), 2165759b3d2Safresh1 '2.7182818284590452353602874713526624977572470936999' 2175759b3d2Safresh1 . '59574966967627724076630353547594571382178525166427', 2185759b3d2Safresh1 qq|$class->new(1)->bexp(100)|); 2195759b3d2Safresh1 2205759b3d2Safresh1is($class->new("12.5")->bexp(91), $class->new(1)->bexp(95)->bpow(12.5, 91), 2215759b3d2Safresh1 qq|$class->new("12.5")->bexp(91)|); 2225759b3d2Safresh1 2235759b3d2Safresh1is($class->new("-118.5")->bexp(20)->bsstr(), 2245759b3d2Safresh1 '34364014567198602057e-71', 2255759b3d2Safresh1 qq|$class->new("-118.5")->bexp(20)->bsstr()|); 2265759b3d2Safresh1 2275759b3d2Safresh1is($class->new("-394.84010945715266885")->bexp(20)->bsstr(), 2285759b3d2Safresh1 '33351796227864913873e-191', 2295759b3d2Safresh1 qq|$class->new("-118.5")->bexp(20)->bsstr()|); 2305759b3d2Safresh1 2315759b3d2Safresh1# all done 2325759b3d2Safresh1 2335759b3d2Safresh1############################################################################### 2345759b3d2Safresh1 2355759b3d2Safresh1sub test_bpow { 2365759b3d2Safresh1 my ($x, $y, $scale, $result) = @_; 2375759b3d2Safresh1 is($class->new($x)->bpow($y, $scale), $result, 2385759b3d2Safresh1 qq|$class->new($x)->bpow($y, | 2395759b3d2Safresh1 . (defined($scale) ? $scale : 'undef') 2405759b3d2Safresh1 . qq|)|); 2415759b3d2Safresh1} 242