use poulpy_hal::{ api::{ModuleN, ScratchAvailable, ScratchTakeBasic, SvpPrepare, VecZnxSwitchRing}, layouts::{Backend, DataMut, Module, ScalarZnx, Scratch}, source::Source, }; use crate::{ ScratchTakeCore, encryption::gglwe_ct::GGLWEEncryptSk, layouts::{ GGLWEInfos, GLWEInfos, GLWESecret, GLWESecretToRef, GLWESwitchingKey, GLWESwitchingKeyToMut, LWEInfos, prepared::GLWESecretPreparedAlloc, }, }; impl GLWESwitchingKey> { pub fn encrypt_sk_tmp_bytes(module: &M, infos: &A) -> usize where A: GGLWEInfos, M: GLWESwitchingKeyEncryptSk, { module.glwe_switching_key_encrypt_sk_tmp_bytes(infos) } pub fn encrypt_pk_tmp_bytes(module: &M, infos: &A) -> usize where A: GGLWEInfos, M: GLWESwitchingKeyEncryptPk, { module.glwe_switching_key_encrypt_pk_tmp_bytes(infos) } } impl GLWESwitchingKey { pub fn encrypt_sk( &mut self, module: &M, sk_in: &S1, sk_out: &S2, source_xa: &mut Source, source_xe: &mut Source, scratch: &mut Scratch, ) where S1: GLWESecretToRef, S2: GLWESecretToRef, M: GLWESwitchingKeyEncryptSk, Scratch: ScratchTakeCore, { module.glwe_switching_key_encrypt_sk(self, sk_in, sk_out, source_xa, source_xe, scratch); } } pub trait GLWESwitchingKeyEncryptSk { fn glwe_switching_key_encrypt_sk_tmp_bytes(&self, infos: &A) -> usize where A: GGLWEInfos; fn glwe_switching_key_encrypt_sk( &self, res: &mut R, sk_in: &S1, sk_out: &S2, source_xa: &mut Source, source_xe: &mut Source, scratch: &mut Scratch, ) where R: GLWESwitchingKeyToMut, S1: GLWESecretToRef, S2: GLWESecretToRef, Scratch: ScratchTakeCore; } impl GLWESwitchingKeyEncryptSk for Module where Self: ModuleN + GGLWEEncryptSk + GLWESecretPreparedAlloc + VecZnxSwitchRing + SvpPrepare, Scratch: ScratchTakeCore, { fn glwe_switching_key_encrypt_sk_tmp_bytes(&self, infos: &A) -> usize where A: GGLWEInfos, { self.gglwe_encrypt_sk_tmp_bytes(infos) .max(ScalarZnx::bytes_of(self.n(), 1)) + ScalarZnx::bytes_of(self.n(), infos.rank_in().into()) + self.bytes_of_glwe_secret_prepared_from_infos(infos) } fn glwe_switching_key_encrypt_sk( &self, res: &mut R, sk_in: &S1, sk_out: &S2, source_xa: &mut Source, source_xe: &mut Source, scratch: &mut Scratch, ) where R: GLWESwitchingKeyToMut, S1: GLWESecretToRef, S2: GLWESecretToRef, { let res: &mut GLWESwitchingKey<&mut [u8]> = &mut res.to_mut(); let sk_in: &GLWESecret<&[u8]> = &sk_in.to_ref(); let sk_out: &GLWESecret<&[u8]> = &sk_out.to_ref(); assert!(sk_in.n().0 <= self.n() as u32); assert!(sk_out.n().0 <= self.n() as u32); assert!( scratch.available() >= self.glwe_switching_key_encrypt_sk_tmp_bytes(res), "scratch.available()={} < GLWESwitchingKey::encrypt_sk_tmp_bytes={}", scratch.available(), self.glwe_switching_key_encrypt_sk_tmp_bytes(res) ); let (mut sk_in_tmp, scratch_1) = scratch.take_scalar_znx(self, sk_in.rank().into()); for i in 0..sk_in.rank().into() { self.vec_znx_switch_ring( &mut sk_in_tmp.as_vec_znx_mut(), i, &sk_in.data.as_vec_znx(), i, ); } let (mut sk_out_tmp, scratch_2) = scratch_1.take_glwe_secret_prepared(self, sk_out.rank()); { let (mut tmp, _) = scratch_2.take_scalar_znx(self, 1); for i in 0..sk_out.rank().into() { self.vec_znx_switch_ring(&mut tmp.as_vec_znx_mut(), 0, &sk_out.data.as_vec_znx(), i); self.svp_prepare(&mut sk_out_tmp.data, i, &tmp, 0); } } res.key.encrypt_sk( self, &sk_in_tmp, &sk_out_tmp, source_xa, source_xe, scratch_2, ); res.sk_in_n = sk_in.n().into(); res.sk_out_n = sk_out.n().into(); } } pub trait GLWESwitchingKeyEncryptPk { fn glwe_switching_key_encrypt_pk_tmp_bytes(&self, infos: &A) -> usize where A: GGLWEInfos; } impl GLWESwitchingKeyEncryptPk for Module { fn glwe_switching_key_encrypt_pk_tmp_bytes(&self, _infos: &A) -> usize where A: GGLWEInfos, { unimplemented!() } }