libxev-sys 0.0.1-rc.2

Low-level FFI bindings to libxev (built from vendored sources via Zig).
const std = @import("std");

inline fn indent(depth: usize, writer: anytype) !void {
    for (0..depth) |_| try writer.writeByte(' ');
}

pub fn describe(comptime T: type, writer: anytype, depth: usize) !void {
    const type_info = @typeInfo(T);
    switch (type_info) {
        .Type,
        .Void,
        .Bool,
        .NoReturn,
        .Int,
        .Float,
        .Pointer,
        .Array,
        .ComptimeFloat,
        .ComptimeInt,
        .Undefined,
        .Null,
        .Optional,
        .ErrorUnion,
        .ErrorSet,
        .Enum,
        .Fn,
        .Opaque,
        .Frame,
        .AnyFrame,
        .Vector,
        .EnumLiteral,
        => {
            try writer.print("{s} ({d} bytes)", .{ @typeName(T), @sizeOf(T) });
        },
        .Union => |s| {
            try writer.print("{s} ({d} bytes) {{\n", .{ @typeName(T), @sizeOf(T) });
            inline for (s.fields) |f| {
                try indent(depth + 4, writer);
                try writer.print("{s}: ", .{f.name});
                try describe(f.type, writer, depth + 4);
                try writer.writeByte('\n');
            }
            try indent(depth, writer);
            try writer.writeByte('}');
        },
        .Struct => |s| {
            try writer.print("{s} ({d} bytes) {{\n", .{ @typeName(T), @sizeOf(T) });
            inline for (s.fields) |f| {
                try indent(depth + 4, writer);
                try writer.print("{s}: ", .{f.name});
                try describe(f.type, writer, depth + 4);
                try writer.writeByte('\n');
            }
            try indent(depth, writer);
            try writer.writeByte('}');
        },
    }
}