my
$config_file
=
'xt/author/lib.ini'
;
my
$config
= Config::Tiny ->
read
(
'xt/author/lib.ini'
)
or
die
Config::Tiny -> errstr();
our
$LIB
=
$config
->{_}->{lib};
die
"No library defined in file '$config_file'"
unless
defined
$LIB
;
die
"Invalid library name '$LIB' in file '$config_file'"
unless
$LIB
=~ /^[A-Za-z]\w*(::\w+)*\z/;
our
$REF
=
$config
->{_}->{
ref
};
die
"No reference type defined in file '$config_file'"
unless
defined
$REF
;
die
"Invalid reference type '$REF' in file '$config_file'"
unless
$REF
=~ /^[A-Za-z]\w*(::\w+)*\z/;
eval
"require $LIB"
;
die
$@
if
$@;
can_ok(
$LIB
,
'_ilog2'
);
sub
ilog2 {
my
$x
=
shift
;
my
$y
=
int
(
log
(
$x
) /
log
(2));
my
$trial
= 2 **
$y
;
return
$y
if
$trial
==
$x
;
while
(
$trial
<
$x
) {
$y
++;
$trial
= 2 **
$y
;
}
while
(
$trial
>
$x
) {
$y
--;
$trial
= 2 **
$y
;
}
return
$y
;
}
my
@data
;
for
(
my
$x
= 1 ;
$x
<= 1022 ; ++
$x
) {
my
$y
= ilog2(
$x
);
my
$status
= 2 **
$y
==
$x
? 1 : 0;
push
@data
, [
$x
,
$y
,
$status
];
}
my
$b
=
$LIB
-> _new(2);
for
(
my
$y
= 10 ;
$y
<= 100 ;
$y
++) {
my
$x
=
$LIB
-> _pow(
$LIB
-> _copy(
$b
),
$LIB
-> _new(
$y
));
my
$x_up
=
$LIB
-> _inc(
$LIB
-> _copy(
$x
));
my
$x_dn
=
$LIB
-> _dec(
$LIB
-> _copy(
$x
));
push
@data
, [
$LIB
-> _str(
$x_dn
),
$y
- 1, 0 ];
push
@data
, [
$LIB
-> _str(
$x
),
$y
, 1 ];
push
@data
, [
$LIB
-> _str(
$x_up
),
$y
, 0 ];
}
for
(
my
$i
= 0 ;
$i
<=
$#data
; ++
$i
) {
my
(
$in0
,
$out0
) = @{
$data
[
$i
] };
my
(
$x
,
$y
,
$got
);
my
$test
=
qq|\$x = $LIB->_new("$in0"); |
.
qq|\$got = $LIB->_ilog2(\$x);|
;
diag(
"\n$test\n\n"
)
if
$ENV
{AUTHOR_DEBUGGING};
eval
$test
;
is($@,
""
,
"'$test' gives emtpy \$\@"
);
subtest
"_ilog2() in list context: $test"
,
sub
{
unless
(
defined
$out0
) {
plan
tests
=> 1;
is(
$got
,
$out0
,
"'$test' output arg has the right value"
);
return
;
}
plan
tests
=> 5;
is(
ref
(
$got
),
$REF
,
"'$test' output arg is a $REF"
);
is(
$LIB
->_check(
$got
), 0,
"'$test' output is valid"
);
is(
$LIB
->_str(
$got
),
$out0
,
"'$test' output arg has the right value"
);
is(
ref
(
$x
),
$REF
,
"'$test' input arg is still a $REF"
);
ok(
$LIB
->_str(
$x
) eq
$out0
||
$LIB
->_str(
$x
) eq
$in0
,
"'$test' input arg has the correct value"
);
};
}
for
(
my
$i
= 0 ;
$i
<=
$#data
; ++
$i
) {
my
(
$in0
,
$out0
,
$out1
) = @{
$data
[
$i
] };
my
(
$x
,
$y
,
@got
);
my
$test
=
qq|\$x = $LIB->_new("$in0"); |
.
qq|\@got = $LIB->_ilog2(\$x);|
;
diag(
"\n$test\n\n"
)
if
$ENV
{AUTHOR_DEBUGGING};
eval
$test
;
is($@,
""
,
"'$test' gives emtpy \$\@"
);
subtest
"_ilog2() in list context: $test"
,
sub
{
unless
(
defined
$out0
) {
plan
tests
=> 1;
is(
$got
[0],
$out0
,
"'$test' output arg has the right value"
);
return
;
}
plan
tests
=> 8;
cmp_ok(
scalar
(
@got
),
'=='
, 2,
"'$test' gives two output args"
);
is(
ref
(
$got
[0]),
$REF
,
"'$test' first output arg is a $REF"
);
is(
$LIB
->_check(
$got
[0]), 0,
"'$test' first output is valid"
);
is(
$LIB
->_str(
$got
[0]),
$out0
,
"'$test' output arg has the right value"
);
is(
ref
(
$x
),
$REF
,
"'$test' first input arg is still a $REF"
);
ok(
$LIB
->_str(
$x
) eq
$out0
||
$LIB
->_str(
$x
) eq
$in0
,
"'$test' first input arg has the correct value"
);
is(
ref
(
$got
[1]),
""
,
"'$test' second output arg is a scalar"
);
is(
$got
[1],
$out1
,
"'$test' second output arg has the right value"
);
};
}