aboutsummaryrefslogtreecommitdiff
path: root/src/main.zig
blob: 367edf8d4ef169f73db013a405bc65035f327b09 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
const std = @import("std");
const codegen = @import("./codegen.zig");
const target = @import("builtin").target;

pub const parse = @import("./parse.zig");
pub const Lexer = @import("./Lexer.zig");
pub const compile = @import("./compile.zig");

pub fn main() !void {
    var arena: std.heap.ArenaAllocator = .init(std.heap.smp_allocator);
    defer arena.deinit();
    const allocator = arena.allocator();

    var args = std.process.args();
    _ = args.next();

    const out_path = args.next();
    const out_file = if (out_path) |path|
        try std.fs.cwd().createFile(path, .{ .mode = 0o777 })
    else
        std.io.getStdOut();
    const run = if (args.next()) |arg| std.mem.eql(u8, arg, "run") else false;

    // var br = std.io.bufferedReader(std.io.getStdIn().reader());
    // const stdin = br.reader();
    //
    // var line: std.ArrayList(u8) = .init(alloc);
    // defer line.deinit();
    // while (true) {
    //     try stdin.streamUntilDelimiter(line.writer(), '\n', null);
    //
    //     const lexer = Lexer{.source = line};
    //
    //     try stdout.print("{s}\n", .{line.items});
    // }

    const source =
        \\{
        \\    let x = 10
        \\    let y = 0
        \\    print(x > y)
        \\    while y < 3 {
        \\        x = x + x
        \\        y = y + 1
        \\    }
        \\    if x {
        \\        # let x = read_int(0)
        \\        # print(18446744073709551615)
        \\        x = x + x
        \\    } else {
        \\        x = 69
        \\    }
        \\    print(x)
        \\}
    ;
    var lexer: Lexer = .{ .source = source };
    std.debug.print("Tokens:\n", .{});
    while (true) {
        const token = lexer.next();
        std.debug.print("    {}\n", .{token});
        if (token.type == .eof) break;
    }
    lexer = .{ .source = source };
    const ast = try parse.block(allocator, &lexer);
    std.debug.print("Parse tree:\n{}\n", .{parse.fmt(ast, source, 0)});
    if (lexer.peek().type != .eof) {
        std.debug.print("Unexpected token {}, expected end of file\n", .{lexer.next()});
    }
    const procedure = try compile.compile(allocator, source, ast);
    std.debug.print("Bytecode instructions:\n{}", .{procedure});
    const elf = try codegen.create_elf(allocator, procedure);
    try out_file.writer().writeAll(elf);
    std.debug.print("Run output:\n", .{});
    if (run) {
        out_file.close();

        const err = std.process.execv(
            allocator,
            if (target.cpu.arch == .riscv64 and target.os.tag == .linux)
                &.{out_path}
            else
                &.{ "qemu-riscv64", out_path.? },
        );
        std.debug.print("{}\n{any}\n", .{err, @errorReturnTrace()});
    }
}

fn HashMapFormatter(HashMap: type) type {
    return std.fmt.Formatter(struct {
        fn formatHashMap(
            hash_map: HashMap,
            comptime fmt: []const u8,
            options: std.fmt.FormatOptions,
            writer: anytype,
        ) !void {
            _ = fmt;
            _ = options;
            try writer.writeAll("{");
            var it = hash_map.iterator();
            var first = true;
            while (it.next()) |kv| {
                try writer.print(
                    "{s} {" ++ (if (@TypeOf(kv.key_ptr.*) == []const u8) "s" else "any") ++ "}: {any}",
                    .{ if (first) "" else ",", kv.key_ptr.*, kv.value_ptr.* },
                );
                first = false;
            }
            try writer.writeAll(if (first) "}" else " }");
        }
    }.formatHashMap);
}

pub fn fmtHashMap(hash_map: anytype) HashMapFormatter(@TypeOf(hash_map)) {
    return .{ .data = hash_map };
}