Skip to content

Commit

Permalink
apply SignedExtendConfig to the places needed msb/is_neg config
Browse files Browse the repository at this point in the history
  • Loading branch information
KimiWu123 committed Nov 1, 2024
1 parent 7990a10 commit 652ac42
Show file tree
Hide file tree
Showing 4 changed files with 33 additions and 56 deletions.
43 changes: 18 additions & 25 deletions ceno_zkvm/src/instructions/riscv/mulh.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,12 +7,12 @@ use crate::{
circuit_builder::CircuitBuilder,
error::ZKVMError,
expression::{Expression, ToExpr},
gadgets::IsLtConfig,
gadgets::SignedExtendConfig,
instructions::{
Instruction,
riscv::{
RIVInstruction,
constants::{BIT_WIDTH, LIMB_BITS, UInt, UIntMul},
constants::{BIT_WIDTH, UInt, UIntMul},
r_insn::RInstructionConfig,
},
},
Expand Down Expand Up @@ -256,7 +256,7 @@ impl<E: ExtensionField> Instruction<E> for MulhInstruction<E> {
/// corresponding signed value, interpreting the bits as a 2s-complement
/// encoding. Gadget allocates 2 `WitIn` values in total.
struct Signed<E: ExtensionField> {
pub is_negative: IsLtConfig,
pub is_negative: SignedExtendConfig,
val: Expression<E>,
}

Expand All @@ -266,23 +266,16 @@ impl<E: ExtensionField> Signed<E> {
name_fn: N,
unsigned_val: &UInt<E>,
) -> Result<Self, ZKVMError> {
cb.namespace(
|| "signed",
|cb| {
let name = name_fn();
// is_lt is set if top limb of val is negative
let is_negative = IsLtConfig::construct_circuit(
cb,
|| name.clone(),
((1u64 << (LIMB_BITS - 1)) - 1).into(),
unsigned_val.expr().last().unwrap().clone(),
1,
)?;
let val = unsigned_val.value() - (1u64 << BIT_WIDTH) * is_negative.expr();

Ok(Self { is_negative, val })
},
)
cb.namespace(name_fn, |cb| {
// is_lt is set if top limb of val is negative
let is_negative = SignedExtendConfig::construct_limb(
cb,
unsigned_val.limbs.iter().last().unwrap().expr(),
)?;
let val = unsigned_val.value() - (1u64 << BIT_WIDTH) * is_negative.expr();

Ok(Self { is_negative, val })
})
}

pub fn assign_instance(
Expand All @@ -291,11 +284,11 @@ impl<E: ExtensionField> Signed<E> {
lkm: &mut LkMultiplicity,
val: &Value<u32>,
) -> Result<i32, ZKVMError> {
let high_limb = *val.limbs.last().unwrap() as u64;
let sign_cutoff = (1u64 << (LIMB_BITS - 1)) - 1;
self.is_negative
.assign_instance(instance, lkm, sign_cutoff, high_limb)?;

self.is_negative.assign_instance::<E>(
instance,
lkm,
*val.as_u16_limbs().last().unwrap() as u64,
)?;
let signed_val = val.as_u32() as i32;

Ok(signed_val)
Expand Down
15 changes: 6 additions & 9 deletions ceno_zkvm/src/instructions/riscv/shift_imm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ use crate::{
circuit_builder::CircuitBuilder,
error::ZKVMError,
expression::{Expression, ToExpr, WitIn},
gadgets::{AssertLTConfig, IsLtConfig},
gadgets::{AssertLTConfig, SignedExtendConfig},
instructions::{
Instruction,
riscv::{constants::UInt, i_insn::IInstructionConfig},
Expand All @@ -26,7 +26,7 @@ pub struct ShiftImmConfig<E: ExtensionField> {
assert_lt_config: AssertLTConfig,

// SRAI
is_lt_config: Option<IsLtConfig>,
is_lt_config: Option<SignedExtendConfig>,
}

pub struct ShiftImmInstruction<E, I>(PhantomData<(E, I)>);
Expand Down Expand Up @@ -84,10 +84,9 @@ impl<E: ExtensionField, I: RIVInstruction> Instruction<E> for ShiftImmInstructio
InsnKind::SRAI | InsnKind::SRLI => {
let (inflow, is_lt_config) = match I::INST_KIND {
InsnKind::SRAI => {
let is_rs1_neg = rs1_read.is_negative(circuit_builder, || "lhs_msb")?;
let msb_expr: Expression<E> = is_rs1_neg.is_lt.expr();
let is_rs1_neg = rs1_read.is_negative(circuit_builder)?;
let ones = imm.expr() - 1;
(msb_expr * ones, Some(is_rs1_neg))
(is_rs1_neg.expr() * ones, Some(is_rs1_neg))
}
InsnKind::SRLI => (Expression::ZERO, None),
_ => unreachable!(),
Expand Down Expand Up @@ -140,12 +139,10 @@ impl<E: ExtensionField, I: RIVInstruction> Instruction<E> for ShiftImmInstructio
InsnKind::SLLI => (rs1_read.as_u64() * imm as u64) >> UInt::<E>::TOTAL_BITS,
InsnKind::SRAI | InsnKind::SRLI => {
if I::INST_KIND == InsnKind::SRAI {
let max_signed_limb_expr = (1 << (UInt::<E>::LIMB_BITS - 1)) - 1;
config.is_lt_config.as_ref().unwrap().assign_instance(
config.is_lt_config.as_ref().unwrap().assign_instance::<E>(
instance,
lk_multiplicity,
max_signed_limb_expr,
rs1_read.as_u64() >> UInt::<E>::LIMB_BITS,
*rs1_read.as_u16_limbs().last().unwrap() as u64,
)?;
}

Expand Down
12 changes: 5 additions & 7 deletions ceno_zkvm/src/instructions/riscv/slti.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ use crate::{
circuit_builder::CircuitBuilder,
error::ZKVMError,
expression::{ToExpr, WitIn},
gadgets::IsLtConfig,
gadgets::{IsLtConfig, SignedExtendConfig},
instructions::Instruction,
set_val,
tables::InsnRecord,
Expand All @@ -32,7 +32,7 @@ pub struct SetLessThanImmConfig<E: ExtensionField> {
lt: IsLtConfig,

// SLTI
is_rs1_neg: Option<IsLtConfig>,
is_rs1_neg: Option<SignedExtendConfig>,
}

pub struct SetLessThanImmInstruction<E, I>(PhantomData<(E, I)>);
Expand Down Expand Up @@ -62,7 +62,7 @@ impl<E: ExtensionField, I: RIVInstruction> Instruction<E> for SetLessThanImmInst
let (value_expr, is_rs1_neg) = match I::INST_KIND {
InsnKind::SLTIU => (rs1_read.value(), None),
InsnKind::SLTI => {
let is_rs1_neg = rs1_read.is_negative(cb, || "lhs_msb")?;
let is_rs1_neg = rs1_read.is_negative(cb)?;
(rs1_read.to_field_expr(is_rs1_neg.expr()), Some(is_rs1_neg))
}
_ => unreachable!("Unsupported instruction kind {:?}", I::INST_KIND),
Expand Down Expand Up @@ -100,7 +100,6 @@ impl<E: ExtensionField, I: RIVInstruction> Instruction<E> for SetLessThanImmInst
config.i_insn.assign_instance(instance, lkm, step)?;

let rs1 = step.rs1().unwrap().value;
let max_signed_limb = (1u64 << (UInt::<E>::LIMB_BITS - 1)) - 1;
let rs1_value = Value::new_unchecked(rs1 as Word);
config
.rs1_read
Expand All @@ -117,11 +116,10 @@ impl<E: ExtensionField, I: RIVInstruction> Instruction<E> for SetLessThanImmInst
.assign_instance(instance, lkm, rs1 as u64, imm as u64)?;
}
InsnKind::SLTI => {
config.is_rs1_neg.as_ref().unwrap().assign_instance(
config.is_rs1_neg.as_ref().unwrap().assign_instance::<E>(
instance,
lkm,
max_signed_limb,
*rs1_value.limbs.last().unwrap() as u64,
*rs1_value.as_u16_limbs().last().unwrap() as u64,
)?;
config
.lt
Expand Down
19 changes: 4 additions & 15 deletions ceno_zkvm/src/uint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,8 @@ use crate::{
circuit_builder::CircuitBuilder,
error::{UtilError, ZKVMError},
expression::{Expression, ToExpr, WitIn},
gadgets::{AssertLTConfig, IsLtConfig},
instructions::riscv::constants::{LIMB_BITS, UInt},
gadgets::{AssertLTConfig, SignedExtendConfig},
instructions::riscv::constants::UInt,
utils::add_one_to_big_num,
witness::LkMultiplicity,
};
Expand All @@ -21,7 +21,6 @@ use goldilocks::SmallField;
use itertools::Itertools;
use std::{
borrow::Cow,
fmt::Display,
mem::{self, MaybeUninit},
ops::Index,
};
Expand Down Expand Up @@ -541,18 +540,8 @@ impl<E: ExtensionField> UInt<E> {
///
/// Also called Most Significant Bit extraction, when
/// interpreted as an unsigned int.
pub fn is_negative<NR: Into<String> + Display + Clone, N: FnOnce() -> NR>(
&self,
cb: &mut CircuitBuilder<E>,
name_fn: N,
) -> Result<IsLtConfig, ZKVMError> {
IsLtConfig::construct_circuit(
cb,
name_fn,
((1u64 << (UInt::<E>::LIMB_BITS - 1)) - 1).into(),
self.expr().last().unwrap().clone(),
LIMB_BITS.div_ceil(LIMB_BITS),
)
pub fn is_negative(&self, cb: &mut CircuitBuilder<E>) -> Result<SignedExtendConfig, ZKVMError> {
SignedExtendConfig::construct_limb(cb, self.limbs.iter().last().unwrap().expr())
}
}

Expand Down

0 comments on commit 652ac42

Please sign in to comment.