The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
#!/usr/bin/perl -w

BEGIN { unshift @INC, './lib', '../lib';
    require Config; import Config;
}

$test = 0;
$| = 1;
print "1..",&last,"\n";

sub test {
  $test++; if (shift) {print "ok $test\n";1} else {print "not ok $test\n";0}
}

use Math::Pari;

test(1);			# 1

$a=new Math::Pari 1;

test($a);			# 2
test(ref $a eq "Math::Pari");	# 3

{
  package Math::Pari;
  ::test(defined &pari2nv); # 4
}

test(defined &Math::Pari::pari2nv); # 5
test(Math::Pari::pari2iv($a)==1); # 6

# This looks like a bug in the Pari conversion to double:

test(abs(Math::Pari::pari2nv($a)-1.0)<2**-49); # 7
#####print "`", Math::Pari::pari2nv($a)-1.0, "'\n";

$add = Math::Pari::loadPari("gadd");

test($add);			# 8
test("$add" =~ /^CODE/);	# 9

$b= &$add (34,67);

test($b);			# 10
test(ref $b eq "Math::Pari");	# 11
test(Math::Pari::pari2iv($b)==101); # 12

$b=Math::Pari::gadd(11,12);	# Uninitialized value here

test($b);			# 13
test(ref $b eq "Math::Pari");	# 14
test(Math::Pari::pari2iv($b)==23); # 15

$c = Math::Pari::gsub(17,93);

test($c);			# 16
test(ref $c eq "Math::Pari");	# 17
test(Math::Pari::pari2iv($c)==-76); # 18

$c = do { package Math::Pari; gsub(19,93) }; # And here (No MORE :-)

test($c);			# 19
test(ref $c eq "Math::Pari");	# 20
test(Math::Pari::pari2iv($c)==-74); # 21

$d=Math::Pari::gadd($b,$c);

test(ref $d eq "Math::Pari");	# 22
test(Math::Pari::pari2iv($d)==-51); # 23

$e=new Math::Pari "matrix(2,2,j,k,j+k)";

test(ref $e eq "Math::Pari");	# 24

$cc=Math::Pari::pari2iv($c);

test(! ref $cc);		# 25

$ee=Math::Pari::pari2pv($e);

test(! ref $ee) || print ref $ee;		# 26

test($ee eq "[2,3;3,4]"); # 27

#$f=Math::Pari::matinvr($e);
$f= $e ** -1;

test(ref $f eq "Math::Pari");	# 28
test(Math::Pari::pari2pv($f) eq "[-4,3;3,-2]"); # 29

# Overload

test("$f" eq "[-4,3;3,-2]") || print "$f\n";	# 30

$f+=1;

test("$f" eq "[-3,3;3,-1]") || print "$f\n";	# 31
test($f == "[-3,3;3,-1]");	# 32

$g=(new Math::Pari "[1,2;3,2]");
$gg=Math::Pari::gpui($g,-1);

test($gg);			# 33
$g=(new Math::Pari "[1,2;3,2]")**-1;

test($g == $gg);		# 34

# Should work eventually

test("$g" eq "[-1/2,1/2;3/4,-1/4]") || print "$g\n"; # 35
#test(1);

test($g == "[-1/2,1/2;3/4,-1/4]"); # 36
#test(1);

$f  = new Math::Pari "[2,3;3,1]";
$ff = PARImat [[2,3],[3,1]];

test(Math::Pari::geq($f,$ff));	# 37
test($f==$ff);			# 38
test(Math::Pari::gne($f,$ff)==0);	# 39
test(!($f!=$ff));		# 40
test("$f" eq "$ff");		# 41
test($f eq $ff);		# 42

$f  = new Math::Pari "[2,3;3,2]";

test($f != $ff);		# 43
test(!($f == $ff));		# 44
test("$f" ne "$ff");		# 45
test($f ne $ff);		# 46

$f = new Math::Pari "[2,3;2,1]";
$fff=$f**2;

test($fff eq "[10,9;6,7]");	# 47
test($fff eq $f*$f);		# 48

test(PARI(3)<4);		# 49
test(3<PARI(4));		# 50
test(!(PARI(3)<2));		# 51
test(!(4<PARI(3)));		# 52

test(PARI(3) lt 4);		# 53
test(3 lt PARI(4));		# 54
test(!(PARI(3) lt 2));		# 55
test(!(4 lt PARI(3)));		# 56

eval <<'EOE';
use Math::Pari qw(I Pi);

#sub Math::Pari::i;
#sub Math::Pari::pi;

# Bug in Pari: depends on the order of execution

# Stupid test, probably version- or machine-specific

test(abs(exp((Pi()+7e-29)*I)+1) >= 1e-29); # 57
test(abs(exp(Pi()*I)+1) <= 1e-28); # 58

test(('x'*I)**2 == "-x^2");	# 59
EOE
;

if ( ($tot, $now, $onstack, $offstack) = Math::Pari::memUsage) {
  print "# total = $tot, now = $now, onstack = $onstack, offstack = $offstack\n";
}

sub incr1 {
  warn "incr1: ", scalar @_, " args" unless @_ == 6 or @_ == 1;	# no proto ==> 6 ???
  my $y = shift;
  print "# Adding $y to $x\n";
  $x += $y;
  #warn "Got $y, Sum $x\n";
}

Math::Pari::installPerlFunction(\&incr1, "incrJ");

$x = 0;
Math::Pari::fordiv(28, 'j', 'incrJ(j)');

test($x == 56) or print "$x\n";	# 60

$x = 0;
Math::Pari::fordiv(28, 'j', 'incr1(j)'); # autoloading Perl->Pari

test($x == 56);			# 61

$x = 0;
$j = PARI 'j';
Math::Pari::fordiv(28, $j, 'incr1(j)'); # autoloading Perl->Pari

test($x == 56);			# 62

$x = 0;
Math::Pari::fordiv(28, 'j', sub { incr1(PARI 'j') } ); # Perl code ->Pari expr

test($x == 56);			# 63
print "# x = '$x'\n" if $x != 56;

$x = 0;
$j = PARIvar 'j';
Math::Pari::fordiv(28, 'j', sub { incr1($j) } ); # Perl code ->Pari expr

test($x == 56);			# 64
print "# x = '$x'\n" if $x != 56;

$x = 0;
$j = PARI 'j';
Math::Pari::fordiv(28, $j, sub { incr1($j) } ); # Perl code ->Pari expr

test($x == 56);			# 65
print "# x = '$x'\n" if $x != 56;

$x = 0;
$j = 3.4;
Math::Pari::fordiv(28, $j, sub { incr1($j) } ); # Perl code ->Pari expr

test($x == 56);			# 66
print "# x = '$x'\n" if $x != 56;

$x = 0;
$j = PARI 'x+O(x^7)';
Math::Pari::fordiv(28, $j, sub { incr1($j) } ); # Perl code ->Pari expr

test($x == 56);			# 67
print "# x = '$x'\n" if $x != 56;

#use Math::Pari 'pi';

$x = 0;
$j = PARI 3.14;
Math::Pari::fordiv(28, $j, sub { incr1($j) } ); # Perl code ->Pari expr

test($x == 56);			# 68
print "# x = '$x'\n" if $x != 56;

undef $x;
eval { $x  = Pi()/0; } ;	# Needs G_KEEPERR patch!
#chomp($@), print "# '$@'\n" if $@;
print "# x = '$x'\n" if defined $x;

test( $@ =~ /^PARI:\s+\*+\s+division\s+by\s+zero\b/i ) or print "$@\n"; # 69

*O=\&Math::Pari::O;

$x = PARI 'x';
$y = 1+O($x,6);
test("$y" eq '1+O(x^6)');	# 70

$x = PARI 'x';
$y = 1+O($x**6);
test("$y" eq '1+O(x^6)');	# 71

$y = 1+O(5,6);
test("$y" eq '1+O(5^6)') or print "$y\n";	# 72

sub counter { $i += shift; }
$i = 145;
PARI 'k=5' ;
Math::Pari::fordiv(28, 'j', 'k=k+counter(j)');
test(PARI('k') == 984);		# 73

sub get2 ($$) {7}
test(PARI('get2(3,-19)==7'));	# 74

