triton_air/
challenge_id.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
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
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
use std::fmt::Debug;
use std::hash::Hash;

use arbitrary::Arbitrary;
use strum::Display;
use strum::EnumCount;
use strum::EnumIter;

/// A `ChallengeId` is a unique, symbolic identifier for a challenge used in
/// Triton VM.
///
/// Since almost all challenges relate to the Processor Table in some form, the
/// words “Processor Table” are usually omitted from the `ChallengeId`'s name.
#[repr(usize)]
#[derive(Debug, Display, Copy, Clone, Eq, PartialEq, Hash, EnumCount, EnumIter, Arbitrary)]
pub enum ChallengeId {
    /// The indeterminate for the [Evaluation Argument][eval] compressing the program digest
    /// into a single extension field element, _i.e._,
    /// [`CompressedProgramDigest`][Self::CompressedProgramDigest].
    /// Relates to program attestation.
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    CompressProgramDigestIndeterminate,

    /// The indeterminate for the [Evaluation Argument][eval] with standard input.
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    StandardInputIndeterminate,

    /// The indeterminate for the [Evaluation Argument][eval] with standard output.
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    StandardOutputIndeterminate,

    /// The indeterminate for the instruction
    /// [Lookup Argument](crate::cross_table_argument::LookupArg)
    /// between the [Processor Table](crate::table::processor) and the
    /// [Program Table](crate::table::program) guaranteeing that the instructions and their
    /// arguments are copied correctly.
    InstructionLookupIndeterminate,

    HashInputIndeterminate,
    HashDigestIndeterminate,
    SpongeIndeterminate,

    OpStackIndeterminate,
    RamIndeterminate,
    JumpStackIndeterminate,

    U32Indeterminate,

    /// The indeterminate for the Lookup Argument between the Processor Table and all memory-like
    /// tables, _i.e._, the OpStack Table, the Ram Table, and the JumpStack Table, guaranteeing
    /// that all clock jump differences are directed forward.
    ClockJumpDifferenceLookupIndeterminate,

    /// The indeterminate for the Contiguity Argument within the Ram Table.
    RamTableBezoutRelationIndeterminate,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `Address` in the Program Table
    /// - `IP` in the Processor Table
    ProgramAddressWeight,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `Instruction` in the Program Table
    /// - `CI` in the Processor Table
    ProgramInstructionWeight,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `Instruction` in the next row in the Program Table
    /// - `NIA` in the Processor Table
    ProgramNextInstructionWeight,

    OpStackClkWeight,
    OpStackIb1Weight,
    OpStackPointerWeight,
    OpStackFirstUnderflowElementWeight,

    RamClkWeight,
    RamPointerWeight,
    RamValueWeight,
    RamInstructionTypeWeight,

    JumpStackClkWeight,
    JumpStackCiWeight,
    JumpStackJspWeight,
    JumpStackJsoWeight,
    JumpStackJsdWeight,

    /// The indeterminate for compressing a [`RATE`][rate]-sized chunk of instructions into a
    /// single extension field element.
    /// Relates to program attestation.
    ///
    /// Used by the evaluation argument [`PrepareChunkEvalArg`][prep] and in the Hash Table.
    ///
    /// [rate]: twenty_first::prelude::tip5::RATE
    /// [prep]: crate::table_column::ProgramAuxColumn::PrepareChunkRunningEvaluation
    ProgramAttestationPrepareChunkIndeterminate,

    /// The indeterminate for the bus over which the [`RATE`][rate]-sized chunks of instructions
    /// are sent. Relates to program attestation.
    /// Used by the evaluation arguments [`SendChunkEvalArg`][send] and
    /// [`ReceiveChunkEvalArg`][recv]. See also:
    /// [`ProgramAttestationPrepareChunkIndeterminate`][ind].
    ///
    /// [rate]: twenty_first::prelude::tip5::RATE
    /// [send]: crate::table_column::ProgramAuxColumn::SendChunkRunningEvaluation
    /// [recv]: crate::table_column::HashAuxColumn::ReceiveChunkRunningEvaluation
    /// [ind]: ChallengeId::ProgramAttestationPrepareChunkIndeterminate
    ProgramAttestationSendChunkIndeterminate,

    HashCIWeight,

    StackWeight0,
    StackWeight1,
    StackWeight2,
    StackWeight3,
    StackWeight4,
    StackWeight5,
    StackWeight6,
    StackWeight7,
    StackWeight8,
    StackWeight9,
    StackWeight10,
    StackWeight11,
    StackWeight12,
    StackWeight13,
    StackWeight14,
    StackWeight15,

    /// The indeterminate for the Lookup Argument between the Hash Table and the Cascade Table.
    HashCascadeLookupIndeterminate,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `*LkIn` in the Hash Table, and
    /// - `2^16·LookInHi + LookInLo` in the Cascade Table.
    HashCascadeLookInWeight,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `*LkOut` in the Hash Table, and
    /// - `2^16·LookOutHi + LookOutLo` in the Cascade Table.
    HashCascadeLookOutWeight,

