my
$inf
=
$Math::Complex::Inf
;
my
$nan
=
$inf
-
$inf
;
Scalar::Util ->
import
(
'refaddr'
)
if
$scalar_util_ok
;
diag
"Skipping some tests since Scalar::Util is not installed."
unless
$scalar_util_ok
;
sub
isinf {
my
$x
=
shift
;
return
$x
==
$inf
||
$x
== -
$inf
;
}
sub
isnan {
my
$x
=
shift
;
return
$x
!=
$x
;
}
sub
pl2mbi {
my
$x
=
shift
;
return
Math::BigInt -> binf(
'+'
)
if
$x
==
$inf
;
return
Math::BigInt -> binf(
'-'
)
if
$x
== -
$inf
;
return
Math::BigInt -> bnan()
if
isnan(
$x
);
return
Math::BigInt -> new(
$x
);
}
sub
fdiv {
die
"Usage: fdiv X Y\n"
if
@_
!= 2;
my
$x
=
shift
;
my
$y
=
shift
;
if
(
$x
=~ /^\s
*nan
\s*$/i) {
$x
=
$nan
;
}
elsif
(
$x
=~ /^\s*([+-]?)inf(inity)?\s*$/i) {
$x
= $1 eq
'-'
? -
$inf
:
$inf
;
}
if
(
$y
=~ /^\s
*nan
\s*$/i) {
$y
=
$nan
;
}
elsif
(
$y
=~ /^\s*([+-]?)inf(inity)?\s*$/i) {
$y
= $1 eq
'-'
? -
$inf
:
$inf
;
}
if
(isnan(
$x
) || isnan(
$y
)) {
return
wantarray
? (
$nan
,
$nan
) :
$nan
;
}
if
(
$y
== 0) {
my
$q
=
$x
< 0 ? -
$inf
:
$x
> 0 ?
$inf
:
$nan
;
my
$r
=
$x
;
return
wantarray
? (
$q
,
$r
) :
$q
;
}
if
(isinf(
$x
)) {
my
$q
=
$x
/
$y
;
my
$r
=
$nan
;
return
wantarray
? (
$q
,
$r
) :
$q
;
if
(isinf(
$y
)) {
return
wantarray
? (
$nan
,
$nan
) :
$nan
;
}
else
{
if
((
$x
<=> 0) == (
$y
<=> 0)) {
return
wantarray
? (
$inf
,
$nan
) :
$inf
;
}
else
{
return
wantarray
? (-
$inf
,
$nan
) : -
$inf
;
}
}
}
if
(isinf(
$y
)) {
if
(
wantarray
) {
if
(
$x
== 0 || (
$x
<=> 0) == (
$y
<=> 0)) {
return
0,
$x
;
}
else
{
return
-1, (
$y
<=> 0) *
$inf
;
}
}
else
{
return
0;
}
}
my
$q
=
int
(
$x
/
$y
);
my
$r
=
$x
-
$y
*
$q
;
if
(
$y
> 0 &&
$r
< 0 ||
$y
< 0 &&
$r
> 0)
{
$q
-= 1;
$r
+=
$y
;
}
return
wantarray
? (
$q
,
$r
) :
$q
;
}
for
my
$num
(-
$inf
, -20 .. 20,
$inf
,
$nan
) {
for
my
$den
(-
$inf
, -20 .. 20,
$inf
,
$nan
) {
my
(
$quo
,
$rem
) = fdiv(
$num
,
$den
);
{
note(
qq|\n(\$quo, \$rem) = |
.
qq|Math::BigInt -> new("$num") -> bfdiv("$den")\n\n|
);
my
$mbi_num
= Math::BigInt -> new(
"$num"
);
my
$mbi_den
= Math::BigInt -> new(
"$den"
);
my
(
$mbi_num_addr
,
$mbi_den_addr
);
$mbi_num_addr
= refaddr(
$mbi_num
)
if
$scalar_util_ok
;
$mbi_den_addr
= refaddr(
$mbi_den
)
if
$scalar_util_ok
;
my
(
$mbi_quo
,
$mbi_rem
) =
$mbi_num
-> bfdiv(
$mbi_den
);
is(
ref
(
$mbi_num
),
'Math::BigInt'
,
"class of numerator is still Math::BigInt"
);
is(
ref
(
$mbi_den
),
'Math::BigInt'
,
"class of denominator is still Math::BigInt"
);
is(
ref
(
$mbi_quo
),
'Math::BigInt'
,
"class of quotient is Math::BigInt"
);
is(
ref
(
$mbi_rem
),
'Math::BigInt'
,
"class of remainder is Math::BigInt"
);
is(
$mbi_quo
, pl2mbi(
$quo
),
"$num / $den = $quo"
);
is(
$mbi_rem
, pl2mbi(
$rem
),
"$num % $den = $rem"
);
is(
$mbi_den
, pl2mbi(
$den
),
"value of denominator has not changed"
);
my
(
$mbi_quo_addr
,
$mbi_rem_addr
);
$mbi_quo_addr
= refaddr(
$mbi_quo
)
if
$scalar_util_ok
;
$mbi_rem_addr
= refaddr(
$mbi_rem
)
if
$scalar_util_ok
;
is(
$mbi_quo_addr
,
$mbi_num_addr
,
"the quotient object is the numerator object"
);
SKIP: {
skip
"Scalar::Util not available"
, 1
unless
$scalar_util_ok
;
ok(
$mbi_rem_addr
!=
$mbi_num_addr
&&
$mbi_rem_addr
!=
$mbi_den_addr
&&
$mbi_rem_addr
!=
$mbi_quo_addr
,
"the remainder object is neither the numerator,"
.
" denominator, nor quotient object"
);
}
}
{
note(
qq|\n\$quo = |
.
qq|Math::BigInt -> new("$num") -> bfdiv("$den")\n\n|
);
my
$mbi_num
= Math::BigInt -> new(
"$num"
);
my
$mbi_den
= Math::BigInt -> new(
"$den"
);
my
(
$mbi_num_addr
,
$mbi_den_addr
);
$mbi_num_addr
= refaddr(
$mbi_num
)
if
$scalar_util_ok
;
$mbi_den_addr
= refaddr(
$mbi_den
)
if
$scalar_util_ok
;
my
$mbi_quo
=
$mbi_num
-> bfdiv(
$mbi_den
);
is(
ref
(
$mbi_num
),
'Math::BigInt'
,
"class of numerator is still Math::BigInt"
);
is(
ref
(
$mbi_den
),
'Math::BigInt'
,
"class of denominator is still Math::BigInt"
);
is(
ref
(
$mbi_quo
),
'Math::BigInt'
,
"class of quotient is Math::BigInt"
);
is(
$mbi_quo
, pl2mbi(
$quo
),
"$num / $den = $quo"
);
is(
$mbi_den
, pl2mbi(
$den
),
"value of numerator has not changed"
);
my
$mbi_quo_addr
;
$mbi_quo_addr
= refaddr(
$mbi_quo
)
if
$scalar_util_ok
;
SKIP: {
skip
"Scalar::Util not available"
, 1
unless
$scalar_util_ok
;
is(
$mbi_quo_addr
,
$mbi_num_addr
,
"the quotient object is the numerator object"
);
}
}
{
note(
qq|\n\$quo = |
.
qq|Math::BigInt -> new("$num") -> bfmod("$den")\n\n|
);
my
$mbi_num
= Math::BigInt -> new(
"$num"
);
my
$mbi_den
= Math::BigInt -> new(
"$den"
);
my
(
$mbi_num_addr
,
$mbi_den_addr
);
$mbi_num_addr
= refaddr(
$mbi_num
)
if
$scalar_util_ok
;
$mbi_den_addr
= refaddr(
$mbi_den
)
if
$scalar_util_ok
;
my
$mbi_rem
=
$mbi_num
-> bfmod(
$mbi_den
);
is(
ref
(
$mbi_num
),
'Math::BigInt'
,
"class of numerator is still Math::BigInt"
);
is(
ref
(
$mbi_den
),
'Math::BigInt'
,
"class of denominator is still Math::BigInt"
);
is(
ref
(
$mbi_rem
),
'Math::BigInt'
,
"class of remainder is Math::BigInt"
);
is(
$mbi_rem
, pl2mbi(
$rem
),
"$num % $den = $rem"
);
is(
$mbi_den
, pl2mbi(
$den
),
"value of denominator has not changed"
);
my
$mbi_rem_addr
;
$mbi_rem_addr
= refaddr(
$mbi_rem
)
if
$scalar_util_ok
;
SKIP: {
skip
"Scalar::Util not available"
, 1
unless
$scalar_util_ok
;
is(
$mbi_rem_addr
,
$mbi_num_addr
,
"the remainder object is the numerator object"
);
}
}
}
}
for
my
$num
(-
$inf
, -20 .. -1, 1 .. 20,
$inf
,
$nan
) {
my
(
$quo
,
$rem
) = fdiv(
$num
,
$num
);
{
note(
qq|\n\$x = Math::BigInt -> new("$num"); |
.
qq|(\$quo, \$rem) = \$x -> bfdiv("\$x")\n\n|
);
my
$mbi_num
= Math::BigInt -> new(
"$num"
);
my
$mbi_num_addr
;
$mbi_num_addr
= refaddr(
$mbi_num
)
if
$scalar_util_ok
;
my
(
$mbi_quo
,
$mbi_rem
) =
$mbi_num
-> bfdiv(
$mbi_num
);
is(
ref
(
$mbi_num
),
'Math::BigInt'
,
"class of numerator is still Math::BigInt"
);
is(
ref
(
$mbi_quo
),
'Math::BigInt'
,
"class of quotient is Math::BigInt"
);
is(
ref
(
$mbi_rem
),
'Math::BigInt'
,
"class of remainder is Math::BigInt"
);
is(
$mbi_quo
, pl2mbi(
$quo
),
"$num / $num = $quo"
);
is(
$mbi_rem
, pl2mbi(
$rem
),
"$num % $num = $rem"
);
my
(
$mbi_quo_addr
,
$mbi_rem_addr
);
$mbi_quo_addr
= refaddr(
$mbi_quo
)
if
$scalar_util_ok
;
$mbi_rem_addr
= refaddr(
$mbi_rem
)
if
$scalar_util_ok
;
SKIP: {
skip
"Scalar::Util not available"
, 2
unless
$scalar_util_ok
;
is(
$mbi_quo_addr
,
$mbi_num_addr
,
"the quotient object is the numerator object"
);
ok(
$mbi_rem_addr
!=
$mbi_num_addr
&&
$mbi_rem_addr
!=
$mbi_quo_addr
,
"the remainder object is neither the numerator,"
.
" denominator, nor quotient object"
);
}
}
{
note(
qq|\n\$x = Math::BigInt -> new("$num"); |
.
qq|\$quo = \$x -> bfdiv(\$x)\n\n|
);
my
$mbi_num
= Math::BigInt -> new(
"$num"
);
my
$mbi_num_addr
;
$mbi_num_addr
= refaddr(
$mbi_num
)
if
$scalar_util_ok
;
my
$mbi_quo
=
$mbi_num
-> bfdiv(
$mbi_num
);
is(
ref
(
$mbi_num
),
'Math::BigInt'
,
"class of numerator is still Math::BigInt"
);
is(
ref
(
$mbi_quo
),
'Math::BigInt'
,
"class of quotient is Math::BigInt"
);
is(
$mbi_quo
, pl2mbi(
$quo
),
"$num / $num = $quo"
);
my
$mbi_quo_addr
;
$mbi_quo_addr
= refaddr(
$mbi_quo
)
if
$scalar_util_ok
;
SKIP: {
skip
"Scalar::Util not available"
, 1
unless
$scalar_util_ok
;
is(
$mbi_quo_addr
,
$mbi_num_addr
,
"the quotient object is the numerator object"
);
}
}
{
note(
qq|\n\$x = Math::BigInt -> new("$num") |
.
qq|\$quo = \$x -> bfmod(\$x)\n\n|
);
my
$mbi_num
= Math::BigInt -> new(
"$num"
);
my
$mbi_num_addr
;
$mbi_num_addr
= refaddr(
$mbi_num
)
if
$scalar_util_ok
;
my
$mbi_rem
=
$mbi_num
-> bfmod(
$mbi_num
);
is(
ref
(
$mbi_num
),
'Math::BigInt'
,
"class of numerator is still Math::BigInt"
);
is(
ref
(
$mbi_rem
),
'Math::BigInt'
,
"class of remainder is Math::BigInt"
);
is(
$mbi_rem
, pl2mbi(
$rem
),
"$num % $num = $rem"
);
my
$mbi_rem_addr
;
$mbi_rem_addr
= refaddr(
$mbi_rem
)
if
$scalar_util_ok
;
SKIP: {
skip
"Scalar::Util not available"
, 1
unless
$scalar_util_ok
;
is(
$mbi_rem_addr
,
$mbi_num_addr
,
"the remainder object is the numerator object"
);
}
}
}