diff options
author | Rich Felker <dalias@aerifal.cx> | 2012-06-19 01:27:26 -0400 |
---|---|---|
committer | Rich Felker <dalias@aerifal.cx> | 2012-06-19 01:27:26 -0400 |
commit | a71e0af25544fd2486e57ea51c6d05abdbf44c3e (patch) | |
tree | 434e8ae22e80ae95c2121a0c89076c7e9388b150 /src/stdio/__stdio_exit.c | |
parent | ca8a4e7fbdeeb05b58ac3d456bae696623b8e312 (diff) | |
download | musl-a71e0af25544fd2486e57ea51c6d05abdbf44c3e.tar.gz musl-a71e0af25544fd2486e57ea51c6d05abdbf44c3e.tar.bz2 musl-a71e0af25544fd2486e57ea51c6d05abdbf44c3e.tar.xz musl-a71e0af25544fd2486e57ea51c6d05abdbf44c3e.zip |
stdio: handle file position correctly at program exit
for seekable files, posix imposed requirements on the offset of the
underlying open file description after a stream is closed. this was
correctly handled (as a side effect of the unconditional fflush call)
when streams were explicitly closed by fclose, but was not handled
correctly at program exit time, where fflush(0) was being used.
the weak symbol hackery is to pull in __stdio_exit if either of
__toread or __towrite is used, but avoid calling it twice so we don't
have to keep extra state. the new __stdio_exit is a streamlined fflush
variant that avoids performing any unnecessary operations and which
never unlocks the files or open file list, so we can be sure no other
threads write new data to a stream's buffer after it's already
flushed.
Diffstat (limited to 'src/stdio/__stdio_exit.c')
-rw-r--r-- | src/stdio/__stdio_exit.c | 23 |
1 files changed, 23 insertions, 0 deletions
diff --git a/src/stdio/__stdio_exit.c b/src/stdio/__stdio_exit.c new file mode 100644 index 00000000..3f87e7ed --- /dev/null +++ b/src/stdio/__stdio_exit.c @@ -0,0 +1,23 @@ +#include "stdio_impl.h" + +static FILE *const dummy_file = 0; +weak_alias(dummy_file, __stdin_used); +weak_alias(dummy_file, __stdout_used); +weak_alias(dummy_file, __stderr_used); + +static void close_file(FILE *f) +{ + if (!f) return; + FLOCK(f); + if (f->wpos > f->wbase) f->write(f, 0, 0); + if (f->rpos < f->rend) f->seek(f, f->rpos-f->rend, SEEK_CUR); +} + +void __stdio_exit(void) +{ + FILE *f; + OFLLOCK(); + for (f=libc.ofl_head; f; f=f->next) close_file(f); + close_file(__stdin_used); + close_file(__stdout_used); +} |