Files
xenia-rs/migration/project-root/ppc-manual/memory/std.md
MechaCat02 e6d43a23ac chore: add migration/ bundle for cross-machine setup
Bundles state that lives OUTSIDE the xenia-rs repo so a fresh clone on
another machine can be brought up to identical configuration via
migration/setup.sh:

  - claude-memory/             ~/.claude/projects/-home-fabi-RE-Project-Sylpheed/memory/
                               (103 files, 1.1 MB - MEMORY.md + every
                                project_xenia_rs_*.md from audits
                                addis_signext through audit-058)
  - project-root/dot-claude/   <project-root>/.claude/settings.json
                               (Stop hook + permissions)
  - project-root/ppc-manual/   <project-root>/ppc-manual/
                               (PowerPC reference docs, 397 files, 3.7 MB)
  - project-root/run-canary.sh <project-root>/run-canary.sh
  - README.md                  Human-readable setup checklist
  - setup.sh                   Idempotent installer (also reclones
                               xenia-canary at pinned HEAD 6de80dffe)
  - MANIFEST.md                Per-file mapping + per-file-not-bundled
                               restoration recipe

Excluded from bundle (not shippable via git):
  - Sylpheed ISO (7.8 GB; copyright; manual copy required)
  - sylpheed.db (395 MB; regenerable from XEX via analysis tooling)
  - target/ build artifacts (rebuild on target)
  - audit-runs probe firehoses (.log/.stdout/.stderr ~11 GB; rerun if needed)
  - audit-runs memory dumps (.bin ~4.5 GB; rerun audit-026/027/029 if needed)
  - xenia-canary checkout (setup.sh reclones from
    git.mc02.dev/fabi/Xenia-Canary.git at HEAD 6de80dffe)

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-10 21:38:38 +02:00

11 KiB
Raw Permalink Blame History

std — Store Doubleword

Category: Memory · Form: DS · Opcode: 0xf8000000

Assembler Mnemonics

Mnemonic XML entry Flags Description
std std Store Doubleword
stdu stdu Store Doubleword with Update
stdux stdux Store Doubleword with Update Indexed
stdx stdx Store Doubleword Indexed

Syntax

std [RS], [ds]([RA0])
stdu [RS], [ds]([RA])
stdux [RS], [RA], [RB]
stdx [RS], [RA0], [RB]

Encoding

std — form DS

  • Opcode word: 0xf8000000
  • Primary opcode (bits 05): 62
  • Extended opcode:
  • Synchronising: no
Bits Field Meaning
05 OPCD primary opcode
610 RT destination GPR (or RS)
1115 RA source GPR (0 ⇒ literal 0)
1629 DS 14-bit signed word-scaled displacement
3031 XO extended opcode

stdu — form DS

  • Opcode word: 0xf8000001
  • Primary opcode (bits 05): 62
  • Extended opcode:
  • Synchronising: no
Bits Field Meaning
05 OPCD primary opcode
610 RT destination GPR (or RS)
1115 RA source GPR (0 ⇒ literal 0)
1629 DS 14-bit signed word-scaled displacement
3031 XO extended opcode

stdux — form X

  • Opcode word: 0x7c00016a
  • Primary opcode (bits 05): 31
  • Extended opcode: 181
  • Synchronising: no
Bits Field Meaning
05 OPCD primary opcode
610 RT/FRT/VRT destination
1115 RA/FRA/VRA source A
1620 RB/FRB/VRB source B
2130 XO extended opcode (10 bits)
31 Rc record-form flag

stdx — form X

  • Opcode word: 0x7c00012a
  • Primary opcode (bits 05): 31
  • Extended opcode: 149
  • Synchronising: no
Bits Field Meaning
05 OPCD primary opcode
610 RT/FRT/VRT destination
1115 RA/FRA/VRA source A
1620 RB/FRB/VRB source B
2130 XO extended opcode (10 bits)
31 Rc record-form flag

Operands

Field Role Description
RS std: read; stdu: read; stdux: read; stdx: read Source GPR (alias for RD in some stores).
RA std: read; stdu: read; stdu: write; stdux: read; stdux: write Source GPR (r0r31).
ds std: read; stdu: read 14-bit signed word-aligned displacement (DS << 2).
RB stdux: read; stdx: read Source GPR.
RA0 stdx: read Source GPR; when the encoded register number is 0 the operand is the literal 64-bit zero, not r0.

Register Effects

std

  • Reads (always): RS, RA, ds
  • Reads (conditional): none
  • Writes (always): none
  • Writes (conditional): none

stdu

  • Reads (always): RS, RA, ds
  • Reads (conditional): none
  • Writes (always): RA
  • Writes (conditional): none

stdux

  • Reads (always): RS, RA, RB
  • Reads (conditional): none
  • Writes (always): RA
  • Writes (conditional): none

