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
use crate::columns::{Column, ColumnIndexer};

use crate::N_LIMBS;

/// Number of columns in the FFA circuits.
pub const FFA_N_COLUMNS: usize = 5 * N_LIMBS;
pub const FFA_NPUB_COLUMNS: usize = N_LIMBS;

/// Column indexer for MSM columns.
///
/// They represent the equation
///   `InputA(i) + InputB(i) = ModulusF(i) * Quotient + Carry(i) * 2^LIMB_SIZE - Carry(i-1)`
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub enum FFAColumn {
    InputA(usize),
    InputB(usize),
    ModulusF(usize),
    Remainder(usize),
    Carry(usize),
    Quotient,
}

impl ColumnIndexer<usize> for FFAColumn {
    const N_COL: usize = FFA_N_COLUMNS;
    fn to_column(self) -> Column<usize> {
        let to_column_inner = |offset, i| {
            assert!(i < N_LIMBS);
            Column::Relation(N_LIMBS * offset + i)
        };
        match self {
            FFAColumn::InputA(i) => to_column_inner(0, i),
            FFAColumn::InputB(i) => to_column_inner(1, i),
            FFAColumn::ModulusF(i) => to_column_inner(2, i),
            FFAColumn::Remainder(i) => to_column_inner(3, i),
            FFAColumn::Carry(i) => {
                assert!(i < N_LIMBS - 1);
                to_column_inner(4, i)
            }
            FFAColumn::Quotient => to_column_inner(4, N_LIMBS - 1),
        }
    }
}