
According to the C Standard, 3.8 a constraint is a restriction, either syntactic or semantic, by which the exposition of language elements is to be interpreted [ISO/IEC 9899:2011]. Despite the similar terms, a runtime-constraint is not a kind of constraint.
Violating any shall within a Constraint clause in the C standard requires an implementation to issue a diagnostic message, the C Standard, 5.1.1.3 [ISO/IEC 9899:2011] states:
A conforming implementation shall produce at least one diagnostic message (identified in an implementation-defined manner) if a preprocessing translation unit or translation unit contains a violation of any syntax rule or constraint, even if the behavior is also explicitly specified as undefined or implementation-defined. Diagnostic messages need not be produced in other circumstances.
The Standard goes on to explain in a footnote:
The intent is that an implementation should identify the nature of, and where possible localize, each violation. Of course, an implementation is free to produce any number of diagnostics as long as a valid program is still correctly translated. It may also successfully translate an invalid program.
Any constraint violation is a violation of this rule because it can result in an invalid program.
Noncompliant Code Example (inline, Internal Linkage)
The Constraints clause in 6.7.4 paragraph 3, states:
An inline definition of a function with external linkage shall not contain a definition of a modifiable object with static or thread storage duration, and shall not contain a reference to an identifier with internal linkage.
The motivation behind this constraint lies in the semantics of inline definitions. Paragraph 7 of 6.7.4 reads, in part:
An inline definition provides an alternative to an external definition, which a translator may use to implement any call to the function in the same translation unit. It is unspecified whether a call to the function uses the inline definition or the external definition.
That is, if a function has an external and inline definition, implementations are free to choose which definition to invoke (two distinct invocations of the function may call different definitions, one the external definition, the other the inline definition). Therefore, issues can arise when these definitions reference internally linked objects, or mutable objects with static or thread storage duration.
This noncompliant code example refers to a file scope static variable with internal linkage from within an external inline function:
static int I = 12; extern inline void func(int a) { int b = a * I; /* ... */ }
Compliant Solution (inline, Internal Linkage)
This compliant solution omits the static
qualifier; consequently, the variable I
has external linkage by default.
int I = 12; extern inline void func(int a) { int b = a * I; /* ... */ }
Noncompliant Code Example (inline, Modifiable Static)
This noncompliant code example defines a modifiable static
variable within an extern inline
function.
extern inline void func(void) { static int I = 12; /* Perform calculations which may modify I */ }
Compliant Solution (inline, Modifiable Static)
This compliant solution removes the static
keyword from the local variable definition. If the modifications to I
must be retained between invocations of func()
, it must be declared at file scope so that it will be defined with external linkage.
extern inline void func(void) { int I = 12; /* Perform calculations which may modify I */ }
Noncompliant Code Example (inline, Modifiable static
)
This noncompliant code example includes two translation units: file1.c
and file2.c
. The first file, file1.c,
defines a random number generation function:
/* file1.c */ /* Externally linked definition of the function get_random */ extern unsigned int get_random() { /* Initialize the seeds */ static unsigned int m_z = 0xdeadbeef; static unsigned int m_w = 0xbaddecaf; /* Compute the next random value and update the seeds */ m_z = 36969 * (m_z & 65535) + (m_z >> 16); m_w = 18000 * (m_w & 65535) + (m_w >> 16); return (m_z << 16) + m_w; }
The second file file2.c
defines an inline version of this function that references mutable static objects, namely objects that maintain the state of the random number generator. Separate invocations of the get_random()
function might call different definitions, each operating on separate static objects, resulting in a faulty random number generator.
/* file2.c */ /* Inline definition of get_random function */ inline unsigned int get_random() { /* Initialize the seeds Constraint violation: static duration storage referenced in non-static inline definition */ static unsigned int m_z = 0xdeadbeef; static unsigned int m_w = 0xbaddecaf; /* Compute the next random value and update the seeds */ m_z = 36969 * (m_z & 65535) + (m_z >> 16); m_w = 18000 * (m_w & 65535) + (m_w >> 16); return (m_z << 16) + m_w; } int main(void) { unsigned int rand_no; for (int ii = 0; ii < 100; ii++) { /* * Get a random number. Implementation defined whether the * inline definition in this file or the external definition * in file2.c is called. */ rand_no = get_random(); /* Use rand_no... */ } /* ... */ /* Get another random number. Behavior is implementation defined. */ rand_no = get_random(); /* Use rand_no... */ return 0; }
Compliant Solution (inline, Modifiable static
)
This compliant solution adds the static
modifier to the inline function definition in file2.c
, giving it internal linkage. All references to get_random()
in file.2.c
will now reference the internally linked definition.
/* file1.c */ /* Externally linked definition of the function get_random. */ extern unsigned int get_random() { /* Initialize the seeds */ static unsigned int m_z = 0xdeadbeef; static unsigned int m_w = 0xbaddecaf; /* Compute the next random value and update the seeds */ m_z = 36969 * (m_z & 65535) + (m_z >> 16); m_w = 18000 * (m_w & 65535) + (m_w >> 16); return (m_z << 16) + m_w; }
/* file2.c */ /* Static inline definition of get_random function */ static inline unsigned int get_random() { /* Initialize the seeds * No more constraint violation, our inline function is now * internally linked. */ static unsigned int m_z = 0xdeadbeef; static unsigned int m_w = 0xbaddecaf; /* Compute the next random value and update the seeds */ m_z = 36969 * (m_z & 65535) + (m_z >> 16); m_w = 18000 * (m_w & 65535) + (m_w >> 16); return (m_z << 16) + m_w; } int main(void) { /* Generate random numbers using get_random()... */ return 0; }
Risk Assessment
Constraint violations are a very broad category of error that can result in unexpected control flow and corrupted data.
Rule | Severity | Likelihood | Remediation Cost | Priority | Level |
---|---|---|---|---|---|
MSC40-C | Low | Unlikely | Medium | P2 | L3 |
Related Vulnerabilities
Search for vulnerabilities resulting from the violation of this rule on the CERT website.
Bibliography
[ISO/IEC 9899:2011] | Clause 4, "Conformance" |