package Math::BigInt::Calc;
use 5.005;
use strict;
use warnings;
require Exporter;
use vars qw/ @ISA @EXPORT $VERSION/;
@ISA = qw(Exporter);
@EXPORT = qw(
_add _mul _div _mod _sub
_new
_str _num _acmp _len
_digit
_is_zero _is_one
_is_even _is_odd
_check _zero _one _copy _zeros
);
$VERSION = '0.06';
# Package to store unsigned big integers in decimal and do math with them
# Internally the numbers are stored in an array with at least 1 element, no
# leading zero parts (except the first) and in base 100000
# todo:
# - fully remove funky $# stuff (maybe)
# - use integer; vs 1e7 as base
# USE_MUL: due to problems on certain os (os390, posix-bc) "* 1e-5" is used
# instead of "/ 1e5" at some places, (marked with USE_MUL). But instead of
# using the reverse only on problematic machines, I used it everytime to avoid
# the costly comparisons. This _should_ work everywhere. Thanx Peter Prymmer
##############################################################################
# global constants, flags and accessory
# constants for easier life
my $nan = 'NaN';
my $BASE_LEN = 5;
my $BASE = int("1e".$BASE_LEN); # var for trying to change it to 1e7
my $RBASE = 1e-5; # see USE_MUL
my $class = 'Math::BigInt::Calc';
##############################################################################
# create objects from various representations
sub _new
{
# (string) return ref to num_array
# Convert a number from string format to internal base 100000 format.
# Assumes normalized value as input.
shift @_ if $_[0] eq $class;
my $d = shift;
# print "_new $d $$d\n";
my $il = CORE::length($$d)-1;
# these leaves '00000' instead of int 0 and will be corrected after any op
return [ reverse(unpack("a" . ($il%5+1) . ("a5" x ($il/5)), $$d)) ];
}
sub _zero
{
# create a zero
return [ 0 ];
}
sub _one
{
# create a one
return [ 1 ];
}
sub _copy
{
shift @_ if $_[0] eq $class;
my $x = shift;
return [ @$x ];
}
##############################################################################
# convert back to string and number
sub _str
{
# (ref to BINT) return num_str
# Convert number from internal base 100000 format to string format.
# internal format is always normalized (no leading zeros, "-0" => "+0")
shift @_ if $_[0] eq $class;
my $ar = shift;
my $ret = "";
my $l = scalar @$ar; # number of parts
return $nan if $l < 1; # should not happen
# handle first one different to strip leading zeros from it (there are no
# leading zero parts in internal representation)
$l --; $ret .= $ar->[$l]; $l--;
# Interestingly, the pre-padd method uses more time
# the old grep variant takes longer (14 to 10 sec)
while ($l >= 0)
{
$ret .= substr('0000'.$ar->[$l],-5); # fastest way I could think of
$l--;
}
return \$ret;
}
sub _num
{
# Make a number (scalar int/float) from a BigInt object
shift @_ if $_[0] eq $class;
my $x = shift;
return $x->[0] if scalar @$x == 1; # below $BASE
my $fac = 1;
my $num = 0;
foreach (@$x)
{
$num += $fac*$_; $fac *= $BASE;
}
return $num;
}
##############################################################################
# actual math code
sub _add
{
# (ref to int_num_array, ref to int_num_array)
# routine to add two base 1e5 numbers
# stolen from Knuth Vol 2 Algorithm A pg 231
# there are separate routines to add and sub as per Knuth pg 233
# This routine clobbers up array x, but not y.
shift @_ if $_[0] eq $class;
my ($x,$y) = @_;
# for each in Y, add Y to X and carry. If after that, something is left in
# X, foreach in X add carry to X and then return X, carry
# Trades one "$j++" for having to shift arrays, $j could be made integer
# but this would impose a limit to number-length of 2**32.
my $i; my $car = 0; my $j = 0;
for $i (@$y)
{
$x->[$j] -= $BASE
if $car = (($x->[$j] += $i + $car) >= $BASE) ? 1 : 0;
$j++;
}
while ($car != 0)
{
$x->[$j] -= $BASE if $car = (($x->[$j] += $car) >= $BASE) ? 1 : 0; $j++;
}
return $x;
}
sub _sub
{
# (ref to int_num_array, ref to int_num_array)
# subtract base 1e5 numbers -- stolen from Knuth Vol 2 pg 232, $x > $y
# subtract Y from X (X is always greater/equal!) by modifying x in place
shift @_ if $_[0] eq $class;
my ($sx,$sy,$s) = @_;
my $car = 0; my $i; my $j = 0;
if (!$s)
{
#print "case 2\n";
for $i (@$sx)
{
last unless defined $sy->[$j] || $car;
#print "x: $i y: $sy->[$j] c: $car\n";
$i += $BASE if $car = (($i -= ($sy->[$j] || 0) + $car) < 0); $j++;
#print "x: $i y: $sy->[$j-1] c: $car\n";
}
# might leave leading zeros, so fix that
__strip_zeros($sx);
return $sx;
}
else
{
#print "case 1 (swap)\n";
for $i (@$sx)
{
last unless defined $sy->[$j] || $car;
#print "$sy->[$j] $i $car => $sx->[$j]\n";
$sy->[$j] += $BASE
if $car = (($sy->[$j] = $i-($sy->[$j]||0) - $car) < 0);
#print "$sy->[$j] $i $car => $sy->[$j]\n";
$j++;
}
# might leave leading zeros, so fix that
__strip_zeros($sy);
return $sy;
}
}
sub _mul
{
# (BINT, BINT) return nothing
# multiply two numbers in internal representation
# modifies first arg, second need not be different from first
shift @_ if $_[0] eq $class;
my ($xv,$yv) = @_;
my @prod = (); my ($prod,$car,$cty,$xi,$yi);
# since multiplying $x with $x fails, make copy in this case
$yv = [@$xv] if "$xv" eq "$yv";
# looping through @$y if $xi == 0 is silly! optimize it!
for $xi (@$xv)
{
$car = 0; $cty = 0;
for $yi (@$yv)
{
$prod = $xi * $yi + ($prod[$cty] || 0) + $car;
$prod[$cty++] =
$prod - ($car = int($prod * 1e-5)) * $BASE; # see USE_MUL
}
$prod[$cty] += $car if $car; # need really to check for 0?
$xi = shift @prod;
}
# for $xi (@$xv)
# {
# $car = 0; $cty = 0;
# # looping through this if $xi == 0 is silly! optimize it!
# if (($xi||0) != 0)
# {
# for $yi (@$yv)
# {
# $prod = $prod[$cty]; $prod += ($car + $xi * $yi); # no ||0 here
# $prod[$cty++] =
# $prod - ($car = int($prod * 1e-5)) * $BASE; # see USE_MUL
# }
# }
# $prod[$cty] += $car if $car; # need really to check for 0?
# $xi = shift @prod;
# }
push @$xv, @prod;
__strip_zeros($xv);
# normalize (handled last to save check for $y->is_zero()
return $xv;
}
sub _div
{
# ref to array, ref to array, modify first array and return remainder if
# in list context
# no longer handles sign
shift @_ if $_[0] eq $class;
my ($x,$yorg) = @_;
my ($car,$bar,$prd,$dd,$xi,$yi,@q,$v2,$v1);
my (@d,$tmp,$q,$u2,$u1,$u0);
$car = $bar = $prd = 0;
my $y = [ @$yorg ];
if (($dd = int($BASE/($y->[-1]+1))) != 1)
{
for $xi (@$x)
{
$xi = $xi * $dd + $car;
$xi -= ($car = int($xi * $RBASE)) * $BASE; # see USE_MUL
}
push(@$x, $car); $car = 0;
for $yi (@$y)
{
$yi = $yi * $dd + $car;
$yi -= ($car = int($yi * $RBASE)) * $BASE; # see USE_MUL
}
}
else
{
push(@$x, 0);
}
@q = (); ($v2,$v1) = @$y[-2,-1];
$v2 = 0 unless $v2;
while ($#$x > $#$y)
{
($u2,$u1,$u0) = @$x[-3..-1];
$u2 = 0 unless $u2;
#warn "oups v1 is 0, u0: $u0 $y->[-2] $y->[-1] l ",scalar @$y,"\n"
# if $v1 == 0;
$q = (($u0 == $v1) ? 99999 : int(($u0*$BASE+$u1)/$v1));
--$q while ($v2*$q > ($u0*1e5+$u1-$q*$v1)*$BASE+$u2);
if ($q)
{
($car, $bar) = (0,0);
for ($yi = 0, $xi = $#$x-$#$y-1; $yi <= $#$y; ++$yi,++$xi)
{
$prd = $q * $y->[$yi] + $car;
$prd -= ($car = int($prd * $RBASE)) * $BASE; # see USE_MUL
$x->[$xi] += 1e5 if ($bar = (($x->[$xi] -= $prd + $bar) < 0));
}
if ($x->[-1] < $car + $bar)
{
$car = 0; --$q;
for ($yi = 0, $xi = $#$x-$#$y-1; $yi <= $#$y; ++$yi,++$xi)
{
$x->[$xi] -= 1e5
if ($car = (($x->[$xi] += $y->[$yi] + $car) > $BASE));
}
}
}
pop(@$x); unshift(@q, $q);
}
if (wantarray)
{
@d = ();
if ($dd != 1)
{
$car = 0;
for $xi (reverse @$x)
{
$prd = $car * $BASE + $xi;
$car = $prd - ($tmp = int($prd / $dd)) * $dd; # see USE_MUL
unshift(@d, $tmp);
}
}
else
{
@d = @$x;
}
@$x = @q;
__strip_zeros($x);
__strip_zeros(\@d);
return ($x,\@d);
}
@$x = @q;
__strip_zeros($x);
return $x;
}
##############################################################################
# testing
sub _acmp
{
# internal absolute post-normalized compare (ignore signs)
# ref to array, ref to array, return <0, 0, >0
# arrays must have at least one entry; this is not checked for
shift @_ if $_[0] eq $class;
my ($cx, $cy) = @_;
#print "$cx $cy\n";
my ($i,$a,$x,$y,$k);
# calculate length based on digits, not parts
$x = _len($cx); $y = _len($cy);
# print "length: ",($x-$y),"\n";
return $x-$y if ($x - $y); # if different in length
#print "full compare\n";
$i = 0; $a = 0;
# first way takes 5.49 sec instead of 4.87, but has the early out advantage
# so grep is slightly faster, but more inflexible. hm. $_ instead of $k
# yields 5.6 instead of 5.5 sec huh?
# manual way (abort if unequal, good for early ne)
my $j = scalar @$cx - 1;
while ($j >= 0)
{
# print "$cx->[$j] $cy->[$j] $a",$cx->[$j]-$cy->[$j],"\n";
last if ($a = $cx->[$j] - $cy->[$j]); $j--;
}
return $a;
# while it early aborts, it is even slower than the manual variant
#grep { return $a if ($a = $_ - $cy->[$i++]); } @$cx;
# grep way, go trough all (bad for early ne)
#grep { $a = $_ - $cy->[$i++]; } @$cx;
#return $a;
}
sub _len
{
# computer number of digits in bigint, minus the sign
# int() because add/sub sometimes leaves strings (like '00005') instead of
# int ('5') in this place, causing length to fail
shift @_ if $_[0] eq $class;
my $cx = shift;
return (@$cx-1)*5+length(int($cx->[-1]));
}
sub _digit
{
# return the nth digit, negative values count backward
# zero is rightmost, so _digit(123,0) will give 3
shift @_ if $_[0] eq $class;
my $x = shift;
my $n = shift || 0;
my $len = _len($x);
$n = $len+$n if $n < 0; # -1 last, -2 second-to-last
$n = abs($n); # if negative was too big
$len--; $n = $len if $n > $len; # n to big?
my $elem = int($n / 5); # which array element
my $digit = $n % 5; # which digit in this element
$elem = '0000'.@$x[$elem]; # get element padded with 0's
return substr($elem,-$digit-1,1);
}
sub _zeros
{
# return amount of trailing zeros in decimal
# check each array elem in _m for having 0 at end as long as elem == 0
# Upon finding a elem != 0, stop
shift @_ if $_[0] eq $class;
my $x = shift;
my $zeros = 0; my $elem;
foreach my $e (@$x)
{
if ($e != 0)
{
$elem = "$e"; # preserve x
$elem =~ s/.*?(0*$)/$1/; # strip anything not zero
$zeros *= 5; # elems * 5
$zeros += CORE::length($elem); # count trailing zeros
last; # early out
}
$zeros ++; # real else branch: 50% slower!
}
return $zeros;
}
##############################################################################
# _is_* routines
sub _is_zero
{
# return true if arg (BINT or num_str) is zero (array '+', '0')
shift @_ if $_[0] eq $class;
my ($x) = shift;
return (((scalar @$x == 1) && ($x->[0] == 0))) <=> 0;
}
sub _is_even
{
# return true if arg (BINT or num_str) is even
shift @_ if $_[0] eq $class;
my ($x) = shift;
return (!($x->[0] & 1)) <=> 0;
}
sub _is_odd
{
# return true if arg (BINT or num_str) is even
shift @_ if $_[0] eq $class;
my ($x) = shift;
return (($x->[0] & 1)) <=> 0;
}
sub _is_one
{
# return true if arg (BINT or num_str) is one (array '+', '1')
shift @_ if $_[0] eq $class;
my ($x) = shift;
return (scalar @$x == 1) && ($x->[0] == 1) <=> 0;
}
sub __strip_zeros
{
# internal normalization function that strips leading zeros from the array
# args: ref to array
#trace(@_);
shift @_ if $_[0] eq $class;
my $s = shift;
my $cnt = scalar @$s; # get count of parts
my $i = $cnt-1;
#print "strip: cnt $cnt i $i\n";
# '0', '3', '4', '0', '0',
# 0 1 2 3 4
# cnt = 5, i = 4
# i = 4
# i = 3
# => fcnt = cnt - i (5-2 => 3, cnt => 5-1 = 4, throw away from 4th pos)
# >= 1: skip first part (this can be zero)
while ($i > 0) { last if $s->[$i] != 0; $i--; }
$i++; splice @$s,$i if ($i < $cnt); # $i cant be 0
return $s;
}
###############################################################################
# check routine to test internal state of corruptions
sub _check
{
# no checks yet, pull it out from the test suite
shift @_ if $_[0] eq $class;
my ($x) = shift;
return "$x is not a reference" if !ref($x);
# are all parts are valid?
my $i = 0; my $j = scalar @$x; my ($e,$try);
while ($i < $j)
{
$e = $x->[$i]; $e = 'undef' unless defined $e;
$try = '=~ /^[\+]?[0-9]+\$/; '."($x, $e)";
last if $e !~ /^[+]?[0-9]+$/;
$try = ' < 0 || >= $BASE; '."($x, $e)";
last if $e <0 || $e >= $BASE;
# this test is disabled, since new/bnorm and certain ops (like early out
# in add/sub) are allowed/expected to leave '00000' in some elements
#$try = '=~ /^00+/; '."($x, $e)";
#last if $e =~ /^00+/;
$i++;
}
return "Illegal part '$e' at pos $i (tested: $try)" if $i < $j;
return 0;
}
1;
__END__
=head1 NAME
Math::BigInt::Calc - Pure Perl module to support Math::BigInt
=head1 SYNOPSIS
Provides support for big integer calculations. Not intended
to be used by other modules. Other modules which export the
same functions can also be used to support Math::Bigint
=head1 DESCRIPTION
In order to allow for multiple big integer libraries, Math::BigInt
was rewritten to use library modules for core math routines. Any
module which follows the same API as this can be used instead by
using the following call:
use Math::BigInt Calc => BigNum;
=head1 EXPORT
The following functions MUST be exported in order to support
the use by Math::BigInt:
_new(string) return ref to new object from ref to decimal string
_zero() return a new object with value 0
_one() return a new object with value 1
_str(obj) return ref to a string representing the object
_num(obj) returns a Perl integer/floating point number
NOTE: because of Perl numeric notation defaults,
the _num'ified obj may lose accuracy due to
machine-dependend floating point size limitations
_add(obj,obj) Simple addition of two objects
_mul(obj,obj) Multiplication of two objects
_div(obj,obj) Division of the 1st object by the 2nd
In list context, returns (result,remainder).
NOTE: this is integer math, so no
fractional part will be returned.
_sub(obj,obj) Simple subtraction of 1 object from another
a third, optional parameter indicates that the params
are swapped. In this case, the first param needs to
be preserved, while you can destroy the second.
sub (x,y,1) => return x - y and keep x intact!
_acmp(obj,obj) <=> operator for objects (return -1, 0 or 1)
_len(obj) returns count of the decimal digits of the object
_digit(obj,n) returns the n'th decimal digit of object
_is_one(obj) return true if argument is +1
_is_zero(obj) return true if argument is 0
_is_even(obj) return true if argument is even (0,2,4,6..)
_is_odd(obj) return true if argument is odd (1,3,5,7..)
_copy return a ref to a true copy of the object
_check(obj) check whether internal representation is still intact
return 0 for ok, otherwise error message as string
The following functions are optional, and can be exported if the underlying lib
has a fast way to do them. If not defined, Math::BigInt will use a pure, but
slow, Perl function as fallback to emulate these:
_from_hex(str) return ref to new object from ref to hexadecimal string
_from_bin(str) return ref to new object from ref to binary string
_rsft(obj,N,B) shift object in base B by N 'digits' right
_lsft(obj,N,B) shift object in base B by N 'digits' left
_xor(obj1,obj2) XOR (bit-wise) object 1 with object 2
Mote: XOR, AND and OR pad with zeros if size mismatches
_and(obj1,obj2) AND (bit-wise) object 1 with object 2
_or(obj1,obj2) OR (bit-wise) object 1 with object 2
_sqrt(obj) return the square root of object
_pow(obj,obj) return object 1 to the power of object 2
_gcd(obj,obj) return Greatest Common Divisor of two objects
_zeros(obj) return number of trailing decimal zeros
_dec(obj) decrement object by one (input is >= 1)
_inc(obj) increment object by one
Input strings come in as unsigned but with prefix (i.e. as '123', '0xabc'
or '0b1101').
Testing of input parameter validity is done by the caller, so you need not
worry about underflow (C<_sub()>, C<_dec()>) nor about division by zero or
similar cases.
=head1 LICENSE
This program is free software; you may redistribute it and/or modify it under
the same terms as Perl itself.
=head1 AUTHORS
Original math code by Mark Biggar, rewritten by Tels L
in late 2000, 2001.
Seperated from BigInt and shaped API with the help of John Peacock.
=head1 SEE ALSO
L, L.
=cut