judofyr/spice
Fine-grained parallelism with sub-nanosecond overhead in Zig
Spice uses heartbeat scheduling to accomplish extremely efficient parallelism in Zig:
(Update, September 2024: Chili is a Rust port of the ideas presented here. Check it out!)
The benchmark in the figure above (summing over the nodes in a binary tree) is typically one of the worst cases for parallelism frameworks: The actual operation is extremely fast so any sort of overhead will have a measurable impact.
Here's the exact same benchmark in Rayon, an excellent library in Rust for doing parallelism. Both implementations follow the same fork/join API which gives code that is very easy to read and reason about. None of the findings here would surprise anyone who deeply knows Rayon and there are ways of getting better performance out of Rayon by using different techniques. This comes at cost of the code becoming more complicated and/or behaving subpar on different types of input. The purpose of this benchmark is to not discourage use of Rayon (on the contrary!), but rather demonstrate that it is possible to have both simple code and good parallelism. See issue #5 for a longer discussion.
The overhead here is roughly ~15 ns (from 7.48 ns to 22.99 ns) which means that at 4 threads we're "back" to the sequential performance - just using four times as much CPU.
Luckily we are able to get linear speed-up (in terms of threads) initially.
These benchmarks were ran on a c4-standard-16
instance in Google Cloud with 16 cores.
Rayon itself shows a nice ~14x speed-up (from 22.99 ns to 1.64 ns) at 16 threads, but compared to the baseline this ends up only being ~4.5x due to the overhead.
In comparison, Spice scales slightly worse: It only got ~11x speed-up when going from 1 to 16 threads. However, due its low overhead this is also essentially the speed-up compared to the baseline.
(It's not entirely clear why the Zig baseline implementation is twice as fast as the Rust implementation. The compiled assembly (godbolt) show that Rust saves five registers on the stack while Zig only saves three, but why? For the purpose of this benchmark it shouldn't matter since we're only comparing against the baseline of each language.)
It becomes even more interesting if we're summing the nodes of a much smaller tree:
In this scenario we have a very short duration of our program: The baseline implementation takes a few microseconds in total to run. For some reason the overhead is a bit higher (~19 ns), but more concerningly we see that performance becomes worse the more threads we're adding. At 32 threads it's in total 60 times slower.
(In this case we're using 32 threads on a machine which only has 16 cores. It's not given that we would see the same slowdown for a machine with 32 cores. Nonetheless, this scaling behavior is concerning.)
The conventional wisdom for parallelism therefore ends up being "it's not worth it unless you have enough work to parallelize". The example above is typically presented as a "bad fit for parallelism". This is understandable and pragmatic, but in practice it makes it a lot more difficult to actually parallelize your code:
The goal of Spice is for you to never have to worry about your program becoming slower by making it parallel. If you're looking to maximize the performance you should of course do elaborate benchmarking, but generally with Spice you can add parallelism and there will be practically no overhead.
The last example of summing over 1000 nodes behaves as follows in Spice:
What's happening here is that it's discovering that the duration is too short so none of the multi-threading kicks in. All the extra threads here are sleeping, giving the cores time to execute other programs.
Spice is primarily a research project. Read along to learn more about it, but if you're considering using it in production you should be aware of its many limitations.
(See the bench/ directory for more details about these specific benchmarks.)
The following example demonstrates how Spice works:
const spice = @import("spice");
// (1) Add task as a parameter.
fn sum(t: *spice.Task, node: *const Node) i64 {
var res: i64 = node.val;
if (node.left) |left_child| {
if (node.right) |right_child| {
var fut = spice.Future(*const Node, i64).init();
// (3) Call `fork` to set up work for another thread.
fut.fork(t, sum, right_child);
// (4) Do some work yourself.
res += t.call(i64, sum, left_child);
if (fut.join(t)) |val| {
// (5) Wait for the other thread to complete the work.
res += val;
} else {
// (6) ... or do it yourself.
res += t.call(i64, sum, right_child);
}
return res;
}
res += t.call(i64, sum, left_child);
}
if (node.right) |right_child| {
// (2) Recursive calls must use `t.call`
res += t.call(i64, sum, right_child);
}
return res;
}
t.call
which will call it for you (in the right way).fork
to set up a piece of work which can be done by a different thread.
This can be called multiple times to set up multiple pieces of work.join
to wait for the work done by the other thread.join
might return null
and this signals that no other thread picked up the work.
In this case you must do the work yourself.Here we repeat ourselves in step 3 and 6:
Both places we refer to sum
and right_child
.
It's possible to hide this duplication by some helper function, but this example demonstrates a core idea behind Spice:
Not every piece of work comes from the queue.
You call fork
to signal that there's something which can be executed by another thread, but if all the other threads are busy then you fallback to executing it as if the fork never happened.
This principle is core to how Spice achieves its low and predictable overhead: If there's no parallelism possible then all Spice is doing on the hot path is pushing and popping the queue (without ever looking at any of the items).
The actually coordination with other threads happens on a fixed heartbeat: Every 100 microsecond or so a thread will look at its current work queue and dispatch the top-most item to another waiting thread. Since the heartbeat happens very infrequently (compared to the clock speed) we also don't need to worry so much about what we're doing during the heartbeat. Even if we spend hundreds of nanoseconds the total overhead becomes small since we do it rarely.
Spice provides the fork/join model which has typically been implementing by using work-stealing. Let's have a look at work-stealing:
However, there's three major sources of inefficiencies in this design:
Every piece of work is a dynamic dispatch. In compiled languages (such as C) function calls are "practically" free due to the capability of statically knowing everything about the called function. This is a scenario which compilers and CPUs have been optimized for decades to execute efficiently. Work-stealing systems don't use this functionality, but instead puts every piece of work into generic "call this dynamic function". It's a small piece of overhead, but it does add up.
The "local" work queue isn't really local. Yes, it's true that every thread have a single queue that they will push work onto, but this is far from a "local" queue as is typically described in concurrent algorithms. This is a queue in which every thread at every point might steal from. In reality, work-stealing systems with N threads have N global queues, where each queue only has a single producer, but everyone is a consumer. Why does this distinction matter? Because all operations on these queues have to use atomic operations. Atomic operations, especially stores, are far more expensive than regular, local stores.
Spinning works great … until it doesn't. The queues in work-stealing systems are typically implemented using spinning: Every thread will optimistically try to acquire a single item from the queue, and if there's a contention with another thread it will try again in a loop. This typically gives great performance … until it doesn't. It can be very hard to reason about this or replicate it since under one set of conditions everything is fine, but suddenly during contention the system will slow down to a halt (i.e. 10x-100x slower).
Spice directly tackles all of these inefficiencies:
while
-loop in Spice which doesn't also contain a wait()
-call which will suspend the thread.
There is no spinning.Let's dive further into how Spice is implemented to achieve its efficient parallelism.
A fork/join program has a set of code blocks which are executed in parallel and once they finish the join
action completes:
join(
fork { code1 }
fork { code2 }
fork { code3 }
)
In Spice this is represented as:
job1 = fork { code1 } // Place on the queue
job2 = fork { code2 } // Place on the queue
code3 // Run right away
if (job2.isExecuting()) {
// Job was picked up by another thread. Wait for it.
job2.wait()
} else {
code2
}
if (job1.isExecuting()) {
// Job was picked up by another thread. Wait for it.
job1.wait()
} else {
code1
}
Notice that code1
and code2
has been duplicated_inside the function.
This is actually a good thing.
Most of the time the job will not be picked up by another thread.
In this case, our program nicely turns into the sequential version (although in reverse order) with a few extra branches which are all very predictable.
This is friendly both for the code optimizer (e.g. it can now inline the function call) and the CPU.
The core idea of heartbeat scheduling is to do scheduling locally and at a low frequency: Every 100 microsecond or so we'd like every thread to look at it local work queue and send work to a different thread. The low frequency is key to eliminating overall overhead. If we're only doing something every 100 microsecond we can actually spend 100 nanoseconds (an eternity!) and still only introduce 0.1% overhead.
Operating systems have built-in support for signaling, but these are very hard to reason about.
The user code gets paused at any random point and it's hard to safely continue running.
For this reason, Spice uses a cooperative approach instead:
The user code have to call tick()
and this detects whether a heartbeat should happen.
This function call is automatically called for you whenever you use the call
-helper.
It's critical that this function is efficient when a heartbeat isn't happening. This is after all the common case (as the heartbeat is only happening every ~100 microsecond).
pub inline fn tick(self: *Task) void {
if (self.worker.heartbeat.load(.monotonic)) {
self.worker.pool.heartbeat(self.worker);
}
}
In Spice we spawn a separate heartbeat thread whose sole purpose is to periodically flip the thread's atomic heartbeat value from false
to true
.
The tick()
function then reads this atomic value and starts its heartbeat code when it's true
.
A key part of reducing the overhead of the ticking is to make sure the heartbeat function itself is marked as cold. This causes the presence of this function call to not use up any registers. Without this the overhead is significantly higher.
If you look inside the codebase of Spice you will find that each thread pool has a single mutex which is locked all over the place. An immediate reaction would be "oh no, a global mutex is terrible" and you might be tempted to replace it.
However, there's no problem with a global mutex until you're being blocked. And you can only be blocked if two conditions occur:
None of these are true for Spice. The heartbeating ensures that typically only a single thread is executing a heartbeat. In addition, no user code is executed while the lock is held. We're only protecting trivial simple memory reads/writes which will complete in constant time.
We're using a doubly-linked list to keep track of the work queue:
fork()
appends to the end, join()
pops from the end (if it's still there), and we pop from the beginning when we want to send work to a background worker.
Appending into a doubly-linked list typically looks like this:
pub fn append(list: *Self, new_node: *Node) void {
if (list.last) |last| {
// Insert after last.
list.insertAfter(last, new_node);
} else {
// Empty list.
list.prepend(new_node);
}
}
Notice that there's a conditional here: If the list is empty we need to do something special. Most of the time the list will of course not be empty. To eliminate the branch we can make sure that the list is never empty. We define a sentinel node (the "head") which always represents the beginning of the list. The tail pointer will start by pointing to this head node.
This means that both pushing and popping is completely branch-free and these are operations we do at every recursive function call.
A Future
in Spice has two possible states: It's either queued or executing.
The heartbeat is responsible for taking a queued future and start executing it.
And as we already know: Heartbeating happens rarely so we expect many futures to be queued without executing.
An early prototype of Spice used a tagged union to store the future on the stack. This turns out to be suboptimal because (1) stack usage matters for performance (at least in this benchmark) and (2) there's quite a lot of additional state needed to keep track of futures which are executing.
To minimize stack usage Spice therefore uses two techniques:
prev
pointer.
Whether the first field is null
therefore decides which of these it is.
(Maybe a smart enough compiler would be able to this optimization for us.)const Future = struct {
prev_or_null: ?*anyopaque,
next_or_state: ?*anyopaque,
}
// A future which is _queued_ has:
// prev_or_null = pointer to prev future
// next_or_state = pointer to next future
// A future which is _executing_ has:
// prev_or_null = null
// next_or_state = ExecuteState
const ExecuteState = struct {
requester: *Worker,
done: std.Thread.ResetEvent = .{},
result: ResultType,
// Any number of fields.
}
Spice works with a Task
struct which has two fields:
A pointer to the owning worker and a pointer to tail of the work queue.
For optimal performance these should be passed as registers across all function boundaries.
However, with LLVM, passing a struct will very often cause it be passed on the stack.
To work around this we define a separate function where worker
and job_tail
are actual parameters.
We place the parameters into a struct and pass a pointer to this into the user-defined function.
This function call we make sure is always being inlined:
fn callWithContext(
worker: *Worker,
job_tail: *Job,
comptime T: type,
func: anytype,
arg: anytype,
) T {
var t = Task{
.worker = worker,
.job_tail = job_tail,
};
return @call(.always_inline, func, .{
&t,
arg,
});
}
This causes the callWithContext
-function to be the actual function which LLVM works on, and since this has pointers are parameters it will happily pass these directly into registers.
The initial development of Spice has been focused around a single benchmark which is described in detail in bench/.
Spice was made possible thanks to the research into heartbeat scheduling:
"The best multicore-parallelization refactoring you've never heard of" gives an excellent introduction into the concepts of heartbeat scheduling. It's a very short paper which focuses entirely on a single use case, but describes everything in a manner which can be generalized. The solution presented in this paper is based around turning all the code into continuation-passing style which enables switching between sequential and parallel execution. Spice started out as an experiment of this approach, but this turned out to have quite high overhead (>10 nanosecond).
Going backwards in time, "Heartbeat scheduling: provable efficiency for nested parallelism" was the first paper introducing "heartbeat scheduling". This paper provides excellent information about the concepts, but the implementation is based around integrating this into an interpreter and focus is primarily on the theoretical guarantees as opposed to raw performance.
"Task parallel assembly language for uncompromising parallelism" is a follow-up paper which improves the performance by defining a custom assembly language and using OS signaling for heartbeats. This is a fascinating line of research, but it's difficult to integrate into an existing language.
There's many limitations of the current implementation of Spice:
fork
and join
).
If you're using it wrong now then weird things could happen.
This should be improved by adding more compile-time checking, debug-mode assertions, or changing the overall API.@panic
extensively.
To be considered a proper Zig library there needs to be way more consideration of how error cases are handled.ReleaseSafe
is an extremely nice feature of Zig.
Further benchmarking and testing is needed to understand how well Spice can work here.Luckily the whole codebase is ~500 lines so it shouldn't be too difficult to make progress on these areas.
There's currently no plans of doing any active development on Spice to improve this (as the original author don't have the time). Any improvements in forks and/or re-implementations in other languages are highly encouraged!
Question: Why is it called "Spice"?
Answer: This project enables fine-grained parallelism. Sand is extremely fine-grained. Sand forms in dunes. Spice. Also: It's a hot take on parallelism.
Question: Why is it implemented in Zig?
Answer: Why not? This describes a generic approach to parallelism that should be possible to implement in multiple languages. Maybe I'll end up implementing something similar in another language as well? I don't know yet. If you think this is interesting for your language of choice I would encourage you to explore this area.
Question: But if you did it in Rust we could have safe parallelism?
Answer: Yeah, that sounds very cool. I'm not at all opposed to it. That said, I've been exploring many different techniques and variants while developing Spice. Many of my initial ideas were definitely not "safe" by any means, but I was able to express these ideas in Zig, look at the assembly and measure the performance in benchmarks. I'd probably only be able to explore a fraction of the ideas if I was limited by Rust's strict semantics in the initial phase of this project. If I have to turn this into a production-ready system I might decide to use Rust.