13.7. Source code
13.7.1. Code style
We intentionally do not have too many code conventions in the Open MPI code base.
13.7.1.1. All languages
4 space tabs. No more, no less.
NEVER use actual tab characters; always use spaces. Both emacs and vim have secret mojo that can automatically use spaces when you hit the
<TAB>
key. This makes the code look the same in every browser, regardless of individual tab display settings.
13.7.1.2. C / C++
When comparing constants for equality or inequality, always put the constant on the left. This is defensive programming: if you have a typo in the test and miss a
!
or=
, you’ll get a compiler error. For example:/* Do this */ if (NULL == foo) { ... } /* Because if you have a typo (i.e., = instead of ==), this will be a compile error rather than a subtle bug */ if (NULL = foo) { ... }
More defensive programming: always include blocks in curly braces
{ }
, even if they are only one line long. For example:/* Do this */ if (whatever) { return OMPI_SUCCESS; } /* Not this */ if (whatever) return OMPI_SUCCESS;
Starting with Open MPI 1.7, Open MPI requires a C99-compliant compiler.
C++-style comments are now allowed (and preferred).
C99-style mixing declarations are allow allowable (and preferred).
ALWAYS include
<level>_config.h
as your first #include file, where<level>
is one ofompi
,oshmem
, oropal
— the level that you’re writing in. There are very, very few cases where this is not true (E.g., some bizarre Windows scenarios). But in 99.9999% of cases, this file should be included first so that it can affect system-level #include files if necessary.Filenames and symbols must follow the prefix rule (see [e-mail thread](http://www.open-mpi.org/community/lists/devel/2009/07/6389.php)):
Filenames must be prefixed with
<framework>_<component>
.Public symbols must be prefixed in components with
<project>_<framework>_<component>
, where<project>
is one ofmca
,ompi
,oshmem
, oropal
. Note that mca used to be the most common, but it has fallen out of favor compared to the other<project>
prefixes. When in doubt about whether a symbol is public, be safe and add the prefix.Non-public symbols must be declared
static
or otherwise made to not appear in the global scope.
ALWAYS #define macros, even for logical values.
The GNU Way is to
#define
a macro when it is “true” and to#undef
it when it is “false”.In Open MPI, we always
#define
a logical macro to be either 0 or 1 — we never#undef
it.The reason for this is defensive programming: if you are only checking if a preprocessor macro is defined (via
#ifdef FOO
or#if defined(FOO)
), you will get no warning when compiling if you accidentally misspell the macro name. However, if you use the logic test#if FOO
with an undefined macro (e.g., because you misspelled it), you’ll get a compiler warning or error.Rationale
Misspelled macro names can be tremendously difficult to find when they are buried in thousands of lines of code; we will take all the help from the preprocessor/compiler that we can get!
/* GNU Way - you will get no warning from the compiler if you misspell "FOO"; the test will simply be false */ #ifdef FOO ... #else ... #endif /* Open MPI Way - you will get a warning from the compiler if you misspell "FOO"; the result of the test is a different value than whether you spelled the macro name right or not */ #if FOO ... #else ... #endif
13.7.1.3. Fortran
We do not have specific coding style guidelines for Fortran. Please read some of the existing Fortran code in the source code tree and try to use a similar style.
13.7.1.4. Shell scripting
Please read some of the existing shell code in the source code tree and try to use a similar style.
Always enclose evaluated shell variables in quotes to ensure that multi-token values are handled properly.
# This is bad if test $foo = bar; then # This is good if test "$foo" = "bar"; then
The one exception to this is that when doing an assignment to a shell variable from another shell variable, it is not necessary to use quotes on the right hand side:
# This is harmless, but unnecessary foo="$bar" # This is actually sufficient, even for multi-token values of $bar foo=$bar
Do not use the
==
operator fortest
— this is a GNU extension and can cause portability problems on BSD systems. Instead, use the single=
operator.# This is bad if test "$foo" == "bar"; then # This is good if test "$foo" = "bar"; then
13.7.1.5. m4
We do not have specific coding style guidelines for m4 (the language
used to create the configure
script). Please read some of the
existing m4 code in the source code tree and try to use a similar
style.
13.7.2. Tree layout
There are a few notable top-level directories in the source tree:
The main sub-projects:
oshmem
: Top-level OpenSHMEM code baseompi
: The Open MPI code baseopal
: The OPAL code base
config
: M4 scripts supporting the top-levelconfigure
scriptmpi.h
etc
: Some miscellaneous text filesdocs
: Source code for Open MPI documentationexamples
: Trivial MPI / OpenSHMEM example programs3rd-party
: Included copies of required core libraries (either via Git submodules in Git clones or via binary tarballs).Note
While it may be considered unusual, we include binary tarballs (instead of Git submodules) for 3rd party projects that are:
Needed by Open MPI for correct operation, and
Not universally included in OS distributions, and
Rarely updated.
Each of the three main source directories (oshmem
, ompi
, and
opal
) generate at least a top-level library named liboshmem
,
libmpi
, and libopen-pal
, respectively. They can be built as
either static or shared libraries. Executables are also produced in
subdirectories of some of the trees.
The libopen-pal
top-level library is built internally in two parts:
libopen-pal_core
Internal “core” portion of OPAL containing the essential source and MCA needed for tools like mpicc/mpirun to link against. The “core” library is not installed.Includes the following MCA frameworks:
backtrace
,dl
,installdirs
,threads
,timer
Includes all of the source under
opal/class
and most ofopal/util
Includes the files suffixed with
_core
inopal/runtime
libopen-pal
Includes “core” plus all of the other OPAL project sources. This is installed.
Each of the sub-project source directories have similar (but not identical) directory structures under them:
class
: C++-like “classes” (using the OPAL class system) specific to this projectinclude
: Top-level include files specific to this projectmca
: MCA frameworks and components specific to this projectruntime
: Startup and shutdown of this project at runtimetools
: Executables specific to this projectutil
: Random utility code
There are other top-level directories in each of the sub-projects,
each having to do with specific logic and code for that project. For
example, the MPI API implementations can be found under
ompi/mpi/LANGUAGE
, where LANGUAGE
is c
, fortran
, or
java
.
The layout of the mca
trees are strictly defined. They are of the
form:
PROJECT/mca/FRAMEWORK/COMPONENT
To be explicit: it is forbidden to have a directory under the mca
trees that does not meet this template (with the exception of base
directories, explained below). Hence, only framework and component
code can be in the mca
trees.
That is, framework and component names must be valid directory names
(and C variables; more on that later). For example, the TCP BTL
component is located in opal/mca/btl/tcp/
.
The name base
is reserved; there cannot be a framework or component
named base
. Directories named base
are reserved for the
implementation of the MCA and frameworks. Here are a few examples (as
of the v5.0.x source tree):
# Main implementation of the MCA
opal/mca/base
# Implementation of the btl framework
opal/mca/btl/base
# Implementation of the sysv framework
oshmem/mcs/sshmem/sysv
# Implementation of the pml framework
ompi/mca/pml/base
Under these mandated directories, frameworks and/or components may have arbitrary directory structures, however.
13.7.3. Symbol Visibility
The *_DECLSPEC
macros provide a method to annotate symbols to indicate
their intended visibility when compiling dynamically shared object files
(e.g., libmpi.so
). The macros are defined on a per project basis:
Open MPI:
OMPI_DECLSPEC
Open PAL:
OPAL_DECLSPEC
OpenSHMEM:
OSHMEM_DECLSPEC
The macros expand to the appropriate compiler and platform flags for marking
whether a symbol should be explicitly made public in the target project’s
library namespace.
The *_DECLSPEC
attributes are used to declare that a symbol is to be
visible outside of that library/DSO’s scope. For example, OMPI_DECLSPEC
is used to control what symbols are visible in the libmpi.so
scope.
Note
This is entirely related to dynamic library compilation and does not apply to static compilation.
Note
The macros were originally introduced when Open MPI supported Windows (circa Open MPI v1.0.0) and are motivated by the Windows __declspec. While support for Windows has been dropped from Open MPI, the symbol visibility macros remain.