use backend::{Backend, FFT64, MatZnxDft, MatZnxDftOps, Module}; use crate::{FourierGLWECiphertext, GGLWECiphertext, GetRow, Infos, SetRow}; pub struct GLWESwitchingKey { pub(crate) key: GGLWECiphertext, pub(crate) sk_in_n: usize, // Degree of sk_in pub(crate) sk_out_n: usize, // Degree of sk_out } impl GLWESwitchingKey, FFT64> { pub fn alloc( module: &Module, basek: usize, k: usize, rows: usize, digits: usize, rank_in: usize, rank_out: usize, ) -> Self { GLWESwitchingKey { key: GGLWECiphertext::alloc(module, basek, k, rows, digits, rank_in, rank_out), sk_in_n: 0, sk_out_n: 0, } } pub fn bytes_of( module: &Module, basek: usize, k: usize, rows: usize, digits: usize, rank_in: usize, rank_out: usize, ) -> usize { GGLWECiphertext::, FFT64>::bytes_of(module, basek, k, rows, digits, rank_in, rank_out) } } impl Infos for GLWESwitchingKey { type Inner = MatZnxDft; fn inner(&self) -> &Self::Inner { self.key.inner() } fn basek(&self) -> usize { self.key.basek() } fn k(&self) -> usize { self.key.k() } } impl GLWESwitchingKey { pub fn rank(&self) -> usize { self.key.data.cols_out() - 1 } pub fn rank_in(&self) -> usize { self.key.data.cols_in() } pub fn rank_out(&self) -> usize { self.key.data.cols_out() - 1 } pub fn digits(&self) -> usize { self.key.digits() } pub fn sk_degree_in(&self) -> usize { self.sk_in_n } pub fn sk_degree_out(&self) -> usize { self.sk_out_n } } impl> GetRow for GLWESwitchingKey { fn get_row + AsRef<[u8]>>( &self, module: &Module, row_i: usize, col_j: usize, res: &mut FourierGLWECiphertext, ) { module.mat_znx_dft_get_row(&mut res.data, &self.key.data, row_i, col_j); } } impl + AsRef<[u8]>> SetRow for GLWESwitchingKey { fn set_row>( &mut self, module: &Module, row_i: usize, col_j: usize, a: &FourierGLWECiphertext, ) { module.mat_znx_dft_set_row(&mut self.key.data, row_i, col_j, &a.data); } }