简体   繁体   中英

intricacies/understanding the stdio buffer and dup2

I am reading this lecture and found this following code sample which I modified to this:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <errno.h>

int main()
{
    int fd;
    char *s, *t;
    off_t ret;

    fd = open("file6", O_WRONLY | O_CREAT | O_TRUNC, 0666);
   
    if (dup2(fd, 1) < 0) { perror("dup2"); exit(1); }
    
    printf("Standard output now goes to file6\n");
    s = "before close\n";
    write(1, s, strlen(s));
    
    close(fd);

    printf("It goes even after we closed file descriptor %d\n", fd);
    
    printf("%ld\t"
        "%ld\n",
        (long int) lseek(fd,0,SEEK_CUR), 
        (long int) lseek(1,0,SEEK_CUR));

    s = "And fwrite\n";
    
    fwrite(s, sizeof(char), strlen(s), stdout);

    printf("%ld\t"
        "%ld\n",
        (long int) lseek(fd,0,SEEK_CUR), 
        (long int) lseek(STDOUT_FILENO,0,SEEK_CUR));
    
    fflush(stdout); 

    s = "And write\n";
    write(1, s, strlen(s));
    
    printf("after:\tAnd wri...: lseek(fd,0,SEEK_CUR)=%ld\t"
        "lseek(STDOUT_FILENO,0,SEEK_CUR)=%ld\n",
        (long int) lseek(fd,0,SEEK_CUR), 
        (long int) lseek(STDOUT_FILENO,0,SEEK_CUR));
    return 0;
}

I am sharing two different outputs with the only change in the code being that the line fflush(stdout) is commented out in first and present in the second run.

Output (with fflush(stdout) commented):

before close
And write
Standard output now goes to file4
It goes even after we closed file descriptor 3
-1  13
And fwrite
-1  13
after:  And wri...: lseek(fd,0,SEEK_CUR)=-1 lseek(STDOUT_FILENO,0,SEEK_CUR)=23

Output with flush(stdout) uncommented:

before close
Standard output now goes to file4
It goes even after we closed file descriptor 3
-1  13
And fwrite
-1  13
And write
after:  And wri...: lseek(fd,0,SEEK_CUR)=-1 lseek(STDOUT_FILENO,0,SEEK_CUR)=127

I have two questions:

  • Why does "And write appears" first when fflush(stdout) is commented?
  • Why lseek prints -1 which I checked separately is an error message corresponding to errno ESPIPE. I am aware that lseek on terminal results in an error. But my current understanding is that since the standard output is dup2 to file6, then, this error shouldn't arise? Shouldn't it ( lseek(STDOUT_FILENO, 0, SEEK_CUR) ) simply return the current lseek pointer in file6, if dup2 is successful?

Why does "And write" appear first when fflush(stdout) is commented?

Because the C stdio buffers haven't filled, so nothing written using stdio APIs is actually sent to the output until the buffers fill, the stdio handle is flushed, or the program ends. Your direct write calls (eg for "And write" ) bypass stdio buffers entirely, and get written immediately, all the buffered stuff doesn't appear until the program ends (or at least, not until after "And write" has already been written).

Why lseek prints -1?

The first lseek was called on fd , which you closed shortly after dup2 ing it over STDOUT_FILENO / 1 , so it fails. If you checked the errno properly (zeroing errno before each lseek , calling the two lseek s separately and storing or printing their errors and errno s separately, so one of them doesn't override the errno of the other before you even see it), you'd see it has a value corresponding to EBADF , not ESPIPE . The second lseek on ( STDOUT_FILENO ) works just fine. A mildly modified version of your code (using stderr so you can see the output for the last couple outputs even when you can't read the actual file, carefully zeroing errno each time, printing it before calling lseek again, and using strerror to show a friendly description of the errno ) shows this clearly: Try it online!

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM