This circuit takes storage requests from storage_application_log_state. Then for each query, it verifies the read value and updates the root_hash is needed. Also, it outputs the hash of storage diffs. Shard_id if enforces to be 0 for now, because we have only one shard.
First part
The circuit begins with allocating input part of the PI.
let StorageApplicationCircuitInstanceWitness {
closed_form_input,
storage_queue_witness,
merkle_paths,
leaf_indexes_for_reads,
} = witness;
let mut structured_input =
StorageApplicationInputOutput::alloc_ignoring_outputs(cs, closed_form_input.clone());
We chose what storage_application_log_state, root_hash and other fields to continue to work with.
let mut current_root_hash = UInt8::<F>::parallel_select(
cs,
start_flag,
&structured_input.observable_input.initial_root_hash,
&structured_input.hidden_fsm_input.current_root_hash,
);
let storage_accesses_queue_state = QueueState::conditionally_select(
cs,
start_flag,
&storage_queue_state_from_input,
&storage_queue_state_from_fsm,
);
...
Main part
Here’s the part, where all the main logic is implemented. Firstly, we take a new storage request if needed.
let (storage_log, _) = storage_accesses_queue.pop_front(cs, parse_next_queue_elem);
Now we update PI output parts and compute a commitment. Then we allocate it as public variables.
let compact_form =
ClosedFormInputCompactForm::from_full_form(cs, &structured_input, round_function);
let input_commitment = commit_variable_length_encodable_item(cs, &compact_form, round_function);
for el in input_commitment.iter() {
let gate = PublicInputGate::new(el.get_variable());
gate.add_to_cs(cs);
}