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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
use ark_ff::FftField;
use ark_poly::{Evaluations, Radix2EvaluationDomain};

use crate::{logup, logup::LookupTableID, witness::Witness};
use kimchi::circuits::{
    berkeley_columns::{BerkeleyChallengeTerm, BerkeleyChallenges},
    domains::{Domain, EvaluationDomains},
    expr::{ColumnEnvironment as TColumnEnvironment, Constants},
};

/// The collection of polynomials (all in evaluation form) and constants
/// required to evaluate an expression as a polynomial.
///
/// All are evaluations.
pub struct ColumnEnvironment<
    'a,
    const N: usize,
    const N_REL: usize,
    const N_DSEL: usize,
    const N_FSEL: usize,
    F: FftField,
    ID: LookupTableID,
> {
    /// The witness column polynomials. Includes relation columns,
    /// fixed selector columns, and dynamic selector columns.
    pub witness: &'a Witness<N, Evaluations<F, Radix2EvaluationDomain<F>>>,
    /// Fixed selectors. These are "predefined" with the circuit, and,
    /// unlike public input or dynamic selectors, are not part of the
    /// witness that users are supposed to change after the circuit is
    /// fixed.
    pub fixed_selectors: &'a [Evaluations<F, Radix2EvaluationDomain<F>>; N_FSEL],
    /// The value `prod_{j != 1} (1 - omega^j)`, used for efficiently
    /// computing the evaluations of the unnormalized Lagrange basis polynomials.
    pub l0_1: F,
    /// Constant values required
    pub constants: Constants<F>,
    /// Challenges from the IOP.
    pub challenges: BerkeleyChallenges<F>,
    /// The domains used in the PLONK argument.
    pub domain: EvaluationDomains<F>,

    /// Lookup specific polynomials
    pub lookup: Option<logup::prover::QuotientPolynomialEnvironment<'a, F, ID>>,
}

impl<
        'a,
        const N_WIT: usize,
        const N_REL: usize,
        const N_DSEL: usize,
        const N_FSEL: usize,
        F: FftField,
        ID: LookupTableID,
    > TColumnEnvironment<'a, F, BerkeleyChallengeTerm, BerkeleyChallenges<F>>
    for ColumnEnvironment<'a, N_WIT, N_REL, N_DSEL, N_FSEL, F, ID>
{
    type Column = crate::columns::Column<usize>;

    fn get_column(
        &self,
        col: &Self::Column,
    ) -> Option<&'a Evaluations<F, Radix2EvaluationDomain<F>>> {
        // TODO: when non-literal constant generics are available, substitute N with N_REG + N_DSEL + N_FSEL
        assert!(N_WIT == N_REL + N_DSEL);
        assert!(N_WIT == self.witness.len());
        match *col {
            // Handling the "relation columns" at the beginning of the witness columns
            Self::Column::Relation(i) => {
                // TODO: add a test for this
                assert!(i < N_REL,"Requested column with index {:?} but the given witness is meant for {:?} relation columns", i, N_REL);
                let res = &self.witness[i];
                Some(res)
            }
            // Handling the "dynamic selector columns" at the end of the witness columns
            Self::Column::DynamicSelector(i) => {
                assert!(i < N_DSEL, "Requested dynamic selector with index {:?} but the given witness is meant for {:?} dynamic selector columns", i, N_DSEL);
                let res = &self.witness[N_REL + i];
                Some(res)
            }
            Self::Column::FixedSelector(i) => {
                assert!(i < N_FSEL, "Requested fixed selector with index {:?} but the given witness is meant for {:?} fixed selector columns", i, N_FSEL);
                let res = &self.fixed_selectors[i];
                Some(res)
            }
            Self::Column::LookupPartialSum((table_id, i)) => {
                if let Some(ref lookup) = self.lookup {
                    let table_id = ID::from_u32(table_id);
                    Some(&lookup.lookup_terms_evals_d8[&table_id][i])
                } else {
                    panic!("No lookup provided")
                }
            }
            Self::Column::LookupAggregation => {
                if let Some(ref lookup) = self.lookup {
                    Some(lookup.lookup_aggregation_evals_d8)
                } else {
                    panic!("No lookup provided")
                }
            }
            Self::Column::LookupMultiplicity((table_id, i)) => {
                if let Some(ref lookup) = self.lookup {
                    Some(&lookup.lookup_counters_evals_d8[&ID::from_u32(table_id)][i])
                } else {
                    panic!("No lookup provided")
                }
            }
            Self::Column::LookupFixedTable(table_id) => {
                if let Some(ref lookup) = self.lookup {
                    Some(&lookup.fixed_tables_evals_d8[&ID::from_u32(table_id)])
                } else {
                    panic!("No lookup provided")
                }
            }
        }
    }

    fn get_domain(&self, d: Domain) -> Radix2EvaluationDomain<F> {
        match d {
            Domain::D1 => self.domain.d1,
            Domain::D2 => self.domain.d2,
            Domain::D4 => self.domain.d4,
            Domain::D8 => self.domain.d8,
        }
    }

    fn column_domain(&self, col: &Self::Column) -> Domain {
        match *col {
            Self::Column::Relation(_)
            | Self::Column::DynamicSelector(_)
            | Self::Column::FixedSelector(_) => {
                let domain_size = match *col {
                    Self::Column::Relation(i) => self.witness[i].domain().size,
                    Self::Column::DynamicSelector(i) => self.witness[N_REL + i].domain().size,
                    Self::Column::FixedSelector(i) => self.fixed_selectors[i].domain().size,
                    _ => panic!("Impossible"),
                };
                if self.domain.d1.size == domain_size {
                    Domain::D1
                } else if self.domain.d2.size == domain_size {
                    Domain::D2
                } else if self.domain.d4.size == domain_size {
                    Domain::D4
                } else if self.domain.d8.size == domain_size {
                    Domain::D8
                } else {
                    panic!("Domain not supported. We do support the following multiple of the domain registered in the environment: 1, 2, 4, 8")
                }
            }
            Self::Column::LookupAggregation
            | Self::Column::LookupFixedTable(_)
            | Self::Column::LookupMultiplicity(_)
            | Self::Column::LookupPartialSum(_) => {
                // When there is a lookup, we do suppose the domain is always D8
                // and we have at leat 6 lookups per row.
                Domain::D8
            }
        }
    }

    fn get_constants(&self) -> &Constants<F> {
        &self.constants
    }

    fn get_challenges(&self) -> &BerkeleyChallenges<F> {
        &self.challenges
    }

    fn vanishes_on_zero_knowledge_and_previous_rows(
        &self,
    ) -> &'a Evaluations<F, Radix2EvaluationDomain<F>> {
        panic!("Not supposed to be used in MSM")
    }

    fn l0_1(&self) -> F {
        self.l0_1
    }
}