$res = 9;
eval { $res = PARI('get2(3)') };
#print "# res='$res' err=`$@'\n";
test($res == 9 and $@ =~ /expected\scharacter:\s\',/); # 75

$res = 9;
eval { $res = PARI('get2(3,1,67)') };
#print "# res='$res' err=`$@'\n";
test($res == 9 and $@ =~ /expected\scharacter:\s\'\)/); # 76

$res = Math::Pari::sum($l,5,9,sub{'kk'**$l},7);
#print "res = `$res'\n";
test("$res" eq 'kk^9+kk^8+kk^7+kk^6+kk^5+7') or print "$res\n";	# 77

$var = PARI 'var';
$var1 = PARIvar 'var';

Math::Pari::changevalue('var', 13);
test(PARI('var') eq '13');	# 78

PARI('var=12');
test(PARI('var') eq '12');	# 79

Math::Pari::changevalue($var1, 15);
test(PARI('var') eq '15');	# 80
test($var1 eq '15');		# 81

Math::Pari::changevalue($var, 51);
test(PARI('var') eq '51');	# 82

my $v = PARI [2,7,11];
$v->[1] = 17;
test("$v" eq "[2,17,11]");	# 83

my $mat = PARImat [[1,2,3],[4,8,5]];
$mat->[1][1] = 18;
test("$mat" eq "[1,4;2,18;3,5]"); # 84

$mat->[0] = [56,67,78];
test("$mat" eq "[56,4;67,18;78,5]"); # 85

eval {$mat->[0] = [56,67]};
test($@ =~ /Assignment of a columns into a matrix of incompatible height/); # 86

eval {$mat->[0] = 56};
test($@ =~ /Not a vector where column of a matrix expected/); # 87

$mat = PARImat [[1,2,3]];
eval {$mat->[0] = [56,67]};
test("$mat" eq "[56;67]");	# 88

if (Math::Pari::pari_version_exp >= 2000018) {
  $vi = PARI 6;
  $v  = PARI 5;
  test(($vi & $v) eq 4);	# 89
  test(($vi | $v) eq 7);	# 90
  test(($vi ^ $v) eq 3);	# 91
  test((~$vi) eq -7);		# 92
} else {
  for (1..4) {
    test(1 eq 1);
  }
}

{
  my $eleven = PARI 11;
  $eleven++;
  test($eleven eq 12);		# 93
  test("$eleven" eq "12");	# 94
}

if ($] < 5.007) {
  test 1;			# 95
} else {
  my $x = PARI '10000000000000000000000000000000000000000011.5';

  test(int($x) eq PARI '10000000000000000000000000000000000000000011');	# 95
}

if (Math::Pari::have_highlevel()) {
  test 't_COL' eq Math::Pari::pari2pv(Math::Pari::type(PARIcol [1,2,3]));	# 96
} else {
  test 1;
}
test 't_COL' eq Math::Pari::type_name(PARIcol [1,2,3]);	# 97
test 4 == Math::Pari::lg(PARIcol [1,2,3]); # 98
test 3 == Math::Pari::lgef(PARI 1); # 99
test 17 == Math::Pari::longword(-PARI(17),2); # 100
test not defined eval { Math::Pari::longword((-PARI(17)),3); 1}; # 101
test not defined eval { Math::Pari::longword((-PARI(17)),-3); 1}; # 102

test 35906707  == (  (PARI(3)**60)  >> 70); # 103
test -35906708 == ((-(PARI(3)**60)) >> 70); # 104
test  4794036  == ( (1 . (0 x 100)) >> PARI(310)); # 105
test -4794037  == ((-1 . (0 x 100)) >> PARI(310)); # 106
test PARI(6)**100 == (PARI(3)**100 << 100); # 107
my $pow3_50 = PARI(3)**50;
test PARI(6)**50  == ("$pow3_50" << PARI 50); # 108

# This causes spurious errors on 5.6.0
if ( $] == 5.006 ) {
    $test++;
    print "ok $test # Skipped: bug in Perl 5.6.0\n";
} else {
  test( (2**(PARI(4)/3)) =~ /^2\.5198420997897463295344212/ );	# 109
}
if (do { local $SIG{__WARN__} = sub {};
	 eval 'local $^W = 0; 0b1001 == 9'}) {
  eval <<'EOE';
   {
     my @res = (11, PARI '6099284589125821621535');	# 330*2^64 + 42131*2^48 + 51818*2^32 + 19189*2^16 + 25887
     use Math::Pari ':hex';
     # printf "# b-hex => %s, expecting %s\n", 0b1011, $res[0];
     test 0b1011 == $res[0];			# 110
     test 0b1010010101010010010010011110010100110101001001010111101010110010100011111
       == 0b00000001010010101010010010010011110010100110101001001010111101010110010100011111; # 111
     # printf "# b-hex => %s, hex => %s, expecting %s\n",
     #  0b1010010101010010010010011110010100110101001001010111101010110010100011111,
     #  0x014aa493ca6a4af5651f, '6099284589125821621535';
     test 0b1010010101010010010010011110010100110101001001010111101010110010100011111
       == 0x014aa493ca6a4af5651f;		# 112
     test 0b1010010101010010010010011110010100110101001001010111101010110010100011111
       == $res[1];				# 113
   }
EOE
} else {
  for (1..4) {
    $test++;
    print "ok $test # Skipped: no support for binary literals\n";
  }
}

