Robert C. Seacord reminds us in “Effective C” that puts
can fail:
#include <stdio.h>
#include <stdlib.h>
int main(void) {
puts("Hello, world!");
return EXIT_SUCCESS;
}
Although it’s unlikely that the puts function will fail and return EOF for our simple program, it’s possible. Because the call to puts can fail and return EOF, it means that your first C program has a bug.
If we want to be strict about it, he suggests we check for EOF
(a macro defined as a negative integer):
#include <stdio.h>
#include <stdlib.h>
int main(void) {
if (puts("Hello, world!") == EOF) {
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
But how could puts
(or indeed printf
) fail? Can we force a write error to see it ourselves?
Redirect to /dev/full
On Linux we can redirect output to /dev/full
, which emulates a full disk:
> ./hello > /dev/full
> echo $?
1
On macOS we have to do something else, since it doesn’t have /dev/full
.
Redirect output to a full or nearly full disk
Let’s make a 1MB disk image, mount it and check available space:
> hdiutil create -size 1M -fs HFS+ -volname SmallDisk ~/Desktop/SmallDisk.dmg
> hdiutil attach ~/Desktop/SmallDisk.dmg
> df -h /Volumes/SmallDisk
Filesystem Size Used Avail Capacity iused ifree %iused Mounted on
/dev/disk4s1 984Ki 144Ki 840Ki 15% 5 4.3G 0% /Volumes/SmallDisk
840 Kibibytes (KiB) or 860,160 bytes are left on our disk after overhead.
We’ll amend our C hello world to output something larger than that, say, a million “a” characters:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(void) {
char *str = malloc(1000001);
if (str == NULL) {
fprintf(stderr, "Memory allocation failed\n");
return EXIT_FAILURE;
}
memset(str, 'a', 1000000);
str[1000000] = '\0';
if (puts(str) == EOF) {
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
Then redirect output from stdout to our little disk at /Volumes/SmallDisk
:
> ./hello > /Volumes/SmallDisk/file.txt
No output. Check the exit code:
> echo $?
1
We made puts
fail! What about our file contents?
> cat /Volumes/SmallDisk/file.txt
>
Empty! puts
returned EOF
before buffer contents flushed to disk. (With printf
, we might expect file.txt
to include some of our “a“s but not all of them.)
When does this matter?
Whenever you’re redirecting stdout to a file and could run out of space. Consider redirecting JSON or YAML output, for example — failed output could leave you with an invalid file.
It also matters when you’re depending on the exit code. The same code above without the EOF
check will always return a 0 exit code, even when puts
fails to write due to disk, permissions, or hardware issues.