Files
xenia-rs/migration/project-root/ppc-manual/memory/ld.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

10 KiB
Raw Blame History

ld — Load Doubleword

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

Assembler Mnemonics

Mnemonic XML entry Flags Description
ld ld Load Doubleword
ldu ldu Load Doubleword with Update
ldux ldux Load Doubleword with Update Indexed
ldx ldx Load Doubleword Indexed

Syntax

ld [RD], [ds]([RA0])
ldu [RD], [ds]([RA])
ldux [RD], [RA], [RB]
ldx [RD], [RA0], [RB]

Encoding

ld — form DS

  • Opcode word: 0xe8000000
  • Primary opcode (bits 05): 58
  • 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

ldu — form DS

  • Opcode word: 0xe8000001
  • Primary opcode (bits 05): 58
  • 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

ldux — form X

  • Opcode word: 0x7c00006a
  • Primary opcode (bits 05): 31
  • Extended opcode: 53
  • 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

ldx — form X

  • Opcode word: 0x7c00002a
  • Primary opcode (bits 05): 31
  • Extended opcode: 21
  • 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
RA0 ld: read; ldx: read Source GPR; when the encoded register number is 0 the operand is the literal 64-bit zero, not r0.
ds ld: read; ldu: read 14-bit signed word-aligned displacement (DS << 2).
RD ld: write; ldu: write; ldux: write; ldx: write Destination GPR.
RA ldu: read; ldu: write; ldux: read; ldux: write Source GPR (r0r31).
RB ldux: read; ldx: read Source GPR.

Register Effects

ld

  • Reads (always): RA0, ds
  • Reads (conditional): none
  • Writes (always): RD
  • Writes (conditional): none

ldu

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

ldux

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

ldx

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

Status-Register Effects

No condition-register or status-register effects.

Operation (pseudocode)

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

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

ld

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::ld => {
            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;
            ctx.gpr[instr.rd()] = mem.read_u64(ea);
            ctx.pc += 4;
        }

ldu

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::ldu => {
            let ea = ctx.gpr[instr.ra()].wrapping_add(instr.ds() as i64 as u64) as u32;
            ctx.gpr[instr.rd()] = mem.read_u64(ea);
            ctx.gpr[instr.ra()] = ea as u64;
            ctx.pc += 4;
        }

ldux

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::ldux => {
            let ea = ctx.gpr[instr.ra()].wrapping_add(ctx.gpr[instr.rb()]) as u32;
            ctx.gpr[instr.rd()] = mem.read_u64(ea);
            ctx.gpr[instr.ra()] = ea as u64;
            ctx.pc += 4;
        }

ldx

xenia-rs interpreter body (frozen snapshot)
        PpcOpcode::ldx => {
            let ea = if instr.ra() == 0 { 0u64 } else { ctx.gpr[instr.ra()] };
            let ea = ea.wrapping_add(ctx.gpr[instr.rb()]) as u32;
            ctx.gpr[instr.rd()] = mem.read_u64(ea);
            ctx.pc += 4;
        }

Special Cases & Edge Conditions

  • DS-form, not D-form. The displacement is 14 bits scaled by 4 (EXTS(ds || 0b00)), giving a signed range of ±32 KiB in 4-byte steps. Bits 3031 are the extended opcode used to distinguish ld (XO=0) from ldu (XO=1). The assembler accepts a normal byte displacement and verifies divisibility by 4.
  • Big-endian read. The 64 bits at EA..EA+7 form the loaded value, most-significant byte first. Xenia-rs's mem.read_u64 returns the host-native value of that big-endian doubleword.
  • No zero/sign-extension question. ld already fills the entire 64-bit register; there is no lda (load doubleword algebraic) — the doubleword is the architectural maximum.
  • RA0 (non-update forms). RA = 0 in ld and ldx means base is literal zero. ld RT, 0x100(0) reads from absolute 0x100.
  • Update-form invalid forms. ldu / ldux invoke "RA = 0" and "RA = RT" as invalid forms. AIX docs say results are undefined; xenia performs the read first, then writes back RA ← EA, which would silently destroy the loaded value if RA == RT.
  • Alignment. Xenon does not enforce doubleword alignment for ld itself — unaligned 8-byte loads are tolerated. However, real POWER cores may take an alignment exception on some implementations; portable code keeps doublewords 8-byte aligned.
  • 64-bit pointer / counter loads. Although Xbox 360 user code is 32-bit, kernel structures and TOC entries are doublewords; ld is the standard load for them.
  • lwz, lhz, lbz — narrower zero-extending loads.
  • lwa, lha — sign-extending loads (no lda exists; ld already fills the register).
  • ldbrx — byte-reversed doubleword load.
  • ldarx / stdcx — load-reserve / store-conditional doubleword pair.
  • std, stdu, stdx, stdux — corresponding stores.

IBM Reference