From ce474970161993b8d37f4ecd91c083a39da05848 Mon Sep 17 00:00:00 2001 From: Federica Date: Mon, 2 Oct 2023 18:57:27 -0300 Subject: [PATCH 01/31] Begin implemneting blake2s --- pkg/hints/hint_utils/blake2s_hash.go | 44 ++++++++++++++++++++++++++++ 1 file changed, 44 insertions(+) create mode 100644 pkg/hints/hint_utils/blake2s_hash.go diff --git a/pkg/hints/hint_utils/blake2s_hash.go b/pkg/hints/hint_utils/blake2s_hash.go new file mode 100644 index 00000000..93a131a5 --- /dev/null +++ b/pkg/hints/hint_utils/blake2s_hash.go @@ -0,0 +1,44 @@ +package hint_utils + +func IV() [8]uint32 { + return [8]uint32{ + 0x6A09E667, + 0xBB67AE85, + 0x3C6EF372, + 0xA54FF53A, + 0x510E527F, + 0x9B05688C, + 0x1F83D9AB, + 0x5BE0CD19} +} + +func SIGMA() [10][16]uint32 { + return [10][16]uint32{ + {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15}, + {14, 10, 4, 8, 9, 15, 13, 6, 1, 12, 0, 2, 11, 7, 5, 3}, + {11, 8, 12, 0, 5, 2, 15, 13, 10, 14, 3, 6, 7, 1, 9, 4}, + {7, 9, 3, 1, 13, 12, 11, 14, 2, 6, 5, 10, 4, 0, 15, 8}, + {9, 0, 5, 7, 2, 4, 10, 15, 14, 1, 11, 12, 6, 8, 3, 13}, + {2, 12, 6, 10, 0, 11, 8, 3, 4, 13, 7, 5, 15, 14, 1, 9}, + {12, 5, 1, 15, 14, 13, 4, 10, 0, 7, 6, 3, 9, 2, 8, 11}, + {13, 11, 7, 14, 12, 1, 3, 9, 5, 0, 15, 4, 8, 6, 2, 10}, + {6, 15, 14, 9, 11, 3, 0, 8, 12, 2, 13, 7, 1, 4, 10, 5}, + {10, 2, 8, 4, 7, 6, 1, 5, 15, 11, 9, 14, 3, 12, 13, 0}, + } +} + +func rightRot(value uint32, n uint32) uint32 { + return (value >> n) | ((value & ((1 << n) - 1)) << (32 - n)) +} + +func mix(a uint32, b uint32, c uint32, d uint32, m0 uint32, m1 uint32) (uint32, uint32, uint32) { + a = a + b + m0 + d = rightRot(d^a, 16) + c = c + d + b = rightRot(b^c, 12) + a = a + b + m1 + d = rightRot(d^a, 8) + c = c + d + b = rightRot(b^c, 7) + return a, b, c +} From bc5f5d4b618753cfb08440b789c8c9bea7457607 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 11:07:45 -0300 Subject: [PATCH 02/31] Finish blake2s impl --- pkg/hints/hint_utils/blake2s_hash.go | 93 +++++++++++++++++++++++++++- 1 file changed, 91 insertions(+), 2 deletions(-) diff --git a/pkg/hints/hint_utils/blake2s_hash.go b/pkg/hints/hint_utils/blake2s_hash.go index 93a131a5..fd92b2ad 100644 --- a/pkg/hints/hint_utils/blake2s_hash.go +++ b/pkg/hints/hint_utils/blake2s_hash.go @@ -31,7 +31,7 @@ func rightRot(value uint32, n uint32) uint32 { return (value >> n) | ((value & ((1 << n) - 1)) << (32 - n)) } -func mix(a uint32, b uint32, c uint32, d uint32, m0 uint32, m1 uint32) (uint32, uint32, uint32) { +func mix(a uint32, b uint32, c uint32, d uint32, m0 uint32, m1 uint32) (uint32, uint32, uint32, uint32) { a = a + b + m0 d = rightRot(d^a, 16) c = c + d @@ -40,5 +40,94 @@ func mix(a uint32, b uint32, c uint32, d uint32, m0 uint32, m1 uint32) (uint32, d = rightRot(d^a, 8) c = c + d b = rightRot(b^c, 7) - return a, b, c + return a, b, c, d +} + +func blakeRound(state []uint32, message [16]uint32, sigma [16]uint32) []uint32 { + state[0], state[4], state[8], state[12] = mix( + state[0], + state[4], + state[8], + state[12], + message[sigma[0]], + message[sigma[1]], + ) + state[1], state[5], state[9], state[13] = mix( + state[1], + state[5], + state[9], + state[13], + message[sigma[2]], + message[sigma[3]], + ) + state[2], state[6], state[10], state[14] = mix( + state[2], + state[6], + state[10], + state[14], + message[sigma[4]], + message[sigma[5]], + ) + state[3], state[7], state[11], state[15] = mix( + state[3], + state[7], + state[11], + state[15], + message[sigma[6]], + message[sigma[7]], + ) + state[0], state[5], state[10], state[15] = mix( + state[0], + state[5], + state[10], + state[15], + message[sigma[8]], + message[sigma[9]], + ) + state[1], state[6], state[11], state[12] = mix( + state[1], + state[6], + state[11], + state[12], + message[sigma[10]], + message[sigma[11]], + ) + state[2], state[7], state[8], state[13] = mix( + state[2], + state[7], + state[8], + state[13], + message[sigma[12]], + message[sigma[13]], + ) + state[3], state[4], state[9], state[14] = mix( + state[3], + state[4], + state[9], + state[14], + message[sigma[14]], + message[sigma[15]], + ) + return state +} + +func Blake2sCompress(h [8]uint32, message [16]uint32, t0 uint32, t1 uint32, f0 uint32, f1 uint32) []uint32 { + iv := IV() + + state := make([]uint32, 0, 16) + + state = append(state, h[:]...) + state = append(state, iv[:4]...) + state = append(state, iv[4]^t0, iv[5]^t1, iv[6]^f0, iv[7]^f1) + + for _, sigmaList := range SIGMA() { + state = blakeRound(state, message, sigmaList) + } + + newState := make([]uint32, 0, 8) + for i := 0; i < 8; i++ { + newState = append(newState, h[i]^state[i]^state[8+i]) + } + + return newState } From 756aec9d47b3555fed042fef262e520fea7109a2 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 11:18:44 -0300 Subject: [PATCH 03/31] Add unit test --- pkg/hints/hint_utils/bake2s_hash_test.go | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) create mode 100644 pkg/hints/hint_utils/bake2s_hash_test.go diff --git a/pkg/hints/hint_utils/bake2s_hash_test.go b/pkg/hints/hint_utils/bake2s_hash_test.go new file mode 100644 index 00000000..6c778c9c --- /dev/null +++ b/pkg/hints/hint_utils/bake2s_hash_test.go @@ -0,0 +1,18 @@ +package hint_utils_test + +import ( + "reflect" + "testing" + + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_utils" +) + +func TestBlake2sCompressA(t *testing.T) { + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} + expectedState := []uint32{412110711, 3234706100, 3894970767, 982912411, 937789635, 742982576, 3942558313, 1407547065} + newState := Blake2sCompress(h, message, 2, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} From 4e89d38d3a77b7c332f853bd850bf7211db403a6 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 11:25:08 -0300 Subject: [PATCH 04/31] Add more unit tests --- pkg/hints/hint_utils/bake2s_hash_test.go | 61 ++++++++++++++++++++++++ 1 file changed, 61 insertions(+) diff --git a/pkg/hints/hint_utils/bake2s_hash_test.go b/pkg/hints/hint_utils/bake2s_hash_test.go index 6c778c9c..0d5a57a0 100644 --- a/pkg/hints/hint_utils/bake2s_hash_test.go +++ b/pkg/hints/hint_utils/bake2s_hash_test.go @@ -16,3 +16,64 @@ func TestBlake2sCompressA(t *testing.T) { t.Error("Wrong state returned by Blake2sCompress") } } + +func TestBlake2sCompressB(t *testing.T) { + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{456710651, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} + expectedState := []uint32{1061041453, 3663967611, 2158760218, 836165556, 3696892209, 3887053585, 2675134684, 2201582556} + newState := Blake2sCompress(h, message, 2, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} + +func TestBlake2sCompressC(t *testing.T) { + //Hashing "Hello World" + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{1819043144, 1870078063, 6581362, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} + expectedState := []uint32{939893662, 3935214984, 1704819782, 3912812968, 4211807320, 3760278243, 674188535, 2642110762} + newState := Blake2sCompress(h, message, 9, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} + +func TestBlake2sCompressD(t *testing.T) { + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{1819043144, 1870078063, 6581362, 274628678, 715791845, 175498643, 871587583, 0, 0, 0, 0, 0, 0, 0, 0, 0} + expectedState := []uint32{3980510537, 3982966407, 1593299263, 2666882356, 3288094120, 2682988286, 1666615862, 378086837} + newState := Blake2sCompress(h, message, 28, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} + +func TestBlake2sCompressE(t *testing.T) { + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{1819043144, 1870078063, 6581362, 274628678, 715791845, 175498643, 871587583, 635963558, 557369694, 1576875962, 215769785, 0, 0, 0, 0, 0} + expectedState := []uint32{3251785223, 1946079609, 2665255093, 3508191500, 3630835628, 3067307230, 3623370123, 656151356} + newState := Blake2sCompress(h, message, 44, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} + +func TestBlake2sCompressF(t *testing.T) { + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{1819043144, 1870078063, 6581362, 274628678, 715791845, 175498643, 871587583, 635963558, 557369694, 1576875962, 215769785, 152379578, 585849303, 764739320, 437383930, 74833930} + expectedState := []uint32{2593218707, 3238077801, 914875393, 3462286058, 4028447058, 3174734057, 2001070146, 3741410512} + newState := Blake2sCompress(h, message, 64, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} + +func TestBlake2sCompressG(t *testing.T) { + h := [8]uint32{1795745351, 3144134277, 1013904242, 2773480762, 1359893119, 2600822924, 528734635, 1541459225} + message := [16]uint32{11563522, 43535528, 653255322, 274628678, 73471943, 17549868, 87158958, 635963558, 343656565, 1576875962, 215769785, 152379578, 585849303, 76473202, 437253230, 74833930} + expectedState := []uint32{3496615692, 3252241979, 3771521549, 2125493093, 3240605752, 2885407061, 3962009872, 3845288240} + newState := Blake2sCompress(h, message, 64, 0, 4294967295, 0) + if !reflect.DeepEqual(expectedState, newState) { + t.Error("Wrong state returned by Blake2sCompress") + } +} From 3073bda7f60d7dd7ca766fb8c1c6ebad13227737 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 11:33:07 -0300 Subject: [PATCH 05/31] Add integration test --- cairo_programs/blake2s_hello_world_hash.cairo | 20 +++++++++++++++++++ pkg/vm/cairo_run/cairo_run_test.go | 4 ++++ 2 files changed, 24 insertions(+) create mode 100644 cairo_programs/blake2s_hello_world_hash.cairo diff --git a/cairo_programs/blake2s_hello_world_hash.cairo b/cairo_programs/blake2s_hello_world_hash.cairo new file mode 100644 index 00000000..9e832e7d --- /dev/null +++ b/cairo_programs/blake2s_hello_world_hash.cairo @@ -0,0 +1,20 @@ +%builtins range_check bitwise + +from starkware.cairo.common.alloc import alloc +from starkware.cairo.common.cairo_blake2s.blake2s import blake2s +from starkware.cairo.common.cairo_builtins import BitwiseBuiltin + +// Computes the hash of "Hello World" +func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { + alloc_locals; + let inputs: felt* = alloc(); + assert inputs[0] = 'Hell'; + assert inputs[1] = 'o Wo'; + assert inputs[2] = 'rld'; + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + let (output) = blake2s{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}(inputs, 9); + assert output.low = 219917655069954262743903159041439073909; + assert output.high = 296157033687865319468534978667166017272; + return (); +} \ No newline at end of file diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index 845a3076..8dea5736 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -348,3 +348,7 @@ func TestCairoKeccak(t *testing.T) { func TestKeccakAddUint256(t *testing.T) { testProgram("keccak_add_uint256", t) } + +func TestBlake2sHelloWorldHash(t *testing.T) { + testProgram("blake2s_hello_world_hash", t) +} From d9564ff8e321098daef95a9b4a33f9f2d53bb4b5 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:03:05 -0300 Subject: [PATCH 06/31] Implement BLAKE2S_COMPUTE --- pkg/hints/blake2s_hints.go | 84 ++++++++++++++++++++++ pkg/hints/hint_codes/blake2s_hint_codes.go | 4 ++ pkg/hints/hint_processor.go | 2 + pkg/lambdaworks/lambdaworks.go | 10 +++ 4 files changed, 100 insertions(+) create mode 100644 pkg/hints/blake2s_hints.go create mode 100644 pkg/hints/hint_codes/blake2s_hint_codes.go diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go new file mode 100644 index 00000000..d66c4794 --- /dev/null +++ b/pkg/hints/blake2s_hints.go @@ -0,0 +1,84 @@ +package hints + +import ( + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_utils" + . "github.com/lambdaclass/cairo-vm.go/pkg/lambdaworks" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm/memory" +) + +func Uint32SliceToMRSlice(u32Slice []uint32) []MaybeRelocatable { + mRSlice := make([]MaybeRelocatable, 0, len(u32Slice)) + for _, u32 := range u32Slice { + mRSlice = append(mRSlice, *NewMaybeRelocatableFelt(FeltFromUint(uint(u32)))) + } + return mRSlice + +} + +func feltSliceToUint32Slice(feltSlice []Felt) ([]uint32, error) { + uint32Slice := make([]uint32, 0, len(feltSlice)) + for _, felt := range feltSlice { + val, err := felt.ToU32() + if err != nil { + return nil, err + } + uint32Slice = append(uint32Slice, val) + } + return uint32Slice, nil +} + +// Returns the range from (baseAddr - baseAddrOffset) to (baseAddr - baseAddrOffset + Size) +func getUint32MemoryRange(baseAddr Relocatable, baseAddrOffset uint, size uint, segments *MemorySegmentManager) ([]uint32, error) { + baseAddr, err := baseAddr.SubUint(baseAddrOffset) + if err != nil { + return nil, err + } + feltRange, err := segments.GetFeltRange(baseAddr, size) + if err != nil { + return nil, err + } + return feltSliceToUint32Slice(feltRange) +} + +// Returns the u32 value at memory[baseAddr - baseAddrOffset] +func getU32FromMemory(baseAddr Relocatable, baseAddrOffset uint, memory *Memory) (uint32, error) { + baseAddr, err := baseAddr.SubUint(baseAddrOffset) + if err != nil { + return 0, err + } + felt, err := memory.GetFelt(baseAddr) + if err != nil { + return 0, err + } + return felt.ToU32() +} + +func blake2sCompute(ids IdsManager, vm *VirtualMachine) error { + output, err := ids.GetRelocatable("output", vm) + if err != nil { + return err + } + h, err := getUint32MemoryRange(output, 26, 8, &vm.Segments) + if err != nil { + return err + } + message, err := getUint32MemoryRange(output, 18, 16, &vm.Segments) + if err != nil { + return err + } + t, err := getU32FromMemory(output, 2, &vm.Segments.Memory) + if err != nil { + return err + } + f, err := getU32FromMemory(output, 1, &vm.Segments.Memory) + if err != nil { + return err + } + + newState := Blake2sCompress([8]uint32(h), [16]uint32(message), t, 0, f, 0) + data := Uint32SliceToMRSlice(newState) + + _, err = vm.Segments.LoadData(output, &data) + return err +} diff --git a/pkg/hints/hint_codes/blake2s_hint_codes.go b/pkg/hints/hint_codes/blake2s_hint_codes.go new file mode 100644 index 00000000..fb26e1f5 --- /dev/null +++ b/pkg/hints/hint_codes/blake2s_hint_codes.go @@ -0,0 +1,4 @@ +package hint_codes + +const BLAKE2S_COMPUTE = `from starkware.cairo.common.cairo_blake2s.blake2s_utils import compute_blake2s_func +compute_blake2s_func(segments=segments, output_ptr=ids.output)` diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index ccac7bd2..3cd88a9d 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -198,6 +198,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return fastEcAddAssignNewX(data.Ids, vm, execScopes, "pt0", "pt1", SECP_P()) case FAST_EC_ADD_ASSIGN_NEW_Y: return fastEcAddAssignNewY(execScopes) + case BLAKE2S_COMPUTE: + return blake2sCompute(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) } diff --git a/pkg/lambdaworks/lambdaworks.go b/pkg/lambdaworks/lambdaworks.go index 3cd29d21..1362c1b9 100644 --- a/pkg/lambdaworks/lambdaworks.go +++ b/pkg/lambdaworks/lambdaworks.go @@ -8,6 +8,7 @@ package lambdaworks import "C" import ( + "math" "math/big" "reflect" "strings" @@ -101,6 +102,15 @@ func (felt Felt) ToUint() (uint, error) { return uint(felt_u64), nil } +// turns a felt to uint32 +func (felt Felt) ToU32() (uint32, error) { + feltU64, err := felt.ToU64() + if err != nil || feltU64 > math.MaxUint32 { + return 0, ConversionError(felt, "uint32") + } + return uint32(feltU64), nil +} + func (felt Felt) ToLeBytes() *[32]byte { var result_c [32]C.uint8_t var value C.felt_t = felt.toC() From 5039f585649599e45f0074dc0302a14c684d4b34 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:11:12 -0300 Subject: [PATCH 07/31] Add unit tests --- pkg/hints/blake2s_hints_test.go | 154 ++++++++++++++++++++++++++++++++ 1 file changed, 154 insertions(+) create mode 100644 pkg/hints/blake2s_hints_test.go diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go new file mode 100644 index 00000000..42b88d26 --- /dev/null +++ b/pkg/hints/blake2s_hints_test.go @@ -0,0 +1,154 @@ +package hints_test + +import ( + "testing" + + . "github.com/lambdaclass/cairo-vm.go/pkg/hints" + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_codes" + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_utils" + . "github.com/lambdaclass/cairo-vm.go/pkg/lambdaworks" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm/memory" +) + +func TestBlake2sComputeOutputOffsetZero(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + output := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "output": {NewMaybeRelocatableRelocatable(output)}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_COMPUTE, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err == nil { + t.Errorf("BLAKE2S_COMPUTE hint test should have failed") + } +} + +func TestBlake2sComputeOutputSegmentEmpty(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + output := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "output": {NewMaybeRelocatableRelocatable(output.AddUint(26))}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_COMPUTE, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err == nil { + t.Errorf("BLAKE2S_COMPUTE hint test should have failed") + } +} + +func TestBlake2sComputeBigOutput(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + output := vm.Segments.AddSegment() + data := []MaybeRelocatable{ + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + *NewMaybeRelocatableFelt(FeltFromDecString("7842562439562793675803603603688959")), + } + output, _ = vm.Segments.LoadData(output, &data) + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "output": {NewMaybeRelocatableRelocatable(output)}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_COMPUTE, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err == nil { + t.Errorf("BLAKE2S_COMPUTE hint test should have failed") + } +} + +func TestBlake2sComputeOk(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + output := vm.Segments.AddSegment() + data := []MaybeRelocatable{ + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + *NewMaybeRelocatableFelt(FeltFromDecString("17")), + } + output, _ = vm.Segments.LoadData(output, &data) + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "output": {NewMaybeRelocatableRelocatable(output)}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_COMPUTE, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("BLAKE2S_COMPUTE hint test failed with error %s", err) + } +} From e32a854feedd8a0f8ca9c2fc1553a1944b4ec02f Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:15:01 -0300 Subject: [PATCH 08/31] Add unit tests --- pkg/lambdaworks/lambdaworks.go | 2 +- pkg/lambdaworks/lambdaworks_test.go | 34 +++++++++++++++++++++++++++++ 2 files changed, 35 insertions(+), 1 deletion(-) diff --git a/pkg/lambdaworks/lambdaworks.go b/pkg/lambdaworks/lambdaworks.go index 1362c1b9..c98606fc 100644 --- a/pkg/lambdaworks/lambdaworks.go +++ b/pkg/lambdaworks/lambdaworks.go @@ -106,7 +106,7 @@ func (felt Felt) ToUint() (uint, error) { func (felt Felt) ToU32() (uint32, error) { feltU64, err := felt.ToU64() if err != nil || feltU64 > math.MaxUint32 { - return 0, ConversionError(felt, "uint32") + return 0, ConversionError(felt, "u32") } return uint32(feltU64), nil } diff --git a/pkg/lambdaworks/lambdaworks_test.go b/pkg/lambdaworks/lambdaworks_test.go index bf7c404d..fdd3cd89 100644 --- a/pkg/lambdaworks/lambdaworks_test.go +++ b/pkg/lambdaworks/lambdaworks_test.go @@ -427,6 +427,40 @@ func TestToU64Fail(t *testing.T) { } } +func TestToU321(t *testing.T) { + felt := lambdaworks.FeltOne() + result, err := felt.ToU32() + + var expected uint32 = 1 + + if expected != result { + t.Errorf("Error in conversion expected: %v, got %v with err: %v", expected, result, err) + } + +} + +func TestToU3210230(t *testing.T) { + felt := lambdaworks.FeltFromUint64(10230) + result, err := felt.ToU32() + + var expected uint32 = 10230 + + if expected != result { + t.Errorf("Error in conversion expected: %v, got %v with err: %v", expected, result, err) + } +} + +func TestTo32Fail(t *testing.T) { + felt := lambdaworks.FeltFromUint64(4294967297) + + _, err := felt.ToU32() + expected_err := lambdaworks.ConversionError(felt, "u32") + + if err.Error() != expected_err.Error() { + t.Errorf("Conversion test should fail with error: %v", expected_err) + } +} + func TestToUint1(t *testing.T) { felt := lambdaworks.FeltOne() result, err := felt.ToUint() From 628dbcd623f7f3e9c52486abc821f100dfeb98bd Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:15:53 -0300 Subject: [PATCH 09/31] Add newline --- cairo_programs/blake2s_hello_world_hash.cairo | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cairo_programs/blake2s_hello_world_hash.cairo b/cairo_programs/blake2s_hello_world_hash.cairo index 9e832e7d..6d591cb9 100644 --- a/cairo_programs/blake2s_hello_world_hash.cairo +++ b/cairo_programs/blake2s_hello_world_hash.cairo @@ -17,4 +17,4 @@ func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { assert output.low = 219917655069954262743903159041439073909; assert output.high = 296157033687865319468534978667166017272; return (); -} \ No newline at end of file +} From e3ae43d81b432ad735f9d4bf4d9267edab238021 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:31:51 -0300 Subject: [PATCH 10/31] Add integration test --- cairo_programs/blake2s_felts.cairo | 35 ++++++++++++++++++++++++++++++ pkg/vm/cairo_run/cairo_run_test.go | 4 ++++ 2 files changed, 39 insertions(+) create mode 100644 cairo_programs/blake2s_felts.cairo diff --git a/cairo_programs/blake2s_felts.cairo b/cairo_programs/blake2s_felts.cairo new file mode 100644 index 00000000..fd913b2a --- /dev/null +++ b/cairo_programs/blake2s_felts.cairo @@ -0,0 +1,35 @@ +%builtins range_check bitwise + +from starkware.cairo.common.bool import TRUE +from starkware.cairo.common.alloc import alloc +from starkware.cairo.common.cairo_blake2s.blake2s import blake2s_felts +from starkware.cairo.common.cairo_builtins import BitwiseBuiltin + +func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { + alloc_locals; + let inputs: felt* = alloc(); + assert inputs[0] = 3456722; + assert inputs[1] = 435425528; + assert inputs[2] = 3232553; + assert inputs[3] = 2576195; + assert inputs[4] = 73471943; + assert inputs[5] = 17549868; + assert inputs[6] = 87158958; + assert inputs[7] = 6353668; + assert inputs[8] = 343656565; + assert inputs[9] = 1255962; + assert inputs[10] = 25439785; + assert inputs[11] = 1154578; + assert inputs[12] = 585849303; + assert inputs[13] = 763502; + assert inputs[14] = 43753647; + assert inputs[15] = 74256930; + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + let (result) = blake2s_felts{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( + 16, inputs, TRUE + ); + assert result.low = 23022179997536219430502258022509199703; + assert result.high = 136831746058902715979837770794974289597; + return (); +} diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index 8dea5736..eaf02fc4 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -352,3 +352,7 @@ func TestKeccakAddUint256(t *testing.T) { func TestBlake2sHelloWorldHash(t *testing.T) { testProgram("blake2s_hello_world_hash", t) } + +func TestBlake2sFelts(t *testing.T) { + testProgram("blake2s_felts", t) +} From a87d704b125cbabd13187e186a79e4c62628ffe9 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:46:14 -0300 Subject: [PATCH 11/31] Implement BLAKE2S_ADD_UINT256_BIGEND hint --- pkg/hints/blake2s_hints.go | 38 ++++++++++++++++++++++ pkg/hints/hint_codes/blake2s_hint_codes.go | 5 +++ pkg/hints/hint_processor.go | 2 ++ 3 files changed, 45 insertions(+) diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go index d66c4794..144ae54f 100644 --- a/pkg/hints/blake2s_hints.go +++ b/pkg/hints/blake2s_hints.go @@ -1,6 +1,8 @@ package hints import ( + "math" + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_utils" . "github.com/lambdaclass/cairo-vm.go/pkg/lambdaworks" . "github.com/lambdaclass/cairo-vm.go/pkg/vm" @@ -82,3 +84,39 @@ func blake2sCompute(ids IdsManager, vm *VirtualMachine) error { _, err = vm.Segments.LoadData(output, &data) return err } + +func blake2sAddUint256Bigend(ids IdsManager, vm *VirtualMachine) error { + // Fetch ids variables + dataPtr, err := ids.GetRelocatable("data", vm) + if err != nil { + return err + } + low, err := ids.GetFelt("low", vm) + if err != nil { + return err + } + high, err := ids.GetFelt("high", vm) + if err != nil { + return err + } + // Hint logic + const MASK = math.MaxUint32 + const B = 32 + mask := FeltFromUint(MASK) + // First batch + data := make([]MaybeRelocatable, 0, 4) + for i := uint(0); i < 4; i++ { + data = append(data, *NewMaybeRelocatableFelt(high.Shr(B * (3 - i)).And(mask))) + } + dataPtr, err = vm.Segments.LoadData(dataPtr, &data) + if err != nil { + return err + } + // Second batch + data = make([]MaybeRelocatable, 0, 4) + for i := uint(0); i < 4; i++ { + data = append(data, *NewMaybeRelocatableFelt(low.Shr(B * (3 - i)).And(mask))) + } + _, err = vm.Segments.LoadData(dataPtr, &data) + return err +} diff --git a/pkg/hints/hint_codes/blake2s_hint_codes.go b/pkg/hints/hint_codes/blake2s_hint_codes.go index fb26e1f5..b6ea16f8 100644 --- a/pkg/hints/hint_codes/blake2s_hint_codes.go +++ b/pkg/hints/hint_codes/blake2s_hint_codes.go @@ -2,3 +2,8 @@ package hint_codes const BLAKE2S_COMPUTE = `from starkware.cairo.common.cairo_blake2s.blake2s_utils import compute_blake2s_func compute_blake2s_func(segments=segments, output_ptr=ids.output)` + +const BLAKE2S_ADD_UINT256_BIGEND = `B = 32 +MASK = 2 ** 32 - 1 +segments.write_arg(ids.data, [(ids.high >> (B * (3 - i))) & MASK for i in range(4)]) +segments.write_arg(ids.data + 4, [(ids.low >> (B * (3 - i))) & MASK for i in range(4)])` diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index 3cd88a9d..136e8c9c 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -200,6 +200,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return fastEcAddAssignNewY(execScopes) case BLAKE2S_COMPUTE: return blake2sCompute(data.Ids, vm) + case BLAKE2S_ADD_UINT256_BIGEND: + return blake2sAddUint256Bigend(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) } From d7004d936c268169fc69a752ed5a777423b549d4 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 12:58:43 -0300 Subject: [PATCH 12/31] Add unit test --- pkg/hints/blake2s_hints_test.go | 39 +++++++++++++++++++++++++++++++++ 1 file changed, 39 insertions(+) diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go index 42b88d26..ca761455 100644 --- a/pkg/hints/blake2s_hints_test.go +++ b/pkg/hints/blake2s_hints_test.go @@ -1,6 +1,7 @@ package hints_test import ( + "reflect" "testing" . "github.com/lambdaclass/cairo-vm.go/pkg/hints" @@ -152,3 +153,41 @@ func TestBlake2sComputeOk(t *testing.T) { t.Errorf("BLAKE2S_COMPUTE hint test failed with error %s", err) } } + +func TestBlake2sAddUint256BigEndOk(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + data := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "data": {NewMaybeRelocatableRelocatable(data)}, + "high": {NewMaybeRelocatableFelt(FeltFromUint(25))}, + "low": {NewMaybeRelocatableFelt(FeltFromUint(20))}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_ADD_UINT256_BIGEND, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("BLAKE2S_ADD_UINT256_BIGEND hint test failed with error %s", err) + } + // Check the data segment + dataSegment, err := vm.Segments.GetFeltRange(data, 8) + expectedDataSegment := []Felt{ + FeltZero(), + FeltZero(), + FeltZero(), + FeltFromUint(25), + FeltZero(), + FeltZero(), + FeltZero(), + FeltFromUint(20), + } + if err != nil || !reflect.DeepEqual(dataSegment, expectedDataSegment) { + t.Errorf("Wrong/No data loaded.\n Expected %v, got %v", expectedDataSegment, dataSegment) + } +} From c55aefdb4b07572916955857ff1c54844c9cb981 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 13:09:16 -0300 Subject: [PATCH 13/31] Add integration test --- cairo_programs/finalize_blake2s.cairo | 18 ++++++++++++++++++ pkg/vm/cairo_run/cairo_run_test.go | 4 ++++ 2 files changed, 22 insertions(+) create mode 100644 cairo_programs/finalize_blake2s.cairo diff --git a/cairo_programs/finalize_blake2s.cairo b/cairo_programs/finalize_blake2s.cairo new file mode 100644 index 00000000..9c7dd7f7 --- /dev/null +++ b/cairo_programs/finalize_blake2s.cairo @@ -0,0 +1,18 @@ +%builtins range_check bitwise + +from starkware.cairo.common.alloc import alloc +from starkware.cairo.common.cairo_blake2s.blake2s import blake2s, finalize_blake2s +from starkware.cairo.common.cairo_builtins import BitwiseBuiltin + +func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { + alloc_locals; + let inputs: felt* = alloc(); + assert inputs[0] = 'Hell'; + assert inputs[1] = 'o Wo'; + assert inputs[2] = 'rld'; + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + let (output) = blake2s{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}(inputs, 9); + finalize_blake2s(blake2s_ptr_start, blake2s_ptr); + return (); +} diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index eaf02fc4..7927f305 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -356,3 +356,7 @@ func TestBlake2sHelloWorldHash(t *testing.T) { func TestBlake2sFelts(t *testing.T) { testProgram("blake2s_felts", t) } + +func TestFinalizeBlake2s(t *testing.T) { + testProgram("finalize_blake2s", t) +} From 6b90c983c5b44eaed5e5c7f95ac1867688c486a4 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 13:31:39 -0300 Subject: [PATCH 14/31] Implement finalize_blake2s hint --- pkg/hints/blake2s_hints.go | 22 ++++++++++++++++++++++ pkg/hints/hint_codes/blake2s_hint_codes.go | 21 +++++++++++++++++++++ pkg/hints/hint_processor.go | 2 ++ 3 files changed, 45 insertions(+) diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go index 144ae54f..8a678812 100644 --- a/pkg/hints/blake2s_hints.go +++ b/pkg/hints/blake2s_hints.go @@ -120,3 +120,25 @@ func blake2sAddUint256Bigend(ids IdsManager, vm *VirtualMachine) error { _, err = vm.Segments.LoadData(dataPtr, &data) return err } + +func blake2sFinalize(ids IdsManager, vm *VirtualMachine) error { + const N_PACKED_INSTANCES = 7 + blake2sPtrEnd, err := ids.GetRelocatable("blake2s_ptr_end", vm) + if err != nil { + return err + } + var message [16]uint32 + modifiedIv := IV() + output := Blake2sCompress(modifiedIv, message, 0, 0, 0xffffffff, 0) + padding := modifiedIv[:] + padding = append(padding, message[:]...) + padding = append(padding, 0, 0xffffffff) + padding = append(padding, output[:]...) + fullPadding := padding + for i := 2; i < N_PACKED_INSTANCES; i++ { + fullPadding = append(fullPadding, padding...) + } + data := Uint32SliceToMRSlice(fullPadding) + _, err = vm.Segments.LoadData(blake2sPtrEnd, &data) + return err +} diff --git a/pkg/hints/hint_codes/blake2s_hint_codes.go b/pkg/hints/hint_codes/blake2s_hint_codes.go index b6ea16f8..c75a9990 100644 --- a/pkg/hints/hint_codes/blake2s_hint_codes.go +++ b/pkg/hints/hint_codes/blake2s_hint_codes.go @@ -7,3 +7,24 @@ const BLAKE2S_ADD_UINT256_BIGEND = `B = 32 MASK = 2 ** 32 - 1 segments.write_arg(ids.data, [(ids.high >> (B * (3 - i))) & MASK for i in range(4)]) segments.write_arg(ids.data + 4, [(ids.low >> (B * (3 - i))) & MASK for i in range(4)])` + +const BLAKE2S_FINALIZE = `# Add dummy pairs of input and output. +from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + +_n_packed_instances = int(ids.N_PACKED_INSTANCES) +assert 0 <= _n_packed_instances < 20 +_blake2s_input_chunk_size_felts = int(ids.INPUT_BLOCK_FELTS) +assert 0 <= _blake2s_input_chunk_size_felts < 100 + +message = [0] * _blake2s_input_chunk_size_felts +modified_iv = [IV[0] ^ 0x01010020] + IV[1:] +output = blake2s_compress( + message=message, + h=modified_iv, + t0=0, + t1=0, + f0=0xffffffff, + f1=0, +) +padding = (modified_iv + message + [0, 0xffffffff] + output) * (_n_packed_instances - 1) +segments.write_arg(ids.blake2s_ptr_end, padding)` diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index 136e8c9c..a43edae5 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -202,6 +202,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return blake2sCompute(data.Ids, vm) case BLAKE2S_ADD_UINT256_BIGEND: return blake2sAddUint256Bigend(data.Ids, vm) + case BLAKE2S_FINALIZE: + return blake2sFinalize(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) } From 2e6a7e0c2024294299b95c2840b85ece1c8334c6 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 15:04:35 -0300 Subject: [PATCH 15/31] Fix removed line --- pkg/hints/blake2s_hints.go | 1 + 1 file changed, 1 insertion(+) diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go index 8a678812..c3382da3 100644 --- a/pkg/hints/blake2s_hints.go +++ b/pkg/hints/blake2s_hints.go @@ -129,6 +129,7 @@ func blake2sFinalize(ids IdsManager, vm *VirtualMachine) error { } var message [16]uint32 modifiedIv := IV() + modifiedIv[0] = modifiedIv[0] ^ 0x01010020 output := Blake2sCompress(modifiedIv, message, 0, 0, 0xffffffff, 0) padding := modifiedIv[:] padding = append(padding, message[:]...) From 3d44715a7b247f954120ec8af5adf948a4574ef3 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 15:05:23 -0300 Subject: [PATCH 16/31] Add unit test --- pkg/hints/blake2s_hints_test.go | 52 +++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go index ca761455..a7d1cbde 100644 --- a/pkg/hints/blake2s_hints_test.go +++ b/pkg/hints/blake2s_hints_test.go @@ -191,3 +191,55 @@ func TestBlake2sAddUint256BigEndOk(t *testing.T) { t.Errorf("Wrong/No data loaded.\n Expected %v, got %v", expectedDataSegment, dataSegment) } } + +func TestBlake2sFinaizeOk(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + data := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "blake2s_ptr_end": {NewMaybeRelocatableRelocatable(data)}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_FINALIZE, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("BLAKE2S_FINALIZE hint test failed with error %s", err) + } + // Check the data segment + dataSegment, err := vm.Segments.GetFeltRange(data, 204) + + expectedDataSegment := []Felt{ + FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + } + if err != nil || !reflect.DeepEqual(dataSegment, expectedDataSegment) { + t.Errorf("Wrong/No data loaded.\n Expected: %v.\n Got: %v", expectedDataSegment, dataSegment) + } +} From 101dc93e0b915fb0acc9a8b6fd61e7e74dbe03a9 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 15:09:19 -0300 Subject: [PATCH 17/31] Fix test values --- pkg/hints/blake2s_hints_test.go | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go index a7d1cbde..d633820e 100644 --- a/pkg/hints/blake2s_hints_test.go +++ b/pkg/hints/blake2s_hints_test.go @@ -215,27 +215,27 @@ func TestBlake2sFinaizeOk(t *testing.T) { dataSegment, err := vm.Segments.GetFeltRange(data, 204) expectedDataSegment := []Felt{ - FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), - FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), - FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), - FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), - FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), - FeltFromUint(179574535), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), } From 909ac06f77c40eb4848cf87e79c5e8392f556bd6 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 15:50:10 -0300 Subject: [PATCH 18/31] Add + expand integration test --- cairo_programs/blake2s_felts.cairo | 12 ++- .../blake2s_integration_tests.cairo | 75 +++++++++++++++++++ pkg/vm/cairo_run/cairo_run_test.go | 4 + 3 files changed, 90 insertions(+), 1 deletion(-) create mode 100644 cairo_programs/blake2s_integration_tests.cairo diff --git a/cairo_programs/blake2s_felts.cairo b/cairo_programs/blake2s_felts.cairo index fd913b2a..6212bb93 100644 --- a/cairo_programs/blake2s_felts.cairo +++ b/cairo_programs/blake2s_felts.cairo @@ -1,6 +1,6 @@ %builtins range_check bitwise -from starkware.cairo.common.bool import TRUE +from starkware.cairo.common.bool import TRUE, FALSE from starkware.cairo.common.alloc import alloc from starkware.cairo.common.cairo_blake2s.blake2s import blake2s_felts from starkware.cairo.common.cairo_builtins import BitwiseBuiltin @@ -26,10 +26,20 @@ func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { assert inputs[15] = 74256930; let (local blake2s_ptr_start) = alloc(); let blake2s_ptr = blake2s_ptr_start; + + // Bigendian let (result) = blake2s_felts{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( 16, inputs, TRUE ); assert result.low = 23022179997536219430502258022509199703; assert result.high = 136831746058902715979837770794974289597; + + // Little endian + let (result) = blake2s_felts{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( + 16, inputs, FALSE + ); + assert result.low = 315510691254085211243916597439546947220; + assert result.high = 42237338665522721102428636006748876126; + return (); } diff --git a/cairo_programs/blake2s_integration_tests.cairo b/cairo_programs/blake2s_integration_tests.cairo new file mode 100644 index 00000000..a4636757 --- /dev/null +++ b/cairo_programs/blake2s_integration_tests.cairo @@ -0,0 +1,75 @@ +%builtins range_check bitwise + +from starkware.cairo.common.alloc import alloc +from starkware.cairo.common.cairo_blake2s.blake2s import blake2s, finalize_blake2s, blake2s_felts +from starkware.cairo.common.cairo_builtins import BitwiseBuiltin +from starkware.cairo.common.uint256 import Uint256 +from starkware.cairo.common.bool import TRUE + +func fill_array(array: felt*, base: felt, step: felt, array_length: felt, iterator: felt) { + if (iterator == array_length) { + return (); + } + assert array[iterator] = base + step * iterator; + return fill_array(array, base, step, array_length, iterator + 1); +} + +func test_integration{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}(iter: felt, last: felt) { + alloc_locals; + if (iter == last) { + return (); + } + + let (data: felt*) = alloc(); + fill_array(data, iter, 2 * iter, 10, 0); + + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + let (res_1: Uint256) = blake2s{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( + data, 9 + ); + + finalize_blake2s(blake2s_ptr_start, blake2s_ptr); + + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + + let (data_2: felt*) = alloc(); + assert data_2[0] = res_1.low; + assert data_2[1] = res_1.high; + + let (res_2) = blake2s_felts{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( + 2, data_2, TRUE + ); + + finalize_blake2s(blake2s_ptr_start, blake2s_ptr); + + if (iter == last - 1 and last == 10) { + assert res_1.low = 327684140823325841083166505949840946643; + assert res_1.high = 28077572547397067729112288485703133566; + assert res_2.low = 323710308182296053867309835081443411626; + assert res_2.high = 159988406782415793602959692147600111481; + } + + if (iter == last - 1 and last == 100) { + assert res_1.low = 26473789897582596397897414631405692327; + assert res_1.high = 35314462001555260569814614879256292984; + assert res_2.low = 256911263205530722270005922452382996929; + assert res_2.high = 248798531786594770765531047659644061441; + } + + return test_integration(iter + 1, last); +} + +func run_tests{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}(last: felt) { + alloc_locals; + test_integration(0, last); + + return (); +} + +func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { + run_tests(10); + + return (); +} diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index 7927f305..dd870078 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -360,3 +360,7 @@ func TestBlake2sFelts(t *testing.T) { func TestFinalizeBlake2s(t *testing.T) { testProgram("finalize_blake2s", t) } + +func TestBlake2sIntegrationTests(t *testing.T) { + testProgram("blake2s_integration_tests", t) +} From c1ef6b02c573c580892b45605ca749f9e60c57a5 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 16:39:24 -0300 Subject: [PATCH 19/31] Implement hint + add quickfix to makefile --- Makefile | 3 +- pkg/hints/blake2s_hints.go | 36 ++++++++++++++++++++ pkg/hints/blake2s_hints_test.go | 38 ++++++++++++++++++++++ pkg/hints/hint_codes/blake2s_hint_codes.go | 5 +++ pkg/hints/hint_processor.go | 2 ++ 5 files changed, 83 insertions(+), 1 deletion(-) diff --git a/Makefile b/Makefile index 9954a52f..30117fef 100644 --- a/Makefile +++ b/Makefile @@ -3,8 +3,9 @@ CAIRO_VM_CLI:=cairo-vm/target/release/cairo-vm-cli +# TODO: clone from last release once the fix is merged $(CAIRO_VM_CLI): - git clone --depth 1 -b v0.8.5 https://github.com/lambdaclass/cairo-vm + git clone --depth 1 -b fix-blake-hint https://github.com/lambdaclass/cairo-vm cd cairo-vm; cargo b --release --bin cairo-vm-cli # Create proof mode programs. diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go index c3382da3..43487a99 100644 --- a/pkg/hints/blake2s_hints.go +++ b/pkg/hints/blake2s_hints.go @@ -85,6 +85,42 @@ func blake2sCompute(ids IdsManager, vm *VirtualMachine) error { return err } +func blake2sAddUint256(ids IdsManager, vm *VirtualMachine) error { + // Fetch ids variables + dataPtr, err := ids.GetRelocatable("data", vm) + if err != nil { + return err + } + low, err := ids.GetFelt("low", vm) + if err != nil { + return err + } + high, err := ids.GetFelt("high", vm) + if err != nil { + return err + } + // Hint logic + const MASK = math.MaxUint32 + const B = 32 + mask := FeltFromUint(MASK) + // First batch + data := make([]MaybeRelocatable, 0, 4) + for i := uint(0); i < 4; i++ { + data = append(data, *NewMaybeRelocatableFelt(low.Shr(B * i).And(mask))) + } + dataPtr, err = vm.Segments.LoadData(dataPtr, &data) + if err != nil { + return err + } + // Second batch + data = make([]MaybeRelocatable, 0, 4) + for i := uint(0); i < 4; i++ { + data = append(data, *NewMaybeRelocatableFelt(high.Shr(B * i).And(mask))) + } + _, err = vm.Segments.LoadData(dataPtr, &data) + return err +} + func blake2sAddUint256Bigend(ids IdsManager, vm *VirtualMachine) error { // Fetch ids variables dataPtr, err := ids.GetRelocatable("data", vm) diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go index d633820e..85fb2577 100644 --- a/pkg/hints/blake2s_hints_test.go +++ b/pkg/hints/blake2s_hints_test.go @@ -154,6 +154,44 @@ func TestBlake2sComputeOk(t *testing.T) { } } +func TestBlake2sAddUint256Ok(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + data := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "data": {NewMaybeRelocatableRelocatable(data)}, + "high": {NewMaybeRelocatableFelt(FeltFromUint(25))}, + "low": {NewMaybeRelocatableFelt(FeltFromUint(20))}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_ADD_UINT256, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("BLAKE2S_ADD_UINT256 hint test failed with error %s", err) + } + // Check the data segment + dataSegment, err := vm.Segments.GetFeltRange(data, 8) + expectedDataSegment := []Felt{ + FeltFromUint(20), + FeltZero(), + FeltZero(), + FeltZero(), + FeltFromUint(25), + FeltZero(), + FeltZero(), + FeltZero(), + } + if err != nil || !reflect.DeepEqual(dataSegment, expectedDataSegment) { + t.Errorf("Wrong/No data loaded.\n Expected %v, got %v", expectedDataSegment, dataSegment) + } +} + func TestBlake2sAddUint256BigEndOk(t *testing.T) { vm := NewVirtualMachine() vm.Segments.AddSegment() diff --git a/pkg/hints/hint_codes/blake2s_hint_codes.go b/pkg/hints/hint_codes/blake2s_hint_codes.go index c75a9990..156bc3a5 100644 --- a/pkg/hints/hint_codes/blake2s_hint_codes.go +++ b/pkg/hints/hint_codes/blake2s_hint_codes.go @@ -3,6 +3,11 @@ package hint_codes const BLAKE2S_COMPUTE = `from starkware.cairo.common.cairo_blake2s.blake2s_utils import compute_blake2s_func compute_blake2s_func(segments=segments, output_ptr=ids.output)` +const BLAKE2S_ADD_UINT256 = `B = 32 +MASK = 2 ** 32 - 1 +segments.write_arg(ids.data, [(ids.low >> (B * i)) & MASK for i in range(4)]) +segments.write_arg(ids.data + 4, [(ids.high >> (B * i)) & MASK for i in range(4)])` + const BLAKE2S_ADD_UINT256_BIGEND = `B = 32 MASK = 2 ** 32 - 1 segments.write_arg(ids.data, [(ids.high >> (B * (3 - i))) & MASK for i in range(4)]) diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index a43edae5..307ed06b 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -200,6 +200,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return fastEcAddAssignNewY(execScopes) case BLAKE2S_COMPUTE: return blake2sCompute(data.Ids, vm) + case BLAKE2S_ADD_UINT256: + return blake2sAddUint256(data.Ids, vm) case BLAKE2S_ADD_UINT256_BIGEND: return blake2sAddUint256Bigend(data.Ids, vm) case BLAKE2S_FINALIZE: From 16982bc0e4f89a5937de6254e806166a36366cce Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 17:13:54 -0300 Subject: [PATCH 20/31] Add hint + integration test --- cairo_programs/finalize_blake2s_v2.cairo | 68 ++++++++++++++++++++++ pkg/hints/hint_codes/blake2s_hint_codes.go | 42 +++++++++++++ pkg/hints/hint_processor.go | 2 +- pkg/vm/cairo_run/cairo_run_test.go | 4 ++ 4 files changed, 115 insertions(+), 1 deletion(-) create mode 100644 cairo_programs/finalize_blake2s_v2.cairo diff --git a/cairo_programs/finalize_blake2s_v2.cairo b/cairo_programs/finalize_blake2s_v2.cairo new file mode 100644 index 00000000..3d80fe13 --- /dev/null +++ b/cairo_programs/finalize_blake2s_v2.cairo @@ -0,0 +1,68 @@ +%builtins range_check bitwise + +from starkware.cairo.common.alloc import alloc +from starkware.cairo.common.cairo_blake2s.blake2s import blake2s, _finalize_blake2s_inner, _get_sigma, INSTANCE_SIZE, INPUT_BLOCK_FELTS +from starkware.cairo.common.cairo_blake2s.packed_blake2s import N_PACKED_INSTANCES, blake2s_compress +from starkware.cairo.common.cairo_builtins import BitwiseBuiltin +from starkware.cairo.common.registers import get_fp_and_pc +from starkware.cairo.common.math import assert_nn_le, split_felt, unsigned_div_rem + +const BLAKE2S_INPUT_CHUNK_SIZE_FELTS = INPUT_BLOCK_FELTS; + +// Verifies that the results of blake2s() are valid. +func finalize_blake2s{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}( + blake2s_ptr_start: felt*, blake2s_ptr_end: felt* +) { + alloc_locals; + + let (__fp__, _) = get_fp_and_pc(); + + let (sigma) = _get_sigma(); + + tempvar n = (blake2s_ptr_end - blake2s_ptr_start) / INSTANCE_SIZE; + if (n == 0) { + return (); + } + + %{ + # Add dummy pairs of input and output. + from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + + _n_packed_instances = int(ids.N_PACKED_INSTANCES) + assert 0 <= _n_packed_instances < 20 + _blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS) + assert 0 <= _blake2s_input_chunk_size_felts < 100 + + message = [0] * _blake2s_input_chunk_size_felts + modified_iv = [IV[0] ^ 0x01010020] + IV[1:] + output = blake2s_compress( + message=message, + h=modified_iv, + t0=0, + t1=0, + f0=0xffffffff, + f1=0, + ) + padding = (modified_iv + message + [0, 0xffffffff] + output) * (_n_packed_instances - 1) + segments.write_arg(ids.blake2s_ptr_end, padding) + %} + + // Compute the amount of chunks (rounded up). + let (local n_chunks, _) = unsigned_div_rem(n + N_PACKED_INSTANCES - 1, N_PACKED_INSTANCES); + let blake2s_ptr = blake2s_ptr_start; + _finalize_blake2s_inner{blake2s_ptr=blake2s_ptr}(n=n_chunks, sigma=sigma); + return (); +} + +func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { + alloc_locals; + let inputs: felt* = alloc(); + assert inputs[0] = 'Hell'; + assert inputs[1] = 'o Wo'; + assert inputs[2] = 'rld'; + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + let (output) = blake2s{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}(inputs, 9); + finalize_blake2s(blake2s_ptr_start, blake2s_ptr); + return (); +} \ No newline at end of file diff --git a/pkg/hints/hint_codes/blake2s_hint_codes.go b/pkg/hints/hint_codes/blake2s_hint_codes.go index 156bc3a5..766f8b4c 100644 --- a/pkg/hints/hint_codes/blake2s_hint_codes.go +++ b/pkg/hints/hint_codes/blake2s_hint_codes.go @@ -33,3 +33,45 @@ output = blake2s_compress( ) padding = (modified_iv + message + [0, 0xffffffff] + output) * (_n_packed_instances - 1) segments.write_arg(ids.blake2s_ptr_end, padding)` + +const BLAKE2S_FINALIZE_V2 = `# Add dummy pairs of input and output. +from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + +_n_packed_instances = int(ids.N_PACKED_INSTANCES) +assert 0 <= _n_packed_instances < 20 +_blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS) +assert 0 <= _blake2s_input_chunk_size_felts < 100 + +message = [0] * _blake2s_input_chunk_size_felts +modified_iv = [IV[0] ^ 0x01010020] + IV[1:] +output = blake2s_compress( + message=message, + h=modified_iv, + t0=0, + t1=0, + f0=0xffffffff, + f1=0, +) +padding = (modified_iv + message + [0, 0xffffffff] + output) * (_n_packed_instances - 1) +segments.write_arg(ids.blake2s_ptr_end, padding)` + +const BLAKE2S_FINALIZE_V3 = `# Add dummy pairs of input and output. +from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + +_n_packed_instances = int(ids.N_PACKED_INSTANCES) +assert 0 <= _n_packed_instances < 20 +_blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS) +assert 0 <= _blake2s_input_chunk_size_felts < 100 + +message = [0] * _blake2s_input_chunk_size_felts +modified_iv = [IV[0] ^ 0x01010020] + IV[1:] +output = blake2s_compress( + message=message, + h=modified_iv, + t0=0, + t1=0, + f0=0xffffffff, + f1=0, +) +padding = (message + modified_iv + [0, 0xffffffff] + output) * (_n_packed_instances - 1) +segments.write_arg(ids.blake2s_ptr_end, padding)` diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index 307ed06b..b0cac47d 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -204,7 +204,7 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return blake2sAddUint256(data.Ids, vm) case BLAKE2S_ADD_UINT256_BIGEND: return blake2sAddUint256Bigend(data.Ids, vm) - case BLAKE2S_FINALIZE: + case BLAKE2S_FINALIZE, BLAKE2S_FINALIZE_V2: return blake2sFinalize(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index dd870078..76470171 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -364,3 +364,7 @@ func TestFinalizeBlake2s(t *testing.T) { func TestBlake2sIntegrationTests(t *testing.T) { testProgram("blake2s_integration_tests", t) } + +func TestFinalizeBlake2sV2(t *testing.T) { + testProgram("finalize_blake2s_v2", t) +} From e94855cf8d16606cc6b2f51f128d2c926d74b358 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 17:27:58 -0300 Subject: [PATCH 21/31] Implement finalize v3 --- pkg/hints/blake2s_hints.go | 23 +++++++++++++++ pkg/hints/blake2s_hints_test.go | 52 +++++++++++++++++++++++++++++++++ pkg/hints/hint_processor.go | 2 ++ 3 files changed, 77 insertions(+) diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go index 43487a99..d029961e 100644 --- a/pkg/hints/blake2s_hints.go +++ b/pkg/hints/blake2s_hints.go @@ -179,3 +179,26 @@ func blake2sFinalize(ids IdsManager, vm *VirtualMachine) error { _, err = vm.Segments.LoadData(blake2sPtrEnd, &data) return err } + +func blake2sFinalizeV3(ids IdsManager, vm *VirtualMachine) error { + const N_PACKED_INSTANCES = 7 + blake2sPtrEnd, err := ids.GetRelocatable("blake2s_ptr_end", vm) + if err != nil { + return err + } + var message [16]uint32 + modifiedIv := IV() + modifiedIv[0] = modifiedIv[0] ^ 0x01010020 + output := Blake2sCompress(modifiedIv, message, 0, 0, 0xffffffff, 0) + padding := message[:] + padding = append(padding, modifiedIv[:]...) + padding = append(padding, 0, 0xffffffff) + padding = append(padding, output[:]...) + fullPadding := padding + for i := 2; i < N_PACKED_INSTANCES; i++ { + fullPadding = append(fullPadding, padding...) + } + data := Uint32SliceToMRSlice(fullPadding) + _, err = vm.Segments.LoadData(blake2sPtrEnd, &data) + return err +} diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go index 85fb2577..05396010 100644 --- a/pkg/hints/blake2s_hints_test.go +++ b/pkg/hints/blake2s_hints_test.go @@ -281,3 +281,55 @@ func TestBlake2sFinaizeOk(t *testing.T) { t.Errorf("Wrong/No data loaded.\n Expected: %v.\n Got: %v", expectedDataSegment, dataSegment) } } + +func TestBlake2sFinaizeV3Ok(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + data := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "blake2s_ptr_end": {NewMaybeRelocatableRelocatable(data)}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: BLAKE2S_FINALIZE_V3, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("BLAKE2S_FINALIZE_V3 hint test failed with error %s", err) + } + // Check the data segment + dataSegment, err := vm.Segments.GetFeltRange(data, 204) + + expectedDataSegment := []Felt{ + FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + + FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), FeltZero(), + FeltFromUint(1795745351), FeltFromUint(3144134277), FeltFromUint(1013904242), FeltFromUint(2773480762), FeltFromUint(1359893119), FeltFromUint(2600822924), FeltFromUint(528734635), FeltFromUint(1541459225), FeltZero(), FeltFromUint(4294967295), FeltFromUint(813310313), + FeltFromUint(2491453561), FeltFromUint(3491828193), FeltFromUint(2085238082), FeltFromUint(1219908895), FeltFromUint(514171180), FeltFromUint(4245497115), FeltFromUint(4193177630), + } + if err != nil || !reflect.DeepEqual(dataSegment, expectedDataSegment) { + t.Errorf("Wrong/No data loaded.\n Expected: %v.\n Got: %v", expectedDataSegment, dataSegment) + } +} diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index b0cac47d..b7985a67 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -206,6 +206,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return blake2sAddUint256Bigend(data.Ids, vm) case BLAKE2S_FINALIZE, BLAKE2S_FINALIZE_V2: return blake2sFinalize(data.Ids, vm) + case BLAKE2S_FINALIZE_V3: + return blake2sFinalizeV3(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) } From c3c77e93e74cadac26cb5768595c0c8f407d4cd5 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 17:31:34 -0300 Subject: [PATCH 22/31] Add integration test --- cairo_programs/example_blake2s.cairo | 654 +++++++++++++++++++++++++++ pkg/vm/cairo_run/cairo_run_test.go | 4 + 2 files changed, 658 insertions(+) create mode 100644 cairo_programs/example_blake2s.cairo diff --git a/cairo_programs/example_blake2s.cairo b/cairo_programs/example_blake2s.cairo new file mode 100644 index 00000000..64db4966 --- /dev/null +++ b/cairo_programs/example_blake2s.cairo @@ -0,0 +1,654 @@ +%builtins range_check bitwise + +// Code taken from https://github.com/starkware-libs/cairo-examples/blob/master/blake2s/blake2s.cairo & https://github.com/starkware-libs/cairo-examples/blob/master/blake2s/packed_blake2s.cairo +from starkware.cairo.common.alloc import alloc +from starkware.cairo.common.registers import get_fp_and_pc +from starkware.cairo.common.cairo_builtins import BitwiseBuiltin +from starkware.cairo.common.math import assert_nn_le, unsigned_div_rem +from starkware.cairo.common.memset import memset +from starkware.cairo.common.pow import pow + +const BLAKE2S_INPUT_CHUNK_SIZE_FELTS = 16; +const BLAKE2S_STATE_SIZE_FELTS = 8; +// Each instance consists of 16 words of message, 8 words for the input state, 8 words +// for the output state and 2 words for t0 and f0. +const BLAKE2S_INSTANCE_SIZE = BLAKE2S_INPUT_CHUNK_SIZE_FELTS + 2 * BLAKE2S_STATE_SIZE_FELTS + 2; + +const N_PACKED_INSTANCES = 7; +const ALL_ONES = 2 ** 251 - 1; +const SHIFTS = 1 + 2 ** 35 + 2 ** (35 * 2) + 2 ** (35 * 3) + 2 ** (35 * 4) + 2 ** (35 * 5) + 2 ** ( + 35 * 6 +); + +// Computes blake2s of 'input'. Inputs of up to 64 bytes are supported. +// To use this function, split the input into (up to) 16 words of 32 bits (little endian). +// For example, to compute blake2s('Hello world'), use: +// input = [1819043144, 1870078063, 6581362] +// where: +// 1819043144 == int.from_bytes(b'Hell', 'little') +// 1870078063 == int.from_bytes(b'o wo', 'little') +// 6581362 == int.from_bytes(b'rld', 'little') +// +// output is an array of 8 32-bit words (little endian). +// +// Assumption: n_bytes <= 64. +// +// Note: You must call finalize_blake2s() at the end of the program. Otherwise, this function +// is not sound and a malicious prover may return a wrong result. +// Note: the interface of this function may change in the future. +func blake2s{range_check_ptr, blake2s_ptr: felt*}(input: felt*, n_bytes: felt) -> (output: felt*) { + assert_nn_le(n_bytes, 64); + let blake2s_start = blake2s_ptr; + _blake2s_input(input=input, n_bytes=n_bytes, n_words=BLAKE2S_INPUT_CHUNK_SIZE_FELTS); + + // Set the initial state to IV (IV[0] is modified). + assert blake2s_ptr[0] = 0x6B08E647; // IV[0] ^ 0x01010020 (config: no key, 32 bytes output). + assert blake2s_ptr[1] = 0xBB67AE85; + assert blake2s_ptr[2] = 0x3C6EF372; + assert blake2s_ptr[3] = 0xA54FF53A; + assert blake2s_ptr[4] = 0x510E527F; + assert blake2s_ptr[5] = 0x9B05688C; + assert blake2s_ptr[6] = 0x1F83D9AB; + assert blake2s_ptr[7] = 0x5BE0CD19; + let blake2s_ptr = blake2s_ptr + BLAKE2S_STATE_SIZE_FELTS; + + assert blake2s_ptr[0] = n_bytes; // n_bytes. + assert blake2s_ptr[1] = 0xffffffff; // Is last byte = True. + let blake2s_ptr = blake2s_ptr + 2; + + let output = blake2s_ptr; + %{ + from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + + _blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS) + assert 0 <= _blake2s_input_chunk_size_felts < 100 + + new_state = blake2s_compress( + message=memory.get_range(ids.blake2s_start, _blake2s_input_chunk_size_felts), + h=[IV[0] ^ 0x01010020] + IV[1:], + t0=ids.n_bytes, + t1=0, + f0=0xffffffff, + f1=0, + ) + + segments.write_arg(ids.output, new_state) + %} + let blake2s_ptr = blake2s_ptr + BLAKE2S_STATE_SIZE_FELTS; + return (output,); +} + +func _blake2s_input{range_check_ptr, blake2s_ptr: felt*}( + input: felt*, n_bytes: felt, n_words: felt +) { + alloc_locals; + + local full_word; + %{ ids.full_word = int(ids.n_bytes >= 4) %} + + if (full_word != 0) { + assert blake2s_ptr[0] = input[0]; + let blake2s_ptr = blake2s_ptr + 1; + return _blake2s_input(input=input + 1, n_bytes=n_bytes - 4, n_words=n_words - 1); + } + + // This is the last input word, so we should fill the rest with zeros. + + if (n_bytes == 0) { + memset(dst=blake2s_ptr, value=0, n=n_words); + let blake2s_ptr = blake2s_ptr + n_words; + return (); + } + + assert_nn_le(n_bytes, 3); + local range_check_ptr = range_check_ptr; + + assert blake2s_ptr[0] = input[0]; + + memset(dst=blake2s_ptr + 1, value=0, n=n_words - 1); + let blake2s_ptr = blake2s_ptr + n_words; + return (); +} + +// Verifies that the results of blake2s() are valid. +func finalize_blake2s{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}( + blake2s_ptr_start: felt*, blake2s_ptr_end: felt* +) { + alloc_locals; + + let (__fp__, _) = get_fp_and_pc(); + + let (sigma) = _get_sigma(); + + tempvar n = (blake2s_ptr_end - blake2s_ptr_start) / BLAKE2S_INSTANCE_SIZE; + if (n == 0) { + return (); + } + + %{ + # Add dummy pairs of input and output. + from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + + _n_packed_instances = int(ids.N_PACKED_INSTANCES) + assert 0 <= _n_packed_instances < 20 + _blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS) + assert 0 <= _blake2s_input_chunk_size_felts < 100 + + message = [0] * _blake2s_input_chunk_size_felts + modified_iv = [IV[0] ^ 0x01010020] + IV[1:] + output = blake2s_compress( + message=message, + h=modified_iv, + t0=0, + t1=0, + f0=0xffffffff, + f1=0, + ) + padding = (message + modified_iv + [0, 0xffffffff] + output) * (_n_packed_instances - 1) + segments.write_arg(ids.blake2s_ptr_end, padding) + %} + + // Compute the amount of chunks (rounded up). + let (local n_chunks, _) = unsigned_div_rem(n + N_PACKED_INSTANCES - 1, N_PACKED_INSTANCES); + let blake2s_ptr = blake2s_ptr_start; + _finalize_blake2s_inner{blake2s_ptr=blake2s_ptr}(n=n_chunks, sigma=sigma); + return (); +} + +func _get_sigma() -> (sigma: felt*) { + alloc_locals; + let (__fp__, _) = get_fp_and_pc(); + local sigma = 0; + local a = 1; + local a = 2; + local a = 3; + local a = 4; + local a = 5; + local a = 6; + local a = 7; + local a = 8; + local a = 9; + local a = 10; + local a = 11; + local a = 12; + local a = 13; + local a = 14; + local a = 15; + local a = 14; + local a = 10; + local a = 4; + local a = 8; + local a = 9; + local a = 15; + local a = 13; + local a = 6; + local a = 1; + local a = 12; + local a = 0; + local a = 2; + local a = 11; + local a = 7; + local a = 5; + local a = 3; + local a = 11; + local a = 8; + local a = 12; + local a = 0; + local a = 5; + local a = 2; + local a = 15; + local a = 13; + local a = 10; + local a = 14; + local a = 3; + local a = 6; + local a = 7; + local a = 1; + local a = 9; + local a = 4; + local a = 7; + local a = 9; + local a = 3; + local a = 1; + local a = 13; + local a = 12; + local a = 11; + local a = 14; + local a = 2; + local a = 6; + local a = 5; + local a = 10; + local a = 4; + local a = 0; + local a = 15; + local a = 8; + local a = 9; + local a = 0; + local a = 5; + local a = 7; + local a = 2; + local a = 4; + local a = 10; + local a = 15; + local a = 14; + local a = 1; + local a = 11; + local a = 12; + local a = 6; + local a = 8; + local a = 3; + local a = 13; + local a = 2; + local a = 12; + local a = 6; + local a = 10; + local a = 0; + local a = 11; + local a = 8; + local a = 3; + local a = 4; + local a = 13; + local a = 7; + local a = 5; + local a = 15; + local a = 14; + local a = 1; + local a = 9; + local a = 12; + local a = 5; + local a = 1; + local a = 15; + local a = 14; + local a = 13; + local a = 4; + local a = 10; + local a = 0; + local a = 7; + local a = 6; + local a = 3; + local a = 9; + local a = 2; + local a = 8; + local a = 11; + local a = 13; + local a = 11; + local a = 7; + local a = 14; + local a = 12; + local a = 1; + local a = 3; + local a = 9; + local a = 5; + local a = 0; + local a = 15; + local a = 4; + local a = 8; + local a = 6; + local a = 2; + local a = 10; + local a = 6; + local a = 15; + local a = 14; + local a = 9; + local a = 11; + local a = 3; + local a = 0; + local a = 8; + local a = 12; + local a = 2; + local a = 13; + local a = 7; + local a = 1; + local a = 4; + local a = 10; + local a = 5; + local a = 10; + local a = 2; + local a = 8; + local a = 4; + local a = 7; + local a = 6; + local a = 1; + local a = 5; + local a = 15; + local a = 11; + local a = 9; + local a = 14; + local a = 3; + local a = 12; + local a = 13; + local a = 0; + return (&sigma,); +} + +// Handles n chunks of N_PACKED_INSTANCES blake2s instances. +func _finalize_blake2s_inner{range_check_ptr, bitwise_ptr: BitwiseBuiltin*, blake2s_ptr: felt*}( + n: felt, sigma: felt* +) { + if (n == 0) { + return (); + } + + alloc_locals; + + local MAX_VALUE = 2 ** 32 - 1; + + let blake2s_start = blake2s_ptr; + + // Load instance data. + let (local data: felt*) = alloc(); + _pack_ints(BLAKE2S_INSTANCE_SIZE, data); + + let message = data; + let input_state = message + BLAKE2S_INPUT_CHUNK_SIZE_FELTS; + let t0_and_f0 = input_state + BLAKE2S_STATE_SIZE_FELTS; + let output_state = t0_and_f0 + 2; + + // Run blake2s on N_PACKED_INSTANCES instances. + local blake2s_ptr: felt* = blake2s_ptr; + local range_check_ptr = range_check_ptr; + blake2s_compress( + h=input_state, + message=data, + t0=t0_and_f0[0], + f0=t0_and_f0[1], + sigma=sigma, + output=output_state, + ); + + local bitwise_ptr: BitwiseBuiltin* = bitwise_ptr; + + let blake2s_ptr = blake2s_start + BLAKE2S_INSTANCE_SIZE * N_PACKED_INSTANCES; + + return _finalize_blake2s_inner(n=n - 1, sigma=sigma); +} + +// Given N_PACKED_INSTANCES sets of m (32-bit) integers in the blake2s implicit argument, +// where each set starts at offset BLAKE2S_INSTANCE_SIZE from the previous set, +// computes m packed integers. +// blake2s_ptr is advanced m steps (just after the first set). +func _pack_ints{range_check_ptr, blake2s_ptr: felt*}(m, packed_values: felt*) { + static_assert N_PACKED_INSTANCES == 7; + alloc_locals; + + local MAX_VALUE = 2 ** 32 - 1; + + // TODO: consider using split_int(). + tempvar packed_values = packed_values; + tempvar blake2s_ptr = blake2s_ptr; + tempvar range_check_ptr = range_check_ptr; + tempvar m = m; + + loop: + tempvar x0 = blake2s_ptr[0 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 0] = x0; + assert [range_check_ptr + 1] = MAX_VALUE - x0; + tempvar x1 = blake2s_ptr[1 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 2] = x1; + assert [range_check_ptr + 3] = MAX_VALUE - x1; + tempvar x2 = blake2s_ptr[2 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 4] = x2; + assert [range_check_ptr + 5] = MAX_VALUE - x2; + tempvar x3 = blake2s_ptr[3 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 6] = x3; + assert [range_check_ptr + 7] = MAX_VALUE - x3; + tempvar x4 = blake2s_ptr[4 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 8] = x4; + assert [range_check_ptr + 9] = MAX_VALUE - x4; + tempvar x5 = blake2s_ptr[5 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 10] = x5; + assert [range_check_ptr + 11] = MAX_VALUE - x5; + tempvar x6 = blake2s_ptr[6 * BLAKE2S_INSTANCE_SIZE]; + assert [range_check_ptr + 12] = x6; + assert [range_check_ptr + 13] = MAX_VALUE - x6; + assert packed_values[0] = x0 + 2 ** 35 * x1 + 2 ** (35 * 2) * x2 + 2 ** (35 * 3) * x3 + 2 ** ( + 35 * 4 + ) * x4 + 2 ** (35 * 5) * x5 + 2 ** (35 * 6) * x6; + + tempvar packed_values = packed_values + 1; + tempvar blake2s_ptr = blake2s_ptr + 1; + tempvar range_check_ptr = range_check_ptr + 14; + tempvar m = m - 1; + jmp loop if m != 0; + + return (); +} + +func mix{bitwise_ptr: BitwiseBuiltin*}(a: felt, b: felt, c: felt, d: felt, m0: felt, m1: felt) -> ( + a: felt, b: felt, c: felt, d: felt +) { + alloc_locals; + + // Defining the following constant as local variables saves some instructions. + local mask32ones = SHIFTS * (2 ** 32 - 1); + + // a = (a + b + m0) % 2**32 + assert bitwise_ptr[0].x = a + b + m0; + assert bitwise_ptr[0].y = mask32ones; + tempvar a = bitwise_ptr[0].x_and_y; + let bitwise_ptr = bitwise_ptr + BitwiseBuiltin.SIZE; + + // d = right_rot((d ^ a), 16) + assert bitwise_ptr[0].x = a; + assert bitwise_ptr[0].y = d; + tempvar a_xor_d = bitwise_ptr[0].x_xor_y; + assert bitwise_ptr[1].x = a_xor_d; + assert bitwise_ptr[1].y = SHIFTS * (2 ** 32 - 2 ** 16); + tempvar d = (2 ** (32 - 16)) * a_xor_d + (1 / 2 ** 16 - 2 ** (32 - 16)) * bitwise_ptr[ + 1 + ].x_and_y; + let bitwise_ptr = bitwise_ptr + 2 * BitwiseBuiltin.SIZE; + + // c = (c + d) % 2**32 + assert bitwise_ptr[0].x = c + d; + assert bitwise_ptr[0].y = mask32ones; + tempvar c = bitwise_ptr[0].x_and_y; + let bitwise_ptr = bitwise_ptr + BitwiseBuiltin.SIZE; + + // b = right_rot((b ^ c), 12) + assert bitwise_ptr[0].x = b; + assert bitwise_ptr[0].y = c; + tempvar b_xor_c = bitwise_ptr[0].x_xor_y; + assert bitwise_ptr[1].x = b_xor_c; + assert bitwise_ptr[1].y = SHIFTS * (2 ** 32 - 2 ** 12); + tempvar b = (2 ** (32 - 12)) * b_xor_c + (1 / 2 ** 12 - 2 ** (32 - 12)) * bitwise_ptr[ + 1 + ].x_and_y; + let bitwise_ptr = bitwise_ptr + 2 * BitwiseBuiltin.SIZE; + + // a = (a + b + m1) % 2**32 + assert bitwise_ptr[0].x = a + b + m1; + assert bitwise_ptr[0].y = mask32ones; + tempvar a = bitwise_ptr[0].x_and_y; + let bitwise_ptr = bitwise_ptr + BitwiseBuiltin.SIZE; + + // d = right_rot((d ^ a), 8) + assert bitwise_ptr[0].x = d; + assert bitwise_ptr[0].y = a; + tempvar d_xor_a = bitwise_ptr[0].x_xor_y; + assert bitwise_ptr[1].x = d_xor_a; + assert bitwise_ptr[1].y = SHIFTS * (2 ** 32 - 2 ** 8); + tempvar d = (2 ** (32 - 8)) * d_xor_a + (1 / 2 ** 8 - 2 ** (32 - 8)) * bitwise_ptr[1].x_and_y; + let bitwise_ptr = bitwise_ptr + 2 * BitwiseBuiltin.SIZE; + + // c = (c + d) % 2**32 + assert bitwise_ptr[0].x = c + d; + assert bitwise_ptr[0].y = mask32ones; + tempvar c = bitwise_ptr[0].x_and_y; + let bitwise_ptr = bitwise_ptr + BitwiseBuiltin.SIZE; + + // b = right_rot((b ^ c), 7) + assert bitwise_ptr[0].x = b; + assert bitwise_ptr[0].y = c; + tempvar b_xor_c = bitwise_ptr[0].x_xor_y; + assert bitwise_ptr[1].x = b_xor_c; + assert bitwise_ptr[1].y = SHIFTS * (2 ** 32 - 2 ** 7); + tempvar b = (2 ** (32 - 7)) * b_xor_c + (1 / 2 ** 7 - 2 ** (32 - 7)) * bitwise_ptr[1].x_and_y; + let bitwise_ptr = bitwise_ptr + 2 * BitwiseBuiltin.SIZE; + + return (a, b, c, d); +} + +func blake_round{bitwise_ptr: BitwiseBuiltin*}(state: felt*, message: felt*, sigma: felt*) -> ( + new_state: felt* +) { + let state0 = state[0]; + let state1 = state[1]; + let state2 = state[2]; + let state3 = state[3]; + let state4 = state[4]; + let state5 = state[5]; + let state6 = state[6]; + let state7 = state[7]; + let state8 = state[8]; + let state9 = state[9]; + let state10 = state[10]; + let state11 = state[11]; + let state12 = state[12]; + let state13 = state[13]; + let state14 = state[14]; + let state15 = state[15]; + + let (state0, state4, state8, state12) = mix( + state0, state4, state8, state12, message[sigma[0]], message[sigma[1]] + ); + let (state1, state5, state9, state13) = mix( + state1, state5, state9, state13, message[sigma[2]], message[sigma[3]] + ); + let (state2, state6, state10, state14) = mix( + state2, state6, state10, state14, message[sigma[4]], message[sigma[5]] + ); + let (state3, state7, state11, state15) = mix( + state3, state7, state11, state15, message[sigma[6]], message[sigma[7]] + ); + + let (state0, state5, state10, state15) = mix( + state0, state5, state10, state15, message[sigma[8]], message[sigma[9]] + ); + let (state1, state6, state11, state12) = mix( + state1, state6, state11, state12, message[sigma[10]], message[sigma[11]] + ); + let (state2, state7, state8, state13) = mix( + state2, state7, state8, state13, message[sigma[12]], message[sigma[13]] + ); + let (state3, state4, state9, state14) = mix( + state3, state4, state9, state14, message[sigma[14]], message[sigma[15]] + ); + + let (new_state: felt*) = alloc(); + assert new_state[0] = state0; + assert new_state[1] = state1; + assert new_state[2] = state2; + assert new_state[3] = state3; + assert new_state[4] = state4; + assert new_state[5] = state5; + assert new_state[6] = state6; + assert new_state[7] = state7; + assert new_state[8] = state8; + assert new_state[9] = state9; + assert new_state[10] = state10; + assert new_state[11] = state11; + assert new_state[12] = state12; + assert new_state[13] = state13; + assert new_state[14] = state14; + assert new_state[15] = state15; + + return (new_state,); +} + +// Performs the blake compression function. +// +// h is a list of 8 32-bit words. +// message is a list of 16 32-bit words. +// t1 and f1 are assumed to be 0. +func blake2s_compress{bitwise_ptr: BitwiseBuiltin*}( + h: felt*, message: felt*, t0: felt, f0: felt, sigma: felt*, output: felt* +) { + alloc_locals; + let (__fp__, _) = get_fp_and_pc(); + + // Compute state[12]. + assert bitwise_ptr[0].x = 0x510e527f * SHIFTS; + assert bitwise_ptr[0].y = t0; + let state12 = bitwise_ptr[0].x_xor_y; + let bitwise_ptr = bitwise_ptr + BitwiseBuiltin.SIZE; + + // Compute state[14]. + assert bitwise_ptr[0].x = 0x1f83d9ab * SHIFTS; + assert bitwise_ptr[0].y = f0; + let state14 = bitwise_ptr[0].x_xor_y; + let bitwise_ptr = bitwise_ptr + BitwiseBuiltin.SIZE; + + local initial_state = h[0]; + local initial_state_ = h[1]; + local initial_state_ = h[2]; + local initial_state_ = h[3]; + local initial_state_ = h[4]; + local initial_state_ = h[5]; + local initial_state_ = h[6]; + local initial_state_ = h[7]; + local initial_state_ = 0x6a09e667 * SHIFTS; + local initial_state_ = 0xbb67ae85 * SHIFTS; + local initial_state_ = 0x3c6ef372 * SHIFTS; + local initial_state_ = 0xa54ff53a * SHIFTS; + local initial_state_ = state12; + local initial_state_ = 0x9b05688c * SHIFTS; + local initial_state_ = state14; + local initial_state_ = 0x5be0cd19 * SHIFTS; + + let state = &initial_state; + + let (state) = blake_round(state, message, sigma + 16 * 0); + let (state) = blake_round(state, message, sigma + 16 * 1); + let (state) = blake_round(state, message, sigma + 16 * 2); + let (state) = blake_round(state, message, sigma + 16 * 3); + let (state) = blake_round(state, message, sigma + 16 * 4); + let (state) = blake_round(state, message, sigma + 16 * 5); + let (state) = blake_round(state, message, sigma + 16 * 6); + let (state) = blake_round(state, message, sigma + 16 * 7); + let (state) = blake_round(state, message, sigma + 16 * 8); + let (state) = blake_round(state, message, sigma + 16 * 9); + + tempvar old_h = h; + tempvar last_state = state; + tempvar new_h = output; + tempvar bitwise_ptr = bitwise_ptr; + tempvar n = 8; + + loop: + assert bitwise_ptr[0].x = old_h[0]; + assert bitwise_ptr[0].y = last_state[0]; + assert bitwise_ptr[1].x = bitwise_ptr[0].x_xor_y; + assert bitwise_ptr[1].y = last_state[8]; + assert new_h[0] = bitwise_ptr[1].x_xor_y; + + tempvar old_h = old_h + 1; + tempvar last_state = last_state + 1; + tempvar new_h = new_h + 1; + tempvar bitwise_ptr = bitwise_ptr + 2 * BitwiseBuiltin.SIZE; + tempvar n = n - 1; + jmp loop if n != 0; + + return (); +} + +func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { + alloc_locals; + let inputs: felt* = alloc(); + assert inputs[0] = 'Hell'; + assert inputs[1] = 'o Wo'; + assert inputs[2] = 'rld'; + let (local blake2s_ptr_start) = alloc(); + let blake2s_ptr = blake2s_ptr_start; + let (output) = blake2s{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}(inputs, 9); + assert output[0] = 3718547061; + assert output[1] = 125168665; + assert output[2] = 1035352101; + assert output[3] = 2775751047; + assert output[4] = 2953291512; + assert output[5] = 1978410869; + assert output[6] = 3956807281; + assert output[7] = 3738027290; + finalize_blake2s(blake2s_ptr_start, blake2s_ptr); + return (); +} diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index 76470171..027c5d04 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -368,3 +368,7 @@ func TestBlake2sIntegrationTests(t *testing.T) { func TestFinalizeBlake2sV2(t *testing.T) { testProgram("finalize_blake2s_v2", t) } + +func TestExampleBlake2s(t *testing.T) { + testProgram("example_blake2s", t) +} From 0346fd58fa1672d039fa977e2a574c6ffe4537d5 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 17:38:41 -0300 Subject: [PATCH 23/31] Implement sha256 input hint --- pkg/hints/hint_codes/sha256_hint_codes.go | 3 +++ pkg/hints/hint_processor.go | 2 ++ pkg/hints/sha256_hints.go | 19 +++++++++++++++++++ 3 files changed, 24 insertions(+) create mode 100644 pkg/hints/hint_codes/sha256_hint_codes.go create mode 100644 pkg/hints/sha256_hints.go diff --git a/pkg/hints/hint_codes/sha256_hint_codes.go b/pkg/hints/hint_codes/sha256_hint_codes.go new file mode 100644 index 00000000..3402a10d --- /dev/null +++ b/pkg/hints/hint_codes/sha256_hint_codes.go @@ -0,0 +1,3 @@ +package hint_codes + +const SHA256_INPUT = "ids.full_word = int(ids.n_bytes >= 4)" diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index b7985a67..89636206 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -208,6 +208,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return blake2sFinalize(data.Ids, vm) case BLAKE2S_FINALIZE_V3: return blake2sFinalizeV3(data.Ids, vm) + case SHA256_INPUT: + return sha256Input(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) } diff --git a/pkg/hints/sha256_hints.go b/pkg/hints/sha256_hints.go new file mode 100644 index 00000000..91d330c9 --- /dev/null +++ b/pkg/hints/sha256_hints.go @@ -0,0 +1,19 @@ +package hints + +import ( + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_utils" + . "github.com/lambdaclass/cairo-vm.go/pkg/lambdaworks" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm/memory" +) + +func sha256Input(ids IdsManager, vm *VirtualMachine) error { + nBytes, err := ids.GetFelt("n_bytes", vm) + if err != nil { + return err + } + if nBytes.Cmp(FeltFromUint(4)) != -1 { + return ids.Insert("full_word", NewMaybeRelocatableFelt(FeltOne()), vm) + } + return ids.Insert("full_word", NewMaybeRelocatableFelt(FeltZero()), vm) +} From 76bc3e5d92fa2956b99ec72e037836edd096fbef Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 3 Oct 2023 17:45:22 -0300 Subject: [PATCH 24/31] Add unit tests --- pkg/hints/sha256_hints_test.go | 64 ++++++++++++++++++++++++++++++++++ 1 file changed, 64 insertions(+) create mode 100644 pkg/hints/sha256_hints_test.go diff --git a/pkg/hints/sha256_hints_test.go b/pkg/hints/sha256_hints_test.go new file mode 100644 index 00000000..87c23091 --- /dev/null +++ b/pkg/hints/sha256_hints_test.go @@ -0,0 +1,64 @@ +package hints_test + +import ( + "testing" + + . "github.com/lambdaclass/cairo-vm.go/pkg/hints" + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_codes" + . "github.com/lambdaclass/cairo-vm.go/pkg/hints/hint_utils" + . "github.com/lambdaclass/cairo-vm.go/pkg/lambdaworks" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm" + . "github.com/lambdaclass/cairo-vm.go/pkg/vm/memory" +) + +func TestSha256InputFalse(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "n_bytes": {NewMaybeRelocatableFelt(FeltFromUint64(2))}, + "full_word": {nil}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: SHA256_INPUT, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("SHA256_INPUT hint test failed with error %s", err) + } + // Check ids.full_word + fullWord, err := idsManager.GetFelt("full_word", vm) + if err != nil || fullWord.Cmp(FeltZero()) != 0 { + t.Error("Wrong/No value inserted into ids.full_word") + } +} + +func TestSha256InputTrue(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "n_bytes": {NewMaybeRelocatableFelt(FeltFromUint64(8))}, + "full_word": {nil}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: SHA256_INPUT, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err != nil { + t.Errorf("SHA256_INPUT hint test failed with error %s", err) + } + // Check ids.full_word + fullWord, err := idsManager.GetFelt("full_word", vm) + if err != nil || fullWord.Cmp(FeltOne()) != 0 { + t.Error("Wrong/No value inserted into ids.full_word") + } +} From fa337136ba08f42cb407fdaf452349d40fe18c42 Mon Sep 17 00:00:00 2001 From: Federica Date: Wed, 4 Oct 2023 10:49:20 -0300 Subject: [PATCH 25/31] Add exmaple blake compress hint --- cairo_programs/finalize_blake2s_v2.cairo | 2 +- pkg/hints/blake2s_hints.go | 31 ++++++++++++++++++++++ pkg/hints/hint_codes/blake2s_hint_codes.go | 16 +++++++++++ pkg/hints/hint_processor.go | 2 ++ 4 files changed, 50 insertions(+), 1 deletion(-) diff --git a/cairo_programs/finalize_blake2s_v2.cairo b/cairo_programs/finalize_blake2s_v2.cairo index 3d80fe13..2d914a14 100644 --- a/cairo_programs/finalize_blake2s_v2.cairo +++ b/cairo_programs/finalize_blake2s_v2.cairo @@ -65,4 +65,4 @@ func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { let (output) = blake2s{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}(inputs, 9); finalize_blake2s(blake2s_ptr_start, blake2s_ptr); return (); -} \ No newline at end of file +} diff --git a/pkg/hints/blake2s_hints.go b/pkg/hints/blake2s_hints.go index d029961e..ec8140c2 100644 --- a/pkg/hints/blake2s_hints.go +++ b/pkg/hints/blake2s_hints.go @@ -202,3 +202,34 @@ func blake2sFinalizeV3(ids IdsManager, vm *VirtualMachine) error { _, err = vm.Segments.LoadData(blake2sPtrEnd, &data) return err } + +func exampleBlake2sCompress(ids IdsManager, vm *VirtualMachine) error { + // Fetch ids variables + blake2sStart, err := ids.GetRelocatable("blake2s_start", vm) + if err != nil { + return err + } + output, err := ids.GetRelocatable("output", vm) + if err != nil { + return err + } + nBytesFelt, err := ids.GetFelt("n_bytes", vm) + if err != nil { + return err + } + nBytes, err := nBytesFelt.ToU32() + if err != nil { + return err + } + // Hint Logic + message, err := getUint32MemoryRange(blake2sStart, 0, 16, &vm.Segments) + if err != nil { + return err + } + modifiedIv := IV() + modifiedIv[0] = modifiedIv[0] ^ 0x01010020 + outputState := Blake2sCompress(modifiedIv, [16]uint32(message), nBytes, 0, 0xffffffff, 0) + outputData := Uint32SliceToMRSlice(outputState) + _, err = vm.Segments.LoadData(output, &outputData) + return err +} diff --git a/pkg/hints/hint_codes/blake2s_hint_codes.go b/pkg/hints/hint_codes/blake2s_hint_codes.go index 766f8b4c..f66b09ca 100644 --- a/pkg/hints/hint_codes/blake2s_hint_codes.go +++ b/pkg/hints/hint_codes/blake2s_hint_codes.go @@ -75,3 +75,19 @@ output = blake2s_compress( ) padding = (message + modified_iv + [0, 0xffffffff] + output) * (_n_packed_instances - 1) segments.write_arg(ids.blake2s_ptr_end, padding)` + +const EXAMPLE_BLAKE2S_COMPRESS = `from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress + +_blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS) +assert 0 <= _blake2s_input_chunk_size_felts < 100 + +new_state = blake2s_compress( + message=memory.get_range(ids.blake2s_start, _blake2s_input_chunk_size_felts), + h=[IV[0] ^ 0x01010020] + IV[1:], + t0=ids.n_bytes, + t1=0, + f0=0xffffffff, + f1=0, +) + +segments.write_arg(ids.output, new_state)` diff --git a/pkg/hints/hint_processor.go b/pkg/hints/hint_processor.go index 89636206..98a7b65b 100644 --- a/pkg/hints/hint_processor.go +++ b/pkg/hints/hint_processor.go @@ -210,6 +210,8 @@ func (p *CairoVmHintProcessor) ExecuteHint(vm *vm.VirtualMachine, hintData *any, return blake2sFinalizeV3(data.Ids, vm) case SHA256_INPUT: return sha256Input(data.Ids, vm) + case EXAMPLE_BLAKE2S_COMPRESS: + return exampleBlake2sCompress(data.Ids, vm) default: return errors.Errorf("Unknown Hint: %s", data.Code) } From 8bcee03c3d0ed3cad20aa44039f5b718b12244de Mon Sep 17 00:00:00 2001 From: Federica Date: Wed, 4 Oct 2023 10:59:31 -0300 Subject: [PATCH 26/31] Add unit test --- pkg/hints/blake2s_hints_test.go | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) diff --git a/pkg/hints/blake2s_hints_test.go b/pkg/hints/blake2s_hints_test.go index 05396010..6628413b 100644 --- a/pkg/hints/blake2s_hints_test.go +++ b/pkg/hints/blake2s_hints_test.go @@ -333,3 +333,27 @@ func TestBlake2sFinaizeV3Ok(t *testing.T) { t.Errorf("Wrong/No data loaded.\n Expected: %v.\n Got: %v", expectedDataSegment, dataSegment) } } + +func TestExampleBlake2sCompressEmptyInput(t *testing.T) { + vm := NewVirtualMachine() + vm.Segments.AddSegment() + output := vm.Segments.AddSegment() + blake2sStart := vm.Segments.AddSegment() + idsManager := SetupIdsForTest( + map[string][]*MaybeRelocatable{ + "output": {NewMaybeRelocatableRelocatable(output)}, + "blake2s_start": {NewMaybeRelocatableRelocatable(blake2sStart)}, + "n_bytes": {NewMaybeRelocatableFelt(FeltOne())}, + }, + vm, + ) + hintProcessor := CairoVmHintProcessor{} + hintData := any(HintData{ + Ids: idsManager, + Code: EXAMPLE_BLAKE2S_COMPRESS, + }) + err := hintProcessor.ExecuteHint(vm, &hintData, nil, nil) + if err == nil { + t.Errorf("EXAMPLE_BLAKE2S_COMPRESS hint test should have failed") + } +} From af1f729795c97bdf020f595798c9636c60982880 Mon Sep 17 00:00:00 2001 From: Federica Date: Thu, 5 Oct 2023 10:29:47 -0300 Subject: [PATCH 27/31] Clone from main branch --- Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Makefile b/Makefile index 30117fef..e7b87b5b 100644 --- a/Makefile +++ b/Makefile @@ -3,9 +3,9 @@ CAIRO_VM_CLI:=cairo-vm/target/release/cairo-vm-cli -# TODO: clone from last release once the fix is merged +# TODO: clone from last release $(CAIRO_VM_CLI): - git clone --depth 1 -b fix-blake-hint https://github.com/lambdaclass/cairo-vm + git clone --depth 1 -b main https://github.com/lambdaclass/cairo-vm cd cairo-vm; cargo b --release --bin cairo-vm-cli # Create proof mode programs. From 206a70b45a8293592785a2a3e6703d7acbf34fd4 Mon Sep 17 00:00:00 2001 From: Federica Date: Thu, 26 Oct 2023 18:15:59 -0300 Subject: [PATCH 28/31] Update cairo version --- Makefile | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/Makefile b/Makefile index e7b87b5b..5805efae 100644 --- a/Makefile +++ b/Makefile @@ -3,9 +3,8 @@ CAIRO_VM_CLI:=cairo-vm/target/release/cairo-vm-cli -# TODO: clone from last release $(CAIRO_VM_CLI): - git clone --depth 1 -b main https://github.com/lambdaclass/cairo-vm + git clone --depth 1 -b v0.9.0 https://github.com/lambdaclass/cairo-vm cd cairo-vm; cargo b --release --bin cairo-vm-cli # Create proof mode programs. From f6927106ef02bc24a9504d403b0322df06cc8638 Mon Sep 17 00:00:00 2001 From: Federica Date: Thu, 26 Oct 2023 18:19:51 -0300 Subject: [PATCH 29/31] fix conflict --- pkg/vm/cairo_run/cairo_run_test.go | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/pkg/vm/cairo_run/cairo_run_test.go b/pkg/vm/cairo_run/cairo_run_test.go index 7cf1557f..5ae9dc3b 100644 --- a/pkg/vm/cairo_run/cairo_run_test.go +++ b/pkg/vm/cairo_run/cairo_run_test.go @@ -357,12 +357,9 @@ func TestKeccakAddUint256(t *testing.T) { testProgram("keccak_add_uint256", t) } -<<<<<<< HEAD -======= func TestReduce(t *testing.T) { testProgram("reduce", t) } ->>>>>>> f08fbfda96b1684c43057448947e444a13b60b55 func TestBlake2sHelloWorldHash(t *testing.T) { testProgram("blake2s_hello_world_hash", t) } @@ -374,11 +371,11 @@ func TestBlake2sFelts(t *testing.T) { func TestFinalizeBlake2s(t *testing.T) { testProgram("finalize_blake2s", t) } -<<<<<<< HEAD func TestBlake2sIntegrationTests(t *testing.T) { testProgram("blake2s_integration_tests", t) -======= +} + func TestUint256Integration(t *testing.T) { testProgram("uint256_integration_tests", t) } @@ -389,5 +386,4 @@ func TestUint256(t *testing.T) { func TestUint256Root(t *testing.T) { testProgram("uint256_root", t) ->>>>>>> f08fbfda96b1684c43057448947e444a13b60b55 } From 85a60e9d6708ae0421657cd39da3003c45d5a321 Mon Sep 17 00:00:00 2001 From: Federica Date: Thu, 26 Oct 2023 18:22:31 -0300 Subject: [PATCH 30/31] fix conflict --- cairo_programs/blake2s_felts.cairo | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/cairo_programs/blake2s_felts.cairo b/cairo_programs/blake2s_felts.cairo index 23047dc8..6212bb93 100644 --- a/cairo_programs/blake2s_felts.cairo +++ b/cairo_programs/blake2s_felts.cairo @@ -1,10 +1,6 @@ %builtins range_check bitwise -<<<<<<< HEAD from starkware.cairo.common.bool import TRUE, FALSE -======= -from starkware.cairo.common.bool import TRUE ->>>>>>> f08fbfda96b1684c43057448947e444a13b60b55 from starkware.cairo.common.alloc import alloc from starkware.cairo.common.cairo_blake2s.blake2s import blake2s_felts from starkware.cairo.common.cairo_builtins import BitwiseBuiltin @@ -30,17 +26,13 @@ func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { assert inputs[15] = 74256930; let (local blake2s_ptr_start) = alloc(); let blake2s_ptr = blake2s_ptr_start; -<<<<<<< HEAD // Bigendian -======= ->>>>>>> f08fbfda96b1684c43057448947e444a13b60b55 let (result) = blake2s_felts{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( 16, inputs, TRUE ); assert result.low = 23022179997536219430502258022509199703; assert result.high = 136831746058902715979837770794974289597; -<<<<<<< HEAD // Little endian let (result) = blake2s_felts{range_check_ptr=range_check_ptr, blake2s_ptr=blake2s_ptr}( @@ -49,7 +41,5 @@ func main{range_check_ptr, bitwise_ptr: BitwiseBuiltin*}() { assert result.low = 315510691254085211243916597439546947220; assert result.high = 42237338665522721102428636006748876126; -======= ->>>>>>> f08fbfda96b1684c43057448947e444a13b60b55 return (); } From f00d78ee807629e42820c4cba01f039683dcbff4 Mon Sep 17 00:00:00 2001 From: Federica Date: Tue, 21 Nov 2023 10:18:08 -0300 Subject: [PATCH 31/31] Update cairo-vm version --- Makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Makefile b/Makefile index 5805efae..e27c83ed 100644 --- a/Makefile +++ b/Makefile @@ -4,7 +4,7 @@ CAIRO_VM_CLI:=cairo-vm/target/release/cairo-vm-cli $(CAIRO_VM_CLI): - git clone --depth 1 -b v0.9.0 https://github.com/lambdaclass/cairo-vm + git clone --depth 1 -b v0.9.1 https://github.com/lambdaclass/cairo-vm cd cairo-vm; cargo b --release --bin cairo-vm-cli # Create proof mode programs.