Reworked io functions to take a configurable reader or writer
This commit is contained in:
parent
35c2f5e626
commit
aa56929ff7
4 changed files with 68 additions and 61 deletions
|
@ -13,19 +13,19 @@ pub const Interpreter = struct {
|
||||||
self.field.deinit();
|
self.field.deinit();
|
||||||
self.allocator.destroy(self);
|
self.allocator.destroy(self);
|
||||||
}
|
}
|
||||||
pub fn init(allocator: std.mem.Allocator, reader: anytype, ioFunctions: io.Functions, args: []const []const u8) !*Interpreter {
|
pub fn init(allocator: std.mem.Allocator, fileReader: anytype, args: []const []const u8) !*Interpreter {
|
||||||
var i = try allocator.create(Interpreter);
|
var i = try allocator.create(Interpreter);
|
||||||
errdefer allocator.destroy(i);
|
errdefer allocator.destroy(i);
|
||||||
i.allocator = allocator;
|
i.allocator = allocator;
|
||||||
i.field = try field.Field.init_from_reader(allocator, reader);
|
i.field = try field.Field.init_from_reader(allocator, fileReader);
|
||||||
errdefer i.field.deinit();
|
errdefer i.field.deinit();
|
||||||
i.pointer = try pointer.Pointer.init(std.testing.allocator, i.field, ioFunctions, args);
|
i.pointer = try pointer.Pointer.init(std.testing.allocator, i.field, args);
|
||||||
errdefer i.pointer.deinit();
|
errdefer i.pointer.deinit();
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
pub fn run(self: *Interpreter) !i64 {
|
pub fn run(self: *Interpreter, ioContext: anytype) !i64 {
|
||||||
while (true) {
|
while (true) {
|
||||||
if (try self.pointer.exec()) |ret| {
|
if (try self.pointer.exec(ioContext)) |ret| {
|
||||||
if (ret.code) |code| {
|
if (ret.code) |code| {
|
||||||
return code;
|
return code;
|
||||||
} else {
|
} else {
|
||||||
|
|
65
src/io.zig
65
src/io.zig
|
@ -1,42 +1,37 @@
|
||||||
const std = @import("std");
|
const std = @import("std");
|
||||||
|
|
||||||
pub const IOErrors = error{
|
pub fn Context(readerType: anytype, writerType: anytype) type {
|
||||||
IOError,
|
return struct {
|
||||||
NotImplemented,
|
reader: readerType,
|
||||||
};
|
writer: writerType,
|
||||||
|
pub fn characterInput(self: @This()) !i64 {
|
||||||
pub const Functions = struct {
|
var buffer = [_]u8{0};
|
||||||
characterInput: fn () IOErrors!i64,
|
var n = try self.reader.read(buffer[0..]);
|
||||||
decimalInput: fn () IOErrors!i64,
|
if (n == 1) {
|
||||||
characterOutput: fn (i64) IOErrors!void,
|
return buffer[0];
|
||||||
decimalOutput: fn (i64) IOErrors!void,
|
}
|
||||||
};
|
return error.IOError;
|
||||||
|
}
|
||||||
pub const defaultFunctions = Functions{
|
pub fn decimalInput(self: @This()) !i64 {
|
||||||
.characterInput = characterInput,
|
_ = self;
|
||||||
.decimalInput = decimalInput,
|
return error.NotImplemented;
|
||||||
.characterOutput = characterOutput,
|
}
|
||||||
.decimalOutput = decimalOutput,
|
pub fn characterOutput(self: @This(), v: i64) !void {
|
||||||
};
|
try self.writer.print("{c}", .{@intCast(u8, v)});
|
||||||
|
return;
|
||||||
fn characterInput() IOErrors!i64 {
|
}
|
||||||
// TODO
|
pub fn decimalOutput(self: @This(), v: i64) !void {
|
||||||
return error.NotImplemented;
|
try self.writer.print("{d}", .{v});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
fn decimalInput() IOErrors!i64 {
|
pub fn context(reader: anytype, writer: anytype) Context(@TypeOf(reader), @TypeOf(writer)) {
|
||||||
// TODO
|
return .{
|
||||||
return error.NotImplemented;
|
.reader = reader,
|
||||||
}
|
.writer = writer,
|
||||||
|
};
|
||||||
fn characterOutput(v: i64) IOErrors!void {
|
|
||||||
std.debug.print("{c}", .{@intCast(u8, v)});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn decimalOutput(v: i64) IOErrors!void {
|
|
||||||
std.debug.print("{d}", .{v});
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
test "all" {
|
test "all" {
|
||||||
|
|
|
@ -15,10 +15,10 @@ pub fn main() anyerror!void {
|
||||||
var file = try std.fs.cwd().openFile("mycology/sanity.bf", .{});
|
var file = try std.fs.cwd().openFile("mycology/sanity.bf", .{});
|
||||||
defer file.close();
|
defer file.close();
|
||||||
|
|
||||||
var i = try interpreter.Interpreter.init(gpa.allocator(), file.reader(), io.defaultFunctions, args);
|
var i = try interpreter.Interpreter.init(gpa.allocator(), file.reader(), args);
|
||||||
defer i.deinit();
|
defer i.deinit();
|
||||||
|
|
||||||
std.os.exit(@intCast(u8, try i.run()));
|
std.os.exit(@intCast(u8, try i.run(io.context(std.io.getStdIn().reader(), std.io.getStdOut().writer()))));
|
||||||
}
|
}
|
||||||
|
|
||||||
test "all" {
|
test "all" {
|
||||||
|
@ -28,7 +28,7 @@ test "sanity" {
|
||||||
var file = try std.fs.cwd().openFile("mycology/sanity.bf", .{});
|
var file = try std.fs.cwd().openFile("mycology/sanity.bf", .{});
|
||||||
defer file.close();
|
defer file.close();
|
||||||
const args = [_][]const u8{"sanity"};
|
const args = [_][]const u8{"sanity"};
|
||||||
var i = try interpreter.Interpreter.init(std.testing.allocator, file.reader(), io.defaultFunctions, args[0..]);
|
var i = try interpreter.Interpreter.init(std.testing.allocator, file.reader(), args[0..]);
|
||||||
defer i.deinit();
|
defer i.deinit();
|
||||||
try std.testing.expectEqual(try i.run(), 0);
|
try std.testing.expectEqual(try i.run(io.context(std.io.getStdIn().reader(), std.io.getStdOut().writer())), 0);
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,6 @@ pub const Pointer = struct {
|
||||||
stringMode: bool = false, // string mode flags
|
stringMode: bool = false, // string mode flags
|
||||||
lastCharWasSpace: bool = false,
|
lastCharWasSpace: bool = false,
|
||||||
ss: *stackStack.StackStack,
|
ss: *stackStack.StackStack,
|
||||||
ioFunctions: io.Functions,
|
|
||||||
argv: []const []const u8,
|
argv: []const []const u8,
|
||||||
rand: *std.rand.Random,
|
rand: *std.rand.Random,
|
||||||
|
|
||||||
|
@ -31,7 +30,7 @@ pub const Pointer = struct {
|
||||||
self.ss.deinit();
|
self.ss.deinit();
|
||||||
self.allocator.destroy(self);
|
self.allocator.destroy(self);
|
||||||
}
|
}
|
||||||
fn eval(p: *Pointer, c: i64) pointerErrorType!?pointerReturn {
|
fn eval(p: *Pointer, ioContext: anytype, c: i64) pointerErrorType!?pointerReturn {
|
||||||
// Returns non nil if the pointer terminated, and a return code if
|
// Returns non nil if the pointer terminated, and a return code if
|
||||||
// the program should terminate completely
|
// the program should terminate completely
|
||||||
switch (c) {
|
switch (c) {
|
||||||
|
@ -67,9 +66,9 @@ pub const Pointer = struct {
|
||||||
p.x = x;
|
p.x = x;
|
||||||
p.y = y;
|
p.y = y;
|
||||||
if (v != ' ' and v != ';') {
|
if (v != ' ' and v != ';') {
|
||||||
if (v == 'q' or v == '@') return try p.eval(v);
|
if (v == 'q' or v == '@') return try p.eval(ioContext, v);
|
||||||
var i: usize = 0;
|
var i: usize = 0;
|
||||||
while (i < n) : (i += 1) _ = try p.eval(v);
|
while (i < n) : (i += 1) _ = try p.eval(ioContext, v);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -168,11 +167,25 @@ pub const Pointer = struct {
|
||||||
const n = p.ss.toss.pop();
|
const n = p.ss.toss.pop();
|
||||||
try p.field.set(v[0] + p.sox, v[1] + p.soy, n);
|
try p.field.set(v[0] + p.sox, v[1] + p.soy, n);
|
||||||
},
|
},
|
||||||
'.' => try p.ioFunctions.decimalOutput(p.ss.toss.pop()),
|
'.' => ioContext.decimalOutput(p.ss.toss.pop()) catch {
|
||||||
',' => try p.ioFunctions.characterOutput(p.ss.toss.pop()),
|
p.reverse();
|
||||||
'&' => try p.ss.toss.push(try p.ioFunctions.decimalInput()),
|
return null;
|
||||||
'~' => try p.ss.toss.push(try p.ioFunctions.characterInput()),
|
},
|
||||||
// TODO
|
',' => ioContext.characterOutput(p.ss.toss.pop()) catch p.reverse(),
|
||||||
|
'&' => {
|
||||||
|
const n = ioContext.decimalInput() catch {
|
||||||
|
p.reverse();
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
p.ss.toss.push(n) catch p.reverse();
|
||||||
|
},
|
||||||
|
'~' => {
|
||||||
|
const n = ioContext.characterInput() catch {
|
||||||
|
p.reverse();
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
p.ss.toss.push(n) catch p.reverse();
|
||||||
|
},
|
||||||
'y' => return error.NotImplemented,
|
'y' => return error.NotImplemented,
|
||||||
'(' => {
|
'(' => {
|
||||||
const n = p.ss.toss.pop();
|
const n = p.ss.toss.pop();
|
||||||
|
@ -208,7 +221,7 @@ pub const Pointer = struct {
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
pub fn exec(self: *Pointer) !?pointerReturn {
|
pub fn exec(self: *Pointer, ioContext: anytype) !?pointerReturn {
|
||||||
// Advances to the next instruction of the field and executes it
|
// Advances to the next instruction of the field and executes it
|
||||||
// Returns non nil if the pointer terminated, and a return code if
|
// Returns non nil if the pointer terminated, and a return code if
|
||||||
// the program should terminate completely
|
// the program should terminate completely
|
||||||
|
@ -233,18 +246,17 @@ pub const Pointer = struct {
|
||||||
if (c == ';') jumpingMode = !jumpingMode;
|
if (c == ';') jumpingMode = !jumpingMode;
|
||||||
c = self.stepAndGet();
|
c = self.stepAndGet();
|
||||||
}
|
}
|
||||||
result = try self.eval(c);
|
result = try self.eval(ioContext, c);
|
||||||
}
|
}
|
||||||
self.step();
|
self.step();
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
pub fn init(allocator: std.mem.Allocator, f: *field.Field, ioFunctions: io.Functions, argv: []const []const u8) !*Pointer {
|
pub fn init(allocator: std.mem.Allocator, f: *field.Field, argv: []const []const u8) !*Pointer {
|
||||||
var p = try allocator.create(Pointer);
|
var p = try allocator.create(Pointer);
|
||||||
errdefer allocator.destroy(p);
|
errdefer allocator.destroy(p);
|
||||||
p.allocator = allocator;
|
p.allocator = allocator;
|
||||||
p.field = f;
|
p.field = f;
|
||||||
p.ss = try stackStack.StackStack.init(allocator);
|
p.ss = try stackStack.StackStack.init(allocator);
|
||||||
p.ioFunctions = ioFunctions;
|
|
||||||
p.argv = argv;
|
p.argv = argv;
|
||||||
p.x = 0;
|
p.x = 0;
|
||||||
p.y = 0;
|
p.y = 0;
|
||||||
|
@ -328,16 +340,16 @@ test "minimal" {
|
||||||
var f = try field.Field.init_from_reader(std.testing.allocator, minimal);
|
var f = try field.Field.init_from_reader(std.testing.allocator, minimal);
|
||||||
defer f.deinit();
|
defer f.deinit();
|
||||||
const argv = [_][]const u8{"minimal"};
|
const argv = [_][]const u8{"minimal"};
|
||||||
var p = try Pointer.init(std.testing.allocator, f, io.defaultFunctions, argv[0..]);
|
var p = try Pointer.init(std.testing.allocator, f, argv[0..]);
|
||||||
defer p.deinit();
|
defer p.deinit();
|
||||||
try std.testing.expectEqual(p.exec(), pointerReturn{});
|
try std.testing.expectEqual(p.exec(io.context(std.io.getStdIn().reader(), std.io.getStdOut().writer())), pointerReturn{});
|
||||||
}
|
}
|
||||||
test "almost minimal" {
|
test "almost minimal" {
|
||||||
const minimal = std.io.fixedBufferStream(" @").reader();
|
const minimal = std.io.fixedBufferStream(" @").reader();
|
||||||
var f = try field.Field.init_from_reader(std.testing.allocator, minimal);
|
var f = try field.Field.init_from_reader(std.testing.allocator, minimal);
|
||||||
defer f.deinit();
|
defer f.deinit();
|
||||||
const argv = [_][]const u8{"minimal"};
|
const argv = [_][]const u8{"minimal"};
|
||||||
var p = try Pointer.init(std.testing.allocator, f, io.defaultFunctions, argv[0..]);
|
var p = try Pointer.init(std.testing.allocator, f, argv[0..]);
|
||||||
defer p.deinit();
|
defer p.deinit();
|
||||||
try std.testing.expectEqual(p.exec(), pointerReturn{});
|
try std.testing.expectEqual(p.exec(io.context(std.io.getStdIn().reader(), std.io.getStdOut().writer())), pointerReturn{});
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue