The C Standard, subclause 7.21.8.2 [ISO/IEC 9899:2011], defines the fwrite() function as follows:
Synopsis
...
size_t fwrite(const void *restrict ptr, size_t size, size_tnitemsnmemb, FILE *restrict stream);Description
The
fwrite()function shall writewrites, from the array pointed to byptr, up tonitemsnmembelements whose size is specified bysize, to the stream pointed to bystream. For each object,sizecalls shall be calls are made to thefputc()function, taking the values (in order) from an array ofunsigned charexactly overlaying the object. The file - position indicator for the stream (if defined) shall be is advanced by the number of bytes successfully written. If an error occurs, the resulting value of the file - position indicator for the stream is unspecifiedindeterminate.
The definition does not state that the fwrite() function will stop copying characters into the file if a null character is encountered. Therefore, when writing a null-terminated byte string to a file using the fwrite() function, always use the length of the string plus 1 (to account for the null character) as the nitems the nmemb parameter.
Noncompliant Code Example
In this noncompliant code example, the size of the buffer is stored in size1, but size2 number of characters are written in to the file. If size2 is greater than size1, write() will not stop copying characters at the null character.
| Code Block | ||||
|---|---|---|---|---|
| ||||
#include <stdio.h> #include <stdlib.h> char *buffer = NULL; size_t size1,; size_t size2; FILE *filedes; /* * Assume size1 and size2 are appropriately initialized */ filedes = fopen("out.txt", "w+"); if (filedes == NULL) { /* Handle error */ } buffer = (char *)calloc( 1, size1); if (buffer == NULL) { /* Handle error */ } fwrite(buffer, 1, size2, filedes); free(buffer); buffer = NULL; fclose(filedes); |
...
This compliant solution ensures that the correct number of characters are written to the file.:
| Code Block | ||||
|---|---|---|---|---|
| ||||
#include <stdio.h> #include <stdlib.h> #include <string.h> char *buffer = NULL; size_t size1,; size_t size2; FILE *filedes; /* * Assume size1 is appropriately initialized */ filedes = fopen("out.txt", "w+"); if (filedes == NULL){ /* Handle error */ } buffer = (char *)calloc( 1, size1); if (buffer == NULL) { /* Handle error */ } /* * Accept characters in to the buffer. * Check for buffer overflow. */ size2 = strlen(buffer) + 1; fwrite(buffer, 1, size2, filedes); free(buffer); buffer = NULL; fclose(filedes); |
...
Failure to follow the recommendation could result in a non-null-terminated string being written to a file. This , which will create problems when the program tries to read it back as a null-terminated byte string.
Recommendation | Severity | Likelihood |
|---|
Detectable | Repairable | Priority | Level |
|---|---|---|---|
FIO18-C | Medium | Probable |
Medium
P8
No | No | P4 | L3 |
Automated Detection
Related Guidelines
...
...
Bibliography
...
| 2011] | Subclause 7. |
...
21.8.2, "The fwrite |
...
| Function" |
Bibliography
http://www.opengroup.org/onlinepubs/009695399/functions/fwrite.html
| [IEEE Std 1003.1:2013] | XSH, System Interfaces, fwrite |
...
09. Input Output (FIO) FIO19-C. Do not use fseek() and ftell() to compute the size of a file