forked from mom-ocean/MOM6
-
Notifications
You must be signed in to change notification settings - Fork 62
Code style guide
Robert Hallberg edited this page May 4, 2018
·
25 revisions
Code style is typically a personal choice. When styles clash
- No tabs
- No trailing white space
- Indents should be consistent (same amount used in contiguous blocks)
- Preferred indent is 2 spaces
- "preferred" might understate the reaction invoked by other indent amounts! 😉
- No white space between a function name and the opening parenthesis
- White space after all language token
-
if(a==0)
is legal fortran but bad style. Useif (a==0)
instead. -
if (a == 0)
is even better, since==
is a language token.
-
- Use space around the equal sign in variable assignment, but not when using a named optional argument
-
a = b
is strongly preferred overa=b
-
call fn(arg1, arg_name=a)
is strongly preferred overcall fn( arg1, arg_name = a )
-
Some compilers handle very long lines of code gracefully, but MOM6 needs to adhere to the Fortran standard, which is 132 characters for code, after any macro expansion. MOM6 does use macros for some memory declarations, so we need to build in some added space in setting MOM6 guidelines:
- The target maximum length for MOM6 code lines (exclusive of comments) is 100 characters
- 80 character lines can be much easier to read if printed; smaller lines are encouraged where they make sense
- No lines of MOM6 code should exceed 120 characters, including comments
- Local variable declarations appear after all the dummy argument declarations. We typically use
! Local variables
to delineate between the argument and local variable declarations. - Local variables should preferably be descriptive multi-character names meaningful in their context, e.g.
del_rho_int
(delta rho at interface). - If using a highly abbreviated or short name, the declaration MUST be commented.
- Multi-word names should use snake_case (e.g.
delta_rho
).
-
i
,j
,k
are used for cell-center, layer-center references, e.g.h(i,j,k)
,T(i+1,j,k)
. -
I
,J
are used for staggered, cell-edge references, e.g.u(I,j,k)
,v(i,J,k)
,q(I,J,k)
,u(I-1,j,k)
. We use a north-east staggering convention so theI
means i+1/2 andI-1
means i-1/2. -
K
is used for the interface (between layer) references, e.g.del_t(i,j,K) = T(i,j,K+1) - T(i,j,K)
. The vertical staggering is such that interfaceK=1
is above layerk=1
so thatK
means k-1/2 andK+1
means k+1/2.
- Absolutely NO!
- There are a few exceptions which are strictly for debugging non-shared memory applications. Do not use these as an excuse for adding module data.
- All needed data is passed via arguments to subroutines and functions, or as the returned value of a function.
- All arguments must have declared intent, with the exception of pointers: i.e.
intent(in)
,intent(out)
,intent(inout)
. - Opaque types are preferred, i.e. referencing members of types defined in other modules is discouraged.
- Do it when you are writing the code in the first place!
- All subroutines, functions, arguments, and elements of public types should be described in with dOxygen comments.
- All real variables should have a full physical description, including units.
- All comments should be clearly written and grammatically correct; American spelling is preferred.