301 lines
11 KiB
Zig
301 lines
11 KiB
Zig
const std = @import("std");
|
|
const assert = std.debug.assert;
|
|
const Build = std.Build;
|
|
const fs = std.fs;
|
|
const mem = std.mem;
|
|
|
|
const Scanner = @import("zig-wayland").Scanner;
|
|
|
|
/// While a river release is in development, this string should contain the version in development
|
|
/// with the "-dev" suffix.
|
|
/// When a release is tagged, the "-dev" suffix should be removed for the commit that gets tagged.
|
|
/// Directly after the tagged commit, the version should be bumped and the "-dev" suffix added.
|
|
const version = "0.4.0-dev";
|
|
|
|
pub fn build(b: *Build) !void {
|
|
const target = b.standardTargetOptions(.{});
|
|
const optimize = b.standardOptimizeOption(.{});
|
|
|
|
const strip = b.option(bool, "strip", "Omit debug information") orelse false;
|
|
const pie = b.option(bool, "pie", "Build a Position Independent Executable") orelse false;
|
|
const llvm = !(b.option(bool, "no-llvm", "(expirimental) Use non-LLVM x86 Zig backend") orelse false);
|
|
|
|
const omit_frame_pointer = switch (optimize) {
|
|
.Debug, .ReleaseSafe => false,
|
|
.ReleaseFast, .ReleaseSmall => true,
|
|
};
|
|
|
|
const man_pages = b.option(
|
|
bool,
|
|
"man-pages",
|
|
"Set to true to build man pages. Requires scdoc. Defaults to true if scdoc is found.",
|
|
) orelse scdoc_found: {
|
|
_ = b.findProgram(&.{"scdoc"}, &.{}) catch |err| switch (err) {
|
|
error.FileNotFound => break :scdoc_found false,
|
|
else => return err,
|
|
};
|
|
break :scdoc_found true;
|
|
};
|
|
|
|
const bash_completion = b.option(
|
|
bool,
|
|
"bash-completion",
|
|
"Set to true to install bash completion for riverctl. Defaults to true.",
|
|
) orelse true;
|
|
|
|
const zsh_completion = b.option(
|
|
bool,
|
|
"zsh-completion",
|
|
"Set to true to install zsh completion for riverctl. Defaults to true.",
|
|
) orelse true;
|
|
|
|
const fish_completion = b.option(
|
|
bool,
|
|
"fish-completion",
|
|
"Set to true to install fish completion for riverctl. Defaults to true.",
|
|
) orelse true;
|
|
|
|
const xwayland = b.option(
|
|
bool,
|
|
"xwayland",
|
|
"Set to true to enable xwayland support",
|
|
) orelse false;
|
|
|
|
const full_version = blk: {
|
|
if (mem.endsWith(u8, version, "-dev")) {
|
|
var ret: u8 = undefined;
|
|
|
|
const git_describe_long = b.runAllowFail(
|
|
&.{ "git", "-C", b.build_root.path orelse ".", "describe", "--long" },
|
|
&ret,
|
|
.Inherit,
|
|
) catch break :blk version;
|
|
|
|
var it = mem.split(u8, mem.trim(u8, git_describe_long, &std.ascii.whitespace), "-");
|
|
_ = it.next().?; // previous tag
|
|
const commit_count = it.next().?;
|
|
const commit_hash = it.next().?;
|
|
assert(it.next() == null);
|
|
assert(commit_hash[0] == 'g');
|
|
|
|
// Follow semantic versioning, e.g. 0.2.0-dev.42+d1cf95b
|
|
break :blk b.fmt(version ++ ".{s}+{s}", .{ commit_count, commit_hash[1..] });
|
|
} else {
|
|
break :blk version;
|
|
}
|
|
};
|
|
|
|
const options = b.addOptions();
|
|
options.addOption(bool, "xwayland", xwayland);
|
|
options.addOption([]const u8, "version", full_version);
|
|
|
|
const scanner = Scanner.create(b, .{});
|
|
|
|
scanner.addSystemProtocol("stable/xdg-shell/xdg-shell.xml");
|
|
scanner.addSystemProtocol("stable/tablet/tablet-v2.xml");
|
|
scanner.addSystemProtocol("staging/cursor-shape/cursor-shape-v1.xml");
|
|
scanner.addSystemProtocol("staging/ext-session-lock/ext-session-lock-v1.xml");
|
|
scanner.addSystemProtocol("staging/tearing-control/tearing-control-v1.xml");
|
|
scanner.addSystemProtocol("unstable/pointer-constraints/pointer-constraints-unstable-v1.xml");
|
|
scanner.addSystemProtocol("unstable/pointer-gestures/pointer-gestures-unstable-v1.xml");
|
|
scanner.addSystemProtocol("unstable/xdg-decoration/xdg-decoration-unstable-v1.xml");
|
|
|
|
scanner.addCustomProtocol(b.path("protocol/river-control-unstable-v1.xml"));
|
|
scanner.addCustomProtocol(b.path("protocol/river-status-unstable-v1.xml"));
|
|
scanner.addCustomProtocol(b.path("protocol/river-layout-v3.xml"));
|
|
scanner.addCustomProtocol(b.path("protocol/wlr-layer-shell-unstable-v1.xml"));
|
|
scanner.addCustomProtocol(b.path("protocol/wlr-output-power-management-unstable-v1.xml"));
|
|
|
|
// Some of these versions may be out of date with what wlroots implements.
|
|
// This is not a problem in practice though as long as river successfully compiles.
|
|
// These versions control Zig code generation and have no effect on anything internal
|
|
// to wlroots. Therefore, the only thnig that can happen due to a version being too
|
|
// old is that river fails to compile.
|
|
scanner.generate("wl_compositor", 4);
|
|
scanner.generate("wl_subcompositor", 1);
|
|
scanner.generate("wl_shm", 1);
|
|
scanner.generate("wl_output", 4);
|
|
scanner.generate("wl_seat", 7);
|
|
scanner.generate("wl_data_device_manager", 3);
|
|
|
|
scanner.generate("xdg_wm_base", 2);
|
|
scanner.generate("zwp_pointer_gestures_v1", 3);
|
|
scanner.generate("zwp_pointer_constraints_v1", 1);
|
|
scanner.generate("zwp_tablet_manager_v2", 1);
|
|
scanner.generate("zxdg_decoration_manager_v1", 1);
|
|
scanner.generate("ext_session_lock_manager_v1", 1);
|
|
scanner.generate("wp_cursor_shape_manager_v1", 1);
|
|
scanner.generate("wp_tearing_control_manager_v1", 1);
|
|
|
|
scanner.generate("zriver_control_v1", 1);
|
|
scanner.generate("zriver_status_manager_v1", 4);
|
|
scanner.generate("river_layout_manager_v3", 2);
|
|
|
|
scanner.generate("zwlr_layer_shell_v1", 4);
|
|
scanner.generate("zwlr_output_power_manager_v1", 1);
|
|
|
|
const wayland = b.createModule(.{ .root_source_file = scanner.result });
|
|
|
|
const xkbcommon = b.dependency("zig-xkbcommon", .{}).module("xkbcommon");
|
|
const pixman = b.dependency("zig-pixman", .{}).module("pixman");
|
|
|
|
const wlroots = b.dependency("zig-wlroots", .{}).module("wlroots");
|
|
wlroots.addImport("wayland", wayland);
|
|
wlroots.addImport("xkbcommon", xkbcommon);
|
|
wlroots.addImport("pixman", pixman);
|
|
|
|
// We need to ensure the wlroots include path obtained from pkg-config is
|
|
// exposed to the wlroots module for @cImport() to work. This seems to be
|
|
// the best way to do so with the current std.Build API.
|
|
wlroots.resolved_target = target;
|
|
wlroots.linkSystemLibrary("wlroots-0.18", .{});
|
|
|
|
const flags = b.createModule(.{ .root_source_file = b.path("common/flags.zig") });
|
|
const globber = b.createModule(.{ .root_source_file = b.path("common/globber.zig") });
|
|
|
|
{
|
|
const river = b.addExecutable(.{
|
|
.name = "river",
|
|
.root_source_file = b.path("river/main.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
.strip = strip,
|
|
.use_llvm = llvm,
|
|
.use_lld = llvm,
|
|
});
|
|
river.root_module.addOptions("build_options", options);
|
|
|
|
river.linkLibC();
|
|
river.linkSystemLibrary("libevdev");
|
|
river.linkSystemLibrary("libinput");
|
|
river.linkSystemLibrary("wayland-server");
|
|
river.linkSystemLibrary("wlroots-0.18");
|
|
river.linkSystemLibrary("xkbcommon");
|
|
river.linkSystemLibrary("pixman-1");
|
|
|
|
river.root_module.addImport("wayland", wayland);
|
|
river.root_module.addImport("xkbcommon", xkbcommon);
|
|
river.root_module.addImport("pixman", pixman);
|
|
river.root_module.addImport("wlroots", wlroots);
|
|
river.root_module.addImport("flags", flags);
|
|
river.root_module.addImport("globber", globber);
|
|
|
|
river.addCSourceFile(.{
|
|
.file = b.path("river/wlroots_log_wrapper.c"),
|
|
.flags = &.{ "-std=c99", "-O2" },
|
|
});
|
|
|
|
// TODO: remove when zig issue #131 is implemented
|
|
scanner.addCSource(river);
|
|
|
|
river.pie = pie;
|
|
river.root_module.omit_frame_pointer = omit_frame_pointer;
|
|
|
|
b.installArtifact(river);
|
|
}
|
|
|
|
{
|
|
const riverctl = b.addExecutable(.{
|
|
.name = "riverctl",
|
|
.root_source_file = b.path("riverctl/main.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
.strip = strip,
|
|
.use_llvm = llvm,
|
|
.use_lld = llvm,
|
|
});
|
|
riverctl.root_module.addOptions("build_options", options);
|
|
|
|
riverctl.root_module.addImport("flags", flags);
|
|
riverctl.root_module.addImport("wayland", wayland);
|
|
riverctl.linkLibC();
|
|
riverctl.linkSystemLibrary("wayland-client");
|
|
|
|
scanner.addCSource(riverctl);
|
|
|
|
riverctl.pie = pie;
|
|
riverctl.root_module.omit_frame_pointer = omit_frame_pointer;
|
|
|
|
b.installArtifact(riverctl);
|
|
}
|
|
|
|
{
|
|
const rivertile = b.addExecutable(.{
|
|
.name = "rivertile",
|
|
.root_source_file = b.path("rivertile/main.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
.strip = strip,
|
|
.use_llvm = llvm,
|
|
.use_lld = llvm,
|
|
});
|
|
rivertile.root_module.addOptions("build_options", options);
|
|
|
|
rivertile.root_module.addImport("flags", flags);
|
|
rivertile.root_module.addImport("wayland", wayland);
|
|
rivertile.linkLibC();
|
|
rivertile.linkSystemLibrary("wayland-client");
|
|
|
|
scanner.addCSource(rivertile);
|
|
|
|
rivertile.pie = pie;
|
|
rivertile.root_module.omit_frame_pointer = omit_frame_pointer;
|
|
|
|
b.installArtifact(rivertile);
|
|
}
|
|
|
|
{
|
|
const wf = Build.Step.WriteFile.create(b);
|
|
const pc_file = wf.add("river-protocols.pc", b.fmt(
|
|
\\prefix={s}
|
|
\\datadir=${{prefix}}/share
|
|
\\pkgdatadir=${{datadir}}/river-protocols
|
|
\\
|
|
\\Name: river-protocols
|
|
\\URL: https://codeberg.org/river/river
|
|
\\Description: protocol files for the river wayland compositor
|
|
\\Version: {s}
|
|
, .{ b.install_prefix, full_version }));
|
|
|
|
b.installFile("protocol/river-layout-v3.xml", "share/river-protocols/river-layout-v3.xml");
|
|
b.getInstallStep().dependOn(&b.addInstallFile(pc_file, "share/pkgconfig/river-protocols.pc").step);
|
|
}
|
|
|
|
if (man_pages) {
|
|
inline for (.{ "river", "riverctl", "rivertile" }) |page| {
|
|
// Workaround for https://github.com/ziglang/zig/issues/16369
|
|
// Even passing a buffer to std.Build.Step.Run appears to be racy and occasionally deadlocks.
|
|
const scdoc = b.addSystemCommand(&.{ "/bin/sh", "-c", "scdoc < doc/" ++ page ++ ".1.scd" });
|
|
// This makes the caching work for the Workaround, and the extra argument is ignored by /bin/sh.
|
|
scdoc.addFileArg(b.path("doc/" ++ page ++ ".1.scd"));
|
|
|
|
const stdout = scdoc.captureStdOut();
|
|
b.getInstallStep().dependOn(&b.addInstallFile(stdout, "share/man/man1/" ++ page ++ ".1").step);
|
|
}
|
|
}
|
|
|
|
if (bash_completion) {
|
|
b.installFile("completions/bash/riverctl", "share/bash-completion/completions/riverctl");
|
|
}
|
|
|
|
if (zsh_completion) {
|
|
b.installFile("completions/zsh/_riverctl", "share/zsh/site-functions/_riverctl");
|
|
}
|
|
|
|
if (fish_completion) {
|
|
b.installFile("completions/fish/riverctl.fish", "share/fish/vendor_completions.d/riverctl.fish");
|
|
}
|
|
|
|
{
|
|
const globber_test = b.addTest(.{
|
|
.root_source_file = b.path("common/globber.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
});
|
|
const run_globber_test = b.addRunArtifact(globber_test);
|
|
|
|
const test_step = b.step("test", "Run the tests");
|
|
test_step.dependOn(&run_globber_test.step);
|
|
}
|
|
}
|