git.s-ol.nu glsl-view / master src / main.zig
master

Tree @master (Download .tar.gz)

main.zig @masterraw · history · blame

const std = @import("std");
const fs = std.fs;
const debug = std.debug;
const panic = debug.panic;
const process = std.process;
const c = @import("c.zig");
const debug_gl = @import("debug_gl.zig");
const cfg = @import("config.zig");
const out = @import("output.zig");
const gl = @import("gl.zig");
const ctrl = @import("control.zig");
const c_allocator = std.heap.c_allocator;

var window: *c.GLFWwindow = undefined;

fn errorCallback(err: c_int, description: [*c]const u8) callconv(.C) void {
    panic("Error {}: {s}\n", .{ err, description });
}

pub fn main() !void {
    var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator);
    defer arena.deinit();

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

    const filename = args.next() orelse "config.yaml";
    var config = try cfg.Config.parse(&arena.allocator(), filename);

    _ = c.glfwSetErrorCallback(errorCallback);

    if (c.glfwInit() == c.GL_FALSE) {
        panic("GLFW init failure\n", .{});
    }
    defer c.glfwTerminate();

    var monitor_count: c_int = 0;
    const monitors = c.glfwGetMonitors(&monitor_count);
    for (monitors[0..@as(usize, @intCast(monitor_count))], 0..) |monitor, i| {
        debug.print(
            "monitor {}: '{s}'\n",
            .{ i, @as([*:0]const u8, @ptrCast(c.glfwGetMonitorName(monitor))) },
        );
    }

    c.glfwWindowHint(c.GLFW_CONTEXT_VERSION_MAJOR, 4);
    c.glfwWindowHint(c.GLFW_CONTEXT_VERSION_MINOR, 2);
    c.glfwWindowHint(c.GLFW_OPENGL_FORWARD_COMPAT, c.GL_TRUE);
    c.glfwWindowHint(c.GLFW_OPENGL_DEBUG_CONTEXT, debug_gl.is_on);
    c.glfwWindowHint(c.GLFW_OPENGL_PROFILE, c.GLFW_OPENGL_CORE_PROFILE);
    c.glfwWindowHint(c.GLFW_DEPTH_BITS, 0);
    c.glfwWindowHint(c.GLFW_STENCIL_BITS, 0);
    c.glfwWindowHint(c.GLFW_VISIBLE, c.GLFW_FALSE);

    window = c.glfwCreateWindow(config.width, config.height, "glsl-view", null, null) orelse {
        panic("unable to create window\n", .{});
    };
    defer c.glfwDestroyWindow(window);

    c.glfwMakeContextCurrent(window);
    c.glfwSwapInterval(1);

    var constants = try gl.Constants.create(window, &config);
    defer constants.destroy();

    var outputs = try std.ArrayList(*out.Output).initCapacity(arena.allocator(), config.outputs.len);
    defer outputs.deinit();
    for (config.outputs) |output_config| {
        try outputs.append(out.Output.create(arena.allocator(), output_config, &constants));
    }
    defer for (outputs.items) |output| {
        output.destroy(output);
    };

    c.glfwMakeContextCurrent(window);
    c.glClearColor(0.0, 0.0, 0.0, 1.0);

    debug_gl.assertNoError();
    debug_gl.init();

    const start_time = c.glfwGetTime();
    var prev_time = start_time;

    constants.normalized_quad.bind(0);

    var main_program = try gl.ShaderProgram.create(
        \\#version 330 core
        \\
        \\layout(location = 0) in vec2 position;
        \\out vec2 uv;
        \\
        \\void main() {
        \\  uv = position / 2.0 + 0.5;
        \\  gl_Position = vec4(position, 0, 1);
        \\}
    ,
        \\#version 330 core
        \\
        \\in vec2 uv;
        \\out vec4 color;
        \\
        \\void main() {
        \\  color = vec4(0,0,0,1);
        \\}
    );
    defer main_program.destroy();

    var fbo = try gl.FramebufferObject.create(config.width, config.height);
    defer fbo.destroy();

    const shader_file = try config.project_root.openFile(config.fragment, .{});
    var last_stat = try shader_file.stat();
    try reloadShader(&config, &main_program, config.fragment, last_stat.size);

    var cache = gl.UniformCache.init(std.heap.c_allocator, &main_program);
    defer cache.deinit();

    const control = try ctrl.ControlServer.init(arena.allocator(), config.osc, &cache);
    defer control.destroy();

    while (c.glfwWindowShouldClose(window) == c.GL_FALSE) {
        c.glfwMakeContextCurrent(window);
        c.glClear(c.GL_COLOR_BUFFER_BIT);

        const now_time = c.glfwGetTime();
        prev_time = now_time;

        const stat = try shader_file.stat();
        if (stat.mtime > last_stat.mtime) {
            try reloadShader(&config, &main_program, config.fragment, stat.size);
            try cache.refresh();
            last_stat = stat;
        }

        control.update();

        fbo.bind();
        c.glClear(c.GL_COLOR_BUFFER_BIT);

        main_program.bind();
        constants.normalized_quad.draw();
        fbo.unbind();

        for (outputs.items, 0..) |output, i| {
            const close = output.update(output, fbo.texture_id);
            if (close) {
                const removed = outputs.swapRemove(i);
                removed.destroy(removed);
                break;
            }
        }

        if (outputs.items.len == 0)
            break;

        c.glfwPollEvents();
    }
}

