The use of make in the MM5 project is necessitated for a number of reasons.
MM5 is written in FORTRAN and organized with the goal of providing a logical structure for code development and to encourage modular development of new options and subroutines. In addition, it is desired to supply the user/developer with some "pointers" as to the location of routines of particular interest. So the hope is to create something that appeared simple to the casual user but allowed more convenient access for the power user.
This structure is implemented implicitly by taking advantage of the Unix file system structure. Since directories are arranged as trees, the subroutines are subdivided into conceptual groups. The include directory contains the include files for the various subroutines. The domain, dynamics, fdda, physics, and memory directories contain the subroutines divided by function. The Run directory holds the main program source code. make is the glue that holds this complicated structure together. As you have seen, make executes commands by spawning shells. These shells can in turn run make in subdirectories. This ability to nest makefiles is very powerful, since it allows you to recursively build an entire directory.
Writing portable code involves not only following language standards, but creating a development structure that is equally standard. Every time code moves to a new machine you not only need to worry about your code but compilers, the operating system and the system environment, available libraries, and options for all of the above.
The answer to this problem is two-fold, namely, use only standard tools and minimize use of esoteric options. make is such a standard tool - you will find make on every working UNIX machine you encounter. While each vendor's make may differ in significant way, they all support a core subset of functionality. This means that a basic makefile with no bells and whistles will work on the vast majority of machines available on the market.
"All makes are equal, but some makes are more equal than others."
Every decent Unix box will have make and cpp. However, they may throw in non-standard options. The quotation above reminds us to keep to standard constructions whenver possible.
One of the stated goals is conditional compilation. This is done in two different ways. make keys off the user's options to skip compilation of those directories not required. When a source file is compiled, cpp is used to avoid including code that is not required. So make skips unnecessary compilation while cpp is used to modify compilation.
Since make needs rules and defined dependencies (sometimes not the default ones), and there are more than 65 makefiles in the MM5 V3 directory structure (more than 70 MM5 subdirectories, over 300 Fortran files, and more than 100 C files), it would be an enomous task to make any change to these makefiles. A simple solution to solve this problem is to define all the rules and dependencies in one file and pass this file to all makefiles when make is executed. These definitions constitute part of the configure.user file. This section explains the rules and dependencies as defined in the configure.user file.
| Macro to define where to locate c pre-processor on the machine. | |
| Macro to define any local libraries that the compiler may access. | |
The following, which appears at the end of the configure.user file, defines the suffix rules a makefile uses. For example, .F.o: defines the rules to go from .F to .o files. In this case, the make will first remove any existing out-of-date .o file, and compile the .F files.
make is a tool that executes "makefiles". Makefiles contain "targets" and "dependencies". A target is what you want to compile. A dependency is what needs to be done to compile the target. We use a 3-tiered makefile structure following the directory struture.
Examples of each makefile follow.
find . -name \*.i -exec rm {} \; ; \
(cd memory; $(MAKE) little_f); \
(cd fdda; $(MAKE) little_f); \
(cd domain; $(MAKE) little_f); \
(cd physics; $(MAKE) little_f); \
(cd dynamics; $(MAKE) little_f); \
./Util/makedeck.csh $(RUNTIME_SYSTEM);
(cd include; $(MAKE) clean); \
(cd dynamics; $(MAKE) clean);\
if [ -f libutil.a ]; then $(RM) libutil.a; fi;
(cd include; $(MAKE) clean); \
(cd dynamics; $(MAKE) clean);\
if [ -f libutil.a ]; then $(RM) libutil.a; fi;
$(CC) -o ./LineNumberer Util/LineNumberer.c;
# To clean after changes to configure.user, type `make mpclean'.
# To uninstall everything relating to MPP option, `make uninstall'.
# To partially remake installation, remove MPP/mpp_install and `make mpp'.
(cd MPP/build ; /bin/rm -fr *.o *.f *.dm *.b)
(sed `/t touch anything below this line/,$$d' configure.user \
$(CPP) -I../../pick ../mpp_objects_all > .tmpobjs ; \
$(MAKE) -f Makefile.$(MPP_LAYER) )
(cd MPP/RSL/RSL ; $(MAKE) $(MPP_TARGET) )
(cd MPP/FLIC ; $(MAKE) ; $(MAKE) clean )
(cd MPP/FLIC/FLIC ; $(MAKE) ; \
sed s+INSTALL_STRING_FLICDIR+`pwd`+ flic.csh > flic ; \
(csh MPP/Makelinks $(MPP_LAYER) $(MPP_TARGET) )
if [ -f libutil.a ]; then $(RM) libutil.a; fi
(cd MPP/FLIC/FLIC; /bin/rm -f dm ; $(MAKE) clean )
(cd MPP/FLIC; $(MAKE) clean ; /bin/rm -fr bin )
(cd MPP/RSL/RSL; $(MAKE) clean ; /bin/rm -f librsl.a )
Note: there are several targets in the top-level makefile: all, code, little_f (for IBM xlf compiler, or any fortran compiler that does not allow the use of cpp), mm5.deck, clean, LineNumberer, and mpclean, mpp, etc. for MPP extension. If a user does not specify a target, the makefile will use the first one it sees. In this case, it is the `all' target. For any target that is not placed first, a user must explicitly specify the target. For example, we use `make mm5.deck' to make a job deck. The command for the target `all' is to cd to a particular directory and execute make (the macro $(MAKE) is defined in configure.user file).
# Makefile for directory physics/pbl_sfc
# Makefile for directory physics/pbl_sfc
(cd bulk; $(MAKE) little_f); \
(cd hirpbl; $(MAKE) little_f); \
(cd btpbl; $(MAKE) little_f); \
(cd btpbl; $(MAKE) little_f); \
(cd mrfpbl; $(MAKE) little_f); \
(cd btpbl; $(MAKE) little_f); \
Note: This example shows how the branching is done with the mid-level makefile. The makefile first echos the string IBLTYP defined in configure.user file to a temporary file, .tmpfile. It then checks, using grep, to see if any of the options exist (in this case, IBLTYP may be 0,1,2,3, or 5). If any of them is defined, it will go to the directory that contains the subroutines for that option and execute the make command there. Again there are several targets in this mid-level makefile: lib, code, little_f, and clean. The default is the target lib.
# Makefile for directory physics/pbl_sfc/mrfpbl
include ../../../configure.user
CURRENT_DIR = $(DEVTOP)/physics/pbl_sfc/mrfpbl
$(AR) $(TARGETDIR)lib$(LIBTARGET).a $(OBJS)
$(AR) $(TARGETDIR)lib$(LIBTARGET).a $(OBJS)
# ---------------------------------------------------------------
# common rules for all Makefiles - do not edit
# ---------------------------------------------------------------
# DO NOT DELETE THIS LINE -- make depend depends on it.
mrfpbl.o: ../../../include/parame.inc ../../../include/rpstar.incl
mrfpbl.o: ../../../include/varia.incl ../../../include/dusolve1.incl
mrfpbl.o: ../../../include/param2.incl ../../../include/param3.incl
mrfpbl.o: ../../../include/pmoist.incl ../../../include/point3d.incl
mrfpbl.o: ../../../include/point2d.incl ../../../include/various.incl
mrfpbl.o: ../../../include/nonhyd.incl ../../../include/nhcnst.incl
mrfpbl.o: ../../../include/soil.incl ../../../include/soilcnst.incl
mrfpbl.o: ../../../include/addrcu.incl ../../../include/pbltb.incl
tridi2.o: ../../../include/parame.incl
Note: In this example, when make is executed (`make -i -r'), it first looks for the target all, for example. It finds that the target `all' depends on a group of object files (defined by the macro OBJS). The rules for making the object files are defined in configure.user file, i.e. the .F.o: rule. The makefile checks whether any .o files are out-of-date w.r.t. the .F files, or w.r.t. any of the include files used in the .F files. The dependencies on include files are at the end of the makefile. After the .o files are made, the command on the following line specifies how to archive them into libutil.a using macro AR defined in configure.user.
The cpp pre-processor is about as old as Unix itself. A pre-processor scans a file and make modifications according to user-supplied definitions. Typically this facility is used for global substitutions, conditional code inclusion, including files, and function templating. We only use the cpp "conditional code inclusion" and "including files" features. Because we use cpp, our Fortran codes are named .F, in contrast to .f. Many machines recognize .F files as the ones that need to be run through cpp first before being compiled.
One cpp directive is "#include <filename>". This directive indicates that filename should be included in the source prior to compilation.
SUBROUTINE SOLVE(IEXEC,INEST,NN)
SUBROUTINE SOLVE(IEXEC,INEST,NN)
C--- ADDITIONAL MEMORY REQUIREMENTS FOR RUNS ,
C--- GRIDDED FDDA RUNS (IFDDAG=1) AND OBS FDDA RUNS (IFDDAO=1),
C--- NONHYDROSTATIC RUNS (INHYD=1), HIGHER ORDER PBL RUNS (INAV=1),
C--- EXPLICIT MOISTURE SCHEME (IEXMS=1), ARAKAWA-SCHUBERT
C--- CONVECTIVE PARAMETERIZATION (IARASC=1), ATMOSPHERIC
C--- RADIATION (IRDDIM=1), MIXED-PHASE ICE SCHEME (IICE=1).
C--- GRAUPEL SCHEME (IICEG=1), KAIN-FRITSCH AND FRITSCH-CHAPPELL.
C--- CONVECTIVE PARAMETERIZATIONS (IKFFC=1), AND GAYNO-SEAMAN PBL (IGSPBL=1).
C--- INTEGER IARASC,IEXMS,IFDDAG,IFDDAO,IICE,IICEG,IKFFC,ILDDIM,INAV
C--- 5-LAYER SOIL (ISLDIM=1,MLX=6), OSU LAND SFC (ILDDIM=1,MLX=4).
INTEGER IARASC,IEXMS,IFDDAG,IFDDAO,IICE,IICEG,IKFFC,ILDDIM,INAV
INTEGER INAV2,INAV3,IGSPBL,INHYD,IRDDIM,ISLDIM,MLX
PARAMETER (IFDDAG=1,IFDDAO=1,INHYD=1,INAV=0,INAV2=0,INAV3=0,
1 IICE=0,IICEG=0,IEXMS=1,IKFFC=0,IARASC=0,IRDDIM=1,
cpp also recognizes conditional directives. You define a macro in your source code using the "define" directive - you can then use the "#ifdef" test on this macro to selectively include code.
Example: In defines.incl, there are statements such as:
CALL CUPARA3(T3D,QV3D,PSB,T3DTEN,QV3DTEN,RAINC,CLDFRA,HT,U3D,
+ V3D,PP3D,INEST,J,IBNES,IENES-1)
................... and so on.
In this example only ICUPA3 is defined (#define ICUPA3 1 in defines.incl), so the call to CUPARA3 will be kept in the final source code. Other cumulus schemes are not selected, so the calls to these schemes won't be included in the source code to be compiled.