Macro replacement lists should be parenthesized to protect any lower-precedence operators from the surrounding expression. See also PRE00-C. Prefer inline or static functions to function-like macros and PRE01-C. Use parentheses within macros around parameter names.
Noncompliant Code Example
This CUBE()
macro definition is noncompliant because it fails to parenthesize the replacement list:
#define CUBE(X) (X) * (X) * (X) int i = 3; int a = 81 / CUBE(i);
As a result, the invocation
int a = 81 / CUBE(i);
expands to
int a = 81 / i * i * i;
which evaluates as
int a = ((81 / i) * i) * i); /* Evaluates to 243 */
which is not the desired behavior.
Compliant Solution
With its replacement list parenthesized, the CUBE()
macro expands correctly for this type of invocation.
#define CUBE(X) ((X) * (X) * (X)) int i = 3; int a = 81 / CUBE(i);
This compliant solution violates PRE00-C. Prefer inline or static functions to function-like macros. Consequently, this solution would be better implemented as an inline function.
Noncompliant Code Example
In this noncompliant code example, END_OF_FILE
is defined as -1
. The macro replacement list consists of a unary negation operator followed by an integer literal 1:
#define END_OF_FILE -1 /* ... */ if (getchar() END_OF_FILE) { /* ... */ }
In this example, the programmer has mistakenly omitted the comparison operator from the conditional statement, which should be getchar() != END_OF_FILE
. (See void MSC02-C. Avoid errors of omission.) After macro expansion, the conditional expression is incorrectly evaluated as a binary operation: getchar()-1
. This statement is syntactically correct, even though it is certainly not what the programmer intended. Note that this example also violates DCL00-C. Const-qualify immutable objects.
Parenthesizing the -1
in the declaration of END_OF_FILE
ensures that the macro expansion is evaluated correctly:
#define END_OF_FILE (-1)
Once this modification is made, the noncompliant code example no longer compiles because the macro expansion results in the conditional expression getchar() (-1)
, which is no longer syntactically valid. Note that there must be a space after END_OF_FILE
because, otherwise, it becomes a function-like macro (and one that is incorrectly formed because −1 cannot be a formal parameter).
Compliant Solution
In this compliant solution, the macro definition is replaced with an enumeration constant in compliance with DCL00-C. Const-qualify immutable objects. In addition, because EOF
is a reserved macro defined in the <stdio.h>
header, the compliant solution must also use a different indentifier in order to comply with DCL37-C. Do not declare or define a reserved identifier.
enum { END_OF_FILE = -1 }; /* ... */ if (getchar() != END_OF_FILE) { /* ... */ }
Exceptions
PRE02-C-EX1: A macro that expands to a single identifier or function call is not affected by the precedence of any operators in the surrounding expression, so its replacement list need not be parenthesized.
#define MY_PID getpid()
PRE02-C-EX2: A macro that expands to an array reference using the array-subscript operator []
, or an expression designating a member of a structure or union object using either the member-access .
or ->
operators is not affected by the precedence of any operators in the surrounding expression, so its replacement list need not be parenthesized.
#define NEXT_FREE block->next_free #define CID customer_record.account.cid #define TOOFAR array[MAX_ARRAY_SIZE]
Risk Assessment
Failing to parenthesize macro replacement lists can cause unexpected results.
Recommendation | Severity | Likelihood | Remediation Cost | Priority | Level |
---|---|---|---|---|---|
PRE02-C | Medium | Probable | Low | P12 | L1 |
Automated Detection
Tool | Version | Checker | Description |
---|---|---|---|
Axivion Bauhaus Suite | 7.2.0 | CertC-PRE02 | |
CodeSonar | 8.1p0 | LANG.PREPROC.MACROEND LANG.PREPROC.MACROSTART | Macro Does Not End With ) or } Macro Does Not Start With ( or { |
1.2 | CC2.PRE02 | Fully implemented | |
Helix QAC | 2024.4 | C3409 | |
Klocwork | 2024.4 | MISRA.DEFINE.BADEXP | |
LDRA tool suite | 9.7.1 | 77 S | Fully implemented |
Parasoft C/C++test | 2023.1 | CERT_C-PRE02-a | Enclose in parentheses whole definition of a function-like macro |
PC-lint Plus | 1.4 | 773, 973 | Fully supported |
Related Vulnerabilities
Search for vulnerabilities resulting from the violation of this rule on the CERT website.
Related Guidelines
SEI CERT C++ Coding Standard | VOID PRE02-CPP. Macro replacement lists should be parenthesized |
ISO/IEC TR 24772:2013 | Operator Precedence/Order of Evaluation [JCW] Pre-processor Directives [NMP] |
Bibliography
[Plum 1985] | Rule 1-1 |
[Summit 2005] | Question 10.1 |
7 Comments
Samium Gromoff
This will disable use of macros which are intended to produce concatenable tokens, for the purposes of name generation.
The reason why we want this, is because sometimes we /have/ to use more than one layer of concatenating macros, just to be able to get the parameter->value mapping we desire.
Robert Seacord
I'm sympathetic to this comment. Can you write an example demonstrating this usage?
Martin Sebor
By defining
EOF
(reserved in<stdio.h>
), the second Compliant Solution violates DCL37-C. Do not use identifiers that are reserved for the implementation, bullet 3.Regarding PRE02-EX1: I'm inclined to discourage the practice of defining object macros that expand to function calls because doing so makes it easy to confuse the macro for an object and introduce subtle bugs. Consider:
Robert Seacord (Manager)
EOF is definitely a problem. Ideally we can replace this with some other function that returns -1 and
#define FUNC_ERROR -1
.Still thinking about the second part of your comment. We have other guidelines which all suggest alternatives to macros. This one is sort of a "defense-in-depth" guideline about parenthesizing macro replacement lists (consistent with the CSCG SG decision that guidelines should be independent).
Martin Sebor
Another reason to remove the exceptions and require that the definitions of all macros be parenthesized is Defect Report #017 which clarifies that
Consider the following snippet:
Since the
MY_PID
macro expands to the name of the function-like macropid
and the next token is(
the behavior of the program is undefined.This would not be the case if the definition of
MY_PID
(or the definition ofgetpid()
) were enclosed in parentheses.David Svoboda
The subtle bugs in Martin's sample code can be fixed by using a no-arg function macro:
Perhaps we should warn against using non-function macros to represent functions? Recommend that macros whose definitions involve side effects or function calls be implemented as f unction macros (possibly with zero arguments).
Dhruv Mohindra
The solution of parenthesizing suggested in the second NCE (EOF example) could perhaps be separated as a CS because it prevents the vulnerability? That way the NCE can only talk about the problem and the CS can also insert the missing =.