Please check the build logs for more information.
See Builds for ideas on how to fix a failed build, or Metadata for how to configure docs.rs builds.
If you believe this is docs.rs' fault, open an issue.
xDy crate
The xdy crate is the core library of the xDy project. It compiles dice expressions into reusable functions, optimizes them, and evaluates them against client-supplied pseudorandom number generators. For a high-level overview of the project, language features, and usage examples, see the project README.
This document covers the internal architecture for contributors and curious power users.
Architecture
Compilation pipeline
Source code flows through a six-stage pipeline. The compile() convenience function drives the full pipeline; compile_unoptimized() skips the optimizer.
graph LR
A["Source Code<br/><code>&str</code>"] --> B["Parser<br/><code>parser::parse</code>"]
B --> C["AST<br/><code>ast::Function</code>"]
C --> D["Compiler<br/><code>Compiler::compile</code>"]
D --> E["Unoptimized IR<br/><code>Function</code>"]
E --> F["Optimizer<br/><code>StandardOptimizer</code>"]
F --> G["Optimized IR<br/><code>Function</code>"]
style A fill:#f9f,stroke:#333,color:#000
style G fill:#9f9,stroke:#333,color:#000
Parser. A nom-based combinator parser recognizes the xDy grammar and produces an abstract syntax tree (ast::Function). The grammar supports operator precedence via recursive descent: add_sub → mul_div_mod → unary → exponent → primary. See parser::combinators for the full set of production rules; the Rustdoc includes a railroad diagram generated from the EBNF grammar.
Compiler. The Compiler implements the ASTVisitor trait and walks the AST in a single pass to emit IR instructions. It uses static single assignment (SSA) form — every instruction writes to a fresh register or rolling record. Parameters are allocated first (in declaration order), then external variables (depth-first, left-to-right), ensuring deterministic register layout.
Optimizer. The StandardOptimizer applies five transformation passes in a fixed-point loop, then a final register coalescing pass. See Optimizer below.
Intermediate representation
The IR is a simple register transfer language (RTL) with no control flow — all instructions reside in a single basic block. The machine model provides two register files:
- Register bank (
@0,@1, …): holdsi32values for parameters, external variables, and computed intermediates. - Rolling record bank (
⚅0,⚅1, …): holds the individual results of dice rolls and range selections, along with drop counters.
Each operand uses one of three addressing modes:
| Mode | Notation | Description |
|---|---|---|
| Immediate | #N |
Constant i32 embedded in the instruction |
| Register | @N |
Index into the register bank |
| RollingRecord | ⚅N |
Index into the rolling record bank |
See Instruction set reference for the complete ISA.
Evaluator
The evaluator is a linear instruction interpreter. For each evaluation it:
- Allocates a register bank (sized at compile time) and a rolling record bank.
- Loads arguments into parameter registers and environment bindings into external variable registers.
- Walks the instruction stream sequentially — there is no branching, so the program counter simply increments.
- Returns the final
Evaluation, which includes both thei32result and the completeVec<RollingRecord>for display.
graph TD
subgraph VM["Evaluator VM"]
direction TB
PC["Program Counter"]
subgraph RF["Register Bank (i32)"]
R0["@0: param"]
R1["@1: extern"]
RN["@N: computed"]
end
subgraph RR["Rolling Record Bank"]
RR0["⚅0: dice results"]
RR1["⚅1: range results"]
end
end
F["Function (IR)"] --> PC
RNG["pRNG"] --> RR
ARGS["Arguments"] --> RF
ENV["Environment"] --> RF
VM --> OUT["Evaluation<br/>result + records"]
style VM fill:#e8f4fd,stroke:#333,color:#000
style RF fill:#d4edda,stroke:#333,color:#000
style RR fill:#fff3cd,stroke:#333,color:#000
All arithmetic saturates to i32::MIN/i32::MAX. Division by zero yields zero. 0^0 = 1. Negative exponents yield zero.
The evaluator also provides bounds(), which computes static min/max bounds and (when possible) the total outcome count — without requiring an RNG.
Optimizer
The StandardOptimizer applies six passes. The first five run in a fixed-point loop; the sixth runs once at the end:
graph TD
A["Input Function (SSA)"] --> B["Common Subexpression Elimination"]
B --> C["Constant Commuting"]
C --> D["Constant Folding"]
D --> E["Strength Reduction"]
E --> F["Dead Code Elimination"]
F --> G{"Changed?"}
G -- Yes --> B
G -- No --> H["Register Coalescing"]
H --> I["Optimized Function"]
style A fill:#f9f,stroke:#333,color:#000
style I fill:#9f9,stroke:#333,color:#000
style G fill:#ff9,stroke:#333,color:#000
| Pass | Effect |
|---|---|
| CSE | Identifies identical instructions and replaces duplicates with references to the first occurrence |
| Constant commuting | Moves immediate operands to the left side of commutative operators, creating folding opportunities |
| Constant folding | Evaluates instructions whose operands are all immediates at compile time |
| Strength reduction | Replaces expensive operations with cheaper equivalents (e.g., multiply by power of two → shift) |
| Dead code elimination | Removes instructions whose results are never consumed |
| Register coalescing | Merges equivalent registers to reduce the register bank size; breaks SSA form, so it runs last |
Histogram engine
The histogram engine computes the complete probability distribution of a dice expression by exhaustive enumeration. It uses a state-machine approach:
- Each
EvaluationStatesuspends at roll/range instructions. - For each possible outcome of the suspended instruction, a successor state is generated.
- States that reach
Returncontribute their result to the histogram. - A configurable gas tank (
MAX_SUCCESSORS = 30) bounds the branching factor per instruction.
Two builders are provided: SerialHistogramBuilder and ParallelHistogramBuilder (the latter requires the parallel-histogram feature and uses rayon).
Diagnostics
The diagnostics module provides structured error reporting with a fix-and-retry strategy. When parsing fails, the diagnostics engine:
- Analyzes the parse error to identify the failure kind (unclosed delimiter, missing operand, etc.).
- Applies a minimal fix to the source text.
- Re-parses the fixed source to discover additional errors.
- Maps error spans back to the original source using an offset map.
This produces rich Diagnostic values with error kinds, source spans, messages, and suggested fixes — designed to power IDE-style error reporting on every keystroke.
Instruction set reference
Addressing modes
| Mode | Syntax | Description |
|---|---|---|
Immediate(N) |
#N |
Constant i32 value |
Register(N) |
@N |
General-purpose register |
RollingRecord(N) |
⚅N |
Rolling record (dice/range results) |
Roll instructions
| Instruction | Syntax | Semantics |
|---|---|---|
RollRange |
⚅N ← roll range S:E |
Select a random value from the inclusive range [S, E] and store it in rolling record N |
RollStandardDice |
⚅N ← roll standard dice CDF |
Roll C standard dice with F faces each, storing all results in rolling record N |
RollCustomDice |
⚅N ← roll custom dice CD[f₁, f₂, …] |
Roll C custom dice with the specified face values, storing all results in rolling record N |
Drop instructions
| Instruction | Syntax | Semantics |
|---|---|---|
DropLowest |
⚅N ← drop lowest K from ⚅N |
Mark the K lowest results in rolling record N as dropped |
DropHighest |
⚅N ← drop highest K from ⚅N |
Mark the K highest results in rolling record N as dropped |
Reduce instructions
| Instruction | Syntax | Semantics |
|---|---|---|
SumRollingRecord |
@N ← sum rolling record ⚅M |
Sum the non-dropped results of rolling record M into register N (saturating) |
Arithmetic instructions
All arithmetic instructions saturate on overflow/underflow.
| Instruction | Syntax | Semantics |
|---|---|---|
Add |
@N ← A + B |
Saturating addition |
Sub |
@N ← A - B |
Saturating subtraction |
Mul |
@N ← A * B |
Saturating multiplication |
Div |
@N ← A / B |
Saturating division; x / 0 = 0 |
Mod |
@N ← A % B |
Saturating remainder; x % 0 = 0 |
Exp |
@N ← A ^ B |
Saturating exponentiation; 0^0 = 1, x^(-n) = 0 |
Neg |
@N ← -A |
Saturating negation |
Control instructions
| Instruction | Syntax | Semantics |
|---|---|---|
Return |
return A |
Set the function result to A and terminate |
Cargo features
| Feature | Default | Description |
|---|---|---|
parallel-histogram |
Yes | Parallel probability distribution computation via rayon |
serde |
Yes | Serialize/Deserialize for IR, evaluation, and histogram types |
Safety
xDy is designed to be well-behaved for all inputs. Dice expression values are i32 and all arithmetic operations saturate on overflow or underflow. Neither the compiler nor evaluator should panic or cause undefined behavior, even for invalid dice expressions and inputs, though client misuse of vector results can lead to panics. The main crate contains no unsafe code.