2019-05-24 22:27:18 +00:00
|
|
|
const std = @import("../std.zig");
|
|
|
|
const builtin = @import("builtin");
|
|
|
|
const os = std.os;
|
|
|
|
const io = std.io;
|
|
|
|
const mem = std.mem;
|
|
|
|
const math = std.math;
|
|
|
|
const assert = std.debug.assert;
|
|
|
|
const windows = os.windows;
|
|
|
|
const Os = builtin.Os;
|
|
|
|
const maxInt = std.math.maxInt;
|
2020-02-06 22:56:40 +00:00
|
|
|
const need_async_thread = std.fs.need_async_thread;
|
2019-05-24 22:27:18 +00:00
|
|
|
|
|
|
|
pub const File = struct {
|
|
|
|
/// The OS-specific file descriptor or file handle.
|
|
|
|
handle: os.fd_t,
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
/// On some systems, such as Linux, file system file descriptors are incapable of non-blocking I/O.
|
|
|
|
/// This forces us to perform asynchronous I/O on a dedicated thread, to achieve non-blocking
|
|
|
|
/// file-system I/O. To do this, `File` must be aware of whether it is a file system file descriptor,
|
|
|
|
/// or, more specifically, whether the I/O is blocking.
|
|
|
|
io_mode: io.Mode,
|
|
|
|
|
|
|
|
/// Even when std.io.mode is async, it is still sometimes desirable to perform blocking I/O, although
|
|
|
|
/// not by default. For example, when printing a stack trace to stderr.
|
|
|
|
async_block_allowed: @TypeOf(async_block_allowed_no) = async_block_allowed_no,
|
|
|
|
|
|
|
|
pub const async_block_allowed_yes = if (io.is_async) true else {};
|
|
|
|
pub const async_block_allowed_no = if (io.is_async) false else {};
|
|
|
|
|
|
|
|
pub const Mode = os.mode_t;
|
2019-05-24 22:27:18 +00:00
|
|
|
|
|
|
|
pub const default_mode = switch (builtin.os) {
|
2020-02-06 22:56:40 +00:00
|
|
|
.windows => 0,
|
2019-05-24 22:27:18 +00:00
|
|
|
else => 0o666,
|
|
|
|
};
|
|
|
|
|
|
|
|
pub const OpenError = windows.CreateFileError || os.OpenError;
|
|
|
|
|
2019-11-30 18:32:11 +00:00
|
|
|
/// TODO https://github.com/ziglang/zig/issues/3802
|
|
|
|
pub const OpenFlags = struct {
|
|
|
|
read: bool = true,
|
|
|
|
write: bool = false,
|
|
|
|
};
|
|
|
|
|
|
|
|
/// TODO https://github.com/ziglang/zig/issues/3802
|
|
|
|
pub const CreateFlags = struct {
|
|
|
|
/// Whether the file will be created with read access.
|
|
|
|
read: bool = false,
|
|
|
|
|
|
|
|
/// If the file already exists, and is a regular file, and the access
|
|
|
|
/// mode allows writing, it will be truncated to length 0.
|
|
|
|
truncate: bool = true,
|
|
|
|
|
|
|
|
/// Ensures that this open call creates the file, otherwise causes
|
|
|
|
/// `error.FileAlreadyExists` to be returned.
|
|
|
|
exclusive: bool = false,
|
|
|
|
|
|
|
|
/// For POSIX systems this is the file system mode the file will
|
|
|
|
/// be created with.
|
|
|
|
mode: Mode = default_mode,
|
|
|
|
};
|
|
|
|
|
2019-05-24 22:27:18 +00:00
|
|
|
/// Upon success, the stream is in an uninitialized state. To continue using it,
|
|
|
|
/// you must use the open() function.
|
|
|
|
pub fn close(self: File) void {
|
2020-02-06 22:56:40 +00:00
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
std.event.Loop.instance.?.close(self.handle);
|
|
|
|
} else {
|
|
|
|
return os.close(self.handle);
|
|
|
|
}
|
2019-05-24 22:27:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Test whether the file refers to a terminal.
|
|
|
|
/// See also `supportsAnsiEscapeCodes`.
|
|
|
|
pub fn isTty(self: File) bool {
|
|
|
|
return os.isatty(self.handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Test whether ANSI escape codes will be treated as such.
|
|
|
|
pub fn supportsAnsiEscapeCodes(self: File) bool {
|
2019-10-24 05:06:03 +00:00
|
|
|
if (builtin.os == .windows) {
|
2019-05-24 22:27:18 +00:00
|
|
|
return os.isCygwinPty(self.handle);
|
|
|
|
}
|
2019-10-23 05:21:16 +00:00
|
|
|
if (self.isTty()) {
|
|
|
|
if (self.handle == os.STDOUT_FILENO or self.handle == os.STDERR_FILENO) {
|
|
|
|
// Use getenvC to workaround https://github.com/ziglang/zig/issues/3511
|
2019-11-20 01:29:08 +00:00
|
|
|
if (os.getenvC("TERM")) |term| {
|
2019-10-23 05:21:16 +00:00
|
|
|
if (std.mem.eql(u8, term, "dumb"))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2019-05-24 22:27:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub const SeekError = os.SeekError;
|
|
|
|
|
|
|
|
/// Repositions read/write file offset relative to the current offset.
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn seekBy(self: File, offset: i64) SeekError!void {
|
|
|
|
return os.lseek_CUR(self.handle, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Repositions read/write file offset relative to the end.
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn seekFromEnd(self: File, offset: i64) SeekError!void {
|
|
|
|
return os.lseek_END(self.handle, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Repositions read/write file offset relative to the beginning.
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn seekTo(self: File, offset: u64) SeekError!void {
|
|
|
|
return os.lseek_SET(self.handle, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const GetPosError = os.SeekError || os.FStatError;
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn getPos(self: File) GetPosError!u64 {
|
|
|
|
return os.lseek_CUR_get(self.handle);
|
|
|
|
}
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn getEndPos(self: File) GetPosError!u64 {
|
2019-10-24 05:06:03 +00:00
|
|
|
if (builtin.os == .windows) {
|
2019-05-24 22:27:18 +00:00
|
|
|
return windows.GetFileSizeEx(self.handle);
|
|
|
|
}
|
2019-07-14 07:06:20 +00:00
|
|
|
return (try self.stat()).size;
|
2019-05-24 22:27:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub const ModeError = os.FStatError;
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn mode(self: File) ModeError!Mode {
|
2019-10-24 05:06:03 +00:00
|
|
|
if (builtin.os == .windows) {
|
2019-05-24 22:27:18 +00:00
|
|
|
return {};
|
|
|
|
}
|
2019-07-14 07:06:20 +00:00
|
|
|
return (try self.stat()).mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const Stat = struct {
|
|
|
|
size: u64,
|
|
|
|
mode: Mode,
|
|
|
|
|
|
|
|
/// access time in nanoseconds
|
2019-07-15 05:41:06 +00:00
|
|
|
atime: i64,
|
2019-07-14 07:06:20 +00:00
|
|
|
|
|
|
|
/// last modification time in nanoseconds
|
2019-07-15 05:41:06 +00:00
|
|
|
mtime: i64,
|
2019-07-14 07:06:20 +00:00
|
|
|
|
|
|
|
/// creation time in nanoseconds
|
2019-07-15 05:41:06 +00:00
|
|
|
ctime: i64,
|
2019-07-14 07:06:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
pub const StatError = os.FStatError;
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-07-14 07:06:20 +00:00
|
|
|
pub fn stat(self: File) StatError!Stat {
|
2019-10-24 05:06:03 +00:00
|
|
|
if (builtin.os == .windows) {
|
2019-07-15 21:54:50 +00:00
|
|
|
var io_status_block: windows.IO_STATUS_BLOCK = undefined;
|
|
|
|
var info: windows.FILE_ALL_INFORMATION = undefined;
|
|
|
|
const rc = windows.ntdll.NtQueryInformationFile(self.handle, &io_status_block, &info, @sizeOf(windows.FILE_ALL_INFORMATION), .FileAllInformation);
|
|
|
|
switch (rc) {
|
2020-01-31 08:47:00 +00:00
|
|
|
.SUCCESS => {},
|
|
|
|
.BUFFER_OVERFLOW => {},
|
|
|
|
.INVALID_PARAMETER => unreachable,
|
|
|
|
.ACCESS_DENIED => return error.AccessDenied,
|
2019-07-15 21:54:50 +00:00
|
|
|
else => return windows.unexpectedStatus(rc),
|
|
|
|
}
|
2019-07-14 07:06:20 +00:00
|
|
|
return Stat{
|
2019-07-15 21:54:50 +00:00
|
|
|
.size = @bitCast(u64, info.StandardInformation.EndOfFile),
|
2020-02-07 19:54:58 +00:00
|
|
|
.mode = 0,
|
2019-07-15 21:54:50 +00:00
|
|
|
.atime = windows.fromSysTime(info.BasicInformation.LastAccessTime),
|
|
|
|
.mtime = windows.fromSysTime(info.BasicInformation.LastWriteTime),
|
|
|
|
.ctime = windows.fromSysTime(info.BasicInformation.CreationTime),
|
2019-07-14 07:06:20 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const st = try os.fstat(self.handle);
|
2019-07-15 16:28:39 +00:00
|
|
|
const atime = st.atime();
|
|
|
|
const mtime = st.mtime();
|
|
|
|
const ctime = st.ctime();
|
2019-07-14 07:06:20 +00:00
|
|
|
return Stat{
|
|
|
|
.size = @bitCast(u64, st.size),
|
|
|
|
.mode = st.mode,
|
2019-11-07 04:25:57 +00:00
|
|
|
.atime = @as(i64, atime.tv_sec) * std.time.ns_per_s + atime.tv_nsec,
|
|
|
|
.mtime = @as(i64, mtime.tv_sec) * std.time.ns_per_s + mtime.tv_nsec,
|
|
|
|
.ctime = @as(i64, ctime.tv_sec) * std.time.ns_per_s + ctime.tv_nsec,
|
2019-07-14 07:06:20 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-07-15 04:02:44 +00:00
|
|
|
pub const UpdateTimesError = os.FutimensError || windows.SetFileTimeError;
|
2019-07-14 07:06:20 +00:00
|
|
|
|
2019-10-16 22:13:40 +00:00
|
|
|
/// The underlying file system may have a different granularity than nanoseconds,
|
|
|
|
/// and therefore this function cannot guarantee any precision will be stored.
|
|
|
|
/// Further, the maximum value is limited by the system ABI. When a value is provided
|
|
|
|
/// that exceeds this range, the value is clamped to the maximum.
|
2020-02-06 22:56:40 +00:00
|
|
|
/// TODO: integrate with async I/O
|
2019-10-16 22:13:40 +00:00
|
|
|
pub fn updateTimes(
|
|
|
|
self: File,
|
|
|
|
/// access timestamp in nanoseconds
|
|
|
|
atime: i64,
|
|
|
|
/// last modification timestamp in nanoseconds
|
|
|
|
mtime: i64,
|
|
|
|
) UpdateTimesError!void {
|
2019-10-24 05:06:03 +00:00
|
|
|
if (builtin.os == .windows) {
|
2019-07-15 04:02:44 +00:00
|
|
|
const atime_ft = windows.nanoSecondsToFileTime(atime);
|
|
|
|
const mtime_ft = windows.nanoSecondsToFileTime(mtime);
|
|
|
|
return windows.SetFileTime(self.handle, null, &atime_ft, &mtime_ft);
|
|
|
|
}
|
2019-07-14 07:06:20 +00:00
|
|
|
const times = [2]os.timespec{
|
|
|
|
os.timespec{
|
2019-10-12 10:21:20 +00:00
|
|
|
.tv_sec = math.cast(isize, @divFloor(atime, std.time.ns_per_s)) catch maxInt(isize),
|
|
|
|
.tv_nsec = math.cast(isize, @mod(atime, std.time.ns_per_s)) catch maxInt(isize),
|
2019-07-14 07:06:20 +00:00
|
|
|
},
|
|
|
|
os.timespec{
|
2019-10-12 10:21:20 +00:00
|
|
|
.tv_sec = math.cast(isize, @divFloor(mtime, std.time.ns_per_s)) catch maxInt(isize),
|
|
|
|
.tv_nsec = math.cast(isize, @mod(mtime, std.time.ns_per_s)) catch maxInt(isize),
|
2019-07-14 07:06:20 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
try os.futimens(self.handle, ×);
|
2019-05-24 22:27:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub const ReadError = os.ReadError;
|
|
|
|
|
|
|
|
pub fn read(self: File, buffer: []u8) ReadError!usize {
|
2020-02-06 22:56:40 +00:00
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.read(self.handle, buffer);
|
|
|
|
}
|
2019-05-24 22:27:18 +00:00
|
|
|
return os.read(self.handle, buffer);
|
|
|
|
}
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
pub fn pread(self: File, buffer: []u8, offset: u64) ReadError!usize {
|
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.pread(self.handle, buffer);
|
|
|
|
}
|
|
|
|
return os.pread(self.handle, buffer, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn readv(self: File, iovecs: []const os.iovec) ReadError!usize {
|
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.readv(self.handle, iovecs);
|
|
|
|
}
|
|
|
|
return os.readv(self.handle, iovecs);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn preadv(self: File, iovecs: []const os.iovec, offset: u64) ReadError!usize {
|
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.preadv(self.handle, iovecs, offset);
|
|
|
|
}
|
|
|
|
return os.preadv(self.handle, iovecs, offset);
|
|
|
|
}
|
|
|
|
|
2019-05-24 22:27:18 +00:00
|
|
|
pub const WriteError = os.WriteError;
|
|
|
|
|
|
|
|
pub fn write(self: File, bytes: []const u8) WriteError!void {
|
2020-02-06 22:56:40 +00:00
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.write(self.handle, bytes);
|
|
|
|
}
|
2019-05-24 22:27:18 +00:00
|
|
|
return os.write(self.handle, bytes);
|
|
|
|
}
|
|
|
|
|
2020-02-06 22:56:40 +00:00
|
|
|
pub fn pwrite(self: File, bytes: []const u8, offset: u64) WriteError!void {
|
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.pwrite(self.handle, bytes, offset);
|
|
|
|
}
|
|
|
|
return os.pwrite(self.handle, bytes, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn writev(self: File, iovecs: []const os.iovec_const) WriteError!void {
|
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.writev(self.handle, iovecs);
|
|
|
|
}
|
|
|
|
return os.writev(self.handle, iovecs);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn pwritev(self: File, iovecs: []const os.iovec_const, offset: usize) WriteError!void {
|
|
|
|
if (need_async_thread and self.io_mode == .blocking and !self.async_block_allowed) {
|
|
|
|
return std.event.Loop.instance.?.pwritev(self.handle, iovecs);
|
2019-08-17 01:29:29 +00:00
|
|
|
}
|
2020-02-06 22:56:40 +00:00
|
|
|
return os.pwritev(self.handle, iovecs);
|
2019-08-17 01:29:29 +00:00
|
|
|
}
|
|
|
|
|
2019-05-24 22:27:18 +00:00
|
|
|
pub fn inStream(file: File) InStream {
|
|
|
|
return InStream{
|
|
|
|
.file = file,
|
|
|
|
.stream = InStream.Stream{ .readFn = InStream.readFn },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn outStream(file: File) OutStream {
|
|
|
|
return OutStream{
|
|
|
|
.file = file,
|
|
|
|
.stream = OutStream.Stream{ .writeFn = OutStream.writeFn },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn seekableStream(file: File) SeekableStream {
|
|
|
|
return SeekableStream{
|
|
|
|
.file = file,
|
|
|
|
.stream = SeekableStream.Stream{
|
|
|
|
.seekToFn = SeekableStream.seekToFn,
|
|
|
|
.seekByFn = SeekableStream.seekByFn,
|
|
|
|
.getPosFn = SeekableStream.getPosFn,
|
|
|
|
.getEndPosFn = SeekableStream.getEndPosFn,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Implementation of io.InStream trait for File
|
|
|
|
pub const InStream = struct {
|
|
|
|
file: File,
|
|
|
|
stream: Stream,
|
|
|
|
|
|
|
|
pub const Error = ReadError;
|
|
|
|
pub const Stream = io.InStream(Error);
|
|
|
|
|
|
|
|
fn readFn(in_stream: *Stream, buffer: []u8) Error!usize {
|
|
|
|
const self = @fieldParentPtr(InStream, "stream", in_stream);
|
|
|
|
return self.file.read(buffer);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Implementation of io.OutStream trait for File
|
|
|
|
pub const OutStream = struct {
|
|
|
|
file: File,
|
|
|
|
stream: Stream,
|
|
|
|
|
|
|
|
pub const Error = WriteError;
|
|
|
|
pub const Stream = io.OutStream(Error);
|
|
|
|
|
|
|
|
fn writeFn(out_stream: *Stream, bytes: []const u8) Error!void {
|
|
|
|
const self = @fieldParentPtr(OutStream, "stream", out_stream);
|
|
|
|
return self.file.write(bytes);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Implementation of io.SeekableStream trait for File
|
|
|
|
pub const SeekableStream = struct {
|
|
|
|
file: File,
|
|
|
|
stream: Stream,
|
|
|
|
|
|
|
|
pub const Stream = io.SeekableStream(SeekError, GetPosError);
|
|
|
|
|
|
|
|
pub fn seekToFn(seekable_stream: *Stream, pos: u64) SeekError!void {
|
|
|
|
const self = @fieldParentPtr(SeekableStream, "stream", seekable_stream);
|
|
|
|
return self.file.seekTo(pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn seekByFn(seekable_stream: *Stream, amt: i64) SeekError!void {
|
|
|
|
const self = @fieldParentPtr(SeekableStream, "stream", seekable_stream);
|
|
|
|
return self.file.seekBy(amt);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn getEndPosFn(seekable_stream: *Stream) GetPosError!u64 {
|
|
|
|
const self = @fieldParentPtr(SeekableStream, "stream", seekable_stream);
|
|
|
|
return self.file.getEndPos();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn getPosFn(seekable_stream: *Stream) GetPosError!u64 {
|
|
|
|
const self = @fieldParentPtr(SeekableStream, "stream", seekable_stream);
|
|
|
|
return self.file.getPos();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|