Skip to content

Commit

Permalink
feat: add proof mode cell checks
Browse files Browse the repository at this point in the history
  • Loading branch information
zmalatrax committed Aug 2, 2024
1 parent aa1b244 commit 117f4c5
Show file tree
Hide file tree
Showing 4 changed files with 208 additions and 39 deletions.
17 changes: 17 additions & 0 deletions src/builtins/keccak.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,23 @@ export const CELLS_PER_KECCAK = 16;
/** Number of input cells for a keccak operation */
export const INPUT_CELLS_PER_KECCAK = 8;

/**
* The diluted cells are:
* - state - 25 rounds times 1600 elements.
* - parity - 24 rounds times 1600/5 elements times 3 auxiliaries.
* - after_theta_rho_pi - 24 rounds times 1600 elements.
* - theta_aux - 24 rounds times 1600 elements.
* - chi_iota_aux - 24 rounds times 1600 elements times 2 auxiliaries.
*
* In total 25 * 1600 + 24 * 320 * 3 + 24 * 1600 + 24 * 1600 + 24 * 1600 * 2 = 216640.
*
* But we actually allocate 4 virtual columns, of dimensions 64 * 1024, in which we embed the
* real cells, and we don't free the unused ones.
*
* So the real number is 4 * 64 * 1024 = 262144.
*/
export const KECCAK_DILUTED_CELLS = 262144;

/**
* Compute the new state of the keccak-f1600 block permutation on 24 rounds
*
Expand Down
214 changes: 175 additions & 39 deletions src/runners/cairoRunner.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,14 +11,27 @@ import {
} from 'errors/cairoRunner';

import { Felt } from 'primitives/felt';
import { SegmentValue } from 'primitives/segmentValue';
import { isFelt, SegmentValue } from 'primitives/segmentValue';
import { Relocatable } from 'primitives/relocatable';
import { CairoProgram, CairoZeroProgram, Program } from 'vm/program';
import { VirtualMachine } from 'vm/virtualMachine';
import { RcLimits, VirtualMachine } from 'vm/virtualMachine';
import { CELLS_PER_INSTANCE, getBuiltin } from 'builtins/builtin';
import { Hint, Hints } from 'hints/hintSchema';
import { isSubsequence, Layout, layouts } from './layout';
import {
isSubsequence,
Layout,
layouts,
MEMORY_UNITS_PER_STEP,
} from './layout';
import { nextPowerOfTwo } from 'primitives/utils';
import { ExpectedFelt } from 'errors/primitives';
import {
INNER_RC_BOUND_MASK,
INNER_RC_BOUND_SHIFT,
RC_N_PARTS,
RC_N_PARTS_96,
} from 'builtins/rangeCheck';
import { KECCAK_DILUTED_CELLS } from 'builtins/keccak';

/**
* Configuration of the run
Expand All @@ -36,6 +49,8 @@ export enum RunnerMode {
ProofModeCairo = 'Proof Mode - Cairo',
}

const MISSING_STEPS_CAPACITY = -1;

export class CairoRunner {
program: Program;
layout: Layout;
Expand Down Expand Up @@ -259,45 +274,15 @@ export class CairoRunner {
/**
* @returns {boolean} Whether there are enough allocated cells for
* generating a proof of this program execution for the chosen layout.
* @throws {} - If the number of allocated cells of the layout is insufficient.
* @throws {InsufficientAllocatedCells} - If the number of allocated cells of the layout is insufficient.
*
*/
checkCellUsage(): boolean {
const builtinChecks = this.builtins
.filter(
(builtin) =>
!['output', 'segment_arena', 'gas_builtin', 'system'].includes(
builtin
)
)
.filter((builtin) => !['gas_builtin', 'system'].includes(builtin))
.map((builtin) => {
// const instancesPerComponent = builtin === 'keccak' ? 16 : 1;
const segment = this.getBuiltinSegment(builtin);
if (!segment) throw new UndefinedBuiltinSegment(builtin);

const ratio = this.layout.ratios[builtin];
const cellsPerInstance = CELLS_PER_INSTANCE[builtin];
// TODO: set size to segment.length - SEGMENT_ARENA_INITIAL_SIZE (3)
// when implementing this builtin
const size = segment.length;
let capacity: number = 0;

if (this.layout.name === 'dynamic') {
const instances = Math.ceil(size / cellsPerInstance);
const instancesPerComponent = builtin === 'keccak' ? 16 : 1;
const components = nextPowerOfTwo(instances / instancesPerComponent);
capacity = cellsPerInstance * instancesPerComponent * components;
} else {
const minStep = ratio * cellsPerInstance;
if (this.vm.currentStep < minStep) {
console.log(
`PROOF MODE (${builtin}): minimum steps (${minStep}) not reached yet: ${this.vm.currentStep} steps`
);
return false;
}
capacity = (this.vm.currentStep / ratio) * cellsPerInstance;
}

const { size, capacity } = this.getSizeAndCapacity(builtin);
if (capacity === MISSING_STEPS_CAPACITY) return false;
if (size > capacity) {
throw new InsufficientAllocatedCells(
this.layout.name,
Expand All @@ -306,8 +291,159 @@ export class CairoRunner {
);
}
return true;
});
return builtinChecks.reduce((prev, curr) => prev && curr);
})
.reduce((prev, curr) => prev && curr);

const initialBounds: RcLimits = { rcMin: 0, rcMax: 1 };
const { rcMin, rcMax }: RcLimits = this.builtins
.filter((builtin) => ['range_check', 'range_check96'].includes(builtin))
.map((builtin) => {
const segment = this.getBuiltinSegment(builtin);
if (!segment) throw new UndefinedBuiltinSegment(builtin);
if (!segment.length) return { rcMin: 0, rcMax: 0 };
return segment.reduce((_, value) => {
if (!isFelt(value)) throw new ExpectedFelt(value);
const nParts = builtin === 'range_check' ? RC_N_PARTS : RC_N_PARTS_96;
return value
.to64BitsWords()
.flatMap((limb) =>
[3, 2, 1, 0].map(
(i) =>
(limb >> BigInt(i * INNER_RC_BOUND_SHIFT)) &
INNER_RC_BOUND_MASK
)
)
.slice(nParts)
.reduce((bounds, curr) => {
const x = Number(curr);
return {
rcMin: Math.min(bounds.rcMin, x),
rcMax: Math.max(bounds.rcMax, x),
};
}, initialBounds);
}, initialBounds);
})
.concat([this.vm.rcLimits])
.reduce((acc, curr) => ({
rcMin: Math.min(acc.rcMin, curr.rcMin),
rcMax: Math.max(curr.rcMin, curr.rcMax),
}));

const usedRcUnits = this.builtins
.filter((builtin) => ['range_check', 'range_check96'].includes(builtin))
.map((builtin) => {
const segment = this.getBuiltinSegment(builtin);
if (!segment) throw new UndefinedBuiltinSegment(builtin);
return builtin === 'range_check'
? segment.length * RC_N_PARTS
: segment.length * RC_N_PARTS_96;
})
.reduce((acc, curr) => acc + curr);

const unusedRcUnits =
(this.layout.rcUnits - 3) * this.vm.currentStep - usedRcUnits;
const rcUnitsCheck = unusedRcUnits >= rcMax - rcMin;

const builtinsCapacity = this.builtins
.map((builtin) => this.getSizeAndCapacity(builtin).capacity)
.reduce((acc, curr) => acc + curr);
const totalMemoryCapacity = this.vm.currentStep * MEMORY_UNITS_PER_STEP;
if (totalMemoryCapacity % this.layout.publicMemoryFraction)
throw new Error(
'Total memory capacity is not a multiple of public memory fraction.'
);
const publicMemoryCapacity =
totalMemoryCapacity / this.layout.publicMemoryFraction;

const instructionCapacity = this.vm.currentStep * 4;
const unusedMemoryCapacity =
totalMemoryCapacity -
(publicMemoryCapacity + instructionCapacity + builtinsCapacity);
const memoryHoles = this.vm.memory.segments.reduce(
(acc, currSegment) =>
acc + currSegment.reduce((acc, _) => acc--, currSegment.length),
0
);
const memoryCheck = unusedMemoryCapacity >= memoryHoles;

