8.7 KiB
dimal — Dimensional Analysis for Zig
A dimensional analysis library for Zig with a unified Tensor API for scalars, vectors, matrices, and higher-dimensional data. All dimension and unit tracking happens at compile time—zero runtime overhead—and all operations use SIMD intrinsics.
If you try to add meters to seconds, it won't compile. That's the point.
Source: git.bouvais.lu/adrien/zig-dimal
Minimum Zig version:0.16.0
Background
Started because I needed i128 positions for a space simulation to avoid floating-point precision loss far from the origin. Grew into a type system for tracking physical dimensions at compile time. It's been useful enough to share.
- Compile-time dimension checking — catch unit mismatches before runtime.
- Unified
TensorAPI — same interface for scalars, vectors, matrices, and higher-rank tensors. - SIMD operations — vector and matrix code automatically uses SIMD instructions.
- Zero runtime cost — all dimension and scale tracking is erased at compile time.
- Supports
i128— useful for high-precision fixed-point integer math.
Features
- Compile-time dimension checking — all physical-unit tracking happens at compile time.
- Automatic unit conversion — use
.to()to convert between compatible units (e.g.km/h→m/s). Scale factors are resolved at comptime. - Unified
TensorAPI — one type for scalars{1}, vectors{N}, matrices{M, N}, and higher-rank tensors. - SIMD operations — vector and matrix code compiles to SIMD instructions automatically.
- Tensor contraction —
.contract(other, axis_a, axis_b)for dot products, matrix multiplication, and general tensor contractions. - Full SI prefix support —
picothroughpeta, plus Imperial units and time scales. - Physical constants — Planck, Boltzmann, speed of light, gravitational constant, etc.
- Pre-built quantities —
Velocity,Acceleration,Force,Energy,Pressure,Charge, and more. - Basic vector operations — cross product, length/magnitude, element-wise arithmetic.
- Formatting — values print with units:
9.81m.s⁻²,0.172km.
Current Limitations
- GPU support not implemented.
- Performance on small tensors is limited by Zig's vector width.
The 7 SI Base Dimensions
| Symbol | Dimension | SI Unit |
|---|---|---|
L |
Length | m |
M |
Mass | g |
T |
Time | s |
I |
Electric Current | A |
Tr |
Temperature | K |
N |
Amount of Substance | mol |
J |
Luminous Intensity | cd |
Installation
1. Add the dependency (Zig 0.14+)
zig fetch --save git+https://git.bouvais.lu/adrien/zig-dimal#0.2.0
2. Wire it up in build.zig
const std = @import("std");
pub fn build(b: *std.Build) void {
const target = b.standardTargetOptions(.{});
const optimize = b.standardOptimizeOption(.{});
const dimal = b.dependency("dimal", .{
.target = target,
.optimize = optimize,
}).module("dimal");
const exe = b.addExecutable(.{
.name = "my_app",
.root_source_file = b.path("src/main.zig"),
.target = target,
.optimize = optimize,
});
exe.root_module.addImport("dimal", dimal);
b.installArtifact(exe);
}
3. Import and use
const dma = @import("dimal");
const Tensor = dma.Tensor;
const Base = dma.Base;
Quick Example: Lunar Descent
Simulate a spacecraft descending to the Moon with correct physics and type safety:
const std = @import("std");
const dma = @import("dimal");
const Tensor = dma.Tensor;
pub fn main() void {
// Define types: m/s² acceleration, m/s velocity, m distance
const Acceleration = dma.Base.Acceleration.Of(f64);
const Velocity = dma.Base.Velocity.Of(f64);
const Distance = dma.Base.Meter.Of(f64);
const Time = dma.Base.Second.Of(f64);
// Initial conditions
const g_moon: Acceleration = .{ .data = @splat(1.62) };
const v_initial: Velocity = .{ .data = @splat(100.0) };
const h_initial: Distance = .{ .data = @splat(10000.0) };
const dt: Time = .{ .data = @splat(1.0) };
var h = h_initial;
var v = v_initial;
var t: f64 = 0;
// Simulate descent
while (h.data[0] > 0 and t < 1000) : (t += 1.0) {
// a = -g (gravity pulls down)
const a = g_moon.mul(-1.0);
// Update: v = v₀ + at
v = v.add(a.mul(dt));
// Update: h = h₀ + vt
h = h.add(v.mul(dt));
if (@mod(t, 100.0) == 0) {
std.debug.print("t={d:.0}s | h={d:.1} | v={d:.1}\n", .{
t,
h,
v,
});
}
}
std.debug.print("Landed in {d:.1}s at h={d:.1}\n", .{ t, h });
}
Output:
t=0s | h=10000m | v=100m.s⁻¹
t=100s | h=8019m | v=-61.8m.s⁻¹
t=200s | h=4174.4m | v=-223.6m.s⁻¹
...
Landed in 323.5s at h=-0.01m
API Overview
Tensors
A Tensor is parameterized by:
T— numeric type:f32,f64,i128, etc.dims— physical dimensions (struct literal):.{.L = 1, .T = -1}means length/time (velocity).scales— SI prefixes or custom scales:.{.L = .k, .T = .hour}means km/h.shape— array shape:&.{1}is a scalar,&.{3}is a 3-vector,&.{3, 3}is a 3×3 matrix.
// Scalar: 1-element tensor
const Meter = Tensor(f64, .{.L = 1}, .{}, &.{1});
const m = Meter{ .data = @splat(5.0) };
// Vector: N-element tensor (SIMD)
const Vec3Meter = Tensor(f64, .{.L = 1}, .{}, &.{3});
const v = Vec3Meter{ .data = @shuffle(f64, [_]f64{1, 2, 3}, [_]f64 undefined, [_]i32{0, 1, 2, 0, 0, 0}) };
// Matrix: M×N tensor (SIMD-accelerated)
const Mat3x3Velocity = Tensor(f32, .{.L = 1, .T = -1}, .{}, &.{3, 3});
const m_vel = Mat3x3Velocity{ .data = @splat(10.0) };
// Higher-rank tensor
const Rank4 = Tensor(f64, .{.M = 1}, .{}, &.{2, 3, 4, 5});
Common Operations
| Operation | Description |
|---|---|
.add(rhs) |
Element-wise addition. Auto-converts scales. |
.sub(rhs) |
Element-wise subtraction. |
.mul(rhs) |
Multiply; dimensions are summed. rhs can be a tensor or bare number. |
.div(rhs) |
Divide; dimensions are subtracted. |
.contract(other, axis_a, axis_b) |
Tensor contraction: dot product, matrix multiply, or general N-D contraction. |
.cross(rhs) |
Cross product (3-vectors only). Returns a 3-vector. |
.length() / .lengthSqr() |
Euclidean length (or squared length) of a vector. Returns a scalar T. |
.product() |
Multiply all elements. Returns a scalar with combined dimensions. |
.abs() |
Element-wise absolute value. Dimensions unchanged. |
.pow(exp) |
Raise to comptime exponent. Dimension exponents multiplied by exp. |
.sqrt() |
Element-wise square root. Compile error if any dimension exponent is odd. |
.to(DestType) |
Convert to another unit of the same dimension. Comptime error on mismatch. |
.eq(rhs) / .ne(rhs) |
Element-wise equality/inequality. |
.gt(rhs) / .gte(rhs) |
Greater-than comparisons. |
.lt(rhs) / .lte(rhs) |
Less-than comparisons. |
Pre-built Types (via dma.Base)
Use .Of(T) for base units, .Scaled(T, scales) for custom scales:
const Velocity = dma.Base.Velocity.Of(f64);
const Kmh = dma.Base.Velocity.Scaled(f64, .{.L = .k, .T = .hour});
const Force = dma.Base.Force.Of(f32);
const Energy = dma.Base.Energy.Of(f64);
Also available: Acceleration, Inertia, Pressure, Power, Area, Volume, Density, Frequency, Viscosity, Charge, Potential, Resistance, MagneticFlux, ThermalCapacity, ThermalConductivity, and many more.
SIMD Performance
Operations on vectors and matrices use Zig's @Vector intrinsics, which compile to SIMD instructions on most platforms. This makes vector operations faster than equivalent scalar loops, but don't expect miracles—SIMD is still limited by memory bandwidth and CPU cache.
Run the included benchmarks to see what you get on your hardware:
zig build benchmark
Next Steps
- GPU support — eventually, for large tensor operations. WebGPU is a target.
- Toy physics language — I've been sketching ideas for a language optimized for numerical physics (tentatively called Éclat). It would use dimal as the foundation. No timeline yet; this is a long-term experiment.
Testing & Benchmarks
zig build test # Run all unit tests
zig build benchmark # Run performance benchmarks
License
See the repository for license details.