mirror of
https://github.com/arnaucube/poulpy.git
synced 2026-02-10 13:16:44 +01:00
added tests for sampling (and indirectly stats)
This commit is contained in:
8
base2k/.vscode/settings.json
vendored
Normal file
8
base2k/.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"github.copilot.enable": {
|
||||||
|
"*": false,
|
||||||
|
"plaintext": false,
|
||||||
|
"markdown": false,
|
||||||
|
"scminput": false
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -271,9 +271,9 @@ mod tests {
|
|||||||
let n: usize = 8;
|
let n: usize = 8;
|
||||||
let module: Module<FFT64> = Module::<FFT64>::new(n);
|
let module: Module<FFT64> = Module::<FFT64>::new(n);
|
||||||
let log_base2k: usize = 17;
|
let log_base2k: usize = 17;
|
||||||
let cols: usize = 5;
|
let limbs: usize = 5;
|
||||||
let log_k: usize = cols * log_base2k - 5;
|
let log_k: usize = limbs * log_base2k - 5;
|
||||||
let mut a: VecZnx = VecZnx::new(&module, 2, cols);
|
let mut a: VecZnx = VecZnx::new(&module, 2, limbs);
|
||||||
let mut source: Source = Source::new([0u8; 32]);
|
let mut source: Source = Source::new([0u8; 32]);
|
||||||
let raw: &mut [i64] = a.raw_mut();
|
let raw: &mut [i64] = a.raw_mut();
|
||||||
raw.iter_mut().enumerate().for_each(|(i, x)| *x = i as i64);
|
raw.iter_mut().enumerate().for_each(|(i, x)| *x = i as i64);
|
||||||
@@ -293,9 +293,9 @@ mod tests {
|
|||||||
let n: usize = 8;
|
let n: usize = 8;
|
||||||
let module: Module<FFT64> = Module::<FFT64>::new(n);
|
let module: Module<FFT64> = Module::<FFT64>::new(n);
|
||||||
let log_base2k: usize = 17;
|
let log_base2k: usize = 17;
|
||||||
let cols: usize = 5;
|
let limbs: usize = 5;
|
||||||
let log_k: usize = cols * log_base2k - 5;
|
let log_k: usize = limbs * log_base2k - 5;
|
||||||
let mut a: VecZnx = VecZnx::new(&module, 2, cols);
|
let mut a: VecZnx = VecZnx::new(&module, 2, limbs);
|
||||||
let mut source = Source::new([0u8; 32]);
|
let mut source = Source::new([0u8; 32]);
|
||||||
let raw: &mut [i64] = a.raw_mut();
|
let raw: &mut [i64] = a.raw_mut();
|
||||||
raw.iter_mut().enumerate().for_each(|(i, x)| *x = i as i64);
|
raw.iter_mut().enumerate().for_each(|(i, x)| *x = i as i64);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use crate::{Backend, Module, VecZnx, ZnxInfos, ZnxLayout};
|
use crate::{Backend, Module, VecZnx, ZnxLayout};
|
||||||
use rand_distr::{Distribution, Normal};
|
use rand_distr::{Distribution, Normal};
|
||||||
use sampling::source::Source;
|
use sampling::source::Source;
|
||||||
|
|
||||||
@@ -59,12 +59,11 @@ impl<B: Backend> Sampling for Module<B> {
|
|||||||
(bound.log2().ceil() as i64)
|
(bound.log2().ceil() as i64)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
let limb: usize = (log_k + log_base2k - 1) / log_base2k - 1;
|
||||||
let log_base2k_rem: usize = log_k % log_base2k;
|
let log_base2k_rem: usize = log_k % log_base2k;
|
||||||
|
|
||||||
if log_base2k_rem != 0 {
|
if log_base2k_rem != 0 {
|
||||||
a.at_poly_mut(col_i, a.limbs() - 1)
|
a.at_poly_mut(col_i, limb).iter_mut().for_each(|a| {
|
||||||
.iter_mut()
|
|
||||||
.for_each(|a| {
|
|
||||||
let mut dist_f64: f64 = dist.sample(source);
|
let mut dist_f64: f64 = dist.sample(source);
|
||||||
while dist_f64.abs() > bound {
|
while dist_f64.abs() > bound {
|
||||||
dist_f64 = dist.sample(source)
|
dist_f64 = dist.sample(source)
|
||||||
@@ -72,9 +71,7 @@ impl<B: Backend> Sampling for Module<B> {
|
|||||||
*a += (dist_f64.round() as i64) << log_base2k_rem;
|
*a += (dist_f64.round() as i64) << log_base2k_rem;
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
a.at_poly_mut(col_i, a.limbs() - 1)
|
a.at_poly_mut(col_i, limb).iter_mut().for_each(|a| {
|
||||||
.iter_mut()
|
|
||||||
.for_each(|a| {
|
|
||||||
let mut dist_f64: f64 = dist.sample(source);
|
let mut dist_f64: f64 = dist.sample(source);
|
||||||
while dist_f64.abs() > bound {
|
while dist_f64.abs() > bound {
|
||||||
dist_f64 = dist.sample(source)
|
dist_f64 = dist.sample(source)
|
||||||
@@ -105,3 +102,70 @@ impl<B: Backend> Sampling for Module<B> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::Sampling;
|
||||||
|
use crate::{FFT64, Module, Stats, VecZnx, ZnxBase, ZnxLayout};
|
||||||
|
use sampling::source::Source;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn fill_uniform() {
|
||||||
|
let n: usize = 4096;
|
||||||
|
let module: Module<FFT64> = Module::<FFT64>::new(n);
|
||||||
|
let log_base2k: usize = 17;
|
||||||
|
let limbs: usize = 5;
|
||||||
|
let mut source: Source = Source::new([0u8; 32]);
|
||||||
|
let cols: usize = 2;
|
||||||
|
let zero: Vec<i64> = vec![0; n];
|
||||||
|
let one_12_sqrt: f64 = 0.28867513459481287;
|
||||||
|
(0..cols).for_each(|col_i| {
|
||||||
|
let mut a: VecZnx = VecZnx::new(&module, cols, limbs);
|
||||||
|
module.fill_uniform(log_base2k, &mut a, col_i, limbs, &mut source);
|
||||||
|
(0..cols).for_each(|col_j| {
|
||||||
|
if col_j != col_i {
|
||||||
|
(0..limbs).for_each(|limb_i| {
|
||||||
|
assert_eq!(a.at_poly(col_j, limb_i), zero);
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
let std: f64 = a.std(col_i, log_base2k);
|
||||||
|
assert!(
|
||||||
|
(std - one_12_sqrt).abs() < 0.01,
|
||||||
|
"std={} ~!= {}",
|
||||||
|
std,
|
||||||
|
one_12_sqrt
|
||||||
|
);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn add_normal() {
|
||||||
|
let n: usize = 4096;
|
||||||
|
let module: Module<FFT64> = Module::<FFT64>::new(n);
|
||||||
|
let log_base2k: usize = 17;
|
||||||
|
let log_k: usize = 2 * 17;
|
||||||
|
let limbs: usize = 5;
|
||||||
|
let sigma: f64 = 3.2;
|
||||||
|
let bound: f64 = 6.0 * sigma;
|
||||||
|
let mut source: Source = Source::new([0u8; 32]);
|
||||||
|
let cols: usize = 2;
|
||||||
|
let zero: Vec<i64> = vec![0; n];
|
||||||
|
let k_f64: f64 = (1u64 << log_k as u64) as f64;
|
||||||
|
(0..cols).for_each(|col_i| {
|
||||||
|
let mut a: VecZnx = VecZnx::new(&module, cols, limbs);
|
||||||
|
module.add_normal(log_base2k, &mut a, col_i, log_k, &mut source, sigma, bound);
|
||||||
|
(0..cols).for_each(|col_j| {
|
||||||
|
if col_j != col_i {
|
||||||
|
(0..limbs).for_each(|limb_i| {
|
||||||
|
assert_eq!(a.at_poly(col_j, limb_i), zero);
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
let std: f64 = a.std(col_i, log_base2k) * k_f64;
|
||||||
|
assert!((std - sigma).abs() < 0.1, "std={} ~!= {}", std, sigma);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -3,11 +3,16 @@ use rug::Float;
|
|||||||
use rug::float::Round;
|
use rug::float::Round;
|
||||||
use rug::ops::{AddAssignRound, DivAssignRound, SubAssignRound};
|
use rug::ops::{AddAssignRound, DivAssignRound, SubAssignRound};
|
||||||
|
|
||||||
impl VecZnx {
|
pub trait Stats {
|
||||||
pub fn std(&self, poly_idx: usize, log_base2k: usize) -> f64 {
|
/// Returns the standard devaition of the i-th polynomial.
|
||||||
let prec: u32 = (self.cols() * log_base2k) as u32;
|
fn std(&self, col_i: usize, log_base2k: usize) -> f64;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Stats for VecZnx {
|
||||||
|
fn std(&self, col_i: usize, log_base2k: usize) -> f64 {
|
||||||
|
let prec: u32 = (self.limbs() * log_base2k) as u32;
|
||||||
let mut data: Vec<Float> = (0..self.n()).map(|_| Float::with_val(prec, 0)).collect();
|
let mut data: Vec<Float> = (0..self.n()).map(|_| Float::with_val(prec, 0)).collect();
|
||||||
self.decode_vec_float(poly_idx, log_base2k, &mut data);
|
self.decode_vec_float(col_i, log_base2k, &mut data);
|
||||||
// std = sqrt(sum((xi - avg)^2) / n)
|
// std = sqrt(sum((xi - avg)^2) / n)
|
||||||
let mut avg: Float = Float::with_val(prec, 0);
|
let mut avg: Float = Float::with_val(prec, 0);
|
||||||
data.iter().for_each(|x| {
|
data.iter().for_each(|x| {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
use crate::ffi::vec_znx_big::{self, vec_znx_big_t};
|
use crate::ffi::vec_znx_big::{self, vec_znx_big_t};
|
||||||
use crate::{Backend, FFT64, Module, VecZnx, VecZnxDft, ZnxBase, ZnxInfos, ZnxLayout, alloc_aligned, assert_alignement};
|
use crate::{Backend, FFT64, Module, VecZnx, ZnxBase, ZnxInfos, ZnxLayout, alloc_aligned, assert_alignement};
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
pub struct VecZnxBig<B: Backend> {
|
pub struct VecZnxBig<B: Backend> {
|
||||||
|
|||||||
Reference in New Issue
Block a user