2019-01-28 13:59:04 +00:00
|
|
|
// Copyright 2019 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2021-08-23 13:01:06 +00:00
|
|
|
#include "include/v8-function.h"
|
2019-05-17 12:13:44 +00:00
|
|
|
#include "src/api/api-inl.h"
|
2019-01-28 13:59:04 +00:00
|
|
|
#include "test/cctest/wasm/wasm-atomics-utils.h"
|
|
|
|
#include "test/common/wasm/test-signatures.h"
|
|
|
|
#include "test/common/wasm/wasm-macro-gen.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace wasm {
|
|
|
|
namespace test_run_wasm_exceptions {
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchThrow) {
|
2019-01-28 13:59:04 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2019-01-28 13:59:04 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
2019-01-30 11:38:56 +00:00
|
|
|
// Build the main test function.
|
2019-01-28 13:59:04 +00:00
|
|
|
BUILD(r, WASM_TRY_CATCH_T(kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
2020-12-17 16:55:33 +00:00
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
2019-01-28 13:59:04 +00:00
|
|
|
WASM_THROW(except))),
|
2020-12-03 18:00:05 +00:00
|
|
|
WASM_STMTS(WASM_I32V(kResult0)), except));
|
2019-01-28 13:59:04 +00:00
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WASM_EXEC_TEST(TryCatchThrowWithValue) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_i());
|
2021-02-11 13:37:03 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r, WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_I32V(kResult0), WASM_THROW(except))),
|
|
|
|
WASM_STMTS(kExprNop), except));
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
2019-01-28 13:59:04 +00:00
|
|
|
}
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryMultiCatchThrow) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except1 = r.builder().AddException(sigs.v_v());
|
|
|
|
byte except2 = r.builder().AddException(sigs.v_v());
|
2021-02-11 13:37:03 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
constexpr uint32_t kResult2 = 51;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(
|
|
|
|
r, kExprTry, static_cast<byte>((kWasmI32).value_type_code()),
|
|
|
|
WASM_STMTS(WASM_I32V(kResult2),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)), WASM_THROW(except1)),
|
|
|
|
WASM_IF(WASM_I32_EQ(WASM_LOCAL_GET(0), WASM_I32V(1)),
|
|
|
|
WASM_THROW(except2))),
|
|
|
|
kExprCatch, except1, WASM_STMTS(WASM_I32V(kResult0)), kExprCatch, except2,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1)), kExprEnd);
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
r.CheckCallViaJS(kResult2, 2);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
2021-02-12 10:35:40 +00:00
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
CHECK_EQ(kResult2, r.CallInterpreter(2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WASM_EXEC_TEST(TryCatchAllThrow) {
|
2021-03-04 09:57:01 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2021-03-04 09:57:01 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r, kExprTry, static_cast<byte>((kWasmI32).value_type_code()),
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1), WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_THROW(except))),
|
|
|
|
kExprCatchAll, WASM_I32V(kResult0), kExprEnd);
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WASM_EXEC_TEST(TryCatchCatchAllThrow) {
|
2021-02-12 10:35:40 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except1 = r.builder().AddException(sigs.v_v());
|
|
|
|
byte except2 = r.builder().AddException(sigs.v_v());
|
2021-02-12 10:35:40 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
constexpr uint32_t kResult2 = 51;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(
|
|
|
|
r, kExprTry, static_cast<byte>((kWasmI32).value_type_code()),
|
|
|
|
WASM_STMTS(WASM_I32V(kResult2),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)), WASM_THROW(except1)),
|
|
|
|
WASM_IF(WASM_I32_EQ(WASM_LOCAL_GET(0), WASM_I32V(1)),
|
|
|
|
WASM_THROW(except2))),
|
2021-03-04 09:57:01 +00:00
|
|
|
kExprCatch, except1, WASM_I32V(kResult0), kExprCatchAll,
|
|
|
|
WASM_I32V(kResult1), kExprEnd);
|
2021-02-12 10:35:40 +00:00
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
r.CheckCallViaJS(kResult2, 2);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
2021-02-11 13:37:03 +00:00
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
CHECK_EQ(kResult2, r.CallInterpreter(2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-12 15:35:46 +00:00
|
|
|
WASM_EXEC_TEST(TryImplicitRethrow) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except1 = r.builder().AddException(sigs.v_v());
|
|
|
|
byte except2 = r.builder().AddException(sigs.v_v());
|
2021-02-12 15:35:46 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
constexpr uint32_t kResult2 = 51;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r,
|
|
|
|
WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_TRY_CATCH_T(kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_THROW(except2))),
|
|
|
|
WASM_STMTS(WASM_I32V(kResult2)), except1),
|
|
|
|
WASM_I32V(kResult0), except2));
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-15 11:04:05 +00:00
|
|
|
WASM_EXEC_TEST(TryDelegate) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2021-02-15 11:04:05 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r,
|
|
|
|
WASM_TRY_CATCH_T(kWasmI32,
|
|
|
|
WASM_TRY_DELEGATE_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_THROW(except))),
|
|
|
|
0),
|
|
|
|
WASM_I32V(kResult0), except));
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
2021-02-16 12:00:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-14 10:46:53 +00:00
|
|
|
WASM_EXEC_TEST(TestCatchlessTry) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t> r(execution_tier);
|
2021-06-14 13:33:46 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_i());
|
2021-06-14 10:46:53 +00:00
|
|
|
BUILD(r,
|
|
|
|
WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_TRY_T(kWasmI32, WASM_STMTS(WASM_I32V(0), WASM_THROW(except))),
|
|
|
|
WASM_NOP, except));
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
r.CheckCallViaJS(0);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(0, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:00:13 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchRethrow) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except1 = r.builder().AddException(sigs.v_v());
|
|
|
|
byte except2 = r.builder().AddException(sigs.v_v());
|
2021-02-16 12:00:13 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
constexpr uint32_t kUnreachable = 51;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r,
|
|
|
|
WASM_TRY_CATCH_CATCH_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32, WASM_THROW(except2),
|
|
|
|
WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32, WASM_THROW(except1),
|
|
|
|
WASM_STMTS(WASM_I32V(kUnreachable),
|
|
|
|
WASM_IF_ELSE(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_RETHROW(1), WASM_RETHROW(2))),
|
|
|
|
except1),
|
|
|
|
except2),
|
|
|
|
except1, WASM_I32V(kResult0), except2, WASM_I32V(kResult1)));
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
2021-02-15 11:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WASM_EXEC_TEST(TryDelegateToCaller) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2021-02-15 11:04:05 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r,
|
|
|
|
WASM_TRY_CATCH_T(kWasmI32,
|
|
|
|
WASM_TRY_DELEGATE_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_THROW(except))),
|
|
|
|
1),
|
|
|
|
WASM_I32V(kResult0), except));
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
constexpr int64_t trap = 0xDEADBEEF;
|
|
|
|
r.CheckCallViaJS(trap, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
constexpr int stopped = 0;
|
|
|
|
CHECK_EQ(stopped, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchCallDirect) {
|
2019-01-30 11:38:56 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2019-01-30 11:38:56 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build a throwing helper function.
|
2019-01-30 15:06:42 +00:00
|
|
|
WasmFunctionCompiler& throw_func = r.NewFunction(sigs.i_ii());
|
2019-01-30 11:38:56 +00:00
|
|
|
BUILD(throw_func, WASM_THROW(except));
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r, WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32,
|
2019-01-30 15:06:42 +00:00
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
2020-12-17 16:55:33 +00:00
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
2019-01-30 15:06:42 +00:00
|
|
|
WASM_STMTS(WASM_CALL_FUNCTION(
|
|
|
|
throw_func.function_index(),
|
|
|
|
WASM_I32V(7), WASM_I32V(9)),
|
|
|
|
WASM_DROP))),
|
2020-12-03 18:00:05 +00:00
|
|
|
WASM_STMTS(WASM_I32V(kResult0)), except));
|
2019-01-30 11:38:56 +00:00
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
2019-01-30 11:38:56 +00:00
|
|
|
}
|
|
|
|
|
2021-03-04 09:57:01 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchAllCallDirect) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2021-03-04 09:57:01 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build a throwing helper function.
|
|
|
|
WasmFunctionCompiler& throw_func = r.NewFunction(sigs.i_ii());
|
|
|
|
BUILD(throw_func, WASM_THROW(except));
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_STMTS(WASM_CALL_FUNCTION(
|
|
|
|
throw_func.function_index(),
|
|
|
|
WASM_I32V(7), WASM_I32V(9)),
|
|
|
|
WASM_DROP))),
|
|
|
|
WASM_STMTS(WASM_I32V(kResult0))));
|
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchCallIndirect) {
|
2019-01-30 11:38:56 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2019-01-30 11:38:56 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build a throwing helper function.
|
2019-01-30 15:06:42 +00:00
|
|
|
WasmFunctionCompiler& throw_func = r.NewFunction(sigs.i_ii());
|
2019-01-30 11:38:56 +00:00
|
|
|
BUILD(throw_func, WASM_THROW(except));
|
|
|
|
|
|
|
|
// Add an indirect function table.
|
|
|
|
uint16_t indirect_function_table[] = {
|
|
|
|
static_cast<uint16_t>(throw_func.function_index())};
|
|
|
|
r.builder().AddIndirectFunctionTable(indirect_function_table,
|
|
|
|
arraysize(indirect_function_table));
|
|
|
|
|
|
|
|
// Build the main test function.
|
2022-07-29 04:29:12 +00:00
|
|
|
BUILD(r, WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(
|
|
|
|
WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_STMTS(WASM_CALL_INDIRECT(
|
|
|
|
throw_func.sig_index(), WASM_I32V(7),
|
|
|
|
WASM_I32V(9), WASM_LOCAL_GET(0)),
|
|
|
|
WASM_DROP))),
|
|
|
|
WASM_I32V(kResult0), except));
|
2021-03-04 09:57:01 +00:00
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WASM_EXEC_TEST(TryCatchAllCallIndirect) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2021-03-04 09:57:01 +00:00
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
|
|
|
|
// Build a throwing helper function.
|
|
|
|
WasmFunctionCompiler& throw_func = r.NewFunction(sigs.i_ii());
|
|
|
|
BUILD(throw_func, WASM_THROW(except));
|
|
|
|
|
|
|
|
// Add an indirect function table.
|
|
|
|
uint16_t indirect_function_table[] = {
|
|
|
|
static_cast<uint16_t>(throw_func.function_index())};
|
|
|
|
r.builder().AddIndirectFunctionTable(indirect_function_table,
|
|
|
|
arraysize(indirect_function_table));
|
|
|
|
|
|
|
|
// Build the main test function.
|
2022-07-29 04:29:12 +00:00
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(
|
|
|
|
WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_STMTS(WASM_CALL_INDIRECT(
|
|
|
|
throw_func.sig_index(), WASM_I32V(7),
|
|
|
|
WASM_I32V(9), WASM_LOCAL_GET(0)),
|
|
|
|
WASM_DROP))),
|
|
|
|
WASM_I32V(kResult0)));
|
2019-01-30 11:38:56 +00:00
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
} else {
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter(0));
|
|
|
|
CHECK_EQ(kResult1, r.CallInterpreter(1));
|
|
|
|
}
|
2019-01-30 11:38:56 +00:00
|
|
|
}
|
|
|
|
|
2020-05-05 10:23:41 +00:00
|
|
|
WASM_COMPILED_EXEC_TEST(TryCatchCallExternal) {
|
2019-01-30 11:38:56 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
HandleScope scope(CcTest::InitIsolateOnce());
|
|
|
|
const char* source = "(function() { throw 'ball'; })";
|
|
|
|
Handle<JSFunction> js_function =
|
|
|
|
Handle<JSFunction>::cast(v8::Utils::OpenHandle(
|
|
|
|
*v8::Local<v8::Function>::Cast(CompileRun(source))));
|
2019-01-30 15:06:42 +00:00
|
|
|
ManuallyImportedJSFunction import = {sigs.i_ii(), js_function};
|
2019-01-30 11:38:56 +00:00
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier, &import);
|
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
constexpr uint32_t kJSFunc = 0;
|
|
|
|
|
|
|
|
// Build the main test function.
|
2020-12-03 18:00:05 +00:00
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
2019-01-30 15:06:42 +00:00
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(
|
|
|
|
WASM_I32V(kResult1),
|
2020-12-17 16:55:33 +00:00
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
2019-01-30 15:06:42 +00:00
|
|
|
WASM_STMTS(WASM_CALL_FUNCTION(kJSFunc, WASM_I32V(7),
|
|
|
|
WASM_I32V(9)),
|
|
|
|
WASM_DROP))),
|
2021-03-04 09:57:01 +00:00
|
|
|
WASM_I32V(kResult0)));
|
|
|
|
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
WASM_COMPILED_EXEC_TEST(TryCatchAllCallExternal) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
HandleScope scope(CcTest::InitIsolateOnce());
|
|
|
|
const char* source = "(function() { throw 'ball'; })";
|
|
|
|
Handle<JSFunction> js_function =
|
|
|
|
Handle<JSFunction>::cast(v8::Utils::OpenHandle(
|
|
|
|
*v8::Local<v8::Function>::Cast(CompileRun(source))));
|
|
|
|
ManuallyImportedJSFunction import = {sigs.i_ii(), js_function};
|
|
|
|
WasmRunner<uint32_t, uint32_t> r(execution_tier, &import);
|
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kResult1 = 42;
|
|
|
|
constexpr uint32_t kJSFunc = 0;
|
|
|
|
|
|
|
|
// Build the main test function.
|
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(
|
|
|
|
WASM_I32V(kResult1),
|
|
|
|
WASM_IF(WASM_I32_EQZ(WASM_LOCAL_GET(0)),
|
|
|
|
WASM_STMTS(WASM_CALL_FUNCTION(kJSFunc, WASM_I32V(7),
|
|
|
|
WASM_I32V(9)),
|
|
|
|
WASM_DROP))),
|
|
|
|
WASM_I32V(kResult0)));
|
2019-01-31 13:35:57 +00:00
|
|
|
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJS(kResult0, 0);
|
|
|
|
r.CheckCallViaJS(kResult1, 1);
|
|
|
|
}
|
|
|
|
|
2019-02-19 14:54:45 +00:00
|
|
|
namespace {
|
|
|
|
|
2020-03-23 12:56:45 +00:00
|
|
|
void TestTrapNotCaught(byte* code, size_t code_size,
|
2020-08-04 11:09:23 +00:00
|
|
|
TestExecutionTier execution_tier) {
|
2019-02-07 10:02:06 +00:00
|
|
|
TestSignatures sigs;
|
2020-03-23 12:56:45 +00:00
|
|
|
WasmRunner<uint32_t> r(execution_tier, nullptr, "main",
|
|
|
|
kRuntimeExceptionSupport);
|
2019-02-19 14:54:45 +00:00
|
|
|
r.builder().AddMemory(kWasmPageSize);
|
2020-03-23 12:56:45 +00:00
|
|
|
constexpr uint32_t kResultSuccess = 23;
|
|
|
|
constexpr uint32_t kResultCaught = 47;
|
2019-02-07 10:02:06 +00:00
|
|
|
|
2021-06-10 14:40:32 +00:00
|
|
|
// Add an indirect function table.
|
|
|
|
const int kTableSize = 2;
|
|
|
|
r.builder().AddIndirectFunctionTable(nullptr, kTableSize);
|
|
|
|
|
2019-02-07 10:02:06 +00:00
|
|
|
// Build a trapping helper function.
|
|
|
|
WasmFunctionCompiler& trap_func = r.NewFunction(sigs.i_ii());
|
2019-02-19 14:54:45 +00:00
|
|
|
trap_func.Build(code, code + code_size);
|
2019-02-07 10:02:06 +00:00
|
|
|
|
|
|
|
// Build the main test function.
|
2020-12-03 18:00:05 +00:00
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
2019-02-07 10:02:06 +00:00
|
|
|
kWasmI32,
|
2020-03-23 12:56:45 +00:00
|
|
|
WASM_STMTS(WASM_I32V(kResultSuccess),
|
|
|
|
WASM_CALL_FUNCTION(trap_func.function_index(),
|
|
|
|
WASM_I32V(7), WASM_I32V(9)),
|
|
|
|
WASM_DROP),
|
2020-12-03 18:00:05 +00:00
|
|
|
WASM_STMTS(WASM_I32V(kResultCaught))));
|
2019-02-07 10:02:06 +00:00
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJSTraps();
|
|
|
|
} else {
|
|
|
|
r.CallInterpreter();
|
|
|
|
}
|
2019-02-07 10:02:06 +00:00
|
|
|
}
|
|
|
|
|
2019-02-19 14:54:45 +00:00
|
|
|
} // namespace
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchTrapUnreachable) {
|
2019-02-19 14:54:45 +00:00
|
|
|
byte code[] = {WASM_UNREACHABLE};
|
2020-03-23 12:56:45 +00:00
|
|
|
TestTrapNotCaught(code, arraysize(code), execution_tier);
|
2019-02-19 14:54:45 +00:00
|
|
|
}
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchTrapMemOutOfBounds) {
|
2019-02-19 14:54:45 +00:00
|
|
|
byte code[] = {WASM_LOAD_MEM(MachineType::Int32(), WASM_I32V_1(-1))};
|
2020-03-23 12:56:45 +00:00
|
|
|
TestTrapNotCaught(code, arraysize(code), execution_tier);
|
2019-02-19 14:54:45 +00:00
|
|
|
}
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchTrapDivByZero) {
|
2020-12-17 16:55:33 +00:00
|
|
|
byte code[] = {WASM_I32_DIVS(WASM_LOCAL_GET(0), WASM_I32V_1(0))};
|
2020-03-23 12:56:45 +00:00
|
|
|
TestTrapNotCaught(code, arraysize(code), execution_tier);
|
2019-02-19 14:54:45 +00:00
|
|
|
}
|
|
|
|
|
2021-02-11 13:37:03 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchTrapRemByZero) {
|
2020-12-17 16:55:33 +00:00
|
|
|
byte code[] = {WASM_I32_REMS(WASM_LOCAL_GET(0), WASM_I32V_1(0))};
|
2020-03-23 12:56:45 +00:00
|
|
|
TestTrapNotCaught(code, arraysize(code), execution_tier);
|
2019-02-19 14:54:45 +00:00
|
|
|
}
|
|
|
|
|
2021-06-10 14:40:32 +00:00
|
|
|
WASM_EXEC_TEST(TryCatchTrapTableFill) {
|
|
|
|
int table_index = 0;
|
|
|
|
int length = 10; // OOB.
|
|
|
|
int start = 10; // OOB.
|
|
|
|
byte code[] = {WASM_TABLE_FILL(table_index, WASM_I32V(length),
|
|
|
|
WASM_REF_NULL(kFuncRefCode), WASM_I32V(start)),
|
|
|
|
WASM_I32V_1(42)};
|
|
|
|
TestTrapNotCaught(code, arraysize(code), execution_tier);
|
|
|
|
}
|
|
|
|
|
2021-05-03 14:29:06 +00:00
|
|
|
namespace {
|
|
|
|
// TODO(cleanup): Define in cctest.h and re-use where appropriate.
|
|
|
|
class IsolateScope {
|
|
|
|
public:
|
|
|
|
IsolateScope() {
|
|
|
|
v8::Isolate::CreateParams create_params;
|
|
|
|
create_params.array_buffer_allocator = CcTest::array_buffer_allocator();
|
|
|
|
isolate_ = v8::Isolate::New(create_params);
|
|
|
|
isolate_->Enter();
|
|
|
|
}
|
|
|
|
|
|
|
|
~IsolateScope() {
|
|
|
|
isolate_->Exit();
|
|
|
|
isolate_->Dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
v8::Isolate* isolate() { return isolate_; }
|
|
|
|
Isolate* i_isolate() { return reinterpret_cast<Isolate*>(isolate_); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
v8::Isolate* isolate_;
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
UNINITIALIZED_WASM_EXEC_TEST(TestStackOverflowNotCaught) {
|
|
|
|
TestSignatures sigs;
|
2022-08-29 10:33:33 +00:00
|
|
|
// v8_flags.stack_size must be set before isolate initialization.
|
2022-09-15 16:54:55 +00:00
|
|
|
FlagScope<int32_t> stack_size(&v8_flags.stack_size, 8);
|
2021-05-03 14:29:06 +00:00
|
|
|
|
|
|
|
IsolateScope isolate_scope;
|
|
|
|
LocalContext context(isolate_scope.isolate());
|
|
|
|
|
|
|
|
WasmRunner<uint32_t> r(execution_tier, nullptr, "main",
|
2021-07-08 09:36:35 +00:00
|
|
|
kRuntimeExceptionSupport, kMemory32,
|
|
|
|
isolate_scope.i_isolate());
|
2021-05-03 14:29:06 +00:00
|
|
|
|
|
|
|
// Build a function that calls itself until stack overflow.
|
|
|
|
WasmFunctionCompiler& stack_overflow = r.NewFunction(sigs.v_v());
|
2021-06-09 22:13:04 +00:00
|
|
|
byte stack_overflow_code[] = {
|
|
|
|
kExprCallFunction, static_cast<byte>(stack_overflow.function_index())};
|
2021-05-03 14:29:06 +00:00
|
|
|
stack_overflow.Build(stack_overflow_code,
|
|
|
|
stack_overflow_code + arraysize(stack_overflow_code));
|
|
|
|
|
|
|
|
// Build the main test function.
|
2021-06-09 22:13:04 +00:00
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(1), kExprCallFunction,
|
|
|
|
static_cast<byte>(stack_overflow.function_index())),
|
|
|
|
WASM_STMTS(WASM_I32V(1))));
|
2021-05-03 14:29:06 +00:00
|
|
|
|
|
|
|
if (execution_tier != TestExecutionTier::kInterpreter) {
|
|
|
|
// Need to call through JS to allow for creation of stack traces.
|
|
|
|
r.CheckCallViaJSTraps();
|
|
|
|
} else {
|
|
|
|
constexpr int stopped = 0;
|
|
|
|
CHECK_EQ(stopped, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-23 16:42:10 +00:00
|
|
|
TEST(Regress1180457) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t> r(TestExecutionTier::kInterpreter);
|
|
|
|
constexpr uint32_t kResult0 = 23;
|
|
|
|
constexpr uint32_t kUnreachable = 42;
|
|
|
|
BUILD(r, WASM_TRY_CATCH_ALL_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_TRY_DELEGATE_T(
|
|
|
|
kWasmI32, WASM_STMTS(WASM_I32V(kResult0), WASM_BR(0)), 0),
|
|
|
|
WASM_I32V(kUnreachable)));
|
|
|
|
|
|
|
|
CHECK_EQ(kResult0, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
|
2021-03-18 09:32:20 +00:00
|
|
|
TEST(Regress1187896) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t> r(TestExecutionTier::kInterpreter);
|
|
|
|
byte try_sig = r.builder().AddSignature(sigs.v_i());
|
|
|
|
constexpr uint32_t kResult = 23;
|
|
|
|
BUILD(r, kExprI32Const, 0, kExprTry, try_sig, kExprDrop, kExprCatchAll,
|
|
|
|
kExprNop, kExprEnd, kExprI32Const, kResult);
|
|
|
|
CHECK_EQ(kResult, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
|
2021-03-22 13:06:30 +00:00
|
|
|
TEST(Regress1190291) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t> r(TestExecutionTier::kInterpreter);
|
|
|
|
byte try_sig = r.builder().AddSignature(sigs.v_i());
|
|
|
|
BUILD(r, kExprUnreachable, kExprTry, try_sig, kExprCatchAll, kExprEnd,
|
|
|
|
kExprI32Const, 0);
|
|
|
|
r.CallInterpreter();
|
|
|
|
}
|
|
|
|
|
2021-03-24 09:41:55 +00:00
|
|
|
TEST(Regress1186795) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<uint32_t> r(TestExecutionTier::kInterpreter);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_i());
|
2021-06-14 13:33:46 +00:00
|
|
|
BUILD(r,
|
|
|
|
WASM_TRY_CATCH_T(
|
|
|
|
kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(0), WASM_I32V(0), WASM_I32V(0), WASM_I32V(0),
|
|
|
|
WASM_I32V(0), WASM_I32V(0), WASM_I32V(0),
|
|
|
|
WASM_TRY_T(kWasmI32,
|
|
|
|
WASM_STMTS(WASM_I32V(0), WASM_THROW(except))),
|
|
|
|
WASM_DROP, WASM_DROP, WASM_DROP, WASM_DROP, WASM_DROP,
|
|
|
|
WASM_DROP, WASM_DROP),
|
|
|
|
WASM_NOP, except));
|
2021-03-24 09:41:55 +00:00
|
|
|
CHECK_EQ(0, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
|
2021-04-12 17:11:55 +00:00
|
|
|
TEST(Regress1197408) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<int32_t, int32_t, int32_t, int32_t> r(
|
|
|
|
TestExecutionTier::kInterpreter);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte sig_id = r.builder().AddSignature(sigs.i_iii());
|
2021-04-12 17:11:55 +00:00
|
|
|
BUILD(r, WASM_STMTS(WASM_I32V(0), WASM_I32V(0), WASM_I32V(0), kExprTry,
|
|
|
|
sig_id, kExprTry, sig_id, kExprCallFunction, 0,
|
|
|
|
kExprDelegate, 0, kExprDelegate, 0));
|
|
|
|
CHECK_EQ(0, r.CallInterpreter(0, 0, 0));
|
|
|
|
}
|
|
|
|
|
2021-05-31 09:26:42 +00:00
|
|
|
TEST(Regress1212396) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<int32_t> r(TestExecutionTier::kInterpreter);
|
2021-06-09 22:13:04 +00:00
|
|
|
byte except = r.builder().AddException(sigs.v_v());
|
2021-05-31 09:26:42 +00:00
|
|
|
BUILD(r, kExprTry, kVoidCode, kExprTry, kVoidCode, kExprI32Const, 0,
|
|
|
|
kExprThrow, except, kExprDelegate, 0, kExprCatch, except, kExprEnd,
|
|
|
|
kExprI32Const, 42);
|
|
|
|
CHECK_EQ(42, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
|
2021-06-24 10:30:22 +00:00
|
|
|
TEST(Regress1219746) {
|
|
|
|
TestSignatures sigs;
|
|
|
|
WasmRunner<int32_t> r(TestExecutionTier::kInterpreter);
|
|
|
|
BUILD(r, kExprTry, kVoidCode, kExprI32Const, 0, kExprEnd);
|
|
|
|
CHECK_EQ(0, r.CallInterpreter());
|
|
|
|
}
|
|
|
|
|
2019-01-28 13:59:04 +00:00
|
|
|
} // namespace test_run_wasm_exceptions
|
|
|
|
} // namespace wasm
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|