commit e56563ce3fb7ae2fb13f66ba6045ffb1f828ae08 (tree)
parent 12cfc96e1b25d4c75fd08b9af72226502982195a
Author: Andrew Kelley <andrew@ziglang.org>
Date: Tue, 13 Jan 2026 21:23:44 -0800
std.Io.File.MultiReader: implementation fixes
Diffstat:
4 files changed, 87 insertions(+), 80 deletions(-)
diff --git a/lib/std/Build/Step.zig b/lib/std/Build/Step.zig
@@ -381,13 +381,15 @@ pub fn addError(step: *Step, comptime fmt: []const u8, args: anytype) error{OutO
pub const ZigProcess = struct {
child: std.process.Child,
+ multi_reader_buffer: Io.File.MultiReader.Buffer(2),
+ multi_reader: Io.File.MultiReader,
progress_ipc_fd: if (std.Progress.have_ipc) ?std.posix.fd_t else void,
pub const StreamEnum = enum { stdout, stderr };
- pub fn deinit(zp: *ZigProcess, gpa: Allocator, io: Io) void {
- _ = gpa;
+ pub fn deinit(zp: *ZigProcess, io: Io) void {
zp.child.kill(io);
+ zp.multi_reader.deinit();
zp.* = undefined;
}
};
@@ -460,14 +462,18 @@ pub fn evalZigProcess(
.request_resource_usage_statistics = true,
.progress_node = prog_node,
}) catch |err| return s.fail("failed to spawn zig compiler {s}: {t}", .{ argv[0], err });
- defer if (!watch) zp.child.kill(io);
zp.* = .{
.child = zp.child,
+ .multi_reader_buffer = undefined,
+ .multi_reader = undefined,
.progress_ipc_fd = if (std.Progress.have_ipc) prog_node.getIpcFd() else {},
};
+ zp.multi_reader.init(gpa, io, zp.multi_reader_buffer.toStreams(), &.{
+ zp.child.stdout.?, zp.child.stderr.?,
+ });
if (watch) s.setZigProcess(zp);
- defer if (!watch) zp.deinit(gpa, io);
+ defer if (!watch) zp.deinit(io);
const result = try zigProcessUpdate(s, zp, watch, web_server, gpa);
@@ -534,18 +540,18 @@ fn zigProcessUpdate(s: *Step, zp: *ZigProcess, watch: bool, web_server: ?*Build.
var result: ?Path = null;
- var multi_reader_buffer: Io.File.MultiReader.Buffer(2) = undefined;
- var multi_reader: Io.File.MultiReader = undefined;
- multi_reader.init(gpa, io, multi_reader_buffer.toStreams(), &.{ zp.child.stdout.?, zp.child.stderr.? });
- defer multi_reader.deinit();
-
- const stdout = multi_reader.reader(0);
- const stderr = multi_reader.reader(1);
+ const stdout = zp.multi_reader.fileReader(0);
while (true) {
const Header = std.zig.Server.Message.Header;
- const header = try stdout.takeStruct(Header, .little);
- const body = try stdout.take(header.bytes_len);
+ const header = stdout.interface.takeStruct(Header, .little) catch |err| switch (err) {
+ error.EndOfStream => break,
+ error.ReadFailed => return stdout.err.?,
+ };
+ const body = stdout.interface.take(header.bytes_len) catch |err| switch (err) {
+ error.EndOfStream => |e| return e,
+ error.ReadFailed => return stdout.err.?,
+ };
switch (header.tag) {
.zig_version => {
if (!std.mem.eql(u8, builtin.zig_version_string, body)) {
@@ -636,7 +642,7 @@ fn zigProcessUpdate(s: *Step, zp: *ZigProcess, watch: bool, web_server: ?*Build.
s.result_duration_ns = timer.read();
- const stderr_contents = stderr.buffered();
+ const stderr_contents = zp.multi_reader.reader(1).buffered();
if (stderr_contents.len > 0) {
try s.result_error_msgs.append(arena, try arena.dupe(u8, stderr_contents));
}
diff --git a/lib/std/Build/Step/Run.zig b/lib/std/Build/Step/Run.zig
@@ -1695,11 +1695,9 @@ fn evalZigTest(
// The runner unexpectedly closed a stdio pipe, which means a crash. Make sure we've captured
// all available stderr to make our error output as useful as possible.
const stderr_fr = multi_reader.fileReader(1);
- while (true) {
- stderr_fr.interface.fillMore() catch |e| switch (e) {
- error.ReadFailed => return stderr_fr.err.?,
- error.EndOfStream => break,
- };
+ while (stderr_fr.interface.fillMore()) |_| {} else |e| switch (e) {
+ error.ReadFailed => return stderr_fr.err.?,
+ error.EndOfStream => {},
}
run.step.result_stderr = try arena.dupe(u8, stderr_fr.interface.buffered());
@@ -1905,7 +1903,7 @@ fn waitZigTest(
.clock = .awake,
} } else .none;
- multi_reader.fill(timeout) catch |err| switch (err) {
+ multi_reader.fill(64, timeout) catch |err| switch (err) {
error.Timeout, error.EndOfStream => return .{ .no_poll = .{
.active_test_index = active_test_index,
.ns_elapsed = if (timer) |*t| t.read() else 0,
@@ -2227,7 +2225,7 @@ fn evalGeneric(run: *Run, spawn_options: process.SpawnOptions) !EvalGenericResul
const stdout_reader = multi_reader.reader(0);
const stderr_reader = multi_reader.reader(1);
- while (multi_reader.fill(.none)) |_| {
+ while (multi_reader.fill(64, .none)) |_| {
if (run.stdio_limit.toInt()) |limit| {
if (stdout_reader.buffered().len > limit)
return error.StdoutStreamTooLong;
diff --git a/lib/std/Io/File/MultiReader.zig b/lib/std/Io/File/MultiReader.zig
@@ -28,18 +28,23 @@ pub const Streams = extern struct {
len: u32,
pub fn contexts(s: *Streams) []Context {
- _ = s;
- @panic("TODO");
+ const base: usize = @intFromPtr(s);
+ const ptr: [*]Context = @ptrFromInt(std.mem.alignForward(usize, base + @sizeOf(Streams), @alignOf(Context)));
+ return ptr[0..s.len];
}
pub fn ring(s: *Streams) []u32 {
- _ = s;
- @panic("TODO");
+ const prev = contexts(s);
+ const end = prev.ptr + prev.len;
+ const ptr: [*]u32 = @ptrFromInt(std.mem.alignForward(usize, @intFromPtr(end), @alignOf(u32)));
+ return ptr[0..s.len];
}
pub fn operations(s: *Streams) []Io.Operation {
- _ = s;
- @panic("TODO");
+ const prev = ring(s);
+ const end = prev.ptr + prev.len;
+ const ptr: [*]Io.Operation = @ptrFromInt(std.mem.alignForward(usize, @intFromPtr(end), @alignOf(Io.Operation)));
+ return ptr[0..s.len];
}
};
@@ -51,6 +56,7 @@ pub fn Buffer(comptime n: usize) type {
operations: [n][@sizeOf(Io.Operation)]u8 align(@alignOf(Io.Operation)),
pub fn toStreams(b: *@This()) *Streams {
+ b.len = n;
return @ptrCast(b);
}
};
@@ -157,61 +163,43 @@ fn stream(r: *Io.Reader, w: *Io.Writer, limit: Io.Limit) Io.Reader.StreamError!u
_ = w;
const fr: *File.Reader = @alignCast(@fieldParentPtr("interface", r));
const context: *Context = @fieldParentPtr("fr", fr);
- const mr = context.mr;
- return fillUntimed(mr, context);
+ try fillUntimed(context, 1);
+ return 0;
}
fn discard(r: *Io.Reader, limit: Io.Limit) Io.Reader.Error!usize {
_ = limit;
const fr: *File.Reader = @alignCast(@fieldParentPtr("interface", r));
const context: *Context = @fieldParentPtr("fr", fr);
- const mr = context.mr;
- return fillUntimed(mr, context);
+ try fillUntimed(context, 1);
+ return 0;
}
fn readVec(r: *Io.Reader, data: [][]u8) Io.Reader.Error!usize {
_ = data;
const fr: *File.Reader = @alignCast(@fieldParentPtr("interface", r));
const context: *Context = @fieldParentPtr("fr", fr);
- const mr = context.mr;
- return fillUntimed(mr, context);
+ try fillUntimed(context, 1);
+ return 0;
}
fn rebase(r: *Io.Reader, capacity: usize) Io.Reader.RebaseError!void {
const fr: *File.Reader = @alignCast(@fieldParentPtr("interface", r));
const context: *Context = @fieldParentPtr("fr", fr);
- const mr = context.mr;
-
- return rebaseGrowing(mr, context, capacity) catch |err| {
- context.err = err;
- return error.ReadFailed;
- };
+ try fillUntimed(context, capacity);
}
-fn rebaseGrowing(mr: *MultiReader, context: *Context, capacity: usize) Allocator.Error!void {
- const gpa = mr.gpa;
- const r = &context.fr.interface;
- if (r.buffer.len >= capacity) {
- const data = r.buffer[r.seek..r.end];
- @memmove(r.buffer[0..data.len], data);
- r.seek = 0;
- r.end = data.len;
- } else {
- const adjusted_capacity = std.ArrayList(u8).growCapacity(capacity);
-
- if (r.seek == 0) {
- if (gpa.remap(r.buffer, adjusted_capacity)) |new_memory| {
- r.buffer = new_memory;
- return;
- }
- }
-
- const data = r.buffer[r.seek..r.end];
- const new = try gpa.alloc(u8, adjusted_capacity);
- @memcpy(new[0..data.len], data);
- r.seek = 0;
- r.end = data.len;
- }
+fn fillUntimed(context: *Context, capacity: usize) Io.Reader.Error!void {
+ fill(context.mr, capacity, .none) catch |err| switch (err) {
+ error.Timeout, error.UnsupportedClock => unreachable,
+ error.Canceled, error.ConcurrencyUnavailable => |e| {
+ context.err = e;
+ return error.ReadFailed;
+ },
+ error.EndOfStream => |e| return e,
+ };
+ if (context.err != null) return error.ReadFailed;
+ if (context.eos) return error.EndOfStream;
}
pub const FillError = Io.Batch.WaitError || error{
@@ -221,7 +209,7 @@ pub const FillError = Io.Batch.WaitError || error{
};
/// Wait until at least one stream receives more data.
-pub fn fill(mr: *MultiReader, timeout: Io.Timeout) FillError!void {
+pub fn fill(mr: *MultiReader, unused_capacity: usize, timeout: Io.Timeout) FillError!void {
const contexts = mr.streams.contexts();
const operations = mr.streams.operations();
const io = contexts[0].fr.io;
@@ -243,14 +231,14 @@ pub fn fill(mr: *MultiReader, timeout: Io.Timeout) FillError!void {
}
const r = &context.fr.interface;
r.end += n;
- if (r.buffer.len - r.end == 0) {
- rebaseGrowing(mr, context, r.bufferedLen() + 1) catch |err| {
+ if (r.buffer.len - r.end < unused_capacity) {
+ rebaseGrowing(mr, context, r.bufferedLen() + unused_capacity) catch |err| {
context.err = err;
continue;
};
assert(r.seek == 0);
- context.vec[0] = r.buffer;
}
+ context.vec[0] = r.buffer[r.end..];
operation.file_read_streaming.status = .{ .unstarted = {} };
mr.batch.add(i);
}
@@ -258,16 +246,30 @@ pub fn fill(mr: *MultiReader, timeout: Io.Timeout) FillError!void {
if (!any_completed) return error.EndOfStream;
}
-fn fillUntimed(mr: *MultiReader, context: *Context) Io.Reader.Error!usize {
- fill(mr, .none) catch |err| switch (err) {
- error.Timeout, error.UnsupportedClock => unreachable,
- error.Canceled, error.ConcurrencyUnavailable => |e| {
- context.err = e;
- return error.ReadFailed;
- },
- error.EndOfStream => |e| return e,
- };
- if (context.err != null) return error.ReadFailed;
- if (context.eos) return error.EndOfStream;
- return 0;
+fn rebaseGrowing(mr: *MultiReader, context: *Context, capacity: usize) Allocator.Error!void {
+ const gpa = mr.gpa;
+ const r = &context.fr.interface;
+ if (r.buffer.len >= capacity) {
+ const data = r.buffer[r.seek..r.end];
+ @memmove(r.buffer[0..data.len], data);
+ r.seek = 0;
+ r.end = data.len;
+ } else {
+ const adjusted_capacity = std.ArrayList(u8).growCapacity(capacity);
+
+ if (r.seek == 0) {
+ if (gpa.remap(r.buffer, adjusted_capacity)) |new_memory| {
+ r.buffer = new_memory;
+ return;
+ }
+ }
+
+ const data = r.buffer[r.seek..r.end];
+ const new = try gpa.alloc(u8, adjusted_capacity);
+ @memcpy(new[0..data.len], data);
+ gpa.free(r.buffer);
+ r.buffer = new;
+ r.seek = 0;
+ r.end = data.len;
+ }
}
diff --git a/lib/std/crypto/tls/Client.zig b/lib/std/crypto/tls/Client.zig
@@ -336,10 +336,11 @@ pub fn init(input: *Reader, output: *Writer, options: Options) InitError!Client
// Ensure the input buffer pointer is stable in this scope.
input.rebase(tls.max_ciphertext_record_len) catch |err| switch (err) {
error.EndOfStream => {}, // We have assurance the remainder of stream can be buffered.
+ error.ReadFailed => |e| return e,
};
const record_header = input.peek(tls.record_header_len) catch |err| switch (err) {
error.EndOfStream => return error.TlsConnectionTruncated,
- error.ReadFailed => return error.ReadFailed,
+ error.ReadFailed => |e| return e,
};
const record_ct = input.takeEnumNonexhaustive(tls.ContentType, .big) catch unreachable; // already peeked
input.toss(2); // legacy_version