use poulpy_hal::{ api::{ ScratchAvailable, SvpApplyDftToDftInplace, SvpPPolAllocBytes, SvpPrepare, TakeScalarZnx, TakeVecZnx, TakeVecZnxDft, VecZnxAddInplace, VecZnxAddNormal, VecZnxAddScalarInplace, VecZnxAutomorphism, VecZnxBigNormalize, VecZnxDftAllocBytes, VecZnxDftApply, VecZnxFillUniform, VecZnxIdftApplyConsume, VecZnxNormalize, VecZnxNormalizeInplace, VecZnxNormalizeTmpBytes, VecZnxSub, VecZnxSubInplace, VecZnxSwitchRing, }, layouts::{Backend, DataMut, Module, Scratch}, source::Source, }; use crate::{ TakeGLWESecret, TakeGLWESecretPrepared, layouts::{ GGLWEAutomorphismKey, GGLWEAutomorphismKeyToMut, GGLWEInfos, GGLWESwitchingKey, GLWEInfos, GLWESecret, GLWESecretToRef, LWEInfos, }, }; impl GGLWEAutomorphismKey> { pub fn encrypt_sk_scratch_space(module: &Module, infos: &A) -> usize where A: GGLWEInfos, Module: SvpPPolAllocBytes + VecZnxNormalizeTmpBytes + VecZnxDftAllocBytes + VecZnxNormalizeTmpBytes, { assert_eq!( infos.rank_in(), infos.rank_out(), "rank_in != rank_out is not supported for GGLWEAutomorphismKey" ); GGLWESwitchingKey::encrypt_sk_scratch_space(module, infos) + GLWESecret::alloc_bytes(&infos.glwe_layout()) } pub fn encrypt_pk_scratch_space(module: &Module, _infos: &A) -> usize where A: GGLWEInfos, { assert_eq!( _infos.rank_in(), _infos.rank_out(), "rank_in != rank_out is not supported for GGLWEAutomorphismKey" ); GGLWESwitchingKey::encrypt_pk_scratch_space(module, _infos) } } pub trait GGLWEAutomorphismKeyEncryptSk { fn gglwe_automorphism_key_encrypt_sk( &self, res: &mut A, p: i64, sk: &B, source_xa: &mut Source, source_xe: &mut Source, scratch: &mut Scratch, ) where A: GGLWEAutomorphismKeyToMut, B: GLWESecretToRef; } impl GGLWEAutomorphismKey where Self: GGLWEAutomorphismKeyToMut, { pub fn encrypt_sk( &mut self, module: &Module, p: i64, sk: &S, source_xa: &mut Source, source_xe: &mut Source, scratch: &mut Scratch, ) where S: GLWESecretToRef, Module: GGLWEAutomorphismKeyEncryptSk, { module.gglwe_automorphism_key_encrypt_sk(self, p, sk, source_xa, source_xe, scratch); } } impl GGLWEAutomorphismKeyEncryptSk for Module where Module: VecZnxAddScalarInplace + VecZnxDftAllocBytes + VecZnxBigNormalize + VecZnxDftApply + SvpApplyDftToDftInplace + VecZnxIdftApplyConsume + VecZnxNormalizeTmpBytes + VecZnxFillUniform + VecZnxSubInplace + VecZnxAddInplace + VecZnxNormalizeInplace + VecZnxAddNormal + VecZnxNormalize + VecZnxSub + SvpPrepare + VecZnxSwitchRing + SvpPPolAllocBytes + VecZnxAutomorphism, Scratch: TakeVecZnxDft + ScratchAvailable + TakeVecZnx + TakeScalarZnx + TakeGLWESecretPrepared, { fn gglwe_automorphism_key_encrypt_sk( &self, res: &mut A, p: i64, sk: &B, source_xa: &mut Source, source_xe: &mut Source, scratch: &mut Scratch, ) where A: GGLWEAutomorphismKeyToMut, B: GLWESecretToRef, { let res: &mut GGLWEAutomorphismKey<&mut [u8]> = &mut res.to_mut(); let sk: &GLWESecret<&[u8]> = &sk.to_ref(); #[cfg(debug_assertions)] { use crate::layouts::{GLWEInfos, LWEInfos}; assert_eq!(res.n(), sk.n()); assert_eq!(res.rank_out(), res.rank_in()); assert_eq!(sk.rank(), res.rank_out()); assert!( scratch.available() >= GGLWEAutomorphismKey::encrypt_sk_scratch_space(self, res), "scratch.available(): {} < AutomorphismKey::encrypt_sk_scratch_space: {:?}", scratch.available(), GGLWEAutomorphismKey::encrypt_sk_scratch_space(self, res) ) } let (mut sk_out, scratch_1) = scratch.take_glwe_secret(sk.n(), sk.rank()); { (0..res.rank_out().into()).for_each(|i| { self.vec_znx_automorphism( self.galois_element_inv(p), &mut sk_out.data.as_vec_znx_mut(), i, &sk.data.as_vec_znx(), i, ); }); } res.key .encrypt_sk(self, sk, &sk_out, source_xa, source_xe, scratch_1); res.p = p; } }