fn escape_dquote(str: []const u8) []const u8 {
    const state = struct {
        var buf: [1024]u8 = undefined;
    };

    const len = std.mem.replacementSize(u8, str, "\"", "\\\"");
    _ = std.mem.replace(u8, str, "\"", "\\\"", state.buf[0..len]);
    return state.buf[0..len];
}

// given a file and the directory it is in, resolve references
fn loadFile(config: *const cfg.Config, writer: anytype, dir: []const u8, filename: []const u8) !void {
    const file_dir = try fs.path.resolve(c_allocator, &[_][]const u8{
        dir,
        fs.path.dirname(filename) orelse "",
    });
    defer c_allocator.free(file_dir);

    const file_path = try fs.path.resolve(c_allocator, &[_][]const u8{ dir, filename });
    defer c_allocator.free(file_path);

    var file = try config.project_root.openFile(file_path, .{});
    defer file.close();
    var reader = file.reader();

    var line_number: u32 = 1;
    var buf: [1024]u8 = undefined;
    var wrote_line = false;
    while (try reader.readUntilDelimiterOrEof(&buf, '\n')) |line| {
        if (std.mem.startsWith(u8, line, "#pragma ")) {
            var parts = std.mem.split(u8, line, " ");
            _ = parts.next();
            const pragma = parts.next().?;

            if (std.mem.eql(u8, pragma, "include")) {
                var include_path = parts.next().?;
                if (include_path[0] != '"' or include_path[include_path.len - 1] != '"') {
                    debug.print("{s}:{}: Invalid #pragma include\n", .{ file_path, line_number });
                    debug.print("{s}:{}:   {s}\n", .{ file_path, line_number, line });
                    continue;
                }
                include_path = include_path[1 .. include_path.len - 1];

                loadFile(config, writer, file_dir, include_path) catch unreachable;
                _ = try writer.print("#line {d} \"{s}\"\n", .{ line_number, escape_dquote(filename) });

                continue;
            } else {
                debug.print("{s}:{}: Unknown #pragma directive '{s}'\n", .{ file_path, line_number, pragma });
                debug.print("{s}:{}:   {s}\n", .{ file_path, line_number, line });
            }
        }

        if (!std.mem.startsWith(u8, line, "#version") and !wrote_line) {
            _ = try writer.print("#line {d} \"{s}\"\n", .{ line_number - 1, escape_dquote(filename) });
            wrote_line = true;
        }

        _ = try writer.write(line);
        _ = try writer.write("\n");

        line_number += 1;
    }
}

fn reloadShader(config: *const cfg.Config, current: *gl.ShaderProgram, frag_filename: []const u8, size: u64) !void {
    var buffer = try std.ArrayList(u8).initCapacity(c_allocator, @as(usize, @intCast(size)));
    errdefer buffer.deinit();

    try loadFile(config, buffer.writer(), "", frag_filename);

    const frag_source = try buffer.toOwnedSlice();
    defer c_allocator.free(frag_source);

    const vert_source =
        \\#version 330 core
        \\
        \\layout(location = 0) in vec2 position;
        \\out vec2 uv;
        \\
        \\void main() {
        \\  uv = position / 2.0 + 0.5;
        \\  gl_Position = vec4(position, 0, 1);
        \\}
    ;

    if (gl.ShaderProgram.create(vert_source, frag_source)) |new_program| {
        current.destroy();
        current.* = new_program;
    } else |err| {
        debug.print("Error while reloading shader: {}\n", .{err});
    }
}