const std = @import("std"); const builtin = @import("builtin"); const str = @import("glue/str.zig"); const RocStr = str.RocStr; const testing = std.testing; const expectEqual = testing.expectEqual; const expect = testing.expect; const maxInt = std.math.maxInt; const mem = std.mem; const Allocator = mem.Allocator; extern fn roc__main_for_host_1_exposed() void; const Align = 2 * @alignOf(usize); extern fn malloc(size: usize) callconv(.C) ?*align(Align) anyopaque; extern fn realloc(c_ptr: [*]align(Align) u8, size: usize) callconv(.C) ?*anyopaque; extern fn free(c_ptr: [*]align(Align) u8) callconv(.C) void; extern fn memcpy(dst: [*]u8, src: [*]u8, size: usize) callconv(.C) void; extern fn memset(dst: [*]u8, value: i32, size: usize) callconv(.C) void; const DEBUG: bool = false; export fn roc_alloc(size: usize, alignment: u32) callconv(.C) ?*anyopaque { if (DEBUG) { const ptr = malloc(size); const stdout = std.io.getStdOut().writer(); stdout.print("alloc: {d} (alignment {d}, size {d})\n", .{ ptr, alignment, size }) catch unreachable; return ptr; } else { return malloc(size); } } export fn roc_realloc(c_ptr: *anyopaque, new_size: usize, old_size: usize, alignment: u32) callconv(.C) ?*anyopaque { if (DEBUG) { const stdout = std.io.getStdOut().writer(); stdout.print("realloc: {d} (alignment {d}, old_size {d})\n", .{ c_ptr, alignment, old_size }) catch unreachable; } return realloc(@as([*]align(Align) u8, @alignCast(@ptrCast(c_ptr))), new_size); } export fn roc_dealloc(c_ptr: *anyopaque, alignment: u32) callconv(.C) void { if (DEBUG) { const stdout = std.io.getStdOut().writer(); stdout.print("dealloc: {d} (alignment {d})\n", .{ c_ptr, alignment }) catch unreachable; } free(@as([*]align(Align) u8, @alignCast(@ptrCast(c_ptr)))); } export fn roc_panic(msg: *RocStr, tag_id: u32) callconv(.C) void { const stderr = std.io.getStdErr().writer(); switch (tag_id) { 0 => { stderr.print("Roc standard library crashed with message\n\n {s}\n\nShutting down\n", .{msg.asSlice()}) catch unreachable; }, 1 => { stderr.print("Application crashed with message\n\n {s}\n\nShutting down\n", .{msg.asSlice()}) catch unreachable; }, else => unreachable, } std.process.exit(1); } export fn roc_dbg(loc: *RocStr, msg: *RocStr, src: *RocStr) callconv(.C) void { const stderr = std.io.getStdErr().writer(); stderr.print("[{s}] {s} = {s}\n", .{ loc.asSlice(), src.asSlice(), msg.asSlice() }) catch unreachable; } export fn roc_memset(dst: [*]u8, value: i32, size: usize) callconv(.C) void { return memset(dst, value, size); } extern fn kill(pid: c_int, sig: c_int) c_int; extern fn shm_open(name: *const i8, oflag: c_int, mode: c_uint) c_int; extern fn mmap(addr: ?*anyopaque, length: c_uint, prot: c_int, flags: c_int, fd: c_int, offset: c_uint) *anyopaque; extern fn getppid() c_int; fn roc_getppid() callconv(.C) c_int { return getppid(); } fn roc_getppid_windows_stub() callconv(.C) c_int { return 0; } fn roc_shm_open(name: *const i8, oflag: c_int, mode: c_uint) callconv(.C) c_int { return shm_open(name, oflag, mode); } fn roc_mmap(addr: ?*anyopaque, length: c_uint, prot: c_int, flags: c_int, fd: c_int, offset: c_uint) callconv(.C) *anyopaque { return mmap(addr, length, prot, flags, fd, offset); } comptime { if (builtin.os.tag == .macos or builtin.os.tag == .linux) { @export(roc_getppid, .{ .name = "roc_getppid", .linkage = .strong }); @export(roc_mmap, .{ .name = "roc_mmap", .linkage = .strong }); @export(roc_shm_open, .{ .name = "roc_shm_open", .linkage = .strong }); } if (builtin.os.tag == .windows) { @export(roc_getppid_windows_stub, .{ .name = "roc_getppid", .linkage = .strong }); } } const Unit = extern struct {}; pub export fn main() u8 { roc__main_for_host_1_exposed(); return 0; } pub export fn roc_fx_put_int(int: i64) i64 { const stdout = std.io.getStdOut().writer(); stdout.print("{d}", .{int}) catch unreachable; stdout.print("\n", .{}) catch unreachable; return 0; } export fn roc_fx_put_line(rocPath: *str.RocStr) callconv(.C) void { const stdout = std.io.getStdOut().writer(); for (rocPath.asSlice()) |char| { stdout.print("{c}", .{char}) catch unreachable; } stdout.print("\n", .{}) catch unreachable; } const GetInt = extern struct { value: i64, is_error: bool, }; comptime { if (@sizeOf(usize) == 8) { @export(roc_fx_get_int_64bit, .{ .name = "roc_fx_get_int" }); } else { @export(roc_fx_get_int_32bit, .{ .name = "roc_fx_get_int" }); } } fn roc_fx_get_int_64bit() callconv(.C) GetInt { if (roc_fx_get_int_help()) |value| { const get_int = GetInt{ .is_error = false, .value = value }; return get_int; } else |err| switch (err) { error.InvalidCharacter => { return GetInt{ .is_error = true, .value = 0 }; }, else => { return GetInt{ .is_error = true, .value = 0 }; }, } return 0; } fn roc_fx_get_int_32bit(output: *GetInt) callconv(.C) void { if (roc_fx_get_int_help()) |value| { const get_int = GetInt{ .is_error = false, .value = value }; output.* = get_int; } else |err| switch (err) { error.InvalidCharacter => { output.* = GetInt{ .is_error = true, .value = 0 }; }, else => { output.* = GetInt{ .is_error = true, .value = 0 }; }, } return; } fn roc_fx_get_int_help() !i64 { const stdout = std.io.getStdOut().writer(); stdout.print("Please enter an integer\n", .{}) catch unreachable; const stdin = std.io.getStdIn().reader(); var buf: [40]u8 = undefined; // make sure to strip `\r` on windows const raw_line: []u8 = (try stdin.readUntilDelimiterOrEof(&buf, '\n')) orelse ""; const line = std.mem.trimRight(u8, raw_line, &std.ascii.whitespace); return std.fmt.parseInt(i64, line, 10); }