Spade 0.13.0
Posted 2025-02-20 by The Spade Developers
Today we're releasing Spade 0.13, one of the biggest Spade releases yet! It
includes gen if
for compile time conditionals and recursion, lots of
namespacing fixes, improvements to handling external Verilog, and lots more.
gen if
🔗
gen if
allows you to conditionally branch based on type parameters. The primary use
case for this is recursively iterating over arrays. For example, you can now write
a function that adds one to every element in an array like this
fn add_one<#W, #N>(x: [int<W>; N]) -> [int<{W+1}>; N] {
gen if N == 0 {
[]
} else {
[x[0] + 1] `concat_arrays` add_one(x[1:N])
}
}
While this is useful for some things in isolation, it is going to lay the groundwork for more complex generic constructs in later releases, especially once lambdas are implemented.
Interacting with external Verilog🔗
This release changes the syntax of the old __builtin__
marker to extern
to make
it less hacky and more clear what it does. The #[no_mangle]
attribute has also
been given a all
parameter which blanket applies #[no_mangle]
to all parameters.
What used to look like this:
entity external_verilog(
#[no_mangle] clk: clock,
#[no_mangle] a: uint<8>,
#[no_mangle] b: uint<8>,
#[no_mangle] out: inv &uint<8>
) __builtin__
now looks like this
#[no_mangle(all)]
extern entity external_verilog(clk: clock, a: uint<8>, b: uint<8>, out: inv &uint<8>);
Including external Verilog in Swim has also been improved, there is now a
[verilog]
and [synthesis.verilog]
section in swim.toml
which allows you
to both include specific Verilog files, and specify include directories.
You can now also use where
clauses on extern units.
Thanks to Ethan for all these changes!
Namespacing changes🔗
The namespacing system has seen several improvements in this release. The most user
facing change is that all modules now need a main.spade
at the root of the module,
and for other files to be included in the project, you need to add mod filename;
to
the corresponding main.spade
. For rust users, this will feel very familiar, what rust calls
main.rs
, lib.rs
and mod.rs
are all called main.spade
.
The namespace of the main.spade
file has also changed, a main.spade
with
fn a() {}
in a project called project
would previously result in project::main::a
, now it
results in project::a;
There have also been several bug fixes and improvements in the name resolution system including
use
statements of undefined names now error on theuse
instead of when using theuse
d name.use
between namespaces now works as expected- No more stack overflows or infinite recursion if
use
-ing a single identifier
Thanks to DasLixou for most of these fixes!
Expressions are now statements🔗
If you wanted to call a unit with "side effects" previously, you would have had to write
let _ = inst unit(some_port);
now you can just write
inst unit(some_port);
Improved const generics🔗
You can use const generics in unit parameter and output types, which means you no longer
have to add "internal" type parameters and where
clauses. For example, you can now
write
fn weird_operation<#uint N>(a: uint<N>, b: uint<{N+5}>) -> uint<{N/2}> {}
Synchronous Resets🔗
Spade now uses synchronous resets for all registers instead of asynchronous.
Other fixes🔗
There have also been numerous small fixes and improvements that don't fit in this blog post, you can read them all in the Changelog
SPADETID!🔗
Frans has started regularly streaming Spade development on
https://www.twitch.tv/thezoq2/. If you want
to be notified when this happens, you can join the
discord and give yourself the SPADETID
role.