@aztec/simulator 5.0.0-private.20260318 → 5.0.0-rc.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (80) hide show
  1. package/dest/private/acvm_wasm.d.ts +1 -1
  2. package/dest/private/acvm_wasm.d.ts.map +1 -1
  3. package/dest/private/acvm_wasm.js +3 -1
  4. package/dest/public/avm/avm_simulator.js +1 -1
  5. package/dest/public/avm/calldata.d.ts +1 -1
  6. package/dest/public/avm/calldata.d.ts.map +1 -1
  7. package/dest/public/avm/calldata.js +3 -2
  8. package/dest/public/avm/fixtures/base_avm_simulation_tester.d.ts +2 -1
  9. package/dest/public/avm/fixtures/base_avm_simulation_tester.d.ts.map +1 -1
  10. package/dest/public/avm/fixtures/base_avm_simulation_tester.js +17 -2
  11. package/dest/public/avm/fixtures/utils.d.ts +1 -1
  12. package/dest/public/avm/fixtures/utils.d.ts.map +1 -1
  13. package/dest/public/avm/fixtures/utils.js +3 -0
  14. package/dest/public/avm/opcodes/contract.d.ts +3 -2
  15. package/dest/public/avm/opcodes/contract.d.ts.map +1 -1
  16. package/dest/public/avm/opcodes/contract.js +5 -1
  17. package/dest/public/avm/opcodes/ec_add.d.ts +2 -4
  18. package/dest/public/avm/opcodes/ec_add.d.ts.map +1 -1
  19. package/dest/public/avm/opcodes/ec_add.js +13 -27
  20. package/dest/public/contracts_db_checkpoint.d.ts +2 -2
  21. package/dest/public/contracts_db_checkpoint.d.ts.map +1 -1
  22. package/dest/public/contracts_db_checkpoint.js +1 -1
  23. package/dest/public/fixtures/bulk_test.d.ts +1 -1
  24. package/dest/public/fixtures/bulk_test.d.ts.map +1 -1
  25. package/dest/public/fixtures/bulk_test.js +31 -4
  26. package/dest/public/fixtures/custom_bytecode_tests.d.ts +3 -1
  27. package/dest/public/fixtures/custom_bytecode_tests.d.ts.map +1 -1
  28. package/dest/public/fixtures/custom_bytecode_tests.js +61 -1
  29. package/dest/public/fixtures/opcode_spammer.d.ts +1 -1
  30. package/dest/public/fixtures/opcode_spammer.d.ts.map +1 -1
  31. package/dest/public/fixtures/opcode_spammer.js +2 -10
  32. package/dest/public/fixtures/public_tx_simulation_tester.d.ts +1 -1
  33. package/dest/public/fixtures/public_tx_simulation_tester.d.ts.map +1 -1
  34. package/dest/public/fixtures/public_tx_simulation_tester.js +4 -2
  35. package/dest/public/fixtures/utils.d.ts +1 -1
  36. package/dest/public/fixtures/utils.d.ts.map +1 -1
  37. package/dest/public/fixtures/utils.js +15 -14
  38. package/dest/public/hinting_db_sources.d.ts +1 -1
  39. package/dest/public/hinting_db_sources.d.ts.map +1 -1
  40. package/dest/public/hinting_db_sources.js +1 -1
  41. package/dest/public/public_db_sources.d.ts +6 -3
  42. package/dest/public/public_db_sources.d.ts.map +1 -1
  43. package/dest/public/public_db_sources.js +16 -10
  44. package/dest/public/public_processor/public_processor.d.ts +4 -3
  45. package/dest/public/public_processor/public_processor.d.ts.map +1 -1
  46. package/dest/public/public_processor/public_processor.js +80 -35
  47. package/dest/public/public_processor/public_processor_metrics.d.ts +4 -1
  48. package/dest/public/public_processor/public_processor_metrics.d.ts.map +1 -1
  49. package/dest/public/public_processor/public_processor_metrics.js +8 -0
  50. package/dest/public/public_tx_simulator/contract_provider_for_cpp.d.ts +1 -1
  51. package/dest/public/public_tx_simulator/contract_provider_for_cpp.d.ts.map +1 -1
  52. package/dest/public/public_tx_simulator/contract_provider_for_cpp.js +3 -2
  53. package/dest/public/public_tx_simulator/public_tx_context.d.ts +7 -3
  54. package/dest/public/public_tx_simulator/public_tx_context.d.ts.map +1 -1
  55. package/dest/public/public_tx_simulator/public_tx_context.js +8 -10
  56. package/dest/public/public_tx_simulator/public_tx_simulator.d.ts +6 -2
  57. package/dest/public/public_tx_simulator/public_tx_simulator.d.ts.map +1 -1
  58. package/dest/public/public_tx_simulator/public_tx_simulator.js +12 -7
  59. package/package.json +16 -15
  60. package/src/private/acvm_wasm.ts +4 -1
  61. package/src/public/avm/avm_simulator.ts +1 -1
  62. package/src/public/avm/calldata.ts +3 -2
  63. package/src/public/avm/fixtures/base_avm_simulation_tester.ts +22 -2
  64. package/src/public/avm/fixtures/utils.ts +3 -0
  65. package/src/public/avm/opcodes/contract.ts +5 -1
  66. package/src/public/avm/opcodes/ec_add.ts +12 -31
  67. package/src/public/avm/opcodes/external_calls.ts +1 -1
  68. package/src/public/contracts_db_checkpoint.ts +1 -1
  69. package/src/public/fixtures/bulk_test.ts +29 -4
  70. package/src/public/fixtures/custom_bytecode_tests.ts +139 -1
  71. package/src/public/fixtures/opcode_spammer.ts +4 -8
  72. package/src/public/fixtures/public_tx_simulation_tester.ts +4 -10
  73. package/src/public/fixtures/utils.ts +17 -19
  74. package/src/public/hinting_db_sources.ts +1 -0
  75. package/src/public/public_db_sources.ts +21 -14
  76. package/src/public/public_processor/public_processor.ts +101 -42
  77. package/src/public/public_processor/public_processor_metrics.ts +12 -0
  78. package/src/public/public_tx_simulator/contract_provider_for_cpp.ts +3 -2
  79. package/src/public/public_tx_simulator/public_tx_context.ts +8 -10
  80. package/src/public/public_tx_simulator/public_tx_simulator.ts +11 -7
