mirror of
https://github.com/nim-lang/Nim.git
synced 2025-12-29 17:34:43 +00:00
443 lines
14 KiB
Nim
443 lines
14 KiB
Nim
#
|
|
#
|
|
# Nim's Runtime Library
|
|
# (c) Copyright 2013 Andreas Rumpf
|
|
#
|
|
# See the file "copying.txt", included in this
|
|
# distribution, for details about the copyright.
|
|
#
|
|
|
|
|
|
# Nim's standard IO library. It contains high-performance
|
|
# routines for reading and writing data to (buffered) files or
|
|
# TTYs.
|
|
|
|
{.push debugger:off .} # the user does not want to trace a part
|
|
# of the standard library!
|
|
|
|
when defined(windows):
|
|
proc c_fdopen(filehandle: cint, mode: cstring): File {.
|
|
importc: "_fdopen", header: "<stdio.h>".}
|
|
else:
|
|
proc c_fdopen(filehandle: cint, mode: cstring): File {.
|
|
importc: "fdopen", header: "<stdio.h>".}
|
|
proc c_fputs(c: cstring, f: File): cint {.
|
|
importc: "fputs", header: "<stdio.h>", tags: [WriteIOEffect].}
|
|
proc c_fgets(c: cstring, n: cint, f: File): cstring {.
|
|
importc: "fgets", header: "<stdio.h>", tags: [ReadIOEffect].}
|
|
proc c_fgetc(stream: File): cint {.
|
|
importc: "fgetc", header: "<stdio.h>", tags: [ReadIOEffect].}
|
|
proc c_ungetc(c: cint, f: File): cint {.
|
|
importc: "ungetc", header: "<stdio.h>", tags: [].}
|
|
proc c_putc(c: cint, stream: File): cint {.
|
|
importc: "putc", header: "<stdio.h>", tags: [WriteIOEffect].}
|
|
proc c_fflush(f: File): cint {.
|
|
importc: "fflush", header: "<stdio.h>".}
|
|
proc c_fclose(f: File): cint {.
|
|
importc: "fclose", header: "<stdio.h>".}
|
|
proc c_clearerr(f: File) {.
|
|
importc: "clearerr", header: "<stdio.h>".}
|
|
proc c_feof(f: File): cint {.
|
|
importc: "feof", header: "<stdio.h>".}
|
|
|
|
when not declared(c_fwrite):
|
|
proc c_fwrite(buf: pointer, size, n: csize, f: File): cint {.
|
|
importc: "fwrite", header: "<stdio.h>".}
|
|
|
|
# C routine that is used here:
|
|
proc c_fread(buf: pointer, size, n: csize, f: File): csize {.
|
|
importc: "fread", header: "<stdio.h>", tags: [ReadIOEffect].}
|
|
when defined(windows):
|
|
when not defined(amd64):
|
|
proc c_fseek(f: File, offset: int64, whence: cint): cint {.
|
|
importc: "fseek", header: "<stdio.h>", tags: [].}
|
|
proc c_ftell(f: File): int64 {.
|
|
importc: "ftell", header: "<stdio.h>", tags: [].}
|
|
else:
|
|
proc c_fseek(f: File, offset: int64, whence: cint): cint {.
|
|
importc: "_fseeki64", header: "<stdio.h>", tags: [].}
|
|
proc c_ftell(f: File): int64 {.
|
|
importc: "_ftelli64", header: "<stdio.h>", tags: [].}
|
|
else:
|
|
proc c_fseek(f: File, offset: int64, whence: cint): cint {.
|
|
importc: "fseeko", header: "<stdio.h>", tags: [].}
|
|
proc c_ftell(f: File): int64 {.
|
|
importc: "ftello", header: "<stdio.h>", tags: [].}
|
|
proc c_ferror(f: File): cint {.
|
|
importc: "ferror", header: "<stdio.h>", tags: [].}
|
|
proc c_setvbuf(f: File, buf: pointer, mode: cint, size: csize): cint {.
|
|
importc: "setvbuf", header: "<stdio.h>", tags: [].}
|
|
|
|
proc raiseEIO(msg: string) {.noinline, noreturn.} =
|
|
sysFatal(IOError, msg)
|
|
|
|
proc raiseEOF() {.noinline, noreturn.} =
|
|
sysFatal(EOFError, "EOF reached")
|
|
|
|
proc checkErr(f: File) =
|
|
if c_ferror(f) != 0:
|
|
c_clearerr(f)
|
|
raiseEIO("Unknown IO Error")
|
|
|
|
{.push stackTrace:off, profiler:off.}
|
|
proc readBuffer(f: File, buffer: pointer, len: Natural): int =
|
|
result = c_fread(buffer, 1, len, f)
|
|
if result != len: checkErr(f)
|
|
|
|
proc readBytes(f: File, a: var openArray[int8|uint8], start, len: Natural): int =
|
|
result = readBuffer(f, addr(a[start]), len)
|
|
|
|
proc readChars(f: File, a: var openArray[char], start, len: Natural): int =
|
|
if (start + len) > len(a):
|
|
raiseEIO("buffer overflow: (start+len) > length of openarray buffer")
|
|
result = readBuffer(f, addr(a[start]), len)
|
|
|
|
proc write(f: File, c: cstring) =
|
|
discard c_fputs(c, f)
|
|
checkErr(f)
|
|
|
|
proc writeBuffer(f: File, buffer: pointer, len: Natural): int =
|
|
result = c_fwrite(buffer, 1, len, f)
|
|
checkErr(f)
|
|
|
|
proc writeBytes(f: File, a: openArray[int8|uint8], start, len: Natural): int =
|
|
var x = cast[ptr UncheckedArray[int8]](a)
|
|
result = writeBuffer(f, addr(x[int(start)]), len)
|
|
proc writeChars(f: File, a: openArray[char], start, len: Natural): int =
|
|
var x = cast[ptr UncheckedArray[int8]](a)
|
|
result = writeBuffer(f, addr(x[int(start)]), len)
|
|
|
|
proc write(f: File, s: string) =
|
|
if writeBuffer(f, cstring(s), s.len) != s.len:
|
|
raiseEIO("cannot write string to file")
|
|
{.pop.}
|
|
|
|
when NoFakeVars:
|
|
when defined(windows):
|
|
const
|
|
IOFBF = cint(0)
|
|
IONBF = cint(4)
|
|
else:
|
|
# On all systems I could find, including Linux, Mac OS X, and the BSDs
|
|
const
|
|
IOFBF = cint(0)
|
|
IONBF = cint(2)
|
|
else:
|
|
var
|
|
IOFBF {.importc: "_IOFBF", nodecl.}: cint
|
|
IONBF {.importc: "_IONBF", nodecl.}: cint
|
|
|
|
const
|
|
BufSize = 4000
|
|
|
|
proc close*(f: File) =
|
|
if not f.isNil:
|
|
discard c_fclose(f)
|
|
|
|
proc readChar(f: File): char =
|
|
let x = c_fgetc(f)
|
|
if x < 0:
|
|
checkErr(f)
|
|
raiseEOF()
|
|
result = char(x)
|
|
|
|
proc flushFile*(f: File) = discard c_fflush(f)
|
|
proc getFileHandle*(f: File): FileHandle = c_fileno(f)
|
|
|
|
proc readLine(f: File, line: var TaintedString): bool =
|
|
var pos = 0
|
|
|
|
# Use the currently reserved space for a first try
|
|
var sp = max(line.string.len, 80)
|
|
line.string.setLen(sp)
|
|
|
|
while true:
|
|
# memset to \L so that we can tell how far fgets wrote, even on EOF, where
|
|
# fgets doesn't append an \L
|
|
nimSetMem(addr line.string[pos], '\L'.ord, sp)
|
|
var fgetsSuccess = c_fgets(addr line.string[pos], sp.cint, f) != nil
|
|
if not fgetsSuccess: checkErr(f)
|
|
let m = c_memchr(addr line.string[pos], '\L'.ord, sp)
|
|
if m != nil:
|
|
# \l found: Could be our own or the one by fgets, in any case, we're done
|
|
var last = cast[ByteAddress](m) - cast[ByteAddress](addr line.string[0])
|
|
if last > 0 and line.string[last-1] == '\c':
|
|
line.string.setLen(last-1)
|
|
return last > 1 or fgetsSuccess
|
|
# We have to distinguish between two possible cases:
|
|
# \0\l\0 => line ending in a null character.
|
|
# \0\l\l => last line without newline, null was put there by fgets.
|
|
elif last > 0 and line.string[last-1] == '\0':
|
|
if last < pos + sp - 1 and line.string[last+1] != '\0':
|
|
dec last
|
|
line.string.setLen(last)
|
|
return last > 0 or fgetsSuccess
|
|
else:
|
|
# fgets will have inserted a null byte at the end of the string.
|
|
dec sp
|
|
# No \l found: Increase buffer and read more
|
|
inc pos, sp
|
|
sp = 128 # read in 128 bytes at a time
|
|
line.string.setLen(pos+sp)
|
|
|
|
proc readLine(f: File): TaintedString =
|
|
result = TaintedString(newStringOfCap(80))
|
|
if not readLine(f, result): raiseEOF()
|
|
|
|
proc write(f: File, i: int) =
|
|
when sizeof(int) == 8:
|
|
if c_fprintf(f, "%lld", i) < 0: checkErr(f)
|
|
else:
|
|
if c_fprintf(f, "%ld", i) < 0: checkErr(f)
|
|
|
|
proc write(f: File, i: BiggestInt) =
|
|
when sizeof(BiggestInt) == 8:
|
|
if c_fprintf(f, "%lld", i) < 0: checkErr(f)
|
|
else:
|
|
if c_fprintf(f, "%ld", i) < 0: checkErr(f)
|
|
|
|
proc write(f: File, b: bool) =
|
|
if b: write(f, "true")
|
|
else: write(f, "false")
|
|
proc write(f: File, r: float32) =
|
|
if c_fprintf(f, "%.16g", r) < 0: checkErr(f)
|
|
proc write(f: File, r: BiggestFloat) =
|
|
if c_fprintf(f, "%.16g", r) < 0: checkErr(f)
|
|
|
|
proc write(f: File, c: char) = discard c_putc(cint(c), f)
|
|
proc write(f: File, a: varargs[string, `$`]) =
|
|
for x in items(a): write(f, x)
|
|
|
|
proc readAllBuffer(file: File): string =
|
|
# This proc is for File we want to read but don't know how many
|
|
# bytes we need to read before the buffer is empty.
|
|
result = ""
|
|
var buffer = newString(BufSize)
|
|
while true:
|
|
var bytesRead = readBuffer(file, addr(buffer[0]), BufSize)
|
|
if bytesRead == BufSize:
|
|
result.add(buffer)
|
|
else:
|
|
buffer.setLen(bytesRead)
|
|
result.add(buffer)
|
|
break
|
|
|
|
proc rawFileSize(file: File): int64 =
|
|
# this does not raise an error opposed to `getFileSize`
|
|
var oldPos = c_ftell(file)
|
|
discard c_fseek(file, 0, 2) # seek the end of the file
|
|
result = c_ftell(file)
|
|
discard c_fseek(file, oldPos, 0)
|
|
|
|
proc endOfFile(f: File): bool =
|
|
var c = c_fgetc(f)
|
|
discard c_ungetc(c, f)
|
|
return c < 0'i32
|
|
#result = c_feof(f) != 0
|
|
|
|
proc readAllFile(file: File, len: int64): string =
|
|
# We acquire the filesize beforehand and hope it doesn't change.
|
|
# Speeds things up.
|
|
result = newString(len)
|
|
let bytes = readBuffer(file, addr(result[0]), len)
|
|
if endOfFile(file):
|
|
if bytes < len:
|
|
result.setLen(bytes)
|
|
else:
|
|
# We read all the bytes but did not reach the EOF
|
|
# Try to read it as a buffer
|
|
result.add(readAllBuffer(file))
|
|
|
|
proc readAllFile(file: File): string =
|
|
var len = rawFileSize(file)
|
|
result = readAllFile(file, len)
|
|
|
|
proc readAll(file: File): TaintedString =
|
|
# Separate handling needed because we need to buffer when we
|
|
# don't know the overall length of the File.
|
|
when declared(stdin):
|
|
let len = if file != stdin: rawFileSize(file) else: -1
|
|
else:
|
|
let len = rawFileSize(file)
|
|
if len > 0:
|
|
result = readAllFile(file, len).TaintedString
|
|
else:
|
|
result = readAllBuffer(file).TaintedString
|
|
|
|
proc writeLn[Ty](f: File, x: varargs[Ty, `$`]) =
|
|
for i in items(x):
|
|
write(f, i)
|
|
write(f, "\n")
|
|
|
|
proc writeLine[Ty](f: File, x: varargs[Ty, `$`]) =
|
|
for i in items(x):
|
|
write(f, i)
|
|
write(f, "\n")
|
|
|
|
when declared(stdout):
|
|
proc rawEcho(x: string) {.inline, compilerproc.} = write(stdout, x)
|
|
proc rawEchoNL() {.inline, compilerproc.} = write(stdout, "\n")
|
|
|
|
# interface to the C procs:
|
|
|
|
include "system/widestrs"
|
|
|
|
when defined(windows) and not defined(useWinAnsi):
|
|
when defined(cpp):
|
|
proc wfopen(filename, mode: WideCString): pointer {.
|
|
importcpp: "_wfopen((const wchar_t*)#, (const wchar_t*)#)", nodecl.}
|
|
proc wfreopen(filename, mode: WideCString, stream: File): File {.
|
|
importcpp: "_wfreopen((const wchar_t*)#, (const wchar_t*)#, #)", nodecl.}
|
|
else:
|
|
proc wfopen(filename, mode: WideCString): pointer {.
|
|
importc: "_wfopen", nodecl.}
|
|
proc wfreopen(filename, mode: WideCString, stream: File): File {.
|
|
importc: "_wfreopen", nodecl.}
|
|
|
|
proc fopen(filename, mode: cstring): pointer =
|
|
var f = newWideCString(filename)
|
|
var m = newWideCString(mode)
|
|
result = wfopen(f, m)
|
|
|
|
proc freopen(filename, mode: cstring, stream: File): File =
|
|
var f = newWideCString(filename)
|
|
var m = newWideCString(mode)
|
|
result = wfreopen(f, m, stream)
|
|
|
|
else:
|
|
proc fopen(filename, mode: cstring): pointer {.importc: "fopen", noDecl.}
|
|
proc freopen(filename, mode: cstring, stream: File): File {.
|
|
importc: "freopen", nodecl.}
|
|
|
|
const
|
|
FormatOpen: array[FileMode, string] = ["rb", "wb", "w+b", "r+b", "ab"]
|
|
#"rt", "wt", "w+t", "r+t", "at"
|
|
# we always use binary here as for Nim the OS line ending
|
|
# should not be translated.
|
|
|
|
when defined(posix) and not defined(nimscript):
|
|
when defined(linux) and defined(amd64):
|
|
type
|
|
Mode {.importc: "mode_t", header: "<sys/types.h>".} = cint
|
|
|
|
# fillers ensure correct size & offsets
|
|
Stat {.importc: "struct stat",
|
|
header: "<sys/stat.h>", final, pure.} = object ## struct stat
|
|
filler_1: array[24, char]
|
|
st_mode: Mode ## Mode of file
|
|
filler_2: array[144 - 24 - 4, char]
|
|
|
|
proc S_ISDIR(m: Mode): bool =
|
|
## Test for a directory.
|
|
(m and 0o170000) == 0o40000
|
|
|
|
else:
|
|
type
|
|
Mode {.importc: "mode_t", header: "<sys/types.h>".} = cint
|
|
|
|
Stat {.importc: "struct stat",
|
|
header: "<sys/stat.h>", final, pure.} = object ## struct stat
|
|
st_mode: Mode ## Mode of file
|
|
|
|
proc S_ISDIR(m: Mode): bool {.importc, header: "<sys/stat.h>".}
|
|
## Test for a directory.
|
|
|
|
proc c_fstat(a1: cint, a2: var Stat): cint {.
|
|
importc: "fstat", header: "<sys/stat.h>".}
|
|
|
|
proc open(f: var File, filename: string,
|
|
mode: FileMode = fmRead,
|
|
bufSize: int = -1): bool =
|
|
var p: pointer = fopen(filename, FormatOpen[mode])
|
|
if p != nil:
|
|
when defined(posix) and not defined(nimscript):
|
|
# How `fopen` handles opening a directory is not specified in ISO C and
|
|
# POSIX. We do not want to handle directories as regular files that can
|
|
# be opened.
|
|
var f2 = cast[File](p)
|
|
var res: Stat
|
|
if c_fstat(getFileHandle(f2), res) >= 0'i32 and S_ISDIR(res.st_mode):
|
|
close(f2)
|
|
return false
|
|
result = true
|
|
f = cast[File](p)
|
|
if bufSize > 0 and bufSize <= high(cint).int:
|
|
discard c_setvbuf(f, nil, IOFBF, bufSize.cint)
|
|
elif bufSize == 0:
|
|
discard c_setvbuf(f, nil, IONBF, 0)
|
|
|
|
proc reopen(f: File, filename: string, mode: FileMode = fmRead): bool =
|
|
var p: pointer = freopen(filename, FormatOpen[mode], f)
|
|
result = p != nil
|
|
|
|
proc open(f: var File, filehandle: FileHandle, mode: FileMode): bool =
|
|
f = c_fdopen(filehandle, FormatOpen[mode])
|
|
result = f != nil
|
|
|
|
proc setFilePos(f: File, pos: int64, relativeTo: FileSeekPos = fspSet) =
|
|
if c_fseek(f, pos, cint(relativeTo)) != 0:
|
|
raiseEIO("cannot set file position")
|
|
|
|
proc getFilePos(f: File): int64 =
|
|
result = c_ftell(f)
|
|
if result < 0: raiseEIO("cannot retrieve file position")
|
|
|
|
proc getFileSize(f: File): int64 =
|
|
var oldPos = getFilePos(f)
|
|
discard c_fseek(f, 0, 2) # seek the end of the file
|
|
result = getFilePos(f)
|
|
setFilePos(f, oldPos)
|
|
|
|
proc readFile(filename: string): TaintedString =
|
|
var f: File
|
|
if open(f, filename):
|
|
try:
|
|
result = readAll(f).TaintedString
|
|
finally:
|
|
close(f)
|
|
else:
|
|
sysFatal(IOError, "cannot open: ", filename)
|
|
|
|
proc writeFile(filename, content: string) =
|
|
var f: File
|
|
if open(f, filename, fmWrite):
|
|
try:
|
|
f.write(content)
|
|
finally:
|
|
close(f)
|
|
else:
|
|
sysFatal(IOError, "cannot open: ", filename)
|
|
|
|
proc setStdIoUnbuffered() =
|
|
when declared(stdout):
|
|
discard c_setvbuf(stdout, nil, IONBF, 0)
|
|
when declared(stderr):
|
|
discard c_setvbuf(stderr, nil, IONBF, 0)
|
|
when declared(stdin):
|
|
discard c_setvbuf(stdin, nil, IONBF, 0)
|
|
|
|
when declared(stdout):
|
|
when defined(windows) and compileOption("threads"):
|
|
var echoLock: SysLock
|
|
initSysLock echoLock
|
|
|
|
proc echoBinSafe(args: openArray[string]) {.compilerProc.} =
|
|
# flockfile deadlocks some versions of Android 5.x.x
|
|
when not defined(windows) and not defined(android) and not defined(nintendoswitch):
|
|
proc flockfile(f: File) {.importc, noDecl.}
|
|
proc funlockfile(f: File) {.importc, noDecl.}
|
|
flockfile(stdout)
|
|
when defined(windows) and compileOption("threads"):
|
|
acquireSys echoLock
|
|
for s in args:
|
|
discard c_fwrite(s.cstring, s.len, 1, stdout)
|
|
const linefeed = "\n" # can be 1 or more chars
|
|
discard c_fwrite(linefeed.cstring, linefeed.len, 1, stdout)
|
|
discard c_fflush(stdout)
|
|
when not defined(windows) and not defined(android) and not defined(nintendoswitch):
|
|
funlockfile(stdout)
|
|
when defined(windows) and compileOption("threads"):
|
|
releaseSys echoLock
|
|
|
|
{.pop.}
|