let dilutedCheck: boolean = true;
const dilutedPool = this.layout.dilutedPool;
if (dilutedPool) {
const dilutedUsedCapacity = this.builtins
.filter((builtin) => ['bitwise', 'keccak'].includes(builtin))
.map((builtin) => {
const multiplier =
this.layout.name === 'dynamic'
? this.vm.currentStep
: this.vm.currentStep / this.layout.ratios[builtin];
if (builtin === 'bitwise') {
const totalNBits = 251;
const { nBits, spacing } = dilutedPool;
const step = nBits * spacing;
const partition: number[] = [];
for (let i = 0; i < totalNBits; i += step) {
for (let j = 0; j < spacing; j++) {
if (i + j < totalNBits) {
partition.push(i + j);
}
}
}
const trimmedNumber = partition.filter(
(value) => value + spacing * (nBits - 1) + 1 > totalNBits
).length;

return (4 * partition.length + trimmedNumber) * multiplier;
}
return (KECCAK_DILUTED_CELLS / dilutedPool.nBits) * multiplier;
})
.reduce((acc, curr) => acc + curr);

const dilutedUnits = this.vm.currentStep * dilutedPool.unitsPerStep;
const unusedDilutedCapacity = dilutedUnits - dilutedUsedCapacity;
dilutedCheck = unusedDilutedCapacity >= 1 << dilutedPool.nBits;
}

return builtinChecks && rcUnitsCheck && memoryCheck && dilutedCheck;
}

/** @returns The size of a builtin and its capacity for the chosen layout. */
private getSizeAndCapacity(builtin: string): {
size: number;
capacity: number;
} {
const segment = this.getBuiltinSegment(builtin);
if (!segment) throw new UndefinedBuiltinSegment(builtin);
const size =
builtin === 'segment_arena' ? segment.length - 3 : segment.length;

if (builtin === 'output' || builtin === 'segment_arena') {
return { size, capacity: size };
}

const ratio = this.layout.ratios[builtin];
const cellsPerInstance = CELLS_PER_INSTANCE[builtin];
let capacity: number = 0;

switch (this.layout.name) {
case 'dynamic':
const instances = Math.ceil(size / cellsPerInstance);
const instancesPerComponent = builtin === 'keccak' ? 16 : 1;
const components = nextPowerOfTwo(instances / instancesPerComponent);
capacity = cellsPerInstance * instancesPerComponent * components;
break;
default:
const minStep = ratio * cellsPerInstance;
if (this.vm.currentStep < minStep) {
// console.log(
// `PROOF MODE (${builtin}): minimum steps (${minStep}) not reached yet: ${this.vm.currentStep} steps`
// );
return { size, capacity: MISSING_STEPS_CAPACITY };
}
capacity = (this.vm.currentStep / ratio) * cellsPerInstance;
break;
}

return { size, capacity };
}

/**
Expand Down
2 changes: 2 additions & 0 deletions src/runners/layout.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,8 @@ export const DEFAULT_DILUTED_POOL: DilutedPool = {
nBits: 16,
};

export const MEMORY_UNITS_PER_STEP = 8;

/**
* Dictionary containing all the available layouts:
* - plain
Expand Down
14 changes: 14 additions & 0 deletions src/vm/virtualMachine.ts
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,12 @@ export type RelocatedMemory = {
value: Felt;
};

/** The bounds of the range_check builtins during a run. */
export type RcLimits = {
rcMin: number;
rcMax: number;
};

export class VirtualMachine {
currentStep: number;
memory: Memory;
Expand All @@ -73,6 +79,7 @@ export class VirtualMachine {
squashedDictManager: SquashedDictManager;
scopeManager: ScopeManager;
trace: TraceEntry[];
rcLimits: RcLimits;
relocatedMemory: RelocatedMemory[];
relocatedTrace: RelocatedTraceEntry[];

Expand All @@ -82,6 +89,7 @@ export class VirtualMachine {
this.currentStep = 0;
this.memory = new Memory();
this.trace = [];
this.rcLimits = { rcMin: 0, rcMax: 0 };
this.relocatedMemory = [];
this.relocatedTrace = [];

Expand Down Expand Up @@ -139,6 +147,12 @@ export class VirtualMachine {

this.trace.push({ pc: this.pc, ap: this.ap, fp: this.fp });

const { dstOffset, op0Offset, op1Offset } = instruction;
this.rcLimits = {
rcMin: Math.min(dstOffset, op0Offset, op1Offset),
rcMax: Math.max(dstOffset, op0Offset, op1Offset),
};

this.updateRegisters(instruction, res, dst);

this.currentStep += 1;
Expand Down

0 comments on commit 117f4c5

Please sign in to comment.