Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Initial trazability of ACIR #1701

Merged
merged 17 commits into from
Aug 22, 2023
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion yarn-project/acir-simulator/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@
"@aztec/circuits.js": "workspace:^",
"@aztec/foundation": "workspace:^",
"@aztec/types": "workspace:^",
"acvm_js": "github:noir-lang/acvm-js-wasm#db/init-sim-backend",
"acvm_js": "github:noir-lang/acvm-js-wasm#arv/0.22+init-pedersen",
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated the dependency from 0.21 + init pedersen to 0.22 + init pedersen. Init pedersen will eventually be released as part of 0.23, let's hope that lands soon!

"levelup": "^5.1.1",
"memdown": "^6.1.1",
"tslib": "^2.4.0"
Expand Down
91 changes: 90 additions & 1 deletion yarn-project/acir-simulator/src/acvm/acvm.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ import {
executeCircuitWithBlackBoxSolver,
} from 'acvm_js';

import { FunctionDebugMetadata } from '../index.js';

/**
* The format for fields on the ACVM.
*/
Expand Down Expand Up @@ -71,6 +73,74 @@ export interface ACIRExecutionResult {
partialWitness: ACVMWitness;
}

/**
* Extracts the opcode location from an ACVM error string.
*/
function extractOpcodeLocationFromError(err: string): string | undefined {
const match = err.match(/^Cannot satisfy constraint (?<opcodeLocation>[0-9]+(?:\.[0-9]+)?)/);
return match?.groups?.opcodeLocation;
}

/**
* The data for a call in the call stack.
*/
interface SourceCodeLocation {
/**
* The path to the source file.
*/
filePath: string;
/**
* The line number of the call.
*/
line: number;
/**
* The source code of the file.
*/
fileSource: string;
/**
* The source code text of the failed constraint.
*/
assertionText: string;
}

/**
* Extracts the call stack from the location of a failing opcode and the debug metadata.
*/
function getCallStackFromOpcodeLocation(opcodeLocation: string, debug: FunctionDebugMetadata): SourceCodeLocation[] {
const { debugSymbols, files } = debug;

const callStack = debugSymbols.locations[opcodeLocation] || [];
return callStack.map(call => {
const { file: fileId, span } = call;

const { path, source } = files[fileId];

const assertionText = source.substring(span.start, span.end + 1);
const precedingText = source.substring(0, span.start);
const line = precedingText.split('\n').length;

return {
filePath: path,
line,
fileSource: source,
assertionText,
};
});
}

/**
* Creates a formatted string for an error stack
* @param callStack - The error stack
* @returns - The formatted string
*/
function printErrorStack(callStack: SourceCodeLocation[]): string {
// TODO experiment with formats of reporting this for better error reporting
return [
'Error: Assertion failed',
callStack.map(call => ` at ${call.filePath}:${call.line} '${call.assertionText}'`),
].join('\n');
}

/**
* The function call that executes an ACIR.
*/
Expand All @@ -79,6 +149,7 @@ export async function acvm(
acir: Buffer,
initialWitness: ACVMWitness,
callback: ACIRCallback,
debug?: FunctionDebugMetadata,
): Promise<ACIRExecutionResult> {
const logger = createDebugLogger('aztec:simulator:acvm');
const partialWitness = await executeCircuitWithBlackBoxSolver(
Expand All @@ -100,7 +171,25 @@ export async function acvm(
throw err;
}
},
);
).catch(err => {
// ACVM_js throws raw string errors
if (typeof err !== 'string') {
throw err;
}

const opcodeLocation = extractOpcodeLocationFromError(err);
if (!opcodeLocation || !debug) {
throw err;
}

const callStack = getCallStackFromOpcodeLocation(opcodeLocation, debug);
logger(printErrorStack(callStack));

// The ACVM only lets string errors pass through so we need to throw a string at the execution level.
// We should probably update the ACVM to let proper errors through.
throw `Assertion failed: '${callStack.pop()?.assertionText ?? 'Unknown'}'`;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But the first if throws the error when it's not a string. Should we convert that to a string or is it actually ok to throw new Error(str)?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The problem is that the acvm js only lets error strings through, so if we emitted new Error(str), if the execution of this ACVM was child of another ACVM, when passing through it'd get transformed to Error in oracle call: unknown or something like that ):

Maybe we can ignore the ACVM error completely if there was an error in a nested call...

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I might try that so we have proper errors from the simulator executions

});

