All standard library functions, including I/O functions and memory allocation functions, return either a valid value or a value of the correct return type that indicates an error (for example, −1 or a null pointer). Assuming that all calls to such functions will succeed and failing to check the return value for an indication of an error is a dangerous practice that may lead to unexpected or undefined behavior when an error occurs. It is essential that programs detect and appropriately handle all errors in accordance with an error-handling policy, as discussed in ERR00-C. Adopt and implement a consistent and comprehensive error-handling policy. In addition to the C standard library functions mentioned in ERR33-C. Detect and handle standard library errors, the following functions defined in POSIX require error checking (list is not all-inclusive).
The successful completion or failure of each of the standard library functions listed in the following table shall be determined either by comparing the function’s return value with the value listed in the column labeled “Error Return” or by calling one of the library functions mentioned in the footnotes to the same column.
Function | Successful Return | Error Return |
|
---|---|---|---|
| Pointer to a |
|
|
| Pointer to a |
|
|
|
| Nonzero | Unchanged |
Setting errno
is a POSIX [ISO/IEC 9945:2008] extension to the C Standard. On error, posix_memalign()
returns a value that corresponds to one of the constants defined in the <errno.h>
header. The function does not set errno
. The posix_memalign()
function is optional and is not required to be provided by POSIX-conforming implementations.
Noncompliant Code Example (POSIX)
In this noncompliant code example, fmemopen()
and open_memstream()
are assumed to succeed. However, if the calls fail, the two file pointers in
and out
will be null and the program will have undefined behavior.
#include <stdio.h> #include <string.h> int main(int argc, char *argv[]) { FILE *out; FILE *in; size_t size; char *ptr; if (argc != 2) { /* Handle error */ } in = fmemopen(argv[1], strlen(argv[1]), "r"); /* Use in */ out = open_memstream(&ptr, &size); /* Use out */ return 0; }
Compliant Solution (POSIX)
A compliant solution avoids assuming that fmemopen()
and open_memstream()
succeed regardless of its arguments and tests the return value of the function before using the file pointers in
and out
:
#include <stdio.h> #include <string.h> int main(int argc, char *argv[]) { FILE *out; FILE *in; size_t size; char *ptr; if (argc != 2) { /* Handle error */ } in = fmemopen(argv[1], strlen(argv[1]), "r"); if (in == NULL){ /* Handle error */ } /* Use in */ out = open_memstream(&ptr, &size); if (out == NULL){ /* Handle error */ } /* Use out */ return 0; }
Exceptions
POS54-C-EX1: This exception has been removed.
POS54-C-EX2: The exception from ERR33-C. Detect and handle standard library errors (that is ERR33-C-EX1) applies to this rule. See that exception for more information.
Risk Assessment
Failing to detect error conditions can lead to unpredictable results, including abnormal program termination and denial-of-service attacks or, in some situations, could even allow an attacker to run arbitrary code.
Rule | Severity | Likelihood | Remediation Cost | Priority | Level |
---|---|---|---|---|---|
POS54-C | High | Likely | Medium | P18 | L1 |
Automated Detection
Tool | Version | Checker | Description |
---|---|---|---|
Astrée | 24.04 | error-information-unused error-information-unused-computed | Supported |
Axivion Bauhaus Suite | 7.2.0 | CertC-POS54 | |
CodeSonar | 8.1p0 | LANG.FUNCS.IRV LANG.ERRCODE.NOTEST LANG.ERRCODE.NZ | Ignored return value Missing Test of Error Code Non-zero Error Code |
Compass/ROSE | Can detect violations of this recommendation when checking for violations of EXP12-C. Do not ignore values returned by functions and EXP34-C. Do not dereference null pointers | ||
2017.07 | CHECKED_RETURN | Finds inconsistencies in how function call return values are handled. Coverity Prevent cannot discover all violations of this recommendation, so further verification is necessary | |
Helix QAC | 2024.2 | C3200 | |
Klocwork | 2024.2 | SV.RVT.RETVAL_NOTTESTED | |
LDRA tool suite | 9.7.1 | 80 D | Partially implemented |
Parasoft C/C++test | 2023.1 | CERT_C-POS54-a | The value returned by a POSIX library function that may return an error should be used |
PC-lint Plus | 1.4 | 413, 534, 613 | Assistance provided |
R2024a | CERT C: Rule POS54-C | Checks for situations where return value of a sensitive function is not checked (rule fully covered) | |
RuleChecker | 24.04 | error-information-unused | Supported |
Related Vulnerabilities
The vulnerability in Adobe Flash [VU#159523] arises because Flash neglects to check the return value from calloc()
. Even when calloc()
returns NULL
, Flash writes to an offset from the return value. Dereferencing NULL
usually results in a program crash, but dereferencing an offset from NULL
allows an exploit to succeed without crashing the program.
Search for vulnerabilities resulting from the violation of this rule on the CERT website.
Related Guidelines
Key here (explains table format and definitions)
Bibliography
[DHS 2006] | Handle All Errors Safely |
[Henricson 1997] | Recommendation 12.1, "Check for All Errors Reported from Functions" |
[ISO/IEC 9899:2011] | Subclause 7.21.7.10, "The ungetc Function" |
10 Comments
Yozo TODA
exception EX2 should be EX1?
(just confirming that it's simply a kind of typo...)
David Svoboda
Confirmed. We just removed EX1 yesterday, but left EX2 and did not want to renumber it.
Markus Elfring
David Svoboda
Markus:
I'm not sure what you are asking.
As you know, ERR33-C enumerates ISO C library functions that return exceptional values, and also enumerates the functions whose values may be ignored. This rule is analogous to ERR33-C but addresses POSIX library functions. However, this rule does not enumerate such functions either in its introduction or the exceptions. We felt that this task was too big at the time. Are you volunteering to do this task?
Markus Elfring
There are further case distinctions to consider.
No, not directly.
I am still trying to adjust the corresponding development awareness (because I came along also special or “exceptional” cases like pause()).
Can a systematic analysis of involved programming interfaces provide desired data automatically?
Markus Elfring
How many functions can you know which are documented in the way that they should provide only a fixed return value?
David Svoboda
You mean like gettimeofday()? I haven't been over the list of POSIX functions. Again, we decided there are too many functions to enumerate them all. It is, however, a straightforward task to do so.
Markus Elfring
David Svoboda
I suspect that your question is better posed to the Austin Group than us. The Austin Group maintains the POSIX standard. FWIW I was able to download an HTML version of the POSIX standard, and I'm sure you can use grep, or other search tools to simplify querying of POSIX functions.
Markus Elfring
Are other contributors in a better position to raise involved abstraction levels and to improve corresponding development tools?