#! /bin/sh # # (C) 2006 by Argonne National Laboratory. # See COPYRIGHT in top-level directory. # # NOTE: this script has been modified to automatically link in the darshan # characterization libraries. See www.mcs.anl.gov/research/projects/darshan # # mpif77 # Simple script to compile and/or link MPI programs. # This script knows the default flags and libraries, and can handle # alternative C compilers and the associated flags and libraries. # The important terms are: # includedir, libdir - Directories containing an *installed* mpich2 # prefix, execprefix - Often used to define includedir and libdir # FC - Fortran 77 compiler # WRAPPER_FFLAGS - Any special flags needed to compile # WRAPPER_LDFLAGS - Any special flags needed to link # MPILIBNAME - Name of the MPI library # MPI_OTHERLIBS - Other libraries needed in order to link # F77_OTHER_LIBS - Yet more libraries, needed just with F77 # # # We assume that (a) the C compiler can both compile and link programs # # Handling of command-line options: # This is a little tricky because some options may contain blanks. # # Special issues with shared libraries - todo # # -------------------------------------------------------------------------- # Set the default values of all variables. # # Directory locations: Fixed for any MPI implementation. # Set from the directory arguments to configure (e.g., --prefix=/usr/local) prefix=/bgsys/drivers/V1R3M0_460_2008-081112P/ppc/comm/default exec_prefix=${prefix} sysconfdir=${prefix}/etc includedir=${prefix}/include libdir=${exec_prefix}/lib CP_WRAPPERS=@CP_WRAPPERS@ DARSHAN_LIB_PATH=@darshan_lib_path@ DARSHAN_BIN_PATH=@darshan_bin_path@ CP_ZLIB_LINK_FLAGS=@__CP_ZLIB_LINK_FLAGS@ # # Default settings for compiler, flags, and libraries # Determined by a combination of environment variables and tests within # configure (e.g., determining whehter -lsocket is needee) FC="/opt/ibmcmp/xlf/bg/11.1/bin/bgxlf2003" FC_LINKPATH_SHL="-Wl,-rpath -Wl," F77CPP="" WRAPPER_FFLAGS="-I/bgsys/drivers/V1R3M0_460_2008-081112P/ppc/comm/sys/include" WRAPPER_LDFLAGS="" MPILIBNAME="mpich.cnk" PMPILIBNAME="pmpich.cnk" MPI_OTHERLIBS=" -L/bgsys/drivers/V1R3M0_460_2008-081112P/ppc/comm/sys/lib -Wl,-rpath,/bgsys/drivers/V1R3M0_460_2008-081112P/ppc/comm/sys/lib -ldcmfcoll.cnk -ldcmf.cnk -lpthread -L/bgsys/drivers/V1R3M0_460_2008-081112P/ppc/runtime/SPI -Wl,-rpath,/bgsys/drivers/V1R3M0_460_2008-081112P/ppc/runtime/SPI -lSPI.cna -lrt" NEEDSPLIB="no" # # MPIVERSION is the version of the MPICH2 library for which mpif77 is intended MPIVERSION="1.0.7" # # Internal variables # Show is set to echo to cause the compilation command to be echoed instead # of executed. Show= # # End of initialization of variables #--------------------------------------------------------------------- # Environment Variables. # The environment variables MPICH_F77 may be used to override the # default choices. # In addition, if there is a file $sysconfdir/mpif77-$F77name.conf, # where F77name is the name of the compiler with all spaces replaced by hyphens # (e.g., "f77 -64" becomes "f77--64", that file is sources, allowing other # changes to the compilation environment. See the variables used by the # script (defined above) if [ -n "$MPICH_F77" ] ; then FC="$MPICH_F77" F77name=`echo $FC | sed 's/ /-/g'` if [ -s $sysconfdir/mpif77-$F77name.conf ] ; then . $sysconfdir/mpif77-$F77name.conf fi fi # Allow a profiling option to be selected through an environment variable if [ -n "$MPIF77_PROFILE" ] ; then profConf=$MPIF77_PROFILE fi # # ------------------------------------------------------------------------ # Argument processing. # This is somewhat awkward because of the handling of arguments within # the shell. We want to handle arguments that include spaces without # loosing the spacing (an alternative would be to use a more powerful # scripting language that would allow us to retain the array of values, # which the basic (rather than enhanced) Bourne shell does not. # # Look through the arguments for arguments that indicate compile only. # If these are *not* found, add the library options linking=yes allargs=("$@") argno=0 cppflags=() for arg in "$@" ; do # Set addarg to no if this arg should be ignored by the C compiler addarg=yes case "$arg" in # ---------------------------------------------------------------- # Compiler options that affect whether we are linking or no -c|-S|-E|-M|-MM) # The compiler links by default linking=no ;; # ---------------------------------------------------------------- # Options that control how we use mpif77 (e.g., -show, # -f77=* -config=* -echo) addarg=no set -x ;; -f77=*) FC=`echo A$arg | sed -e 's/A-f77=//g'` addarg=no ;; -fc=*) FC=`echo A$arg | sed -e 's/A-fc=//g'` addarg=no ;; -show) addarg=no Show=echo ;; -config=*) addarg=no F77name=`echo A$arg | sed -e 's/A-config=//g'` if [ -s "$sysconfdir/mpif77-$F77name.conf" ] ; then . "$sysconfdir/mpif77-$F77name.conf" else echo "Configuration file mpif77-$F77name.conf not found" fi ;; -compile-info|-compile_info) # -compile_info included for backward compatibility Show=echo addarg=no ;; -link-info|-link_info) # -link_info included for backward compatibility Show=echo addarg=no ;; -v) # Pass this argument to the compiler as well. echo "mpif77 for $MPIVERSION" linking=no ;; -profile=*) # Pass the name of a profiling configuration. As # a special case, lib.so or lib.la may be used # if the library is in $libdir profConf=`echo A$arg | sed -e 's/A-profile=//g'` addarg=no # Loading the profConf file is handled below ;; -mpe=*) # Pass the name of a profiling configuration; this is a special # case for the MPE libs. See -profile profConf=`echo A$arg | sed -e 's/A-mpe=//g'` profConf="mpe_$profConf" addarg=no # Loading the profConf file is handled below ;; # -help -help) ;; # The following are special args used to handle .F files when the # Fortran compiler itself does not handle these options -I*) cppflags[${#cppflags}]="$arg" ;; -D*) cppflags[${#cppflags}]="$arg" ;; *.F|*.F90|*.fpp|*.FPP) # If F77CPP is not empty, then we need to do the following: # If any input files have the .F or .F90 extension, then # If F77CPP = false, then # generate an error message and exit # Use F77CPP to convert the file from .F to .f, using # $TMPDIR/f$$-$count.f as the output file name # Replace the input file with this name in the args # This is needed only for very broken systems # if [ -n "$F77CPP" ] ; then if [ "$F77CPP" = "false" ] ; then echo "This Fortran compiler does not accept .F or .F90 files" exit 1 fi addarg=no # Remove and directory names and extension $ext=`expr "$arg" : '.*\(\..*\)'` bfile=`basename $arg $ext` # TMPDIR=${TMPDIR:-/tmp} tmpfile=$TMPDIR/f$$-$bfile.f if $F77CPP "${cppflags[@]}" $arg > $tmpfile ; then # Add this file to the commandline list count=`expr $count + 1` allargs[${#allargs}]="$tmpfile" rmfiles="$rmfiles $tmpfile" else echo "Aborting compilation because of failure in preprocessing step" echo "for file $arg ." exit 1 fi fi # Otherwise, just accept the argument ;; # - end of special handling for .F files esac if [ $addarg = no ] ; then unset allargs[$argno] fi # Some versions of bash do not accept ((argno++)) argno=`expr $argno + 1` done # ----------------------------------------------------------------------- # Derived variables. These are assembled from variables set from the # default, environment, configuration file (if any) and command-line # options (if any) if [ "$NEEDSPLIB" = yes ] ; then mpilibs="-l$PMPILIBNAME -l$MPILIBNAME" else mpilibs="-l$MPILIBNAME" fi # # Init with the ones needed by MPI FFLAGS="$WRAPPER_FFLAGS" LDFLAGS="$WRAPPER_LDFLAGS" # # Handle the case of a profile switch if [ -n "$profConf" ] ; then profConffile= if [ -s "$libdir/lib$profConf.a" -o -s "$libdir/lib$profConf.so" ] ; then mpilibs="-l$profConf $mpilibs" elif [ -s "$sysconfdir/$profConf.conf" ] ; then profConffile="$sysconfdir/$profConf.conf" elif [ -s "$profConf.conf" ] ; then profConffile="$profConf.conf" else echo "Profiling configuration file $profConf.conf not found in $sysconfdir" fi if [ -n "$profConffile" -a -s "$profConffile" ] ; then . $profConffile if [ -n "$PROFILE_INCPATHS" ] ; then FFLAGS="$PROFILE_INCPATHS $FFLAGS" fi if [ -n "$PROFILE_PRELIB" ] ; then mpilibs="$PROFILE_PRELIB $mpilibs" fi if [ -n "$PROFILE_POSTLIB" ] ; then mpilibs="$mpilibs $PROFILE_POSTLIB" fi fi fi # # A temporary statement to invoke the compiler # Place the -L before any args incase there are any mpi libraries in there. # Eventually, we'll want to move this after any non-MPI implementation # libraries # if [ "$linking" = yes ] ; then if [ -n "$FC_LINKPATH_SHL" ] ; then # Prepend the path for the shared libraries to the library list mpilibs="$FC_LINKPATH_SHL$libdir $mpilibs" fi # check to see if the command line refers to any profilers that might # use PMPI. profiler_check=`echo "${allargs[@]}" | ${DARSHAN_BIN_PATH}/darshan-pmpi-filter.sh` if [ "$profiler_check" = "" ] ; then # If we link link darshan-mpi-io before system mpi lib, we don't get a # trace but if we don't link it early enough, we get undefined MPI symbols. # Seems like listing the darshan libs twice does the trick $Show $FC "${allargs[@]}" -L${DARSHAN_LIB_PATH} -lfmpich.cnk \ -ldarshan-posix -ldarshan-mpi-io \ -I$includedir $FFLAGS ${CP_WRAPPERS} $LDFLAGS -L$libdir $mpilibs $MPI_OTHERLIBS \ -L${DARSHAN_LIB_PATH} ${CP_ZLIB_LINK_FLAGS} -ldarshan-posix -ldarshan-mpi-io -lz else $Show $FC "${allargs[@]}" -I$includedir $FFLAGS $LDFLAGS -L$libdir $mpilibs $MPI_OTHERLIBS fi rc=$? else $Show $FC "${allargs[@]}" -I$includedir $FFLAGS rc=$? fi if [ -n "$rmfiles" ] ; then for file in $rmfiles ; do objfile=`basename $file .f` if [ -s "${objfile}.o" ] ; then # Rename destfile=`echo $objfile | sed -e "s/.*$$-//"` mv -f ${objfile}.o ${destfile}.o fi rm -f $file done rm -f $rmfiles fi exit $rc