$| = 1;
my
@cases
=
(
[ 0, -1, 0 ],
[ 0, -1.5, 0 ],
[ 0, -1, 0 ],
[ 7, -1, 14 ],
[ 7, -1, 14 ],
[ 7.5, -1, 14 ],
[ 7, -1.5, 14 ],
[ 7.5, -1.5, 14 ],
[ -7, -1, -14 ],
[ -7.5, -1, -14 ],
[ -7, -1.5, -14 ],
[ -7.5, -1.5, -14 ],
[ 7, 1, 3 ],
[ 7.5, 1, 3 ],
[ 7, 1.5, 3 ],
[ 7.5, 1.5, 3 ],
[ -7, 1, -4 ],
[ -7.5, 1, -4 ],
[ -7, 1.5, -4 ],
[ -7.5, 1.5, -4 ],
[
"NaN"
, 0,
"NaN"
],
[
"NaN"
, -1,
"NaN"
],
[
"NaN"
, -1.5,
"NaN"
],
[
"NaN"
,
"-inf"
,
"NaN"
],
[
"NaN"
, 1,
"NaN"
],
[
"NaN"
, 1.5,
"NaN"
],
[
"NaN"
,
"inf"
,
"NaN"
],
[
"NaN"
,
"NaN"
,
"NaN"
],
[ 0,
"NaN"
,
"NaN"
],
[ 1,
"NaN"
,
"NaN"
],
[ 1.5,
"NaN"
,
"NaN"
],
[
"inf"
,
"NaN"
,
"NaN"
],
[ -1,
"NaN"
,
"NaN"
],
[ -1.5,
"NaN"
,
"NaN"
],
[
"-inf"
,
"NaN"
,
"NaN"
],
[
"NaN"
,
"NaN"
,
"NaN"
],
[
"inf"
, 0,
"inf"
],
[
"inf"
, -1,
"inf"
],
[
"inf"
, -1.5,
"inf"
],
[
"inf"
,
"-inf"
,
"inf"
],
[
"inf"
, 1,
"inf"
],
[
"inf"
, 1.5,
"inf"
],
[
"inf"
,
"inf"
,
"NaN"
],
[ 0,
"-inf"
,
"NaN"
],
[ 1,
"-inf"
,
"inf"
],
[ 1.5,
"-inf"
,
"inf"
],
[ -1,
"-inf"
,
"-inf"
],
[ -1.5,
"-inf"
,
"-inf"
],
[
"-inf"
, 0,
"-inf"
],
[
"-inf"
, -1,
"-inf"
],
[
"-inf"
, -1.5,
"-inf"
],
[
"-inf"
,
"-inf"
,
"-inf"
],
[
"-inf"
, 1,
"-inf"
],
[
"-inf"
, 1.5,
"-inf"
],
[
"-inf"
,
"inf"
,
"NaN"
],
[ 0,
"inf"
, 0 ],
[ 1,
"inf"
, 0 ],
[ 1.5,
"inf"
, 0 ],
[ -1,
"inf"
, -1 ],
[ -1.5,
"inf"
, -1 ],
);
if
(1) {
my
@x
= (
"-inf"
,
"inf"
,
"NaN"
,
map
{
$_
/ 4} -25 .. 25);
my
@y
= (
"-inf"
,
"inf"
,
"NaN"
,
map
{
$_
/ 4} -25 .. 25);
for
my
$x
(
@x
) {
for
my
$y
(
@y
) {
my
$xint
= Math::BigFloat -> new(
$x
) -> as_int();
my
$yint
= Math::BigFloat -> new(
$y
) -> as_int();
my
$z
=
$yint
< 0 ?
$xint
-> bmul(Math::BigInt -> new(2) -> bpow(-
$yint
))
:
$xint
-> bdiv(Math::BigInt -> new(2) -> bpow(
$yint
));
$z
=
$z
-> bint();
$z
=
$z
-> is_nan() ?
"NaN"
:
$z
-> is_inf(
"+"
) ?
"inf"
:
$z
-> is_inf(
"-"
) ?
"-inf"
:
$z
-> numify();
push
@cases
, [
$x
,
$y
,
$z
];
}
}
}
for
my
$upg
(
undef
,
"Math::BigRat"
) {
for
my
$dng
(
undef
,
"Math::BigInt"
) {
Math::BigInt -> upgrade(
$upg
);
Math::BigRat -> downgrade(
$dng
);
for
my
$case
(
@cases
) {
my
(
$xscl
,
$yscl
,
$zscl
) =
@$case
;
my
@xref
= (
'Math::BigRat'
);
my
@yref
= (
'Math::BigRat'
,
''
);
unshift
@xref
,
'Math::BigInt'
unless
$xscl
=~ /\./;
unshift
@yref
,
'Math::BigInt'
unless
$yscl
=~ /\./;
for
my
$xref
(
@xref
) {
for
my
$yref
(
@yref
) {
my
$zref
=
$dng
?
"Math::BigInt"
:
$xref
;
note
"\n"
;
note
"Math::BigInt -> upgrade("
,
defined
(
$upg
) ?
"\"$upg\""
:
"undef"
,
");"
,
" Math::BigRat -> downgrade("
,
defined
(
$dng
) ?
"\"$dng\""
:
"undef"
,
");"
,
$xref
?
" \$x = $xref -> new(\"$xscl\");"
:
" \$x = $xscl;"
,
$yref
?
" \$y = $yref -> new(\"$yscl\");"
:
" \$y = $yscl;"
,
" \$z = \$x -> bbrsft(\$y);"
,
" print \$z\n"
;
note
"\n"
;
{
my
$x
=
$xref
?
$xref
-> new(
$xscl
) :
$xscl
;
my
$y
=
$yref
?
$yref
-> new(
$yscl
) :
$yscl
;
my
$z
=
eval
{
$x
-> bbrsft(
$y
) };
is($@,
''
,
"eval succeeded"
);
is(
ref
(
$z
),
$zref
,
"output class is $zref"
);
is(
$z
,
$zscl
,
"output value is $zscl"
);
is(
$x
,
$z
,
"invocand value $z is the output"
);
}
note
"\n"
;
note
"Math::BigInt -> upgrade("
,
defined
(
$upg
) ?
"\"$upg\""
:
"undef"
,
");"
,
" Math::BigRat -> downgrade("
,
defined
(
$dng
) ?
"\"$dng\""
:
"undef"
,
");"
,
$xref
?
" \$x = $xref -> new(\"$xscl\");"
:
" \$x = $xscl;"
,
$yref
?
" \$y = $yref -> new(\"$yscl\");"
:
" \$y = $yscl;"
,
" \$z = \$x >>= \$y;"
,
" print \$z\n"
;
note
"\n"
;
{
my
$x
=
$xref
?
$xref
-> new(
$xscl
) :
$xscl
;
my
$y
=
$yref
?
$yref
-> new(
$yscl
) :
$yscl
;
my
$z
=
eval
{
$x
>>=
$y
};
is($@,
''
,
"eval succeeded"
);
is(
ref
(
$z
),
$zref
,
"output class is $zref"
);
is(
$z
,
$zscl
,
"output value is $zscl"
);
is(
$x
,
$z
,
"invocand value $z is the output"
);
}
note
"\n"
;
note
"Math::BigInt -> upgrade("
,
defined
(
$upg
) ?
"\"$upg\""
:
"undef"
,
");"
,
" Math::BigRat -> downgrade("
,
defined
(
$dng
) ?
"\"$dng\""
:
"undef"
,
");"
,
$xref
?
" \$x = $xref -> new(\"$xscl\");"
:
" \$x = $xscl;"
,
$yref
?
" \$y = $yref -> new(\"$yscl\");"
:
" \$y = $yscl;"
,
" \$z = \$x >> \$y;"
,
" print \$z\n"
;
note
"\n"
;
{
my
$x
=
$xref
?
$xref
-> new(
$xscl
) :
$xscl
;
my
$y
=
$yref
?
$yref
-> new(
$yscl
) :
$yscl
;
my
$z
=
eval
{
$x
>>
$y
};
is($@,
''
,
"eval succeeded"
);
is(
ref
(
$z
),
$zref
,
"output class is $zref"
);
is(
$z
,
$zscl
,
"output value is $zscl"
);
if
(
$xscl
==
$xscl
) {
cmp_ok(
$x
,
"=="
,
$xscl
,
"invocand value $xscl is unmodified"
);
}
else
{
is(
$x
,
$xscl
,
"invocand value $xscl is unmodified"
);
}
}
}
}
}
}
}
for
my
$upg
(
undef
,
"Math::BigRat"
) {
for
my
$dng
(
undef
,
"Math::BigInt"
) {
Math::BigInt -> upgrade(
$upg
);
Math::BigRat -> downgrade(
$dng
);
for
my
$ref
(
"Math::BigInt"
,
"Math::BigRat"
) {
for
my
$case
(
@cases
) {
my
(
$xscl
,
$yscl
,
$zscl
) =
@$case
;
my
@xref
= (
'Math::BigRat'
,
''
);
my
@yref
= (
'Math::BigRat'
,
''
);
unshift
@xref
,
'Math::BigInt'
unless
$xscl
=~ /\./;
unshift
@yref
,
'Math::BigInt'
unless
$yscl
=~ /\./;
for
my
$xref
(
@xref
) {
for
my
$yref
(
@yref
) {
my
$zref
=
$dng
?
"Math::BigInt"
:
$ref
;
my
$x
=
$xref
?
$xref
-> new(
$xscl
) :
$xscl
;
my
$y
=
$yref
?
$yref
-> new(
$yscl
) :
$yscl
;
note
"\n"
;
note
"Math::BigInt -> upgrade("
,
defined
(
$upg
) ?
"\"$upg\""
:
"undef"
,
");"
,
" Math::BigRat -> downgrade("
,
defined
(
$dng
) ?
"\"$dng\""
:
"undef"
,
");"
,
$xref
?
" \$x = $xref -> new(\"$xscl\");"
:
" \$x = \"$xscl\";"
,
$yref
?
" \$y = $yref -> new(\"$yscl\");"
:
" \$y = \"$yscl\";"
,
" \$z = $ref -> bbrsft(\$x, \$y);"
,
" print \$z\n"
;
note
"\n"
;
my
$z
=
eval
{
$ref
-> bbrsft(
$x
,
$y
) };
is($@,
''
,
"eval succeeded"
);
is(
ref
(
$z
),
$zref
,
"output class is $zref"
);
is(
$z
,
$zscl
,
"output value is $zscl"
);
if
(
$ref
eq
"Math::BigInt"
&&
$xref
eq
"Math::BigInt"
||
$ref
eq
"Math::BigRat"
&&
$xref
eq
"Math::BigRat"
&& !
$dng
)
{
is(
$x
,
$z
,
"invocand is the output (value is $zscl)"
);
}
}
}
}
}
}
}
done_testing();