#include "file.h" #include #include #include #include #include #include #include #include static FILE _stdin_null = { .fd = STDIN_FILENO }; static FILE _stdout_null = { .fd = STDOUT_FILENO }; static FILE _stderr_null = { .fd = STDERR_FILENO }; FILE *const stdin = &_stdin_null; FILE *const stdout = &_stdout_null; FILE *const stderr = &_stderr_null; FILE *fopen(const char *path, const char *mode) { FILE *f; handle_t h; int flags = 0; char *tmppath = NULL; if (!path) { errno = 1; return NULL; } else if (path[0] == '!') { /* special handling for "!files" */ path++; if (!strcmp(path, "stdin")) return file_clone(stdin, mode); if (!strcmp(path, "stdout")) return file_clone(stdout, mode); if (!strcmp(path, "stderr")) return file_clone(stderr, mode); errno = 1; return NULL; } if (path && path[0] != '/') { size_t len = absolutepath(NULL, path, 0); tmppath = malloc(len); if (!tmppath) return NULL; absolutepath(tmppath, path, len); path = tmppath; } if (strchr(mode, 'e')) { /* camellia extension: open as executable */ flags |= OPEN_EXEC; } else if (strchr(mode, 'r')) { flags |= OPEN_READ; if (strchr(mode, '+')) flags |= OPEN_WRITE; } else { flags |= OPEN_WRITE | OPEN_CREATE; } h = _syscall_open(path, strlen(path), flags); if (tmppath) free(tmppath); if (h < 0) { errno = -h; return NULL; } if (mode[0] == 'w') _syscall_write(h, NULL, 0, 0, WRITE_TRUNCATE); f = fdopen(h, mode); if (!f) close(h); return f; } FILE *freopen(const char *path, const char *mode, FILE *f) { /* partially based on the musl implementation of freopen */ FILE *f2; if (!path) goto fail; f2 = fopen(path, mode); if (!f2) goto fail; if (f->fd == f2->fd) { f2->fd = -1; } else { if (_syscall_dup(f2->fd, f->fd, 0) < 0) goto fail2; } f->pos = f2->pos; f->eof = f2->eof; fclose(f2); return f; fail2: fclose(f2); fail: fclose(f); return NULL; } FILE *fdopen(int fd, const char *mode) { FILE *f; f = malloc(sizeof *f); if (!f) return NULL; f->fd = fd; f->pos = mode[0] == 'a' ? -1 : 0; f->eof = false; f->error = false; f->extflags = 0; return f; } FILE *file_clone(const FILE *f, const char *mode) { handle_t h = _syscall_dup(f->fd, -1, 0); FILE *f2; if (h < 0) return NULL; f2 = fdopen(h, mode); if (!f2) { close(h); return NULL; } f2->pos = f->pos; f2->eof = f->eof; f2->fd = h; return f2; } // TODO popen / pclose FILE *popen(const char *cmd, const char *mode) { (void)cmd; (void)mode; errno = ENOSYS; return NULL; } int pclose(FILE *f) { (void)f; errno = ENOSYS; return -1; } // TODO tmpfile() FILE *tmpfile(void) { errno = ENOSYS; return NULL; } int fextflags(FILE *f, int extflags) { int old = f->extflags; f->extflags = extflags; return old; } int setvbuf(FILE *restrict f, char *restrict buf, int type, size_t size) { (void)f; (void)buf; (void)size; if (type == _IONBF) return 0; errno = ENOSYS; return -1; } static void fadvance(long amt, FILE *f) { bool pos_neg = f->pos < 0; f->pos += amt; if (pos_neg && f->pos >= 0) f->pos = -1; } size_t fread(void *restrict ptr, size_t size, size_t nitems, FILE *restrict f) { size_t total = size*nitems, pos = 0; unsigned char *buf = ptr; if (f->fd < 0) { errno = EBADF; return 0; } if (size == 0) return 0; while (pos < total) { long res = _syscall_read(f->fd, buf + pos, total - pos, f->pos); if (res < 0) { f->error = true; errno = -res; break; } else if (res == 0) { f->eof = true; break; } pos += res; fadvance(res, f); if (f->extflags & FEXT_NOFILL) break; } return pos == total ? nitems : (pos/size); } size_t fwrite(const void *restrict ptr, size_t size, size_t nitems, FILE *restrict f) { size_t total = size*nitems, pos = 0; const unsigned char *buf = ptr; if (f->fd < 0) { errno = EBADF; return 0; } if (size == 0) return 0; while (pos < total) { long res = _syscall_write(f->fd, buf + pos, total - pos, f->pos, 0); if (res < 0) { f->error = true; errno = -res; return pos/size; } else if (res == 0) { f->eof = true; return pos/size; } else { pos += res; fadvance(res, f); } } return nitems; } int fputs(const char *s, FILE *f) { return fprintf(f, "%s\n", s); } char *fgets(char *buf, int size, FILE *f) { char c = '\0'; long pos = 0; while (pos < (size-1) && c != '\n' && fread(&c, 1, 1, f)) buf[pos++] = c; buf[pos++] = '\0'; if (f->eof && pos == 1) return NULL; if (f->error) return NULL; return buf; } int fgetc(FILE *f) { char c; return fread(&c, 1, 1, f) ? c : EOF; } int getc(FILE *f) { return fgetc(f); } int fputc(int c, FILE *f) { return fwrite(&c, 1, 1, f) ? c : EOF; } int putc(int c, FILE *f) { return fputc(c, f); } // TODO ungetc int ungetc(int c, FILE *f) { (void)c; (void)f; __libc_panic("unimplemented"); } int fseek(FILE *f, long offset, int whence) { return fseeko(f, offset, whence); } int fseeko(FILE *f, off_t offset, int whence) { if (fflush(f)) return -1; long base; switch (whence) { case SEEK_SET: base = 0; break; case SEEK_CUR: base = f->pos; break; case SEEK_END: base = _syscall_getsize(f->fd); if (base < 0) base = -1; break; default: errno = EINVAL; return -1; } if (base >= 0 && base + offset < 0) { /* underflow */ errno = EINVAL; return -1; } else if (base < 0 && base + offset >= 0) { /* overflow - went from a negative offset (relative to EOF) * to a positive offset (from start of file). * can only happen when getsize() is unsupported */ errno = EINVAL; return -1; } f->pos = base + offset; f->eof = false; return 0; } long ftell(FILE *f) { return ftello(f); } off_t ftello(FILE *f) { return f->pos; } int fclose(FILE *f) { fflush(f); if (f->fd > 0) close(f->fd); if (f != &_stdin_null && f != &_stdout_null && f != &_stderr_null) free(f); return 0; } int fflush(FILE *f) { (void)f; return 0; } int feof(FILE *f) { return f->eof; } int ferror(FILE *f) { return f->error; } void clearerr(FILE *f) { f->error = false; f->eof = false; }