Skip to content

Commit 483c057

Browse files
committed
Merge branch 'clean up writeFileAllUnseekable by using readers'
closes #7156
2 parents 3468872 + d68adc5 commit 483c057

File tree

5 files changed

+66
-20
lines changed

5 files changed

+66
-20
lines changed

CMakeLists.txt

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -375,10 +375,11 @@ set(ZIG_STAGE2_SOURCES
375375
"${CMAKE_SOURCE_DIR}/lib/std/io/buffered_atomic_file.zig"
376376
"${CMAKE_SOURCE_DIR}/lib/std/io/buffered_writer.zig"
377377
"${CMAKE_SOURCE_DIR}/lib/std/io/change_detection_stream.zig"
378-
"${CMAKE_SOURCE_DIR}/lib/std/io/counting_writer.zig"
379378
"${CMAKE_SOURCE_DIR}/lib/std/io/counting_reader.zig"
379+
"${CMAKE_SOURCE_DIR}/lib/std/io/counting_writer.zig"
380380
"${CMAKE_SOURCE_DIR}/lib/std/io/find_byte_writer.zig"
381381
"${CMAKE_SOURCE_DIR}/lib/std/io/fixed_buffer_stream.zig"
382+
"${CMAKE_SOURCE_DIR}/lib/std/io/limited_reader.zig"
382383
"${CMAKE_SOURCE_DIR}/lib/std/io/reader.zig"
383384
"${CMAKE_SOURCE_DIR}/lib/std/io/seekable_stream.zig"
384385
"${CMAKE_SOURCE_DIR}/lib/std/io/writer.zig"

lib/std/fs/file.zig

Lines changed: 9 additions & 18 deletions
Original file line numberDiff line numberDiff line change
@@ -698,7 +698,7 @@ pub const File = struct {
698698
header_count: usize = 0,
699699
};
700700

701-
pub const WriteFileError = ReadError || WriteError;
701+
pub const WriteFileError = ReadError || error{EndOfStream} || WriteError;
702702

703703
pub fn writeFileAll(self: File, in_file: File, args: WriteFileOptions) WriteFileError!void {
704704
return self.writeFileAllSendfile(in_file, args) catch |err| switch (err) {
@@ -722,23 +722,14 @@ pub const File = struct {
722722

723723
try self.writevAll(headers);
724724

725-
var buffer: [4096]u8 = undefined;
726-
{
727-
var index: usize = 0;
728-
// Skip in_offset bytes.
729-
while (index < args.in_offset) {
730-
const ask = math.min(buffer.len, args.in_offset - index);
731-
const amt = try in_file.read(buffer[0..ask]);
732-
index += amt;
733-
}
734-
}
735-
const in_len = args.in_len orelse math.maxInt(u64);
736-
var index: usize = 0;
737-
while (index < in_len) {
738-
const ask = math.min(buffer.len, in_len - index);
739-
const amt = try in_file.read(buffer[0..ask]);
740-
if (amt == 0) break;
741-
index += try self.write(buffer[0..amt]);
725+
try in_file.reader().skipBytes(args.in_offset, .{ .buf_size = 4096 });
726+
727+
var fifo = std.fifo.LinearFifo(u8, .{ .Static = 4096 }).init();
728+
if (args.in_len) |len| {
729+
var stream = std.io.limitedReader(in_file.reader(), len);
730+
try fifo.pump(stream.reader(), self.writer());
731+
} else {
732+
try fifo.pump(in_file.reader(), self.writer());
742733
}
743734

744735
try self.writevAll(trailers);

lib/std/io.zig

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -125,6 +125,9 @@ pub const fixedBufferStream = @import("io/fixed_buffer_stream.zig").fixedBufferS
125125
pub const CWriter = @import("io/c_writer.zig").CWriter;
126126
pub const cWriter = @import("io/c_writer.zig").cWriter;
127127

128+
pub const LimitedReader = @import("io/limited_reader.zig").LimitedReader;
129+
pub const limitedReader = @import("io/limited_reader.zig").limitedReader;
130+
128131
pub const CountingWriter = @import("io/counting_writer.zig").CountingWriter;
129132
pub const countingWriter = @import("io/counting_writer.zig").countingWriter;
130133
pub const CountingReader = @import("io/counting_reader.zig").CountingReader;

lib/std/io/limited_reader.zig

Lines changed: 50 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
// SPDX-License-Identifier: MIT
2+
// Copyright (c) 2015-2020 Zig Contributors
3+
// This file is part of [zig](https://ziglang.org/), which is MIT licensed.
4+
// The MIT license requires this copyright notice to be included in all copies
5+
// and substantial portions of the software.
6+
const std = @import("../std.zig");
7+
const io = std.io;
8+
const assert = std.debug.assert;
9+
const testing = std.testing;
10+
11+
pub fn LimitedReader(comptime ReaderType: type) type {
12+
return struct {
13+
inner_reader: ReaderType,
14+
bytes_left: u64,
15+
16+
pub const Error = ReaderType.Error;
17+
pub const Reader = io.Reader(*Self, Error, read);
18+
19+
const Self = @This();
20+
21+
pub fn read(self: *Self, dest: []u8) Error!usize {
22+
const max_read = std.math.min(self.bytes_left, dest.len);
23+
const n = try self.inner_reader.read(dest[0..max_read]);
24+
self.bytes_left -= n;
25+
return n;
26+
}
27+
28+
pub fn reader(self: *Self) Reader {
29+
return .{ .context = self };
30+
}
31+
};
32+
}
33+
34+
/// Returns an initialised `LimitedReader`
35+
/// `bytes_left` is a `u64` to be able to take 64 bit file offsets
36+
pub fn limitedReader(inner_reader: anytype, bytes_left: u64) LimitedReader(@TypeOf(inner_reader)) {
37+
return .{ .inner_reader = inner_reader, .bytes_left = bytes_left };
38+
}
39+
40+
test "basic usage" {
41+
const data = "hello world";
42+
var fbs = std.io.fixedBufferStream(data);
43+
var early_stream = limitedReader(fbs.reader(), 3);
44+
45+
var buf: [5]u8 = undefined;
46+
testing.expectEqual(@as(usize, 3), try early_stream.reader().read(&buf));
47+
testing.expectEqualSlices(u8, data[0..3], buf[0..3]);
48+
testing.expectEqual(@as(usize, 0), try early_stream.reader().read(&buf));
49+
testing.expectError(error.EndOfStream, early_stream.reader().skipBytes(10, .{}));
50+
}

lib/std/io/reader.zig

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -271,8 +271,9 @@ pub fn Reader(
271271
buf_size: usize = 512,
272272
};
273273

274+
// `num_bytes` is a `u64` to match `off_t`
274275
/// Reads `num_bytes` bytes from the stream and discards them
275-
pub fn skipBytes(self: Self, num_bytes: usize, comptime options: SkipBytesOptions) !void {
276+
pub fn skipBytes(self: Self, num_bytes: u64, comptime options: SkipBytesOptions) !void {
276277
var buf: [options.buf_size]u8 = undefined;
277278
var remaining = num_bytes;
278279

0 commit comments

Comments
 (0)