diff --git a/test/contracts/answer-locals/answer-locals.wat b/test/contracts/answer-locals/answer-locals.wat index 22fd8d03f..02ad07671 100644 --- a/test/contracts/answer-locals/answer-locals.wat +++ b/test/contracts/answer-locals/answer-locals.wat @@ -2,12 +2,14 @@ (type (;0;) (func (param i64))) (type (;1;) (func)) (import "env" "int64finish" (func (;0;) (type 0))) - (func (;1;) (type 1) - (local i64 i64 i64 i64 i64) + (func (;1;) (type 1)) + (func (;2;) (type 1) + (local i64 i64 i64 i64) i64.const 42 call 0) (table (;0;) 1 1 funcref) (memory (;0;) 2) (global (;0;) (mut i32) (i32.const 66560)) (export "memory" (memory 0)) - (export "answer" (func 1))) + (export "init" (func 1)) + (export "answer" (func 2))) diff --git a/test/contracts/answer-locals/output/answer-locals.wasm b/test/contracts/answer-locals/output/answer-locals.wasm index fe42715f7..b6b8d92dc 100644 Binary files a/test/contracts/answer-locals/output/answer-locals.wasm and b/test/contracts/answer-locals/output/answer-locals.wasm differ diff --git a/test/contracts/answer/answer.c b/test/contracts/answer/answer.c index e30b5a5c9..dc1a89135 100644 --- a/test/contracts/answer/answer.c +++ b/test/contracts/answer/answer.c @@ -1,5 +1,7 @@ #include "../mxvm/context.h" +void init() {} + void answer() { int64finish(42); } diff --git a/test/contracts/answer/answer.export b/test/contracts/answer/answer.export index fe49af451..11ab8bfb4 100644 --- a/test/contracts/answer/answer.export +++ b/test/contracts/answer/answer.export @@ -1 +1,2 @@ answer +init diff --git a/test/contracts/bad-extra/output/bad-extra.wasm b/test/contracts/bad-extra/output/bad-extra.wasm new file mode 100644 index 000000000..b23213faa Binary files /dev/null and b/test/contracts/bad-extra/output/bad-extra.wasm differ diff --git a/test/contracts/bad-recursive/bad-recursive.c b/test/contracts/bad-recursive/bad-recursive.c new file mode 100644 index 000000000..1d40cef48 --- /dev/null +++ b/test/contracts/bad-recursive/bad-recursive.c @@ -0,0 +1,30 @@ +typedef unsigned char byte; +typedef unsigned int i32; +typedef unsigned long long i64; +typedef unsigned int bigInt; + +bigInt bigIntNew(long long value); +void bigIntFinishUnsigned(bigInt reference); + +void init() +{ +} + +i64 doStackoverflow(i64 a) +{ + if (a % 2 == 0) + { + return 42; + } + + i64 x = doStackoverflow(a * 8 + 1); + i64 y = doStackoverflow(a * 2 + 1); + return x + y + a; +} + +void badRecursive() +{ + i64 result = doStackoverflow(1); + bigInt resultBig = bigIntNew(result); + bigIntFinishUnsigned(resultBig); +} diff --git a/test/contracts/bad-recursive/bad-recursive.export b/test/contracts/bad-recursive/bad-recursive.export new file mode 100644 index 000000000..01ce9bc04 --- /dev/null +++ b/test/contracts/bad-recursive/bad-recursive.export @@ -0,0 +1,2 @@ +init +badRecursive diff --git a/test/contracts/bad-recursive/output/bad-recursive.wasm b/test/contracts/bad-recursive/output/bad-recursive.wasm new file mode 100755 index 000000000..32b6eaa74 Binary files /dev/null and b/test/contracts/bad-recursive/output/bad-recursive.wasm differ diff --git a/test/contracts/counter/counter.c b/test/contracts/counter/counter.c index f0c4230fc..91dbac918 100644 --- a/test/contracts/counter/counter.c +++ b/test/contracts/counter/counter.c @@ -7,6 +7,10 @@ void init() { int64storageStore(COUNTER_KEY, COUNTER_KEY_LEN, 1); } +void upgrade() { + int64storageStore(COUNTER_KEY, COUNTER_KEY_LEN, 1); +} + void increment() { i64 counter = int64storageLoad(COUNTER_KEY, COUNTER_KEY_LEN); counter++; diff --git a/test/contracts/counter/counter.export b/test/contracts/counter/counter.export index 91723bf1b..82ad4d916 100644 --- a/test/contracts/counter/counter.export +++ b/test/contracts/counter/counter.export @@ -1,4 +1,5 @@ init +upgrade increment decrement get diff --git a/test/contracts/counter/output/counter.wasm b/test/contracts/counter/output/counter.wasm old mode 100755 new mode 100644 index 4f31dbb4b..97d854c17 Binary files a/test/contracts/counter/output/counter.wasm and b/test/contracts/counter/output/counter.wasm differ diff --git a/test/contracts/counter/output/counter.wat b/test/contracts/counter/output/counter.wat index 50471c3bd..4403b2732 100644 --- a/test/contracts/counter/output/counter.wat +++ b/test/contracts/counter/output/counter.wat @@ -1,56 +1,197 @@ (module - (type $t0 (func (param i32 i32 i64) (result i32))) - (type $t1 (func (param i32 i32) (result i64))) - (type $t2 (func (param i64))) - (type $t3 (func)) - (import "env" "int64storageStore" (func $env.int64storageStore (type $t0))) - (import "env" "int64storageLoad" (func $env.int64storageLoad (type $t1))) - (import "env" "int64finish" (func $env.int64finish (type $t2))) - (func $init (type $t3) + (type (;0;) (func (param i32 i32 i64) (result i32))) + (type (;1;) (func (param i32 i32) (result i64))) + (type (;2;) (func (param i64))) + (type (;3;) (func)) + (import "env" "int64storageStore" (func (;0;) (type 0))) + (import "env" "int64storageLoad" (func (;1;) (type 1))) + (import "env" "int64finish" (func (;2;) (type 2))) + (func (;3;) (type 3) + (local i32 i32 i64) i32.const 1024 + local.set 0 i32.const 7 + local.set 1 i64.const 1 - call $env.int64storageStore - drop) - (func $increment (type $t3) - (local $l0 i64) + local.set 2 + local.get 0 + local.get 1 + local.get 2 + call 0 + drop + return) + (func (;4;) (type 3) + (local i32 i32 i64) i32.const 1024 + local.set 0 i32.const 7 + local.set 1 + i64.const 1 + local.set 2 + local.get 0 + local.get 1 + local.get 2 + call 0 + drop + return) + (func (;5;) (type 3) + (local i32 i32 i32 i32 i32 i64 i64 i64 i64 i64 i64 i32 i32) + global.get 0 + local.set 0 + i32.const 16 + local.set 1 + local.get 0 + local.get 1 + i32.sub + local.set 2 + local.get 2 + global.set 0 i32.const 1024 + local.set 3 i32.const 7 - call $env.int64storageLoad + local.set 4 + local.get 3 + local.get 4 + call 1 + local.set 5 + local.get 2 + local.get 5 + i64.store offset=8 + local.get 2 + i64.load offset=8 + local.set 6 i64.const 1 + local.set 7 + local.get 6 + local.get 7 i64.add - local.tee $l0 - call $env.int64storageStore + local.set 8 + local.get 2 + local.get 8 + i64.store offset=8 + local.get 2 + i64.load offset=8 + local.set 9 + local.get 3 + local.get 4 + local.get 9 + call 0 drop - local.get $l0 - call $env.int64finish) - (func $decrement (type $t3) - (local $l0 i64) - i32.const 1024 - i32.const 7 + local.get 2 + i64.load offset=8 + local.set 10 + local.get 10 + call 2 + i32.const 16 + local.set 11 + local.get 2 + local.get 11 + i32.add + local.set 12 + local.get 12 + global.set 0 + return) + (func (;6;) (type 3) + (local i32 i32 i32 i32 i32 i64 i64 i64 i64 i64 i64 i32 i32) + global.get 0 + local.set 0 + i32.const 16 + local.set 1 + local.get 0 + local.get 1 + i32.sub + local.set 2 + local.get 2 + global.set 0 i32.const 1024 + local.set 3 i32.const 7 - call $env.int64storageLoad + local.set 4 + local.get 3 + local.get 4 + call 1 + local.set 5 + local.get 2 + local.get 5 + i64.store offset=8 + local.get 2 + i64.load offset=8 + local.set 6 i64.const -1 + local.set 7 + local.get 6 + local.get 7 i64.add - local.tee $l0 - call $env.int64storageStore + local.set 8 + local.get 2 + local.get 8 + i64.store offset=8 + local.get 2 + i64.load offset=8 + local.set 9 + local.get 3 + local.get 4 + local.get 9 + call 0 drop - local.get $l0 - call $env.int64finish) - (func $get (type $t3) + local.get 2 + i64.load offset=8 + local.set 10 + local.get 10 + call 2 + i32.const 16 + local.set 11 + local.get 2 + local.get 11 + i32.add + local.set 12 + local.get 12 + global.set 0 + return) + (func (;7;) (type 3) + (local i32 i32 i32 i32 i32 i64 i64 i32 i32) + global.get 0 + local.set 0 + i32.const 16 + local.set 1 + local.get 0 + local.get 1 + i32.sub + local.set 2 + local.get 2 + global.set 0 i32.const 1024 + local.set 3 i32.const 7 - call $env.int64storageLoad - call $env.int64finish) - (table $T0 1 1 funcref) - (memory $memory 2) - (global $g0 (mut i32) (i32.const 66576)) + local.set 4 + local.get 3 + local.get 4 + call 1 + local.set 5 + local.get 2 + local.get 5 + i64.store offset=8 + local.get 2 + i64.load offset=8 + local.set 6 + local.get 6 + call 2 + i32.const 16 + local.set 7 + local.get 2 + local.get 7 + i32.add + local.set 8 + local.get 8 + global.set 0 + return) + (table (;0;) 1 1 funcref) + (memory (;0;) 2) + (global (;0;) (mut i32) (i32.const 66576)) (export "memory" (memory 0)) - (export "init" (func $init)) - (export "increment" (func $increment)) - (export "decrement" (func $decrement)) - (export "get" (func $get)) - (data $d0 (i32.const 1024) "COUNTER\00")) + (export "init" (func 3)) + (export "upgrade" (func 4)) + (export "increment" (func 5)) + (export "decrement" (func 6)) + (export "get" (func 7)) + (data (;0;) (i32.const 1024) "COUNTER\00")) diff --git a/vmhost/hostCore/execution.go b/vmhost/hostCore/execution.go index 51c95912d..64aded75b 100644 --- a/vmhost/hostCore/execution.go +++ b/vmhost/hostCore/execution.go @@ -1150,7 +1150,7 @@ func (host *vmHost) callUpgradeFunction() error { func (host *vmHost) callSCFunction(functionName string) error { runtime := host.Runtime() if !runtime.HasFunction(functionName) { - return nil + return executor.ErrFuncNotFound } err := runtime.CallSCFunction(functionName) diff --git a/vmhost/hosttest/bad_test.go b/vmhost/hosttest/bad_test.go index 6abc63cfe..a4cad194d 100644 --- a/vmhost/hosttest/bad_test.go +++ b/vmhost/hosttest/bad_test.go @@ -1,12 +1,17 @@ package hostCoretest import ( + "context" "testing" + "time" "github.com/multiversx/mx-chain-vm-go/executor" contextmock "github.com/multiversx/mx-chain-vm-go/mock/context" test "github.com/multiversx/mx-chain-vm-go/testcommon" "github.com/multiversx/mx-chain-vm-go/vmhost" + "github.com/multiversx/mx-chain-vm-go/wasmer" + "github.com/multiversx/mx-chain-vm-go/wasmer2" + "github.com/stretchr/testify/require" ) func TestBadContract_NoPanic_Memoryfault(t *testing.T) { @@ -251,3 +256,73 @@ func TestBadContract_NoPanic_NonExistingFunction(t *testing.T) { HasRuntimeErrorAndInfo(executor.ErrInvalidFunction.Error(), "thisDoesNotExist") }) } + +func TestBadContractExtra_LongIntLoop_Wasmer1(t *testing.T) { + testBadContractExtraLongIntLoop(t, wasmer.ExecutorFactory()) +} + +func TestBadContractExtra_LongIntLoop_Wasmer2(t *testing.T) { + testBadContractExtraLongIntLoop(t, wasmer2.ExecutorFactory()) +} + +func testBadContractExtraLongIntLoop(t *testing.T, executorFactory executor.ExecutorAbstractFactory) { + testCase := test.BuildInstanceCallTest(t).WithContracts( + test.CreateInstanceContract(test.ParentAddress). + WithCode(test.GetTestSCCode("bad-extra", "../../"))). + WithInput(test.CreateTestContractCallInputBuilder(). + WithRecipientAddr(test.ParentAddress). + WithGasProvided(test.GasProvided). + WithFunction("bigLoop"). + Build()). + WithExecutorFactory(executorFactory). + WithWasmerSIGSEGVPassthrough(false) + + ctx, cancel := context.WithTimeout(context.Background(), time.Second) + defer cancel() + + done := make(chan struct{}) + go func() { + testCase.AndAssertResults(func(_ vmhost.VMHost, _ *contextmock.BlockchainHookStub, verify *test.VMOutputVerifier) { + verify.OutOfGas() + }) + close(done) + }() + + select { + case <-done: + return + case <-ctx.Done(): + require.FailNow(t, "test timed out") + } +} + +func TestBadContractExtra_NoPanic_BadRecursive(t *testing.T) { + if testing.Short() { + t.Skip("not a short test") + } + + testCase := test.BuildInstanceCallTest(t). + WithWasmerSIGSEGVPassthrough(false). + WithContracts( + test.CreateInstanceContract(test.ParentAddress). + WithCode(test.GetTestSCCode("bad-recursive", "../../")). + WithBalance(1000)). + WithExecutorFactory(wasmer2.ExecutorFactory()) + + input := test.CreateTestContractCallInputBuilder(). + WithRecipientAddr(test.ParentAddress). + WithGasProvided(10000000). + WithFunction("badRecursive"). + Build() + + repetitions := 25_000 + + for i := 0; i < repetitions; i++ { + testCase. + WithInput(input). + AndAssertResultsWithoutReset(func(host vmhost.VMHost, stubBlockchainHook *contextmock.BlockchainHookStub, verify *test.VMOutputVerifier) { + verify.ReturnMessage("execution failed") + verify.ExecutionFailed() + }) + } +} diff --git a/vmhost/hosttest/execution_test.go b/vmhost/hosttest/execution_test.go index 4db839d08..fc74f012f 100644 --- a/vmhost/hosttest/execution_test.go +++ b/vmhost/hosttest/execution_test.go @@ -210,7 +210,7 @@ func TestExecution_DeployWASM_Popcnt(t *testing.T) { func TestExecution_DeployWASM_AtMaximumLocals(t *testing.T) { test.BuildInstanceCreatorTest(t). WithInput(test.CreateTestContractCreateInputBuilder(). - WithGasProvided(1000). + WithGasProvided(100000). WithCallValue(88). WithContractCode(makeBytecodeWithLocals(WASMLocalsLimit)). Build()). @@ -3687,8 +3687,9 @@ func TestExecution_Mocked_OnSameFollowedByOnDest(t *testing.T) { // number of i64 locals it instantiates func makeBytecodeWithLocals(numLocals uint64) []byte { originalCode := test.GetTestSCCode("answer-locals", "../../") - firstSlice := originalCode[:0x5B] - secondSlice := originalCode[0x5C:] + + firstSlice := originalCode[:0x66] + secondSlice := originalCode[0x67:] encodedNumLocals := vmhost.U64ToLEB128(numLocals) extraBytes := len(encodedNumLocals) - 1 @@ -3698,8 +3699,8 @@ func makeBytecodeWithLocals(numLocals uint64) []byte { result = append(result, encodedNumLocals...) result = append(result, secondSlice...) - result[0x57] = byte(int(result[0x57]) + extraBytes) - result[0x59] = byte(int(result[0x59]) + extraBytes) + result[0x5F] = byte(int(result[0x5F]) + extraBytes) + result[0x64] = byte(int(result[0x64]) + extraBytes) return result } diff --git a/wasmer/libwasmer_darwin_amd64.dylib b/wasmer/libwasmer_darwin_amd64.dylib index 9701b3da1..6c89493cc 100755 Binary files a/wasmer/libwasmer_darwin_amd64.dylib and b/wasmer/libwasmer_darwin_amd64.dylib differ diff --git a/wasmer2/libvmexeccapi.dylib b/wasmer2/libvmexeccapi.dylib index 27f527009..c064ffdf2 100644 Binary files a/wasmer2/libvmexeccapi.dylib and b/wasmer2/libvmexeccapi.dylib differ diff --git a/wasmer2/libvmexeccapi.h b/wasmer2/libvmexeccapi.h index f4a0da9a2..c7b77ad8f 100644 --- a/wasmer2/libvmexeccapi.h +++ b/wasmer2/libvmexeccapi.h @@ -317,7 +317,7 @@ vm_exec_result_t vm_check_signatures(vm_exec_instance_t *instance_ptr); * * C API function, works with raw object pointers. */ -void vm_exec_executor_destroy(vm_exec_executor_t *executor); +void vm_exec_executor_destroy(vm_exec_executor_t *executor_ptr); /** * Sets the data that can be hold by an instance context. @@ -380,7 +380,7 @@ vm_exec_result_t vm_exec_instance_call(vm_exec_instance_t *instance_ptr, const c * * C API function, works with raw object pointers. */ -void vm_exec_instance_destroy(vm_exec_instance_t *instance); +void vm_exec_instance_destroy(vm_exec_instance_t *instance_ptr); /** * Creates a new VM executor instance from cache. diff --git a/wasmer2/libvmexeccapi.so b/wasmer2/libvmexeccapi.so index 7788f9de4..4f6ac7d96 100644 Binary files a/wasmer2/libvmexeccapi.so and b/wasmer2/libvmexeccapi.so differ