@@ -63,7 +63,7 @@ export class AvmSimulator implements AvmSimulatorInterface {
63
63
  public static async build(context: AvmContext): Promise<AvmSimulator> {
64
64
  const simulator = new AvmSimulator(context);
65
65
  const fnName = await context.persistableState.getPublicFunctionDebugName(context.environment);
66
- simulator.log = createLogger(`simulator:avm(f:${fnName})`);
66
+ simulator.log = createLogger(`simulator:avm(f:${fnName.slice(0, 128)})`);
67
67
 
68
68
  return simulator;
69
69
  }
@@ -33,13 +33,14 @@ export class LazyReaderMemory implements LazyReader {
33
33
  }
34
34
 
35
35
  public slice(start: number, end: number): Fr[] {
36
- const clampedEnd = Math.min(end, this.size);
36
+ const clampedEnd = Math.min(end, this.size, TaggedMemory.MAX_MEMORY_SIZE - this.offset);
37
37
  const length = Math.max(0, clampedEnd - start);
38
38
  return this.memory.getSlice(this.offset + start, length).map(word => word.toFr());
39
39
  }
40
40
 
41
41
  public readAll(): Fr[] {
42
- return this.memory.getSlice(this.offset, this.size).map(word => word.toFr());
42
+ const size = Math.min(this.size, TaggedMemory.MAX_MEMORY_SIZE - this.offset);
43
+ return this.memory.getSlice(this.offset, size).map(word => word.toFr());
43
44
  }
44
45
 
45
46
  public length(): number {
@@ -2,9 +2,10 @@ import { CONTRACT_INSTANCE_REGISTRY_CONTRACT_ADDRESS } from '@aztec/constants';
2
2
  import { Fr } from '@aztec/foundation/curves/bn254';
3
3
  import { createLogger } from '@aztec/foundation/log';
4
4
  import { ProtocolContractAddress } from '@aztec/protocol-contracts';
5
- import { getCanonicalAuthRegistry } from '@aztec/protocol-contracts/auth-registry';
5
+ import { getCanonicalClassRegistry } from '@aztec/protocol-contracts/class-registry';
6
6
  import { computeFeePayerBalanceStorageSlot, getCanonicalFeeJuice } from '@aztec/protocol-contracts/fee-juice';
7
7
  import { getCanonicalInstanceRegistry } from '@aztec/protocol-contracts/instance-registry';
8
+ import { getStandardAuthRegistry } from '@aztec/standard-contracts/auth-registry';
8
9
  import type { ContractArtifact } from '@aztec/stdlib/abi';
9
10
  import { PublicDataWrite } from '@aztec/stdlib/avm';
10
11
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
@@ -91,7 +92,7 @@ export abstract class BaseAvmSimulationTester {
91
92
  }
92
93
 
93
94
  async registerAuthContract(): Promise<ContractInstanceWithAddress> {
94
- const authRegistry = await getCanonicalAuthRegistry();
95
+ const authRegistry = await getStandardAuthRegistry();
95
96
  const authRegistryContractClassPublic = {
96
97
  ...authRegistry.contractClass,
97
98
  privateFunctions: [],
@@ -102,9 +103,28 @@ export abstract class BaseAvmSimulationTester {
102
103
  authRegistryContractClassPublic,
103
104
  authRegistry.instance,
104
105
  );
106
+ // AuthRegistry is a standard contract (not a protocol contract), so the AVM enforces a
107
+ // deployment-nullifier check on every instance retrieval. Insert the nullifier here so
108
+ // calls to AuthRegistry don't EXCEPTIONAL_HALT during simulation.
109
+ await this.insertContractAddressNullifier(authRegistry.instance.address);
105
110
  return authRegistry.instance;
106
111
  }
107
112
 
113
+ async registerClassRegistryContract(): Promise<ContractInstanceWithAddress> {
114
+ const classRegistry = await getCanonicalClassRegistry();
115
+ const classRegistryContractClassPublic = {
116
+ ...classRegistry.contractClass,
117
+ privateFunctions: [],
118
+ utilityFunctions: [],
119
+ };
120
+ await this.contractDataSource.addNewContract(
121
+ classRegistry.artifact,
122
+ classRegistryContractClassPublic,
123
+ classRegistry.instance,
124
+ );
125
+ return classRegistry.instance;
126
+ }
127
+
108
128
  async registerInstanceRegistryContract(): Promise<ContractInstanceWithAddress> {
109
129
  const instanceRegistry = await getCanonicalInstanceRegistry();
110
130
  const instanceRegistryContractClassPublic = {
@@ -128,17 +128,20 @@ export async function createContractClassAndInstance(
128
128
  const constructorAbi = getContractFunctionAbi('constructor', contractArtifact);
129
129
  const { publicKeys } = await deriveKeys(new Fr(seed));
130
130
  const initializationHash = await computeInitializationHash(constructorAbi, constructorArgs);
131
+ const immutablesHash = new Fr(seed + 1);
131
132
  const contractInstance =
132
133
  originalContractClassId === undefined
133
134
  ? await makeContractInstanceFromClassId(contractClass.id, seed, {
134
135
  deployer,
135
136
  initializationHash,
137
+ immutablesHash,
136
138
  publicKeys,
137
139
  })
138
140
  : await makeContractInstanceFromClassId(originalContractClassId, seed, {
139
141
  deployer,
140
142
  initializationHash,
141
143
  currentClassId: contractClass.id,
144
+ immutablesHash,
142
145
  publicKeys,
143
146
  });
144
147
 
@@ -9,6 +9,7 @@ export enum ContractInstanceMember {
9
9
  DEPLOYER,
10
10
  CLASS_ID,
11
11
  INIT_HASH,
12
+ IMMUTABLES_HASH,
12
13
  }
13
14
 
14
15
  export class GetContractInstance extends Instruction {
@@ -41,7 +42,7 @@ export class GetContractInstance extends Instruction {
41
42
  );
42
43
 
43
44
  if (!(this.memberEnum in ContractInstanceMember)) {
44
- throw new InstructionExecutionError(`Invalid GETCONSTRACTINSTANCE member enum ${this.memberEnum}`);
45
+ throw new InstructionExecutionError(`Invalid GETCONTRACTINSTANCE member enum ${this.memberEnum}`);
45
46
  }
46
47
 
47
48
  const operands = [this.addressOffset, this.dstOffset];
@@ -64,6 +65,9 @@ export class GetContractInstance extends Instruction {
64
65
  case ContractInstanceMember.INIT_HASH:
65
66
  memberValue = new Field(instance.initializationHash);
66
67
  break;
68
+ case ContractInstanceMember.IMMUTABLES_HASH:
69
+ memberValue = new Field(instance.immutablesHash);
70
+ break;
67
71
  }
68
72
  }
69
73
 
@@ -2,7 +2,7 @@ import { Grumpkin } from '@aztec/foundation/crypto/grumpkin';
2
2
  import { Point } from '@aztec/foundation/curves/grumpkin';
3
3
 
4
4
  import type { AvmContext } from '../avm_context.js';
5
- import { Field, TypeTag, Uint1 } from '../avm_memory_types.js';
5
+ import { Field, TypeTag } from '../avm_memory_types.js';
6
6
  import { EcAddPointNotOnCurveError } from '../errors.js';
7
7
  import { Opcode, OperandType } from '../serialization/instruction_serialization.js';
8
8
  import { Addressing } from './addressing_mode.js';
@@ -18,10 +18,8 @@ export class EcAdd extends Instruction {
18
18
  OperandType.UINT16, // indirect
19
19
  OperandType.UINT16, // p1X
20
20
  OperandType.UINT16, // p1Y
21
- OperandType.UINT16, // p1IsInfinite
22
21
  OperandType.UINT16, // p2X
23
22
  OperandType.UINT16, // p2Y
24
- OperandType.UINT16, // p2IsInfinite
25
23
  OperandType.UINT16, // dst
26
24
  ];
27
25
 
@@ -29,10 +27,8 @@ export class EcAdd extends Instruction {
29
27
  private addressingMode: number,
30
28
  private p1XOffset: number,
31
29
  private p1YOffset: number,
32
- private p1IsInfiniteOffset: number,
33
30
  private p2XOffset: number,
34
31
  private p2YOffset: number,
35
- private p2IsInfiniteOffset: number,
36
32
  private dstOffset: number,
37
33
  ) {
38
34
  super();
@@ -46,51 +42,38 @@ export class EcAdd extends Instruction {
46
42
  this.baseGasCost(addressing.indirectOperandsCount(), addressing.relativeOperandsCount()),
47
43
  );
48
44
 
49
- const operands = [
50
- this.p1XOffset,
51
- this.p1YOffset,
52
- this.p1IsInfiniteOffset,
53
- this.p2XOffset,
54
- this.p2YOffset,
55
- this.p2IsInfiniteOffset,
56
- this.dstOffset,
57
- ];
58
- const [p1XOffset, p1YOffset, p1IsInfiniteOffset, p2XOffset, p2YOffset, p2IsInfiniteOffset, dstOffset] =
59
- addressing.resolve(operands, memory);
45
+ const operands = [this.p1XOffset, this.p1YOffset, this.p2XOffset, this.p2YOffset, this.dstOffset];
46
+ const [p1XOffset, p1YOffset, p2XOffset, p2YOffset, dstOffset] = addressing.resolve(operands, memory);
60
47
 
61
48
  memory.checkTags(TypeTag.FIELD, p1XOffset, p1YOffset, p2XOffset, p2YOffset);
62
- memory.checkTags(TypeTag.UINT1, p1IsInfiniteOffset, p2IsInfiniteOffset);
63
49
 
64
50
  const p1X = memory.get(p1XOffset);
65
51
  const p1Y = memory.get(p1YOffset);
66
- const p1IsInfinite = memory.get(p1IsInfiniteOffset).toNumber() === 1;
67
- const p1 = new Point(p1X.toFr(), p1Y.toFr(), p1IsInfinite);
68
- if (!p1.isOnGrumpkin()) {
52
+ const p1 = new Point(p1X.toFr(), p1Y.toFr());
53
+ if (!p1.isOnCurve()) {
69
54
  throw new EcAddPointNotOnCurveError(/*pointIndex=*/ 1, p1);
70
55
  }
71
56
 
72
57
  const p2X = memory.get(p2XOffset);
73
58
  const p2Y = memory.get(p2YOffset);
74
- // unused. Point doesn't store this information
75
- const p2IsInfinite = memory.get(p2IsInfiniteOffset).toNumber() === 1;
76
- const p2 = new Point(p2X.toFr(), p2Y.toFr(), p2IsInfinite);
77
- if (!p2.isOnGrumpkin()) {
59
+ const p2 = new Point(p2X.toFr(), p2Y.toFr());
60
+ if (!p2.isOnCurve()) {
78
61
  throw new EcAddPointNotOnCurveError(/*pointIndex=*/ 2, p2);
79
62
  }
80
63
 
81
64
  let dest;
82
- if (p1IsInfinite && p2IsInfinite) {
83
- dest = Point.ZERO;
84
- } else if (p1IsInfinite) {
65
+ if (p1.isInfinite && p2.isInfinite) {
66
+ dest = Point.INFINITY;
67
+ } else if (p1.isInfinite) {
85
68
  dest = p2;
86
- } else if (p2IsInfinite) {
69
+ } else if (p2.isInfinite) {
87
70
  dest = p1;
88
71
  } else {
89
72
  // TS<>BB ecc add communication is broken for points that add up to infinity.
90
73
  // However, here we know that both points are on the curve, and that none is infinity
91
74
  // so we can check for the case where you add p + (-p) = infinity.
92
75
  if (p1.x.equals(p2.x) && !p1.y.equals(p2.y)) {
93
- dest = Point.ZERO;
76
+ dest = Point.INFINITY;
94
77
  } else {
95
78
  dest = await Grumpkin.add(p1, p2);
96
79
  }
@@ -99,7 +82,5 @@ export class EcAdd extends Instruction {
99
82
  // Important to use setSlice() and not set() in the two following statements as
100
83
  // this checks that the offsets lie within memory range.
101
84
  memory.setSlice(dstOffset, [new Field(dest.x), new Field(dest.y)]);
102
- // Check representation of infinity for grumpkin
103
- memory.setSlice(dstOffset + 2, [new Uint1(dest.equals(Point.ZERO) ? 1 : 0)]);
104
85
  }
105
86
  }
@@ -14,8 +14,8 @@ abstract class ExternalCall extends Instruction {
14
14
  OperandType.UINT16, // L2 gas offset
15
15
  OperandType.UINT16, // DA gas offset
16
16
  OperandType.UINT16, // Address offset
17
- OperandType.UINT16, // Args offset
18
17
  OperandType.UINT16, // Args size offset
18
+ OperandType.UINT16, // Args offset
19
19
  ];
20
20
 
21
21
  constructor(
@@ -31,7 +31,7 @@ export class ContractsDbCheckpoint {
31
31
  return this.bytecodeCommitments.get(classId.toString());
32
32
  }
33
33
 
34
- public deepCopy(): ContractsDbCheckpoint {
34
+ public fork(): ContractsDbCheckpoint {
35
35
  const copy = new ContractsDbCheckpoint();
36
36
  this.instances.forEach((value, key) => copy.instances.set(key, value));
37
37
  this.classes.forEach((value, key) => copy.classes.set(key, value));
@@ -25,14 +25,26 @@ export async function bulkTest(
25
25
  // Register multiple different protocol contracts (to ensure we don't dedup bytecode hashing events):
26
26
  await tester.registerAuthContract();
27
27
  await tester.registerInstanceRegistryContract();
28
+ await tester.registerClassRegistryContract();
28
29
 
29
30
  // Get a deployed contract instance to pass to the contract
30
31
  // for it to use as "expected" values when testing contract instance retrieval.
31
32
  const expectContractInstance = avmTestContract;
32
33
  const argsField = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(x => new Fr(x));
33
- const argsU8 = [1, 2, 3, 4, 5, 6, 7, 8].map(x => new Fr(x));
34
- argsU8.push(new Fr(2n ** 128n + 9n)); // Trigger truncation from large (> 128 bits) value (canonical decomposition event)
35
- argsU8.push(new Fr(2n ** 125n + 10n)); // Trigger truncation from small (< 128 bits) value (no canonical decomposition event)
34
+ const argsU8 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(x => new Fr(x));
35
+ // Pinned grumpkin-Poseidon2 Schnorr signature (mirrors the C++ `pinned_test_vector_large`
36
+ // and noir-lang/schnorr v0.4.0's `pinned_vector_large`). Passing these in as calldata
37
+ // (rather than baking them into Noir as constants) keeps MSM + Poseidon2 from being folded
38
+ // by the Noir compiler.
39
+ const schnorrInputs = [
40
+ Fr.fromHexString('0x065812e335a97c2108ea8cf4ccfe2f9dd6b117a0714f5e18461575be93f61da6'), // pubkey.x
41
+ Fr.fromHexString('0x1a915003e8ec534f9a15d926a7ded478e178468ccc4f28e236e67450a55ac622'), // pubkey.y
42
+ Fr.fromHexString('0xf3bc3b7147acb9c621fd9f72dbf15ffa'), // sig_s.lo
43
+ Fr.fromHexString('0x08599f379f0301dfefdbd0272554454d'), // sig_s.hi
44
+ Fr.fromHexString('0x97065383ebbbd76620398792bd259bc2'), // sig_e.lo
45
+ Fr.fromHexString('0x2ceaee87f45b7a417f0ffb05451a8c92'), // sig_e.hi
46
+ Fr.fromHexString('0x0123456789abcdef0fedcba9876543210123456789abcdef0fedcba987654321'), // message
47
+ ];
36
48
  const args = [
37
49
  argsField,
38
50
  argsU8,
@@ -40,6 +52,8 @@ export async function bulkTest(
40
52
  /*expectedDeployer=*/ expectContractInstance.deployer,
41
53
  /*expectedClassId=*/ expectContractInstance.currentContractClassId,
42
54
  /*expectedInitializationHash=*/ expectContractInstance.initializationHash,
55
+ /*expectedImmutablesHash=*/ expectContractInstance.immutablesHash,
56
+ /*schnorrInputs=*/ schnorrInputs,
43
57
  /*skip_strictly_limited_side_effects=*/ false,
44
58
  ];
45
59
 
@@ -62,7 +76,7 @@ export async function bulkTest(
62
76
  {
63
77
  address: avmTestContract.address,
64
78
  fnName: 'assert_calldata_copy',
65
- args: [argsField.slice(3), /* with_selector: */ true],
79
+ args: [argsField.slice(0, 3), /* with_selector: */ true],
66
80
  },
67
81
  {
68
82
  address: avmTestContract.address,
@@ -123,6 +137,15 @@ export async function megaBulkTest(
123
137
  //const argsField7 = [15, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(x => new Fr(x));
124
138
  //const argsField8 = [17, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(x => new Fr(x));
125
139
  const argsU8 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(x => new Fr(x));
140
+ const schnorrInputs = [
141
+ Fr.fromHexString('0x065812e335a97c2108ea8cf4ccfe2f9dd6b117a0714f5e18461575be93f61da6'), // pubkey.x
142
+ Fr.fromHexString('0x1a915003e8ec534f9a15d926a7ded478e178468ccc4f28e236e67450a55ac622'), // pubkey.y
143
+ Fr.fromHexString('0xf3bc3b7147acb9c621fd9f72dbf15ffa'), // sig_s.lo
144
+ Fr.fromHexString('0x08599f379f0301dfefdbd0272554454d'), // sig_s.hi
145
+ Fr.fromHexString('0x97065383ebbbd76620398792bd259bc2'), // sig_e.lo
146
+ Fr.fromHexString('0x2ceaee87f45b7a417f0ffb05451a8c92'), // sig_e.hi
147
+ Fr.fromHexString('0x0123456789abcdef0fedcba9876543210123456789abcdef0fedcba987654321'), // message
148
+ ];
126
149
  const genArgs = (argsField: Fr[]) => [
127
150
  argsField,
128
151
  argsU8,
@@ -130,6 +153,8 @@ export async function megaBulkTest(
130
153
  /*expectedDeployer=*/ expectContractInstance.deployer.toField(),
131
154
  /*expectedClassId=*/ expectContractInstance.currentContractClassId.toField(),
132
155
  /*expectedInitializationHash=*/ expectContractInstance.initializationHash.toField(),
156
+ /*expectedImmutablesHash=*/ expectContractInstance.immutablesHash.toField(),
157
+ /*schnorrInputs=*/ schnorrInputs,
133
158
  // Must skip strictly limited side effects (logs, messages) so we can spam the bulk test several times.
134
159
  /*skip_strictly_limited_side_effects=*/ true,
135
160
  ];
@@ -2,7 +2,7 @@ import { strict as assert } from 'assert';
2
2
 
3
3
  import { TypeTag } from '../avm/avm_memory_types.js';
4
4
  import { Addressing, AddressingMode } from '../avm/opcodes/addressing_mode.js';
5
- import { Add, CalldataCopy, Jump, Return, Set } from '../avm/opcodes/index.js';
5
+ import { Add, CalldataCopy, Cast, Jump, Return, Set } from '../avm/opcodes/index.js';
6
6
  import { encodeToBytecode } from '../avm/serialization/bytecode_serialization.js';
7
7
  import {
8
8
  MAX_OPCODE_VALUE,
@@ -208,6 +208,144 @@ export async function invalidTagValueAndInstructionTruncatedTest(tester: PublicT
208
208
  return await deployAndExecuteCustomBytecode(bytecode, tester, txLabel);
209
209
  }
210
210
 
211
+ // Exercise SET truncation: set values whose widths exceed the target tag and
212
+ // rely on `buildFromTagTruncating` to truncate to the low bits of the tag.
213
+ // Covers sources larger than 128 bits (via SET_FF) and sources in (32, 128]
214
+ // bits (via SET_64) against destination tags U1/U8/U16/U32/U64/U128.
215
+ export async function setTruncationTest(tester: PublicTxSimulationTester) {
216
+ // 200-bit value: forces truncation for every target tag up to U128.
217
+ const LARGE_FIELD_VALUE = (1n << 200n) + 0x1234567890abcdef1234567890abcdefn;
218
+ // 40-bit value: forces truncation for target tags up to U32.
219
+ const LARGE_U64_VALUE = (1n << 40n) + 0xdeadbeefn;
220
+
221
+ const bytecode = encodeToBytecode([
222
+ // Zero U32 at offset 0 — used as the Return copy-size slot.
223
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 0, TypeTag.UINT32, /*value=*/ 0).as(Opcode.SET_8, Set.wireFormat8),
224
+
225
+ // Source >128 bits (via SET_FF) truncated to smaller target tags.
226
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 1, TypeTag.UINT128, LARGE_FIELD_VALUE).as(
227
+ Opcode.SET_FF,
228
+ Set.wireFormatFF,
229
+ ),
230
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 2, TypeTag.UINT64, LARGE_FIELD_VALUE).as(
231
+ Opcode.SET_FF,
232
+ Set.wireFormatFF,
233
+ ),
234
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 3, TypeTag.UINT32, LARGE_FIELD_VALUE).as(
235
+ Opcode.SET_FF,
236
+ Set.wireFormatFF,
237
+ ),
238
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 4, TypeTag.UINT16, LARGE_FIELD_VALUE).as(
239
+ Opcode.SET_FF,
240
+ Set.wireFormatFF,
241
+ ),
242
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 5, TypeTag.UINT8, LARGE_FIELD_VALUE).as(
243
+ Opcode.SET_FF,
244
+ Set.wireFormatFF,
245
+ ),
246
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 6, TypeTag.UINT1, LARGE_FIELD_VALUE).as(
247
+ Opcode.SET_FF,
248
+ Set.wireFormatFF,
249
+ ),
250
+
251
+ // Source in (32, 128] bits (via SET_64) truncated to smaller target tags.
252
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 7, TypeTag.UINT32, LARGE_U64_VALUE).as(
253
+ Opcode.SET_64,
254
+ Set.wireFormat64,
255
+ ),
256
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 8, TypeTag.UINT16, LARGE_U64_VALUE).as(
257
+ Opcode.SET_64,
258
+ Set.wireFormat64,
259
+ ),
260
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 9, TypeTag.UINT8, LARGE_U64_VALUE).as(
261
+ Opcode.SET_64,
262
+ Set.wireFormat64,
263
+ ),
264
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 10, TypeTag.UINT1, LARGE_U64_VALUE).as(
265
+ Opcode.SET_64,
266
+ Set.wireFormat64,
267
+ ),
268
+
269
+ new Return(/*addressing_mode=*/ 0, /*copySizeOffset=*/ 0, /*returnOffset=*/ 0),
270
+ ]);
271
+
272
+ const txLabel = 'SetTruncation';
273
+ return await deployAndExecuteCustomBytecode(bytecode, tester, txLabel);
274
+ }
275
+
276
+ // Exercise CAST truncation: store a wide source value in memory then CAST it
277
+ // to smaller destination tags. Covers sources larger than 128 bits (FIELD
278
+ // source) and sources in (32, 128] bits (UINT64 source) against destination
279
+ // tags U1/U8/U16/U32/U64/U128.
280
+ export async function castTruncationTest(tester: PublicTxSimulationTester) {
281
+ // 200-bit source: stored as FIELD so that CASTs to any integer tag truncate.
282
+ const LARGE_FIELD_VALUE = (1n << 200n) + 0x1234567890abcdef1234567890abcdefn;
283
+ // 40-bit source: stored as UINT64 so CASTs to U1/U8/U16/U32 truncate.
284
+ const LARGE_U64_VALUE = (1n << 40n) + 0xdeadbeefn;
285
+
286
+ const bytecode = encodeToBytecode([
287
+ // Zero U32 at offset 0 — used as the Return copy-size slot.
288
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 0, TypeTag.UINT32, /*value=*/ 0).as(Opcode.SET_8, Set.wireFormat8),
289
+
290
+ // Store wide FIELD source at offset 10, then CAST to smaller tags.
291
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 10, TypeTag.FIELD, LARGE_FIELD_VALUE).as(
292
+ Opcode.SET_FF,
293
+ Set.wireFormatFF,
294
+ ),
295
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 10, /*dstOffset=*/ 11, TypeTag.UINT128).as(
296
+ Opcode.CAST_8,
297
+ Cast.wireFormat8,
298
+ ),
299
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 10, /*dstOffset=*/ 12, TypeTag.UINT64).as(
300
+ Opcode.CAST_8,
301
+ Cast.wireFormat8,
302
+ ),
303
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 10, /*dstOffset=*/ 13, TypeTag.UINT32).as(
304
+ Opcode.CAST_8,
305
+ Cast.wireFormat8,
306
+ ),
307
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 10, /*dstOffset=*/ 14, TypeTag.UINT16).as(
308
+ Opcode.CAST_8,
309
+ Cast.wireFormat8,
310
+ ),
311
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 10, /*dstOffset=*/ 15, TypeTag.UINT8).as(
312
+ Opcode.CAST_8,
313
+ Cast.wireFormat8,
314
+ ),
315
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 10, /*dstOffset=*/ 16, TypeTag.UINT1).as(
316
+ Opcode.CAST_8,
317
+ Cast.wireFormat8,
318
+ ),
319
+
320
+ // Store UINT64 source at offset 20, then CAST to smaller integer tags.
321
+ new Set(/*addressing_mode=*/ 0, /*dstOffset=*/ 20, TypeTag.UINT64, LARGE_U64_VALUE).as(
322
+ Opcode.SET_64,
323
+ Set.wireFormat64,
324
+ ),
325
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 20, /*dstOffset=*/ 21, TypeTag.UINT32).as(
326
+ Opcode.CAST_8,
327
+ Cast.wireFormat8,
328
+ ),
329
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 20, /*dstOffset=*/ 22, TypeTag.UINT16).as(
330
+ Opcode.CAST_8,
331
+ Cast.wireFormat8,
332
+ ),
333
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 20, /*dstOffset=*/ 23, TypeTag.UINT8).as(
334
+ Opcode.CAST_8,
335
+ Cast.wireFormat8,
336
+ ),
337
+ new Cast(/*addressing_mode=*/ 0, /*srcOffset=*/ 20, /*dstOffset=*/ 24, TypeTag.UINT1).as(
338
+ Opcode.CAST_8,
339
+ Cast.wireFormat8,
340
+ ),
341
+
342
+ new Return(/*addressing_mode=*/ 0, /*copySizeOffset=*/ 0, /*returnOffset=*/ 0),
343
+ ]);
344
+
345
+ const txLabel = 'CastTruncation';
346
+ return await deployAndExecuteCustomBytecode(bytecode, tester, txLabel);
347
+ }
348
+
211
349
  /**
212
350
  * Returns the offset of the tag in an instruction.
213
351
  * @details Loops over the wire format operand type entries until it finds the tag.
@@ -1339,20 +1339,16 @@ export const SPAM_CONFIGS: Partial<Record<Opcode, SpamConfig[]>> = {
1339
1339
  setup: [
1340
1340
  { offset: 0, value: new Field(Grumpkin.generator.x) }, // p1X = G.x
1341
1341
  { offset: 1, value: new Field(Grumpkin.generator.y) }, // p1Y = G.y
1342
- { offset: 2, value: new Uint1(0n) }, // p1IsInfinite = false
1343
- { offset: 3, value: new Field(Grumpkin.generator.x) }, // p2X = G.x
1344
- { offset: 4, value: new Field(Grumpkin.generator.y) }, // p2Y = G.y
1345
- { offset: 5, value: new Uint1(0n) }, // p2IsInfinite = false
1342
+ { offset: 2, value: new Field(Grumpkin.generator.x) }, // p2X = G.x
1343
+ { offset: 3, value: new Field(Grumpkin.generator.y) }, // p2Y = G.y
1346
1344
  ],
1347
1345
  targetInstructions: () => [
1348
1346
  new EcAdd(
1349
1347
  /*addressing_mode=*/ 0,
