4.0 KiB
Ring Benchmark Results
Generated by lib/erlang/bench_ring.sh against sx_server.exe on the
synchronous Erlang-on-SX scheduler.
| N (processes) | Hops | Wall-clock | Throughput |
|---|---|---|---|
| 10 | 10 | 907ms | 11 hops/s |
| 50 | 50 | 2107ms | 24 hops/s |
| 100 | 100 | 3827ms | 26 hops/s |
| 500 | 500 | 17004ms | 29 hops/s |
| 1000 | 1000 | 29832ms | 34 hops/s |
(Each Nm row spawns N processes connected in a ring and passes a
single token N hops total — i.e. the token completes one full lap.)
Status of the 1M-process target
Phase 3's stretch goal in plans/erlang-on-sx.md is a million-process
ring benchmark. That target is not met in the current synchronous
scheduler; extrapolating from the table above, 1M hops would take
~30 000 s. Correctness is fine — the program runs at every measured
size — but throughput is bound by per-hop overhead.
Per-hop cost is dominated by:
er-env-copyper fun clause attempt (whole-dict copy each time)call/cccapture +raise/guardunwind on everyreceiveer-q-delete-at!rebuilds the mailbox backing list on every matchdict-set!/dict-has?lookups in the global processes table
To reach 1M-process throughput in this architecture would need at least: persistent (path-copying) envs, an inline scheduler that doesn't call/cc on the common path (msg-already-in-mailbox), and a linked-list mailbox. None of those are in scope for the Phase 3 checkbox — captured here as the floor we're starting from.
Phase 9 status (2026-05-14)
Specialized opcodes 9b–9f landed as stub dispatchers in
lib/erlang/vm/dispatcher.sx: OP_PATTERN_TUPLE/LIST/BINARY,
OP_PERFORM/HANDLE, OP_RECEIVE_SCAN, OP_SPAWN/SEND, and ten
OP_BIF_* hot dispatch entries. Each opcode's handler is a thin
wrapper over the existing er-match-* / er-bif-* / runtime impls,
so the perf numbers above are unchanged — same per-hop cost, same
scheduler. The stubs exist to nail down opcode IDs, operand contracts,
and tests against er-match! parity before 9a (the OCaml
opcode-extension mechanism in hosts/ocaml/evaluator/) lands.
When 9a integrates and the bytecode compiler can emit these opcodes at hot call sites, the real speedup story (~3000× ring throughput, ~1000× spawn) starts. Until then this file documents the pre-integration ceiling. 72 vm-suite tests guard the stub correctness; full conformance is 709/709 with the stub infrastructure loaded.
Phase 9g — post-integration bench (2026-05-15)
9a (vm-ext mechanism), 9h (erlang_ext.ml registering erlang.OP_*
ids 222-239), and 9i (SX dispatcher consulting extension-opcode-id)
are now integrated and built into hosts/ocaml/_build/default/bin/sx_server.exe.
Re-ran the ring ladder on that binary:
| N (processes) | Hops | Wall-clock | Throughput |
|---|---|---|---|
| 10 | 10 | 938ms | 11 hops/s |
| 100 | 100 | 2772ms | 36 hops/s |
| 500 | 500 | 14190ms | 35 hops/s |
| 1000 | 1000 | 31814ms | 31 hops/s |
Numbers are unchanged from the pre-integration baseline — and that
is the expected, correct result. The opcode handlers (both the SX stub
dispatcher and the OCaml erlang_ext module) wrap the existing
er-match-* / er-bif-* / scheduler implementations 1-to-1, and the
bytecode compiler does not yet emit erlang.OP_* opcodes, so every
hop still goes through the general CEK path exactly as before. The
unchanged numbers therefore double as a no-regression check: the full
extension wiring (cherry-picked vm-ext A-E + force-link + erlang_ext +
SX bridge) added zero per-hop cost. Conformance 715/715 on this
binary.
The ~3000×/~1000× targets remain gated on a future phase (Phase 10 —
bytecode emission): teach lib/compiler.sx (or the Erlang
transpiler) to emit erlang.OP_PATTERN_TUPLE etc. at hot call sites,
then give erlang_ext.ml real register-machine handlers instead of the
current honest not-wired raise. That is a substantial standalone phase,
tracked in plans/erlang-on-sx.md. 9g's deliverable — honest
measurement + recorded numbers on the integrated binary — is complete.