diff options
author | Leo Tenenbaum <pommicket@gmail.com> | 2020-05-08 21:41:45 -0400 |
---|---|---|
committer | Leo Tenenbaum <pommicket@gmail.com> | 2020-05-08 21:41:45 -0400 |
commit | d31a2f61610fc23b63b643018167f518ce3e8b01 (patch) | |
tree | caed61c229c2a6b4bee9713fefa5d5d161aa43df /std | |
parent | 6886389e7c55b382545026b668dd400949edf337 (diff) |
fixed a few bugs starting to make buffered io
Diffstat (limited to 'std')
-rw-r--r-- | std/base.toc | 4 | ||||
-rw-r--r-- | std/io.toc | 94 |
2 files changed, 49 insertions, 49 deletions
diff --git a/std/base.toc b/std/base.toc index bdb91e0..08c5e53 100644 --- a/std/base.toc +++ b/std/base.toc @@ -20,4 +20,8 @@ PLATFORM ::= #builtin("platform"); libc ::= "libc.so.6"; } +PLATFORM_IS_UNIX :: bool = PLATFORM == PLATFORM_LINUX || PLATFORM == PLATFORM_OSX || PLATFORM == PLATFORM_FREEBSD + || PLATFORM == PLATFORM_OPENBSD || PLATFORM == PLATFORM_MISC_UNIX; +// @TODO: write your own +memcpy ::= #foreign("memcpy", libc) fn(&void, #C &"const void", #C size_t) &void; @@ -1,61 +1,57 @@ -/* @TODO: use write / WriteFile */ - #include "std/base.toc", base; -putchar ::= #foreign("putchar", base.libc) fn(#C int) #C int; -toc_putchar ::= fn(x: char) { - putchar(x as #C int); -} -printf ::= #foreign("printf", base.libc) fn (#C &"const char", #C ..) #C int; - -/* -unfortunately, we can't use fwrite because MSVC doesn't like it when you -use a file handle that's not from the DLL. (i.e. you can't pass your stdout -to the imported version of fwrite) -*/ -writes ::= fn(x: []char) { - for c := x { - toc_putchar(c); +#if base.PLATFORM_IS_UNIX { + // @TODO: use syscall instead + write ::= #foreign("write", "libc.so.6") fn(#C int, #C &"const void", #C size_t) #C long; + RawFile ::= int; + raw_file_write ::= fn(file: RawFile, buf: &void, size: int) int { + write(file as #C int, buf, size as #C size_t) as int + } + get_stdout ::= fn() RawFile { + 1 + } +} else { + // @TODO: on windows, use WriteFile + fwrite ::= #foreign("fwrite", base.libc) fn(#C &"const void", #C size_t, #C size_t, &void) #C size_t; + RawFile ::= &void; + raw_file_write ::= fn(file: RawFile, buf: &void, size: int) int { + fwrite(buf, 1, size, file); + } + get_stdout ::= fn() RawFile { + #builtin("stdout") } } -puts ::= fn(x: []char) { - writes(x); - toc_putchar('\n'); -} -writei ::= fn(x: int) { - if x < 0 { - toc_putchar('-'); - // NOTE: don't do x = -x; here to make sure I64_MIN works - } - if x == 0 { - toc_putchar('0'); - } else { - abs ::= fn(x: int) int { if x < 0 { -x } else { x } }; - scan_digit := 1000000000000000000; - started := false; - while scan_digit > 0 { - digit := abs((x / scan_digit) % 10); - if digit > 0 { started = true; } - if started { - toc_putchar((('0' as int) + digit) as char); - } - scan_digit /= 10; - } - } +File ::= struct { + BUFSZ ::= 4096; + raw : RawFile; + written : int; // ranges from 0 to FILE_BUFSZ-1 + buffer : [BUFSZ]char; } -puti ::= fn(x: int) { - writei(x); - toc_putchar('\n'); +raw_file_to_file ::= fn(raw : RawFile) f: File { + f.raw = raw; } -writef ::= fn(x: float) { - fmt := "%f\0"; - printf(&fmt[0], x as f64); +std_out := raw_file_to_file(get_stdout()); + +file_flush ::= fn(use f: &File) { + raw_file_write(raw, &buffer[0], written); + written = 0; } -putf ::= fn(x: float) { - writef(x); - toc_putchar('\n'); + +file_writes ::= fn(use f: &File, s : []char) { + if written + s.len > BUFSZ { + file_flush(f); + if s.len > BUFSZ { + raw_file_write(raw, &s[0], s.len); + } else { + base.memcpy(&buffer[0], &s[0], s.len as #C size_t); + written = s.len; + } + } else { + base.memcpy(&buffer[written], &s[0], s.len as #C size_t); + written += s.len; + } } |