binius_core/transparent/
powers.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
// Copyright 2024-2025 Irreducible Inc.

use std::iter::successors;

use binius_field::{Field, PackedField, TowerField};
use binius_math::MultilinearExtension;
use binius_utils::bail;
use bytemuck::zeroed_vec;
use itertools::{izip, Itertools};
use rayon::prelude::*;

use crate::polynomial::{Error, MultivariatePoly};

/// A transparent multilinear polynomial whose evaluation at index $i$ is $g^i$ for
/// some field element $g$.
#[derive(Debug)]
pub struct Powers<F: Field> {
	n_vars: usize,
	base: F,
}

impl<F: Field> Powers<F> {
	pub fn new(n_vars: usize, base: F) -> Self {
		Powers { n_vars, base }
	}

	pub fn multilinear_extension<P: PackedField<Scalar = F>>(
		&self,
	) -> Result<MultilinearExtension<P>, Error> {
		let mut values = zeroed_vec(1 << self.n_vars.saturating_sub(P::LOG_WIDTH));

		const CHUNK_LOG_SIZE: usize = 10;
		values
			.par_chunks_mut(1 << CHUNK_LOG_SIZE)
			.enumerate()
			.for_each(|(chunk_index, chunk)| {
				let start_power = (chunk_index as u64) << CHUNK_LOG_SIZE;
				let powers = successors(Some(self.base.pow_vartime([start_power])), |&power| {
					Some(power * self.base)
				})
				.take(1 << self.n_vars);

				for (dest, values) in izip!(chunk, &powers.chunks(P::WIDTH)) {
					*dest = P::from_scalars(values);
				}
			});

		Ok(MultilinearExtension::new(self.n_vars, values)?)
	}
}

impl<F: TowerField, P: PackedField<Scalar = F>> MultivariatePoly<P> for Powers<F> {
	fn n_vars(&self) -> usize {
		self.n_vars
	}

	fn degree(&self) -> usize {
		self.n_vars
	}

	fn evaluate(&self, query: &[P]) -> Result<P, Error> {
		let n_vars = self.n_vars;
		if query.len() != self.n_vars {
			bail!(Error::IncorrectQuerySize { expected: n_vars });
		}

		let mut result = P::one();
		let mut base_power = self.base;
		for &q_i in query {
			result *= P::one() - q_i + q_i * base_power;
			base_power = base_power.square();
		}

		Ok(result)
	}

	fn binary_tower_level(&self) -> usize {
		F::TOWER_LEVEL
	}
}

#[cfg(test)]
mod tests {
	use binius_field::{BinaryField32b, Field, PackedBinaryField4x32b, PackedField, TowerField};
	use rand::{prelude::StdRng, SeedableRng};

	use super::Powers;
	use crate::polynomial::MultivariatePoly;

	fn test_consistency_helper<P: PackedField<Scalar: TowerField>>(n_vars: usize, base: P::Scalar) {
		let powers = Powers::new(n_vars, base);
		let mle = powers.multilinear_extension::<P>().unwrap();
		let mut power = P::Scalar::ONE;
		for i in 0..1 << n_vars {
			let query = (0..n_vars)
				.map(|j| {
					if (i >> j) & 1 == 0 {
						P::zero()
					} else {
						P::one()
					}
				})
				.collect::<Vec<_>>();
			assert_eq!(powers.evaluate(&query).unwrap(), P::broadcast(power));
			assert_eq!(mle.evaluate_on_hypercube(i).unwrap(), power);
			power *= base;
		}
	}

	#[test]
	fn test_consistency() {
		type F = BinaryField32b;
		type P = PackedBinaryField4x32b;
		let mut rng = StdRng::seed_from_u64(0);
		for n_vars in 0..12 {
			test_consistency_helper::<P>(n_vars, <F as Field>::random(&mut rng));
		}
	}
}