use std::fmt::Debug; use backend::hal::{ api::{FillUniform, VecZnxAlloc, VecZnxAllocBytes, VecZnxCopy, VecZnxFillUniform, ZnxInfos, ZnxZero}, layouts::{Backend, Data, DataMut, DataRef, Module, ReaderFrom, VecZnx, VecZnxToMut, VecZnxToRef, WriterTo}, }; use sampling::source::Source; use crate::{GLWEOps, Infos, SetMetaData}; #[derive(PartialEq, Eq, Clone)] pub struct GLWECiphertext { pub data: VecZnx, pub basek: usize, pub k: usize, } impl Debug for GLWECiphertext { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!( f, "GLWECiphertext: basek={} k={}: {}", self.basek(), self.k(), self.data ) } } impl ZnxZero for GLWECiphertext where VecZnx: ZnxZero, { fn zero(&mut self) { self.data.zero() } fn zero_at(&mut self, i: usize, j: usize) { self.data.zero_at(i, j); } } impl FillUniform for GLWECiphertext where VecZnx: FillUniform, { fn fill_uniform(&mut self, source: &mut Source) { self.data.fill_uniform(source); } } impl GLWECiphertext> { pub fn alloc(module: &Module, basek: usize, k: usize, rank: usize) -> Self where Module: VecZnxAlloc, { Self { data: module.vec_znx_alloc(rank + 1, k.div_ceil(basek)), basek, k, } } pub fn bytes_of(module: &Module, basek: usize, k: usize, rank: usize) -> usize where Module: VecZnxAllocBytes, { module.vec_znx_alloc_bytes(rank + 1, k.div_ceil(basek)) } } impl Infos for GLWECiphertext { type Inner = VecZnx; fn inner(&self) -> &Self::Inner { &self.data } fn basek(&self) -> usize { self.basek } fn k(&self) -> usize { self.k } } impl GLWECiphertext { pub fn rank(&self) -> usize { self.cols() - 1 } } impl GLWECiphertext { pub fn clone(&self) -> GLWECiphertext> { GLWECiphertext { data: self.data.clone(), basek: self.basek(), k: self.k(), } } } impl SetMetaData for GLWECiphertext { fn set_k(&mut self, k: usize) { self.k = k } fn set_basek(&mut self, basek: usize) { self.basek = basek } } pub trait GLWECiphertextToRef: Infos { fn to_ref(&self) -> GLWECiphertext<&[u8]>; } impl GLWECiphertextToRef for GLWECiphertext { fn to_ref(&self) -> GLWECiphertext<&[u8]> { GLWECiphertext { data: self.data.to_ref(), basek: self.basek, k: self.k, } } } pub trait GLWECiphertextToMut: Infos { fn to_mut(&mut self) -> GLWECiphertext<&mut [u8]>; } impl GLWECiphertextToMut for GLWECiphertext { fn to_mut(&mut self) -> GLWECiphertext<&mut [u8]> { GLWECiphertext { data: self.data.to_mut(), basek: self.basek, k: self.k, } } } impl GLWEOps for GLWECiphertext where GLWECiphertext: GLWECiphertextToMut + Infos + SetMetaData {} use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt}; impl ReaderFrom for GLWECiphertext { fn read_from(&mut self, reader: &mut R) -> std::io::Result<()> { self.k = reader.read_u64::()? as usize; self.basek = reader.read_u64::()? as usize; self.data.read_from(reader) } } impl WriterTo for GLWECiphertext { fn write_to(&self, writer: &mut W) -> std::io::Result<()> { writer.write_u64::(self.k as u64)?; writer.write_u64::(self.basek as u64)?; self.data.write_to(writer) } } #[derive(PartialEq, Eq, Clone)] pub struct GLWECiphertextCompressed { pub(crate) data: VecZnx, pub(crate) basek: usize, pub(crate) k: usize, pub(crate) rank: usize, pub(crate) seed: [u8; 32], } impl Debug for GLWECiphertextCompressed { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!( f, "GLWECiphertext: basek={} k={}: {}", self.basek(), self.k(), self.data ) } } impl ZnxZero for GLWECiphertextCompressed where VecZnx: ZnxZero, { fn zero(&mut self) { self.data.zero() } fn zero_at(&mut self, i: usize, j: usize) { self.data.zero_at(i, j); } } impl FillUniform for GLWECiphertextCompressed where VecZnx: FillUniform, { fn fill_uniform(&mut self, source: &mut Source) { self.data.fill_uniform(source); } } impl Infos for GLWECiphertextCompressed { type Inner = VecZnx; fn inner(&self) -> &Self::Inner { &self.data } fn basek(&self) -> usize { self.basek } fn k(&self) -> usize { self.k } } impl GLWECiphertextCompressed { pub fn rank(&self) -> usize { self.rank } } impl GLWECiphertextCompressed> { pub fn alloc(module: &Module, basek: usize, k: usize, rank: usize) -> Self where Module: VecZnxAlloc, { Self { data: module.vec_znx_alloc(1, k.div_ceil(basek)), basek, k, rank, seed: [0u8; 32], } } pub fn bytes_of(module: &Module, basek: usize, k: usize) -> usize where Module: VecZnxAllocBytes, { GLWECiphertext::bytes_of(module, basek, k, 1) } } impl ReaderFrom for GLWECiphertextCompressed { fn read_from(&mut self, reader: &mut R) -> std::io::Result<()> { self.k = reader.read_u64::()? as usize; self.basek = reader.read_u64::()? as usize; self.rank = reader.read_u64::()? as usize; reader.read(&mut self.seed)?; self.data.read_from(reader) } } impl WriterTo for GLWECiphertextCompressed { fn write_to(&self, writer: &mut W) -> std::io::Result<()> { writer.write_u64::(self.k as u64)?; writer.write_u64::(self.basek as u64)?; writer.write_u64::(self.rank as u64)?; writer.write_all(&self.seed)?; self.data.write_to(writer) } } impl GLWECiphertext { pub fn decompress(&mut self, module: &Module, other: &GLWECiphertextCompressed) where DataOther: DataRef, Module: VecZnxFillUniform + VecZnxCopy, { #[cfg(debug_assertions)] { assert_eq!( self.n(), other.data.n(), "invalid receiver: self.n()={} != other.n()={}", self.n(), other.data.n() ); assert_eq!( self.size(), other.size(), "invalid receiver: self.size()={} != other.size()={}", self.size(), other.size() ); assert_eq!( self.rank(), other.rank(), "invalid receiver: self.rank()={} != other.rank()={}", self.rank(), other.rank() ); let mut source: Source = Source::new(other.seed); self.decompress_internal(module, other, &mut source); } } pub(crate) fn decompress_internal( &mut self, module: &Module, other: &GLWECiphertextCompressed, source: &mut Source, ) where DataOther: DataRef, Module: VecZnxFillUniform + VecZnxCopy, { let k: usize = other.k; let basek: usize = other.basek; let cols: usize = other.cols(); module.vec_znx_copy(&mut self.data, 0, &other.data, 0); (1..cols).for_each(|i| { module.vec_znx_fill_uniform(basek, &mut self.data, i, k, source); }); self.basek = basek; self.k = k; } }