Establish English as the canonical ADR language with Korean translations held in a parallel docs/adr-ko/ tree as derived artifacts (1:1 mirror). Promotion from adr-proposed/ to adr/ now writes English to adr/ and the Korean to adr-ko/; bidirectional sync rule documented in CLAUDE.md. - Migrate 30 ADRs in docs/adr/: 28 Korean-only translated to English, 2 bilingual pairs (ADR-0020, ADR-0023) consolidated (.en.md suffix dropped). ADR-0023 EN regenerated against KO source which had newer HW Realization Notes (D16-D23) section. - docs/adr-history/ left frozen by design (transitional state). - CLAUDE.md (Part 2): update ADR Lifecycle for 4-folder layout, mark docs/adr-ko/ as a Derived Artifact, add ADR Translation Discipline section covering bidirectional sync, conflict resolution (EN wins), and proposed-language freedom. - tools/verify_adr_lang_pairs.py: new verification tool checking pair completeness, filename mirroring, ADR-ID match, Status byte-equality. Pre-commit hook intentionally not added; run on demand or in CI. - tests/test_verify_adr_lang_pairs.py: 11 cases including CRLF/LF normalization, em-dash title separator, underscore-slug edge case. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
8.2 KiB
ADR-0036: IO_CPU Component Model
Status
Accepted
Context
IO_CPU is the IO chiplet's host-facing endpoint inside the simulation graph. PCIE_EP receives host messages from the runtime API and routes them via the io_noc; for command-bearing requests (KernelLaunch, MmuMap/Unmap) the io_noc forwards to IO_CPU, which:
- Fans out the request to per-cube M_CPUs.
- Aggregates per-cube responses into a single host-visible completion.
- For kernel launches, stamps a global
target_start_nsbarrier so every PE across every targeted cube begins kernel body execution at the same simulated time (ADR-0009 D5).
Memory R/W traffic bypasses IO_CPU per ADR-0015 D4 / ADR-0016 D3; this component therefore handles only command-plane traffic in normal operation.
This ADR documents the IO_CPU component implementation that realizes those responsibilities.
Decision
D1. Role
IO_CPU is the host-facing endpoint of the IO chiplet. It has two primary responsibilities:
- Multi-cube fan-out — distribute KernelLaunchMsg / MmuMapMsg / MmuUnmapMsg to per-cube M_CPUs.
- Response aggregation — collect per-cube ResponseMsg, signal
parent
txn.donewhen all targeted cubes have responded.
A third, narrower responsibility applies only to KernelLaunchMsg:
target_start_ns global barrier stamping (D3).
The component does not:
- Decide routing — paths are pre-computed by the router (ADR-0002).
- Decode tensor or kernel internals — those concerns belong to M_CPU / PE_CPU / engines.
- Handle PE-level fan-out — M_CPU fans out within a cube (ADR-0009 D3).
- Handle Memory R/W data path — those bypass IO_CPU per ADR-0015 D4
and ADR-0016 D3 (Memory R/W resolution code in
_resolve_cube_targetsexists as a defensive fallback only).
Per invocation (run()): applies the configured overhead_ns once
per incoming Transaction (D8).
D2. Forward path — multi-cube fan-out
When a non-response Transaction arrives, the worker:
- Pays
overhead_nsviarun(). - Calls
_resolve_cube_targetsto derive the list of(sip, cube)targets from the request (D5). - For each target:
- Resolves M_CPU node id via
ctx.resolver.find_m_cpu(sip, cube). - Resolves the path via
ctx.router.find_node_path(io_cpu, m_cpu). - Creates a per-cube sub-Transaction with
pathpopulated and forwards it topath[1](the first hop on the io_noc).
- Resolves M_CPU node id via
- Registers aggregation state:
_pending[request_id] = (expected, received=0, parent_done).
D3. KernelLaunch target_start_ns global barrier (ADR-0009 D5)
IO_CPU is the canonical stamper for target_start_ns. When the
request is a KernelLaunchMsg, IO_CPU computes a single global
barrier covering every targeted PE across every targeted cube:
for (sip, cube) in cube_targets:
leg1 = compute_path_latency_ns(io_cpu → m_cpu(sip, cube), nbytes=0)
for pe_id in target_pe_ids:
leg2 = compute_path_latency_ns(m_cpu → pe_cpu(sip, cube, pe_id),
nbytes=0)
latency = leg1 + leg2 - io_overhead_ns - m_overhead_ns
global_max = max(global_max, latency)
target_start_ns = env.now + global_max
The request is then replaced (via dataclasses.replace) so the
stamped value propagates through the fan-out.
Two overhead corrections:
io_overhead_nsis subtracted because IO_CPU has already paid it inrun()before this method runs.m_overhead_nsis subtracted once because it appears as the endpoint of leg1 and the start of leg2 in path latency, but M_CPU pays it only once at run time.
Every downstream PE_CPU yields until target_start_ns before
beginning kernel body execution; all PEs therefore start at the same
simulated time regardless of how long their individual dispatch path
took.
D4. KernelLaunch sub-Transactions carry nbytes=0
Per-cube sub-Transactions for KernelLaunchMsg force nbytes=0,
overriding the parent txn.nbytes:
- Kernel launch is a control message; payload size is irrelevant at the data-fabric level.
- If
nbytes > 0, every per-cube sub-txn occupies fabric BW on the io_noc's shared first hop. With 16 cubes this serializes fan-out, pushing far M_CPUs pasttarget_start_nsand breaking the D3 invariant.
Non-KernelLaunch sub-Transactions preserve txn.nbytes (only relevant
for the defensive Memory R/W fallback path, which carries actual
payload sizes).
D5. Per-request-type cube target resolution
_resolve_cube_targets dispatches by request type:
| Request type | Source of (sip, cube) |
target_cubes="all" semantics |
|---|---|---|
MemoryWriteMsg |
dst_sip, dst_cube (or PhysAddr.decode(dst_pa).die_id fallback) |
single cube derived from PA decode |
MemoryReadMsg |
src_sip, src_cube (or PhysAddr.decode(src_pa).die_id fallback) |
single cube derived from PA decode |
KernelLaunchMsg |
tensor shards filtered by shard.sip == my_sip |
every cube that owns a shard on this SIP |
MmuMapMsg / MmuUnmapMsg |
target_cubes list, filtered to this SIP |
range(cubes_per_sip) from spec |
Each IO_CPU instance fans out only within its own SIP — _my_sip()
parses the SIP id from the node id (e.g., sip0.io0.io_cpu → 0).
The Memory R/W rows exist for defensive completeness; the engine's
normal path routes Memory R/W via _process_memory_direct() /
find_memory_path(), bypassing IO_CPU entirely (ADR-0015 D4 /
ADR-0016 D3).
D6. Response aggregation
_pending: dict[request_id → (expected, received, parent_done)]:
- On dispatch: register
(len(cube_targets), 0, txn.done). _workerrecognises responses byis_response=Trueand routes them to_collect_response._collect_responseincrementsreceived; whenreceived >= expected,parent_done.succeed()is invoked and the entry is removed from_pending.
This is a simple per-request counter. There is no per-cube identity tracking and no partial-failure handling — a missing response indefinitely stalls the parent done. Production-style failure paths are out of scope for the current simulator model.
D7. target_pe resolution helper
_resolve_pe_ids(target_pe):
int→[target_pe].tuple[int, ...]→list(target_pe)."all"→range(n_slices), wheren_slicescomes from cubememory_map.hbm_slices_per_cube(default 8).
Used in D3's barrier computation to enumerate every PE target per cube.
D8. Configurable overhead_ns
A single attribute drives per-instance latency:
| Site | impl name | overhead_ns |
|---|---|---|
IO chiplet io_cpu |
builtin.io_cpu |
10.0 |
Applied once in run() per Transaction. Models command
interpretation + dispatch-decision time at IO_CPU.
Consequences
Positive
- Cross-cube and cross-SIP kernel launches share a single global barrier (D3 + D4) — no per-cube divergence in start time.
- nbytes=0 invariant keeps fan-out off the shared first-hop fabric BW, preserving the barrier's accuracy at scale (16 cubes).
- Response aggregation via a single counter → minimal state, deterministic ordering of completion.
- Per-SIP scoping (
_my_sip()) keeps IO_CPUs in different SIPs cleanly independent.
Negative
- No partial-failure semantics — a missing per-cube response indefinitely stalls the parent. Adequate for simulation but not suitable as a production-style endpoint.
_pendingis a regular dict; in-flight requests accumulate state. Acceptable for current benchmark workloads (few concurrent outstanding launches); unbounded in principle.- The Memory R/W resolution branches in
_resolve_cube_targetsare dead code in the normal engine path. Kept defensively but invite drift if the bypass path ever changes.
Links
- ADR-0002 (Routing distance — path computation)
- ADR-0009 D1 (Kernel launch is an endpoint request to IO_CPU)
- ADR-0009 D3 (M_CPU fans out within a cube; IO_CPU fans out across cubes)
- ADR-0009 D5 (target_start_ns canonical stamping at IO_CPU)
- ADR-0011 D-VA3 (MmuMapMsg routes through IO_CPU for cube fan-out)
- ADR-0012 (Host ↔ IO_CPU message schema)
- ADR-0015 D4 (Memory R/W bypasses IO_CPU; Kernel Launch via IO_CPU)
- ADR-0016 D1 (IO chiplet io_noc — IO_CPU attaches here)
- ADR-0016 D3 (Memory R/W path bypasses IO_CPU)
- ADR-0016 D4 (Kernel Launch path through IO_CPU for command interpretation)