xref: /openbsd-src/gnu/usr.bin/perl/cpan/Math-BigInt-FastCalc/t/biglog.t (revision f2a19305cfc49ea4d1a5feb55cd6c283c6f1e031)
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