Skip to content

Commit

Permalink
Explain conversion of AUIPC and LUI
Browse files Browse the repository at this point in the history
  • Loading branch information
matthiasgoergens committed Dec 14, 2024
1 parent 0cd4234 commit 6fa2529
Showing 1 changed file with 27 additions and 0 deletions.
27 changes: 27 additions & 0 deletions ceno_emul/src/disassemble/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -250,6 +250,14 @@ impl InstructionProcessor for InstructionTranspiler {
}

/// Convert LUI to ADDI.
///
/// RiscV's load-upper-immediate instruction is necessary to build arbitrary constants,
/// because its ADDI can only have a relatively small immediate value: there's just not
/// enough space in the 32 bits for more.
///
/// Our internal ADDI does not have this limitation, so we can convert LUI to ADDI.
/// See [`InstructionTranspiler::process_auipc`] for more background on the conversion.
///
fn process_lui(&mut self, dec_insn: UType) -> Self::InstructionResult {
// Verify assumption that the immediate is already shifted left by 12 bits.
assert_eq!(dec_insn.imm & 0xfff, 0);
Expand All @@ -264,6 +272,25 @@ impl InstructionProcessor for InstructionTranspiler {
}

/// Convert AUIPC to ADDI.
///
/// RiscV's instructions are designed to be (mosty) position-independent. AUIPC is used
/// to get access to the current program counter, even if the code has been moved around
/// by the linker.
///
/// Our conversion here happens after the linker has done its job, so we can safely hardcode
/// the current program counter into the immediate value of our internal ADDI.
///
/// Note that our internal ADDI can have arbitrary intermediate values, not just 12 bits.
///
/// ADDI is slightly more general than LUI or AUIPC, because you can also specify an
/// input register rs1. That generality might cost us sligthtly in the non-recursive proof,
/// but we suspect dropping the number of instructions will speed up the recursive proof.
///
/// In any case, AUIPC and LUI together make up ~0.1% of instructions executed in typical
/// real world scenarios like a `reth` run.
///
/// TODO(Matthias): run benchmarks to verify the impact on recursion, once we have a working
/// recursion.
fn process_auipc(&mut self, dec_insn: UType) -> Self::InstructionResult {
let pc = self.pc;
// Verify our assumption that the immediate is already shifted left by 12 bits.
Expand Down

0 comments on commit 6fa2529

Please sign in to comment.