From 4d134a01f5a8baae346783f19d9b5db8c8256d32 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 12:21:29 -0600
Subject: [PATCH 01/16] Move debug.global_allocator to testing.allocator
---
doc/docgen.zig | 2 +-
lib/std/array_list.zig | 14 +++----
lib/std/buffer.zig | 6 +--
lib/std/debug.zig | 7 ++--
lib/std/fifo.zig | 4 +-
lib/std/fmt.zig | 10 ++---
lib/std/fs/path.zig | 16 ++++----
lib/std/heap.zig | 4 ++
lib/std/linked_list.zig | 6 +--
lib/std/os/test.zig | 2 +-
lib/std/priority_queue.zig | 38 +++++++++----------
lib/std/process.zig | 8 ++--
lib/std/special/test_runner.zig | 1 +
lib/std/testing.zig | 5 +++
lib/std/unicode.zig | 12 +++---
lib/std/zig/ast.zig | 2 +-
test/compare_output.zig | 4 +-
test/compile_errors.zig | 2 +-
test/stage1/behavior/const_slice_child.zig | 2 +-
test/standalone/brace_expansion/main.zig | 4 +-
test/standalone/cat/main.zig | 2 +-
test/standalone/empty_env/main.zig | 2 +-
test/standalone/load_dynamic_library/main.zig | 4 +-
23 files changed, 84 insertions(+), 73 deletions(-)
diff --git a/doc/docgen.zig b/doc/docgen.zig
index 0218f50419..678eb26043 100644
--- a/doc/docgen.zig
+++ b/doc/docgen.zig
@@ -672,7 +672,7 @@ const TermState = enum {
test "term color" {
const input_bytes = "A\x1b[32;1mgreen\x1b[0mB";
- const result = try termColor(std.debug.global_allocator, input_bytes);
+ const result = try termColor(std.testing.allocator, input_bytes);
testing.expectEqualSlices(u8, "AgreenB", result);
}
diff --git a/lib/std/array_list.zig b/lib/std/array_list.zig
index 64f13eff9b..be99a54e2b 100644
--- a/lib/std/array_list.zig
+++ b/lib/std/array_list.zig
@@ -320,7 +320,7 @@ test "std.ArrayList.basic" {
}
test "std.ArrayList.orderedRemove" {
- var list = ArrayList(i32).init(debug.global_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -347,7 +347,7 @@ test "std.ArrayList.orderedRemove" {
}
test "std.ArrayList.swapRemove" {
- var list = ArrayList(i32).init(debug.global_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -374,7 +374,7 @@ test "std.ArrayList.swapRemove" {
}
test "std.ArrayList.swapRemoveOrError" {
- var list = ArrayList(i32).init(debug.global_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
// Test just after initialization
@@ -402,7 +402,7 @@ test "std.ArrayList.swapRemoveOrError" {
}
test "std.ArrayList.insert" {
- var list = ArrayList(i32).init(debug.global_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -416,7 +416,7 @@ test "std.ArrayList.insert" {
}
test "std.ArrayList.insertSlice" {
- var list = ArrayList(i32).init(debug.global_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -443,7 +443,7 @@ const Item = struct {
};
test "std.ArrayList: ArrayList(T) of struct T" {
- var root = Item{ .integer = 1, .sub_items = ArrayList(Item).init(debug.global_allocator) };
- try root.sub_items.append(Item{ .integer = 42, .sub_items = ArrayList(Item).init(debug.global_allocator) });
+ var root = Item{ .integer = 1, .sub_items = ArrayList(Item).init(testing.allocator) };
+ try root.sub_items.append(Item{ .integer = 42, .sub_items = ArrayList(Item).init(testing.allocator) });
testing.expect(root.sub_items.items[0].integer == 42);
}
diff --git a/lib/std/buffer.zig b/lib/std/buffer.zig
index 42bf8e8142..ada3590616 100644
--- a/lib/std/buffer.zig
+++ b/lib/std/buffer.zig
@@ -150,7 +150,7 @@ pub const Buffer = struct {
};
test "simple Buffer" {
- var buf = try Buffer.init(debug.global_allocator, "");
+ var buf = try Buffer.init(testing.allocator, "");
testing.expect(buf.len() == 0);
try buf.append("hello");
try buf.append(" ");
@@ -169,14 +169,14 @@ test "simple Buffer" {
}
test "Buffer.initSize" {
- var buf = try Buffer.initSize(debug.global_allocator, 3);
+ var buf = try Buffer.initSize(testing.allocator, 3);
testing.expect(buf.len() == 3);
try buf.append("hello");
testing.expect(mem.eql(u8, buf.toSliceConst()[3..], "hello"));
}
test "Buffer.initCapacity" {
- var buf = try Buffer.initCapacity(debug.global_allocator, 10);
+ var buf = try Buffer.initCapacity(testing.allocator, 10);
testing.expect(buf.len() == 0);
testing.expect(buf.capacity() >= 10);
const old_cap = buf.capacity();
diff --git a/lib/std/debug.zig b/lib/std/debug.zig
index d035707a54..f2dcdd7e77 100644
--- a/lib/std/debug.zig
+++ b/lib/std/debug.zig
@@ -20,7 +20,7 @@ const windows = std.os.windows;
pub const leb = @import("debug/leb128.zig");
pub const FailingAllocator = @import("debug/failing_allocator.zig").FailingAllocator;
-pub const failing_allocator = &FailingAllocator.init(global_allocator, 0).allocator;
+pub const failing_allocator = &FailingAllocator.init(&global_fixed_allocator.allocator, 0).allocator;
pub const runtime_safety = switch (builtin.mode) {
.Debug, .ReleaseSafe => true,
@@ -2192,8 +2192,9 @@ fn readInitialLength(comptime E: type, in_stream: *io.InStream(E), is_64: *bool)
}
}
-/// This should only be used in temporary test programs.
-pub const global_allocator = &global_fixed_allocator.allocator;
+pub const global_allocator = blk: {
+ @compileError("Please switch to std.testing.allocator.");
+};
var global_fixed_allocator = std.heap.ThreadSafeFixedBufferAllocator.init(global_allocator_mem[0..]);
var global_allocator_mem: [100 * 1024]u8 = undefined;
diff --git a/lib/std/fifo.zig b/lib/std/fifo.zig
index 7870979abb..ac8589f2ce 100644
--- a/lib/std/fifo.zig
+++ b/lib/std/fifo.zig
@@ -347,7 +347,7 @@ pub fn LinearFifo(
}
test "LinearFifo(u8, .Dynamic)" {
- var fifo = LinearFifo(u8, .Dynamic).init(debug.global_allocator);
+ var fifo = LinearFifo(u8, .Dynamic).init(testing.allocator);
defer fifo.deinit();
try fifo.write("HELLO");
@@ -422,7 +422,7 @@ test "LinearFifo" {
var fifo = switch (bt) {
.Static => FifoType.init(),
.Slice => FifoType.init(buf[0..]),
- .Dynamic => FifoType.init(debug.global_allocator),
+ .Dynamic => FifoType.init(testing.allocator),
};
defer fifo.deinit();
diff --git a/lib/std/fmt.zig b/lib/std/fmt.zig
index cf3322a995..9757658447 100644
--- a/lib/std/fmt.zig
+++ b/lib/std/fmt.zig
@@ -1598,7 +1598,7 @@ test "hexToBytes" {
test "formatIntValue with comptime_int" {
const value: comptime_int = 123456789123456789;
- var buf = try std.Buffer.init(std.debug.global_allocator, "");
+ var buf = try std.Buffer.init(std.testing.allocator, "");
try formatIntValue(value, "", FormatOptions{}, &buf, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append);
std.testing.expect(mem.eql(u8, buf.toSlice(), "123456789123456789"));
}
@@ -1652,19 +1652,19 @@ test "formatType max_depth" {
inst.a = &inst;
inst.tu.ptr = &inst.tu;
- var buf0 = try std.Buffer.init(std.debug.global_allocator, "");
+ var buf0 = try std.Buffer.init(std.testing.allocator, "");
try formatType(inst, "", FormatOptions{}, &buf0, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 0);
std.testing.expect(mem.eql(u8, buf0.toSlice(), "S{ ... }"));
- var buf1 = try std.Buffer.init(std.debug.global_allocator, "");
+ var buf1 = try std.Buffer.init(std.testing.allocator, "");
try formatType(inst, "", FormatOptions{}, &buf1, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 1);
std.testing.expect(mem.eql(u8, buf1.toSlice(), "S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }"));
- var buf2 = try std.Buffer.init(std.debug.global_allocator, "");
+ var buf2 = try std.Buffer.init(std.testing.allocator, "");
try formatType(inst, "", FormatOptions{}, &buf2, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 2);
std.testing.expect(mem.eql(u8, buf2.toSlice(), "S{ .a = S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ ... } }, .e = E.Two, .vec = (10.200,2.220) }"));
- var buf3 = try std.Buffer.init(std.debug.global_allocator, "");
+ var buf3 = try std.Buffer.init(std.testing.allocator, "");
try formatType(inst, "", FormatOptions{}, &buf3, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 3);
std.testing.expect(mem.eql(u8, buf3.toSlice(), "S{ .a = S{ .a = S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ ... } }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ .ptr = TU{ ... } } }, .e = E.Two, .vec = (10.200,2.220) }"));
}
diff --git a/lib/std/fs/path.zig b/lib/std/fs/path.zig
index b3ad3e9f7a..6c29a3950c 100644
--- a/lib/std/fs/path.zig
+++ b/lib/std/fs/path.zig
@@ -665,7 +665,7 @@ pub fn resolvePosix(allocator: *Allocator, paths: []const []const u8) ![]u8 {
}
test "resolve" {
- const cwd = try process.getCwdAlloc(debug.global_allocator);
+ const cwd = try process.getCwdAlloc(testing.allocator);
if (builtin.os == .windows) {
if (windowsParsePath(cwd).kind == WindowsPath.Kind.Drive) {
cwd[0] = asciiUpper(cwd[0]);
@@ -683,11 +683,11 @@ test "resolveWindows" {
return error.SkipZigTest;
}
if (builtin.os == .windows) {
- const cwd = try process.getCwdAlloc(debug.global_allocator);
+ const cwd = try process.getCwdAlloc(testing.allocator);
const parsed_cwd = windowsParsePath(cwd);
{
const result = testResolveWindows(&[_][]const u8{ "/usr/local", "lib\\zig\\std\\array_list.zig" });
- const expected = try join(debug.global_allocator, &[_][]const u8{
+ const expected = try join(testing.allocator, &[_][]const u8{
parsed_cwd.disk_designator,
"usr\\local\\lib\\zig\\std\\array_list.zig",
});
@@ -698,7 +698,7 @@ test "resolveWindows" {
}
{
const result = testResolveWindows(&[_][]const u8{ "usr/local", "lib\\zig" });
- const expected = try join(debug.global_allocator, &[_][]const u8{
+ const expected = try join(testing.allocator, &[_][]const u8{
cwd,
"usr\\local\\lib\\zig",
});
@@ -738,11 +738,11 @@ test "resolvePosix" {
}
fn testResolveWindows(paths: []const []const u8) []u8 {
- return resolveWindows(debug.global_allocator, paths) catch unreachable;
+ return resolveWindows(testing.allocator, paths) catch unreachable;
}
fn testResolvePosix(paths: []const []const u8) []u8 {
- return resolvePosix(debug.global_allocator, paths) catch unreachable;
+ return resolvePosix(testing.allocator, paths) catch unreachable;
}
/// If the path is a file in the current directory (no directory component)
@@ -1166,11 +1166,11 @@ test "relative" {
}
fn testRelativePosix(from: []const u8, to: []const u8, expected_output: []const u8) void {
- const result = relativePosix(debug.global_allocator, from, to) catch unreachable;
+ const result = relativePosix(testing.allocator, from, to) catch unreachable;
testing.expectEqualSlices(u8, expected_output, result);
}
fn testRelativeWindows(from: []const u8, to: []const u8, expected_output: []const u8) void {
- const result = relativeWindows(debug.global_allocator, from, to) catch unreachable;
+ const result = relativeWindows(testing.allocator, from, to) catch unreachable;
testing.expectEqualSlices(u8, expected_output, result);
}
diff --git a/lib/std/heap.zig b/lib/std/heap.zig
index f25d2f0172..e7196f82f2 100644
--- a/lib/std/heap.zig
+++ b/lib/std/heap.zig
@@ -711,6 +711,10 @@ pub const ThreadSafeFixedBufferAllocator = blk: {
fn shrink(allocator: *Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
return old_mem[0..new_size];
}
+
+ pub fn reset(self: *ThreadSafeFixedBufferAllocator) void {
+ self.end_index = 0;
+ }
};
}
};
diff --git a/lib/std/linked_list.zig b/lib/std/linked_list.zig
index 6495653150..a21c9a83eb 100644
--- a/lib/std/linked_list.zig
+++ b/lib/std/linked_list.zig
@@ -143,7 +143,7 @@ pub fn SinglyLinkedList(comptime T: type) type {
}
test "basic SinglyLinkedList test" {
- const allocator = debug.global_allocator;
+ const allocator = testing.allocator;
var list = SinglyLinkedList(u32).init();
var one = try list.createNode(1, allocator);
@@ -404,7 +404,7 @@ pub fn TailQueue(comptime T: type) type {
}
test "basic TailQueue test" {
- const allocator = debug.global_allocator;
+ const allocator = testing.allocator;
var list = TailQueue(u32).init();
var one = try list.createNode(1, allocator);
@@ -456,7 +456,7 @@ test "basic TailQueue test" {
}
test "TailQueue concatenation" {
- const allocator = debug.global_allocator;
+ const allocator = testing.allocator;
var list1 = TailQueue(u32).init();
var list2 = TailQueue(u32).init();
diff --git a/lib/std/os/test.zig b/lib/std/os/test.zig
index f71e80c7d7..0d3be9e7e4 100644
--- a/lib/std/os/test.zig
+++ b/lib/std/os/test.zig
@@ -9,7 +9,7 @@ const elf = std.elf;
const File = std.fs.File;
const Thread = std.Thread;
-const a = std.debug.global_allocator;
+const a = std.testing.allocator;
const builtin = @import("builtin");
const AtomicRmwOp = builtin.AtomicRmwOp;
diff --git a/lib/std/priority_queue.zig b/lib/std/priority_queue.zig
index 6c56f469f9..e726a07a88 100644
--- a/lib/std/priority_queue.zig
+++ b/lib/std/priority_queue.zig
@@ -1,10 +1,10 @@
const std = @import("std.zig");
const Allocator = std.mem.Allocator;
-const debug = std.debug;
-const assert = debug.assert;
-const expect = std.testing.expect;
-const expectEqual = std.testing.expectEqual;
-const expectError = std.testing.expectError;
+const assert = std.debug.assert;
+const testing = std.testing;
+const expect = testing.expect;
+const expectEqual = testing.expectEqual;
+const expectError = testing.expectError;
/// Priority queue for storing generic data. Initialize with `init`.
pub fn PriorityQueue(comptime T: type) type {
@@ -239,7 +239,7 @@ fn greaterThan(a: u32, b: u32) bool {
const PQ = PriorityQueue(u32);
test "std.PriorityQueue: add and remove min heap" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(54);
@@ -257,7 +257,7 @@ test "std.PriorityQueue: add and remove min heap" {
}
test "std.PriorityQueue: add and remove same min heap" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(1);
@@ -275,14 +275,14 @@ test "std.PriorityQueue: add and remove same min heap" {
}
test "std.PriorityQueue: removeOrNull on empty" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
expect(queue.removeOrNull() == null);
}
test "std.PriorityQueue: edge case 3 elements" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(9);
@@ -294,7 +294,7 @@ test "std.PriorityQueue: edge case 3 elements" {
}
test "std.PriorityQueue: peek" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
expect(queue.peek() == null);
@@ -306,7 +306,7 @@ test "std.PriorityQueue: peek" {
}
test "std.PriorityQueue: sift up with odd indices" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
for (items) |e| {
@@ -320,7 +320,7 @@ test "std.PriorityQueue: sift up with odd indices" {
}
test "std.PriorityQueue: addSlice" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
try queue.addSlice(items[0..]);
@@ -333,8 +333,8 @@ test "std.PriorityQueue: addSlice" {
test "std.PriorityQueue: fromOwnedSlice" {
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
- const heap_items = try std.mem.dupe(debug.global_allocator, u32, items[0..]);
- var queue = PQ.fromOwnedSlice(debug.global_allocator, lessThan, heap_items[0..]);
+ const heap_items = try std.mem.dupe(testing.allocator, u32, items[0..]);
+ var queue = PQ.fromOwnedSlice(testing.allocator, lessThan, heap_items[0..]);
defer queue.deinit();
const sorted_items = [_]u32{ 1, 2, 5, 6, 7, 7, 11, 12, 13, 14, 15, 15, 16, 21, 22, 24, 24, 25 };
@@ -344,7 +344,7 @@ test "std.PriorityQueue: fromOwnedSlice" {
}
test "std.PriorityQueue: add and remove max heap" {
- var queue = PQ.init(debug.global_allocator, greaterThan);
+ var queue = PQ.init(testing.allocator, greaterThan);
defer queue.deinit();
try queue.add(54);
@@ -362,7 +362,7 @@ test "std.PriorityQueue: add and remove max heap" {
}
test "std.PriorityQueue: add and remove same max heap" {
- var queue = PQ.init(debug.global_allocator, greaterThan);
+ var queue = PQ.init(testing.allocator, greaterThan);
defer queue.deinit();
try queue.add(1);
@@ -380,8 +380,8 @@ test "std.PriorityQueue: add and remove same max heap" {
}
test "std.PriorityQueue: iterator" {
- var queue = PQ.init(debug.global_allocator, lessThan);
- var map = std.AutoHashMap(u32, void).init(debug.global_allocator);
+ var queue = PQ.init(testing.allocator, lessThan);
+ var map = std.AutoHashMap(u32, void).init(testing.allocator);
defer {
queue.deinit();
map.deinit();
@@ -402,7 +402,7 @@ test "std.PriorityQueue: iterator" {
}
test "std.PriorityQueue: remove at index" {
- var queue = PQ.init(debug.global_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(3);
diff --git a/lib/std/process.zig b/lib/std/process.zig
index 0ce3cabc19..2758e25ac0 100644
--- a/lib/std/process.zig
+++ b/lib/std/process.zig
@@ -114,7 +114,7 @@ pub fn getEnvMap(allocator: *Allocator) !BufMap {
}
test "os.getEnvMap" {
- var env = try getEnvMap(std.debug.global_allocator);
+ var env = try getEnvMap(std.testing.allocator);
defer env.deinit();
}
@@ -165,7 +165,7 @@ pub fn getEnvVarOwned(allocator: *mem.Allocator, key: []const u8) GetEnvVarOwned
}
test "os.getEnvVarOwned" {
- var ga = std.debug.global_allocator;
+ var ga = std.testing.allocator;
testing.expectError(error.EnvironmentVariableNotFound, getEnvVarOwned(ga, "BADENV"));
}
@@ -492,10 +492,10 @@ test "windows arg parsing" {
fn testWindowsCmdLine(input_cmd_line: [*]const u8, expected_args: []const []const u8) void {
var it = ArgIteratorWindows.initWithCmdLine(input_cmd_line);
for (expected_args) |expected_arg| {
- const arg = it.next(std.debug.global_allocator).? catch unreachable;
+ const arg = it.next(std.testing.allocator).? catch unreachable;
testing.expectEqualSlices(u8, expected_arg, arg);
}
- testing.expect(it.next(std.debug.global_allocator) == null);
+ testing.expect(it.next(std.testing.allocator) == null);
}
pub const UserInfo = struct {
diff --git a/lib/std/special/test_runner.zig b/lib/std/special/test_runner.zig
index 0335c8562d..286dfbed1b 100644
--- a/lib/std/special/test_runner.zig
+++ b/lib/std/special/test_runner.zig
@@ -13,6 +13,7 @@ pub fn main() anyerror!void {
};
for (test_fn_list) |test_fn, i| {
+ std.testing.allocator_instance.reset();
var test_node = root_node.start(test_fn.name, null);
test_node.activate();
progress.refresh();
diff --git a/lib/std/testing.zig b/lib/std/testing.zig
index 7a261e0751..02d7cdb393 100644
--- a/lib/std/testing.zig
+++ b/lib/std/testing.zig
@@ -2,6 +2,11 @@ const builtin = @import("builtin");
const TypeId = builtin.TypeId;
const std = @import("std.zig");
+/// This should only be used in temporary test programs.
+pub const allocator = &allocator_instance.allocator;
+pub var allocator_instance = std.heap.ThreadSafeFixedBufferAllocator.init(allocator_mem[0..]);
+var allocator_mem: [100 * 1024]u8 = undefined;
+
/// This function is intended to be used only in tests. It prints diagnostics to stderr
/// and then aborts when actual_error_union is not expected_error.
pub fn expectError(expected_error: anyerror, actual_error_union: var) void {
diff --git a/lib/std/unicode.zig b/lib/std/unicode.zig
index 3d16748447..638b5f5e2f 100644
--- a/lib/std/unicode.zig
+++ b/lib/std/unicode.zig
@@ -501,14 +501,14 @@ test "utf16leToUtf8" {
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 'A');
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 'a');
- const utf8 = try utf16leToUtf8Alloc(std.debug.global_allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
testing.expect(mem.eql(u8, utf8, "Aa"));
}
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0x80);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xffff);
- const utf8 = try utf16leToUtf8Alloc(std.debug.global_allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
testing.expect(mem.eql(u8, utf8, "\xc2\x80" ++ "\xef\xbf\xbf"));
}
@@ -516,7 +516,7 @@ test "utf16leToUtf8" {
// the values just outside the surrogate half range
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xd7ff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xe000);
- const utf8 = try utf16leToUtf8Alloc(std.debug.global_allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
testing.expect(mem.eql(u8, utf8, "\xed\x9f\xbf" ++ "\xee\x80\x80"));
}
@@ -524,7 +524,7 @@ test "utf16leToUtf8" {
// smallest surrogate pair
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xd800);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdc00);
- const utf8 = try utf16leToUtf8Alloc(std.debug.global_allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
testing.expect(mem.eql(u8, utf8, "\xf0\x90\x80\x80"));
}
@@ -532,14 +532,14 @@ test "utf16leToUtf8" {
// largest surrogate pair
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xdbff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdfff);
- const utf8 = try utf16leToUtf8Alloc(std.debug.global_allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
testing.expect(mem.eql(u8, utf8, "\xf4\x8f\xbf\xbf"));
}
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xdbff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdc00);
- const utf8 = try utf16leToUtf8Alloc(std.debug.global_allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
testing.expect(mem.eql(u8, utf8, "\xf4\x8f\xb0\x80"));
}
}
diff --git a/lib/std/zig/ast.zig b/lib/std/zig/ast.zig
index dc9a6bc7ca..e339343823 100644
--- a/lib/std/zig/ast.zig
+++ b/lib/std/zig/ast.zig
@@ -2287,7 +2287,7 @@ pub const Node = struct {
test "iterate" {
var root = Node.Root{
.base = Node{ .id = Node.Id.Root },
- .decls = Node.Root.DeclList.init(std.debug.global_allocator),
+ .decls = Node.Root.DeclList.init(std.testing.allocator),
.eof_token = 0,
};
var base = &root.base;
diff --git a/test/compare_output.zig b/test/compare_output.zig
index db20709afa..7a41d46f54 100644
--- a/test/compare_output.zig
+++ b/test/compare_output.zig
@@ -445,7 +445,7 @@ pub fn addCases(cases: *tests.CompareOutputContext) void {
\\const std = @import("std");
\\const io = std.io;
\\const os = std.os;
- \\const allocator = std.debug.global_allocator;
+ \\const allocator = std.testing.allocator;
\\
\\pub fn main() !void {
\\ var args_it = std.process.args();
@@ -486,7 +486,7 @@ pub fn addCases(cases: *tests.CompareOutputContext) void {
\\const std = @import("std");
\\const io = std.io;
\\const os = std.os;
- \\const allocator = std.debug.global_allocator;
+ \\const allocator = std.testing.allocator;
\\
\\pub fn main() !void {
\\ var args_it = std.process.args();
diff --git a/test/compile_errors.zig b/test/compile_errors.zig
index 89e187aed2..341062e161 100644
--- a/test/compile_errors.zig
+++ b/test/compile_errors.zig
@@ -5765,7 +5765,7 @@ pub fn addCases(cases: *tests.CompileErrorContext) void {
\\
\\export fn entry() void {
\\ const a = MdNode.Header {
- \\ .text = MdText.init(&std.debug.global_allocator),
+ \\ .text = MdText.init(&std.testing.allocator),
\\ .weight = HeaderWeight.H1,
\\ };
\\}
diff --git a/test/stage1/behavior/const_slice_child.zig b/test/stage1/behavior/const_slice_child.zig
index 6720667187..cf650ae88e 100644
--- a/test/stage1/behavior/const_slice_child.zig
+++ b/test/stage1/behavior/const_slice_child.zig
@@ -22,7 +22,7 @@ fn foo(args: [][]const u8) void {
}
fn bar(argc: usize) void {
- const args = debug.global_allocator.alloc([]const u8, argc) catch unreachable;
+ const args = testing.allocator.alloc([]const u8, argc) catch unreachable;
for (args) |_, i| {
const ptr = argv[i];
args[i] = ptr[0..strlen(ptr)];
diff --git a/test/standalone/brace_expansion/main.zig b/test/standalone/brace_expansion/main.zig
index cbb328541e..c6bfd02e79 100644
--- a/test/standalone/brace_expansion/main.zig
+++ b/test/standalone/brace_expansion/main.zig
@@ -201,7 +201,7 @@ pub fn main() !void {
}
test "invalid inputs" {
- global_allocator = std.debug.global_allocator;
+ global_allocator = std.testing.allocator;
expectError("}ABC", error.InvalidInput);
expectError("{ABC", error.InvalidInput);
@@ -222,7 +222,7 @@ fn expectError(test_input: []const u8, expected_err: anyerror) void {
}
test "valid inputs" {
- global_allocator = std.debug.global_allocator;
+ global_allocator = std.testing.allocator;
expectExpansion("{x,y,z}", "x y z");
expectExpansion("{A,B}{x,y}", "Ax Ay Bx By");
diff --git a/test/standalone/cat/main.zig b/test/standalone/cat/main.zig
index f0cd9728ab..34439f9c24 100644
--- a/test/standalone/cat/main.zig
+++ b/test/standalone/cat/main.zig
@@ -4,7 +4,7 @@ const process = std.process;
const fs = std.fs;
const mem = std.mem;
const warn = std.debug.warn;
-const allocator = std.debug.global_allocator;
+const allocator = std.testing.allocator;
pub fn main() !void {
var args_it = process.args();
diff --git a/test/standalone/empty_env/main.zig b/test/standalone/empty_env/main.zig
index d6d5ecb5af..f4ebf56136 100644
--- a/test/standalone/empty_env/main.zig
+++ b/test/standalone/empty_env/main.zig
@@ -1,6 +1,6 @@
const std = @import("std");
pub fn main() void {
- const env_map = std.process.getEnvMap(std.debug.global_allocator) catch @panic("unable to get env map");
+ const env_map = std.process.getEnvMap(std.testing.allocator) catch @panic("unable to get env map");
std.testing.expect(env_map.count() == 0);
}
diff --git a/test/standalone/load_dynamic_library/main.zig b/test/standalone/load_dynamic_library/main.zig
index 197e3ca47c..70fc4986ac 100644
--- a/test/standalone/load_dynamic_library/main.zig
+++ b/test/standalone/load_dynamic_library/main.zig
@@ -1,8 +1,8 @@
const std = @import("std");
pub fn main() !void {
- const args = try std.process.argsAlloc(std.debug.global_allocator);
- defer std.process.argsFree(std.debug.global_allocator, args);
+ const args = try std.process.argsAlloc(std.testing.allocator);
+ defer std.process.argsFree(std.testing.allocator, args);
const dynlib_name = args[1];
From aa9caf5064b3fc01579b7bd8c765ff33323a073a Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 13:18:04 -0600
Subject: [PATCH 02/16] Create leak_count_allocator
---
doc/docgen.zig | 2 +-
lib/std/array_list.zig | 15 +-
lib/std/buffer.zig | 11 +-
lib/std/debug.zig | 2 +-
lib/std/fifo.zig | 4 +-
lib/std/fmt.zig | 15 +-
lib/std/fs/path.zig | 162 +++++++++---------
lib/std/linked_list.zig | 6 +-
lib/std/os/test.zig | 2 +-
lib/std/priority_queue.zig | 28 +--
lib/std/process.zig | 9 +-
lib/std/special/test_runner.zig | 3 +
lib/std/testing.zig | 42 +++++
lib/std/unicode.zig | 18 +-
lib/std/zig/ast.zig | 2 +-
test/compare_output.zig | 4 +-
test/compile_errors.zig | 2 +-
test/stage1/behavior/const_slice_child.zig | 2 +-
test/standalone/brace_expansion/main.zig | 4 +-
test/standalone/cat/main.zig | 2 +-
test/standalone/empty_env/main.zig | 2 +-
test/standalone/load_dynamic_library/main.zig | 4 +-
22 files changed, 206 insertions(+), 135 deletions(-)
diff --git a/doc/docgen.zig b/doc/docgen.zig
index 678eb26043..128314696e 100644
--- a/doc/docgen.zig
+++ b/doc/docgen.zig
@@ -672,7 +672,7 @@ const TermState = enum {
test "term color" {
const input_bytes = "A\x1b[32;1mgreen\x1b[0mB";
- const result = try termColor(std.testing.allocator, input_bytes);
+ const result = try termColor(std.testing.leak_count_allocator, input_bytes);
testing.expectEqualSlices(u8, "AgreenB", result);
}
diff --git a/lib/std/array_list.zig b/lib/std/array_list.zig
index be99a54e2b..57c0d706a2 100644
--- a/lib/std/array_list.zig
+++ b/lib/std/array_list.zig
@@ -320,7 +320,7 @@ test "std.ArrayList.basic" {
}
test "std.ArrayList.orderedRemove" {
- var list = ArrayList(i32).init(testing.allocator);
+ var list = ArrayList(i32).init(testing.leak_count_allocator);
defer list.deinit();
try list.append(1);
@@ -347,7 +347,7 @@ test "std.ArrayList.orderedRemove" {
}
test "std.ArrayList.swapRemove" {
- var list = ArrayList(i32).init(testing.allocator);
+ var list = ArrayList(i32).init(testing.leak_count_allocator);
defer list.deinit();
try list.append(1);
@@ -374,7 +374,7 @@ test "std.ArrayList.swapRemove" {
}
test "std.ArrayList.swapRemoveOrError" {
- var list = ArrayList(i32).init(testing.allocator);
+ var list = ArrayList(i32).init(testing.leak_count_allocator);
defer list.deinit();
// Test just after initialization
@@ -402,7 +402,7 @@ test "std.ArrayList.swapRemoveOrError" {
}
test "std.ArrayList.insert" {
- var list = ArrayList(i32).init(testing.allocator);
+ var list = ArrayList(i32).init(testing.leak_count_allocator);
defer list.deinit();
try list.append(1);
@@ -416,7 +416,7 @@ test "std.ArrayList.insert" {
}
test "std.ArrayList.insertSlice" {
- var list = ArrayList(i32).init(testing.allocator);
+ var list = ArrayList(i32).init(testing.leak_count_allocator);
defer list.deinit();
try list.append(1);
@@ -443,7 +443,8 @@ const Item = struct {
};
test "std.ArrayList: ArrayList(T) of struct T" {
- var root = Item{ .integer = 1, .sub_items = ArrayList(Item).init(testing.allocator) };
- try root.sub_items.append(Item{ .integer = 42, .sub_items = ArrayList(Item).init(testing.allocator) });
+ var root = Item{ .integer = 1, .sub_items = ArrayList(Item).init(testing.leak_count_allocator) };
+ defer root.sub_items.deinit();
+ try root.sub_items.append(Item{ .integer = 42, .sub_items = ArrayList(Item).init(testing.leak_count_allocator) });
testing.expect(root.sub_items.items[0].integer == 42);
}
diff --git a/lib/std/buffer.zig b/lib/std/buffer.zig
index ada3590616..cc1166e0e7 100644
--- a/lib/std/buffer.zig
+++ b/lib/std/buffer.zig
@@ -150,7 +150,9 @@ pub const Buffer = struct {
};
test "simple Buffer" {
- var buf = try Buffer.init(testing.allocator, "");
+ var buf = try Buffer.init(testing.leak_count_allocator, "");
+ defer buf.deinit();
+
testing.expect(buf.len() == 0);
try buf.append("hello");
try buf.append(" ");
@@ -159,6 +161,7 @@ test "simple Buffer" {
testing.expect(mem.eql(u8, mem.toSliceConst(u8, buf.toSliceConst().ptr), buf.toSliceConst()));
var buf2 = try Buffer.initFromBuffer(buf);
+ defer buf2.deinit();
testing.expect(buf.eql(buf2.toSliceConst()));
testing.expect(buf.startsWith("hell"));
@@ -169,14 +172,16 @@ test "simple Buffer" {
}
test "Buffer.initSize" {
- var buf = try Buffer.initSize(testing.allocator, 3);
+ var buf = try Buffer.initSize(testing.leak_count_allocator, 3);
+ defer buf.deinit();
testing.expect(buf.len() == 3);
try buf.append("hello");
testing.expect(mem.eql(u8, buf.toSliceConst()[3..], "hello"));
}
test "Buffer.initCapacity" {
- var buf = try Buffer.initCapacity(testing.allocator, 10);
+ var buf = try Buffer.initCapacity(testing.leak_count_allocator, 10);
+ defer buf.deinit();
testing.expect(buf.len() == 0);
testing.expect(buf.capacity() >= 10);
const old_cap = buf.capacity();
diff --git a/lib/std/debug.zig b/lib/std/debug.zig
index f2dcdd7e77..7e1a99a395 100644
--- a/lib/std/debug.zig
+++ b/lib/std/debug.zig
@@ -2193,7 +2193,7 @@ fn readInitialLength(comptime E: type, in_stream: *io.InStream(E), is_64: *bool)
}
pub const global_allocator = blk: {
- @compileError("Please switch to std.testing.allocator.");
+ @compileError("Please switch to std.testing.leak_count_allocator.");
};
var global_fixed_allocator = std.heap.ThreadSafeFixedBufferAllocator.init(global_allocator_mem[0..]);
var global_allocator_mem: [100 * 1024]u8 = undefined;
diff --git a/lib/std/fifo.zig b/lib/std/fifo.zig
index ac8589f2ce..dd429289d1 100644
--- a/lib/std/fifo.zig
+++ b/lib/std/fifo.zig
@@ -347,7 +347,7 @@ pub fn LinearFifo(
}
test "LinearFifo(u8, .Dynamic)" {
- var fifo = LinearFifo(u8, .Dynamic).init(testing.allocator);
+ var fifo = LinearFifo(u8, .Dynamic).init(testing.leak_count_allocator);
defer fifo.deinit();
try fifo.write("HELLO");
@@ -422,7 +422,7 @@ test "LinearFifo" {
var fifo = switch (bt) {
.Static => FifoType.init(),
.Slice => FifoType.init(buf[0..]),
- .Dynamic => FifoType.init(testing.allocator),
+ .Dynamic => FifoType.init(testing.leak_count_allocator),
};
defer fifo.deinit();
diff --git a/lib/std/fmt.zig b/lib/std/fmt.zig
index 9757658447..529131065c 100644
--- a/lib/std/fmt.zig
+++ b/lib/std/fmt.zig
@@ -1598,7 +1598,8 @@ test "hexToBytes" {
test "formatIntValue with comptime_int" {
const value: comptime_int = 123456789123456789;
- var buf = try std.Buffer.init(std.testing.allocator, "");
+ var buf = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ defer buf.deinit();
try formatIntValue(value, "", FormatOptions{}, &buf, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append);
std.testing.expect(mem.eql(u8, buf.toSlice(), "123456789123456789"));
}
@@ -1652,19 +1653,23 @@ test "formatType max_depth" {
inst.a = &inst;
inst.tu.ptr = &inst.tu;
- var buf0 = try std.Buffer.init(std.testing.allocator, "");
+ var buf0 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ defer buf0.deinit();
try formatType(inst, "", FormatOptions{}, &buf0, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 0);
std.testing.expect(mem.eql(u8, buf0.toSlice(), "S{ ... }"));
- var buf1 = try std.Buffer.init(std.testing.allocator, "");
+ var buf1 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ defer buf1.deinit();
try formatType(inst, "", FormatOptions{}, &buf1, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 1);
std.testing.expect(mem.eql(u8, buf1.toSlice(), "S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }"));
- var buf2 = try std.Buffer.init(std.testing.allocator, "");
+ var buf2 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ defer buf2.deinit();
try formatType(inst, "", FormatOptions{}, &buf2, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 2);
std.testing.expect(mem.eql(u8, buf2.toSlice(), "S{ .a = S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ ... } }, .e = E.Two, .vec = (10.200,2.220) }"));
- var buf3 = try std.Buffer.init(std.testing.allocator, "");
+ var buf3 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ defer buf3.deinit();
try formatType(inst, "", FormatOptions{}, &buf3, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 3);
std.testing.expect(mem.eql(u8, buf3.toSlice(), "S{ .a = S{ .a = S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ ... } }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ .ptr = TU{ ... } } }, .e = E.Two, .vec = (10.200,2.220) }"));
}
diff --git a/lib/std/fs/path.zig b/lib/std/fs/path.zig
index 6c29a3950c..424e9f5759 100644
--- a/lib/std/fs/path.zig
+++ b/lib/std/fs/path.zig
@@ -665,15 +665,16 @@ pub fn resolvePosix(allocator: *Allocator, paths: []const []const u8) ![]u8 {
}
test "resolve" {
- const cwd = try process.getCwdAlloc(testing.allocator);
+ const cwd = try process.getCwdAlloc(testing.leak_count_allocator);
+ defer testing.leak_count_allocator.free(cwd);
if (builtin.os == .windows) {
if (windowsParsePath(cwd).kind == WindowsPath.Kind.Drive) {
cwd[0] = asciiUpper(cwd[0]);
}
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{"."}), cwd));
+ try testResolveWindows(&[_][]const u8{"."}, cwd);
} else {
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "a/b/c/", "../../.." }), cwd));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{"."}), cwd));
+ try testResolvePosix(&[_][]const u8{ "a/b/c/", "../../.." }, cwd);
+ try testResolvePosix(&[_][]const u8{"."}, cwd);
}
}
@@ -683,66 +684,71 @@ test "resolveWindows" {
return error.SkipZigTest;
}
if (builtin.os == .windows) {
- const cwd = try process.getCwdAlloc(testing.allocator);
+ const cwd = try process.getCwdAlloc(testing.leak_count_allocator);
+ defer testing.leak_count_allocator.free(cwd);
const parsed_cwd = windowsParsePath(cwd);
{
- const result = testResolveWindows(&[_][]const u8{ "/usr/local", "lib\\zig\\std\\array_list.zig" });
- const expected = try join(testing.allocator, &[_][]const u8{
+ const expected = try join(testing.leak_count_allocator, &[_][]const u8{
parsed_cwd.disk_designator,
"usr\\local\\lib\\zig\\std\\array_list.zig",
});
+ defer testing.leak_count_allocator.free(expected);
if (parsed_cwd.kind == WindowsPath.Kind.Drive) {
expected[0] = asciiUpper(parsed_cwd.disk_designator[0]);
}
- testing.expect(mem.eql(u8, result, expected));
+ try testResolveWindows(&[_][]const u8{ "/usr/local", "lib\\zig\\std\\array_list.zig" }, expected);
}
{
- const result = testResolveWindows(&[_][]const u8{ "usr/local", "lib\\zig" });
- const expected = try join(testing.allocator, &[_][]const u8{
+ const expected = try join(testing.leak_count_allocator, &[_][]const u8{
cwd,
"usr\\local\\lib\\zig",
});
+ defer testing.leak_count_allocator.free(expected);
if (parsed_cwd.kind == WindowsPath.Kind.Drive) {
expected[0] = asciiUpper(parsed_cwd.disk_designator[0]);
}
- testing.expect(mem.eql(u8, result, expected));
+ try testResolveWindows(&[_][]const u8{ "usr/local", "lib\\zig" }, expected);
}
}
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:\\a\\b\\c", "/hi", "ok" }), "C:\\hi\\ok"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/blah\\blah", "d:/games", "c:../a" }), "C:\\blah\\a"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/blah\\blah", "d:/games", "C:../a" }), "C:\\blah\\a"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/ignore", "d:\\a/b\\c/d", "\\e.exe" }), "D:\\e.exe"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/ignore", "c:/some/file" }), "C:\\some\\file"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "d:/ignore", "d:some/dir//" }), "D:\\ignore\\some\\dir"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "//server/share", "..", "relative\\" }), "\\\\server\\share\\relative"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/", "//" }), "C:\\"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/", "//dir" }), "C:\\dir"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/", "//server/share" }), "\\\\server\\share\\"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/", "//server//share" }), "\\\\server\\share\\"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "c:/", "///some//dir" }), "C:\\some\\dir"));
- testing.expect(mem.eql(u8, testResolveWindows(&[_][]const u8{ "C:\\foo\\tmp.3\\", "..\\tmp.3\\cycles\\root.js" }), "C:\\foo\\tmp.3\\cycles\\root.js"));
+ try testResolveWindows(&[_][]const u8{ "c:\\a\\b\\c", "/hi", "ok" }, "C:\\hi\\ok");
+ try testResolveWindows(&[_][]const u8{ "c:/blah\\blah", "d:/games", "c:../a" }, "C:\\blah\\a");
+ try testResolveWindows(&[_][]const u8{ "c:/blah\\blah", "d:/games", "C:../a" }, "C:\\blah\\a");
+ try testResolveWindows(&[_][]const u8{ "c:/ignore", "d:\\a/b\\c/d", "\\e.exe" }, "D:\\e.exe");
+ try testResolveWindows(&[_][]const u8{ "c:/ignore", "c:/some/file" }, "C:\\some\\file");
+ try testResolveWindows(&[_][]const u8{ "d:/ignore", "d:some/dir//" }, "D:\\ignore\\some\\dir");
+ try testResolveWindows(&[_][]const u8{ "//server/share", "..", "relative\\" }, "\\\\server\\share\\relative");
+ try testResolveWindows(&[_][]const u8{ "c:/", "//" }, "C:\\");
+ try testResolveWindows(&[_][]const u8{ "c:/", "//dir" }, "C:\\dir");
+ try testResolveWindows(&[_][]const u8{ "c:/", "//server/share" }, "\\\\server\\share\\");
+ try testResolveWindows(&[_][]const u8{ "c:/", "//server//share" }, "\\\\server\\share\\");
+ try testResolveWindows(&[_][]const u8{ "c:/", "///some//dir" }, "C:\\some\\dir");
+ try testResolveWindows(&[_][]const u8{ "C:\\foo\\tmp.3\\", "..\\tmp.3\\cycles\\root.js" }, "C:\\foo\\tmp.3\\cycles\\root.js");
}
test "resolvePosix" {
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/a/b", "c" }), "/a/b/c"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/a/b", "c", "//d", "e///" }), "/d/e"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/a/b/c", "..", "../" }), "/a"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/", "..", ".." }), "/"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{"/a/b/c/"}), "/a/b/c"));
+ try testResolvePosix(&[_][]const u8{ "/a/b", "c" }, "/a/b/c");
+ try testResolvePosix(&[_][]const u8{ "/a/b", "c", "//d", "e///" }, "/d/e");
+ try testResolvePosix(&[_][]const u8{ "/a/b/c", "..", "../" }, "/a");
+ try testResolvePosix(&[_][]const u8{ "/", "..", ".." }, "/");
+ try testResolvePosix(&[_][]const u8{"/a/b/c/"}, "/a/b/c");
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/var/lib", "../", "file/" }), "/var/file"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/var/lib", "/../", "file/" }), "/file"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/some/dir", ".", "/absolute/" }), "/absolute"));
- testing.expect(mem.eql(u8, testResolvePosix(&[_][]const u8{ "/foo/tmp.3/", "../tmp.3/cycles/root.js" }), "/foo/tmp.3/cycles/root.js"));
+ try testResolvePosix(&[_][]const u8{ "/var/lib", "../", "file/" }, "/var/file");
+ try testResolvePosix(&[_][]const u8{ "/var/lib", "/../", "file/" }, "/file");
+ try testResolvePosix(&[_][]const u8{ "/some/dir", ".", "/absolute/" }, "/absolute");
+ try testResolvePosix(&[_][]const u8{ "/foo/tmp.3/", "../tmp.3/cycles/root.js" }, "/foo/tmp.3/cycles/root.js");
}
-fn testResolveWindows(paths: []const []const u8) []u8 {
- return resolveWindows(testing.allocator, paths) catch unreachable;
+fn testResolveWindows(paths: []const []const u8, expected: []const u8) !void {
+ const actual = try resolveWindows(testing.leak_count_allocator, paths);
+ defer testing.leak_count_allocator.free(actual);
+ return testing.expect(mem.eql(u8, actual, expected));
}
-fn testResolvePosix(paths: []const []const u8) []u8 {
- return resolvePosix(testing.allocator, paths) catch unreachable;
+fn testResolvePosix(paths: []const []const u8, expected: []const u8) !void {
+ const actual = try resolvePosix(testing.leak_count_allocator, paths);
+ defer testing.leak_count_allocator.free(actual);
+ return testing.expect(mem.eql(u8, actual, expected));
}
/// If the path is a file in the current directory (no directory component)
@@ -1126,51 +1132,53 @@ test "relative" {
// TODO https://github.com/ziglang/zig/issues/3288
return error.SkipZigTest;
}
- testRelativeWindows("c:/blah\\blah", "d:/games", "D:\\games");
- testRelativeWindows("c:/aaaa/bbbb", "c:/aaaa", "..");
- testRelativeWindows("c:/aaaa/bbbb", "c:/cccc", "..\\..\\cccc");
- testRelativeWindows("c:/aaaa/bbbb", "c:/aaaa/bbbb", "");
- testRelativeWindows("c:/aaaa/bbbb", "c:/aaaa/cccc", "..\\cccc");
- testRelativeWindows("c:/aaaa/", "c:/aaaa/cccc", "cccc");
- testRelativeWindows("c:/", "c:\\aaaa\\bbbb", "aaaa\\bbbb");
- testRelativeWindows("c:/aaaa/bbbb", "d:\\", "D:\\");
- testRelativeWindows("c:/AaAa/bbbb", "c:/aaaa/bbbb", "");
- testRelativeWindows("c:/aaaaa/", "c:/aaaa/cccc", "..\\aaaa\\cccc");
- testRelativeWindows("C:\\foo\\bar\\baz\\quux", "C:\\", "..\\..\\..\\..");
- testRelativeWindows("C:\\foo\\test", "C:\\foo\\test\\bar\\package.json", "bar\\package.json");
- testRelativeWindows("C:\\foo\\bar\\baz-quux", "C:\\foo\\bar\\baz", "..\\baz");
- testRelativeWindows("C:\\foo\\bar\\baz", "C:\\foo\\bar\\baz-quux", "..\\baz-quux");
- testRelativeWindows("\\\\foo\\bar", "\\\\foo\\bar\\baz", "baz");
- testRelativeWindows("\\\\foo\\bar\\baz", "\\\\foo\\bar", "..");
- testRelativeWindows("\\\\foo\\bar\\baz-quux", "\\\\foo\\bar\\baz", "..\\baz");
- testRelativeWindows("\\\\foo\\bar\\baz", "\\\\foo\\bar\\baz-quux", "..\\baz-quux");
- testRelativeWindows("C:\\baz-quux", "C:\\baz", "..\\baz");
- testRelativeWindows("C:\\baz", "C:\\baz-quux", "..\\baz-quux");
- testRelativeWindows("\\\\foo\\baz-quux", "\\\\foo\\baz", "..\\baz");
- testRelativeWindows("\\\\foo\\baz", "\\\\foo\\baz-quux", "..\\baz-quux");
- testRelativeWindows("C:\\baz", "\\\\foo\\bar\\baz", "\\\\foo\\bar\\baz");
- testRelativeWindows("\\\\foo\\bar\\baz", "C:\\baz", "C:\\baz");
+ try testRelativeWindows("c:/blah\\blah", "d:/games", "D:\\games");
+ try testRelativeWindows("c:/aaaa/bbbb", "c:/aaaa", "..");
+ try testRelativeWindows("c:/aaaa/bbbb", "c:/cccc", "..\\..\\cccc");
+ try testRelativeWindows("c:/aaaa/bbbb", "c:/aaaa/bbbb", "");
+ try testRelativeWindows("c:/aaaa/bbbb", "c:/aaaa/cccc", "..\\cccc");
+ try testRelativeWindows("c:/aaaa/", "c:/aaaa/cccc", "cccc");
+ try testRelativeWindows("c:/", "c:\\aaaa\\bbbb", "aaaa\\bbbb");
+ try testRelativeWindows("c:/aaaa/bbbb", "d:\\", "D:\\");
+ try testRelativeWindows("c:/AaAa/bbbb", "c:/aaaa/bbbb", "");
+ try testRelativeWindows("c:/aaaaa/", "c:/aaaa/cccc", "..\\aaaa\\cccc");
+ try testRelativeWindows("C:\\foo\\bar\\baz\\quux", "C:\\", "..\\..\\..\\..");
+ try testRelativeWindows("C:\\foo\\test", "C:\\foo\\test\\bar\\package.json", "bar\\package.json");
+ try testRelativeWindows("C:\\foo\\bar\\baz-quux", "C:\\foo\\bar\\baz", "..\\baz");
+ try testRelativeWindows("C:\\foo\\bar\\baz", "C:\\foo\\bar\\baz-quux", "..\\baz-quux");
+ try testRelativeWindows("\\\\foo\\bar", "\\\\foo\\bar\\baz", "baz");
+ try testRelativeWindows("\\\\foo\\bar\\baz", "\\\\foo\\bar", "..");
+ try testRelativeWindows("\\\\foo\\bar\\baz-quux", "\\\\foo\\bar\\baz", "..\\baz");
+ try testRelativeWindows("\\\\foo\\bar\\baz", "\\\\foo\\bar\\baz-quux", "..\\baz-quux");
+ try testRelativeWindows("C:\\baz-quux", "C:\\baz", "..\\baz");
+ try testRelativeWindows("C:\\baz", "C:\\baz-quux", "..\\baz-quux");
+ try testRelativeWindows("\\\\foo\\baz-quux", "\\\\foo\\baz", "..\\baz");
+ try testRelativeWindows("\\\\foo\\baz", "\\\\foo\\baz-quux", "..\\baz-quux");
+ try testRelativeWindows("C:\\baz", "\\\\foo\\bar\\baz", "\\\\foo\\bar\\baz");
+ try testRelativeWindows("\\\\foo\\bar\\baz", "C:\\baz", "C:\\baz");
- testRelativePosix("/var/lib", "/var", "..");
- testRelativePosix("/var/lib", "/bin", "../../bin");
- testRelativePosix("/var/lib", "/var/lib", "");
- testRelativePosix("/var/lib", "/var/apache", "../apache");
- testRelativePosix("/var/", "/var/lib", "lib");
- testRelativePosix("/", "/var/lib", "var/lib");
- testRelativePosix("/foo/test", "/foo/test/bar/package.json", "bar/package.json");
- testRelativePosix("/Users/a/web/b/test/mails", "/Users/a/web/b", "../..");
- testRelativePosix("/foo/bar/baz-quux", "/foo/bar/baz", "../baz");
- testRelativePosix("/foo/bar/baz", "/foo/bar/baz-quux", "../baz-quux");
- testRelativePosix("/baz-quux", "/baz", "../baz");
- testRelativePosix("/baz", "/baz-quux", "../baz-quux");
+ try testRelativePosix("/var/lib", "/var", "..");
+ try testRelativePosix("/var/lib", "/bin", "../../bin");
+ try testRelativePosix("/var/lib", "/var/lib", "");
+ try testRelativePosix("/var/lib", "/var/apache", "../apache");
+ try testRelativePosix("/var/", "/var/lib", "lib");
+ try testRelativePosix("/", "/var/lib", "var/lib");
+ try testRelativePosix("/foo/test", "/foo/test/bar/package.json", "bar/package.json");
+ try testRelativePosix("/Users/a/web/b/test/mails", "/Users/a/web/b", "../..");
+ try testRelativePosix("/foo/bar/baz-quux", "/foo/bar/baz", "../baz");
+ try testRelativePosix("/foo/bar/baz", "/foo/bar/baz-quux", "../baz-quux");
+ try testRelativePosix("/baz-quux", "/baz", "../baz");
+ try testRelativePosix("/baz", "/baz-quux", "../baz-quux");
}
-fn testRelativePosix(from: []const u8, to: []const u8, expected_output: []const u8) void {
- const result = relativePosix(testing.allocator, from, to) catch unreachable;
+fn testRelativePosix(from: []const u8, to: []const u8, expected_output: []const u8) !void {
+ const result = try relativePosix(testing.leak_count_allocator, from, to);
+ defer testing.leak_count_allocator.free(result);
testing.expectEqualSlices(u8, expected_output, result);
}
-fn testRelativeWindows(from: []const u8, to: []const u8, expected_output: []const u8) void {
- const result = relativeWindows(testing.allocator, from, to) catch unreachable;
+fn testRelativeWindows(from: []const u8, to: []const u8, expected_output: []const u8) !void {
+ const result = try relativeWindows(testing.leak_count_allocator, from, to);
+ defer testing.leak_count_allocator.free(result);
testing.expectEqualSlices(u8, expected_output, result);
}
diff --git a/lib/std/linked_list.zig b/lib/std/linked_list.zig
index a21c9a83eb..d1652a8047 100644
--- a/lib/std/linked_list.zig
+++ b/lib/std/linked_list.zig
@@ -143,7 +143,7 @@ pub fn SinglyLinkedList(comptime T: type) type {
}
test "basic SinglyLinkedList test" {
- const allocator = testing.allocator;
+ const allocator = testing.leak_count_allocator;
var list = SinglyLinkedList(u32).init();
var one = try list.createNode(1, allocator);
@@ -404,7 +404,7 @@ pub fn TailQueue(comptime T: type) type {
}
test "basic TailQueue test" {
- const allocator = testing.allocator;
+ const allocator = testing.leak_count_allocator;
var list = TailQueue(u32).init();
var one = try list.createNode(1, allocator);
@@ -456,7 +456,7 @@ test "basic TailQueue test" {
}
test "TailQueue concatenation" {
- const allocator = testing.allocator;
+ const allocator = testing.leak_count_allocator;
var list1 = TailQueue(u32).init();
var list2 = TailQueue(u32).init();
diff --git a/lib/std/os/test.zig b/lib/std/os/test.zig
index 0d3be9e7e4..2e60fe500e 100644
--- a/lib/std/os/test.zig
+++ b/lib/std/os/test.zig
@@ -9,7 +9,7 @@ const elf = std.elf;
const File = std.fs.File;
const Thread = std.Thread;
-const a = std.testing.allocator;
+const a = std.testing.leak_count_allocator;
const builtin = @import("builtin");
const AtomicRmwOp = builtin.AtomicRmwOp;
diff --git a/lib/std/priority_queue.zig b/lib/std/priority_queue.zig
index e726a07a88..9c07d20798 100644
--- a/lib/std/priority_queue.zig
+++ b/lib/std/priority_queue.zig
@@ -239,7 +239,7 @@ fn greaterThan(a: u32, b: u32) bool {
const PQ = PriorityQueue(u32);
test "std.PriorityQueue: add and remove min heap" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
try queue.add(54);
@@ -257,7 +257,7 @@ test "std.PriorityQueue: add and remove min heap" {
}
test "std.PriorityQueue: add and remove same min heap" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
try queue.add(1);
@@ -275,14 +275,14 @@ test "std.PriorityQueue: add and remove same min heap" {
}
test "std.PriorityQueue: removeOrNull on empty" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
expect(queue.removeOrNull() == null);
}
test "std.PriorityQueue: edge case 3 elements" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
try queue.add(9);
@@ -294,7 +294,7 @@ test "std.PriorityQueue: edge case 3 elements" {
}
test "std.PriorityQueue: peek" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
expect(queue.peek() == null);
@@ -306,7 +306,7 @@ test "std.PriorityQueue: peek" {
}
test "std.PriorityQueue: sift up with odd indices" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
for (items) |e| {
@@ -320,7 +320,7 @@ test "std.PriorityQueue: sift up with odd indices" {
}
test "std.PriorityQueue: addSlice" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
try queue.addSlice(items[0..]);
@@ -333,8 +333,8 @@ test "std.PriorityQueue: addSlice" {
test "std.PriorityQueue: fromOwnedSlice" {
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
- const heap_items = try std.mem.dupe(testing.allocator, u32, items[0..]);
- var queue = PQ.fromOwnedSlice(testing.allocator, lessThan, heap_items[0..]);
+ const heap_items = try std.mem.dupe(testing.leak_count_allocator, u32, items[0..]);
+ var queue = PQ.fromOwnedSlice(testing.leak_count_allocator, lessThan, heap_items[0..]);
defer queue.deinit();
const sorted_items = [_]u32{ 1, 2, 5, 6, 7, 7, 11, 12, 13, 14, 15, 15, 16, 21, 22, 24, 24, 25 };
@@ -344,7 +344,7 @@ test "std.PriorityQueue: fromOwnedSlice" {
}
test "std.PriorityQueue: add and remove max heap" {
- var queue = PQ.init(testing.allocator, greaterThan);
+ var queue = PQ.init(testing.leak_count_allocator, greaterThan);
defer queue.deinit();
try queue.add(54);
@@ -362,7 +362,7 @@ test "std.PriorityQueue: add and remove max heap" {
}
test "std.PriorityQueue: add and remove same max heap" {
- var queue = PQ.init(testing.allocator, greaterThan);
+ var queue = PQ.init(testing.leak_count_allocator, greaterThan);
defer queue.deinit();
try queue.add(1);
@@ -380,8 +380,8 @@ test "std.PriorityQueue: add and remove same max heap" {
}
test "std.PriorityQueue: iterator" {
- var queue = PQ.init(testing.allocator, lessThan);
- var map = std.AutoHashMap(u32, void).init(testing.allocator);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var map = std.AutoHashMap(u32, void).init(testing.leak_count_allocator);
defer {
queue.deinit();
map.deinit();
@@ -402,7 +402,7 @@ test "std.PriorityQueue: iterator" {
}
test "std.PriorityQueue: remove at index" {
- var queue = PQ.init(testing.allocator, lessThan);
+ var queue = PQ.init(testing.leak_count_allocator, lessThan);
defer queue.deinit();
try queue.add(3);
diff --git a/lib/std/process.zig b/lib/std/process.zig
index 2758e25ac0..19f0eac235 100644
--- a/lib/std/process.zig
+++ b/lib/std/process.zig
@@ -114,7 +114,7 @@ pub fn getEnvMap(allocator: *Allocator) !BufMap {
}
test "os.getEnvMap" {
- var env = try getEnvMap(std.testing.allocator);
+ var env = try getEnvMap(std.testing.leak_count_allocator);
defer env.deinit();
}
@@ -165,7 +165,7 @@ pub fn getEnvVarOwned(allocator: *mem.Allocator, key: []const u8) GetEnvVarOwned
}
test "os.getEnvVarOwned" {
- var ga = std.testing.allocator;
+ var ga = std.testing.leak_count_allocator;
testing.expectError(error.EnvironmentVariableNotFound, getEnvVarOwned(ga, "BADENV"));
}
@@ -492,10 +492,11 @@ test "windows arg parsing" {
fn testWindowsCmdLine(input_cmd_line: [*]const u8, expected_args: []const []const u8) void {
var it = ArgIteratorWindows.initWithCmdLine(input_cmd_line);
for (expected_args) |expected_arg| {
- const arg = it.next(std.testing.allocator).? catch unreachable;
+ const arg = it.next(std.testing.leak_count_allocator).? catch unreachable;
+ defer std.testing.leak_count_allocator.free(arg);
testing.expectEqualSlices(u8, expected_arg, arg);
}
- testing.expect(it.next(std.testing.allocator) == null);
+ testing.expect(it.next(std.testing.leak_count_allocator) == null);
}
pub const UserInfo = struct {
diff --git a/lib/std/special/test_runner.zig b/lib/std/special/test_runner.zig
index 286dfbed1b..e519f7f766 100644
--- a/lib/std/special/test_runner.zig
+++ b/lib/std/special/test_runner.zig
@@ -14,6 +14,7 @@ pub fn main() anyerror!void {
for (test_fn_list) |test_fn, i| {
std.testing.allocator_instance.reset();
+
var test_node = root_node.start(test_fn.name, null);
test_node.activate();
progress.refresh();
@@ -36,6 +37,8 @@ pub fn main() anyerror!void {
return err;
},
}
+
+ try std.testing.leak_count_allocator_instance.validate();
}
root_node.end();
if (ok_count == test_fn_list.len) {
diff --git a/lib/std/testing.zig b/lib/std/testing.zig
index 02d7cdb393..58298708c3 100644
--- a/lib/std/testing.zig
+++ b/lib/std/testing.zig
@@ -7,6 +7,48 @@ pub const allocator = &allocator_instance.allocator;
pub var allocator_instance = std.heap.ThreadSafeFixedBufferAllocator.init(allocator_mem[0..]);
var allocator_mem: [100 * 1024]u8 = undefined;
+pub const leak_count_allocator = &leak_count_allocator_instance.allocator;
+pub var leak_count_allocator_instance = LeakCountAllocator.init(allocator);
+const LeakCountAllocator = struct {
+ count: usize,
+ allocator: std.mem.Allocator,
+ internal_allocator: *std.mem.Allocator,
+
+ fn init(allo: *std.mem.Allocator) LeakCountAllocator {
+ return .{
+ .count = 0,
+ .allocator = .{
+ .reallocFn = realloc,
+ .shrinkFn = shrink,
+ },
+ .internal_allocator = allo,
+ };
+ }
+
+ fn realloc(allo: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) ![]u8 {
+ const self = @fieldParentPtr(LeakCountAllocator, "allocator", allo);
+ if (old_mem.len == 0) {
+ self.count += 1;
+ }
+ return self.internal_allocator.reallocFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
+ }
+
+ fn shrink(allo: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
+ const self = @fieldParentPtr(LeakCountAllocator, "allocator", allo);
+ if (new_size == 0) {
+ self.count -= 1;
+ }
+ return self.internal_allocator.shrinkFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
+ }
+
+ fn validate(self: LeakCountAllocator) !void {
+ if (self.count > 0) {
+ std.debug.warn("Detected leaked allocations without matching free: {}\n", .{self.count});
+ return error.Leak;
+ }
+ }
+};
+
/// This function is intended to be used only in tests. It prints diagnostics to stderr
/// and then aborts when actual_error_union is not expected_error.
pub fn expectError(expected_error: anyerror, actual_error_union: var) void {
diff --git a/lib/std/unicode.zig b/lib/std/unicode.zig
index 638b5f5e2f..1884796835 100644
--- a/lib/std/unicode.zig
+++ b/lib/std/unicode.zig
@@ -501,14 +501,16 @@ test "utf16leToUtf8" {
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 'A');
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 'a');
- const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
+ defer std.testing.leak_count_allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "Aa"));
}
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0x80);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xffff);
- const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
+ defer std.testing.leak_count_allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xc2\x80" ++ "\xef\xbf\xbf"));
}
@@ -516,7 +518,8 @@ test "utf16leToUtf8" {
// the values just outside the surrogate half range
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xd7ff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xe000);
- const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
+ defer std.testing.leak_count_allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xed\x9f\xbf" ++ "\xee\x80\x80"));
}
@@ -524,7 +527,8 @@ test "utf16leToUtf8" {
// smallest surrogate pair
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xd800);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdc00);
- const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
+ defer std.testing.leak_count_allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xf0\x90\x80\x80"));
}
@@ -532,14 +536,16 @@ test "utf16leToUtf8" {
// largest surrogate pair
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xdbff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdfff);
- const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
+ defer std.testing.leak_count_allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xf4\x8f\xbf\xbf"));
}
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xdbff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdc00);
- const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
+ defer std.testing.leak_count_allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xf4\x8f\xb0\x80"));
}
}
diff --git a/lib/std/zig/ast.zig b/lib/std/zig/ast.zig
index e339343823..c967c78789 100644
--- a/lib/std/zig/ast.zig
+++ b/lib/std/zig/ast.zig
@@ -2287,7 +2287,7 @@ pub const Node = struct {
test "iterate" {
var root = Node.Root{
.base = Node{ .id = Node.Id.Root },
- .decls = Node.Root.DeclList.init(std.testing.allocator),
+ .decls = Node.Root.DeclList.init(std.testing.leak_count_allocator),
.eof_token = 0,
};
var base = &root.base;
diff --git a/test/compare_output.zig b/test/compare_output.zig
index 7a41d46f54..d743e0784a 100644
--- a/test/compare_output.zig
+++ b/test/compare_output.zig
@@ -445,7 +445,7 @@ pub fn addCases(cases: *tests.CompareOutputContext) void {
\\const std = @import("std");
\\const io = std.io;
\\const os = std.os;
- \\const allocator = std.testing.allocator;
+ \\const allocator = std.testing.leak_count_allocator;
\\
\\pub fn main() !void {
\\ var args_it = std.process.args();
@@ -486,7 +486,7 @@ pub fn addCases(cases: *tests.CompareOutputContext) void {
\\const std = @import("std");
\\const io = std.io;
\\const os = std.os;
- \\const allocator = std.testing.allocator;
+ \\const allocator = std.testing.leak_count_allocator;
\\
\\pub fn main() !void {
\\ var args_it = std.process.args();
diff --git a/test/compile_errors.zig b/test/compile_errors.zig
index 341062e161..2f3f0a9f1f 100644
--- a/test/compile_errors.zig
+++ b/test/compile_errors.zig
@@ -5765,7 +5765,7 @@ pub fn addCases(cases: *tests.CompileErrorContext) void {
\\
\\export fn entry() void {
\\ const a = MdNode.Header {
- \\ .text = MdText.init(&std.testing.allocator),
+ \\ .text = MdText.init(&std.testing.leak_count_allocator),
\\ .weight = HeaderWeight.H1,
\\ };
\\}
diff --git a/test/stage1/behavior/const_slice_child.zig b/test/stage1/behavior/const_slice_child.zig
index cf650ae88e..848a4780e5 100644
--- a/test/stage1/behavior/const_slice_child.zig
+++ b/test/stage1/behavior/const_slice_child.zig
@@ -22,7 +22,7 @@ fn foo(args: [][]const u8) void {
}
fn bar(argc: usize) void {
- const args = testing.allocator.alloc([]const u8, argc) catch unreachable;
+ const args = testing.leak_count_allocator.alloc([]const u8, argc) catch unreachable;
for (args) |_, i| {
const ptr = argv[i];
args[i] = ptr[0..strlen(ptr)];
diff --git a/test/standalone/brace_expansion/main.zig b/test/standalone/brace_expansion/main.zig
index c6bfd02e79..33fe826d18 100644
--- a/test/standalone/brace_expansion/main.zig
+++ b/test/standalone/brace_expansion/main.zig
@@ -201,7 +201,7 @@ pub fn main() !void {
}
test "invalid inputs" {
- global_allocator = std.testing.allocator;
+ global_allocator = std.testing.leak_count_allocator;
expectError("}ABC", error.InvalidInput);
expectError("{ABC", error.InvalidInput);
@@ -222,7 +222,7 @@ fn expectError(test_input: []const u8, expected_err: anyerror) void {
}
test "valid inputs" {
- global_allocator = std.testing.allocator;
+ global_allocator = std.testing.leak_count_allocator;
expectExpansion("{x,y,z}", "x y z");
expectExpansion("{A,B}{x,y}", "Ax Ay Bx By");
diff --git a/test/standalone/cat/main.zig b/test/standalone/cat/main.zig
index 34439f9c24..10ec00d6e8 100644
--- a/test/standalone/cat/main.zig
+++ b/test/standalone/cat/main.zig
@@ -4,7 +4,7 @@ const process = std.process;
const fs = std.fs;
const mem = std.mem;
const warn = std.debug.warn;
-const allocator = std.testing.allocator;
+const allocator = std.testing.leak_count_allocator;
pub fn main() !void {
var args_it = process.args();
diff --git a/test/standalone/empty_env/main.zig b/test/standalone/empty_env/main.zig
index f4ebf56136..dc2ac73738 100644
--- a/test/standalone/empty_env/main.zig
+++ b/test/standalone/empty_env/main.zig
@@ -1,6 +1,6 @@
const std = @import("std");
pub fn main() void {
- const env_map = std.process.getEnvMap(std.testing.allocator) catch @panic("unable to get env map");
+ const env_map = std.process.getEnvMap(std.testing.leak_count_allocator) catch @panic("unable to get env map");
std.testing.expect(env_map.count() == 0);
}
diff --git a/test/standalone/load_dynamic_library/main.zig b/test/standalone/load_dynamic_library/main.zig
index 70fc4986ac..5a2e55e937 100644
--- a/test/standalone/load_dynamic_library/main.zig
+++ b/test/standalone/load_dynamic_library/main.zig
@@ -1,8 +1,8 @@
const std = @import("std");
pub fn main() !void {
- const args = try std.process.argsAlloc(std.testing.allocator);
- defer std.process.argsFree(std.testing.allocator, args);
+ const args = try std.process.argsAlloc(std.testing.leak_count_allocator);
+ defer std.process.argsFree(std.testing.leak_count_allocator, args);
const dynlib_name = args[1];
From c4e6e5fad600f9277ce2cd76b7f2cf85922c6fd3 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 14:47:17 -0600
Subject: [PATCH 03/16] Add explicit free to docs
---
doc/docgen.zig | 1 +
1 file changed, 1 insertion(+)
diff --git a/doc/docgen.zig b/doc/docgen.zig
index 128314696e..cc4993d65f 100644
--- a/doc/docgen.zig
+++ b/doc/docgen.zig
@@ -673,6 +673,7 @@ const TermState = enum {
test "term color" {
const input_bytes = "A\x1b[32;1mgreen\x1b[0mB";
const result = try termColor(std.testing.leak_count_allocator, input_bytes);
+ defer std.testing.leak_count_allocator.free(result);
testing.expectEqualSlices(u8, "AgreenB", result);
}
From ffd30dbe28efce0b971d69df06ab684ceef0f881 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 15:30:13 -0600
Subject: [PATCH 04/16] Fix stage1 test
---
test/stage1/behavior/const_slice_child.zig | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/test/stage1/behavior/const_slice_child.zig b/test/stage1/behavior/const_slice_child.zig
index 848a4780e5..7d969ae468 100644
--- a/test/stage1/behavior/const_slice_child.zig
+++ b/test/stage1/behavior/const_slice_child.zig
@@ -1,6 +1,7 @@
const std = @import("std");
const debug = std.debug;
-const expect = std.testing.expect;
+const testing = std.testing;
+const expect = testing.expect;
var argv: [*]const [*]const u8 = undefined;
@@ -23,6 +24,7 @@ fn foo(args: [][]const u8) void {
fn bar(argc: usize) void {
const args = testing.leak_count_allocator.alloc([]const u8, argc) catch unreachable;
+ defer testing.leak_count_allocator.free(args);
for (args) |_, i| {
const ptr = argv[i];
args[i] = ptr[0..strlen(ptr)];
From 0c137934cbd10528c2dced898b6c5485ab528e6d Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 17:26:10 -0600
Subject: [PATCH 05/16] Move FailingAllocator to testing
---
lib/std/debug.zig | 10 +---
lib/std/io.zig | 2 +-
lib/std/testing.zig | 45 ++----------------
.../{debug => testing}/failing_allocator.zig | 0
lib/std/testing/leak_count_allocator.zig | 46 +++++++++++++++++++
lib/std/zig/parser_test.zig | 4 +-
6 files changed, 56 insertions(+), 51 deletions(-)
rename lib/std/{debug => testing}/failing_allocator.zig (100%)
create mode 100644 lib/std/testing/leak_count_allocator.zig
diff --git a/lib/std/debug.zig b/lib/std/debug.zig
index 7e1a99a395..32aba0d9b4 100644
--- a/lib/std/debug.zig
+++ b/lib/std/debug.zig
@@ -19,8 +19,8 @@ const windows = std.os.windows;
pub const leb = @import("debug/leb128.zig");
-pub const FailingAllocator = @import("debug/failing_allocator.zig").FailingAllocator;
-pub const failing_allocator = &FailingAllocator.init(&global_fixed_allocator.allocator, 0).allocator;
+pub const global_allocator = @compileError("Please switch to std.testing.leak_count_allocator.");
+pub const failing_allocator = @compileError("Please switch to std.testing.failing_allocator.");
pub const runtime_safety = switch (builtin.mode) {
.Debug, .ReleaseSafe => true,
@@ -2192,12 +2192,6 @@ fn readInitialLength(comptime E: type, in_stream: *io.InStream(E), is_64: *bool)
}
}
-pub const global_allocator = blk: {
- @compileError("Please switch to std.testing.leak_count_allocator.");
-};
-var global_fixed_allocator = std.heap.ThreadSafeFixedBufferAllocator.init(global_allocator_mem[0..]);
-var global_allocator_mem: [100 * 1024]u8 = undefined;
-
/// TODO multithreaded awareness
var debug_info_allocator: ?*mem.Allocator = null;
var debug_info_arena_allocator: std.heap.ArenaAllocator = undefined;
diff --git a/lib/std/io.zig b/lib/std/io.zig
index 93f2ae7680..a0e58c373d 100644
--- a/lib/std/io.zig
+++ b/lib/std/io.zig
@@ -891,7 +891,7 @@ pub fn readLineSlice(slice: []u8) ![]u8 {
pub fn readLineSliceFrom(stream: var, slice: []u8) ![]u8 {
// We cannot use Buffer.fromOwnedSlice, as it wants to append a null byte
// after taking ownership, which would always require an allocation.
- var buf = std.Buffer{ .list = std.ArrayList(u8).fromOwnedSlice(debug.failing_allocator, slice) };
+ var buf = std.Buffer{ .list = std.ArrayList(u8).fromOwnedSlice(testing.failing_allocator, slice) };
try buf.resize(0);
return try readLineFrom(stream, &buf);
}
diff --git a/lib/std/testing.zig b/lib/std/testing.zig
index 58298708c3..07479a3852 100644
--- a/lib/std/testing.zig
+++ b/lib/std/testing.zig
@@ -7,47 +7,12 @@ pub const allocator = &allocator_instance.allocator;
pub var allocator_instance = std.heap.ThreadSafeFixedBufferAllocator.init(allocator_mem[0..]);
var allocator_mem: [100 * 1024]u8 = undefined;
-pub const leak_count_allocator = &leak_count_allocator_instance.allocator;
+pub const FailingAllocator = @import("testing/failing_allocator.zig").FailingAllocator;
+pub const failing_allocator = &FailingAllocator.init(allocator, 0).allocator;
+
+pub const LeakCountAllocator = @import("testing/leak_count_allocator.zig").LeakCountAllocator;
pub var leak_count_allocator_instance = LeakCountAllocator.init(allocator);
-const LeakCountAllocator = struct {
- count: usize,
- allocator: std.mem.Allocator,
- internal_allocator: *std.mem.Allocator,
-
- fn init(allo: *std.mem.Allocator) LeakCountAllocator {
- return .{
- .count = 0,
- .allocator = .{
- .reallocFn = realloc,
- .shrinkFn = shrink,
- },
- .internal_allocator = allo,
- };
- }
-
- fn realloc(allo: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) ![]u8 {
- const self = @fieldParentPtr(LeakCountAllocator, "allocator", allo);
- if (old_mem.len == 0) {
- self.count += 1;
- }
- return self.internal_allocator.reallocFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
- }
-
- fn shrink(allo: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
- const self = @fieldParentPtr(LeakCountAllocator, "allocator", allo);
- if (new_size == 0) {
- self.count -= 1;
- }
- return self.internal_allocator.shrinkFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
- }
-
- fn validate(self: LeakCountAllocator) !void {
- if (self.count > 0) {
- std.debug.warn("Detected leaked allocations without matching free: {}\n", .{self.count});
- return error.Leak;
- }
- }
-};
+pub const leak_count_allocator = &leak_count_allocator_instance.allocator;
/// This function is intended to be used only in tests. It prints diagnostics to stderr
/// and then aborts when actual_error_union is not expected_error.
diff --git a/lib/std/debug/failing_allocator.zig b/lib/std/testing/failing_allocator.zig
similarity index 100%
rename from lib/std/debug/failing_allocator.zig
rename to lib/std/testing/failing_allocator.zig
diff --git a/lib/std/testing/leak_count_allocator.zig b/lib/std/testing/leak_count_allocator.zig
new file mode 100644
index 0000000000..84c3248178
--- /dev/null
+++ b/lib/std/testing/leak_count_allocator.zig
@@ -0,0 +1,46 @@
+const std = @import("../std.zig");
+
+/// This allocator is used in front of another allocator and counts the numbers of allocs and frees.
+/// The test runner asserts every alloc has a corresponding free at the end of each test.
+///
+/// The detection algorithm is incredibly primitive and only accounts for number of calls.
+/// This should be replaced by the general purpose debug allocator.
+pub const LeakCountAllocator = struct {
+ count: usize,
+ allocator: std.mem.Allocator,
+ internal_allocator: *std.mem.Allocator,
+
+ pub fn init(allocator: *std.mem.Allocator) LeakCountAllocator {
+ return .{
+ .count = 0,
+ .allocator = .{
+ .reallocFn = realloc,
+ .shrinkFn = shrink,
+ },
+ .internal_allocator = allocator,
+ };
+ }
+
+ fn realloc(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) ![]u8 {
+ const self = @fieldParentPtr(LeakCountAllocator, "allocator", allocator);
+ if (old_mem.len == 0) {
+ self.count += 1;
+ }
+ return self.internal_allocator.reallocFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
+ }
+
+ fn shrink(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
+ const self = @fieldParentPtr(LeakCountAllocator, "allocator", allocator);
+ if (new_size == 0) {
+ self.count -= 1;
+ }
+ return self.internal_allocator.shrinkFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
+ }
+
+ pub fn validate(self: LeakCountAllocator) !void {
+ if (self.count > 0) {
+ std.debug.warn("Detected leaked allocations without matching free: {}\n", .{self.count});
+ return error.Leak;
+ }
+ }
+};
diff --git a/lib/std/zig/parser_test.zig b/lib/std/zig/parser_test.zig
index c57540ade9..1640d09992 100644
--- a/lib/std/zig/parser_test.zig
+++ b/lib/std/zig/parser_test.zig
@@ -2773,7 +2773,7 @@ fn testTransform(source: []const u8, expected_source: []const u8) !void {
const needed_alloc_count = x: {
// Try it once with unlimited memory, make sure it works
var fixed_allocator = std.heap.FixedBufferAllocator.init(fixed_buffer_mem[0..]);
- var failing_allocator = std.debug.FailingAllocator.init(&fixed_allocator.allocator, maxInt(usize));
+ var failing_allocator = std.testing.FailingAllocator.init(&fixed_allocator.allocator, maxInt(usize));
var anything_changed: bool = undefined;
const result_source = try testParse(source, &failing_allocator.allocator, &anything_changed);
if (!mem.eql(u8, result_source, expected_source)) {
@@ -2797,7 +2797,7 @@ fn testTransform(source: []const u8, expected_source: []const u8) !void {
var fail_index: usize = 0;
while (fail_index < needed_alloc_count) : (fail_index += 1) {
var fixed_allocator = std.heap.FixedBufferAllocator.init(fixed_buffer_mem[0..]);
- var failing_allocator = std.debug.FailingAllocator.init(&fixed_allocator.allocator, fail_index);
+ var failing_allocator = std.testing.FailingAllocator.init(&fixed_allocator.allocator, fail_index);
var anything_changed: bool = undefined;
if (testParse(source, &failing_allocator.allocator, &anything_changed)) |_| {
return error.NondeterministicMemoryUsage;
From 70ad84c8209ae7ab14472f77f9452820bc47c30e Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 17:38:52 -0600
Subject: [PATCH 06/16] Use defer/panic to better account for test failure
---
lib/std/special/test_runner.zig | 7 +++++--
1 file changed, 5 insertions(+), 2 deletions(-)
diff --git a/lib/std/special/test_runner.zig b/lib/std/special/test_runner.zig
index e519f7f766..a5bbfff6fb 100644
--- a/lib/std/special/test_runner.zig
+++ b/lib/std/special/test_runner.zig
@@ -14,6 +14,11 @@ pub fn main() anyerror!void {
for (test_fn_list) |test_fn, i| {
std.testing.allocator_instance.reset();
+ defer {
+ std.testing.leak_count_allocator_instance.validate() catch |err| {
+ @panic(@errorName(err));
+ };
+ }
var test_node = root_node.start(test_fn.name, null);
test_node.activate();
@@ -37,8 +42,6 @@ pub fn main() anyerror!void {
return err;
},
}
-
- try std.testing.leak_count_allocator_instance.validate();
}
root_node.end();
if (ok_count == test_fn_list.len) {
From b077f3ab7d55374d2770336db236ecd7b18238a8 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 21:22:01 -0600
Subject: [PATCH 07/16] Promoted "leak_count_allocator" to the main
testing.allocator
---
doc/docgen.zig | 4 +--
lib/std/array_list.zig | 14 ++++----
lib/std/buffer.zig | 6 ++--
lib/std/debug.zig | 2 +-
lib/std/fifo.zig | 4 +--
lib/std/fmt.zig | 10 +++---
lib/std/fs/path.zig | 32 +++++++++----------
lib/std/linked_list.zig | 6 ++--
lib/std/os/test.zig | 2 +-
lib/std/priority_queue.zig | 28 ++++++++--------
lib/std/process.zig | 10 +++---
lib/std/special/test_runner.zig | 7 ++--
lib/std/testing.zig | 16 +++++-----
lib/std/testing/leak_count_allocator.zig | 2 +-
lib/std/unicode.zig | 24 +++++++-------
lib/std/zig/ast.zig | 2 +-
test/compare_output.zig | 4 +--
test/compile_errors.zig | 2 +-
test/stage1/behavior/const_slice_child.zig | 4 +--
test/standalone/brace_expansion/main.zig | 4 +--
test/standalone/cat/main.zig | 2 +-
test/standalone/empty_env/main.zig | 2 +-
test/standalone/load_dynamic_library/main.zig | 4 +--
23 files changed, 96 insertions(+), 95 deletions(-)
diff --git a/doc/docgen.zig b/doc/docgen.zig
index cc4993d65f..e94f3500e7 100644
--- a/doc/docgen.zig
+++ b/doc/docgen.zig
@@ -672,8 +672,8 @@ const TermState = enum {
test "term color" {
const input_bytes = "A\x1b[32;1mgreen\x1b[0mB";
- const result = try termColor(std.testing.leak_count_allocator, input_bytes);
- defer std.testing.leak_count_allocator.free(result);
+ const result = try termColor(std.testing.allocator, input_bytes);
+ defer std.testing.allocator.free(result);
testing.expectEqualSlices(u8, "AgreenB", result);
}
diff --git a/lib/std/array_list.zig b/lib/std/array_list.zig
index 57c0d706a2..b028390465 100644
--- a/lib/std/array_list.zig
+++ b/lib/std/array_list.zig
@@ -320,7 +320,7 @@ test "std.ArrayList.basic" {
}
test "std.ArrayList.orderedRemove" {
- var list = ArrayList(i32).init(testing.leak_count_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -347,7 +347,7 @@ test "std.ArrayList.orderedRemove" {
}
test "std.ArrayList.swapRemove" {
- var list = ArrayList(i32).init(testing.leak_count_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -374,7 +374,7 @@ test "std.ArrayList.swapRemove" {
}
test "std.ArrayList.swapRemoveOrError" {
- var list = ArrayList(i32).init(testing.leak_count_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
// Test just after initialization
@@ -402,7 +402,7 @@ test "std.ArrayList.swapRemoveOrError" {
}
test "std.ArrayList.insert" {
- var list = ArrayList(i32).init(testing.leak_count_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -416,7 +416,7 @@ test "std.ArrayList.insert" {
}
test "std.ArrayList.insertSlice" {
- var list = ArrayList(i32).init(testing.leak_count_allocator);
+ var list = ArrayList(i32).init(testing.allocator);
defer list.deinit();
try list.append(1);
@@ -443,8 +443,8 @@ const Item = struct {
};
test "std.ArrayList: ArrayList(T) of struct T" {
- var root = Item{ .integer = 1, .sub_items = ArrayList(Item).init(testing.leak_count_allocator) };
+ var root = Item{ .integer = 1, .sub_items = ArrayList(Item).init(testing.allocator) };
defer root.sub_items.deinit();
- try root.sub_items.append(Item{ .integer = 42, .sub_items = ArrayList(Item).init(testing.leak_count_allocator) });
+ try root.sub_items.append(Item{ .integer = 42, .sub_items = ArrayList(Item).init(testing.allocator) });
testing.expect(root.sub_items.items[0].integer == 42);
}
diff --git a/lib/std/buffer.zig b/lib/std/buffer.zig
index cc1166e0e7..077b2b615f 100644
--- a/lib/std/buffer.zig
+++ b/lib/std/buffer.zig
@@ -150,7 +150,7 @@ pub const Buffer = struct {
};
test "simple Buffer" {
- var buf = try Buffer.init(testing.leak_count_allocator, "");
+ var buf = try Buffer.init(testing.allocator, "");
defer buf.deinit();
testing.expect(buf.len() == 0);
@@ -172,7 +172,7 @@ test "simple Buffer" {
}
test "Buffer.initSize" {
- var buf = try Buffer.initSize(testing.leak_count_allocator, 3);
+ var buf = try Buffer.initSize(testing.allocator, 3);
defer buf.deinit();
testing.expect(buf.len() == 3);
try buf.append("hello");
@@ -180,7 +180,7 @@ test "Buffer.initSize" {
}
test "Buffer.initCapacity" {
- var buf = try Buffer.initCapacity(testing.leak_count_allocator, 10);
+ var buf = try Buffer.initCapacity(testing.allocator, 10);
defer buf.deinit();
testing.expect(buf.len() == 0);
testing.expect(buf.capacity() >= 10);
diff --git a/lib/std/debug.zig b/lib/std/debug.zig
index 32aba0d9b4..8468c4eaaf 100644
--- a/lib/std/debug.zig
+++ b/lib/std/debug.zig
@@ -19,7 +19,7 @@ const windows = std.os.windows;
pub const leb = @import("debug/leb128.zig");
-pub const global_allocator = @compileError("Please switch to std.testing.leak_count_allocator.");
+pub const global_allocator = @compileError("Please switch to std.testing.allocator.");
pub const failing_allocator = @compileError("Please switch to std.testing.failing_allocator.");
pub const runtime_safety = switch (builtin.mode) {
diff --git a/lib/std/fifo.zig b/lib/std/fifo.zig
index dd429289d1..ac8589f2ce 100644
--- a/lib/std/fifo.zig
+++ b/lib/std/fifo.zig
@@ -347,7 +347,7 @@ pub fn LinearFifo(
}
test "LinearFifo(u8, .Dynamic)" {
- var fifo = LinearFifo(u8, .Dynamic).init(testing.leak_count_allocator);
+ var fifo = LinearFifo(u8, .Dynamic).init(testing.allocator);
defer fifo.deinit();
try fifo.write("HELLO");
@@ -422,7 +422,7 @@ test "LinearFifo" {
var fifo = switch (bt) {
.Static => FifoType.init(),
.Slice => FifoType.init(buf[0..]),
- .Dynamic => FifoType.init(testing.leak_count_allocator),
+ .Dynamic => FifoType.init(testing.allocator),
};
defer fifo.deinit();
diff --git a/lib/std/fmt.zig b/lib/std/fmt.zig
index 529131065c..627d8bc7ab 100644
--- a/lib/std/fmt.zig
+++ b/lib/std/fmt.zig
@@ -1598,7 +1598,7 @@ test "hexToBytes" {
test "formatIntValue with comptime_int" {
const value: comptime_int = 123456789123456789;
- var buf = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ var buf = try std.Buffer.init(std.testing.allocator, "");
defer buf.deinit();
try formatIntValue(value, "", FormatOptions{}, &buf, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append);
std.testing.expect(mem.eql(u8, buf.toSlice(), "123456789123456789"));
@@ -1653,22 +1653,22 @@ test "formatType max_depth" {
inst.a = &inst;
inst.tu.ptr = &inst.tu;
- var buf0 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ var buf0 = try std.Buffer.init(std.testing.allocator, "");
defer buf0.deinit();
try formatType(inst, "", FormatOptions{}, &buf0, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 0);
std.testing.expect(mem.eql(u8, buf0.toSlice(), "S{ ... }"));
- var buf1 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ var buf1 = try std.Buffer.init(std.testing.allocator, "");
defer buf1.deinit();
try formatType(inst, "", FormatOptions{}, &buf1, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 1);
std.testing.expect(mem.eql(u8, buf1.toSlice(), "S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }"));
- var buf2 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ var buf2 = try std.Buffer.init(std.testing.allocator, "");
defer buf2.deinit();
try formatType(inst, "", FormatOptions{}, &buf2, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 2);
std.testing.expect(mem.eql(u8, buf2.toSlice(), "S{ .a = S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ ... } }, .e = E.Two, .vec = (10.200,2.220) }"));
- var buf3 = try std.Buffer.init(std.testing.leak_count_allocator, "");
+ var buf3 = try std.Buffer.init(std.testing.allocator, "");
defer buf3.deinit();
try formatType(inst, "", FormatOptions{}, &buf3, @TypeOf(std.Buffer.append).ReturnType.ErrorSet, std.Buffer.append, 3);
std.testing.expect(mem.eql(u8, buf3.toSlice(), "S{ .a = S{ .a = S{ .a = S{ ... }, .tu = TU{ ... }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ ... } }, .e = E.Two, .vec = (10.200,2.220) }, .tu = TU{ .ptr = TU{ .ptr = TU{ ... } } }, .e = E.Two, .vec = (10.200,2.220) }"));
diff --git a/lib/std/fs/path.zig b/lib/std/fs/path.zig
index 424e9f5759..66e24a4805 100644
--- a/lib/std/fs/path.zig
+++ b/lib/std/fs/path.zig
@@ -665,8 +665,8 @@ pub fn resolvePosix(allocator: *Allocator, paths: []const []const u8) ![]u8 {
}
test "resolve" {
- const cwd = try process.getCwdAlloc(testing.leak_count_allocator);
- defer testing.leak_count_allocator.free(cwd);
+ const cwd = try process.getCwdAlloc(testing.allocator);
+ defer testing.allocator.free(cwd);
if (builtin.os == .windows) {
if (windowsParsePath(cwd).kind == WindowsPath.Kind.Drive) {
cwd[0] = asciiUpper(cwd[0]);
@@ -684,26 +684,26 @@ test "resolveWindows" {
return error.SkipZigTest;
}
if (builtin.os == .windows) {
- const cwd = try process.getCwdAlloc(testing.leak_count_allocator);
- defer testing.leak_count_allocator.free(cwd);
+ const cwd = try process.getCwdAlloc(testing.allocator);
+ defer testing.allocator.free(cwd);
const parsed_cwd = windowsParsePath(cwd);
{
- const expected = try join(testing.leak_count_allocator, &[_][]const u8{
+ const expected = try join(testing.allocator, &[_][]const u8{
parsed_cwd.disk_designator,
"usr\\local\\lib\\zig\\std\\array_list.zig",
});
- defer testing.leak_count_allocator.free(expected);
+ defer testing.allocator.free(expected);
if (parsed_cwd.kind == WindowsPath.Kind.Drive) {
expected[0] = asciiUpper(parsed_cwd.disk_designator[0]);
}
try testResolveWindows(&[_][]const u8{ "/usr/local", "lib\\zig\\std\\array_list.zig" }, expected);
}
{
- const expected = try join(testing.leak_count_allocator, &[_][]const u8{
+ const expected = try join(testing.allocator, &[_][]const u8{
cwd,
"usr\\local\\lib\\zig",
});
- defer testing.leak_count_allocator.free(expected);
+ defer testing.allocator.free(expected);
if (parsed_cwd.kind == WindowsPath.Kind.Drive) {
expected[0] = asciiUpper(parsed_cwd.disk_designator[0]);
}
@@ -740,14 +740,14 @@ test "resolvePosix" {
}
fn testResolveWindows(paths: []const []const u8, expected: []const u8) !void {
- const actual = try resolveWindows(testing.leak_count_allocator, paths);
- defer testing.leak_count_allocator.free(actual);
+ const actual = try resolveWindows(testing.allocator, paths);
+ defer testing.allocator.free(actual);
return testing.expect(mem.eql(u8, actual, expected));
}
fn testResolvePosix(paths: []const []const u8, expected: []const u8) !void {
- const actual = try resolvePosix(testing.leak_count_allocator, paths);
- defer testing.leak_count_allocator.free(actual);
+ const actual = try resolvePosix(testing.allocator, paths);
+ defer testing.allocator.free(actual);
return testing.expect(mem.eql(u8, actual, expected));
}
@@ -1172,13 +1172,13 @@ test "relative" {
}
fn testRelativePosix(from: []const u8, to: []const u8, expected_output: []const u8) !void {
- const result = try relativePosix(testing.leak_count_allocator, from, to);
- defer testing.leak_count_allocator.free(result);
+ const result = try relativePosix(testing.allocator, from, to);
+ defer testing.allocator.free(result);
testing.expectEqualSlices(u8, expected_output, result);
}
fn testRelativeWindows(from: []const u8, to: []const u8, expected_output: []const u8) !void {
- const result = try relativeWindows(testing.leak_count_allocator, from, to);
- defer testing.leak_count_allocator.free(result);
+ const result = try relativeWindows(testing.allocator, from, to);
+ defer testing.allocator.free(result);
testing.expectEqualSlices(u8, expected_output, result);
}
diff --git a/lib/std/linked_list.zig b/lib/std/linked_list.zig
index d1652a8047..a21c9a83eb 100644
--- a/lib/std/linked_list.zig
+++ b/lib/std/linked_list.zig
@@ -143,7 +143,7 @@ pub fn SinglyLinkedList(comptime T: type) type {
}
test "basic SinglyLinkedList test" {
- const allocator = testing.leak_count_allocator;
+ const allocator = testing.allocator;
var list = SinglyLinkedList(u32).init();
var one = try list.createNode(1, allocator);
@@ -404,7 +404,7 @@ pub fn TailQueue(comptime T: type) type {
}
test "basic TailQueue test" {
- const allocator = testing.leak_count_allocator;
+ const allocator = testing.allocator;
var list = TailQueue(u32).init();
var one = try list.createNode(1, allocator);
@@ -456,7 +456,7 @@ test "basic TailQueue test" {
}
test "TailQueue concatenation" {
- const allocator = testing.leak_count_allocator;
+ const allocator = testing.allocator;
var list1 = TailQueue(u32).init();
var list2 = TailQueue(u32).init();
diff --git a/lib/std/os/test.zig b/lib/std/os/test.zig
index 2e60fe500e..0d3be9e7e4 100644
--- a/lib/std/os/test.zig
+++ b/lib/std/os/test.zig
@@ -9,7 +9,7 @@ const elf = std.elf;
const File = std.fs.File;
const Thread = std.Thread;
-const a = std.testing.leak_count_allocator;
+const a = std.testing.allocator;
const builtin = @import("builtin");
const AtomicRmwOp = builtin.AtomicRmwOp;
diff --git a/lib/std/priority_queue.zig b/lib/std/priority_queue.zig
index 9c07d20798..e726a07a88 100644
--- a/lib/std/priority_queue.zig
+++ b/lib/std/priority_queue.zig
@@ -239,7 +239,7 @@ fn greaterThan(a: u32, b: u32) bool {
const PQ = PriorityQueue(u32);
test "std.PriorityQueue: add and remove min heap" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(54);
@@ -257,7 +257,7 @@ test "std.PriorityQueue: add and remove min heap" {
}
test "std.PriorityQueue: add and remove same min heap" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(1);
@@ -275,14 +275,14 @@ test "std.PriorityQueue: add and remove same min heap" {
}
test "std.PriorityQueue: removeOrNull on empty" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
expect(queue.removeOrNull() == null);
}
test "std.PriorityQueue: edge case 3 elements" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(9);
@@ -294,7 +294,7 @@ test "std.PriorityQueue: edge case 3 elements" {
}
test "std.PriorityQueue: peek" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
expect(queue.peek() == null);
@@ -306,7 +306,7 @@ test "std.PriorityQueue: peek" {
}
test "std.PriorityQueue: sift up with odd indices" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
for (items) |e| {
@@ -320,7 +320,7 @@ test "std.PriorityQueue: sift up with odd indices" {
}
test "std.PriorityQueue: addSlice" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
try queue.addSlice(items[0..]);
@@ -333,8 +333,8 @@ test "std.PriorityQueue: addSlice" {
test "std.PriorityQueue: fromOwnedSlice" {
const items = [_]u32{ 15, 7, 21, 14, 13, 22, 12, 6, 7, 25, 5, 24, 11, 16, 15, 24, 2, 1 };
- const heap_items = try std.mem.dupe(testing.leak_count_allocator, u32, items[0..]);
- var queue = PQ.fromOwnedSlice(testing.leak_count_allocator, lessThan, heap_items[0..]);
+ const heap_items = try std.mem.dupe(testing.allocator, u32, items[0..]);
+ var queue = PQ.fromOwnedSlice(testing.allocator, lessThan, heap_items[0..]);
defer queue.deinit();
const sorted_items = [_]u32{ 1, 2, 5, 6, 7, 7, 11, 12, 13, 14, 15, 15, 16, 21, 22, 24, 24, 25 };
@@ -344,7 +344,7 @@ test "std.PriorityQueue: fromOwnedSlice" {
}
test "std.PriorityQueue: add and remove max heap" {
- var queue = PQ.init(testing.leak_count_allocator, greaterThan);
+ var queue = PQ.init(testing.allocator, greaterThan);
defer queue.deinit();
try queue.add(54);
@@ -362,7 +362,7 @@ test "std.PriorityQueue: add and remove max heap" {
}
test "std.PriorityQueue: add and remove same max heap" {
- var queue = PQ.init(testing.leak_count_allocator, greaterThan);
+ var queue = PQ.init(testing.allocator, greaterThan);
defer queue.deinit();
try queue.add(1);
@@ -380,8 +380,8 @@ test "std.PriorityQueue: add and remove same max heap" {
}
test "std.PriorityQueue: iterator" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
- var map = std.AutoHashMap(u32, void).init(testing.leak_count_allocator);
+ var queue = PQ.init(testing.allocator, lessThan);
+ var map = std.AutoHashMap(u32, void).init(testing.allocator);
defer {
queue.deinit();
map.deinit();
@@ -402,7 +402,7 @@ test "std.PriorityQueue: iterator" {
}
test "std.PriorityQueue: remove at index" {
- var queue = PQ.init(testing.leak_count_allocator, lessThan);
+ var queue = PQ.init(testing.allocator, lessThan);
defer queue.deinit();
try queue.add(3);
diff --git a/lib/std/process.zig b/lib/std/process.zig
index 19f0eac235..dd5e124654 100644
--- a/lib/std/process.zig
+++ b/lib/std/process.zig
@@ -114,7 +114,7 @@ pub fn getEnvMap(allocator: *Allocator) !BufMap {
}
test "os.getEnvMap" {
- var env = try getEnvMap(std.testing.leak_count_allocator);
+ var env = try getEnvMap(std.testing.allocator);
defer env.deinit();
}
@@ -165,7 +165,7 @@ pub fn getEnvVarOwned(allocator: *mem.Allocator, key: []const u8) GetEnvVarOwned
}
test "os.getEnvVarOwned" {
- var ga = std.testing.leak_count_allocator;
+ var ga = std.testing.allocator;
testing.expectError(error.EnvironmentVariableNotFound, getEnvVarOwned(ga, "BADENV"));
}
@@ -492,11 +492,11 @@ test "windows arg parsing" {
fn testWindowsCmdLine(input_cmd_line: [*]const u8, expected_args: []const []const u8) void {
var it = ArgIteratorWindows.initWithCmdLine(input_cmd_line);
for (expected_args) |expected_arg| {
- const arg = it.next(std.testing.leak_count_allocator).? catch unreachable;
- defer std.testing.leak_count_allocator.free(arg);
+ const arg = it.next(std.testing.allocator).? catch unreachable;
+ defer std.testing.allocator.free(arg);
testing.expectEqualSlices(u8, expected_arg, arg);
}
- testing.expect(it.next(std.testing.leak_count_allocator) == null);
+ testing.expect(it.next(std.testing.allocator) == null);
}
pub const UserInfo = struct {
diff --git a/lib/std/special/test_runner.zig b/lib/std/special/test_runner.zig
index a5bbfff6fb..9a2d68a73d 100644
--- a/lib/std/special/test_runner.zig
+++ b/lib/std/special/test_runner.zig
@@ -13,10 +13,11 @@ pub fn main() anyerror!void {
};
for (test_fn_list) |test_fn, i| {
- std.testing.allocator_instance.reset();
+ std.testing.base_allocator_instance.reset();
defer {
- std.testing.leak_count_allocator_instance.validate() catch |err| {
- @panic(@errorName(err));
+ std.testing.allocator_instance.validate() catch |err| switch (err) {
+ error.Leak => std.debug.panic("", .{}),
+ else => std.debug.panic("error.{}", .{@errorName(err)}),
};
}
diff --git a/lib/std/testing.zig b/lib/std/testing.zig
index 07479a3852..4850f2e9b9 100644
--- a/lib/std/testing.zig
+++ b/lib/std/testing.zig
@@ -2,18 +2,18 @@ const builtin = @import("builtin");
const TypeId = builtin.TypeId;
const std = @import("std.zig");
+pub const LeakCountAllocator = @import("testing/leak_count_allocator.zig").LeakCountAllocator;
+pub const FailingAllocator = @import("testing/failing_allocator.zig").FailingAllocator;
+
/// This should only be used in temporary test programs.
pub const allocator = &allocator_instance.allocator;
-pub var allocator_instance = std.heap.ThreadSafeFixedBufferAllocator.init(allocator_mem[0..]);
+pub var allocator_instance = LeakCountAllocator.init(&base_allocator_instance.allocator);
+
+pub const failing_allocator = &FailingAllocator.init(&base_allocator_instance.allocator, 0).allocator;
+
+pub var base_allocator_instance = std.heap.ThreadSafeFixedBufferAllocator.init(allocator_mem[0..]);
var allocator_mem: [100 * 1024]u8 = undefined;
-pub const FailingAllocator = @import("testing/failing_allocator.zig").FailingAllocator;
-pub const failing_allocator = &FailingAllocator.init(allocator, 0).allocator;
-
-pub const LeakCountAllocator = @import("testing/leak_count_allocator.zig").LeakCountAllocator;
-pub var leak_count_allocator_instance = LeakCountAllocator.init(allocator);
-pub const leak_count_allocator = &leak_count_allocator_instance.allocator;
-
/// This function is intended to be used only in tests. It prints diagnostics to stderr
/// and then aborts when actual_error_union is not expected_error.
pub fn expectError(expected_error: anyerror, actual_error_union: var) void {
diff --git a/lib/std/testing/leak_count_allocator.zig b/lib/std/testing/leak_count_allocator.zig
index 84c3248178..6cdcfa2678 100644
--- a/lib/std/testing/leak_count_allocator.zig
+++ b/lib/std/testing/leak_count_allocator.zig
@@ -39,7 +39,7 @@ pub const LeakCountAllocator = struct {
pub fn validate(self: LeakCountAllocator) !void {
if (self.count > 0) {
- std.debug.warn("Detected leaked allocations without matching free: {}\n", .{self.count});
+ std.debug.warn("error - detected leaked allocations without matching free: {}\n", .{self.count});
return error.Leak;
}
}
diff --git a/lib/std/unicode.zig b/lib/std/unicode.zig
index 1884796835..77783c3edd 100644
--- a/lib/std/unicode.zig
+++ b/lib/std/unicode.zig
@@ -501,16 +501,16 @@ test "utf16leToUtf8" {
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 'A');
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 'a');
- const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
- defer std.testing.leak_count_allocator.free(utf8);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ defer std.testing.allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "Aa"));
}
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0x80);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xffff);
- const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
- defer std.testing.leak_count_allocator.free(utf8);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ defer std.testing.allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xc2\x80" ++ "\xef\xbf\xbf"));
}
@@ -518,8 +518,8 @@ test "utf16leToUtf8" {
// the values just outside the surrogate half range
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xd7ff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xe000);
- const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
- defer std.testing.leak_count_allocator.free(utf8);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ defer std.testing.allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xed\x9f\xbf" ++ "\xee\x80\x80"));
}
@@ -527,8 +527,8 @@ test "utf16leToUtf8" {
// smallest surrogate pair
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xd800);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdc00);
- const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
- defer std.testing.leak_count_allocator.free(utf8);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ defer std.testing.allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xf0\x90\x80\x80"));
}
@@ -536,16 +536,16 @@ test "utf16leToUtf8" {
// largest surrogate pair
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xdbff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdfff);
- const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
- defer std.testing.leak_count_allocator.free(utf8);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ defer std.testing.allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xf4\x8f\xbf\xbf"));
}
{
mem.writeIntSliceLittle(u16, utf16le_as_bytes[0..], 0xdbff);
mem.writeIntSliceLittle(u16, utf16le_as_bytes[2..], 0xdc00);
- const utf8 = try utf16leToUtf8Alloc(std.testing.leak_count_allocator, &utf16le);
- defer std.testing.leak_count_allocator.free(utf8);
+ const utf8 = try utf16leToUtf8Alloc(std.testing.allocator, &utf16le);
+ defer std.testing.allocator.free(utf8);
testing.expect(mem.eql(u8, utf8, "\xf4\x8f\xb0\x80"));
}
}
diff --git a/lib/std/zig/ast.zig b/lib/std/zig/ast.zig
index c967c78789..e339343823 100644
--- a/lib/std/zig/ast.zig
+++ b/lib/std/zig/ast.zig
@@ -2287,7 +2287,7 @@ pub const Node = struct {
test "iterate" {
var root = Node.Root{
.base = Node{ .id = Node.Id.Root },
- .decls = Node.Root.DeclList.init(std.testing.leak_count_allocator),
+ .decls = Node.Root.DeclList.init(std.testing.allocator),
.eof_token = 0,
};
var base = &root.base;
diff --git a/test/compare_output.zig b/test/compare_output.zig
index d743e0784a..7a41d46f54 100644
--- a/test/compare_output.zig
+++ b/test/compare_output.zig
@@ -445,7 +445,7 @@ pub fn addCases(cases: *tests.CompareOutputContext) void {
\\const std = @import("std");
\\const io = std.io;
\\const os = std.os;
- \\const allocator = std.testing.leak_count_allocator;
+ \\const allocator = std.testing.allocator;
\\
\\pub fn main() !void {
\\ var args_it = std.process.args();
@@ -486,7 +486,7 @@ pub fn addCases(cases: *tests.CompareOutputContext) void {
\\const std = @import("std");
\\const io = std.io;
\\const os = std.os;
- \\const allocator = std.testing.leak_count_allocator;
+ \\const allocator = std.testing.allocator;
\\
\\pub fn main() !void {
\\ var args_it = std.process.args();
diff --git a/test/compile_errors.zig b/test/compile_errors.zig
index 2f3f0a9f1f..341062e161 100644
--- a/test/compile_errors.zig
+++ b/test/compile_errors.zig
@@ -5765,7 +5765,7 @@ pub fn addCases(cases: *tests.CompileErrorContext) void {
\\
\\export fn entry() void {
\\ const a = MdNode.Header {
- \\ .text = MdText.init(&std.testing.leak_count_allocator),
+ \\ .text = MdText.init(&std.testing.allocator),
\\ .weight = HeaderWeight.H1,
\\ };
\\}
diff --git a/test/stage1/behavior/const_slice_child.zig b/test/stage1/behavior/const_slice_child.zig
index 7d969ae468..92e5121026 100644
--- a/test/stage1/behavior/const_slice_child.zig
+++ b/test/stage1/behavior/const_slice_child.zig
@@ -23,8 +23,8 @@ fn foo(args: [][]const u8) void {
}
fn bar(argc: usize) void {
- const args = testing.leak_count_allocator.alloc([]const u8, argc) catch unreachable;
- defer testing.leak_count_allocator.free(args);
+ const args = testing.allocator.alloc([]const u8, argc) catch unreachable;
+ defer testing.allocator.free(args);
for (args) |_, i| {
const ptr = argv[i];
args[i] = ptr[0..strlen(ptr)];
diff --git a/test/standalone/brace_expansion/main.zig b/test/standalone/brace_expansion/main.zig
index 33fe826d18..c6bfd02e79 100644
--- a/test/standalone/brace_expansion/main.zig
+++ b/test/standalone/brace_expansion/main.zig
@@ -201,7 +201,7 @@ pub fn main() !void {
}
test "invalid inputs" {
- global_allocator = std.testing.leak_count_allocator;
+ global_allocator = std.testing.allocator;
expectError("}ABC", error.InvalidInput);
expectError("{ABC", error.InvalidInput);
@@ -222,7 +222,7 @@ fn expectError(test_input: []const u8, expected_err: anyerror) void {
}
test "valid inputs" {
- global_allocator = std.testing.leak_count_allocator;
+ global_allocator = std.testing.allocator;
expectExpansion("{x,y,z}", "x y z");
expectExpansion("{A,B}{x,y}", "Ax Ay Bx By");
diff --git a/test/standalone/cat/main.zig b/test/standalone/cat/main.zig
index 10ec00d6e8..34439f9c24 100644
--- a/test/standalone/cat/main.zig
+++ b/test/standalone/cat/main.zig
@@ -4,7 +4,7 @@ const process = std.process;
const fs = std.fs;
const mem = std.mem;
const warn = std.debug.warn;
-const allocator = std.testing.leak_count_allocator;
+const allocator = std.testing.allocator;
pub fn main() !void {
var args_it = process.args();
diff --git a/test/standalone/empty_env/main.zig b/test/standalone/empty_env/main.zig
index dc2ac73738..f4ebf56136 100644
--- a/test/standalone/empty_env/main.zig
+++ b/test/standalone/empty_env/main.zig
@@ -1,6 +1,6 @@
const std = @import("std");
pub fn main() void {
- const env_map = std.process.getEnvMap(std.testing.leak_count_allocator) catch @panic("unable to get env map");
+ const env_map = std.process.getEnvMap(std.testing.allocator) catch @panic("unable to get env map");
std.testing.expect(env_map.count() == 0);
}
diff --git a/test/standalone/load_dynamic_library/main.zig b/test/standalone/load_dynamic_library/main.zig
index 5a2e55e937..70fc4986ac 100644
--- a/test/standalone/load_dynamic_library/main.zig
+++ b/test/standalone/load_dynamic_library/main.zig
@@ -1,8 +1,8 @@
const std = @import("std");
pub fn main() !void {
- const args = try std.process.argsAlloc(std.testing.leak_count_allocator);
- defer std.process.argsFree(std.testing.leak_count_allocator, args);
+ const args = try std.process.argsAlloc(std.testing.allocator);
+ defer std.process.argsFree(std.testing.allocator, args);
const dynlib_name = args[1];
From 898ef82d65af9a00aadc7ed63e5e1c0bc6c52bde Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 21:35:08 -0600
Subject: [PATCH 08/16] Patch in arena to cleanup leaks
---
test/standalone/brace_expansion/main.zig | 10 ++++++++--
1 file changed, 8 insertions(+), 2 deletions(-)
diff --git a/test/standalone/brace_expansion/main.zig b/test/standalone/brace_expansion/main.zig
index c6bfd02e79..e1d2ee6673 100644
--- a/test/standalone/brace_expansion/main.zig
+++ b/test/standalone/brace_expansion/main.zig
@@ -201,7 +201,10 @@ pub fn main() !void {
}
test "invalid inputs" {
- global_allocator = std.testing.allocator;
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
+ defer arena.deinit();
+
+ global_allocator = &arena.allocator;
expectError("}ABC", error.InvalidInput);
expectError("{ABC", error.InvalidInput);
@@ -222,7 +225,10 @@ fn expectError(test_input: []const u8, expected_err: anyerror) void {
}
test "valid inputs" {
- global_allocator = std.testing.allocator;
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
+ defer arena.deinit();
+
+ global_allocator = &arena.allocator;
expectExpansion("{x,y,z}", "x y z");
expectExpansion("{A,B}{x,y}", "Ax Ay Bx By");
From 837877ea373d295cb30c4c1f4ebc2ff561d52a79 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 21:39:08 -0600
Subject: [PATCH 09/16] Update docs to reflect new testing.allocator usage
---
doc/langref.html.in | 9 ++++-----
1 file changed, 4 insertions(+), 5 deletions(-)
diff --git a/doc/langref.html.in b/doc/langref.html.in
index 6789b25a63..47f13f9e05 100644
--- a/doc/langref.html.in
+++ b/doc/langref.html.in
@@ -5359,7 +5359,7 @@ const std = @import("std");
const assert = std.debug.assert;
test "turn HashMap into a set with void" {
- var map = std.HashMap(i32, void, hash_i32, eql_i32).init(std.debug.global_allocator);
+ var map = std.HashMap(i32, void, hash_i32, eql_i32).init(std.testing.allocator);
defer map.deinit();
_ = try map.put(1, {});
@@ -9281,9 +9281,8 @@ fn concat(allocator: *Allocator, a: []const u8, b: []const u8) ![]u8 {
In the above example, 100 bytes of stack memory are used to initialize a
{#syntax#}FixedBufferAllocator{#endsyntax#}, which is then passed to a function.
As a convenience there is a global {#syntax#}FixedBufferAllocator{#endsyntax#}
- available for quick tests at {#syntax#}std.debug.global_allocator{#endsyntax#},
- however it is deprecated and should be avoided in favor of directly using a
- {#syntax#}FixedBufferAllocator{#endsyntax#} as in the example above.
+ available for quick tests at {#syntax#}std.testing.allocator{#endsyntax#},
+ which will also do perform basic leak detection.
Currently Zig has no general purpose allocator, but there is
@@ -9341,7 +9340,7 @@ pub fn main() !void {
Are you writing a test, and you want to make sure {#syntax#}error.OutOfMemory{#endsyntax#}
- is handled correctly? In this case, use {#syntax#}std.debug.FailingAllocator{#endsyntax#}.
+ is handled correctly? In this case, use {#syntax#}std.testing.FailingAllocator{#endsyntax#}.
Finally, if none of the above apply, you need a general purpose allocator. Zig does not
From ad93ad3e6077cdaa7111c7aefbfb1305ae3edfb5 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 21:46:06 -0600
Subject: [PATCH 10/16] Fix errant reference to page_allocator
---
lib/std/process.zig | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/lib/std/process.zig b/lib/std/process.zig
index dd5e124654..ca5ca3a3bb 100644
--- a/lib/std/process.zig
+++ b/lib/std/process.zig
@@ -79,7 +79,7 @@ pub fn getEnvMap(allocator: *Allocator) !BufMap {
// https://github.com/WebAssembly/WASI/issues/27
var environ = try allocator.alloc(?[*:0]u8, environ_count + 1);
defer allocator.free(environ);
- var environ_buf = try std.heap.page_allocator.alloc(u8, environ_buf_size);
+ var environ_buf = try allocator.alloc(u8, environ_buf_size);
defer allocator.free(environ_buf);
const environ_get_ret = os.wasi.environ_get(environ.ptr, environ_buf.ptr);
From b7a236d68e042ffcf2b0642e951ecca33ed842a4 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 22:06:26 -0600
Subject: [PATCH 11/16] Convert a bunch of page_allocator to testing.allocator
---
lib/std/buf_map.zig | 1 +
lib/std/buf_set.zig | 2 +-
lib/std/build.zig | 3 +++
lib/std/hash/auto_hash.zig | 10 +++++-----
lib/std/hash_map.zig | 6 +++---
lib/std/os/test.zig | 4 ++--
lib/std/packed_int_array.zig | 4 ++--
lib/std/segmented_list.zig | 2 +-
src-self-hosted/c_tokenizer.zig | 4 ++--
test/stage1/behavior/async_fn.zig | 6 +++---
10 files changed, 23 insertions(+), 19 deletions(-)
diff --git a/lib/std/buf_map.zig b/lib/std/buf_map.zig
index 06f00608bb..df90025bae 100644
--- a/lib/std/buf_map.zig
+++ b/lib/std/buf_map.zig
@@ -83,6 +83,7 @@ pub const BufMap = struct {
};
test "BufMap" {
+ // TODO: uncomment and fix the leak
var bufmap = BufMap.init(std.heap.page_allocator);
defer bufmap.deinit();
diff --git a/lib/std/buf_set.zig b/lib/std/buf_set.zig
index 7ac8136e7d..89df0478ff 100644
--- a/lib/std/buf_set.zig
+++ b/lib/std/buf_set.zig
@@ -65,7 +65,7 @@ pub const BufSet = struct {
};
test "BufSet" {
- var bufset = BufSet.init(std.heap.page_allocator);
+ var bufset = BufSet.init(std.testing.allocator);
defer bufset.deinit();
try bufset.put("x");
diff --git a/lib/std/build.zig b/lib/std/build.zig
index d310f53c06..40dd4a3484 100644
--- a/lib/std/build.zig
+++ b/lib/std/build.zig
@@ -1059,7 +1059,10 @@ pub const Builder = struct {
};
test "builder.findProgram compiles" {
+ // TODO: uncomment and fix the leak
+ // const builder = try Builder.create(std.testing.allocator, "zig", "zig-cache", "zig-cache");
const builder = try Builder.create(std.heap.page_allocator, "zig", "zig-cache", "zig-cache");
+ defer builder.destroy();
_ = builder.findProgram(&[_][]const u8{}, &[_][]const u8{}) catch null;
}
diff --git a/lib/std/hash/auto_hash.zig b/lib/std/hash/auto_hash.zig
index e60a1c1055..adde71875e 100644
--- a/lib/std/hash/auto_hash.zig
+++ b/lib/std/hash/auto_hash.zig
@@ -234,8 +234,8 @@ test "hash pointer" {
test "hash slice shallow" {
// Allocate one array dynamically so that we're assured it is not merged
// with the other by the optimization passes.
- const array1 = try std.heap.page_allocator.create([6]u32);
- defer std.heap.page_allocator.destroy(array1);
+ const array1 = try std.testing.allocator.create([6]u32);
+ defer std.testing.allocator.destroy(array1);
array1.* = [_]u32{ 1, 2, 3, 4, 5, 6 };
const array2 = [_]u32{ 1, 2, 3, 4, 5, 6 };
const a = array1[0..];
@@ -250,8 +250,8 @@ test "hash slice shallow" {
test "hash slice deep" {
// Allocate one array dynamically so that we're assured it is not merged
// with the other by the optimization passes.
- const array1 = try std.heap.page_allocator.create([6]u32);
- defer std.heap.page_allocator.destroy(array1);
+ const array1 = try std.testing.allocator.create([6]u32);
+ defer std.testing.allocator.destroy(array1);
array1.* = [_]u32{ 1, 2, 3, 4, 5, 6 };
const array2 = [_]u32{ 1, 2, 3, 4, 5, 6 };
const a = array1[0..];
@@ -278,7 +278,7 @@ test "hash struct deep" {
}
};
- const allocator = std.heap.page_allocator;
+ const allocator = std.testing.allocator;
const foo = try Foo.init(allocator, 123, 1.0, true);
const bar = try Foo.init(allocator, 123, 1.0, true);
const baz = try Foo.init(allocator, 123, 1.0, false);
diff --git a/lib/std/hash_map.zig b/lib/std/hash_map.zig
index 57a2f58b92..80ca218df6 100644
--- a/lib/std/hash_map.zig
+++ b/lib/std/hash_map.zig
@@ -419,7 +419,7 @@ pub fn HashMap(comptime K: type, comptime V: type, comptime hash: fn (key: K) u3
}
test "basic hash map usage" {
- var map = AutoHashMap(i32, i32).init(std.heap.page_allocator);
+ var map = AutoHashMap(i32, i32).init(std.testing.allocator);
defer map.deinit();
testing.expect((try map.put(1, 11)) == null);
@@ -463,7 +463,7 @@ test "basic hash map usage" {
}
test "iterator hash map" {
- var reset_map = AutoHashMap(i32, i32).init(std.heap.page_allocator);
+ var reset_map = AutoHashMap(i32, i32).init(std.testing.allocator);
defer reset_map.deinit();
try reset_map.putNoClobber(1, 11);
@@ -509,7 +509,7 @@ test "iterator hash map" {
}
test "ensure capacity" {
- var map = AutoHashMap(i32, i32).init(std.heap.page_allocator);
+ var map = AutoHashMap(i32, i32).init(std.testing.allocator);
defer map.deinit();
try map.ensureCapacity(20);
diff --git a/lib/std/os/test.zig b/lib/std/os/test.zig
index 0d3be9e7e4..a872c03fde 100644
--- a/lib/std/os/test.zig
+++ b/lib/std/os/test.zig
@@ -235,8 +235,8 @@ test "pipe" {
}
test "argsAlloc" {
- var args = try std.process.argsAlloc(std.heap.page_allocator);
- std.process.argsFree(std.heap.page_allocator, args);
+ var args = try std.process.argsAlloc(std.testing.allocator);
+ std.process.argsFree(std.testing.allocator, args);
}
test "memfd_create" {
diff --git a/lib/std/packed_int_array.zig b/lib/std/packed_int_array.zig
index bc29e985b5..3dfa55e74b 100644
--- a/lib/std/packed_int_array.zig
+++ b/lib/std/packed_int_array.zig
@@ -604,7 +604,7 @@ test "PackedIntArray at end of available memory" {
p: PackedArray,
};
- const allocator = std.heap.page_allocator;
+ const allocator = std.testing.allocator;
var pad = try allocator.create(Padded);
defer allocator.destroy(pad);
@@ -618,7 +618,7 @@ test "PackedIntSlice at end of available memory" {
}
const PackedSlice = PackedIntSlice(u11);
- const allocator = std.heap.page_allocator;
+ const allocator = std.testing.allocator;
var page = try allocator.alloc(u8, std.mem.page_size);
defer allocator.free(page);
diff --git a/lib/std/segmented_list.zig b/lib/std/segmented_list.zig
index 80e4666b6a..d7d137175c 100644
--- a/lib/std/segmented_list.zig
+++ b/lib/std/segmented_list.zig
@@ -339,7 +339,7 @@ pub fn SegmentedList(comptime T: type, comptime prealloc_item_count: usize) type
}
test "std.SegmentedList" {
- var a = std.heap.page_allocator;
+ var a = std.testing.allocator;
try testSegmentedList(0, a);
try testSegmentedList(1, a);
diff --git a/src-self-hosted/c_tokenizer.zig b/src-self-hosted/c_tokenizer.zig
index 7cff969c35..7e085bcf78 100644
--- a/src-self-hosted/c_tokenizer.zig
+++ b/src-self-hosted/c_tokenizer.zig
@@ -866,7 +866,7 @@ fn expectTokens(tl: *TokenList, src: [*:0]const u8, expected: []CToken) void {
}
test "tokenize macro" {
- var tl = TokenList.init(std.heap.page_allocator);
+ var tl = TokenList.init(std.testing.allocator);
defer tl.deinit();
expectTokens(&tl, "TEST(0\n", &[_]CToken{
@@ -904,7 +904,7 @@ test "tokenize macro" {
}
test "tokenize macro ops" {
- var tl = TokenList.init(std.heap.page_allocator);
+ var tl = TokenList.init(std.testing.allocator);
defer tl.deinit();
expectTokens(&tl, "ADD A + B", &[_]CToken{
diff --git a/test/stage1/behavior/async_fn.zig b/test/stage1/behavior/async_fn.zig
index 1b21ba7eff..c7ea183b6f 100644
--- a/test/stage1/behavior/async_fn.zig
+++ b/test/stage1/behavior/async_fn.zig
@@ -410,8 +410,8 @@ test "heap allocated async function frame" {
var x: i32 = 42;
fn doTheTest() !void {
- const frame = try std.heap.page_allocator.create(@Frame(someFunc));
- defer std.heap.page_allocator.destroy(frame);
+ const frame = try std.testing.allocator.create(@Frame(someFunc));
+ defer std.testing.allocator.destroy(frame);
expect(x == 42);
frame.* = async someFunc();
@@ -671,7 +671,7 @@ fn testAsyncAwaitTypicalUsage(
}
fn amain() !void {
- const allocator = std.heap.page_allocator; // TODO once we have the debug allocator, use that, so that this can detect leaks
+ const allocator = std.testing.allocator;
var download_frame = async fetchUrl(allocator, "https://example.com/");
var download_awaited = false;
errdefer if (!download_awaited) {
From 5c8e85f3884c581b17c8df1f19852189fe7c9412 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Wed, 29 Jan 2020 22:17:15 -0600
Subject: [PATCH 12/16] Fix BufMap value leak
---
lib/std/buf_map.zig | 8 ++++----
1 file changed, 4 insertions(+), 4 deletions(-)
diff --git a/lib/std/buf_map.zig b/lib/std/buf_map.zig
index df90025bae..e8bc735b57 100644
--- a/lib/std/buf_map.zig
+++ b/lib/std/buf_map.zig
@@ -43,9 +43,10 @@ pub const BufMap = struct {
pub fn set(self: *BufMap, key: []const u8, value: []const u8) !void {
const value_copy = try self.copy(value);
errdefer self.free(value_copy);
- // Avoid copying key if it already exists
const get_or_put = try self.hash_map.getOrPut(key);
- if (!get_or_put.found_existing) {
+ if (get_or_put.found_existing) {
+ self.free(get_or_put.kv.value);
+ } else {
get_or_put.kv.key = self.copy(key) catch |err| {
_ = self.hash_map.remove(key);
return err;
@@ -83,8 +84,7 @@ pub const BufMap = struct {
};
test "BufMap" {
- // TODO: uncomment and fix the leak
- var bufmap = BufMap.init(std.heap.page_allocator);
+ var bufmap = BufMap.init(std.testing.allocator);
defer bufmap.deinit();
try bufmap.set("x", "1");
From 184128fd9e5bfcf9b9a1c1021cf1d8db03147c28 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Thu, 30 Jan 2020 00:14:17 -0600
Subject: [PATCH 13/16] Fix testing.allocator wiring
---
lib/std/special/test_runner.zig | 10 ++++------
lib/std/testing/leak_count_allocator.zig | 3 ++-
2 files changed, 6 insertions(+), 7 deletions(-)
diff --git a/lib/std/special/test_runner.zig b/lib/std/special/test_runner.zig
index 9a2d68a73d..07c46ee43c 100644
--- a/lib/std/special/test_runner.zig
+++ b/lib/std/special/test_runner.zig
@@ -14,12 +14,6 @@ pub fn main() anyerror!void {
for (test_fn_list) |test_fn, i| {
std.testing.base_allocator_instance.reset();
- defer {
- std.testing.allocator_instance.validate() catch |err| switch (err) {
- error.Leak => std.debug.panic("", .{}),
- else => std.debug.panic("error.{}", .{@errorName(err)}),
- };
- }
var test_node = root_node.start(test_fn.name, null);
test_node.activate();
@@ -30,6 +24,10 @@ pub fn main() anyerror!void {
if (test_fn.func()) |_| {
ok_count += 1;
test_node.end();
+ std.testing.allocator_instance.validate() catch |err| switch (err) {
+ error.Leak => std.debug.panic("", .{}),
+ else => std.debug.panic("error.{}", .{@errorName(err)}),
+ };
if (progress.terminal == null) std.debug.warn("OK\n", .{});
} else |err| switch (err) {
error.SkipZigTest => {
diff --git a/lib/std/testing/leak_count_allocator.zig b/lib/std/testing/leak_count_allocator.zig
index 6cdcfa2678..1d65b8c560 100644
--- a/lib/std/testing/leak_count_allocator.zig
+++ b/lib/std/testing/leak_count_allocator.zig
@@ -23,10 +23,11 @@ pub const LeakCountAllocator = struct {
fn realloc(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) ![]u8 {
const self = @fieldParentPtr(LeakCountAllocator, "allocator", allocator);
+ var data = try self.internal_allocator.reallocFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
if (old_mem.len == 0) {
self.count += 1;
}
- return self.internal_allocator.reallocFn(self.internal_allocator, old_mem, old_align, new_size, new_align);
+ return data;
}
fn shrink(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
From c1fb97aef6d0f7343e9226b2f19fbe9d315afa94 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Thu, 30 Jan 2020 00:20:11 -0600
Subject: [PATCH 14/16] Remove unnecessary allocator from mutex
---
lib/std/mutex.zig | 6 ------
1 file changed, 6 deletions(-)
diff --git a/lib/std/mutex.zig b/lib/std/mutex.zig
index 7fbe4fde18..30ac835f9f 100644
--- a/lib/std/mutex.zig
+++ b/lib/std/mutex.zig
@@ -306,12 +306,6 @@ const TestContext = struct {
};
test "std.Mutex" {
- var plenty_of_memory = try std.heap.page_allocator.alloc(u8, 300 * 1024);
- defer std.heap.page_allocator.free(plenty_of_memory);
-
- var fixed_buffer_allocator = std.heap.ThreadSafeFixedBufferAllocator.init(plenty_of_memory);
- var a = &fixed_buffer_allocator.allocator;
-
var mutex = Mutex.init();
defer mutex.deinit();
From 46d84a1b639eb8e2c11b1936e6897a6352802325 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Thu, 30 Jan 2020 00:26:54 -0600
Subject: [PATCH 15/16] Convert a few more page_allocator
---
lib/std/heap/logging_allocator.zig | 2 +-
lib/std/testing.zig | 2 +-
test/cli.zig | 2 +-
test/stage1/behavior/async_fn.zig | 4 ++--
4 files changed, 5 insertions(+), 5 deletions(-)
diff --git a/lib/std/heap/logging_allocator.zig b/lib/std/heap/logging_allocator.zig
index d1b7f4776d..7dce7bc20a 100644
--- a/lib/std/heap/logging_allocator.zig
+++ b/lib/std/heap/logging_allocator.zig
@@ -57,7 +57,7 @@ test "LoggingAllocator" {
var slice_stream = std.io.SliceOutStream.init(buf[0..]);
const stream = &slice_stream.stream;
- const allocator = &LoggingAllocator.init(std.heap.page_allocator, @ptrCast(*AnyErrorOutStream, stream)).allocator;
+ const allocator = &LoggingAllocator.init(std.testing.allocator, @ptrCast(*AnyErrorOutStream, stream)).allocator;
const ptr = try allocator.alloc(u8, 10);
allocator.free(ptr);
diff --git a/lib/std/testing.zig b/lib/std/testing.zig
index 4850f2e9b9..f8247b5a9d 100644
--- a/lib/std/testing.zig
+++ b/lib/std/testing.zig
@@ -12,7 +12,7 @@ pub var allocator_instance = LeakCountAllocator.init(&base_allocator_instance.al
pub const failing_allocator = &FailingAllocator.init(&base_allocator_instance.allocator, 0).allocator;
pub var base_allocator_instance = std.heap.ThreadSafeFixedBufferAllocator.init(allocator_mem[0..]);
-var allocator_mem: [100 * 1024]u8 = undefined;
+var allocator_mem: [512 * 1024]u8 = undefined;
/// This function is intended to be used only in tests. It prints diagnostics to stderr
/// and then aborts when actual_error_union is not expected_error.
diff --git a/test/cli.zig b/test/cli.zig
index 0527b5c923..b7d03d9e21 100644
--- a/test/cli.zig
+++ b/test/cli.zig
@@ -8,7 +8,7 @@ const ChildProcess = std.ChildProcess;
var a: *std.mem.Allocator = undefined;
pub fn main() !void {
- var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator);
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
defer arena.deinit();
var arg_it = process.args();
diff --git a/test/stage1/behavior/async_fn.zig b/test/stage1/behavior/async_fn.zig
index c7ea183b6f..fb740ead42 100644
--- a/test/stage1/behavior/async_fn.zig
+++ b/test/stage1/behavior/async_fn.zig
@@ -935,12 +935,12 @@ fn recursiveAsyncFunctionTest(comptime suspending_implementation: bool) type {
_ = async amain(&result);
return result;
} else {
- return fib(std.heap.page_allocator, 10) catch unreachable;
+ return fib(std.testing.allocator, 10) catch unreachable;
}
}
fn amain(result: *u32) void {
- var x = async fib(std.heap.page_allocator, 10);
+ var x = async fib(std.testing.allocator, 10);
result.* = (await x) catch unreachable;
}
};
From 9c196efa2afe0e337ac0b16bd1138e89393f6106 Mon Sep 17 00:00:00 2001
From: Benjamin Feng
Date: Thu, 30 Jan 2020 01:12:21 -0600
Subject: [PATCH 16/16] Add explicit error message for too many frees
---
lib/std/testing/leak_count_allocator.zig | 3 +++
1 file changed, 3 insertions(+)
diff --git a/lib/std/testing/leak_count_allocator.zig b/lib/std/testing/leak_count_allocator.zig
index 1d65b8c560..65244e529b 100644
--- a/lib/std/testing/leak_count_allocator.zig
+++ b/lib/std/testing/leak_count_allocator.zig
@@ -33,6 +33,9 @@ pub const LeakCountAllocator = struct {
fn shrink(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
const self = @fieldParentPtr(LeakCountAllocator, "allocator", allocator);
if (new_size == 0) {
+ if (self.count == 0) {
+ std.debug.panic("error - too many calls to free, most likely double free", .{});
+ }
self.count -= 1;
}
return self.internal_allocator.shrinkFn(self.internal_allocator, old_mem, old_align, new_size, new_align);