1350
1348
  /*p1XOffset=*/ 0,
1351
1349
  /*p1YOffset=*/ 1,
1352
- /*p1IsInfiniteOffset=*/ 2,
1353
- /*p2XOffset=*/ 3,
1354
- /*p2YOffset=*/ 4,
1355
- /*p2IsInfiniteOffset=*/ 5,
1350
+ /*p2XOffset=*/ 2,
1351
+ /*p2YOffset=*/ 3,
1356
1352
  /*dstOffset=*/ 0,
1357
1353
  ),
1358
1354
  ],
@@ -1,9 +1,4 @@
1
- import {
2
- DEFAULT_TEARDOWN_DA_GAS_LIMIT,
3
- DEFAULT_TEARDOWN_L2_GAS_LIMIT,
4
- PUBLIC_TX_L2_GAS_OVERHEAD,
5
- TX_DA_GAS_OVERHEAD,
6
- } from '@aztec/constants';
1
+ import { PUBLIC_TX_L2_GAS_OVERHEAD, TX_DA_GAS_OVERHEAD } from '@aztec/constants';
7
2
  import { asyncMap } from '@aztec/foundation/async-map';
8
3
  import { BlockNumber } from '@aztec/foundation/branded-types';
9
4
  import { Fr } from '@aztec/foundation/curves/bn254';
