use backend::hal::{ api::{ ScratchOwnedAlloc, ScratchOwnedBorrow, VecZnxAddScalarInplace, VecZnxAutomorphismInplace, VecZnxEncodeCoeffsi64, VecZnxSwithcDegree, ZnxView, }, layouts::{Backend, Module, ScratchOwned}, oep::{ ScratchAvailableImpl, ScratchOwnedAllocImpl, ScratchOwnedBorrowImpl, TakeScalarZnxImpl, TakeSvpPPolImpl, TakeVecZnxBigImpl, TakeVecZnxDftImpl, TakeVecZnxImpl, }, }; use sampling::source::Source; use crate::{ GGLWEEncryptSkFamily, GGLWEExecLayoutFamily, GLWECiphertext, GLWEDecryptFamily, GLWEKeyswitchFamily, GLWEPlaintext, GLWESecret, GLWESecretExec, GLWEToLWESwitchingKey, GLWEToLWESwitchingKeyExec, Infos, LWECiphertext, LWEPlaintext, LWESecret, LWESwitchingKey, LWESwitchingKeyExec, LWEToGLWESwitchingKey, LWEToGLWESwitchingKeyExec, }; pub(crate) trait LWETestModuleFamily = GGLWEEncryptSkFamily + GLWEDecryptFamily + VecZnxSwithcDegree + VecZnxAddScalarInplace + GGLWEExecLayoutFamily + GLWEKeyswitchFamily + VecZnxEncodeCoeffsi64 + VecZnxAutomorphismInplace; pub(crate) trait LWETestScratchFamily = TakeScalarZnxImpl + TakeVecZnxDftImpl + ScratchAvailableImpl + TakeVecZnxImpl + TakeVecZnxBigImpl + TakeSvpPPolImpl + ScratchOwnedAllocImpl + ScratchOwnedBorrowImpl; pub(crate) fn test_lwe_to_glwe(module: &Module) where Module: LWETestModuleFamily, B: LWETestScratchFamily, { let n: usize = module.n(); let basek: usize = 17; let sigma: f64 = 3.2; let rank: usize = 2; let n_lwe: usize = 22; let k_lwe_ct: usize = 2 * basek; let k_lwe_pt: usize = 8; let k_glwe_ct: usize = 3 * basek; let k_ksk: usize = k_lwe_ct + basek; let mut source_xs: Source = Source::new([0u8; 32]); let mut source_xa: Source = Source::new([0u8; 32]); let mut source_xe: Source = Source::new([0u8; 32]); let mut scratch: ScratchOwned = ScratchOwned::alloc( LWEToGLWESwitchingKey::encrypt_sk_scratch_space(module, n, basek, k_ksk, rank) | GLWECiphertext::from_lwe_scratch_space(module, n, basek, k_lwe_ct, k_glwe_ct, k_ksk, rank) | GLWECiphertext::decrypt_scratch_space(module, n, basek, k_glwe_ct), ); let mut sk_glwe: GLWESecret> = GLWESecret::alloc(n, rank); sk_glwe.fill_ternary_prob(0.5, &mut source_xs); let sk_glwe_exec: GLWESecretExec, B> = GLWESecretExec::from(module, &sk_glwe); let mut sk_lwe: LWESecret> = LWESecret::alloc(n_lwe); sk_lwe.fill_ternary_prob(0.5, &mut source_xs); let data: i64 = 17; let mut lwe_pt: LWEPlaintext> = LWEPlaintext::alloc(basek, k_lwe_pt); module.encode_coeff_i64(basek, &mut lwe_pt.data, 0, k_lwe_pt, 0, data, k_lwe_pt); let mut lwe_ct: LWECiphertext> = LWECiphertext::alloc(n_lwe, basek, k_lwe_ct); lwe_ct.encrypt_sk( module, &lwe_pt, &sk_lwe, &mut source_xa, &mut source_xe, sigma, ); let mut ksk: LWEToGLWESwitchingKey> = LWEToGLWESwitchingKey::alloc(n, basek, k_ksk, lwe_ct.size(), rank); ksk.encrypt_sk( module, &sk_lwe, &sk_glwe, &mut source_xa, &mut source_xe, sigma, scratch.borrow(), ); let mut glwe_ct: GLWECiphertext> = GLWECiphertext::alloc(n, basek, k_glwe_ct, rank); let ksk_exec: LWEToGLWESwitchingKeyExec, B> = LWEToGLWESwitchingKeyExec::from(module, &ksk, scratch.borrow()); glwe_ct.from_lwe(module, &lwe_ct, &ksk_exec, scratch.borrow()); let mut glwe_pt: GLWEPlaintext> = GLWEPlaintext::alloc(n, basek, k_glwe_ct); glwe_ct.decrypt(module, &mut glwe_pt, &sk_glwe_exec, scratch.borrow()); assert_eq!(glwe_pt.data.at(0, 0)[0], lwe_pt.data.at(0, 0)[0]); } pub(crate) fn test_glwe_to_lwe(module: &Module) where Module: LWETestModuleFamily, B: LWETestScratchFamily, { let n: usize = module.n(); let basek: usize = 17; let sigma: f64 = 3.2; let rank: usize = 2; let n_lwe: usize = 22; let k_lwe_ct: usize = 2 * basek; let k_lwe_pt: usize = 8; let k_glwe_ct: usize = 3 * basek; let k_ksk: usize = k_lwe_ct + basek; let mut source_xs: Source = Source::new([0u8; 32]); let mut source_xa: Source = Source::new([0u8; 32]); let mut source_xe: Source = Source::new([0u8; 32]); let mut scratch: ScratchOwned = ScratchOwned::alloc( LWEToGLWESwitchingKey::encrypt_sk_scratch_space(module, n, basek, k_ksk, rank) | LWECiphertext::from_glwe_scratch_space(module, n, basek, k_lwe_ct, k_glwe_ct, k_ksk, rank) | GLWECiphertext::decrypt_scratch_space(module, n, basek, k_glwe_ct), ); let mut sk_glwe: GLWESecret> = GLWESecret::alloc(n, rank); sk_glwe.fill_ternary_prob(0.5, &mut source_xs); let sk_glwe_exec: GLWESecretExec, B> = GLWESecretExec::from(module, &sk_glwe); let mut sk_lwe = LWESecret::alloc(n_lwe); sk_lwe.fill_ternary_prob(0.5, &mut source_xs); let data: i64 = 17; let mut glwe_pt: GLWEPlaintext> = GLWEPlaintext::alloc(n, basek, k_glwe_ct); module.encode_coeff_i64(basek, &mut glwe_pt.data, 0, k_lwe_pt, 0, data, k_lwe_pt); let mut glwe_ct = GLWECiphertext::alloc(n, basek, k_glwe_ct, rank); glwe_ct.encrypt_sk( module, &glwe_pt, &sk_glwe_exec, &mut source_xa, &mut source_xe, sigma, scratch.borrow(), ); let mut ksk: GLWEToLWESwitchingKey> = GLWEToLWESwitchingKey::alloc(n, basek, k_ksk, glwe_ct.size(), rank); ksk.encrypt_sk( module, &sk_lwe, &sk_glwe, &mut source_xa, &mut source_xe, sigma, scratch.borrow(), ); let mut lwe_ct: LWECiphertext> = LWECiphertext::alloc(n_lwe, basek, k_lwe_ct); let ksk_exec: GLWEToLWESwitchingKeyExec, B> = GLWEToLWESwitchingKeyExec::from(module, &ksk, scratch.borrow()); lwe_ct.from_glwe(module, &glwe_ct, &ksk_exec, scratch.borrow()); let mut lwe_pt: LWEPlaintext> = LWEPlaintext::alloc(basek, k_lwe_ct); lwe_ct.decrypt(module, &mut lwe_pt, &sk_lwe); assert_eq!(glwe_pt.data.at(0, 0)[0], lwe_pt.data.at(0, 0)[0]); } pub(crate) fn test_keyswitch(module: &Module) where Module: LWETestModuleFamily, B: LWETestScratchFamily, { let n: usize = module.n(); let basek: usize = 17; let sigma: f64 = 3.2; let n_lwe_in: usize = 22; let n_lwe_out: usize = 30; let k_lwe_ct: usize = 2 * basek; let k_lwe_pt: usize = 8; let k_ksk: usize = k_lwe_ct + basek; let mut source_xs: Source = Source::new([0u8; 32]); let mut source_xa: Source = Source::new([0u8; 32]); let mut source_xe: Source = Source::new([0u8; 32]); let mut scratch: ScratchOwned = ScratchOwned::alloc( LWESwitchingKey::encrypt_sk_scratch_space(module, n, basek, k_ksk) | LWECiphertext::keyswitch_scratch_space(module, n, basek, k_lwe_ct, k_lwe_ct, k_ksk), ); let mut sk_lwe_in: LWESecret> = LWESecret::alloc(n_lwe_in); sk_lwe_in.fill_ternary_prob(0.5, &mut source_xs); let mut sk_lwe_out: LWESecret> = LWESecret::alloc(n_lwe_out); sk_lwe_out.fill_ternary_prob(0.5, &mut source_xs); let data: i64 = 17; let mut lwe_pt_in: LWEPlaintext> = LWEPlaintext::alloc(basek, k_lwe_pt); module.encode_coeff_i64(basek, &mut lwe_pt_in.data, 0, k_lwe_pt, 0, data, k_lwe_pt); let mut lwe_ct_in: LWECiphertext> = LWECiphertext::alloc(n_lwe_in, basek, k_lwe_ct); lwe_ct_in.encrypt_sk( module, &lwe_pt_in, &sk_lwe_in, &mut source_xa, &mut source_xe, sigma, ); let mut ksk: LWESwitchingKey> = LWESwitchingKey::alloc(n, basek, k_ksk, lwe_ct_in.size()); ksk.encrypt_sk( module, &sk_lwe_in, &sk_lwe_out, &mut source_xa, &mut source_xe, sigma, scratch.borrow(), ); let mut lwe_ct_out: LWECiphertext> = LWECiphertext::alloc(n_lwe_out, basek, k_lwe_ct); let ksk_exec: LWESwitchingKeyExec, B> = LWESwitchingKeyExec::from(module, &ksk, scratch.borrow()); lwe_ct_out.keyswitch(module, &lwe_ct_in, &ksk_exec, scratch.borrow()); let mut lwe_pt_out: LWEPlaintext> = LWEPlaintext::alloc(basek, k_lwe_ct); lwe_ct_out.decrypt(module, &mut lwe_pt_out, &sk_lwe_out); assert_eq!(lwe_pt_in.data.at(0, 0)[0], lwe_pt_out.data.at(0, 0)[0]); }