From 0f6cf13d45ce45a6ee011caa81020d542ff1b833 Mon Sep 17 00:00:00 2001 From: Ivan Kalinin Date: Fri, 24 Jan 2025 15:41:43 +0100 Subject: [PATCH] fix(vm): fix saving `ret` on deep jump (#1487) --- crypto/vm/vm.cpp | 9 +++++++-- crypto/vm/vm.h | 9 +++++++++ 2 files changed, 16 insertions(+), 2 deletions(-) diff --git a/crypto/vm/vm.cpp b/crypto/vm/vm.cpp index 77d5d8f8..3c1118c6 100644 --- a/crypto/vm/vm.cpp +++ b/crypto/vm/vm.cpp @@ -247,6 +247,11 @@ int VmState::jump(Ref cont) { // general jump to continuation cont int VmState::jump(Ref cont, int pass_args) { + cont = adjust_jump_cont(std::move(cont), pass_args); + return jump_to(std::move(cont)); +} + +Ref VmState::adjust_jump_cont(Ref cont, int pass_args) { const ControlData* cont_data = cont->get_cdata(); if (cont_data) { // first do the checks @@ -287,7 +292,7 @@ int VmState::jump(Ref cont, int pass_args) { consume_stack_gas(copy); } } - return jump_to(std::move(cont)); + return cont; } else { // have no continuation data, situation is somewhat simpler if (pass_args >= 0) { @@ -299,7 +304,7 @@ int VmState::jump(Ref cont, int pass_args) { consume_stack_gas(pass_args); } } - return jump_to(std::move(cont)); + return cont; } } diff --git a/crypto/vm/vm.h b/crypto/vm/vm.h index 04c5e576..7aaf1e91 100644 --- a/crypto/vm/vm.h +++ b/crypto/vm/vm.h @@ -347,6 +347,7 @@ class VmState final : public VmStateInterface { int call(Ref cont, int pass_args, int ret_args = -1); int jump(Ref cont); int jump(Ref cont, int pass_args); + Ref adjust_jump_cont(Ref cont, int pass_args); int ret(); int ret(int ret_args); int ret_alt(); @@ -374,6 +375,14 @@ class VmState final : public VmStateInterface { if (cnt > free_nested_cont_jump && global_version >= 9) { consume_gas(1); } + + if (cont.not_null()) { + const ControlData* cont_data = cont->get_cdata(); + if (cont_data && (cont_data->stack.not_null() || cont_data->nargs >= 0)) { + // if cont has non-empty stack or expects fixed number of arguments, jump is not simple + cont = adjust_jump_cont(std::move(cont), -1); + } + } } return res; }