snarkvm-synthesizer-process 4.7.3

A process for a decentralized virtual machine
Documentation
// Copyright (c) 2019-2026 Provable Inc.
// This file is part of the snarkVM library.

// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at:

// http://www.apache.org/licenses/LICENSE-2.0

// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

use super::*;

#[derive(Clone, Debug)]
pub struct InclusionAssignment<N: Network> {
    pub(crate) state_path: StatePath<N>,
    commitment: Field<N>,
    gamma: Group<N>,
    serial_number: Field<N>,
    is_record_block_height_reached: bool,
    upgrade_block_height: u32,
    local_state_root: N::TransactionID,
    is_global: bool,
}

impl<N: Network> InclusionAssignment<N> {
    /// Initializes a new inclusion assignment.
    pub fn new(
        state_path: StatePath<N>,
        commitment: Field<N>,
        gamma: Group<N>,
        serial_number: Field<N>,
        is_record_block_height_reached: bool,
        upgrade_block_height: u32,
        local_state_root: N::TransactionID,
        is_global: bool,
    ) -> Self {
        Self {
            state_path,
            commitment,
            gamma,
            serial_number,
            is_record_block_height_reached,
            upgrade_block_height,
            local_state_root,
            is_global,
        }
    }

    /// The circuit for state path verification.
    ///
    /// # Diagram
    /// The `[[ ]]` notation is used to denote public inputs.
    /// ```ignore
    ///             [[ global_state_root ]] || [[ local_state_root ]] [[ is_record_block_height_reached ]]
    ///                        |                          |                            |
    ///                        -------- is_global --------                             |
    ///                                     |                                          |
    ///                                state_path ------- record_block_height ------ compare ------ [[ upgrade_block_height ]]
    ///                                    |                                           |
    ///                                    |                                           |
    ///                                    |                        is_global ------ check
    ///                                    |
    /// [[ serial_number ]] := Commit( commitment || Hash( COFACTOR * gamma ) )
    /// ```
    pub fn to_circuit_assignment<A: circuit::Aleo<Network = N>>(&self) -> Result<circuit::Assignment<N::Field>> {
        use circuit::Inject;

        // Ensure the circuit environment is clean.
        assert_eq!(A::count(), (0, 1, 0, 0, (0, 0, 0)));
        A::reset();

        // Inject the state path as `Mode::Private` (with a global state root as `Mode::Public`).
        let state_path = circuit::StatePath::<A>::new(circuit::Mode::Private, self.state_path.clone());
        // Inject the commitment as `Mode::Private`.
        let commitment = circuit::Field::<A>::new(circuit::Mode::Private, self.commitment);
        // Inject the gamma as `Mode::Private`.
        let gamma = circuit::Group::<A>::new(circuit::Mode::Private, self.gamma);

        // Inject the local state root as `Mode::Public`.
        let local_state_root = circuit::Field::<A>::new(circuit::Mode::Public, *self.local_state_root);
        // Inject the 'is_global' flag as `Mode::Private`.
        let is_global = circuit::Boolean::<A>::new(circuit::Mode::Private, self.is_global);

        // Inject the serial number as `Mode::Public`.
        let serial_number = circuit::Field::<A>::new(circuit::Mode::Public, self.serial_number);

        // Inject the `is_record_block_height_reached` flag as `Mode::Public`.
        let is_record_block_height_reached =
            circuit::Boolean::<A>::new(circuit::Mode::Public, self.is_record_block_height_reached);
        // Inject the `upgrade_block_height` as `Mode::Public`.
        // This is cast into a u32 to prevent requiring 32 field elements as input.
        let upgrade_block_height_field = circuit::Field::<A>::new(
            circuit::Mode::Public,
            console::types::Field::<N>::from_u32(self.upgrade_block_height),
        );
        let upgrade_block_height = circuit::U32::from_field(upgrade_block_height_field);

        // Compute the candidate serial number.
        let candidate_serial_number =
            circuit::Record::<A, circuit::Plaintext<A>>::serial_number_from_gamma(&gamma, commitment.clone());
        // Enforce that the candidate serial number is equal to the serial number.
        A::assert_eq(candidate_serial_number, serial_number)?;

        // Enforce the starting leaf is the claimed commitment.
        A::assert_eq(state_path.transition_leaf().id(), commitment)?;
        // Enforce the state path from leaf to root is correct.
        A::assert(state_path.verify(&is_global, &local_state_root))?;

        // Fetch the record block height from the state path.
        let record_block_height = U32::<A>::from_bits_le(&state_path.block_path().leaf_index().to_bits_le());
        // Determine if the record block height is at or past the upgrade height.
        let is_record_block_height_past_upgrade_block_height =
            record_block_height.is_greater_than_or_equal(&upgrade_block_height);
        // Determine if the record block height is correctly evaluated.
        let is_block_height_check_valid =
            is_record_block_height_reached.is_equal(&is_record_block_height_past_upgrade_block_height);
        // Check that the height is valid if the record is from a global state path.
        A::assert(is_global.not().bitor(is_block_height_check_valid))?;

        Stack::log_circuit::<A>(
            format_args!("State Path for {}", self.serial_number),
            "InclusionAssignment".to_string(),
        );

        // Eject the assignment and reset the circuit environment.
        Ok(A::eject_assignment_and_reset())
    }
}