@@ -32,6 +27,8 @@ import { SimpleContractDataSource } from './simple_contract_data_source.js';
32
27
  import { type TestPrivateInsertions, createTxForPublicCalls } from './utils.js';
33
28
 
34
29
  const DEFAULT_GAS_FEES = new GasFees(2, 3);
30
+ const TEARDOWN_DA_GAS_LIMIT = 98_304;
31
+ const TEARDOWN_L2_GAS_LIMIT = 817_500;
35
32
 
36
33
  export type TestEnqueuedCall = {
37
34
  sender?: AztecAddress;
@@ -137,10 +134,7 @@ export class PublicTxSimulationTester extends BaseAvmSimulationTester {
137
134
  teardownCallRequest,
138
135
  feePayer,
139
136
  /*gasUsedByPrivate*/ teardownCall
140
- ? new Gas(
141
- DEFAULT_TEARDOWN_DA_GAS_LIMIT + TX_DA_GAS_OVERHEAD,
142
- DEFAULT_TEARDOWN_L2_GAS_LIMIT + PUBLIC_TX_L2_GAS_OVERHEAD,
143
- )
137
+ ? new Gas(TEARDOWN_DA_GAS_LIMIT + TX_DA_GAS_OVERHEAD, TEARDOWN_L2_GAS_LIMIT + PUBLIC_TX_L2_GAS_OVERHEAD)
144
138
  : new Gas(TX_DA_GAS_OVERHEAD, PUBLIC_TX_L2_GAS_OVERHEAD),
145
139
  defaultGlobals(),
146
140
  gasLimits,
@@ -2,10 +2,8 @@ import {
2
2
  CONTRACT_CLASS_PUBLISHED_MAGIC_VALUE,
3
3
  CONTRACT_CLASS_REGISTRY_CONTRACT_ADDRESS,
4
4
  CONTRACT_INSTANCE_REGISTRY_CONTRACT_ADDRESS,
5
- DEFAULT_DA_GAS_LIMIT,
6
- DEFAULT_L2_GAS_LIMIT,
7
- DEFAULT_TEARDOWN_DA_GAS_LIMIT,
8
- DEFAULT_TEARDOWN_L2_GAS_LIMIT,
5
+ MAX_PROCESSABLE_DA_GAS_PER_CHECKPOINT,
6
+ MAX_PROCESSABLE_L2_GAS,
9
7
  PRIVATE_LOG_SIZE_IN_FIELDS,
10
8
  } from '@aztec/constants';
11
9
  import { padArrayEnd } from '@aztec/foundation/collection';
@@ -38,6 +36,9 @@ import {
38
36
 
39
37
  import { strict as assert } from 'assert';
40
38
 
39
+ const TEARDOWN_DA_GAS_LIMIT = 98_304;
40
+ const TEARDOWN_L2_GAS_LIMIT = 817_500;
41
+
41
42
  export type TestPrivateInsertions = {
42
43
  revertible?: {
43
44
  nullifiers?: Fr[];
@@ -69,7 +70,7 @@ export async function createTxForPublicCalls(
69
70
  "Can't create public tx with no enqueued calls",
70
71
  );
71
72
  // use max limits
72
- gasLimits = gasLimits ?? new Gas(DEFAULT_DA_GAS_LIMIT, DEFAULT_L2_GAS_LIMIT);
73
+ gasLimits = gasLimits ?? new Gas(MAX_PROCESSABLE_DA_GAS_PER_CHECKPOINT, MAX_PROCESSABLE_L2_GAS);
73
74
 
74
75
  const forPublic = PartialPrivateTailPublicInputsForPublic.empty();
75
76
 
@@ -128,9 +129,7 @@ export async function createTxForPublicCalls(
128
129
  }
129
130
 
130
131
  const maxFeesPerGas = feePayer.isZero() ? GasFees.empty() : new GasFees(10, 10);
131
- const teardownGasLimits = teardownCallRequest
132
- ? new Gas(DEFAULT_TEARDOWN_DA_GAS_LIMIT, DEFAULT_TEARDOWN_L2_GAS_LIMIT)
133
- : Gas.empty();
132
+ const teardownGasLimits = teardownCallRequest ? new Gas(TEARDOWN_DA_GAS_LIMIT, TEARDOWN_L2_GAS_LIMIT) : Gas.empty();
134
133
  const gasSettings = new GasSettings(gasLimits, teardownGasLimits, maxFeesPerGas, GasFees.empty());
135
134
  const txContext = new TxContext(Fr.zero(), Fr.zero(), gasSettings);
136
135
  const header = BlockHeader.empty({ globalVariables: globals });
@@ -164,7 +163,7 @@ export async function createTxForPrivateOnly(
164
163
  gasUsedByPrivate: Gas = new Gas(10, 10),
165
164
  ): Promise<Tx> {
166
165
  // use max limits
167
- const gasLimits = new Gas(DEFAULT_DA_GAS_LIMIT, DEFAULT_L2_GAS_LIMIT);
166
+ const gasLimits = new Gas(MAX_PROCESSABLE_DA_GAS_PER_CHECKPOINT, MAX_PROCESSABLE_L2_GAS);
168
167
 
169
168
  const forRollup = PartialPrivateTailPublicInputsForRollup.empty();
170
169
 
@@ -229,17 +228,15 @@ export async function addNewContractInstanceToTx(
229
228
  contractInstance: ContractInstanceWithAddress,
230
229
  skipNullifierInsertion = false,
231
230
  ) {
232
- // can't use publicKeys.toFields() because it includes isInfinite which
233
- // is not broadcast in such private logs
231
+ // Only ivpk_m is broadcast as a point (x, y); the other five keys are hashes.
234
232
  const publicKeysAsFields = [
235
- contractInstance.publicKeys.masterNullifierPublicKey.x,
236
- contractInstance.publicKeys.masterNullifierPublicKey.y,
237
- contractInstance.publicKeys.masterIncomingViewingPublicKey.x,
238
- contractInstance.publicKeys.masterIncomingViewingPublicKey.y,
239
- contractInstance.publicKeys.masterOutgoingViewingPublicKey.x,
240
- contractInstance.publicKeys.masterOutgoingViewingPublicKey.y,
241
- contractInstance.publicKeys.masterTaggingPublicKey.x,
242
- contractInstance.publicKeys.masterTaggingPublicKey.y,
233
+ contractInstance.publicKeys.npkMHash,
234
+ contractInstance.publicKeys.ivpkM.x,
235
+ contractInstance.publicKeys.ivpkM.y,
236
+ contractInstance.publicKeys.ovpkMHash,
237
+ contractInstance.publicKeys.tpkMHash,
238
+ contractInstance.publicKeys.mspkMHash,
239
+ contractInstance.publicKeys.fbpkMHash,
243
240
  ];
244
241
  const logFields = [
245
242
  CONTRACT_INSTANCE_PUBLISHED_EVENT_TAG,
@@ -248,6 +245,7 @@ export async function addNewContractInstanceToTx(
248
245
  new Fr(contractInstance.salt),
249
246
  contractInstance.currentContractClassId,
250
247
  contractInstance.initializationHash,
248
+ contractInstance.immutablesHash,
251
249
  ...publicKeysAsFields,
252
250
  contractInstance.deployer.toField(),
253
251
  ];
@@ -83,6 +83,7 @@ export class HintingPublicContractsDB implements PublicContractsDBInterface {
83
83
  instance.currentContractClassId,
84
84
  instance.originalContractClassId,
85
85
  instance.initializationHash,
86
+ instance.immutablesHash,
86
87
  instance.publicKeys,
87
88
  ),
88
89
  );