Blame test/mpi/maint/testmerge.in

Packit 0848f5
#! @PERL@
Packit 0848f5
# -*- Mode: Perl; -*-
Packit 0848f5
#
Packit 0848f5
#  (C) 2004 by Argonne National Laboratory.
Packit 0848f5
#      See COPYRIGHT in top-level directory.
Packit 0848f5
#
Packit 0848f5
#
Packit 0848f5
# This file provides a way to merge a template file with a set of
Packit 0848f5
# code fragments.  This makes it simple to generate many related tests
Packit 0848f5
# from a single test harness, while ending up with relatively simple code
Packit 0848f5
# in case it is necessary to debug the code.
Packit 0848f5
# The template file uses an XML-like notation to mark off places for code.
Packit 0848f5
# Specifically, lines of the form
Packit 0848f5
#    <name/>
Packit 0848f5
# are replaced from a definition file.  The template is XML-like because
Packit 0848f5
# angle brackets and ampersands can be freely used as long as they don't
Packit 0848f5
# match the perl pattern <\w*\/> (which no valid C, C++, or Fortran code will)
Packit 0848f5
#
Packit 0848f5
# The file of definitions has the following form, also in an XML-like 
Packit 0848f5
# format (for simple parsing)
Packit 0848f5
#    <TESTDEFN filename="fname">
Packit 0848f5
#    <blockname>
Packit 0848f5
#      definition
Packit 0848f5
#    </blockname>
Packit 0848f5
#    ...
Packit 0848f5
#    </TESTDEFN>
Packit 0848f5
# where "blockname" is an arbitrary name (matching the perl expression \w*)
Packit 0848f5
# that matches the names in the template file.  
Packit 0848f5
# TESTDEFN is a required field
Packit 0848f5
#
Packit 0848f5
# Possible extensions:
Packit 0848f5
#    Common definitions for all files (allows a common template for
Packit 0848f5
#      multiple sets of merges)
Packit 0848f5
#    Allow the *template* to define some names (eg, value)
Packit 0848f5
#      that are replaced in the generated file.
Packit 0848f5
#    
Packit 0848f5
# ----------------------------------------------------------------------------
Packit 0848f5
# Global variables
Packit 0848f5
$debug = 0;
Packit 0848f5
Packit 0848f5
$lang = "Fortran";   
Packit 0848f5
%knownLang = ( "Fortran" => 1, "C" => 1, "C++" => 1 );
Packit 0848f5
#$lang = "C";
Packit 0848f5
#$lang = "C++";
Packit 0848f5
Packit 0848f5
# Definitions
Packit 0848f5
%Definitions = ();
Packit 0848f5
# Global definitions are for all files, and can contain standard comments,
Packit 0848f5
# initializations, and other data
Packit 0848f5
%GlobalDefinitions = ();
Packit 0848f5
# Read a definition file
Packit 0848f5
# ReadDefinition( filename )
Packit 0848f5
# Places the definitions into %Definitions{name} => content
Packit 0848f5
sub ReadDefinition {
Packit 0848f5
    my $DFD = $_[0];
Packit 0848f5
Packit 0848f5
    while (<$DFD>) {
Packit 0848f5
	# Check for end-of-description
Packit 0848f5
	if (/<\/TESTDEFN>/) { last; }
Packit 0848f5
Packit 0848f5
	# match definition name
Packit 0848f5
	if (/<(\w*)>/) {
Packit 0848f5
	    my $name = $1;
Packit 0848f5
	    my $defn = "";
Packit 0848f5
	    my $found = 0;
Packit 0848f5
	    while (<$DFD>) {
Packit 0848f5
		if (/<\/$name>/) { $found = 1; last; }
Packit 0848f5
		s/\r//g;   # Remove any extraneous characters
Packit 0848f5
		$defn .= $_;
Packit 0848f5
	    }
Packit 0848f5
	    # If we didn't close the definition, generate an error message
Packit 0848f5
	    if (! $found) {
Packit 0848f5
		print STDERR "Read to end-of-file while looking for </$name>\n";
Packit 0848f5
	    }
Packit 0848f5
	    $Definitions{$name} = $defn;
Packit 0848f5
	}
Packit 0848f5
	else {
Packit 0848f5
	    # Skip (blank space, comment, etc)
Packit 0848f5
	    next;
Packit 0848f5
	}
Packit 0848f5
    }
Packit 0848f5
    
Packit 0848f5
}
Packit 0848f5
Packit 0848f5
# ReadGlobalDefinitions( filename )
Packit 0848f5
sub ReadGlobalDefinitions {
Packit 0848f5
    my $filename = $_[0];
Packit 0848f5
    # Save Definitions, if any
Packit 0848f5
    my %saveDefinitions = %Definitions;
Packit 0848f5
    # Reset Definitions to hold the current global set
Packit 0848f5
    %Definitions = %GlobalDefinitions;
Packit 0848f5
Packit 0848f5
    my $DFD = "DFD";
Packit 0848f5
    open $DFD, "<$filename" || die "Could not open global definition file $filename\n";
Packit 0848f5
    print "Opening $filename..\n";
Packit 0848f5
    &ReadDefinition( $DFD );
Packit 0848f5
    close $DFD;
Packit 0848f5
    %GlobalDefinitions = %Definitions;
Packit 0848f5
    %Definitions = %saveDefinitions;
Packit 0848f5
    # Print the new definitions if requested.
Packit 0848f5
    if ($debug) {
Packit 0848f5
	foreach my $key (keys(%GlobalDefinitions)) {
Packit 0848f5
	    my $val = $GlobalDefinitions{$key};
Packit 0848f5
	    print "$key => $val\n";
Packit 0848f5
	}
Packit 0848f5
    }
Packit 0848f5
}
Packit 0848f5
Packit 0848f5
# ---------------------------------------------------------------------------
Packit 0848f5
# MergeTemplate
Packit 0848f5
# Read a template and merge the output
Packit 0848f5
# MergeTemplate( template file, output file )
Packit 0848f5
# Preserve indentation
Packit 0848f5
sub MergeTemplate {
Packit 0848f5
    my $templateFilename = $_[0];
Packit 0848f5
    my $outputFilename   = $_[1];
Packit 0848f5
Packit 0848f5
    open IFD, "<$templateFilename" || die "Cannot open $templateFilename\n";
Packit 0848f5
    open OFD, ">$outputFilename" || die "Cannot open $outputFilename\n";
Packit 0848f5
Packit 0848f5
    while (<IFD>) {
Packit 0848f5
	s/\r//;
Packit 0848f5
	my $loopLimit = 20;
Packit 0848f5
	while (/(\s*)<(\w*)\/>/) {
Packit 0848f5
	    my $indent = $1;
Packit 0848f5
	    my $name = $2;
Packit 0848f5
	    $indent =~ s/\s*\n//g;
Packit 0848f5
	    if ($loopLimit-- <= 0) {
Packit 0848f5
		print STDERR "Exceeded loop limit while writing $outputFilename\n";
Packit 0848f5
		print STDERR "Searching for $name in $_";
Packit 0848f5
		last;
Packit 0848f5
	    }
Packit 0848f5
	    if (defined($Definitions{$name})) {
Packit 0848f5
		my $defn = $Definitions{$name};
Packit 0848f5
		my $newdefn = "";
Packit 0848f5
		# Add indentation to definition; substitute any defintions
Packit 0848f5
		foreach my $line (split(/\n/,$defn)) {
Packit 0848f5
		    print "Looking at |$line|\n" if $debug;
Packit 0848f5
		    $newdefn .= $indent . $line . "\n";
Packit 0848f5
		}
Packit 0848f5
		chop $newdefn;
Packit 0848f5
		print "Replacing <$name> with |$newdefn|\n" if $debug;
Packit 0848f5
		s/$indent<$name\/>/$newdefn/;   # Only do one at a time
Packit 0848f5
	    }
Packit 0848f5
	    elsif (defined($GlobalDefinitions{$name})) {
Packit 0848f5
		# local definitions can override any global definitions
Packit 0848f5
		my $defn = $GlobalDefinitions{$name};
Packit 0848f5
		my $newdefn = "";
Packit 0848f5
		# Add indentation to definition
Packit 0848f5
		foreach my $line (split(/\n/,$defn)) {
Packit 0848f5
		    $newdefn .= $indent . $line . "\n";
Packit 0848f5
		}
Packit 0848f5
		print "Replacing <$name> with |$newdefn|\n" if $debug;
Packit 0848f5
		chop $newdefn;
Packit 0848f5
		s/$indent<$name\/>/$newdefn/;   # Only do one at a time
Packit 0848f5
	    }
Packit 0848f5
	    else {
Packit 0848f5
		# Unknown name!
Packit 0848f5
		print STDERR "Unknown name $name in template file when creating $outputFilename!\n";
Packit 0848f5
		last;
Packit 0848f5
	    }
Packit 0848f5
	}
Packit 0848f5
	&printLine( OFD, $_ );
Packit 0848f5
    }
Packit 0848f5
    close OFD;
Packit 0848f5
    close IFD;
Packit 0848f5
}
Packit 0848f5
# ---------------------------------------------------------------------------
Packit 0848f5
# ReadAndMerge( description file, template file )
Packit 0848f5
sub ReadAndMerge {
Packit 0848f5
    my $filename = $_[0];
Packit 0848f5
    my $templateFile = $_[1];
Packit 0848f5
Packit 0848f5
    $DFD = "DFD";
Packit 0848f5
    open $DFD, "<$filename" || die "Can not open $filename\n";
Packit 0848f5
    
Packit 0848f5
    while (<$DFD>) {
Packit 0848f5
	s/#.*//;            # Remove comments
Packit 0848f5
	# Read until a TESTDEFN line
Packit 0848f5
	if (/<TESTDEFN\s+filename=\"(.*)\">/) {
Packit 0848f5
	    my $outputFile = $1;
Packit 0848f5
	    %Definitions = ();
Packit 0848f5
	    # Read until </TESTDEFN>
Packit 0848f5
	    &ReadDefinition( $DFD );
Packit 0848f5
	    # Create the merged file
Packit 0848f5
	    &MergeTemplate( $templateFile, $outputFile );
Packit 0848f5
	}
Packit 0848f5
	elsif (/<LANG>([\w\+]*)<\/LANG>/) {
Packit 0848f5
	    # Special for language definition
Packit 0848f5
	    $lang = $1;
Packit 0848f5
	    if (!defined($knownLang{$lang})) {
Packit 0848f5
		print STDERR "Unknown language $lang\n"; 
Packit 0848f5
	    }
Packit 0848f5
	}
Packit 0848f5
	elsif (/<(\w*)>/) {
Packit 0848f5
	    my $name = $1;
Packit 0848f5
	    my $defn = "";
Packit 0848f5
	    # read this as a global definition
Packit 0848f5
	    while (<$DFD>) {
Packit 0848f5
		if (/<\/$name>/) { last; }
Packit 0848f5
		s/\r//;
Packit 0848f5
		$defn .= $_;
Packit 0848f5
	    }
Packit 0848f5
	    if (eof($DFD)) { print STDERR "found EOF before end of $name\n"; }
Packit 0848f5
	    $GlobalDefinitions{$name} = $defn;
Packit 0848f5
	}
Packit 0848f5
    }
Packit 0848f5
    close $DFD;
Packit 0848f5
Packit 0848f5
}
Packit 0848f5
# --------------------------------------------------------------------------
Packit 0848f5
# Debug
Packit 0848f5
sub PrintDefinitions {
Packit 0848f5
    foreach my $name (keys(%Definitions)) {
Packit 0848f5
        print "<$name>\n";
Packit 0848f5
        my $defn = $Definitions{$name};
Packit 0848f5
        # Here we could consider doing replacement for embedded <name>...</name>,
Packit 0848f5
        # for things like arguments.
Packit 0848f5
        print $defn;
Packit 0848f5
        print "</$name>\n";
Packit 0848f5
    }
Packit 0848f5
}
Packit 0848f5
# --------------------------------------------------------------------------
Packit 0848f5
# printLine handles any continuation conventions
Packit 0848f5
# printLine ( FD, lines )
Packit 0848f5
# Note that a very simple approach works for Fortran because blanks
Packit 0848f5
 # are ignored.  However, we will try to make the code easier to read
