| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737 | #!/usr/bin/env perl# Copyright (C) 1999 Free Software Foundation, Inc.# This file is part of the GNU C Library.# Contributed by Andreas Jaeger <aj@suse.de>, 1999.# The GNU C Library is free software; you can redistribute it and/or# modify it under the terms of the GNU Lesser General Public# License as published by the Free Software Foundation; either# version 2.1 of the License, or (at your option) any later version.# The GNU C Library is distributed in the hope that it will be useful,# but WITHOUT ANY WARRANTY; without even the implied warranty of# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU# Lesser General Public License for more details.# You should have received a copy of the GNU Lesser General Public# License along with the GNU C Library; see the file COPYING.LIB.  If# not, see <http://www.gnu.org/licenses/>.# This file needs to be tidied up# Note that functions and tests share the same namespace.# Information about tests are stored in: %results# $results{$test}{"kind"} is either "fct" or "test" and flags whether this# is a maximal error of a function or a single test.# $results{$test}{"type"} is the result type, e.g. normal or complex.# $results{$test}{"has_ulps"} is set if deltas exist.# $results{$test}{"has_fails"} is set if exptected failures exist.# In the following description $type and $float are:# - $type is either "normal", "real" (for the real part of a complex number)#   or "imag" (for the imaginary part # of a complex number).# - $float is either of float, ifloat, double, idouble, ldouble, ildouble;#   It represents the underlying floating point type (float, double or long#   double) and if inline functions (the leading i stands for inline)#   are used.# $results{$test}{$type}{"fail"}{$float} is defined and has a 1 if# the test is expected to fail# $results{$test}{$type}{"ulp"}{$float} is defined and has a delta as valueuse Getopt::Std;use strict;use vars qw ($input $output);use vars qw (%results);use vars qw (@tests @functions);use vars qw ($count);use vars qw (%beautify @all_floats);use vars qw ($output_dir $ulps_file);# all_floats is sorted and contains all recognised float types@all_floats = ('double', 'float', 'idouble',	       'ifloat', 'ildouble', 'ldouble');%beautify =  ( "minus_zero" => "-0",    "plus_zero" => "+0",    "minus_infty" => "-inf",    "plus_infty" => "inf",    "nan_value" => "NaN",    "M_El" => "e",    "M_E2l" => "e^2",    "M_E3l" => "e^3",    "M_LOG10El", "log10(e)",    "M_PIl" => "pi",    "M_PI_34l" => "3/4 pi",    "M_PI_2l" => "pi/2",    "M_PI_4l" => "pi/4",    "M_PI_6l" => "pi/6",    "M_PI_34_LOG10El" => "3/4 pi*log10(e)",    "M_PI_LOG10El" => "pi*log10(e)",    "M_PI2_LOG10El" => "pi/2*log10(e)",    "M_PI4_LOG10El" => "pi/4*log10(e)",    "M_LOG_SQRT_PIl" => "log(sqrt(pi))",    "M_LOG_2_SQRT_PIl" => "log(2*sqrt(pi))",    "M_2_SQRT_PIl" => "2 sqrt (pi)",    "M_SQRT_PIl" => "sqrt (pi)",    "INVALID_EXCEPTION" => "invalid exception",    "DIVIDE_BY_ZERO_EXCEPTION" => "division by zero exception",    "INVALID_EXCEPTION_OK" => "invalid exception allowed",    "DIVIDE_BY_ZERO_EXCEPTION_OK" => "division by zero exception allowed",    "EXCEPTIONS_OK" => "exceptions allowed",    "IGNORE_ZERO_INF_SIGN" => "sign of zero/inf not specified","INVALID_EXCEPTION|IGNORE_ZERO_INF_SIGN" => "invalid exception and sign of zero/inf not specified"  );# get Options# Options:# u: ulps-file# h: help# o: output-directory# n: generate new ulps fileuse vars qw($opt_u $opt_h $opt_o $opt_n);getopts('u:o:nh');$ulps_file = 'libm-test-ulps';$output_dir = '';if ($opt_h) {  print "Usage: gen-libm-test.pl [OPTIONS]\n";  print " -h         print this help, then exit\n";  print " -o DIR     directory where generated files will be placed\n";  print " -n         only generate sorted file NewUlps from libm-test-ulps\n";  print " -u FILE    input file with ulps\n";  exit 0;}$ulps_file = $opt_u if ($opt_u);$output_dir = $opt_o if ($opt_o);$input = "libm-test.inc";$output = "${output_dir}libm-test.c";$count = 0;&parse_ulps ($ulps_file);&generate_testfile ($input, $output) unless ($opt_n);&output_ulps ("${output_dir}libm-test-ulps.h", $ulps_file) unless ($opt_n);&print_ulps_file ("${output_dir}NewUlps") if ($opt_n);# Return a nicer representationsub beautify {  my ($arg) = @_;  my ($tmp);  if (exists $beautify{$arg}) {    return $beautify{$arg};  }  if ($arg =~ /^-/) {    $tmp = $arg;    $tmp =~ s/^-//;    if (exists $beautify{$tmp}) {      return '-' . $beautify{$tmp};    }  }  if ($arg =~ /[0-9]L$/) {    $arg =~ s/L$//;  }  return $arg;}# Return a nicer representation of a complex numbersub build_complex_beautify {  my ($r, $i) = @_;  my ($str1, $str2);  $str1 = &beautify ($r);  $str2 = &beautify ($i);  if ($str2 =~ /^-/) {    $str2 =~ s/^-//;    $str1 .= ' - ' . $str2;  } else {    $str1 .= ' + ' . $str2;  }  $str1 .= ' i';  return $str1;}# Return name of a variablesub get_variable {  my ($number) = @_;  return "x" if ($number == 1);  return "y" if ($number == 2);  return "z" if ($number == 3);  # return x1,x2,...  $number =-3;  return "x$number";}# Add a new test to internal data structures and fill in the# ulps, failures and exception information for the C line.sub new_test {  my ($test, $exception) = @_;  my $rest;  # Add ulp, xfail  if (exists $results{$test}{'has_ulps'}) {    $rest = ", DELTA$count";  } else {    $rest = ', 0';  }  if (exists $results{$test}{'has_fails'}) {    $rest .= ", FAIL$count";  } else {    $rest .= ', 0';  }  if (defined $exception) {    $rest .= ", $exception";  } else {    $rest .= ', 0';  }  $rest .= ");\n";  # We must increment here to keep @tests and count in sync  push @tests, $test;  ++$count;  return $rest;}# Treat some functions especially.# Currently only sincos needs extra treatment.sub special_functions {  my ($file, $args) = @_;  my (@args, $str, $test, $cline);  @args = split /,\s*/, $args;  unless ($args[0] =~ /sincos/) {    die ("Don't know how to handle $args[0] extra.");  }  print $file "  FUNC (sincos) ($args[1], &sin_res, &cos_res);\n";  $str = 'sincos (' . &beautify ($args[1]) . ', &sin_res, &cos_res)';  # handle sin  $test = $str . ' puts ' . &beautify ($args[2]) . ' in sin_res';  if ($#args == 4) {    $test .= " plus " . &beautify ($args[4]);  }  $cline = "  check_float (\"$test\", sin_res, $args[2]";  $cline .= &new_test ($test, $args[4]);  print $file $cline;  # handle cos  $test = $str . ' puts ' . &beautify ($args[3]) . ' in cos_res';  $cline = "  check_float (\"$test\", cos_res, $args[3]";  # only tests once for exception  $cline .= &new_test ($test, undef);  print $file $cline;}# Parse the arguments to TEST_x_ysub parse_args {  my ($file, $descr, $args) = @_;  my (@args, $str, $descr_args, $descr_res, @descr);  my ($current_arg, $cline, $i);  my ($pre, $post, @special);  my ($extra_var, $call, $c_call);  if ($descr eq 'extra') {    &special_functions ($file, $args);    return;  }  ($descr_args, $descr_res) = split /_/,$descr, 2;  @args = split /,\s*/, $args;  $call = "$args[0] (";  # Generate first the string that's shown to the user  $current_arg = 1;  $extra_var = 0;  @descr = split //,$descr_args;  for ($i = 0; $i <= $#descr; $i++) {    if ($i >= 1) {      $call .= ', ';    }    # FLOAT, int, long int, long long int    if ($descr[$i] =~ /f|i|l|L/) {      $call .= &beautify ($args[$current_arg]);      ++$current_arg;      next;    }    # &FLOAT, &int - argument is added here    if ($descr[$i] =~ /F|I/) {      ++$extra_var;      $call .= '&' . &get_variable ($extra_var);      next;    }    # complex    if ($descr[$i] eq 'c') {      $call .= &build_complex_beautify ($args[$current_arg], $args[$current_arg+1]);      $current_arg += 2;      next;    }    die ("$descr[$i] is unknown");  }  $call .= ')';  $str = "$call == ";  # Result  @descr = split //,$descr_res;  foreach (@descr) {    if ($_ =~ /f|i|l|L/) {      $str .= &beautify ($args[$current_arg]);      ++$current_arg;    } elsif ($_ eq 'c') {      $str .= &build_complex_beautify ($args[$current_arg], $args[$current_arg+1]);      $current_arg += 2;    } elsif ($_ eq 'b') {      # boolean      $str .= ($args[$current_arg] == 0) ? "false" : "true";      ++$current_arg;    } elsif ($_ eq '1') {      ++$current_arg;    } else {      die ("$_ is unknown");    }  }  # consistency check  if ($current_arg == $#args) {    die ("wrong number of arguments")      unless ($args[$current_arg] =~ /EXCEPTION|IGNORE_ZERO_INF_SIGN/);  } elsif ($current_arg < $#args) {    die ("wrong number of arguments");  } elsif ($current_arg > ($#args+1)) {    die ("wrong number of arguments");  }  # check for exceptions  if ($current_arg <= $#args) {    $str .= " plus " . &beautify ($args[$current_arg]);  }  # Put the C program line together  # Reset some variables to start again  $current_arg = 1;  $extra_var = 0;  if (substr($descr_res,0,1) eq 'f') {    $cline = 'check_float'  } elsif (substr($descr_res,0,1) eq 'b') {    $cline = 'check_bool';  } elsif (substr($descr_res,0,1) eq 'c') {    $cline = 'check_complex';  } elsif (substr($descr_res,0,1) eq 'i') {    $cline = 'check_int';  } elsif (substr($descr_res,0,1) eq 'l') {    $cline = 'check_long';  } elsif (substr($descr_res,0,1) eq 'L') {    $cline = 'check_longlong';  }  # Special handling for some macros:  $cline .= " (\"$str\", ";  if ($args[0] =~ /fpclassify|isnormal|isfinite|signbit/) {    $c_call = "$args[0] (";  } else {    $c_call = " FUNC($args[0]) (";  }  @descr = split //,$descr_args;  for ($i=0; $i <= $#descr; $i++) {    if ($i >= 1) {      $c_call .= ', ';    }    # FLOAT, int, long int, long long int    if ($descr[$i] =~ /f|i|l|L/) {      $c_call .= $args[$current_arg];      $current_arg++;      next;    }    # &FLOAT, &int    if ($descr[$i] =~ /F|I/) {      ++$extra_var;      $c_call .= '&' . &get_variable ($extra_var);      next;    }    # complex    if ($descr[$i] eq 'c') {      $c_call .= "BUILD_COMPLEX ($args[$current_arg], $args[$current_arg+1])";      $current_arg += 2;      next;    }  }  $c_call .= ')';  $cline .= "$c_call, ";  @descr = split //,$descr_res;  foreach (@descr) {    if ($_ =~ /b|f|i|l|L/ ) {      $cline .= $args[$current_arg];      $current_arg++;    } elsif ($_ eq 'c') {      $cline .= "BUILD_COMPLEX ($args[$current_arg], $args[$current_arg+1])";      $current_arg += 2;    } elsif ($_ eq '1') {      push @special, $args[$current_arg];      ++$current_arg;    }  }  # Add ulp, xfail  $cline .= &new_test ($str, ($current_arg <= $#args) ? $args[$current_arg] : undef);  # special treatment for some functions  if ($args[0] eq 'frexp') {    if (defined $special[0] && $special[0] ne "IGNORE") {      my ($str) = "$call sets x to $special[0]";      $post = "  check_int (\"$str\", x, $special[0]";      $post .= &new_test ($str, undef);    }  } elsif ($args[0] eq 'gamma' || $args[0] eq 'lgamma') {    $pre = "  signgam = 0;\n";    if (defined $special[0] && $special[0] ne "IGNORE") {      my ($str) = "$call sets signgam to $special[0]";      $post = "  check_int (\"$str\", signgam, $special[0]";      $post .= &new_test ($str, undef);    }  } elsif ($args[0] eq 'modf') {    if (defined $special[0] && $special[0] ne "IGNORE") {      my ($str) = "$call sets x to $special[0]";      $post = "  check_float (\"$str\", x, $special[0]";      $post .= &new_test ($str, undef);    }  } elsif ($args[0] eq 'remquo') {    if (defined $special[0] && $special[0] ne "IGNORE") {      my ($str) = "$call sets x to $special[0]";      $post = "  check_int (\"$str\", x, $special[0]";      $post .= &new_test ($str, undef);    }  }  print $file $pre if (defined $pre);  print $file "  $cline";  print $file $post if (defined $post);}# Generate libm-test.csub generate_testfile {  my ($input, $output) = @_;  my ($lasttext);  my (@args, $i, $str);  open INPUT, $input or die ("Can't open $input: $!");  open OUTPUT, ">$output" or die ("Can't open $output: $!");  # Replace the special macros  while (<INPUT>) {    # TEST_...    if (/^\s*TEST_/) {      my ($descr, $args);      chop;      ($descr, $args) = ($_ =~ /TEST_(\w+)\s*\((.*)\)/);      &parse_args (\*OUTPUT, $descr, $args);      next;    }    # START (function)    if (/START/) {      print OUTPUT "  init_max_error ();\n";      next;    }    # END (function)    if (/END/) {      my ($fct, $line, $type);      if (/complex/) {	s/,\s*complex\s*//;	$type = 'complex';      } else {	$type = 'normal';      }      ($fct) = ($_ =~ /END\s*\((.*)\)/);      if ($type eq 'complex') {	$line = "  print_complex_max_error (\"$fct\", ";      } else {	$line = "  print_max_error (\"$fct\", ";      }      if (exists $results{$fct}{'has_ulps'}) {	$line .= "DELTA$fct";      } else {	$line .= '0';      }      if (exists $results{$fct}{'has_fails'}) {	$line .= ", FAIL$fct";      } else {	$line .= ', 0';      }      $line .= ");\n";      print OUTPUT $line;      push @functions, $fct;      next;    }    print OUTPUT;  }  close INPUT;  close OUTPUT;}# Parse ulps filesub parse_ulps {  my ($file) = @_;  my ($test, $type, $float, $eps, $kind);  # $type has the following values:  # "normal": No complex variable  # "real": Real part of complex result  # "imag": Imaginary part of complex result  open ULP, $file  or die ("Can't open $file: $!");  while (<ULP>) {    chop;    # ignore comments and empty lines    next if /^#/;    next if /^\s*$/;    if (/^Test/) {      if (/Real part of:/) {	s/Real part of: //;	$type = 'real';      } elsif (/Imaginary part of:/) {	s/Imaginary part of: //;	$type = 'imag';      } else {	$type = 'normal';      }      s/^.+\"(.*)\".*$/$1/;      $test = $_;      $kind = 'test';      next;    }    if (/^Function: /) {      if (/Real part of/) {	s/Real part of //;	$type = 'real';      } elsif (/Imaginary part of/) {	s/Imaginary part of //;	$type = 'imag';      } else {	$type = 'normal';      }      ($test) = ($_ =~ /^Function:\s*\"([a-zA-Z0-9_]+)\"/);      $kind = 'fct';      next;    }    if (/^i?(float|double|ldouble):/) {      ($float, $eps) = split /\s*:\s*/,$_,2;      if ($eps eq 'fail') {	$results{$test}{$type}{'fail'}{$float} = 1;	$results{$test}{'has_fails'} = 1;      } elsif ($eps eq "0") {	# ignore	next;      } else {	$results{$test}{$type}{'ulp'}{$float} = $eps;	$results{$test}{'has_ulps'} = 1;      }      if ($type =~ /^real|imag$/) {	$results{$test}{'type'} = 'complex';      } elsif ($type eq 'normal') {	$results{$test}{'type'} = 'normal';      }      $results{$test}{'kind'} = $kind;      next;    }    print "Skipping unknown entry: `$_'\n";  }  close ULP;}# Clean up a floating point numbersub clean_up_number {  my ($number) = @_;  # Remove trailing zeros  $number =~ s/0+$//;  $number =~ s/\.$//;  return $number;}# Output a file which can be read in as ulps file.sub print_ulps_file {  my ($file) = @_;  my ($test, $type, $float, $eps, $fct, $last_fct);  $last_fct = '';  open NEWULP, ">$file" or die ("Can't open $file: $!");  print NEWULP "# Begin of automatic generation\n";  # first the function calls  foreach $test (sort keys %results) {    next if ($results{$test}{'kind'} ne 'test');    foreach $type ('real', 'imag', 'normal') {      if (exists $results{$test}{$type}) {	if (defined $results{$test}) {	  ($fct) = ($test =~ /^(\w+)\s/);	  if ($fct ne $last_fct) {	    $last_fct = $fct;	    print NEWULP "\n# $fct\n";	  }	}	if ($type eq 'normal') {	  print NEWULP "Test \"$test\":\n";	} elsif ($type eq 'real') {	  print NEWULP "Test \"Real part of: $test\":\n";	} elsif ($type eq 'imag') {	  print NEWULP "Test \"Imaginary part of: $test\":\n";	}	foreach $float (@all_floats) {	  if (exists $results{$test}{$type}{'ulp'}{$float}) {	    print NEWULP "$float: ",	    &clean_up_number ($results{$test}{$type}{'ulp'}{$float}),	    "\n";	  }	  if (exists $results{$test}{$type}{'fail'}{$float}) {	    print NEWULP "$float: fail\n";	  }	}      }    }  }  print NEWULP "\n# Maximal error of functions:\n";  foreach $fct (sort keys %results) {    next if ($results{$fct}{'kind'} ne 'fct');    foreach $type ('real', 'imag', 'normal') {      if (exists $results{$fct}{$type}) {	if ($type eq 'normal') {	  print NEWULP "Function: \"$fct\":\n";	} elsif ($type eq 'real') {	  print NEWULP "Function: Real part of \"$fct\":\n";	} elsif ($type eq 'imag') {	  print NEWULP "Function: Imaginary part of \"$fct\":\n";	}	foreach $float (@all_floats) {	  if (exists $results{$fct}{$type}{'ulp'}{$float}) {	    print NEWULP "$float: ",	    &clean_up_number ($results{$fct}{$type}{'ulp'}{$float}),	    "\n";	  }	  if (exists $results{$fct}{$type}{'fail'}{$float}) {	    print NEWULP "$float: fail\n";	  }	}	print NEWULP "\n";      }    }  }  print NEWULP "# end of automatic generation\n";  close NEWULP;}sub get_ulps {  my ($test, $type, $float) = @_;  if ($type eq 'complex') {    my ($res);    # Return 0 instead of BUILD_COMPLEX (0,0)    if (!exists $results{$test}{'real'}{'ulp'}{$float} &&	!exists $results{$test}{'imag'}{'ulp'}{$float}) {      return "0";    }    $res = 'BUILD_COMPLEX (';    $res .= (exists $results{$test}{'real'}{'ulp'}{$float}	     ? $results{$test}{'real'}{'ulp'}{$float} : "0");    $res .= ', ';    $res .= (exists $results{$test}{'imag'}{'ulp'}{$float}	     ? $results{$test}{'imag'}{'ulp'}{$float} : "0");    $res .= ')';    return $res;  }  return (exists $results{$test}{'normal'}{'ulp'}{$float}	  ? $results{$test}{'normal'}{'ulp'}{$float} : "0");}sub get_failure {  my ($test, $type, $float) = @_;  if ($type eq 'complex') {    # return x,y    my ($res);    # Return 0 instead of BUILD_COMPLEX_INT (0,0)    if (!exists $results{$test}{'real'}{'ulp'}{$float} &&	!exists $results{$test}{'imag'}{'ulp'}{$float}) {      return "0";    }    $res = 'BUILD_COMPLEX_INT (';    $res .= (exists $results{$test}{'real'}{'fail'}{$float}	     ? $results{$test}{'real'}{'fail'}{$float} : "0");    $res .= ', ';    $res .= (exists $results{$test}{'imag'}{'fail'}{$float}	     ? $results{$test}{'imag'}{'fail'}{$float} : "0");    $res .= ')';    return $res;  }  return (exists $results{$test}{'normal'}{'fail'}{$float}	  ? $results{$test}{'normal'}{'fail'}{$float} : "0");}# Output the defines for a single testsub output_test {  my ($file, $test, $name) = @_;  my ($ldouble, $double, $float, $ildouble, $idouble, $ifloat);  my ($type);  # Do we have ulps/failures?  if (!exists $results{$test}{'type'}) {    return;  }  $type = $results{$test}{'type'};  if (exists $results{$test}{'has_ulps'}) {    # XXX use all_floats (change order!)    $ldouble = &get_ulps ($test, $type, "ldouble");    $double = &get_ulps ($test, $type, "double");    $float = &get_ulps ($test, $type, "float");    $ildouble = &get_ulps ($test, $type, "ildouble");    $idouble = &get_ulps ($test, $type, "idouble");    $ifloat = &get_ulps ($test, $type, "ifloat");    print $file "#define DELTA$name CHOOSE($ldouble, $double, $float, $ildouble, $idouble, $ifloat)\t/* $test  */\n";  }  if (exists $results{$test}{'has_fails'}) {    $ldouble = &get_failure ($test, "ldouble");    $double = &get_failure ($test, "double");    $float = &get_failure ($test, "float");    $ildouble = &get_failure ($test, "ildouble");    $idouble = &get_failure ($test, "idouble");    $ifloat = &get_failure ($test, "ifloat");    print $file "#define FAIL$name CHOOSE($ldouble, $double, $float $ildouble, $idouble, $ifloat)\t/* $test  */\n";  }}# Print include filesub output_ulps {  my ($file, $ulps_filename) = @_;  my ($i, $fct);  open ULP, ">$file" or die ("Can't open $file: $!");  print ULP "/* This file is automatically generated\n";  print ULP "   from $ulps_filename with gen-libm-test.pl.\n";  print ULP "   Don't change it - change instead the master files.  */\n\n";  print ULP "\n/* Maximal error of functions.  */\n";  foreach $fct (@functions) {    output_test (\*ULP, $fct, $fct);  }  print ULP "\n/* Error of single function calls.  */\n";  for ($i = 0; $i < $count; $i++) {    output_test (\*ULP, $tests[$i], $i);  }  close ULP;}
 |