From 06328b3a834cbb269d6b30640493bfd0ac222b9d Mon Sep 17 00:00:00 2001 From: Nikita Orlov Date: Wed, 19 Jun 2024 16:30:21 +0200 Subject: [PATCH] Fix field arithmetic benchmark cairo related bug + perf (#518) fix constants bug + update field arithmetic cairo + optimize split 64/128 --- .gitignore | 1 + ...ield_arithmetic_get_square_benchmark.cairo | 6 +++--- scripts/benchmarks.sh | 11 +++++----- scripts/build_cairo_programs.sh | 11 ++++++++-- src/hint_processor/field_arithmetic.zig | 1 + src/hint_processor/hint_utils.zig | 6 +++++- src/hint_processor/uint256_utils.zig | 11 +++------- src/hint_processor/uint384.zig | 12 ++++++----- src/vm/core.zig | 6 +++--- src/vm/memory/memory.zig | 15 ++++++-------- src/vm/memory/relocatable.zig | 20 +++++++++++++++++++ 11 files changed, 64 insertions(+), 36 deletions(-) diff --git a/.gitignore b/.gitignore index b33977b5..858f4559 100644 --- a/.gitignore +++ b/.gitignore @@ -18,6 +18,7 @@ docsite/ cairo-vm-env cairo-vm tmp/ +profile.json # we dont need compiled cairo programs in repo /cairo_programs/*.json diff --git a/cairo_programs/benchmarks/field_arithmetic_get_square_benchmark.cairo b/cairo_programs/benchmarks/field_arithmetic_get_square_benchmark.cairo index 57cefc1e..f70a0d30 100644 --- a/cairo_programs/benchmarks/field_arithmetic_get_square_benchmark.cairo +++ b/cairo_programs/benchmarks/field_arithmetic_get_square_benchmark.cairo @@ -1,8 +1,8 @@ from starkware.cairo.common.cairo_builtins import BitwiseBuiltin from starkware.cairo.common.bool import TRUE -from uint384 import u384, Uint384 -from uint384_extension import u384_ext -from field_arithmetic import field_arithmetic +from cairo_programs.uint384 import u384, Uint384 +from cairo_programs.uint384_extension import u384_ext +from cairo_programs.field_arithmetic import field_arithmetic func run_get_square{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}( prime: Uint384, generator: Uint384, num: Uint384, iterations: felt diff --git a/scripts/benchmarks.sh b/scripts/benchmarks.sh index 98e26ba5..2e4a0edd 100644 --- a/scripts/benchmarks.sh +++ b/scripts/benchmarks.sh @@ -3,14 +3,15 @@ set -e . ../cairo-vm-env/bin/activate -BENCH_DIR=../cairo_programs/benchmarks -CAIRO_DIR=../cairo_programs/ -CAIRO_VM_CLI=../cairo-vm/target/release/cairo-vm-cli -ZIG_CLI=../zig-out/bin/ziggy-starkdust +cd .. +BENCH_DIR=cairo_programs/benchmarks +CAIRO_DIR=cairo_programs/ +CAIRO_VM_CLI=cairo-vm/target/release/cairo-vm-cli +ZIG_CLI=zig-out/bin/ziggy-starkdust for file in $(ls ${BENCH_DIR} | grep .cairo | sed -E 's/\.cairo//'); do echo "Compiling ${file} program..." - cairo-compile --cairo_path="${CAIRO_DIR}" ${BENCH_DIR}/${file}.cairo --output ${BENCH_DIR}/${file}.json --proof_mode + cairo-compile --cairo_path="${CAIRO_DIR}:../${CAIRO_DIR}" ${BENCH_DIR}/${file}.cairo --output ${BENCH_DIR}/${file}.json --proof_mode echo "Running ${file} benchmark" hyperfine --show-output --warmup 2 \ -n "cairo-vm (Rust)" "${CAIRO_VM_CLI} ${BENCH_DIR}/${file}.json --memory_file /dev/null --trace_file /dev/null --layout all_cairo" \ diff --git a/scripts/build_cairo_programs.sh b/scripts/build_cairo_programs.sh index 903e17ff..7287e3ab 100644 --- a/scripts/build_cairo_programs.sh +++ b/scripts/build_cairo_programs.sh @@ -6,10 +6,17 @@ set -e cd .. CAIRO_PROGRAMS_DIR=cairo_programs +CAIRO_PROGRAMS_BENCHMARK_DIR=cairo_programs/benchmarks CAIRO_DIR=cairo_programs -for file in $(ls ${CAIRO_PROGRAMS_DIR} | grep .cairo | sed -E 's/\.cairo//'); do +echo "Compiling benchmark cairo programs" +for file in $(ls ${CAIRO_PROGRAMS_BENCHMARK_DIR} | grep .cairo | sed -E 's/\.cairo//'); do echo "Compiling ${file} program..." - cairo-compile --cairo_path="${CAIRO_DIR}:" ${CAIRO_PROGRAMS_DIR}/${file}.cairo --output ${CAIRO_PROGRAMS_DIR}/${file}.json --proof_mode + cairo-compile --cairo_path="${CAIRO_DIR}:../${CAIRO_DIR}" ${CAIRO_PROGRAMS_BENCHMARK_DIR}/${file}.cairo --output ${CAIRO_PROGRAMS_BENCHMARK_DIR}/${file}.json --proof_mode done +echo "Compiling cairo programs" +for file in $(ls ${CAIRO_PROGRAMS_DIR} | grep .cairo | sed -E 's/\.cairo//'); do + echo "Compiling ${file} program..." + cairo-compile --cairo_path="${CAIRO_DIR}:../${CAIRO_DIR}" ${CAIRO_PROGRAMS_DIR}/${file}.cairo --output ${CAIRO_PROGRAMS_DIR}/${file}.json --proof_mode +done diff --git a/src/hint_processor/field_arithmetic.zig b/src/hint_processor/field_arithmetic.zig index 6edf1aa2..d9a97afc 100644 --- a/src/hint_processor/field_arithmetic.zig +++ b/src/hint_processor/field_arithmetic.zig @@ -162,6 +162,7 @@ pub fn bigIntIntGetSquareRoot( (try field_helper.sqrtPrimePower(allocator, gx, p)) orelse try Int.initSet(allocator, 0) else try Int.initSet(allocator, 0); + defer root_gx.deinit(); if (!x.eqlZero() and (@intFromBool(success_x) ^ @intFromBool(success_gx)) == 0) diff --git a/src/hint_processor/hint_utils.zig b/src/hint_processor/hint_utils.zig index 30c325e1..335f6939 100644 --- a/src/hint_processor/hint_utils.zig +++ b/src/hint_processor/hint_utils.zig @@ -36,7 +36,11 @@ pub fn getConstantFromVarName( var it = constants.iterator(); while (it.next()) |k| { if (k.key_ptr.*.len < var_name.len) continue; - if (std.mem.eql(u8, var_name, k.key_ptr.*[k.key_ptr.*.len - var_name.len ..])) + var iter = + std.mem.splitBackwardsSequence(u8, k.key_ptr.*, "."); + + const v = iter.next() orelse continue; + if (std.mem.eql(u8, var_name, v)) return k.value_ptr.*; } diff --git a/src/hint_processor/uint256_utils.zig b/src/hint_processor/uint256_utils.zig index c65f3665..bead8096 100644 --- a/src/hint_processor/uint256_utils.zig +++ b/src/hint_processor/uint256_utils.zig @@ -255,15 +255,10 @@ pub fn split64( ap_tracking: ApTracking, ) !void { const a = try hint_utils.getIntegerFromVarName("a", vm, ids_data, ap_tracking); - const digits = a.toLeDigits(); - var bytes = [_]u8{0} ** 32; + const numb = a.toU256(); - inline for (1..4) |i| { - std.mem.writeInt(u64, bytes[(i - 1) * 8 .. i * 8], digits[i], .little); - } - - const low = Felt252.fromInt(u64, digits[0]); - const high = Felt252.fromBytesLe(bytes); + const low = Felt252.fromInt(u256, numb & ((1 << 64) - 1)); + const high = Felt252.fromInt(u256, numb >> 64); try hint_utils.insertValueFromVarName(allocator, "high", MaybeRelocatable.fromFelt(high), vm, ids_data, ap_tracking); try hint_utils.insertValueFromVarName(allocator, "low", MaybeRelocatable.fromFelt(low), vm, ids_data, ap_tracking); diff --git a/src/hint_processor/uint384.zig b/src/hint_processor/uint384.zig index 2716d6c5..45843d1d 100644 --- a/src/hint_processor/uint384.zig +++ b/src/hint_processor/uint384.zig @@ -93,13 +93,15 @@ pub fn uint384Split128( ids_data: std.StringHashMap(HintReference), ap_tracking: ApTracking, ) !void { - const bound = Felt252.pow2Const(128); const a = try hint_utils.getIntegerFromVarName("a", vm, ids_data, ap_tracking); - const high, const low = a.divRem(bound); + const number = a.toU256(); - try hint_utils.insertValueFromVarName(allocator, "low", MaybeRelocatable.fromFelt(low), vm, ids_data, ap_tracking); - try hint_utils.insertValueFromVarName(allocator, "high", MaybeRelocatable.fromFelt(high), vm, ids_data, ap_tracking); + const low = number & ((1 << 128) - 1); + const high = number >> 128; + + try hint_utils.insertValueFromVarName(allocator, "low", MaybeRelocatable.fromInt(u256, low), vm, ids_data, ap_tracking); + try hint_utils.insertValueFromVarName(allocator, "high", MaybeRelocatable.fromInt(u256, high), vm, ids_data, ap_tracking); } // Implements Hint: @@ -129,7 +131,7 @@ pub fn addNoUint384Check( var buffer: [20]u8 = undefined; inline for (0..3) |i| { - var sum = try a.limbs[i].toInt(u258) + try b.limbs[i].toInt(u258); + var sum = try a.limbs[i].toInt(u512) + try b.limbs[i].toInt(u512); if (prev_carry) sum += 1; diff --git a/src/vm/core.zig b/src/vm/core.zig index c1f3adea..7f370a02 100644 --- a/src/vm/core.zig +++ b/src/vm/core.zig @@ -1407,11 +1407,11 @@ pub const OperandsResult = struct { /// The second operand value. op_1: MaybeRelocatable = undefined, /// The relocatable address of the destination operand. - dst_addr: Relocatable = .{}, + dst_addr: Relocatable = undefined, /// The relocatable address of the first operand. - op_0_addr: Relocatable = .{}, + op_0_addr: Relocatable = undefined, /// The relocatable address of the second operand. - op_1_addr: Relocatable = .{}, + op_1_addr: Relocatable = undefined, /// Indicator for deduced operands. deduced_operands: u8 = 0, diff --git a/src/vm/memory/memory.zig b/src/vm/memory/memory.zig index e1868ce6..a022ce46 100644 --- a/src/vm/memory/memory.zig +++ b/src/vm/memory/memory.zig @@ -442,15 +442,12 @@ pub const Memory = struct { // TODO: rewrite all on self rel address // Return null if either the segment index or offset is not valid. // Otherwise, return the maybe_relocatable value at the specified address. - return if (data.items[segment_index].items[address.offset].getValue()) |val| - switch (val) { - .relocatable => |addr| self.relAddress( - addr, - ) catch unreachable, - else => |_| val, - } - else - null; + return switch (data.items[segment_index].items[address.offset].getValue() orelse return null) { + .relocatable => |addr| self.relAddress( + addr, + ) catch unreachable, + else => |v| v, + }; } /// Retrieves a `Felt252` value from the memory at the specified relocatable address. diff --git a/src/vm/memory/relocatable.zig b/src/vm/memory/relocatable.zig index f71368e1..96bd4672 100644 --- a/src/vm/memory/relocatable.zig +++ b/src/vm/memory/relocatable.zig @@ -280,6 +280,26 @@ pub const MaybeRelocatable = union(enum) { relocatable: Relocatable, felt: Felt252, + /// This method is interface method, when format is calling for this struct, this method is called + /// + pub fn format( + self: Self, + comptime fmt: []const u8, + options: std.fmt.FormatOptions, + writer: anytype, + ) !void { + _ = fmt; + _ = options; + switch (self) { + .relocatable => |r| { + try writer.print("MaybeRelocatable(Relocatable(offset={},index={}))", .{ r.offset, r.segment_index }); + }, + .felt => |f| { + try writer.print("MaybeRelocatable(Felt({}))", .{f.toU256()}); + }, + } + } + /// Determines if two `MaybeRelocatable` instances are equal. /// /// This method compares the variant type and the contained value. If both the variant