Packit 0848f5
sub printLine {
Packit 0848f5
    my $OFD   = $_[0];
Packit 0848f5
    my $lines = $_[1];
Packit 0848f5
Packit 0848f5
    # Make sure that we get the current conventions
Packit 0848f5
    if ($lang eq "Fortran") {
Packit 0848f5
	$maxPrintLine = 72;
Packit 0848f5
	$postLine     = "  &";
Packit 0848f5
	$preLine      = "     &";
Packit 0848f5
    }
Packit 0848f5
    elsif ($lang eq "C" || $lang eq "C++") {
Packit 0848f5
	$maxPrintLine = 180;
Packit 0848f5
	$postLine     = "";
Packit 0848f5
	$preLine      = "\t";
Packit 0848f5
    }
Packit 0848f5
    
Packit 0848f5
    foreach my $line (split(/\n/,$lines)) {
Packit 0848f5
	# Compute length
Packit 0848f5
	my $length = length($line);
Packit 0848f5
	while ($length > $maxPrintLine) {
Packit 0848f5
	    # For Fortran 90 and C/C++, lines must be
Packit 0848f5
	    # broken at whitespace.  Fortran 77 ignores whitespace,
Packit 0848f5
	    my $subline = substr $line, 0, $maxPrintLine;
Packit 0848f5
	    # Now, break subline at the last non-letter
Packit 0848f5
	    if ($subline =~ /(.*)([^\w]\S*)$/) {
Packit 0848f5
		$subline = $1;
Packit 0848f5
		$line    = $2 . $line;
Packit 0848f5
		# Add blanks to end of the subline to match what was stripped
Packit 0848f5
		# off
Packit 0848f5
		my $len = length($2);
Packit 0848f5
		for (my $i=0; $i < $len; $i++) { 
Packit 0848f5
		    $subline .= " ";
Packit 0848f5
		}
Packit 0848f5
	    }
Packit 0848f5
	    print $OFD $subline . $postLine . "\n";
Packit 0848f5
Packit 0848f5
	    $line = substr $line, $maxPrintLine;
Packit 0848f5
	    $line = $preLine . $line;
Packit 0848f5
	    $length = length($line);
Packit 0848f5
	}
Packit 0848f5
        print $OFD $line . "\n";
Packit 0848f5
    }
Packit 0848f5
}
Packit 0848f5
# --------------------------------------------------------------------------
Packit 0848f5
# Process the file
Packit 0848f5
# Still to do:
Packit 0848f5
#   Allow multiple definition files, to allow for common definitions
Packit 0848f5
#   for such things as headers
Packit 0848f5
my $defnFile = "";
Packit 0848f5
my $tmplFile = "";
Packit 0848f5
my $posCount = 0;
Packit 0848f5
for (@ARGV) {
Packit 0848f5
    if (/-defn=(.*)/) {
Packit 0848f5
	my $extraFile = $1;
Packit 0848f5
	&ReadGlobalDefinitions( $extraFile );
Packit 0848f5
    }
Packit 0848f5
    elsif (/-lang=(.*)/) {
Packit 0848f5
	$lang = $1;
Packit 0848f5
	if (!defined($knownLang{$lang})) {
Packit 0848f5
	    print STDERR "Unknown language $lang\n";
Packit 0848f5
	    exit 1;
Packit 0848f5
	}
Packit 0848f5
    }
Packit 0848f5
    elsif (/^-/) {
Packit 0848f5
	print STDERR "Unrecognized argument $_\n";
Packit 0848f5
	exit 1;
Packit 0848f5
    }
Packit 0848f5
    else {
Packit 0848f5
	if ($posCount == 0) {
Packit 0848f5
	    $defnFile = $_;
Packit 0848f5
	}
Packit 0848f5
	elsif ($posCount == 1) {
Packit 0848f5
	    $tmplFile = $_;
Packit 0848f5
	}
Packit 0848f5
	else {
Packit 0848f5
	    print STDERR "Too many arguments ($_)\n";
Packit 0848f5
	    exit 1;
Packit 0848f5
	}
Packit 0848f5
	$posCount ++;
Packit 0848f5
    }
Packit 0848f5
}
Packit 0848f5
Packit 0848f5
if ($defnFile eq "" || $tmplFile eq "") {
Packit 0848f5
    print STDERR "testmerge [ -defn=name ] defintion-file template-file \n";
Packit 0848f5
    exit 1;
Packit 0848f5
}
Packit 0848f5
Packit 0848f5
&ReadAndMerge( $defnFile, $tmplFile );
Packit 0848f5