stdx

  • Reads (always): RS, RA0, RB
  • Reads (conditional): none
  • Writes (always): none
  • Writes (conditional): none

Status-Register Effects

No condition-register or status-register effects.

Operation (pseudocode)

EA <- (RA|0) + EXTS(ds || 0b00)
MEM(EA, 8) <- (RS)

C Translation Example

/* C translation: the xenia-rs interpreter arm below in           */
/* Implementation References is the authoritative semantic        */
/* snapshot. Translate it line-by-line:                            */
/*   - ctx.gpr[N]  -> r[N]       (or f[]/v[] for FPRs/VRs)        */
/*   - mem.read_u*/write_u* -> mem_read_u*_be / mem_write_u*_be   */
/*   - ctx.update_cr_signed(fld, v) -> update_cr_signed(fld, v)   */
/*   - ctx.xer_ca / xer_ov / xer_so -> xer.CA / xer.OV / xer.SO   */
/* The Register Effects and Status-Register Effects tables above  */
/* enumerate every side effect a faithful translation must emit.  */

Implementation References

std

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::std => {
            let ea = if instr.ra() == 0 { 0u64 } else { ctx.gpr[instr.ra()] };
            let ea = ea.wrapping_add(instr.ds() as i64 as u64) as u32;
            if let Some(t) = ctx.reservation_table.as_ref().filter(|t| t.is_enabled()) {
                if t.has_active_reservers() { t.invalidate_for_write(ea); }
            }
            mem.write_u64(ea, ctx.gpr[instr.rs()]);
            ctx.pc += 4;
        }

stdu

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::stdu => {
            let ea = ctx.gpr[instr.ra()].wrapping_add(instr.ds() as i64 as u64) as u32;
            if let Some(t) = ctx.reservation_table.as_ref().filter(|t| t.is_enabled()) {
                if t.has_active_reservers() { t.invalidate_for_write(ea); }
            }
            mem.write_u64(ea, ctx.gpr[instr.rs()]);
            ctx.gpr[instr.ra()] = ea as u64;
            ctx.pc += 4;
        }

stdux

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::stdux => {
            let ea = ctx.gpr[instr.ra()].wrapping_add(ctx.gpr[instr.rb()]) as u32;
            if let Some(t) = ctx.reservation_table.as_ref().filter(|t| t.is_enabled()) {
                if t.has_active_reservers() { t.invalidate_for_write(ea); }
            }
            mem.write_u64(ea, ctx.gpr[instr.rs()]);
            ctx.gpr[instr.ra()] = ea as u64;
            ctx.pc += 4;
        }

stdx

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::stdx => {
            let ea = if instr.ra() == 0 { 0u64 } else { ctx.gpr[instr.ra()] };
            let ea = ea.wrapping_add(ctx.gpr[instr.rb()]) as u32;
            if let Some(t) = ctx.reservation_table.as_ref().filter(|t| t.is_enabled()) {
                if t.has_active_reservers() { t.invalidate_for_write(ea); }
            }
            mem.write_u64(ea, ctx.gpr[instr.rs()]);
            ctx.pc += 4;
        }

Special Cases & Edge Conditions

  • DS-form, not D-form. Like ld, std uses a 14-bit signed displacement scaled by 4 (EXTS(ds || 0b00)). Bits 3031 are the extended opcode used to distinguish std (XO=0) from stdu (XO=1). Assemblers verify the byte displacement is a multiple of 4.
  • Big-endian write. The 64-bit value of RS is written most-significant-byte-first: RS[0:7] to EA, RS[56:63] to EA+7. Xenia's mem.write_u64 performs the host-side byte swap if needed.
  • RA0 for std and stdx. When RA = 0, base is the literal zero. Update forms stdu / stdux invoke RA = 0 as an invalid form (no RA = RS collision possible — RS and RA are independent encoding fields, and even if equal the store reads RS first).
  • Update-form post-write. stdu / stdux write EA to RA after the store. Order is store-then-update.
  • Alignment. Xenon tolerates unaligned doubleword stores. PowerISA permits implementations to raise alignment exceptions; portable code keeps doublewords 8-byte aligned. Cache-inhibited storage may force alignment.
  • Cache-line behaviour. A doubleword store fits inside one Xenon cache line (128 B), so it's a single line write. A doubleword store that straddles a line boundary triggers two line accesses — keep doublewords 8-byte aligned to avoid the cost.
  • 64-bit pointer / counter stores. Xbox 360 user code is 32-bit, but kernel structures, TOC entries, and 64-bit counters are stored with std.
  • stw, sth, stb — narrower integer stores.
  • stdbrx — byte-reversed doubleword store.
  • stdcx — store-conditional (the doubleword reservation pair end).
  • ld, ldu, ldx, ldux — corresponding loads.
  • stfd — FP doubleword store (same width, different register file).

IBM Reference