my $lift = PARI(1)->_can('lift');
test  $lift;					# 114
test !PARI(1)->_can('raft');			# 115
test  PARI(1)->_can('O');			# 116

test &$lift(Math::Pari::Mod(67,17)) == 16;	# 117

{
  package ttt;
  @ISA = 'Math::Pari';
  sub new { shift; bless ::PARI(shift) }	# Not complete...
}
my $denom = 'ttt'->new(1)->can('denominator');
my $lift1 = 'ttt'->new(1)->can('lift');

test  'ttt' eq ref 'ttt'->new(1);			# 118
test  $lift1;					# 119
test  $denom;					# 120

test  'ttt' eq ref 'ttt'->new(Math::Pari::Mod(67,17));	# 121
test &$lift1('ttt'->new(Math::Pari::Mod(67,17))) == 16;	# 122
# my $d = &$denom('ttt'->new(4)/68);
# print "# d=$d\n";
test &$denom('ttt'->new(4)/78) == 39;		# 123
test 'ttt'->new(4)->can('can');			# 124
test &{ 'ttt'->new(4)->can('isa') }( 'ttt'->new(6), 'Math::Pari') == 1;	# 125
test 'ttt'->new(4)->can('isa');			# 126
test !'ttt'->new(4)->can('foobar');		# 127

my $try = 0.1;
my $try1 = abs($try);

test 0.1 == PARI $try;		# 128

eval <<'EOE' or warn $@;	# 129..131
  use Math::Pari ':int', ':float';
  my $i = PARI 12345678901234567890123;
  test "$i" eq '12345678901234567890123';
  $i = 12345678901234567890123;
  test "$i" eq '12345678901234567890123';
  $i = 2**10000;
  my $ii = PARI(2)**PARI(10000);
  test "$i" eq "$ii";
EOE

my $ff = Math::Pari::factor(24);
test "[2,3;3,1]" eq "$ff";	# 132

parse_as_gp <<'EOP';

add1(x) = x + 1
  \\ a comment in between
add_1(x) = x + 1  \\ with comment

add___1(x) = x++
int_div(a,b) = a \ b

str_ex() = "foo \" bar"

add__1(x) = 
  x + \\ broken when possible
    1
  \\ nothing more
EOP

test PARI("add1(3)") == 4;	# 133
test PARI("add_1(3)") == 4;	# 134
test PARI("add__1(3)") == 4;	# 135
test parse_as_gp("yt=17\nadd___1(yt)+yt") == 35;	# 136

*$_ = Math::Pari::__wrap_PARI_macro($_) and test 1 for qw(add1 add___1 str_ex int_div); # 137..140
test add1(35) == 36;		# 141
my $xT = 44;
test add___1($xT) == 45;	# 142
test $xT == 44;			# 143 (increment affects a local-to-function copy???)
test str_ex() eq '"foo \" bar"'; # 144
test int_div(13,3) == 4;	# 145
#warn "str_ex=<<", str_ex(), ">>\n";	# <<foo " bar>>

for my $l (0..22) {		# + 23*9*2
  for my $d (1..9, -9..-1) {
    my $n = $d . (0 x $l);
    my $p = eval "PARI($n)";
    my $pn = sprintf '%.0f', $n;
    my $op = $p = "$p";
    $p =~ s/\.0*$//;
    print "# `$n' ==> Perl `$pn' ==> Math::Pari `$op'\n" if $n ne $p;
    test ($n eq $p or $p =~ /(0{15}|9{15})\d{0,3}e\d\d$/i);
  }
}

my $ow;
BEGIN {
  $ow = $^W;
  $^W = 1;
}

# check warnings with double import
package xx;
use Math::Pari 'centerlift';
package yy;
use Math::Pari 'centerlift';

package main;

BEGIN {
  $^W = $ow;
}

sub last {145 + 23*9*2}