return Promise.resolve({ partialWitness });
}

Expand Down
28 changes: 26 additions & 2 deletions yarn-project/acir-simulator/src/client/db_oracle.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { CompleteAddress, HistoricBlockData, PrivateKey, PublicKey } from '@aztec/circuits.js';
import { FunctionAbi } from '@aztec/foundation/abi';
import { DebugFileMap, DebugInfo, FunctionAbi } from '@aztec/foundation/abi';
import { AztecAddress } from '@aztec/foundation/aztec-address';
import { EthAddress } from '@aztec/foundation/eth-address';
import { Fr } from '@aztec/foundation/fields';
Expand Down Expand Up @@ -67,6 +67,30 @@ export interface CommitmentDataOracleInputs {
index: bigint;
}

/**
* Debug metadata for a function.
*/
export interface FunctionDebugMetadata {
/**
* Maps opcodes to source code pointers
*/
debugSymbols: DebugInfo;
/**
* Maps the file IDs to the file contents to resolve pointers
*/
files: DebugFileMap;
}

/**
* A function ABI with optional debug metadata
*/
export interface FunctionAbiWithDebugMetadata extends FunctionAbi {
/**
* Debug metadata for the function.
*/
debug?: FunctionDebugMetadata;
}

/**
* The database oracle interface.
*/
Expand Down Expand Up @@ -109,7 +133,7 @@ export interface DBOracle extends CommitmentsDB {
* @param functionSelector - The Buffer containing the function selector bytes.
* @returns A Promise that resolves to a FunctionAbi object containing the ABI information of the target function.
*/
getFunctionABI(contractAddress: AztecAddress, functionSelector: Buffer): Promise<FunctionAbi>;
getFunctionABI(contractAddress: AztecAddress, functionSelector: Buffer): Promise<FunctionAbiWithDebugMetadata>;

/**
* Retrieves the portal contract address associated with the given contract address.
Expand Down
40 changes: 20 additions & 20 deletions yarn-project/acir-simulator/src/client/private_execution.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ import { MockProxy, mock } from 'jest-mock-extended';
import { default as levelup } from 'levelup';
import { type MemDown, default as memdown } from 'memdown';

import { buildL1ToL2Message } from '../test/utils.js';
import { buildL1ToL2Message, getFunctionAbi } from '../test/utils.js';
import { computeSlotForMapping } from '../utils.js';
import { DBOracle } from './db_oracle.js';
import { AcirSimulator } from './simulator.js';
Expand Down Expand Up @@ -154,7 +154,7 @@ describe('Private Execution test suite', () => {

describe('empty constructor', () => {
it('should run the empty constructor', async () => {
const abi = TestContractAbi.functions[0];
const abi = getFunctionAbi(TestContractAbi, 'constructor');
const contractDeploymentData = makeContractDeploymentData(100);
const txContext = { isContractDeploymentTx: true, contractDeploymentData };
const result = await runSimulator({ abi, txContext });
Expand Down Expand Up @@ -251,7 +251,7 @@ describe('Private Execution test suite', () => {
});

it('should a constructor with arguments that inserts notes', async () => {
const abi = PrivateTokenAirdropContractAbi.functions.find(f => f.name === 'constructor')!;
const abi = getFunctionAbi(PrivateTokenAirdropContractAbi, 'constructor');

const result = await runSimulator({ args: [140, owner], abi });

Expand All @@ -269,7 +269,7 @@ describe('Private Execution test suite', () => {
});

it('should run the mint function', async () => {
const abi = PrivateTokenAirdropContractAbi.functions.find(f => f.name === 'mint')!;
const abi = getFunctionAbi(PrivateTokenAirdropContractAbi, 'mint');

const result = await runSimulator({ args: [140, owner], abi });

Expand All @@ -288,7 +288,7 @@ describe('Private Execution test suite', () => {

it('should run the transfer function', async () => {
const amountToTransfer = 100n;
const abi = PrivateTokenAirdropContractAbi.functions.find(f => f.name === 'transfer')!;
const abi = getFunctionAbi(PrivateTokenAirdropContractAbi, 'transfer');

const storageSlot = computeSlotForMapping(new Fr(1n), owner.toField(), circuitsWasm);
const recipientStorageSlot = computeSlotForMapping(new Fr(1n), recipient.toField(), circuitsWasm);
Expand Down Expand Up @@ -333,7 +333,7 @@ describe('Private Execution test suite', () => {
it('should be able to transfer with dummy notes', async () => {
const amountToTransfer = 100n;
const balance = 160n;
const abi = PrivateTokenAirdropContractAbi.functions.find(f => f.name === 'transfer')!;
const abi = getFunctionAbi(PrivateTokenAirdropContractAbi, 'transfer');

const storageSlot = computeSlotForMapping(new Fr(1n), owner.toField(), circuitsWasm);

Expand All @@ -360,7 +360,7 @@ describe('Private Execution test suite', () => {
it('Should be able to claim a note by providing the correct secret', async () => {
const amount = 100n;
const secret = Fr.random();
const abi = PrivateTokenAirdropContractAbi.functions.find(f => f.name === 'claim')!;
const abi = getFunctionAbi(PrivateTokenAirdropContractAbi, 'claim');
const storageSlot = new Fr(2n);
// choose nonzero nonce otherwise reads will be interpreted as transient (inner note hash instead of unique+siloed)
const nonce = new Fr(1n);
Expand Down Expand Up @@ -479,7 +479,7 @@ describe('Private Execution test suite', () => {
});

it('should a constructor with arguments that inserts notes', async () => {
const abi = PrivateTokenContractAbi.functions.find(f => f.name === 'constructor')!;
const abi = getFunctionAbi(PrivateTokenContractAbi, 'constructor');

const result = await runSimulator({ args: [140, owner], abi });

Expand All @@ -497,7 +497,7 @@ describe('Private Execution test suite', () => {
});

it('should run the mint function', async () => {
const abi = PrivateTokenContractAbi.functions.find(f => f.name === 'mint')!;
const abi = getFunctionAbi(PrivateTokenContractAbi, 'mint');

const result = await runSimulator({ args: [140, owner], abi });

Expand All @@ -516,7 +516,7 @@ describe('Private Execution test suite', () => {

it('should run the transfer function', async () => {
const amountToTransfer = 100n;
const abi = PrivateTokenContractAbi.functions.find(f => f.name === 'transfer')!;
const abi = getFunctionAbi(PrivateTokenContractAbi, 'transfer');

const storageSlot = computeSlotForMapping(new Fr(1n), owner.toField(), circuitsWasm);
const recipientStorageSlot = computeSlotForMapping(new Fr(1n), recipient.toField(), circuitsWasm);
Expand Down Expand Up @@ -561,7 +561,7 @@ describe('Private Execution test suite', () => {
it('should be able to transfer with dummy notes', async () => {
const amountToTransfer = 100n;
const balance = 160n;
const abi = PrivateTokenContractAbi.functions.find(f => f.name === 'transfer')!;
const abi = getFunctionAbi(PrivateTokenContractAbi, 'transfer');

const storageSlot = computeSlotForMapping(new Fr(1n), owner.toField(), circuitsWasm);

Expand Down Expand Up @@ -591,15 +591,15 @@ describe('Private Execution test suite', () => {

it('child function should be callable', async () => {
const initialValue = 100n;
const abi = ChildContractAbi.functions.find(f => f.name === 'value')!;
const abi = getFunctionAbi(ChildContractAbi, 'value');
const result = await runSimulator({ args: [initialValue], abi });

expect(result.callStackItem.publicInputs.returnValues[0]).toEqual(new Fr(initialValue + privateIncrement));
});

it('parent should call child', async () => {
const childAbi = ChildContractAbi.functions.find(f => f.name === 'value')!;
const parentAbi = ParentContractAbi.functions.find(f => f.name === 'entryPoint')!;
const childAbi = getFunctionAbi(ChildContractAbi, 'value');
const parentAbi = getFunctionAbi(ParentContractAbi, 'entryPoint');
const parentAddress = AztecAddress.random();
const childAddress = AztecAddress.random();
const childSelector = generateFunctionSelector(childAbi.name, childAbi.parameters);
Expand Down Expand Up @@ -686,7 +686,7 @@ describe('Private Execution test suite', () => {

it('Should be able to consume a dummy cross chain message', async () => {
const bridgedAmount = 100n;
const abi = NonNativeTokenContractAbi.functions.find(f => f.name === 'mint')!;
const abi = getFunctionAbi(NonNativeTokenContractAbi, 'mint');

const secret = new Fr(1n);
const canceller = EthAddress.random();
Expand Down Expand Up @@ -720,7 +720,7 @@ describe('Private Execution test suite', () => {

it('Should be able to consume a dummy public to private message', async () => {
const amount = 100n;
const abi = NonNativeTokenContractAbi.functions.find(f => f.name === 'redeemShield')!;
const abi = getFunctionAbi(NonNativeTokenContractAbi, 'redeemShield');

const wasm = await CircuitsWasm.get();
const secret = new Fr(1n);
Expand Down Expand Up @@ -998,7 +998,7 @@ describe('Private Execution test suite', () => {
describe('get public key', () => {
it('gets the public key for an address', async () => {
// Tweak the contract ABI so we can extract return values
const abi = TestContractAbi.functions.find(f => f.name === 'getPublicKey')!;
const abi = getFunctionAbi(TestContractAbi, 'getPublicKey');
abi.returnTypes = [{ kind: 'array', length: 2, type: { kind: 'field' } }];

// Generate a partial address, pubkey, and resulting address
Expand All @@ -1018,7 +1018,7 @@ describe('Private Execution test suite', () => {
const aztecAddressToQuery = AztecAddress.random();

// Tweak the contract ABI so we can extract return values
const abi = TestContractAbi.functions.find(f => f.name === 'getPortalContractAddress')!;
const abi = getFunctionAbi(TestContractAbi, 'getPortalContractAddress');
abi.returnTypes = [{ kind: 'field' }];

const args = [aztecAddressToQuery.toField()];
Expand All @@ -1033,7 +1033,7 @@ describe('Private Execution test suite', () => {
const contractAddress = AztecAddress.random();

// Tweak the contract ABI so we can extract return values
const abi = TestContractAbi.functions.find(f => f.name === 'getThisAddress')!;
const abi = getFunctionAbi(TestContractAbi, 'getThisAddress');
abi.returnTypes = [{ kind: 'field' }];

// Overwrite the oracle return value
Expand All @@ -1045,7 +1045,7 @@ describe('Private Execution test suite', () => {
const portalContractAddress = EthAddress.random();

// Tweak the contract ABI so we can extract return values
const abi = TestContractAbi.functions.find(f => f.name === 'getThisPortalAddress')!;
const abi = getFunctionAbi(TestContractAbi, 'getThisPortalAddress');
abi.returnTypes = [{ kind: 'field' }];

// Overwrite the oracle return value
Expand Down
Loading