I read in Advanced Programming in Unix Environment this:
Line buffering comes with two caveats. First, the size of the buffer that the standard I/O library uses to collect each line is fixed, so I/O might take place if we fill this buffer before writing a newline. Second, whenever input is requested through the standard I/O library from either (a) an unbuffered stream or (b) a line-buffered stream (that requires data to be requested from the kernel), all line-buffered output streams are flushed. The reason for the qualifier on (b) is that the requested data may already be in the buffer, which doesn’t require data to be read from the kernel. Obviously, any input from an unbuffered stream, item (a), requires data to be obtained from the kernel.
https://www.gnu.org/software/libc/manual/html_node/Flushing-Buffers.html states:
Flushing output on a buffered stream means transmitting all accumulated characters to the file. There are many circumstances when buffered output on a stream is flushed automatically:
When you try to do output and the output buffer is full.
When the stream is closed. See Closing Streams.
When the program terminates by calling exit. See Normal Termination. When a newline is written, if the stream is line buffered.
Whenever an input operation on any stream actually reads data from its file.
So I've written a program to test this:
#include <stdio.h>
#include <unistd.h>
int
main()
{
printf("Hello");
FILE *fp = fopen("hugefile", "r");
if (fp == NULL) {
fprintf(stderr, "Cannot open file\n");
return -1;
}
setvbuf(fp, NULL, _IONBF, 0);
sleep(2);
char buf[10];
while (fread(buf, sizeof(char), 10, fp) == 10) {
}
if (ferror(fp)) {
fprintf(stderr, "Read error\n");
return -1;
}; // I expect "Hello" to appear on a screen but it doesn't
fclose(fp);
sleep(50);
}
But fread() doesn't trigger flashing of the output. Am I missing something?
I don't know what might be relevant here but I use:
gcc version 14.2.1 20240910
glibc 2.40+r16+gaa533d58ff-2
hugefile
?setvbuf()
. It may be rejected, as it is probably physically impossible to read a block-buffered device without a (hidden) block-sized buffer. Maybe running understrace
would be helpful.