File: source-code.rst

package info (click to toggle)
openmpi 5.0.8-3
  • links: PTS, VCS
  • area: main
  • in suites:
  • size: 201,692 kB
  • sloc: ansic: 613,078; makefile: 42,353; sh: 11,194; javascript: 9,244; f90: 7,052; java: 6,404; perl: 5,179; python: 1,859; lex: 740; fortran: 61; cpp: 20; tcl: 12
file content (234 lines) | stat: -rw-r--r-- 7,445 bytes parent folder | download | duplicates (12)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
Source code
===========

Code style
----------

We intentionally do not have too many code conventions in the PMIx
code base.

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.

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:

  .. code-block:: c

     /* 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:

  .. code-block:: c

     /* Do this */
     if (whatever) {
         return OMPI_SUCCESS;
     }

     /* Not this */
     if (whatever)
        return OMPI_SUCCESS;

* PMIx requires a C99-compliant compiler.

  * C++-style comments are now allowed (and preferred).
  * C99-style mixing declarations are allow allowable (and preferred).

* **ALWAYS** include ``pmix_config.h`` as your first #include file.
  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
    ``pmix_<framework>_<component>``. 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 PMIx, 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.

    .. admonition:: Rationale
       :class: tip

       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!

  .. code-block:: c

     /* GNU Way - you will get no warning from the compiler if you
        misspell "FOO"; the test will simply be false */
     #ifdef FOO
     ...
     #else
     ...
     #endif

     /* PMIx 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


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.

  .. code-block:: sh

     # 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:

    .. code-block:: sh

       # 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 for ``test`` |mdash| this is a GNU
  extension and can cause portability problems on BSD systems.
  Instead, use the single ``=`` operator.

  .. code-block:: sh

     # This is bad
     if test "$foo" == "bar"; then

     # This is good
     if test "$foo" = "bar"; then

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.

Tree layout
-----------

There are a few notable top-level directories in the source
tree:

* The main PMIx source is under the ``src`` directory
* ``config``: M4 scripts supporting the top-level ``configure`` script
* ``etc``: Some miscellaneous text files
* ``docs``: Source code for PMIx documentation
* ``examples``: Trivial example programs
* ``include``: The public PMIx headers

The ``src`` directory generates a top-level library named ``libpmix``.
It can be built as either a static or shared library. The directory
structure under it includes:

* ``class``: C++-like "classes" (using the OPAL class system)
  specific to this project
* ``include``: Top-level internal include files
* ``mca``: MCA frameworks and components specific to PMIx
* ``runtime``: Startup and shutdown of PMIx at runtime
* ``tools``: Executables specific to PMIx
* ``util``: Random utility code

The layout of the ``mca`` tree is strictly defined to be of the
form:

.. code-block:: text

    mca/FRAMEWORK/COMPONENT

To be explicit: it is forbidden to have a directory under the ``mca``
tree 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`` tree.

That is, framework and component names must be valid directory names
(and C variables; more on that later).  For example, the CLIENT PTL
component is located in ``mca/ptl/client/``.

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 |opmix_series| source tree):

.. code-block:: sh

    # Main implementation of the MCA
    mca/base

    # Implementation of the ptl framework
    mca/ptl/base

    # Implementation of the client component of the ptl framework
    mca/ptl/client

Under these mandated directories, frameworks and/or components may have
arbitrary directory structures, however.

Symbol Visibility
-----------------

The ``PMIX_EXPORT`` macro provides a method to annotate symbols to indicate
their intended visibility when compiling dynamically shared object files
(e.g., ``libpmix.so``).

The macro expands to the appropriate compiler and platform flags for marking
whether a symbol should be explicitly made public in the PMIx
library namespace.
The ``PMIX_EXPORT`` attribute is used to declare that a symbol is to be
visible outside of the PMIx DSO's scope.

.. note:: This is entirely related to dynamic library compilation and does not
   apply to static compilation.