0xV3NOMx
Linux ip-172-26-7-228 5.4.0-1103-aws #111~18.04.1-Ubuntu SMP Tue May 23 20:04:10 UTC 2023 x86_64



Your IP : 3.16.47.126


Current Path : /proc/thread-self/root/usr/bin/
Upload File :
Current File : //proc/thread-self/root/usr/bin/mathspic

#!/usr/bin/perl
#
#(c) Copyright 2005-2010  VERSION 1.13 April 26, 2010  
#                    Apostolos Syropoulos   &  R.W.D. Nickalls
#                    asyropoulos@yahoo.com     dick@nickalls.org
#
# This program can be redistributed and/or modified under the terms
# of the LaTeX Project Public License Distributed from CTAN
# archives in directory macros/latex/base/lppl.txt; either
# version 1 of the License, or any later version.
#
package DummyFH;
my $index = 0;
sub TIEHANDLE {
    my $class = shift;
    my $self= shift;
    bless $self, $class;
}
sub READLINE {
    my $self = shift;
    #shift @$self;
    if ($index > $#$self) {
      $index = 0;
      return undef;
    }
    else {
      return $self->[$index++];
    }
}

package main;
use Math::Trig;
our $version_number = "1.13 Apr 26, 2010";
our $commandLineArgs = join(" ", @ARGV);
our $command  = "";
our $curr_in_file = "";
our %PointTable = ();
our %VarTable = ();
our %ConstTable = ();
our $no_errors = 0;
our $xunits = "1pt";
our $yunits = "1pt";
our $units = "pt|pc|in|bp|cm|mm|dd|cc|sp";
our $defaultsymbol = "\$\\bullet\$";
our $defaultLFradius = 0;
use constant PI => atan2(1,1)*4;
use constant R2D => 180 / PI;
use constant D2R => PI / 180;
our $arrowLength = 2;
our $arrowLengthUnits = "mm";
our $arrowAngleB = 30;
our $arrowAngleC = 40; 
our %DimOfPoint = ();
our $GlobalDimOfPoints = 0;
our @Macros = ();
our $LineThickness = 0.4;

  sub mpp {
    my $in_line;
    chomp($in_line = shift);
    my $LC = shift;
    my $out_line = $in_line;
    my $macro_name = "";
    my @macro_param = ();
    my $macro_code = "";
    if ($in_line =~ s/^%def\s*//) {
      if ($in_line =~ s/^(\w+)\s*//){
        $macro_name = $1;
      }
      else {
        PrintErrorMessage("No macro name has been found",$LC);
        return ""
      }
      if ($in_line =~ s/^\(\s*//) {
        # do nothing
      }
      else {
        PrintErrorMessage("No left parenthesis after macro name has been found",$LC);
        return "";
      }
      if ($in_line =~ s/^\)//) {
        # Macro has no parameters!
      }
      else {
        MACROS: while (1) {
          if ($in_line =~ s/^(\w+)\s*//) {
            push (@macro_param, $1);
          }
          else {
            PrintErrorMessage("No macro parameter name has been found",$LC);
            return "";
          }
          if ($in_line =~ s/^,\s*//) {
            next MACROS;
          }
          else {
            last MACROS;
          } 
        }
        if ($in_line =~ s/^\)//) {
          # do nothing!
        }
        else {
          PrintErrorMessage("No closing parenthesis after macro parameters",$LC);
          return "";
        }
      }
      $in_line =~ s/([^%]+)(%.*)/$1/;
      $macro_code = $in_line;
      push ( @Macros , { 'macro_name' => $macro_name,
                         'macro_code' => $macro_code,
                         'macro_param' => \@macro_param }); 
      return $out_line;
    }
    elsif ($in_line =~ s/^%undef\s*//) {
      if ($in_line =~ s/^(\w+)//) {
        my $undef_macro = $1;
        for(my $i = $#Macros; $i >= 0; $i--) {
          if ($Macros[$i]->{'macro_name'} eq $undef_macro) {
           splice(@Macros,$i,1);
          }   
        }
      }
      return $out_line;
    }
    elsif ($in_line =~ s/^\s*%//) {
      return $out_line;
    }
    else {
      my $comment = $2 if $in_line =~ s/([^%]+)(%.+)/$1/;
      EXPANSIONLOOP: while () {
        my $org_in_line = $in_line;
        for(my $i = $#Macros; $i >= 0; $i--) {
          my $macro_name = $Macros[$i]->{'macro_name'};
          if ($in_line =~ /&$macro_name\b/) {       ############################
            my $num_of_macro_args = @{$Macros[$i]->{'macro_param'}};
            if ( $num_of_macro_args > 0 ) { 
            # Macro with parameters
              my $pattern = "&$macro_name\\(";
              foreach my $p ( 1..$num_of_macro_args ) {
                my $comma = ($p == $num_of_macro_args) ? "\\s*" : "\\s*,\\s*";
                $pattern .= "\\s*[^\\s\\)]+$comma";
              }
              $pattern .= "\\)";
              while($in_line =~ /&$macro_name\b/) {
                if ($in_line =~ /$pattern/) {   
                  my $before = $`;
                  my $after = $';
                  my $match = $&;
                  my $new_code = $Macros[$i]->{'macro_code'};
                  $match =~ s/^&$macro_name\(\s*//;
                  $match =~ s/\)$//; 
                  foreach my $arg ( 0..($num_of_macro_args - 1) ) {
                    my $old = $Macros[$i]->{'macro_param'}->[$arg];
                    my $comma = ($arg == ($num_of_macro_args - 1)) ? "" : ",";
                    $match =~ s/^\s*([^\s,]+)\s*$comma//;
                    my $new = $1; 
                    # 'g': Parameter may occur several times
                    # in $new_code.
                    # '\b': Substitute only whole words
                    # not x in xA
                    $new_code =~ s/\b$old\b/$new/g;
                  }
                  $in_line = "$before$new_code$after"; 
                }
                else {
                  PrintErrorMessage("Usage of macro &$macro_name does not " .
                                    "match its definition", $LC); 
                  return "";
                }
              }
            }
            else {
              # Macro without parameters
              my $replacement = $Macros[$i]->{'macro_code'};
              # '\b': Substitute only whole words
              # not x in xA
              $in_line =~ s/&$macro_name\b/$replacement/g;
            }
          }
        } 
        last EXPANSIONLOOP if ( $org_in_line eq $in_line );
      }
      return "$in_line$comment";   
    }
  }

      sub PrintErrorMessage {
        my $errormessage = shift;
        my $error_line   = shift;
        my ($l,$A);
        $l = 1+length($command)-length;
        $A = substr($command,0,$l);
        $l += 7 +length($error_line);

        for my $fh (STDOUT, LOG) {
          print $fh "$curr_in_file", "Line $error_line: $A\n";
          print $fh " " x $l  ,$_,"***Error: $errormessage\n";
        }
        if ($comments_on) {  #print to output file file
          print OUT "%% *** $curr_in_file", "Line $error_line: $A\n";
          print OUT "%% *** "," " x $l  ,$_,"%% ... Error: $errormessage\n";
        }
        $no_errors++;
      }

      sub PrintWarningMessage {
        my $warningMessage = shift;
        my $warning_line   = shift;
        my ($l,$A);
        $l = 1+length($command)-length;
        $A = substr($command,0,$l);
        $l += 7 +length($warning_line);

        for my $fh (STDOUT, LOG) {
          print $fh "$curr_in_file", "Line $warning_line: $A\n";
          print $fh " " x $l  ,$_,"***Warning: $warningMessage\n";
        }
        if ($comments_on) {  #print to output file file
          print OUT "%% *** $curr_in_file", "Line $warning_line: $A\n";
          print OUT "%% *** "," " x $l  ,$_,"%% ... Warning: $warningMessage\n";
        }
      }

      sub PrintFatalError {
        my $FatalMessage = shift;
        my $fatal_line   = shift;
        my ($l,$A);
        $l = 1+length($command)-length;
        $A = substr($command,0,$l);
        $l += 7 +length($fatal_line);

        die "$curr_in_file", "Line $fatal_line: $A\n" .
            (" " x $l) . $_ . "***Fatal Error: $FatalMessage\n";
      }

  sub chk_lparen {
    my $token = $_[0];
    my $lc    = $_[1];
    s/\s*//;
    if (/^[^\(]/) {
      PrintErrorMessage("Missing ( after $token",$lc);
    }
    else {
      s/^\(\s*//;
    }
  }

  sub chk_rparen {
    my $token = $_[0];
    my $lc    = $_[1];
    s/\s*//;
    if (s/^\)//) {
      s/\s*//;
    }
    else {
      PrintErrorMessage("Missing ) after $token",$lc);
    }
  }


  sub chk_lcb {
    my $token = $_[0];
    my $lc    = $_[1];
    s/\s*//;
    if ($_ !~ /^\{/) {
      PrintErrorMessage("Missing { after $token",$lc);
    }
    else {
      s/^{\s*//;
    }
  }

  sub chk_rcb {
    my $token = $_[0];
    my $lc    = $_[1];
    if ($_ !~ /^\s*\}/) {
      PrintErrorMessage("Missing } after $token",$lc);
    }
    else {
      s/^\s*}\s*//;
    }
  }

  sub chk_lsb {
     my $token = $_[0];
     my $lc    = $_[1];

     s/\s*//;
     if ($_ !~ /^\[/) {
       PrintErrorMessage("Missing [ after  $token",$lc);
     }
     else {
        s/^\[\s*//;
     }
  }

  sub chk_rsb {
     my $token = $_[0];
     my $lc    = $_[1];

     s/\s*//;
     if ($_ !~ /^\]/) {
       PrintErrorMessage("Missing ] after  $token",$lc);
     }
     else {
       s/^\]\s*//;
     }
  }

  sub chk_comma {
    my $lc  = $_[0];

    s/\s*//;
    if (/^[^,]/) {
      PrintErrorMessage("Did not find expected comma",$lc);
    }
    else {
      s/^,\s*//;
    }
  }

  sub chk_comment {
     my $lc = $_[0];

     s/\s*//;
     if (/^%/) {
       # do nothing!
     }
     elsif (/^[^%]/) {
       PrintWarningMessage("Trailing text is ignored",$lc);
     }
  }

    sub print_headers
    {
       my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime;
       $year+=1900;
       $mon+=1;
       $now_string = "$year/" . ($mon>9 ? "$mon/" : "0$mon/") .
                     ($mday>9 ? "$mday    " : "0$mday    ") .
                     ($hour>9 ? "$hour:" : "0$hour:") .
                     ($min>9 ? "$min:" : "0$min:") .
                     ($sec>9 ? "$sec" : "0$sec");
       print OUT "%* -----------------------------------------------\n";
       print OUT "%* mathspic (Perl version $version_number)\n";
       print OUT "%* A filter program for use with PiCTeX\n";
       print OUT "%* Copyright (c) 2005-2010 A Syropoulos & RWD Nickalls \n";
       print OUT "%* Command line: $0 $commandLineArgs\n";
       print OUT "%* Input filename : $source_file\n";
       print OUT "%* Output filename: $out_file\n";
       print OUT "%* Date & time: $now_string\n";
       print OUT "%* -----------------------------------------------\n";
       #
       print LOG "----\n";
       print LOG "$now_string\n";
       print LOG "mathspic (Perl version $version_number)\n";
       print LOG "Copyright (c) 2005-2010 A Syropoulos & RWD Nickalls \n";
       print LOG "Input file  = $source_file\n";
       print LOG "Output file = $out_file\n";
       print LOG "Log file    = $log_file\n";
       print LOG "----\n";
    }

     sub get_point {

         my ($lc) = $_[0];
         my ($PointName);

         if (s/^([^\W\d_]\d{0,4})\s*//i) { #point name
           $PointName = $1;
           if (!exists($PointTable{lc($PointName)})) {
             PrintErrorMessage("Undefined point $PointName",$lc);
             return "_undef_";
           }
           else {
             return lc($PointName);
           }
         }
         else {
           PrintErrorMessage("Point name expected",$lc);
           return "_undef_";
         }
     }

    sub perpendicular {
        my ($xP, $yP, $xA, $yA, $xB, $yB) = @_;
        my ($xF, $yF, $deltax, $deltay, $m1, $m2, $c1, $c2, $factor);

        $deltax = $xA - $xB;
        return ($xA, $yP) if abs($deltax) < 0.0000001;
        $deltay = $yA - $yB;
        return ($xP, $yA) if abs($deltay) < 0.0000001;
        $m1 = $deltay / $deltax;
        eval { $m2 = (-1) / $m1;};
        PrintFatalError("Division by zero",$lc) if $@;
        $c1 = $yA - $m1 * $xA;
        $c2 = $yP - $m2 * $xP;
        eval { $factor = 1 / ($m1 - $m2)};
        PrintFatalError("Division by zero",$lc) if $@;
        return (($c2 - $c1) * $factor, ($m1 * $c2 - $m2 * $c1) * $factor);
    }

    sub Length {
        my ($xA, $yA, $xB, $yB)=@_;
        return sqrt(($xB - $xA)**2 + ($yB - $yA)**2);
    }

    sub triangleArea {
        my ($xA, $yA, $xB, $yB, $xC, $yC)=@_;
        my ($lenAB, $lenBC, $lenCA, $s);

        $lenAB = Length($xA,$yA,$xB,$yB);
        $lenBC = Length($xB,$yB,$xC,$yC);
        $lenCA = Length($xC,$yC,$xA,$yA);
        $s = ($lenAB + $lenBC + $lenCA) / 2;
        return sqrt($s * ($s - $lenAB)*($s - $lenBC)*($s - $lenCA));
    }

    sub pointOnLine {
        my ($xA, $yA, $xB, $yB, $dist)=@_;
        my ($deltax, $deltay, $xPol, $yPol);

        $deltax = $xB - $xA;
        $deltay = $yB - $yA;
        $xPol = $xA + ($dist * $deltax / &Length($xA,$yA,$xB,$yB));
        $yPol = $yA + ($dist * $deltay / &Length($xA,$yA,$xB,$yB));
        return ($xPol, $yPol);
    }


  
    sub circumCircleCenter {
       my ($xA, $yA, $xB, $yB, $xC, $yC, $lc)=@_;
       my ($deltay12, $deltax12, $xs12, $ys12);
       my ($deltay23, $deltax23, $xs23, $ys23);
       my ($xcc, $ycc);
       my ($m23, $mr23, $c23, $m12, $mr12, $c12);
       my ($sideA, $sideB, $sideC, $a, $radius);

       if (abs(triangleArea($xA, $yA, $xB, $yB, $xC, $yC)) < 0.0000001)
       {
          PrintErrorMessage("Area of triangle is zero!",$lc);
          return (0,0,0);
       }
       $deltay12 = $yB - $yA;
       $deltax12 = $xB - $xA;
       $xs12 = $xA + $deltax12 / 2;
       $ys12 = $yA + $deltay12 / 2;
       #
       $deltay23 = $yC - $yB;
       $deltax23 = $xC - $xB;
       $xs23 = $xB + $deltax23 / 2;
       $ys23 = $yB + $deltay23 / 2;
       #
       CCXYLINE:{
       if (abs($deltay12) < 0.0000001)
       {
          $xcc = $xs12;
          if (abs($deltax23) < 0.0000001)
          {
             $ycc = $ys23;
             last CCXYLINE;
          }
          else
          {
             $m23 = $deltay23 / $deltax23;
             $mr23 = -1 / $m23;
             $c23 = $ys23 - $mr23 * $xs23;
             $ycc = $mr23 * $xs12 + $c23;
             last CCXYLINE;
          }
       }
       if (abs($deltax12) < 0.0000001)
       {
          $ycc = $ys12;
          if (abs($deltay23) < 0.0000001)
          {
             $xcc = $xs23;
             last CCXYLINE;
          }
          else
          {
             $m23 = $deltay23 / $deltax23;
             $mr23 = -1 / $m23;
             $c23 = $ys23 - $mr23 * $xs23;
             $xcc = ($ys12 - $c23) / $mr23;
             last CCXYLINE;
          }
       }
       if (abs($deltay23) < 0.0000001)
       {
          $xcc = $xs23;
          if (abs($deltax12) < 0.0000001)
          {
             $ycc = $ys12;
             last CCXYLINE;
          }
          else
          {
             $m12 = $deltay12 / $deltax12;
             $mr12 = -1 / $m12;
             $c12 = $ys12 - $mr12 * $xs12;
             $ycc = $mr12 * $xcc + $c12;
             last CCXYLINE;
          }
       }
       if (abs($deltax23) < 0.0000001)
       {
          $ycc = $ys23;
          if (abs($deltay12) < 0.0000001)
          {
             $xcc = $xs12;
             last CCXYLINE;
          }
          else
          {
             $m12 = $deltay12 / $deltax12;
             $mr12 = -1 / $m12;
             $c12 = $ys12 - $mr12 * $xs12;
             $xcc = ($ycc - $c12) / $mr12;
             last CCXYLINE;
          }
       }
       $m12 = $deltay12 / $deltax12;
       $mr12 = -1 / $m12;
       $c12 = $ys12 - $mr12 * $xs12;
       #-----
       $m23 = $deltay23 / $deltax23;
       $mr23 = -1 / $m23;
       $c23 = $ys23 - $mr23 * $xs23;
       $xcc = ($c23 - $c12) / ($mr12 - $mr23);
       $ycc = ($c23 * $mr12 - $c12 * $mr23) / ($mr12 - $mr23);
       }
       #
       $sideA = &Length($xA,$yA,$xB,$yB);
       $sideB = &Length($xB,$yB,$xC,$yC);
       $sideC = &Length($xC,$yC,$xA,$yA);
       $a = triangleArea($xA, $yA, $xB, $yB, $xC, $yC);
       $radius = ($sideA * $sideB * $sideC) / (4 * $a);
       #
       return ($xcc, $ycc, $radius);
    }

    sub ComputeDist {
       my ($lc) = $_[0];
       my ($v1, $v2);

       if (s/^((\+|-)?\d+(\.\d+)?([eE](\+|-)?\d+)?)//) #is it a number?
       {
           return ($1, 1);
       }
       elsif (/^[^\W\d_]\d{0,4}[^\W\d_]\d{0,4}/) #it is a pair of IDs?
       {
          s/^([^\W\d_]\d{0,4})//i;
          $v1 = $1;
          if (!exists($PointTable{lc($v1)})) {  
             if (exists($VarTable{lc($v1)})) {
                return ($VarTable{lc($v1)}, 1);
             }
             PrintErrorMessage("Point $v1 has not been defined", $lc);
             s/^\s*[^\W\d_]\d{0,4}//i;
             return (0,0);
          }
          $v1 = lc($v1);
          s/^\s*([^\W\d_]\d{0,4})//i;
          $v2 = $1;
          if (!exists($PointTable{lc($v2)}))
          {
             PrintErrorMessage("Point $v2 has not been defined", $lc);
             return (0,0);
          }
          $v2 = lc($v2);
          my ($x1,$y1,$pSV1,$pS1) = unpack("d3A*",$PointTable{$v1});
          my ($x2,$y2,$pSV2,$pS2) = unpack("d3A*",$PointTable{$v2});
          return (Length($x1,$y1,$x2,$y2), 1);
       }
       elsif (s/^([^\W\d_]\d{0,4})//i) # it is a single id
       {
         $v1 = $1;
         if (!exists($VarTable{lc($v1)})) #it isn't a variable
         {
           PrintErrorMessage("Variable $v1 has not been defined", $lc);
           return (0,0);
         }
         return ($VarTable{lc($v1)}, 1);
       }
       else
       {
          PrintErrorMessage("Unexpected token", $lc);
          return (0,0);
       }
     }

     sub intersection4points {
       my ($x1, $y1, $x2, $y2, $x3, $y3, $x4, $y4) = @_;
       my ($deltay12, $deltax12, $deltay34, $deltax34);
       my ($xcc, $ycc, $m34, $c34, $m12, $c12);

       $deltay12 = $y2 - $y1;
       $deltax12 = $x2 - $x1;
       #
       $deltay34 = $y4 - $y3;
       $deltax34 = $x4 - $x3;
       I4PXYLINE:{
          if (abs($deltay12) < 0.0000001)
          {
             $ycc = $y1;
             if (abs($deltax34) < 0.0000001)
             {
                $xcc = $x3;
                last I4PXYLINE;
             }
             else
             {
                $m34 = $deltay34 / $deltax34;
                $c34 = $y3 - $m34 * $x3;
                $xcc = ($ycc - $c34) / $m34;
                last I4PXYLINE;
             }
          }
          if (abs($deltax12) < 0.0000001)
          {
             $xcc = $x1;
             if (abs($deltay34) < 0.0000001)
             {
                $ycc = $y3;
                last I4PXYLINE;
             }
             else
             {
                $m34 = $deltay34 / $deltax34;
                $c34 = $y3 - $m34 * $x3;
                $ycc = $m34 * $xcc + $c34;
                last I4PXYLINE;
             }
          }
          if (abs($deltay34) < 0.0000001)
          {
             $ycc = $y3;
             if (abs($deltax12) < 0.0000001)
             {
                $xcc = $x1;
                last I4PXYLINE;
             }
             else
             {
                $m12 = $deltay12 / $deltax12;
                $c12 = $y1 - $m12 * $x1;
                $xcc = ($ycc - $c12) / $m12;
                last I4PXYLINE;
             }
          }
          if (abs($deltax34) < 0.0000001)
          {
             $xcc = $x3;
             if (abs($deltay12) < 0.0000001)
             {
                $ycc = $y1;
                last I4PXYLINE;
             }
             else
             {
                $m12 = $deltay12 / $deltax12;
                $c12 = $y1 - $m12 * $x1;
                $ycc = $m12 * $xcc + $c12;
                last I4PXYLINE;
             }
          }
          $m12 = $deltay12 / $deltax12;
          $c12 = $y1 - $m12 * $x1;
          $m34 = $deltay34 / $deltax34;
          $c34 = $y3 - $m34 * $x3;
          $xcc = ($c34 - $c12) / ($m12 - $m34);
          $ycc = ($c34 * $m12 - $c12 * $m34) / ($m12 - $m34);
       }
       return ($xcc, $ycc);
     }

    sub IncircleCenter {
       my ($Ax, $Ay, $Bx, $By, $Cx, $Cy) = @_;
       my ($sideA, $sideB, $sideC);
       my ($ba1, $xA1, $yA1, $cb1, $ac1, $xB1, $yB1, $xC1, $yC1, $a, $s, $r);

       #determine the lengths of the sides
       $sideA = Length($Bx, $By, $Cx, $Cy);
       $sideB = Length($Cx, $Cy, $Ax, $Ay);
       $sideC = Length($Ax, $Ay, $Bx, $By);
       #
       $ba1 = ($sideC * $sideA) / ($sideB + $sideC);
       ($xA1, $yA1) = pointOnLine($Bx, $By, $Cx, $Cy, $ba1);
       $cb1 = ($sideA * $sideB) / ($sideC + $sideA);
       ($xB1, $yB1) = pointOnLine($Cx, $Cy, $Ax, $Ay, $cb1);
       $ac1 = ($sideB * $sideC) / ($sideA + $sideB);
       ($xC1, $yC1) = pointOnLine($Ax, $Ay, $Bx, $By, $ac1);
       ($xcenter, $ycenter) = &intersection4points($Ax, $Ay, $xA1, $yA1,
                                                   $Bx, $By, $xB1, $yB1);
       # get radius
       $a = &triangleArea($Ax, $Ay, $Bx, $By, $Cx, $Cy);
       $s = ($sideA + $sideB +$sideC) / 2;
       $r = $a / $s;
       return ($xcenter, $ycenter, $r);
    }

    sub Angle {
      my ($Ax, $Ay, $Bx, $By, $Cx, $Cy) = @_;
      my ($RAx, $RAy, $RBx, $RBy, $RCx, $RCy, $deltax, $deltay);
      my ($lineBA, $lineBC, $lineAC, $k, $kk, $angle);
      my ($T, $cosT, $sinT) = (0.3, cos(0.3), sin(0.3));

      $RAx = $Ax * $cosT + $Ay * $sinT;
      $RAy = -$Ax * $sinT + $Ay * $cosT;
      $RBx = $Bx * $cosT + $By * $sinT;
      $RBy = -$Bx * $sinT + $By * $cosT;
      $RCx = $Cx * $cosT + $Cy * $sinT;
      $RCy = -$Cx * $sinT + $Cy * $cosT;
      $deltax = $RBx - $RAx;
      $deltay = $RBy - $RAy;
      $lineBA = sqrt($deltax*$deltax + $deltay*$deltay);
      if ($lineBA < 0.0000001)
      {
         return -500;
      }
      $deltax = $RBx - $RCx;
      $deltay = $RBy - $RCy;
      $lineBC = sqrt($deltax*$deltax + $deltay*$deltay);
      if ($lineBC < 0.0000001)
      {
         return -500;
      }
      $deltax = $RAx - $RCx;
      $deltay = $RAy - $RCy;
      $lineAC = sqrt($deltax*$deltax + $deltay*$deltay);
      if ($lineAC < 0.0000001)
      {
         return -500;
      }
      $k = ($lineBA*$lineBA + $lineBC*$lineBC - $lineAC*$lineAC ) /
           (2 * $lineBA * $lineBC);
      $k = -1 if $k < -0.99999;
      $k = 1 if $k > 0.99999;
      $kk = $k * $k;
      if (($kk * $kk) == 1)
      {
         $angle = PI if $k == -1;
         $angle = 0 if $k == 1;
      }
      else
      {
         $angle = (PI / 2) - atan2($k / sqrt(1 - $kk),1);
      }
      return $angle * 180 / PI;
    }

    sub excircle {
      my ($A, $B, $C, $D, $E) = @_;
      my ($Ax,$Ay,$Bx,$By,$Dx,$Dy,$Ex,$Ey,$ASVA,$ASA);
      ($Ax,$Ay,$ASVA,$ASA)=unpack("d3A*",$PointTable{$A});
      ($Bx,$By,$ASVA,$ASA)=unpack("d3A*",$PointTable{$B});
      ($Cx,$Cy,$ASVA,$ASA)=unpack("d3A*",$PointTable{$C});
      ($Dx,$Dy,$ASVA,$ASA)=unpack("d3A*",$PointTable{$D});
      ($Ex,$Ey,$ASVA,$ASA)=unpack("d3A*",$PointTable{$E});
      my ($sideA, $sideB, $sideC, $s, $R, $theAdeg, $d);
      my ($Xmypoint, $Ymypoint, $deltax, $deltay, $mylength, $xc, $yc);

      $sideA = &Length($Bx, $By, $Cx, $Cy);
      $sideB = &Length($Cx, $Cy, $Ax, $Ay);
      $sideC = &Length($Ax, $Ay, $Bx, $By);
      $s = ($sideA + $sideB + $sideC) / 2;
      $R = triangleArea($Ax, $Ay, $Bx, $By, $Cx, $Cy) /
           ($s - &Length($Dx, $Dy, $Ex, $Ey));
      if (($D eq $A && $E eq $B) || ($D eq $B && $E eq $A))
      {
        $theAdeg = &Angle($Bx, $By, $Cx, $Cy, $Ax, $Ay);
        $Xmypoint = $Cx;
        $Ymypoint = $Cy;
      }
      elsif (($D eq $B && $E eq $C) || ($D eq $C && $E eq $B))
      {
        $theAdeg = &Angle($Cx, $Cy, $Ax, $Ay, $Bx, $By);
        $Xmypoint = $Ax;
        $Ymypoint = $Ay;
      }
      elsif (($D eq $C && $E eq $A) || ($D eq $A && $E eq $C))
      {
        $theAdeg = &Angle($Ax, $Ay, $Bx, $By, $Cx, $Cy);
        $Xmypoint = $Bx;
        $Ymypoint = $By;
      }
      else
      {
         return (0,0,0);
      }
      $d = $R  / sin($theAdeg * PI / 180 / 2);
      my ($xIn, $yIn, $rin) = &IncircleCenter($Ax, $Ay, $Bx, $By, $Cx, $Cy);
      $deltax = $xIn - $Xmypoint;
      $deltay = $yIn - $Ymypoint;
      $mylength = sqrt($deltax*$deltax + $deltay*$deltay);
      $xc = $Xmypoint + $d * $deltax / $mylength;
      $yc = $Ymypoint + $d * $deltay / $mylength;
      return ($xc, $yc, $R);
    }

     sub DrawLineOrArrow {      
       my $draw_Line = shift;
       my $lc = shift;
       my $lineLength = -1;
       my $stacklen = 0;
       my @PP = ();
  #     if ($draw_Line != 2) {
  #       s/\s*//;
  #       if (s/^\[\s*//) { # optional length specifier 
  #         $lineLength = expr($lc);
  #         if ($lineLength <= 0) {
  #           PrintErrorMessage("length must greater than zero",$lc);
  #           $lineLength = -1;
  #         } 
  #         chk_rsb("optional part",$lc);
  #       }
  #     } 
       chk_lparen("$cmd",$lc);
       DRAWLINES:while(1) {
         @PP = () ;
         while(1) {
           if (s/^([^\W\d_]\d{0,4})\s*//i) { #point name
             $P = $1;
             if (!exists($PointTable{lc($P)})) {
               PrintErrorMessage("Undefined point $P",$lc);
             }
             else {
               push (@PP,$P);
             }
           }
           else {
             $stacklen = @PP;
             if ($draw_Line != 2) {
               if ($stacklen <= 1) {
                 PrintErrorMessage("Wrong number of points",$lc);
               }
               else {
                 push(@PP,$lc);
                 if ($draw_Line == 0) {
                   drawarrows(@PP); 
                 }
                 elsif ($draw_Line == 1) {
                   drawlines(@PP); 
                 }
               }
             }
             if (s/^,\s*// and $draw_Line != 2) {
               next DRAWLINES;
             }
             else {
               last DRAWLINES;
             }
           }
         }
       }
       if ($draw_Line == 2) {
         $stacklen =  @PP;
         if ($stacklen < 2) {
           PrintErrorMessage("Wrong number of points",$lc);
         }
         elsif ($stacklen % 2 == 0) {
           PrintErrorMessage("Number of points must be odd",$lc);
         }
         else {
           drawCurve(@PP);        
         }
       }
       chk_rparen("arguments of $cmd",$lc);
       chk_comment($lc);
     }

    sub drawarrows {
      my ($NoArgs);
      $NoArgs = @_;
      my ($lc) = $_[$NoArgs-1]; #line number is the last argument
      my ($NumberOfPoints, $p, $q, $r12, $d12);
      my ($px,$py,$pSV,$pS, $qx,$qy,$qSV,$qS);

      $NumberOfPoints = $NoArgs - 1;
      LOOP: for(my $i=0; $i < $NumberOfPoints - 1; $i++)
      {
         $p = $_[$i];
         $q = $_[$i+1];
         ($px,$py,$pSV,$pS) = unpack("d3A*",$PointTable{lc($p)});
         ($qx,$qy,$qSV,$qS) = unpack("d3A*",$PointTable{lc($q)});
         $pSV = $defaultLFradius if $pSV == 0;
         $qSV = $defaultLFradius if $qSV == 0;
         $r12 = $pSV + $qSV;
         $d12 = Length($px,$py,$qx,$qy);
         if ($d12 <= $r12)
         {
            if($d12 == 0)
            {
               PrintErrorMessage("points $p and $q are the same", $lc);
               next LOOP;
            }
            PrintWarningMessage("arrow $p$q not drawn: points too close or ".
                                  "radii too big", $lc);
            next LOOP;
         }
         ($px, $py) = pointOnLine($px, $py, $qx, $qy, $pSV) if $pSV > 0;
         ($qx, $qy) = pointOnLine($qx, $qy, $px, $py, $qSV) if $qSV > 0;
         my ($beta, $gamma);
         $beta  = tan($arrowAngleB * D2R / 2);
         $gamma = 2 * tan($arrowAngleC * D2R / 2);
         printf OUT "\\arrow <%.5f%s> [%.5f,%.5f] from %.5f %.5f to %.5f %.5f\n",
                $arrowLength, $arrowLengthUnits, $beta, $gamma, $px, $py, $qx, $qy;
      }
    }

    sub drawlines {
      my ($NoArgs);
      $NoArgs = @_;
      my ($lc) = $_[$NoArgs-1]; #line number is the last argument
      my ($NumberOfPoints, $p, $q, $r12, $d12);
      my ($px,$py,$pSV,$pS, $qx,$qy,$qSV,$qS);

      $NumberOfPoints = $NoArgs - 1;
      LOOP: for(my $i=0; $i < $NumberOfPoints - 1; $i++)
      {
         $p = $_[$i];
         $q = $_[$i+1];
         ($px,$py,$pSV,$pS) = unpack("d3A*",$PointTable{lc($p)});
         ($qx,$qy,$qSV,$qS) = unpack("d3A*",$PointTable{lc($q)});
         $pSV = $defaultLFradius if $pSV == 0;
         $qSV = $defaultLFradius if $qSV == 0;
         $r12 = $pSV + $qSV;
         $d12 = Length($px,$py,$qx,$qy);
         if ($d12 <= $r12)
         {
            if($d12 == 0)
            {
               PrintErrorMessage("points $p and $q are the same", $lc);
               next LOOP;
            }
            PrintWarningMessage("line $p$q not drawn: points too close or ".
                                  "radii too big", $lc);
            next LOOP;
         }
         ($px, $py) = pointOnLine($px, $py, $qx, $qy, $pSV) if $pSV > 0;
         ($qx, $qy) = pointOnLine($qx, $qy, $px, $py, $qSV) if $qSV > 0;
         if ($px == $qx || $py == $qy)
         {
            printf OUT "\\putrule from %.5f %.5f to %.5f %.5f %%%% %s%s\n",
                       $px,$py,$qx,$qy,$p,$q;
         }
         else
         {
            printf OUT "\\plot %.5f %.5f\t%.5f %.5f / %%%% %s%s\n",
                       $px, $py,$qx,$qy,$p,$q;
         }
      }
    }

    sub drawCurve {
      my ($NoArgs);
      $NoArgs = @_;
      my ($lc) = $_[$NoArgs-1]; #line number is the last argument
      my ($NumberOfPoints, $p);

      $NumberOfPoints = $NoArgs - 1;
      print OUT "\\setquadratic\n\\plot\n";
      for(my $i=0; $i <= $NumberOfPoints; $i++)
      {
         $p = $_[$i];
         my ($px,$py,$pSV,$pS) = unpack("d3A*",$PointTable{lc($p)});
         printf OUT "\t%0.5f  %0.5f", $px, $py;
         print OUT (($i == $NumberOfPoints) ? " / %$p\n" : " %$p\n");  
      }
      print OUT "\\setlinear\n";
    }

    sub drawpoints {
      my ($NumberOfPoints,$p);
      $NumberOfPoints = @_;
      my ($px,$py,$pSV,$pS);

      for($i=0; $i < $NumberOfPoints; $i++)
      {
         $p = $_[$i];
         ($px,$py,$pSV,$pS) = unpack("d3A*",$PointTable{lc($p)});
         if ($pS eq "" and $defaultsymbol =~ /circle|square/) {
           $pS = $defaultsymbol;
         }
         POINTSWITCH: {
           if ($pS eq "") # no plot symbol specified
           {
             printf OUT "\\put {%s} at %.5f %.5f %%%% %s\n",
                        $defaultsymbol, $px, $py, $p;
             last POINTSWITCH;
           }
           if ($pS eq "circle") # plot symbol is a circle
           { 
             my $radius = (defined($DimOfPoint{lc($p)})) ? $DimOfPoint{lc($p)} : 
                           $GlobalDimOfPoints; 
             if ($radius > 0) # draw a circle using the current units
             {
                if ($radius == 1.5) # use \bigcirc
                {
                  printf OUT "\\put {\$\\bigcirc\$} at %.5f %.5f  %%%% %s\n",
                             $px, $py, $p;
                }
                else
                {
                  printf OUT "\\circulararc 360 degrees from %.5f %.5f center at %.5f %.5f %%%% %s\n",
                             $px+$radius, $py, $px, $py, $p;
                }
             }
             else #use \circ symbol
             {
               printf OUT "\\put {\$\\circ\$} at %.5f %.5f %%%% %s\n",
                           $px,$py,$p;
             }
             last POINTSWITCH;
           }
           if ($pS eq "square")
           {
             my $side = (defined($DimOfPoint{lc($p)})) ? $DimOfPoint{lc($p)} : 
                           $GlobalDimOfPoints;
             printf OUT "\\put {%s} at %.5f %.5f %%%% %s\n",
                           drawsquare($side), $px, $py, $p;
             last POINTSWITCH;
           }
           printf OUT "\\put {%s} at %.5f %.5f %%%% %s\n", $pS,$px,$py,$p;
         }
      }
    }

  sub drawAngleArc {
    my ($P1, $P2, $P3, $radius, $inout, $direction) = @_;
    my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$P1});
    my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$P2});
    my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$P3});

    my $internalAngle = Angle($x1, $y1, $x2, $y2, $x3, $y3);
    my $externalAngle = 360 - $internalAngle;
    my ($x, $y) = pointOnLine($x2, $y2, $x1, $y1, $radius);
    my $code = "";
    if ($inout eq "internal" and $direction eq "clockwise" ) {
       $code = sprintf "\\circulararc %.5f degrees from %.5f %.5f center at %.5f %.5f\n",
               -1 * $internalAngle, $x, $y, $x2, $y2;
    }
    elsif ($inout eq "internal" and $direction eq "anticlockwise" ) {
       $code = sprintf "\\circulararc %.5f degrees from %.5f %.5f center at %.5f %.5f\n",
               $internalAngle, $x, $y, $x2, $y2;
    }
    elsif ($inout eq "external" and $direction eq "clockwise" ) {
       $code = sprintf "\\circulararc %.5f degrees from %.5f %.5f center at %.5f %.5f\n",
               -1 * $externalAngle, $x, $y, $x2, $y2;
    }
    elsif ($inout eq "external" and $direction eq "anticlockwise" ) {
       $code = sprintf "\\circulararc %.5f degrees from %.5f %.5f center at %.5f %.5f\n",
               $externalAngle, $x, $y, $x2, $y2;
    }
    return $code;
  }

  sub drawAngleArrow {
     my ($P1, $P2, $P3, $radius, $inout, $direction) = @_;
     my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$P1});
     my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$P2});
     my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$P3});

     my $code = drawAngleArc($P1, $P2, $P3, $radius, $inout, $direction);

     my ($xqp, $yqp) = pointOnLine($x2, $y2, $x1, $y1, $radius);
     my ($deltax, $deltay) = ($x1 - $x2, $y1 - $y2);
     my $AL;

     if ($xunits =~ /mm/) {
       $AL = 1;
     }
     elsif ($xunits =~ /cm/) {
       $AL = 0.1;
     }
     elsif ($xunits =~ /pt/) {
       $AL = 2.845;
     }
     elsif ($xunits =~ /bp/) {
       $AL = 2.835;
     }
     elsif ($xunits =~ /pc/) {
       $AL = 0.2371;
     }
     elsif ($xunits =~ /in/) {
       $AL = 0.03937;
     }
     elsif ($xunits =~ /dd/) {
       $AL = 2.659;
     }
     elsif ($xunits =~ /cc/) {
       $AL = 0.2216;
     }
     elsif ($xunits =~ /sp/) {
       $AL = 186467.98;
     }
     my $halfAL = $AL / 2;
     my $d = sqrt($radius * $radius - $halfAL * $halfAL);
     my $alpha = atan2($d / $halfAL, 1) * R2D;
     my $beta = 2 * (90 - $alpha);
     my $thetaqr;
     if (abs($deltay) < 0.00001) {
       if ($deltax > 0 ) {$thetaqr = 0 }
       elsif ($deltax < 0) {$thetaqr = -180}
     }
     else {
       if (abs($deltax) < 0.00001) {
          $thetaqr = 90;
       }
       else {
         $thetaqr = atan2($deltay / $deltax, 1) * R2D;
       }
     }
     my ($xqr, $yqr) = pointOnLine($x2, $y2, $x3, $y3, $radius);
     $deltax = $x3 - $x2;
     $deltay = $y3 - $y2;
     $alpha = atan2(sqrt($radius * $radius - $halfAL * $halfAL) / $halfAL, 1) /
              D2R;
     $beta = 2 * (90 - $alpha);
     LINE2 : {
       if (abs($deltax) < 0.00001) {
         if ($deltay > 0) { $thetaqr = 90 }
         elsif ($deltay < 0) { $thetaqr = - 90 }
         last LINE2;
       }
       else {
         $thetaqr = atan2($deltay / $deltax, 1) * R2D;
       }
       if (abs($deltay) < 0.00001) {
         if ($deltax > 0)    { $thetaqr = 0 }
         elsif ($deltax < 0) { $thetaqr = -180 }
         last LINE2;
       }
       else {
         $thetaqr = atan2($deltay / $deltax, 1) * R2D;
       }
       if ($deltax < 0 and $deltay > 0) { $thetaqr += 180 }
       elsif ($deltax < 0 and $deltay < 0) { $thetaqr += 180 }
       elsif ($deltax > 0 and $deltay < 0) { $thetaqr += 360 }
     }
     my $xqrleft  = $x2 + $radius * cos(($thetaqr + $beta) * D2R);
     my $yqrleft  = $y2 + $radius * sin(($thetaqr + $beta) * D2R);
     my $xqrright = $x2 + $radius * cos(($thetaqr - $beta) * D2R);
     my $yqrright = $y2 + $radius * sin(($thetaqr - $beta) * D2R);
     if ($inout eq "internal" and $direction eq "clockwise") {
       $code .= sprintf "\\arrow <1.5mm> [0.5, 1] from %.5f %.5f to %.5f %.5f\n",
                $xqrleft, $yqrleft, $xqr, $yqr;
     }
     elsif ($inout eq "internal" and $direction eq "anticlockwise") {
       $code .= sprintf "\\arrow <1.5mm> [0.5, 1] from %.5f %.5f to %.5f %.5f\n",
                $xqrright, $yqrright, $xqr, $yqr;
     }
     elsif ($inout eq "external" and $direction eq "clockwise") {
       $code .= sprintf "\\arrow <1.5mm> [0.5, 1] from %.5f %.5f to %.5f %.5f\n",
                $xqrleft, $yqrleft, $xqr, $yqr;
     }
     elsif ($inout eq "external" and $direction eq "anticlockwise") {
       $code .= sprintf "\\arrow <1.5mm> [0.5, 1] from %.5f %.5f to %.5f %.5f\n",
                $xqrright, $yqrright, $xqr, $yqr;
     }
     return $code;
  }

  sub expr {
    my $lc = $_[0];
    my($left,$op,$right);

    $left = term($lc);
    while ($op = addop()) {
      $right = term($lc);
      if ($op eq '+')
        { $left += $right }
      else
        { $left -= $right }
    }
    return $left;
  }

  sub addop {
    s/^([+-])// && $1;
  }

  sub term {
    my $lc = $_[0];
    my ($left, $op, $right);
    $left = factor($lc);
    while ($op = mulop()) {
      $right = factor($lc);
      if ($op eq '*')
        { $left *= $right }
      elsif ($op =~ /rem/i) {
        eval {$left %= $right};
        PrintFatalError("Division by zero", $lc) if $@;
      }
      else {
        eval {$left /= $right};
        PrintFatalError("Division by zero", $lc) if $@;
      }
    }
    return $left;
  }

  sub mulop {
    (s#^([*/])## || s/^(rem)//i) && lc($1);
  }

  sub factor {
    my $lc = $_[0];
    my ($left);

    $left = primitive($lc);
    if (s/^\*\*//) {
      $left **= factor($lc);
    }
    return $left;
  }

  sub primitive {
    my $lc = $_[0];
    my $val;
    s/\s*//;
    if (s/^\(//) {  #is it an expr in parentheses
      $val = expr($lc);
      s/^\)// || PrintErrorMessage("Missing right parenthesis", $lc);
    }
    elsif (s/^-//) { # is it a negated primitive
       $val = - primitive();
    }
    elsif (s/^\+//) { # is it a positive primitive
       $val = primitive();
    }
    elsif (s/^angledeg//i) {
       chk_lparen("angledeg",$lc);
       my $point_1 = get_point($lc);
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
       my $point_2 = get_point($lc);
       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
       my $point_3 = get_point($lc);
       my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point_3});
       my $d12 = Length($x1, $y1, $x2, $y2);
       my $d23 = Length($x2, $y2, $x3, $y3);
       my $d31 = Length($x3, $y3, $x1, $y1);
       if ( $d12 == 0 ) {
         PrintErrorMessage("points `$point_1' and `$point_2' are the same", $lc);
         $val = 0;
       }
       elsif ( $d23 == 0 ) {
         PrintErrorMessage("points `$point_2' and `$point_3' are the same", $lc);
         $val = 0;
       }
       elsif ( $d31 == 0 ) {
         PrintErrorMessage("points `$point_1' and `$point_3' are the same", $lc);
         $val = 0;
       }
       else {  
         $val = Angle($x1, $y1, $x2, $y2, $x3, $y3);
         $val = 0 if $val == -500;
       }
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^angle//i) {
       chk_lparen("angle",$lc);
       my $point_1 = get_point($lc);
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
       my $point_2 = get_point($lc);
       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
       my $point_3 = get_point($lc);
       my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point_3});
       my $d12 = Length($x1, $y1, $x2, $y2);
       my $d23 = Length($x2, $y2, $x3, $y3);
       my $d31 = Length($x3, $y3, $x1, $y1);
       if ( $d12 == 0 ) {
         PrintErrorMessage("points `$point_1' and `$point_2' are the same", $lc);
         $val = 0;
       }
       elsif ( $d23 == 0 ) {
         PrintErrorMessage("points `$point_2' and `$point_3' are the same", $lc);
         $val = 0;
       }
       elsif ( $d31 == 0 ) {
         PrintErrorMessage("points `$point_1' and `$point_3' are the same", $lc);
         $val = 0;
       }
       else {
         $val =  Angle($x1, $y1, $x2, $y2, $x3, $y3);
         if ($val == -500) {
           $val = 0;
         }
         else {
           $val = D2R * $val;
         }
       }
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^area//i) {
       chk_lparen("angledeg",$lc);
       my $point_1 = get_point($lc);
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
       my $point_2 = get_point($lc);
       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
       my $point_3 = get_point($lc);
       my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point_3});
       $val = triangleArea($x1, $y1, $x2, $y2, $x3, $y3);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^asin//i) {
       chk_lparen("asin");
       $val = expr();
       PrintFatalError("Can't take asin of $val", $lc) if $val < -1 || $val > 1;
       $val = asin($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^acos//i) {
       chk_lparen("acos");
       $val = expr();
       PrintFatalError("Can't take acos of $val", $lc) if $val < -1 || $val > 1 ;
       $val = acos($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^atan//i) {
       chk_lparen("atan");
       $val = expr();
       $val = atan($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^cos//i) {
       chk_lparen("cos");
       $val = expr();
       $val = cos($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^directiondeg//i) {
       chk_lparen("directiondeg",$lc);
       my $point_1 = get_point($lc);
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
       my $point_2 = get_point($lc);
       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
       my $x3 = $x1+1;
       if ( ($y2 - $y1) >= 0) {
         $val = Angle($x3, $y1, $x1, $y1, $x2, $y2);
         $val = 0 if $val == -500;
       }
       else {
         $val = 360 - Angle($x3, $y1, $x1, $y1, $x2, $y2);
         $val = 0 if $val == -500;  
       }
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^direction//i) {
       chk_lparen("direction",$lc);
       my $point_1 = get_point($lc);
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
       my $point_2 = get_point($lc);
       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
       my $x3 = $x1+1;
       if ( ($y2 - $y1) >= 0) {
         $val = Angle($x3, $y1, $x1, $y1, $x2, $y2);
         $val = 0 if $val == -500;
         $val = D2R * $val;
       }
       else {
         $val = 360 - Angle($x3, $y1, $x1, $y1, $x2, $y2);
         $val = 0 if $val == -500;
         $val = D2R * $val;  
       }
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^exp//i) {
       chk_lparen("exp");
       $val = expr();
       $val = exp($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^int//i) {
       chk_lparen("int");
       $val = expr();
       $val = int($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^log//i) {
       chk_lparen("log");
       $val = expr();
       PrintFatalError("Can't take log of $val", $lc) if $val <= 0;
       $val = log($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^sin//i) {
       chk_lparen("sin");
       $val = expr();
       $val = sin($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^sgn//i) {
       chk_lparen("sgn");
       $val = expr();
       if ($val > 0) { 
         $val = 1;
       }
       elsif ($val == 0) {
         $val = 0;
       }
       else {
         $val = -1;
       }
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^sqrt//i) {
       chk_lparen("sqrt");
       $val = expr();
       $val = sqrt($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^tan//i) {
       chk_lparen("tan");
       $val = expr();
       $val = sin($val)/cos($val);
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^xcoord//i) {
       chk_lparen("xcoord");
       my $point_name = get_point;
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_name});
       $val = $x1;
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^ycoord//i) {
       chk_lparen("ycoord");
       my $point_name = get_point;
       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_name});
       $val = $y1;
       chk_rparen("Missing right parenthesis", $lc);
    }
    elsif (s/^_pi_//i) {
       $val = PI;
    }
    elsif (s/^_e_//i) {
       $val = 2.71828182845905;
    }
    elsif (s/^_linethickness_//i) {
       $val = $LineThickness / $xunits;
    }
    else  {
      my $err_code;
      ($val,$err_code) = ComputeDist($lc);
    }
    s/\s*//;
    return $val;
  }

  sub memberOf {
     my $elem = shift(@_);

     my $found = 0;
     foreach $item (@_){
         if ($item eq $elem){
            $found = 1;
            last;
         }
     }
     return $found;
  }

  sub midpoint {
    my ($x1, $y1, $x2, $y2)=@_;
    return ($x1 + ($x2 - $x1)/2, 
            $y1 + ($y2 - $y1)/2);
  }


  sub tand {
    my $d = $_[0];
    $d = $d * PI / 180;
    return sin($d)/cos($d);
  }

  sub get_string {
    my $string = shift;
    my $lc = shift;

    $string =~ s/^\s+//;
    if ($string !~ s/^\"//) {
      PrintErrorMessage("No starting \" found",$lc);
      return (1,$string,$string);
    }
    my @ch = split //,$string;
    my @cmd;
    while (@ch and $ch[0] ne "\"") {
      if ($ch[0] eq "\\" and (defined $ch[1]) and $ch[1] eq "\"") {
         shift @ch;
         push @cmd, $ch[0];
         shift @ch;
      }
      else {
        push @cmd, $ch[0];
        shift @ch;
      }
    }
    if (! defined $ch[0]) {
       PrintErrorMessage("No closing \" found",$lc);
       return (1,join("",@cmd), join("",@ch))
    }
    else {
      shift @ch;
      return (0, join("",@cmd), join("",@ch))
    }
  }

  sub is_tainted {
    my $arg = shift;
    my $nada = substr($arg,0,0);
    local $@;
    eval { eval "# $nada"};
    return length($@) != 0;
  }

  sub noOfDigits {
    my $num = $_[0];

    if ($num =~ /^[\+-]?\d+(?!\.)/) {
      return 0;
    }
    elsif ($num =~ /^[\+-]\d+\.(\d+)?/) {
      return length($1);
    }
  }

  sub drawsquare {
     my $s = $_[0];
     #$s *= sqrt(2);
     $s = sprintf "%.5f", $s;
     my $code = "\\setlength{\\unitlength}{$xunits}%\n";
     $code .= "\\begin{picture}($s,$s)\\put(0,0)" .
              "{\\framebox($s,$s){}}\\end{picture}";
     return $code;
  }

  sub X2sp {
     my $LT = shift;
     my $units = shift;

     if ($units eq "pc") {
       return $LT * 786432;
     }
     elsif ($units eq "pt") {
       return $LT * 65536;
     }
     elsif ($units eq "in") {
       return $LT * 4736286.72;
     }
     elsif ($units eq "bp") {
       return $LT * 65781.76;
     }
     elsif ($units eq "cm") {
       return $LT * 1864679.811023622;
     }
     elsif ($units eq "mm") {
       return $LT * 186467.981102362;
     }
     elsif ($units eq "dd") {
       return $LT * 70124.086430424;
     }
     elsif ($units eq "cc") {
       return $LT * 841489.037165082;
     }
     elsif ($units eq "sp") {
       return $LT;
     }
  }


  sub sp2X {
     my $LT = shift;
     my $units = shift;

     if ($units eq "pc") {
       return $LT / 786432;
     }
     elsif ($units eq "pt") {
       return $LT / 65536;
     }
     elsif ($units eq "in") {
       return $LT / 4736286.72;
     }
     elsif ($units eq "bp") {
       return $LT / 65781.76;
     }
     elsif ($units eq "cm") {
       return $LT / 1864679.811023622;
     }
     elsif ($units eq "mm") {
       return $LT / 186467.981102362;
     }
     elsif ($units eq "dd") {
       return $LT / 70124.086430424;
     }
     elsif ($units eq "cc") {
       return $LT / 841489.037165082;
     }
     elsif ($units eq "sp") {
       return $LT;
     }
  }

  sub setLineThickness {
    my $Xunits = shift;
    my $LT = shift;
    $Xunits =~ s/^((\+|-)?\d+(\.\d+)?([eE](\+|-)?\d+)?)//;
    my $xlength = "$1";
    $Xunits =~ s/^\s*($units)//;
    my $x_in_units = $1;
    $LT =~ s/^((\+|-)?\d+(\.\d+)?([eE](\+|-)?\d+)?)//;
    my $LTlength = "$1";
    $LT =~ s/^\s*($units)//;
    my $LT_in_units = $1;
    $LTlength = X2sp($LTlength,$LT_in_units);
    $LTlength = sp2X($LTlength,$x_in_units);
    return $LTlength;
  }

    sub process_input {
      my ($INFILE,$currInFile) = @_;
      my $lc = 0;
      my $no_output = 0;
      $curr_in_file = $currInFile;
      LINE: while(<$INFILE>) {
        $lc++;
        chomp($command = $_);
        s/^\s+//;
        if (/^beginSkip\s*/i) {
          $no_output = 1;
          print OUT "%%$_" if $comments_on;
          next LINE;
        }
        elsif (/^endSkip\s*/i) {
          if ($no_output == 0) {
            PrintErrorMessage("endSkip without beginSkip",$lc);
          }
          else {
            $no_output = 0;
          }
          print OUT "%%$_" if $comments_on and !$no_output;
          next LINE;
       }
       elsif ($no_output == 1) {
         next LINE;
       }
       else {
          if (/^[^\\]/) {
            my $out_line  = mpp($command,$lc) unless /^\\/;  #call macro pre-processor
            $_ = "$out_line\n";
          }
         
           if (/^\s*%/)
           {
              print OUT "$_" if $comments_on;
           }
           elsif (s/^\s*(beginloop(?=\W))//i) {
             s/\s+//;
             my $times = expr($lc);
             print OUT "%% BEGINLOOP $times\n" if $comments_on;
             my @C = ();
             REPEATCOMMS: while (<$INFILE>) {
               if (/^\s*endloop/i) {
                 last REPEATCOMMS;
               }
               else {
                 push @C, $_;
               }
             }
             if (! /^\s*endloop/i) {
               PrintFatalError("unexpected end of file",$lc);
             }
             else {
               s/^\s*endloop//i;
               for(my $i=1; $i<=$times; $i++) {
                 tie *DUMMY, 'DummyFH', \@C;
                 process_input(DUMMY, $currInFile);
                 untie *DUMMY;
               }
               print OUT "%% ENDLOOP\n" if $comments_on;
             }
           }
           elsif (s/^\s*(ArrowShape(?=\W))//i)
           {
               my $cmd = $1;
               print OUT "%% $cmd$_" if $comments_on;
              
                      chk_lparen("$cmd",$lc);
                      if (s/^default//i) {
                        $arrowLength = 2;
                        $arrowLengthUnits = "mm";
                        $arrowAngleB = 30;
                        $arrowAngleC = 40;
                      }
                      else {
                        my ($LocalArrowLength, $LocalArrowAngleB ,$LocalArrowAngleC) = (0,0,0);
                        $LocalArrowLength = expr($lc);
                        if (s/^\s*($units)//i) {
                          $arrowLengthUnits = "$1";
                        }
                        else {
                          $xunits =~ /(\d+(\.\d+)?)\s*($units)/;
                          $LocalArrowLength *= $1;
                          $arrowLengthUnits = "$3";
                        }
                        chk_comma($lc);
                        $LocalArrowAngleB = expr($lc);
                        chk_comma($lc);
                        $LocalArrowAngleC = expr($lc);
                        $arrowLength = ($LocalArrowLength == 0 ? 2  : $LocalArrowLength);
                        $arrowLengthUnits = ($LocalArrowLength == 0 ? "mm" : $arrowLengthUnits);
                        $arrowAngleB = ($LocalArrowAngleB == 0 ? 30 : $LocalArrowAngleB);
                        $arrowAngleC = ($LocalArrowAngleC == 0 ? 40 : $LocalArrowAngleC);
                      }
                      chk_rparen("after $cmd arguments",$lc);
                      chk_comment("after $cmd command",$lc);
                      print OUT "%% arrowLength = $arrowLength$arrowLengthUnits, ",
                                "arrowAngleB = $arrowAngleB ",
                                "and arrowAngleC = $arrowAngleC\n" if $comments_on;

           }
           elsif (s/^\s*(const(?=\W))//i)
           {
              print OUT "%% $1$_" if $comments_on;
                  do{
                    s/\s*//;
                    PrintErrorMessage("no identifier found after token const",$lc)
                      if $_ !~ s/^([^\W\d_]\d{0,4})//i;
                    my $Constname = $1;
                    my $constname = lc($Constname);
                    if (exists $ConstTable{$constname}) {
                      PrintErrorMessage("Redefinition of constant $constname",$lc);
                    }
                    s/\s*//; #remove leading white space
                    PrintErrorMessage("did not find expected = sign",$lc)
                      if $_ !~ s/^[=]//i;
                    my $val = expr($lc);
                    $VarTable{$constname} = $val;
                    $ConstTable{$constname} = 1;
                    print OUT "%% $Constname = $val\n" if $comments_on;
                  }while (s/^,//);
                  chk_comment($lc);
                  s/\s*//;
                  if (/^[^%]/) {
                    PrintWarningMessage("Trailing text is ignored",$lc);
                  }

           }
           elsif (s/^\s*(dasharray(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                    chk_lparen($cmd,$lc);
                    my @DashArray = ();
                    my $dash = "";
                    my $dashpattern = "";
                    PATTERN: while (1) {
                      $dash = sprintf("%.5f", expr($lc));
                      if (s/^\s*($units)//i) {
                         push (@DashArray, "$dash$1");
                      }
                      else {
                        PrintErrorMessage("Did not found unit after expression", $lc);
                      }
                      s/\s*//;
                      if (/^[^,]/) {
                        last PATTERN;
                      }
                      else {
                        s/^,\s*//;
                      }
                    }
                    print OUT "\\setdashpattern <";
                    while (@DashArray) {
                      $dashpattern .= shift @DashArray;
                      $dashpattern .= ",";
                    }
                    $dashpattern =~ s/,$//;
                    print OUT $dashpattern, ">\n";
                    chk_rparen("arguments of $cmd",$lc);
                    chk_comment($lc);

           }
           elsif (s/^\s*(drawAngleArc(?=\W))//i or s/^\s*(drawAngleArrow(?=\W))//i )
           {
              my $cmd = $1;
              print OUT "%% $cmd$_" if $comments_on;
              
                    chk_lcb($cmd,$lc);
                          my ($P1, $P2, $P3);
                          if (s/^angle(?=\W)//i) {
                            chk_lparen("token angle of command $cmd",$lc);
                            $P1 = get_point($lc);
                            next LINE if $P1 eq "_undef_";
                            $P2 = get_point($lc);
                            next LINE if $P2 eq "_undef_";
                            $P3 = get_point($lc);
                            next LINE if $P3 eq "_undef_";
                            my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$P1});
                            my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$P2});
                            my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$P3});
                            my $Angle = Angle($x1, $y1, $x2, $y2, $x3, $y3);
                            if ($Angle <= 0) {
                              if ($Angle == 0) {
                                PrintErrorMessage("Angle is equal to zero",$lc);
                                next LINE;
                              }
                              elsif ($Angle < -400) {
                                PrintErrorMessage("Something is wrong with the points",$lc);
                                next LINE;
                              }
                            }
                            chk_rparen("angle part of command $cmd",$lc);
                          }
                          else {
                            PrintErrorMessage("Did not find expected angle part",$lc);
                            next LINE;
                          }

                    s/^,\s*// or s/\s*//; #parse optional comma
                    
                         my $radius;
                         if (s/^radius(?=\W)//i) {
                           chk_lparen("token radius of command $cmd",$lc);
                           $radius = expr($lc);
                           chk_rparen("radius part of command $cmd",$lc);
                         }
                         else {
                           PrintErrorMessage("Did not found expected angle part",$lc);
                           next LINE;
                         }

                    s/^,\s*// or s/\s*//; #parse optional comma
                    my $inout = "";
                    if (s/^(internal(?=\W))//i or s/^(external(?=\W))//i) {
                      $inout = $1;
                    }
                    else {
                      PrintErrorMessage("Did not find expected 'internal' specifier", $lc);
                      next LINE;
                    }
                    s/^,\s*// or s/\s*//; #parse optional comma
                    my $direction = "";
                    if (s/^(clockwise(?=\W))//i or s/^(anticlockwise(?=\W))//i) {
                      $direction = $1;
                    }
                    else {
                      PrintErrorMessage("Did not find expected 'direction' specifier", $lc);
                      next LINE;
                    }
                    chk_rcb("arguments of $cmd",$lc);
                    chk_comment($lc);
                    my $code;
                    if (lc($cmd) eq "drawanglearc") {
                      $code = drawAngleArc($P1, $P2, $P3, $radius, $inout, $direction);
                    }
                    else {
                      $code = drawAngleArrow($P1, $P2, $P3, $radius, $inout, $direction);
                    }
                    print OUT $code if $code ne "";

           }
           elsif (s/^\s*(drawArrow(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              DrawLineOrArrow(0,$lc);
           }
           elsif (s/^\s*(drawcircle(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                    chk_lparen("drawcircle",$lc);
                    my $Point = get_point($lc);
                    next LINE if $Point eq "_undef_";
                    chk_comma($lc);
                    my $R = expr($lc);
                    if ($R <= 0) {
                       PrintErrorMessage("Radius must be greater than zero",$lc);
                       next LINE;
                    }
                    my ($x,$y,$pSV,$pS)=unpack("d3A*",$PointTable{lc($Point)});
                    printf OUT "\\circulararc 360 degrees from %.5f %.5f center at %.5f %.5f\n",
                                $x+$R, $y, $x, $y;
                    chk_rparen("arguments of $cmd",$lc);
                    chk_comment($lc);

           }
           elsif (s/^\s*(drawcurve(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              DrawLineOrArrow(2,$lc);
           }
           elsif (s/^\s*(drawcircumcircle(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                    chk_lparen("drawcircumcircle",$lc);
                    my $point1 = get_point($lc);
                    next LINE if $point1 eq "_undef_";
                    my $point2 = get_point($lc);
                    next LINE if $point2 eq "_undef_";
                    my $point3 = get_point($lc);
                    next LINE if $point3 eq "_undef_";
                    my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point1});
                    my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point2});
                    my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point3});
                    my ($xc, $yc,$r) = circumCircleCenter($x1,$y1,$x2,$y2,$x3,$y3,$lc);
                    next LINE if $xc == 0 and  $yc == 0 and $r == 0;
                    print OUT "%% circumcircle center = ($xc,$yc), radius = $r\n" if $comments_on;
                    printf OUT "\\circulararc 360 degrees from %.5f %.5f center at %.5f %.5f\n",
                                $xc+$r, $yc, $xc, $yc;
                    chk_rparen("arguments of $cmd",$lc);
                    chk_comment($lc);

           }
           elsif (s/^\s*(drawexcircle(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                   chk_lparen("drawexcircle",$lc);
                   my $point1 = get_point($lc);
                   next LINE if $point1 eq "_undef_";
                   my $point2 = get_point($lc);
                   next LINE if $point2 eq "_undef_";
                   my $point3 = get_point($lc);
                   next LINE if $point3 eq "_undef_";
                   my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point1});
                   my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point2});
                   my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point3});
                   if (triangleArea($x1, $y1, $x2, $y2, $x3, $y3) < 0.0001) {
                     PrintErrorMessage("Area of triangle is zero!",$lc);
                     next LINE;
                   }
                   chk_comma($lc);
                   my $point4 = get_point($lc);
                   if (!memberOf($point4, $point1, $point2, $point3)) {
                     PrintErrorMessage("Current point isn't a side point",$lc);
                     next LINE;
                   }
                   next LINE if $point4 eq "_undef_";
                   my $point5 = get_point($lc);
                   next LINE if $point5 eq "_undef_";
                   if (!memberOf($point5, $point1, $point2, $point3)) {
                     PrintErrorMessage("Current point isn't a side point",$lc);
                     next LINE;
                   }
                   if ($point4 eq $point5) {
                     PrintErrorMessage("Side points are identical",$lc);
                     next LINE;
                   }
                   chk_rparen("arguments of $cmd",$lc);
                   my ($xc, $yc, $r) = excircle($point1, $point2, $point3,
                                                $point4, $point5);
                   my $R=$r;
                   if (s/^\s*\[\s*//) {
                      $R += expr($lc);
                      if ($R < 0.0001) {
                        PrintErrorMessage("Radius has become equal to zero!",$lc);
                        next LINE;
                      }
                      chk_rsb($lc);
                   }
                   if ($R > (500 / 2.845)) {
                     PrintErrorMessage("Radius is greater than 175mm!",$lc);
                     next LINE;
                   }
                   print OUT "%% excircle center = ($xc,$yc) radius = $R\n" if $comments_on;
                   printf OUT "\\circulararc 360 degrees from %.5f %.5f center at %.5f %.5f\n",
                                $xc+$R, $yc, $xc, $yc;
                   chk_comment($lc);

           }
           elsif (s/^\s*(drawincircle(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                   chk_lparen("drawincircle",$lc);
                   my $point1 = get_point($lc);
                   next LINE if $point1 eq "_undef_";
                   my $point2 = get_point($lc);
                   next LINE if $point2 eq "_undef_";
                   my $point3 = get_point($lc);
                   next LINE if $point3 eq "_undef_";
                   my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point1});
                   my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point2});
                   my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point3});
                   if (triangleArea($x1, $y1, $x2, $y2, $x3, $y3) < 0.0001) {
                     PrintErrorMessage("Area of triangle is zero!",$lc);
                     next LINE;
                   }
                   my ($xc, $yc, $r) = IncircleCenter($x1,$y1,$x2,$y2,$x3,$y3);
                   my $R=$r;
                   if (s/^\s*\[\s*//) {
                      $R += expr($lc);
                      if ($R < 0.0001) {
                        PrintErrorMessage("Radius has become equal to zero!",$lc);
                        next LINE;
                      }
                      chk_rsb($lc);
                   }
                   if ($R > (500 / 2.845)) {
                     PrintErrorMessage("Radius is greater than 175mm!",$lc);
                     next LINE;
                   }
                   print OUT "%% incircle center = ($xc,$yc) radius = $R\n" if $comments_on;
                   printf OUT "\\circulararc 360 degrees from %.5f %.5f center at %.5f %.5f\n",
                                $xc+$R, $yc, $xc, $yc;
                   chk_rparen("arguments of $cmd",$lc);
                   chk_comment($lc);

           }
           elsif (s/^\s*(drawline(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              DrawLineOrArrow(1,$lc);
           }
           elsif (s/^\s*(drawthickarrow(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              print OUT "\\setplotsymbol  ({\\usefont{OT1}{cmr}{m}{n}\\large .})%\n";
              print OUT "{\\setbox1=\\hbox{\\usefont{OT1}{cmr}{m}{n}\\large .}%\n";
              print OUT " \\global\\linethickness=0.31\\wd1}%\n";
              DrawLineOrArrow(0,$lc);
              print OUT "\\setlength{\\linethickness}{0.4pt}%\n";
              print OUT "\\setplotsymbol  ({\\usefont{OT1}{cmr}{m}{n}\\tiny .})%\n";
           }
           elsif (s/^\s*(drawthickline(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              print OUT "\\setplotsymbol  ({\\usefont{OT1}{cmr}{m}{n}\\large .})%\n";
              print OUT "{\\setbox1=\\hbox{\\usefont{OT1}{cmr}{m}{n}\\large .}%\n";
              print OUT " \\global\\linethickness=0.31\\wd1}%\n";
              DrawLineOrArrow(1,$lc);
              print OUT "\\setlength{\\linethickness}{0.4pt}%\n";
              print OUT "\\setplotsymbol  ({\\usefont{OT1}{cmr}{m}{n}\\tiny .})%\n";
           }
           elsif (s/^\s*(drawperpendicular(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              
                    chk_lparen($cmd,$lc);
                    my $A = get_point($lc);
                    next LINE if $A eq "_undef_";
                    chk_comma($lc);
                    my $B = get_point($lc);
                    next LINE if $A eq "_undef_";
                    s/\s*//; #ignore white space
                    my $C = get_point($lc);
                    next LINE if $A eq "_undef_";
                    chk_rparen("arguments of $cmd",$lc);
                    chk_comment($lc);
                    #
                    #start actual computation
                    #
                    my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$A});
                    my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$B});
                    my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$C});
                    my ($xF, $yF) = perpendicular($x1, $y1, $x2, $y2, $x3, $y3);
                    printf OUT "\\plot  %.5f %.5f    %.5f %.5f  /\n",
                           $x1, $y1, $xF, $yF;

           }
           elsif (s/^\s*(drawpoint(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                    my ($stacklen);
                    chk_lparen("$cmd",$lc);
                    if (/^\)/) {
                      PrintErrorMessage("There are no point to draw",$lc);
                      next LINE;
                    }
                    my(@PP);
                    DRAWPOINTS:while(1) {
                      if (s/^([^\W\d_]\d{0,4})//i) { #point name
                        $P = $1;
                        if (!exists($PointTable{lc($P)})) {
                          PrintErrorMessage("Undefined point $P",$lc);
                          next DRAWPOINTS;
                        }
                        else {
                          push (@PP,$P);
                          s/\s*//;
                        }
                      }
                      else {
                        last DRAWPOINTS;
                      }
                    }
                    drawpoints(@PP);
                    chk_rparen("arguments of $cmd",$lc);
                    chk_comment($lc);

           }
           elsif (s/^\s*(drawRightAngle(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              
                   chk_lparen("drawRightAngle",$lc);
                   my $point1 = get_point($lc);
                   next LINE if $point1 eq "_undef_";
                   my $point2 = get_point($lc);
                   next LINE if $point2 eq "_undef_";
                   my $point3 = get_point($lc);
                   next LINE if $point3 eq "_undef_";
                   my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point1});
                   my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point2});
                   my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point3});
                   chk_comma($lc);
                   my $dist = expr($lc);
                   chk_rparen("arguments of $cmd",$lc);
                   chk_comment($lc);
                   #
                   #actual computation
                   #
                   my ($Px, $Py) = pointOnLine($x2, $y2, $x1, $y1, $dist);
                   my ($Qx, $Qy) = pointOnLine($x2, $y2, $x3, $y3, $dist);
                   my ($Tx, $Ty) = midpoint($Px, $Py, $Qx, $Qy);
                   my ($Ux, $Uy) = pointOnLine($x2, $y2, $Tx, $Ty, 2*Length($x2, $y2, $Tx, $Ty));
                   if ($Px == $Ux || $Py == $Uy) {
                     printf OUT "\\putrule from %.5f %.5f to %.5f %.5f \n", $Px,$Py,$Ux,$Uy;
                   }
                   else {
                     printf OUT "\\plot %.5f %.5f\t%.5f %.5f / \n", $Px, $Py,$Ux,$Uy;
                   }
                   if ($Ux == $Qx || $Uy == $Qy) {
                     printf OUT "\\putrule from %.5f %.5f to %.5f %.5f \n", $Ux,$Uy,$Qx,$Qy;
                   }
                   else {
                     printf OUT "\\plot %.5f %.5f\t%.5f %.5f / \n", $Ux, $Uy,$Qx,$Qy;
                   }


           }
           elsif (s/^\s*(drawsquare(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                 chk_lparen("drawSquare",$lc);
                 my $p = get_point($lc);
                 chk_comma($lc);
                 my $side = expr($lc);
                 $side = $side - (1.1 * $LineThickness/$xunits); #Suggested by RWDN
                 my ($x,$y,$pSV,$pS) = unpack("d3A*",$PointTable{$p});
                 printf OUT "\\put {%s} at %.5f %.5f %%drawsquare\n", drawsquare($side), $x, $y;
                 chk_rparen("arguments of $cmd",$lc);
                 chk_comment($lc);

           }
           elsif (s/^\s*inputfile\*//i)
           {
                   chk_lparen("inputfile*",$lc);
                   my $row_in = "";
                   if (s/^((\w|-|\.)+)//) {
                     $row_in = $1;
                   }
                   else {
                     PrintErrorMessage("No input file name found",$lc);
                     next LINE;
                   }
                   if (!(-e $row_in)) {
                     PrintErrorMessage("File $row_in does not exist",$lc);
                     next LINE;
                   }
                   open(ROW, "$row_in")|| die "Can't open file $row_in\n";
                   while (defined($in_line=<ROW>)) { print OUT $in_line; }
                   print OUT "%% ... end of input file <$row_in>\n";
                   close ROW;
                   chk_rparen("input file name",$lc);
                   chk_comment($lc);


           }
           elsif (s/^\s*(inputfile(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
              
                   chk_lparen("inputfile",$lc);
                   my $comm_in = "";
                   if (s/^((\w|-|\.)+)//) {
                     $comm_in = $1;
                   }
                   else {
                     PrintErrorMessage("No input file name found",$lc);
                     next LINE;
                   }
                   if (!(-e $comm_in)) {
                     PrintErrorMessage("File $comm_in does not exist",$lc);
                     next LINE;
                   }
                   chk_rparen("input file name",$lc);
                   my $input_times = 1; #default value
                   if (s/^\[//) {
                     $input_times = expr($lc);       
                     chk_rsb("optional argument",$lc);
                   }
                   print OUT "%% ... start of file <$comm_in> loop [$input_times]\n";
                   for (my $i=0; $i<int($input_times); $i++) {
                     open(COMM,"$comm_in") or die "Can't open file $comm_in\n";
                     print OUT "%%% Iteration number: ",$i+1,"\n";
                     my $old_file_name = $curr_in_file;
                     process_input(COMM,"File $comm_in, ");
                     $curr_in_file = $old_file_name;
                     close COMM;
                   }
                   print OUT "%% ... end of file <$comm_in> loop [$input_times]\n";
                   chk_comment($lc);

           }
           elsif (s/^\s*(linethickness(?=\W))//i)
           {
              my $cmd = $1;
              print OUT "%% $cmd$_" if $comments_on;
                   chk_lparen("linethickness", $lc);
                   if (s/^default//i) {
                     print OUT "\\linethickness=0.4pt\\Linethickness{0.4pt}%%\n";
                     print OUT "\\setplotsymbol  ({\\usefont{OT1}{cmr}{m}{n}\\tiny .})%\n";
                     $LineThickness = setLineThickness($xunits,"0.4pt");
                   }
                   else {
                     my $length = expr($lc);
                     if (s/^\s*($units)//i) {
                       my $units = $1;
                       printf OUT "\\linethickness=%.5f%s\\Linethickness{%.5f%s}%%\n",
                              $length, $units, $length, $units;
                       $LineThickness = setLineThickness($xunits,"$length$units");
                       my $mag;
                       if ($units eq "pc") {
                         $mag = $length * 12;
                       }
                       elsif ($units eq "in") {
                         $mag = $length * 72.27;
                       }
                       elsif ($units eq "bp") {
                         $mag = $length * 1.00375;
                       }
                       elsif ($units eq "cm") {
                         $mag = $length * 28.45275;
                       }
                       elsif ($units eq "mm") {
                         $mag = $length * 2.845275;
                       }
                       elsif ($units eq "dd") {
                         $mag = $length * 1.07001;
                       }
                       elsif ($units eq "cc") {
                         $mag = $length * 0.08917;
                       }
                       elsif ($units eq "sp") {
                         $mag = $length * 0.000015259;
                       }
                       elsif ($units eq "pt") {
                         $mag = $length;
                       }
                       $mag = 10 * $mag / 1.00278219;
                       printf OUT "\\font\\CM=cmr10 at %.5fpt%%\n", $mag;
                       print OUT "\\setplotsymbol  ({\\CM .})%\n";
                     }
                     else {
                       PrintErrorMessage("Did not found expect units part",$lc);
                     }
                   }
                   chk_rparen("linethickness", $lc);
                   chk_comment($lc);


           }
           elsif (s/^\s*(paper(?=\W))//i)
           {
              my ($cmd) = $1;
              print OUT "%% $cmd$_" if $comments_on;
                   chk_lcb("paper", $lc);
                   if (s/^units(?=\W)//i)
                   {
                      
                              chk_lparen("units",$lc);
                              if(s/^\)//)
                              {
                                PrintWarningMessage("Missing value in \"units\"--default is 1pt",
                                                       $lc);
                                $xunits = "1pt";          
                              }
                              else {
                                $xunits = expr($lc);
                                s/\s*//;
                                if (s/^($units)//i) {
                                  $xunits .= "$1";
                                  $LineThickness = setLineThickness($xunits,"0.4pt");
                                }
                                elsif(s/^(\w)+//i) {
                                  PrintErrorMessage("$1 is not a valid mathspic unit",$lc);
                                  $xunits = "1pt";
                                }
                                else {
                                  PrintErrorMessage("No x-units found",$lc);
                                  $xunits = "1pt";
                                }
                                s/\s*//; #ignore white space
                                if (s/^,//) {  # there is a comma so expect an y-units
                                  s/\s*//; #ignore white space
                                  $yunits = expr($lc);
                                  s/\s*//; #ignore white space
                                  if (s/^($units)//i) {
                                    $yunits .= "$1";
                                  }
                                  elsif(s/^(\w)+//i) {
                                    PrintErrorMessage("$1 is not a valid mathspic unit",$lc);
                                    $yunits = "1pt";
                                  }
                                  else {
                                    PrintErrorMessage("No y-units found",$lc);
                                    $yunits = $xunits;
                                  }
                                }
                                else {
                                  $yunits = $xunits;
                                }
                                chk_rparen("units",$lc);
                              }

                      $nounits = 0;
                   }
                   else
                   {
                      $nounits = 1;
                   }
                   s/^,\s*// or s/\s*//;
                   if (s/^xrange//i)
                   {
                      
                              chk_lparen("xrange",$lc);
                              my $ec;
                              ($xlow,$ec) = ComputeDist($lc);
                              next LINE if $ec == 0;
                              chk_comma($lc);
                              ($xhigh,$ec) = ComputeDist($lc);
                              next LINE if $ec == 0;
                              if ($xlow >= $xhigh)
                              {
                                 PrintErrorMessage("xlow >= xhigh in xrange",$lc);
                                 next LINE;
                              }
                              chk_rparen("$xhigh",$lc);

                      $noxrange = 0;
                   }
                   else
                   {
                      $noxrange = 1;
                   }
                   s/^,\s*// or s/\s*//;
                   if (s/^yrange//i)
                   {
                      
                              chk_lparen("yrange",$lc);
                              my $ec;
                              ($ylow,$ec) = ComputeDist($lc);
                              next LINE if $ec == 0;
                              chk_comma($lc);
                              ($yhigh,$ec) = ComputeDist($lc);
                              next LINE if $ec == 0;
                              if ($ylow >= $yhigh)
                              {
                                 PrintErrorMessage("ylow >= yhigh in yrange",$lc);
                                 next LINE;
                              }
                              chk_rparen("$yhigh",$lc);

                      $noyrange = 0;
                   }
                   else
                   {
                      $noyrange = 1;
                   }
                   
                       if (!$nounits)
                       {
                          printf OUT "\\setcoordinatesystem units <%s,%s>\n",
                                     $xunits,$yunits;
                       }
                       if(!$noxrange && !$noyrange)
                       {
                          printf OUT "\\setplotarea x from %.5f to %.5f, y from %.5f to %.5f\n",
                                     $xlow, $xhigh, $ylow, $yhigh;

                       }

                   s/^,\s*// or s/\s*//;
                   $axis = "";
                   if (s/^ax[ei]s(?=\W)//i)
                   {
                      
                              chk_lparen("axis",$lc);
                              while(/^[^\)]/)
                              {
                                  if (s/^([lrtbxy]{1}\*?)//i)
                                  {
                                     $axis .= $1;
                                  }
                                  elsif (s/^([^lrtbxy])//i)
                                  {
                                     PrintErrorMessage("Non-valid character \"$1\" in axis()",$lc);
                                  }
                                  s/\s*//;
                              }
                              chk_rparen("axis(arguments",$lc);

                   }
                   $axis = uc($axis);
                   s/^,\s*// or s/\s*//;
                   if (s/^ticks(?=\W)//i)
                   {
                              chk_lparen("ticks",$lc);
                              my $ec;
                              ($xticks,$ec) = ComputeDist($lc);
                              next LINE if $ec == 0;
                              chk_comma($lc);
                              ($yticks,$ec) = ComputeDist($lc);
                              next LINE if $ec == 0;
                              chk_rparen("ticks(arguments",$lc);

                   }
                   else
                   {
                      $xticks = $yticks = 0;
                   }
                   chk_rcb("paper", $lc);
                   YBRANCH: {
                      if (index($axis, "Y")>-1)
                      {
                         if (index($axis, "Y*")>-1)
                         {
                            print OUT "\\axis left shiftedto x=0 / \n";
                            last YBRANCH;
                         }
                         if ($yticks > 0)
                         {
                            if (index($axis, "T")>-1 && index($axis, "B")==-1)
                            {
                               print OUT "\\axis left shiftedto x=0 ticks numbered from ";
                               print OUT "$ylow to -$yticks by $yticks\n      from $yticks to ";
                               print OUT $yhigh-$yticks," by $yticks /\n";
                            }
                            elsif (index($axis, "T")==-1 && index($axis, "B")>-1)
                            {
                               print OUT "\\axis left shiftedto x=0 ticks numbered from ";
                               print OUT $ylow+$yticks," to -$yticks by $yticks\n      from ";
                               print OUT "$yticks to $yhigh by $yticks /\n";
                            }
                            elsif (index($axis, "T")>-1 && index($axis, "B")>-1)
                            {
                               print OUT "\\axis left shiftedto x=0 ticks numbered from ";
                               print OUT $ylow+$yticks," to -$yticks by $yticks\n      from ";
                               print OUT "$yticks to ",$yhigh-$yticks," by $yticks /\n";
                            }
                            else
                            {
                               print OUT "\\axis left shiftedto x=0 ticks numbered from ";
                               print OUT "$ylow to -$yticks by $yticks\n      from ";
                               print OUT "$yticks to $yhigh by $yticks /\n";
                            }
                         }
                         else
                         {
                            print OUT "\\axis left shiftedto x=0 /\n";
                         }
                      }
                      }
                      XBRANCH: { if (index($axis, "X")>-1)
                      {
                         if (index($axis, "X*")>-1)
                         {
                            print OUT "\\axis bottom shiftedto y=0 /\n";
                            last XBRANCH;
                         }
                         if ($xticks > 0)
                         {
                            if (index($axis, "L")>-1 && index($axis, "R")==1)
                            {
                               print OUT "\\axis bottom shiftedto y=0 ticks numbered from ";
                               print OUT $xlow + $xticks," to -$xticks by $xticks\n      from";
                               print OUT " $xticks to $xhigh by $xticks /\n";
                            }
                            elsif (index($axis, "L")==-1 && index($axis, "R")>-1)
                            {
                               print OUT "\\axis bottom shiftedto y=0 ticks numbered from ";
                               print OUT "$xlow to -$xticks by $xticks\n      from ";
                               print OUT "$xticks to ",$xhigh-$xticks," by $xticks /\n";
                            }
                            elsif (index($axis, "L")>-1 && index($axis, "R")>-1)
                            {
                               print OUT "\\axis bottom shiftedto y=0 ticks numbered from ";
                               print OUT $xlow + $xticks," to -$xticks by $xticks\n      from ";
                               print OUT "$xticks to ",$xhigh - $xticks," by $xticks /\n";
                            }
                            else
                            {
                               print OUT "\\axis bottom shiftedto y=0 ticks numbered from ";
                               print OUT "$xlow to -$xticks by  $xticks\n      from ";
                               print OUT "$xticks to $xhigh by $xticks /\n";
                            }
                         }
                         else
                         {
                            print OUT "\\axis bottom shiftedto y=0 /\n";
                         }
                      } }
                      LBRANCH: {if (index($axis, "L")>-1)
                      {
                         if (index($axis, "L")>-1)
                         {
                            if (index($axis, "L*")>-1)
                            {
                               print OUT "\\axis left /\n";
                               last LBRANCH;
                            }
                            if ($yticks > 0)
                            {
                               print OUT "\\axis left ticks numbered from ";
                               print OUT "$ylow to $yhigh by $yticks /\n";
                            }
                            else
                            {
                               print OUT "\\axis left /\n";
                            }
                         }
                      } }
                      RBRANCH: { if (index($axis, "R")>-1)
                      {
                         if (index($axis, "R*")>-1)
                         {
                            print OUT "\\axis right /\n";
                            last RBRANCH;
                         }
                         if ($yticks > 0)
                         {
                            print OUT "\\axis right ticks numbered from $ylow to $yhigh by ";
                            print OUT "$yticks /\n";
                         }
                         else
                         {
                            print OUT "\\axis right /\n";
                         }
                      } }
                      TBRANCH: { if (index($axis, "T")>-1)
                      {
                         if (index($axis, "T*")>-1)
                         {
                            print OUT "\\axis top /\n";
                            last TBRANCH;
                         }
                         if ($xticks > 0)
                         {
                            print OUT "\\axis top ticks numbered from $xlow to $xhigh by ";
                            print OUT "$xticks /\n";
                         }
                         else
                         {
                            print OUT "\\axis top /\n";
                         }
                      } }
                      BBRANCH: { if (index($axis, "B")>-1)
                      {
                         if (index($axis, "B*")>-1)
                         {
                            print OUT "\\axis bottom /\n";
                            last BBRANCH;
                         }
                         if ($xticks > 0)
                         {
                            print OUT "\\axis bottom ticks numbered from $xlow to $xhigh by ";
                            print OUT "$xticks /\n";
                         }
                         else
                         {
                            print OUT "\\axis bottom /\n";
                         }
                      } }


                   chk_comment($lc);

           }
           elsif (s/^\s*(PointSymbol(?=\W))//i)
           {
               my $cmd = $1;
               print OUT "%% $cmd$_" if $comments_on;
              
                      chk_lparen("$cmd",$lc);
                      if (s/^default//i) #default point symbol
                      {
                         $defaultsymbol = "\$\\bullet\$";
                      }
                      elsif (s/^(circle|square)//i) {
                        $defaultsymbol = $1;
                        chk_lparen($defaultsymbol, $lc);
                        $GlobalDimOfPoints = expr($lc);
                        chk_rparen("expression", $lc);          
                      }
                      elsif (s/^(((\\,){1}|(\\\)){1}|(\\\s){1}|[^\),\s])+)//) #arbitrary LaTeX point
                      {
                         $defaultsymbol = $1;
                         $defaultsymbol=~ s/\\\)/\)/g;
                         $defaultsymbol=~ s/\\,/,/g;
                         $defaultsymbol=~ s/\\ / /g;
                      }
                      else
                      {
                         PrintErrorMessage("unrecognized point symbol",$lc);
                      }
                      if (s/\s*,\s*//) {
                        $defaultLFradius = expr($lc);
                      }
                      chk_rparen("after $cmd arguments",$lc);
                      chk_comment("after $cmd command",$lc);

           }
           elsif (s/^\s*point(?=\W)//i)
           {
              my ($Point_Line);
              chomp($Point_Line=$_);
                   my ($pointStar, $PointName, $origPN);
                   $pointStar = 0; # default value: he have a point command
                   $pointStar = 1 if s/^\*//;
                   chk_lparen("point" . (($pointStar)?"*":""),$lc);
                   if (s/^([^\W\d_](?![^\W\d_])\d{0,4})//i) {
                   #
                   # Note: the regular expression (foo)(?!bar) means that we are
                   # looking a foo not followed by a bar. Moreover, the regular
                   # expression [^\W\d_] means that we are looking for letter.
                   #
                     $origPN = $1;
                     $PointName = lc($1);
                   }
                   else {
                     PrintErrorMessage("Invalid point name",$lc);
                     next LINE;
                   }
                   #if ($pointStar and !exists($PointTable{$PointName})) {
                   #  PrintWarningMessage("Point $origPN has not been defined",$lc);       
                   #}
                   if (!$pointStar and exists($PointTable{$PointName})) {
                      PrintWarningMessage("Point $origPN has been used already",$lc);
                   }
                   chk_rparen("point" . (($pointStar)?"*":""). "($origPN",$lc);
                   chk_lcb("point" . (($pointStar)?"*":""). "($origPN)",$lc);
                   my ($Px, $Py);
                          if (s/^perpendicular(?=\W)//i) {
                                      chk_lparen("perpendicular",$lc);
                                      my $FirstPoint = &get_point($lc);
                                      next LINE if $FirstPoint eq "_undef_";
                                      chk_comma($lc);
                                      my $SecondPoint = &get_point($lc);
                                      next LINE if $SecondPoint eq "_undef_";
                                      my $ThirdPoint = &get_point($lc);
                                      next LINE if $ThirdPoint eq "_undef_";
                                      chk_rparen("No closing parenthesis found",$lc);
                                      my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$FirstPoint});
                                      my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$SecondPoint});
                                      my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$ThirdPoint});
                                      ($Px, $Py) = perpendicular($x1,$y1,$x2,$y2,$x3,$y3);

                          }
                          elsif (s/^intersection(?=\W)//i) {
                                       chk_lparen("intersection",$lc);
                                       my $FirstPoint = get_point($lc);
                                       next LINE if $FirstPoint eq "_undef_";
                                       my $SecondPoint = get_point($lc);
                                       next LINE if $SecondPoint eq "_undef_";
                                       chk_comma($lc);
                                       my $ThirdPoint = get_point($lc);
                                       next LINE if $ThirdPoint eq "_undef_";
                                       my $ForthPoint = get_point($lc);
                                       next LINE if $ForthPoint eq "_undef_";
                                       chk_rparen("No closing parenthesis found",$lc);
                                       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$FirstPoint});
                                       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$SecondPoint});
                                       my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$ThirdPoint});
                                       my ($x4,$y4,$pSV4,$pS4)=unpack("d3A*",$PointTable{$ForthPoint});
                                       ($Px, $Py) = intersection4points($x1,$y1,$x2,$y2,$x3,$y3,$x4,$y4);


                          }
                          elsif (s/^midpoint(?=\W)//i) {
                                      chk_lparen("midpoint",$lc);
                                      my $FirstPoint = &get_point($lc);
                                      next LINE if $FirstPoint eq "_undef_";
                                      my $SecondPoint = &get_point($lc);
                                      next LINE if $SecondPoint eq "_undef_";
                                      chk_rparen("No closing parenthesis found",$lc);
                                      my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$FirstPoint});
                                      my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$SecondPoint});
                                      ($Px, $Py) = midpoint($x1, $y1, $x2, $y2);

                          }
                          elsif (s/^pointonline(?=\W)//i) {
                                      chk_lparen("pointonline",$lc);
                                      my $FirstPoint = &get_point($lc);
                                      next LINE if $FirstPoint eq "_undef_";
                                      my $SecondPoint = &get_point($lc);
                                      next LINE if $SecondPoint eq "_undef_";
                                      chk_comma($lc);
                                      # now get the distance
                                      my $distance = expr($lc);
                                      chk_rparen("No closing parenthesis found",$lc);
                                      my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$FirstPoint});
                                      my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$SecondPoint});
                                      ($Px, $Py) = pointOnLine($x1,$y1,$x2,$y2,$distance);

                          }
                          elsif (s/^circumcircleCenter(?=\W)//i) {
                                       chk_lparen("circumCircleCenter",$lc);
                                       my $FirstPoint = &get_point($lc);
                                       next LINE if $FirstPoint eq "_undef_";
                                       my $SecondPoint = &get_point($lc);
                                       next LINE if $SecondPoint eq "_undef_";
                                       my $ThirdPoint = &get_point($lc);
                                       next LINE if $ThirdPoint eq "_undef_";
                                       chk_rparen("No closing parenthesis found",$lc);
                                       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$FirstPoint});
                                       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$SecondPoint});
                                       my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$ThirdPoint});
                                       ($Px, $Py,$r) = &circumCircleCenter($x1,$y1,$x2,$y2,$x3,$y3,$lc);
                                       next LINE if $Px == 0 and $Py == 0 and $r == 0;

                          }
                          elsif (s/^IncircleCenter(?=\W)//i) {
                                       chk_lparen("IncircleCenter",$lc);
                                       my $FirstPoint = &get_point($lc);
                                       next LINE if $FirstPoint eq "_undef_";
                                       my $SecondPoint = &get_point($lc);
                                       next LINE if $SecondPoint eq "_undef_";
                                       my $ThirdPoint = &get_point($lc);
                                       next LINE if $ThirdPoint eq "_undef_";
                                       chk_rparen("No closing parenthesis found",$lc);
                                       my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$FirstPoint});
                                       my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$SecondPoint});
                                       my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$ThirdPoint});
                                       ($Px, $Py, $r) = IncircleCenter($x1,$y1,$x2,$y2,$x3,$y3);

                          }
                          elsif (s/^ExcircleCenter(?=\W)//i) {
                                       chk_lparen("ExcircleCenter",$lc);
                                       my $PointA = get_point($lc);
                                       next LINE if $PointA eq "_undef_";
                                       my $PointB = get_point($lc);
                                       next LINE if $PointB eq "_undef_";
                                       my $PointC = get_point($lc);
                                       next LINE if $PointC eq "_undef_";
                                       chk_comma($lc);
                                       my $PointD = &get_point($lc);
                                       next LINE if $PointD eq "_undef_";
                                       if (!memberOf($PointD, $PointA, $PointB, $PointC)) {
                                         PrintErrorMessage("Current point isn't a side point",$lc);
                                         next LINE;
                                       }
                                       my $PointE = get_point($lc);
                                       next LINE if $PointE eq "_undef_";
                                       if (!memberOf($PointE, $PointA, $PointB, $PointC)) {
                                         PrintErrorMessage("Current point isn't a side point",$lc);
                                         next LINE;
                                       }
                                       if ($PointD eq $PointE) {
                                         PrintErrorMessage("Side points are identical",$lc);
                                         next LINE;
                                       }
                                       ($Px, $Py, $r) = excircle($PointA, $PointB, $PointC,
                                                                  $PointD, $PointE);
                                       chk_rparen("after coordinates part",$lc);

                          }
                          elsif (/^[^\W\d_]\d{0,4}\s*[^,\w]/) {
                            m/^([^\W\d_]\d{0,4})\s*/i;
                            if (exists($PointTable{lc($1)})) {         
                              my $Tcoord = get_point($lc);
                              my ($x,$y,$pSV,$pS)=unpack("d3A*",$PointTable{$Tcoord});
                              $Px = $x;
                              $Py = $y;
                            }
                            else {
                              $Px = expr();
                              chk_comma($lc);
                              $Py = expr();
                            }
                          }
                          elsif (/[^\W\d_]\d{0,4}\s*,\s*shift|polar|rotate|vector/i) { #a point?
                            s/^([^\W\d_]\d{0,4})//i;
                            my $PointA = $1;
                            if (exists($PointTable{lc($PointA)})) {
                              s/\s*//;
                              if (s/^,//) {
                                s/\s*//;
                                if (s/^shift(?=\W)//i) {
                                  
                                            chk_lparen("shift",$lc);
                                            my $dist1 = expr($lc);
                                            chk_comma($lc);
                                            my $dist2 = expr($lc);
                                            my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{lc($PointA)});
                                            $Px = $x1 + $dist1;
                                            $Py = $y1 + $dist2;
                                            chk_rparen("shift part",$lc);

                                }
                                elsif (s/^polar(?=\W)//i) {
                                            chk_lparen("polar",$lc);
                                            my ($R1, $Theta1);
                                            $R1 = expr($lc);
                                            chk_comma($lc);
                                            $Theta1 = expr($lc);
                                            my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{lc($PointA)});
                                            s/\s*//;
                                            if (s/^rad(?=\W)//i) {
                                               # do nothing!
                                            }
                                            elsif (s/^deg(?=\W)//i) {
                                              $Theta1 = $Theta1 * PI / 180;
                                            }
                                            else {
                                              #$Theta1 = $Theta1 * PI / 180; 
                                            }
                                            $Px = $x1 + $R1 * cos($Theta1);
                                            $Py = $y1 + $R1 * sin($Theta1);
                                            chk_rparen("after polar part",$lc);

                                }
                                elsif (s/^rotate(?=\W)//i) {
                                            chk_lparen("rotate",$lc);
                                            my $Q = lc($PointA);
                                            my $P = get_point($lc);
                                            next LINE if $P eq "_undef_";
                                            chk_comma($lc);
                                            my $Theta1 = expr($lc);
                                            my ($xP,$yP,$pSV1,$pS1)=unpack("d3A*",$PointTable{$P});
                                            my ($xQ,$yQ,$pSV2,$pS2)=unpack("d3A*",$PointTable{$Q});
                                            s/\s*//;
                                            if (s/^rad(?=\W)//i)
                                            {
                                               # do nothing!
                                            }
                                            elsif (s/^deg(?=\W)//i)
                                            {
                                                $Theta1 = $Theta1 * PI / 180;
                                            }
                                            else
                                            {
                                                $Theta1 = $Theta1 * PI / 180;
                                            }
                                            # shift origin to P
                                            $xQ -= $xP;
                                            $yQ -= $yP;
                                            # do the rotation
                                            $Px = $xQ * cos($Theta1) - $yQ * sin($Theta1);
                                            $Py = $xQ * sin($Theta1) + $yQ * cos($Theta1);
                                            # return origin back to original origin
                                            $Px += $xP;
                                            $Py += $yP;
                                            chk_rparen("after rotate part",$lc);

                                }
                                elsif (s/^vector(?=\W)//i) {
                                            chk_lparen("vector",$lc);
                                            my ($x0,$y0,$pSV0,$pS0) = unpack("d3A*",$PointTable{lc($PointA)});
                                            my $P = get_point($lc);
                                            my $Q = get_point($lc);
                                            my ($x1,$y1,$pSV1,$pS1) = unpack("d3A*",$PointTable{$P});
                                            my ($x2,$y2,$pSV2,$pS2) = unpack("d3A*",$PointTable{$Q});
                                            $Px = $x0 + $x2 - $x1;
                                            $Py = $y0 + $y2 - $y1;
                                            chk_rparen("vector part",$lc);


                                }
                                else {
                                  PrintErrorMessage("unexpected token",$lc);
                                  next LINE;
                                }
                              }
                              else {
                                my ($xA,$yA,$pSVA,$pSA)=unpack("d3A*",$PointTable{lc($PointA)});
                                $Px = $xA;
                                $Py = $yA;
                              }
                            }
                            else {
                              PrintErrorMessage("Undefined point $PointA",$lc);
                              next LINE;
                            }
                          }
                          else {
                            $Px = expr();
                            chk_comma($lc);
                            $Py = expr();
                          }

                   chk_rcb("coordinates part",$lc);
                   my $sv = $defaultsymbol;
                   my $sh = $defaultLFradius;
                   my $side_or_radius = undef;
                   if (s/^\[\s*//) { # the user has opted to specify the optional part
                                if (/^(symbol|radius|side)\s*/i) {
                                  my @previous_options = ();
                                  my $number_of_options = 1;
                                  my $symbol_set = 0;        
                                  while (s/^(symbol|radius)\s*//i and $number_of_options <= 2) {
                                    my $option = lc($1);
                                    if (s/^=\s*//) {
                                      if (memberOf($option,@previous_options)) {
                                        PrintErrorMessage("Option \"$option\" has been already defined", $lc);
                                        my $dummy = expr($lc);
                                      } 
                                      elsif ($option eq "radius") {
                                        $sh = expr($lc);
                                        $sv = $defaultsymbol if ! $symbol_set;
                                      }
                                      elsif ($option eq "symbol") {
                                        if (s/^circle\s*//i) {
                                          $sv = "circle";
                                          chk_lparen("after token circle",$lc);
                                          $side_or_radius = expr($lc);
                                          chk_rparen("expression",$lc);
                                        }
                                        elsif (s/^square\s*//i) {
                                          $sv = "square";
                                          chk_lparen("after token square",$lc);
                                          $side_or_radius = expr($lc);
                                          chk_rparen("expression",$lc);
                                        }
                                        elsif (s/^(((\\\]){1}|(\\,){1}|(\\\s){1}|[^\],\s])+)//) {
                                          $sv = $1;
                                          $sv =~ s/\\\]/\]/g;
                                          $sv =~ s/\\,/,/g;
                                          $sv =~ s/\\ / /g;
                                          s/\s*//;
                                        }
                                        $symbol_set = 1;
                                      }
                                    }
                                    else {
                                      PrintErrorMessage("unexpected token", $lc);
                                      next LINE;
                                    }
                                    $number_of_options++;
                                    push (@previous_options, $option);
                                    s/^,\s*//;
                                  }
                                }
                                else {
                                  PrintErrorMessage("unexpected token", $lc);
                                  next LINE;
                                }

                      chk_rsb("optional part",$lc);
                   }
                   # to avoid truncation problems introduced by the pack function, we
                   # round each number up to five decimal digits
                   $Px = sprintf("%.5f", $Px);
                   $Py = sprintf("%.5f", $Py);
                   print OUT "%% point$Point_Line \t$origPN = ($Px, $Py)\n" if $comments_on;
                   chk_comment($lc);
                   $PointTable{$PointName} = pack("d3A*",$Px,$Py,$sh,$sv);
                   if (defined($side_or_radius)) {
                     $DimOfPoint{$PointName} = $side_or_radius;
                   } 

           }
           elsif (/^\s*setPointNumber(?=\W)/i)
           {
              PrintWarningMessage("Command setPointNumber is ignored",$lc);
              next LINE;
           }
           elsif (s/^\s*(showAngle(?=\W))//i)
           {
                     chk_lparen("showangle",$lc);
                     my $point_1 = get_point($lc);
                     my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
                     my $point_2 = get_point($lc);
                     my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
                     my $point_3 = get_point($lc);
                     my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point_3});
                     my $angle = Angle($x1, $y1, $x2, $y2, $x3, $y3);
                     $angle = 0 if $angle == -500;
                     printf OUT "%%%% angle(%s%s%s) = %.5f deg ( %.5f rad)\n", $point_1,
                            $point_2, $point_3, $angle, $angle*D2R;
                     chk_rparen("Missing right parenthesis", $lc);

           }
           elsif (s/^\s*(showArea(?=\W))//i)
           {
                     chk_lparen("showarea",$lc);
                     my $point_1 = get_point($lc);
                     my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
                     my $point_2 = get_point($lc);
                     my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
                     my $point_3 = get_point($lc);
                     my ($x3,$y3,$pSV3,$pS3)=unpack("d3A*",$PointTable{$point_3});
                     print OUT "%% area($point_1$point_2$point_3) = ",
                           triangleArea($x1, $y1, $x2, $y2, $x3, $y3), "\n";
                     chk_rparen("Missing right parenthesis", $lc);

           }
           elsif (s/^\s*(showLength(?=\W))//i)
           {
                     chk_lparen("showlength",$lc);
                     my $point_1 = get_point($lc);
                     my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{$point_1});
                     my $point_2 = get_point($lc);
                     my ($x2,$y2,$pSV2,$pS2)=unpack("d3A*",$PointTable{$point_2});
                     print OUT "%% length($point_1$point_2) = ",
                           Length($x1, $y1, $x2, $y2), "\n";
                     chk_rparen("Missing right parenthesis", $lc);


           }
           elsif (/^\s*showPoints(?=\W)/i)
           {
              print OUT "%%-------------------------------------------------\n";
              print OUT "%%            L I S T  O F  P O I N T S            \n";
              print OUT "%%-------------------------------------------------\n";
              foreach my $p (keys(%PointTable)) {
                my ($x, $y, $pSV, $pS) = unpack("d3A*",$PointTable{$p});
                printf OUT "%%%%\t%s\t= ( %.5f, %.5f ), LF-radius = %.5f, symbol = %s\n",
                        $p, $x, $y, $pSV, $pS;
              }
              print OUT "%%-------------------------------------------------\n";
              print OUT "%%      E N D  O F  L I S T  O F  P O I N T S      \n";
              print OUT "%%-------------------------------------------------\n";
              next LINE;
           }
           elsif (/^\s*showVariables(?=\W)/i)
           {
              print OUT "%%-------------------------------------------------\n";
              print OUT "%%       L I S T  O F  V A R I A B L E S           \n";
              print OUT "%%-------------------------------------------------\n";
              foreach my $var (keys(%VarTable)) {
                print OUT "%%\t", $var, "\t=\t", $VarTable{$var}, "\n";
              }
              print OUT "%%-------------------------------------------------\n";
              print OUT "%%   E N D  O F  L I S T  O F  V A R I A B L E S   \n";
              print OUT "%%-------------------------------------------------\n";
              next LINE;
           }
           elsif (s/^\s*(system(?=\W))//i)
           {
              print OUT "%% $1$_" if $comments_on;
              
                   chk_lparen("$cmd",$lc);
                   my ($error, $command, $rest) = get_string($_);
                   next LINE if $error == 1;
                   $_ = $rest;
                   if (! is_tainted($command)) {
                      system($command);
                   }
                   else {
                      PrintErrorMessage("String \"$command\" has tainted data", $lc);
                      next LINE;
                   }
                   chk_rparen("after $cmd arguments",$lc);
                   chk_comment("after $cmd command",$lc);

           }
           elsif (s/^\s*(text(?=\W))//i)
           {
              print OUT "%% $1$_" if $comments_on;
              
                      chk_lparen("text",$lc);
                      my ($level,$text)=(1,"");
                      TEXTLOOP: while (1)
                      {
                        $level++ if /^\(/;
                        $level-- if /^\)/;
                        s/^(.)//;
                        last TEXTLOOP if $level==0;
                        $text .= $1;
                      }
                      chk_lcb("text part",$lc);
                      my ($Px, $Py,$dummy,$pos);
                      $pos="";
                      s/\s*//;
                      
                        if (/^[^\W\d_]\d{0,4}\s*[^,\w]/) {
                          my $Tcoord = get_point($lc);
                          my ($x,$y,$pSV,$pS)=unpack("d3A*",$PointTable{$Tcoord});
                          $Px = $x;
                          $Py = $y;
                        }
                        elsif (/[^\W\d_]\d{0,4}\s*,\s*shift|polar/i) {
                          s/^([^\W\d_]\d{0,4})//i;
                          my $PointA = $1;
                          if (exists($PointTable{lc($PointA)})) {
                            s/\s*//;
                            if (s/^,//) {
                              s/\s*//;
                              if (s/^shift(?=\W)//i) {
                                
                                          chk_lparen("shift",$lc);
                                          my $dist1 = expr($lc);
                                          chk_comma($lc);
                                          my $dist2 = expr($lc);
                                          my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{lc($PointA)});
                                          $Px = $x1 + $dist1;
                                          $Py = $y1 + $dist2;
                                          chk_rparen("shift part",$lc);

                              }
                              elsif (s/^polar(?=\W)//i) {
                                          chk_lparen("polar",$lc);
                                          my ($R1, $Theta1);
                                          $R1 = expr($lc);
                                          chk_comma($lc);
                                          $Theta1 = expr($lc);
                                          my ($x1,$y1,$pSV1,$pS1)=unpack("d3A*",$PointTable{lc($PointA)});
                                          s/\s*//;
                                          if (s/^rad(?=\W)//i) {
                                             # do nothing!
                                          }
                                          elsif (s/^deg(?=\W)//i) {
                                            $Theta1 = $Theta1 * PI / 180;
                                          }
                                          else {
                                            #$Theta1 = $Theta1 * PI / 180; 
                                          }
                                          $Px = $x1 + $R1 * cos($Theta1);
                                          $Py = $y1 + $R1 * sin($Theta1);
                                          chk_rparen("after polar part",$lc);

                              }
                            }
                          }
                          else {
                            PrintErrorMessage("undefined point/var",$lc);
                            next LINE;
                          }
                        }
                        else {
                          $Px = expr();
                          chk_comma($lc);
                          $Py = expr();
                        }

                      chk_rcb("coordinates part of text command",$lc);
                      if (s/^\[//)
                      {
                        s/\s*//;
                        
                          if (s/^(\w{1})\s*//) {
                            $pos .= $1;
                            if (memberOf($pos, "l", "r")) {
                              if (s/^(\w{1})\s*//) {
                                my $np = $1;
                                if (memberOf($np, "t", "b", "B")) {
                                  $pos .= $np;
                                }
                                else {
                                  if (memberOf($np, "l", "r")) {
                                     PrintErrorMessage("$np can't follow 'l' or 'r'", $lc);
                                  }
                                  else {
                                    PrintErrorMessage("$np is not a valid positioning option", $lc);
                                  }
                                  next LINE;
                                }
                              }
                            }
                            elsif (memberOf($pos, "t", "b", "B")) {
                              if (s/^(\w{1})\s*//) {
                                my $np = $1;
                                if (memberOf($np, "l", "r")) {
                                  $pos .= $np;
                                }
                                else {
                                  if (memberOf($np, "t", "b", "B")) {
                                     PrintErrorMessage("$np can't follow 't', 'b', or 'B'", $lc);
                                  }
                                  else {
                                    PrintErrorMessage("$np is not a valid positioning option", $lc);
                                  }
                                  next LINE;
                                }
                              }
                            }
                            else {
                              PrintErrorMessage("$pos is not a valid positioning option", $lc);
                              next LINE;
                            }
                          }
                          else {
                            PrintErrorMessage("illegal token in optional part of text command",$lc);
                            next LINE;
                          }

                        s/\s*//;
                        chk_rsb("optional part of text command",$lc);
                      }
                      chk_comment($lc);
                      if ($pos eq "")
                      {
                         printf OUT "\\put {%s} at %f %f\n", $text, $Px, $Py;
                      }
                      else
                      {
                         printf OUT "\\put {%s} [%s] at %f %f\n", $text, $pos, $Px, $Py;
                      }

           }
           elsif (s/^\s*(var(?=\W))//i)
           {
              print OUT "%% $1$_" if $comments_on;
                  do{
                    s/\s*//;
                    PrintErrorMessage("no identifier found after token var",$lc)
                      if $_ !~ s/^([^\W\d_]\d{0,4})//i;
                    my $Varname = $1;
                    my $varname = lc($Varname);
                    if (exists $ConstTable{$varname}) {
                      PrintErrorMessage("Redefinition of constant $varname",$lc);
                    }
                    s/\s*//; #remove leading white space
                    PrintErrorMessage("did not find expected = sign",$lc)
                      if $_ !~ s/^[=]//i;
                    my $val = expr($lc);
                    $VarTable{$varname} = $val;
                    print OUT "%% $Varname = $val\n" if $comments_on;
                  }while (s/^,//);
                  chk_comment($lc);
                  s/\s*//;
                  if (/^[^%]/) {
                    PrintWarningMessage("Trailing text is ignored",$lc);
                  }
           }
           elsif (/^\s*\\(.+)/)
           {
              my $line = $1;
              if ($line =~ /^\s+(.+)/)
              {
                 print OUT " $line\n";
              }
              else
              {
                print OUT "\\$line\n";
              }
              next LINE;
           }
           elsif (0==length) #empty line
           {
              next LINE;
           }
           else {
             PrintErrorMessage("command not recognized",$lc);
             next LINE;
           }

       }
     }
   }


our $alarm="";
our $comments_on=1;
our $out_file="default";
our $argc=@ARGV;
if ($argc == 0 || $argc > 5 ){ # no command line arguments or more than 4
                               # arguments
  die "\nmathspic version $version_number\n" .
      "Usage: mathspic [-h] [-b] [-c] [-o <out file>] <in file>\n\n";
}
else {
     our $file = "";
     SWITCHES:
     while($_ = $ARGV[0]) {
       shift;     
       if (/^-h$/) {
         die "\nThis is mathspic version $version_number\n" .
             "Type \"man mathspic\" for detailed help\n".
             "Usage:\tmathspic  [-h] [-b] [-c] [-o <out file>] <in file>\n" . 
             "\twhere,\n" . 
             "\t[-b]\tenables bell sound if error exists\n" .
             "\t[-c]\tdisables comments in output file\n" .
             "\t[-h]\tgives this help listing\n" .
             "\t[-o]\tcreates specified output file\n\n"; 
       }
       elsif (/^-b$/) {
         $alarm = chr(7);
       }
       elsif (/^-c$/) {
         $comments_on = 0;
       }
       elsif (/^-o$/) {
         die "No output file specified!\n" if !@ARGV;
         $out_file = $ARGV[0];
         shift;
       }
       elsif (/^-\w+/) {
         die "$_: Illegal command line switch!\n";
       }
       else {
         $file = $_;
       }
     }my ($xA, $yA, $xB, $yB, $dist)=@_;
     die "No input file specified!\n" if $file eq "";

  print "This is mathspic version $version_number\n";
}
   our ($source_file, $log_file);
   if (! -e $file) {
      die "$file: no such file!\n" if (! (-e "$file.m"));
      $source_file = "$file.m";
   }
   else {
      $source_file = $file;
      $file = $1 if $file =~ /(\w[\w-\.]+)\.\w+/;
   }
   $out_file= "$file.mt" if $out_file eq "default";
   $log_file= "$file.mlg";


  open(IN,"$source_file")||die "Can't open source file: $source_file\n";
  open(OUT,">$out_file")||die "Can't open output file: $out_file\n";
  open(LOG,">$log_file")||die "Can't open log file: $log_file\n";
  print_headers;
  process_input(IN,"");

print $alarm if $no_errors > 0;
__END__