/* * Copyright (c) 2013 Travis Geiselbrecht * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files * (the "Software"), to deal in the Software without restriction, * including without limitation the rights to use, copy, modify, merge, * publish, distribute, sublicense, and/or sell copies of the Software, * and to permit persons to whom the Software is furnished to do so, * subject to the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #include #include #include #include #include #include #if LK_LIBC_IMPLEMENTATION_IS_LK #define DEFINE_STDIO_DESC(id) \ [(id)] = { \ .io = &console_io, \ } __WEAK FILE __stdio_FILEs[3] = { DEFINE_STDIO_DESC(0), /* stdin */ DEFINE_STDIO_DESC(1), /* stdout */ DEFINE_STDIO_DESC(2), /* stderr */ }; #undef DEFINE_STDIO_DESC #endif static size_t lock_write_commit_unlock(FILE *fp, const char* s, size_t length) { size_t bytes_written; io_handle_t *io = file_io_handle(fp); io_lock(io); bytes_written = io_write(io, s, length); io_write_commit(io); io_unlock(io); return bytes_written; } int fputc(int _c, FILE *fp) { unsigned char c = _c; return lock_write_commit_unlock(fp, (char *)&c, 1); } int putchar(int c) { return fputc(c, stdout); } int puts(const char *str) { int err = fputs(str, stdout); if (err >= 0) err = fputc('\n', stdout); return err; } int fputs(const char *s, FILE *fp) { size_t len = strlen(s); return lock_write_commit_unlock(fp, s, len); } size_t fwrite(const void *ptr, size_t size, size_t count, FILE *fp) { size_t bytes_written; if (size == 0 || count == 0) return 0; // fast path for size == 1 if (likely(size == 1)) { return lock_write_commit_unlock(fp, ptr, count); } bytes_written = lock_write_commit_unlock(fp, ptr, size * count); return bytes_written / size; } int getc(FILE *fp) { char c; io_handle_t *io = file_io_handle(fp); ssize_t ret = io_read(io, &c, sizeof(c)); return (ret > 0) ? c : ret; } int getchar(void) { return getc(stdin); } static int _fprintf_output_func(const char *str, size_t len, void *state) { io_handle_t *io = file_io_handle((FILE *)state); return io_write(io, str, len); } int vfprintf_worker(FILE *fp, const char *fmt, va_list ap, int filtered_on_release) { io_handle_t *io = file_io_handle(fp); io_lock(io); int result = _printf_engine(&_fprintf_output_func, (void *)fp, fmt, ap); io_write_commit(io); io_unlock(io); return result; } int fprintf(FILE *fp, const char *fmt, ...) { va_list ap; int err; va_start(ap, fmt); err = vfprintf(fp, fmt, ap); va_end(ap); return err; } int printf(const char *fmt, ...) { #if DISABLE_DEBUG_OUTPUT return 0; #else va_list ap; int err; va_start(ap, fmt); err = vfprintf(stdout, fmt, ap); va_end(ap); return err; #endif } int vprintf(const char *fmt, va_list ap) { #if DISABLE_DEBUG_OUTPUT return 0; #else return vfprintf(stdout, fmt, ap); #endif } #if LK_LIBC_IMPLEMENTATION_IS_LK int fflush(FILE *stream) { /* nothing to flush; output streams aren't buffered */ return 0; } #endif