Files
poulpy/base2k/src/module.rs
Jean-Philippe Bossuat 19758b4736 removed Bytes
2025-02-14 18:27:32 +01:00

64 lines
1.5 KiB
Rust

use crate::ffi::module::{delete_module_info, module_info_t, new_module_info, MODULE};
use crate::{Free, GALOISGENERATOR};
pub type MODULETYPE = u8;
pub const FFT64: u8 = 0;
pub const NTT120: u8 = 1;
pub struct Module(pub *mut MODULE, pub usize);
impl Module {
// Instantiates a new module.
pub fn new<const MODULETYPE: MODULETYPE>(n: usize) -> Self {
unsafe {
let m: *mut module_info_t = new_module_info(n as u64, MODULETYPE as u32);
if m.is_null() {
panic!("Failed to create module.");
}
Self(m, n)
}
}
pub fn n(&self) -> usize {
self.1
}
pub fn log_n(&self) -> usize {
(usize::BITS - (self.n() - 1).leading_zeros()) as _
}
pub fn cyclotomic_order(&self) -> u64 {
(self.n() << 1) as _
}
// GALOISGENERATOR^|gen| * sign(gen)
pub fn galois_element(&self, gen: i64) -> i64 {
if gen == 0 {
return 1;
}
let mut gal_el: u64 = 1;
let mut gen_1_pow: u64 = GALOISGENERATOR;
let mut e: usize = gen.abs() as usize;
while e > 0 {
if e & 1 == 1 {
gal_el = gal_el.wrapping_mul(gen_1_pow);
}
gen_1_pow = gen_1_pow.wrapping_mul(gen_1_pow);
e >>= 1;
}
gal_el &= self.cyclotomic_order() - 1;
(gal_el as i64) * gen.signum()
}
}
impl Free for Module {
fn free(self) {
unsafe { delete_module_info(self.0) }
drop(self);
}
}