use ark_ec::{short_weierstrass::Affine, AffineRepr, CurveConfig};
use mina_curves::pasta::curves::{
pallas::{LegacyPallasParameters, PallasParameters},
vesta::{LegacyVestaParameters, VestaParameters},
};
use mina_poseidon::poseidon::ArithmeticSpongeParams;
use once_cell::sync::Lazy;
use poly_commitment::{
commitment::{CommitmentCurve, EndoCurve},
ipa::endos,
};
pub trait KimchiCurve: CommitmentCurve + EndoCurve {
const NAME: &'static str;
fn sponge_params() -> &'static ArithmeticSpongeParams<Self::ScalarField>;
fn other_curve_sponge_params() -> &'static ArithmeticSpongeParams<Self::BaseField>;
fn endos() -> &'static (Self::BaseField, Self::ScalarField);
fn other_curve_endo() -> &'static Self::ScalarField;
fn other_curve_generator() -> (Self::ScalarField, Self::ScalarField);
}
pub fn vesta_endos() -> &'static (
<VestaParameters as CurveConfig>::BaseField,
<VestaParameters as CurveConfig>::ScalarField,
) {
static VESTA_ENDOS: Lazy<(
<VestaParameters as CurveConfig>::BaseField,
<VestaParameters as CurveConfig>::ScalarField,
)> = Lazy::new(endos::<Affine<VestaParameters>>);
&VESTA_ENDOS
}
pub fn pallas_endos() -> &'static (
<PallasParameters as CurveConfig>::BaseField,
<PallasParameters as CurveConfig>::ScalarField,
) {
static PALLAS_ENDOS: Lazy<(
<PallasParameters as CurveConfig>::BaseField,
<PallasParameters as CurveConfig>::ScalarField,
)> = Lazy::new(endos::<Affine<PallasParameters>>);
&PALLAS_ENDOS
}
impl KimchiCurve for Affine<VestaParameters> {
const NAME: &'static str = "vesta";
fn sponge_params() -> &'static ArithmeticSpongeParams<Self::ScalarField> {
mina_poseidon::pasta::fp_kimchi::static_params()
}
fn other_curve_sponge_params() -> &'static ArithmeticSpongeParams<Self::BaseField> {
mina_poseidon::pasta::fq_kimchi::static_params()
}
fn endos() -> &'static (Self::BaseField, Self::ScalarField) {
vesta_endos()
}
fn other_curve_endo() -> &'static Self::ScalarField {
&pallas_endos().0
}
fn other_curve_generator() -> (Self::ScalarField, Self::ScalarField) {
Affine::<PallasParameters>::generator()
.to_coordinates()
.unwrap()
}
}
impl KimchiCurve for Affine<PallasParameters> {
const NAME: &'static str = "pallas";
fn sponge_params() -> &'static ArithmeticSpongeParams<Self::ScalarField> {
mina_poseidon::pasta::fq_kimchi::static_params()
}
fn other_curve_sponge_params() -> &'static ArithmeticSpongeParams<Self::BaseField> {
mina_poseidon::pasta::fp_kimchi::static_params()
}
fn endos() -> &'static (Self::BaseField, Self::ScalarField) {
pallas_endos()
}
fn other_curve_endo() -> &'static Self::ScalarField {
&vesta_endos().0
}
fn other_curve_generator() -> (Self::ScalarField, Self::ScalarField) {
Affine::<VestaParameters>::generator()
.to_coordinates()
.unwrap()
}
}
impl KimchiCurve for Affine<LegacyVestaParameters> {
const NAME: &'static str = "legacy_vesta";
fn sponge_params() -> &'static ArithmeticSpongeParams<Self::ScalarField> {
mina_poseidon::pasta::fp_legacy::static_params()
}
fn other_curve_sponge_params() -> &'static ArithmeticSpongeParams<Self::BaseField> {
mina_poseidon::pasta::fq_legacy::static_params()
}
fn endos() -> &'static (Self::BaseField, Self::ScalarField) {
vesta_endos()
}
fn other_curve_endo() -> &'static Self::ScalarField {
&pallas_endos().0
}
fn other_curve_generator() -> (Self::ScalarField, Self::ScalarField) {
Affine::<PallasParameters>::generator()
.to_coordinates()
.unwrap()
}
}
impl KimchiCurve for Affine<LegacyPallasParameters> {
const NAME: &'static str = "legacy_pallas";
fn sponge_params() -> &'static ArithmeticSpongeParams<Self::ScalarField> {
mina_poseidon::pasta::fq_legacy::static_params()
}
fn other_curve_sponge_params() -> &'static ArithmeticSpongeParams<Self::BaseField> {
mina_poseidon::pasta::fp_legacy::static_params()
}
fn endos() -> &'static (Self::BaseField, Self::ScalarField) {
pallas_endos()
}
fn other_curve_endo() -> &'static Self::ScalarField {
&vesta_endos().0
}
fn other_curve_generator() -> (Self::ScalarField, Self::ScalarField) {
Affine::<VestaParameters>::generator()
.to_coordinates()
.unwrap()
}
}
#[cfg(feature = "bn254")]
use mina_poseidon::dummy_values::kimchi_dummy;
#[cfg(feature = "bn254")]
impl KimchiCurve for Affine<ark_bn254::g1::Config> {
const NAME: &'static str = "bn254";
fn sponge_params() -> &'static ArithmeticSpongeParams<Self::ScalarField> {
static PARAMS: Lazy<ArithmeticSpongeParams<ark_bn254::Fr>> = Lazy::new(kimchi_dummy);
&PARAMS
}
fn other_curve_sponge_params() -> &'static ArithmeticSpongeParams<Self::BaseField> {
static PARAMS: Lazy<ArithmeticSpongeParams<ark_bn254::Fq>> = Lazy::new(kimchi_dummy);
&PARAMS
}
fn endos() -> &'static (Self::BaseField, Self::ScalarField) {
static ENDOS: Lazy<(ark_bn254::Fq, ark_bn254::Fr)> =
Lazy::new(endos::<ark_bn254::G1Affine>);
&ENDOS
}
fn other_curve_endo() -> &'static Self::ScalarField {
static ENDO: Lazy<ark_bn254::Fr> = Lazy::new(|| 13u64.into());
&ENDO
}
fn other_curve_generator() -> (Self::ScalarField, Self::ScalarField) {
(44u64.into(), 88u64.into())
}
}