mirror of
https://github.com/ghostty-org/ghostty.git
synced 2026-06-08 12:54:28 +00:00
54 lines
1.7 KiB
Zig
54 lines
1.7 KiB
Zig
const std = @import("std");
|
|
const ghostty_vt = @import("ghostty-vt");
|
|
const mem = @import("mem.zig");
|
|
const Terminal = ghostty_vt.Terminal;
|
|
const ReadonlyStream = ghostty_vt.ReadonlyStream;
|
|
|
|
/// Use a single global allocator for simplicity and to avoid heap
|
|
/// allocation overhead in the fuzzer. The allocator is backed by a fixed
|
|
/// buffer, and every fuzz input resets the bump pointer to the start.
|
|
var fuzz_alloc: mem.FuzzAllocator(64 * 1024 * 1024) = .{};
|
|
|
|
pub export fn zig_fuzz_init() callconv(.c) void {
|
|
fuzz_alloc.init();
|
|
}
|
|
|
|
pub export fn zig_fuzz_test(
|
|
buf: [*]const u8,
|
|
len: usize,
|
|
) callconv(.c) void {
|
|
// Do not test zero-length input paths.
|
|
if (len == 0) return;
|
|
|
|
fuzz_alloc.reset();
|
|
const alloc = fuzz_alloc.allocator();
|
|
const input = buf[0..len];
|
|
|
|
// Allocate a terminal; if we run out of fixed-buffer space just
|
|
// skip this input (not a bug, just a very large allocation).
|
|
var t = Terminal.init(alloc, .{
|
|
.cols = 80,
|
|
.rows = 24,
|
|
.max_scrollback = 100,
|
|
}) catch return;
|
|
defer t.deinit(alloc);
|
|
|
|
var stream: ReadonlyStream = t.vtStream();
|
|
defer stream.deinit();
|
|
|
|
// Use the first byte to decide between the scalar and slice paths
|
|
// so both code paths get exercised by the fuzzer.
|
|
const mode = input[0];
|
|
const data = input[1..];
|
|
|
|
if (mode & 1 == 0) {
|
|
// Slice path — exercises SIMD fast-path if enabled
|
|
stream.nextSlice(data) catch |err|
|
|
std.debug.panic("nextSlice: {}", .{err});
|
|
} else {
|
|
// Scalar path — exercises byte-at-a-time UTF-8 decoding
|
|
for (data) |byte| _ = stream.next(byte) catch |err|
|
|
std.debug.panic("next: {}", .{err});
|
|
}
|
|
}
|