it should be noted that only the actual underlying buffer flush and fill operations are cancellable, not reads from or writes to the buffer. this behavior is compatible with POSIX, which makes all cancellation points in stdio optional, and it achieves the goal of allowing cancellation of a thread that's "stuck" on IO (due to a non-responsive socket/pipe peer, slow/stuck hardware, etc.) without imposing any measurable performance cost.
34 lines
689 B
C
34 lines
689 B
C
#include "stdio_impl.h"
|
|
|
|
FILE *fopen(const char *filename, const char *mode)
|
|
{
|
|
FILE *f;
|
|
int fd;
|
|
int flags;
|
|
int plus = !!strchr(mode, '+');
|
|
|
|
/* Check for valid initial mode character */
|
|
if (!strchr("rwa", *mode)) {
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
|
|
/* Compute the flags to pass to open() */
|
|
if (plus) flags = O_RDWR;
|
|
else if (*mode == 'r') flags = O_RDONLY;
|
|
else flags = O_WRONLY;
|
|
if (*mode != 'r') flags |= O_CREAT;
|
|
if (*mode == 'w') flags |= O_TRUNC;
|
|
if (*mode == 'a') flags |= O_APPEND;
|
|
|
|
fd = syscall_cp(SYS_open, filename, flags|O_LARGEFILE, 0666);
|
|
if (fd < 0) return 0;
|
|
|
|
f = __fdopen(fd, mode);
|
|
if (f) return f;
|
|
|
|
__syscall(SYS_close, fd);
|
|
return 0;
|
|
}
|
|
|
|
LFS64(fopen);
|