use core::ops::{Add, Neg};
use hex_literal::hex;
#[cfg(not(target_os = "zkvm"))]
use lazy_static::lazy_static;
#[cfg(not(target_os = "zkvm"))]
use num_bigint::BigUint;
use openvm_algebra_guest::{Field, IntMod};
use openvm_algebra_moduli_setup::moduli_declare;
use openvm_ecc_sw_setup::sw_declare;
use super::group::{CyclicGroup, Group};
use crate::weierstrass::{CachedMulTable, IntrinsicCurve};
#[cfg(not(target_os = "zkvm"))]
lazy_static! {
pub static ref SECP256K1_MODULUS: BigUint = BigUint::from_bytes_be(&hex!(
"FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFE FFFFFC2F"
));
pub static ref SECP256K1_ORDER: BigUint = BigUint::from_bytes_be(&hex!(
"FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFE BAAEDCE6 AF48A03B BFD25E8C D0364141"
));
}
pub const SECP256K1_NUM_LIMBS: usize = 32;
pub const SECP256K1_LIMB_BITS: usize = 8;
pub const SECP256K1_BLOCK_SIZE: usize = 32;
const CURVE_B: Secp256k1Coord = Secp256k1Coord::from_const_bytes(seven_le());
const fn seven_le() -> [u8; 32] {
let mut buf = [0u8; 32];
buf[0] = 7;
buf
}
moduli_declare! {
Secp256k1Coord { modulus = "0xFFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFE FFFFFC2F" },
Secp256k1Scalar { modulus = "0xFFFFFFFF FFFFFFFF FFFFFFFF FFFFFFFE BAAEDCE6 AF48A03B BFD25E8C D0364141" },
}
sw_declare! {
Secp256k1Point { mod_type = Secp256k1Coord, b = CURVE_B },
}
impl Field for Secp256k1Coord {
const ZERO: Self = <Self as IntMod>::ZERO;
const ONE: Self = <Self as IntMod>::ONE;
type SelfRef<'a> = &'a Self;
fn double_assign(&mut self) {
IntMod::double_assign(self);
}
fn square_assign(&mut self) {
IntMod::square_assign(self);
}
}
impl CyclicGroup for Secp256k1Point {
const GENERATOR: Self = Secp256k1Point {
x: Secp256k1Coord::from_const_bytes(hex!(
"9817F8165B81F259D928CE2DDBFC9B02070B87CE9562A055ACBBDCF97E66BE79"
)),
y: Secp256k1Coord::from_const_bytes(hex!(
"B8D410FB8FD0479C195485A648B417FDA808110EFCFBA45D65C4A32677DA3A48"
)),
};
const NEG_GENERATOR: Self = Secp256k1Point {
x: Secp256k1Coord::from_const_bytes(hex!(
"9817F8165B81F259D928CE2DDBFC9B02070B87CE9562A055ACBBDCF97E66BE79"
)),
y: Secp256k1Coord::from_const_bytes(hex!(
"7727EF046F2FB863E6AB7A59B74BE80257F7EEF103045BA29A3B5CD98825C5B7"
)),
};
}
impl IntrinsicCurve for k256::Secp256k1 {
type Scalar = Secp256k1Scalar;
type Point = Secp256k1Point;
fn msm(coeffs: &[Self::Scalar], bases: &[Self::Point]) -> Self::Point
where
for<'a> &'a Self::Point: Add<&'a Self::Point, Output = Self::Point>,
{
if coeffs.len() < 25 {
let table = CachedMulTable::<Self>::new_with_prime_order(bases, 4);
table.windowed_mul(coeffs)
} else {
crate::msm(coeffs, bases)
}
}
}