    /// The indeterminate for the Lookup Argument between the Cascade Table and the Lookup Table.
    CascadeLookupIndeterminate,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `LkIn*` in the Cascade Table, and
    /// - `LookIn` in the Lookup Table.
    LookupTableInputWeight,

    /// A weight for linearly combining multiple elements. Applies to
    /// - `LkOut*` in the Cascade Table, and
    /// - `LookOut` in the Lookup Table.
    LookupTableOutputWeight,

    /// The indeterminate for the public Evaluation Argument establishing correctness of the
    /// Lookup Table.
    LookupTablePublicIndeterminate,

    U32LhsWeight,
    U32RhsWeight,
    U32CiWeight,
    U32ResultWeight,

    // Derived challenges.
    //
    // When modifying this, be sure to add to the compile-time assertions in the
    // `#[test] const fn compile_time_index_assertions() { … }`
    // at the end of this file.
    /// The terminal for the [`EvaluationArgument`][eval] with standard input.
    /// Makes use of challenge
    /// [`StandardInputIndeterminate`][Self::StandardInputIndeterminate].
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    StandardInputTerminal,

    /// The terminal for the [`EvaluationArgument`][eval] with standard output.
    /// Makes use of challenge
    /// [`StandardOutputIndeterminate`][Self::StandardOutputIndeterminate].
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    StandardOutputTerminal,

    /// The terminal for the [`EvaluationArgument`][eval] establishing correctness of the
    /// [Lookup Table](crate::table::lookup::LookupTable).
    /// Makes use of challenge
    /// [`LookupTablePublicIndeterminate`][Self::LookupTablePublicIndeterminate].
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    LookupTablePublicTerminal,

    /// The digest of the program to be executed, compressed into a single extension field element.
    /// The compression happens using an [`EvaluationArgument`][eval] under challenge
    /// [`CompressProgramDigestIndeterminate`][Self::CompressProgramDigestIndeterminate].
    /// Relates to program attestation.
    ///
    /// [eval]: crate::cross_table_argument::EvalArg
    CompressedProgramDigest,
}

impl ChallengeId {
    /// The number of challenges derived from other challenges.
    ///
    /// The IDs of the derived challenges are guaranteed to be larger than the
    /// challenges they are derived from.
    pub const NUM_DERIVED_CHALLENGES: usize = 4;

    pub const fn index(&self) -> usize {
        *self as usize
    }
}

impl From<ChallengeId> for usize {
    fn from(id: ChallengeId) -> Self {
        id.index()
    }
}

#[cfg(test)]
pub(crate) mod tests {
    use super::*;

    /// Terminal challenges are computed from public information, such as public
    /// input or public output, and other challenges. Because these other challenges
    /// are used to compute the terminal challenges, the terminal challenges must be
    /// inserted into the challenges vector after the used challenges.
    #[test]
    const fn compile_time_index_assertions() {
        const DERIVED: [ChallengeId; ChallengeId::NUM_DERIVED_CHALLENGES] = [
            ChallengeId::StandardInputTerminal,
            ChallengeId::StandardOutputTerminal,
            ChallengeId::LookupTablePublicTerminal,
            ChallengeId::CompressedProgramDigest,
        ];

        assert!(ChallengeId::StandardInputIndeterminate.index() < DERIVED[0].index());
        assert!(ChallengeId::StandardInputIndeterminate.index() < DERIVED[1].index());
        assert!(ChallengeId::StandardInputIndeterminate.index() < DERIVED[2].index());
        assert!(ChallengeId::StandardInputIndeterminate.index() < DERIVED[3].index());

        assert!(ChallengeId::StandardOutputIndeterminate.index() < DERIVED[0].index());
        assert!(ChallengeId::StandardOutputIndeterminate.index() < DERIVED[1].index());
        assert!(ChallengeId::StandardOutputIndeterminate.index() < DERIVED[2].index());
        assert!(ChallengeId::StandardOutputIndeterminate.index() < DERIVED[3].index());

        assert!(ChallengeId::CompressProgramDigestIndeterminate.index() < DERIVED[0].index());
        assert!(ChallengeId::CompressProgramDigestIndeterminate.index() < DERIVED[1].index());
        assert!(ChallengeId::CompressProgramDigestIndeterminate.index() < DERIVED[2].index());
        assert!(ChallengeId::CompressProgramDigestIndeterminate.index() < DERIVED[3].index());

        assert!(ChallengeId::LookupTablePublicIndeterminate.index() < DERIVED[0].index());
        assert!(ChallengeId::LookupTablePublicIndeterminate.index() < DERIVED[1].index());
        assert!(ChallengeId::LookupTablePublicIndeterminate.index() < DERIVED[2].index());
        assert!(ChallengeId::LookupTablePublicIndeterminate.index() < DERIVED[3].index());
    }

    // Ensure the compile-time assertions are actually executed by the compiler.
    const _: () = compile_time_index_assertions();
}