2018-09-21 09:14:51 +02:00
|
|
|
// Copyright 2018 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.
|
|
|
|
|
2019-08-01 08:38:30 +02:00
|
|
|
#include "src/codegen/macro-assembler.h"
|
|
|
|
#include "src/execution/simulator.h"
|
2018-09-21 09:14:51 +02:00
|
|
|
#include "test/common/assembler-tester.h"
|
2020-05-05 09:19:02 +02:00
|
|
|
#include "test/unittests/test-utils.h"
|
2018-09-21 09:14:51 +02:00
|
|
|
#include "testing/gtest-support.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
2023-03-30 12:11:08 +02:00
|
|
|
#define __ masm.
|
2018-09-21 09:14:51 +02:00
|
|
|
|
2023-10-05 08:46:07 +02:00
|
|
|
// Test the ia32 assembler by compiling some simple functions into
|
2018-09-21 09:14:51 +02:00
|
|
|
// a buffer and executing them. These tests do not initialize the
|
|
|
|
// V8 library, create a context, or use any V8 objects.
|
|
|
|
|
2023-03-30 12:11:08 +02:00
|
|
|
class MacroAssemblerTest : public TestWithIsolate {};
|
2020-05-05 09:19:02 +02:00
|
|
|
|
2023-03-30 12:11:08 +02:00
|
|
|
TEST_F(MacroAssemblerTest, TestHardAbort) {
|
2019-03-12 09:01:49 +01:00
|
|
|
auto buffer = AllocateAssemblerBuffer();
|
2023-03-30 12:11:08 +02:00
|
|
|
MacroAssembler masm(isolate(), AssemblerOptions{}, CodeObjectRequired::kNo,
|
2019-03-12 09:01:49 +01:00
|
|
|
buffer->CreateView());
|
2020-05-05 09:19:02 +02:00
|
|
|
__ set_root_array_available(false);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Initialize the root register, as we need it for `Abort()`. Since `Abort()`
|
|
|
|
// does not return properly, we don't need to restore `kRootRegister`, even
|
|
|
|
// though it's a callee-saved register.
|
|
|
|
__ LoadAddress(kRootRegister, ExternalReference::isolate_root(isolate()));
|
|
|
|
__ set_root_array_available(true);
|
2018-09-21 09:14:51 +02:00
|
|
|
__ set_abort_hard(true);
|
|
|
|
|
|
|
|
__ Abort(AbortReason::kNoReason);
|
|
|
|
|
|
|
|
CodeDesc desc;
|
2023-03-30 12:11:08 +02:00
|
|
|
masm.GetCode(isolate(), &desc);
|
2019-03-12 09:01:49 +01:00
|
|
|
buffer->MakeExecutable();
|
2020-05-05 09:19:02 +02:00
|
|
|
auto f = GeneratedCode<void>::FromBuffer(isolate(), buffer->start());
|
2018-09-21 09:14:51 +02:00
|
|
|
|
2025-04-29 08:03:15 +02:00
|
|
|
ASSERT_DEATH_IF_SUPPORTED(
|
|
|
|
{ f.Call(); }, v8_flags.debug_code ? "abort: no reason" : "");
|
2018-09-21 09:14:51 +02:00
|
|
|
}
|
|
|
|
|
2023-03-30 12:11:08 +02:00
|
|
|
TEST_F(MacroAssemblerTest, TestCheck) {
|
2019-03-12 09:01:49 +01:00
|
|
|
auto buffer = AllocateAssemblerBuffer();
|
2023-03-30 12:11:08 +02:00
|
|
|
MacroAssembler masm(isolate(), AssemblerOptions{}, CodeObjectRequired::kNo,
|
2019-03-12 09:01:49 +01:00
|
|
|
buffer->CreateView());
|
2020-05-05 09:19:02 +02:00
|
|
|
__ set_root_array_available(false);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Initialize the root register, as we need it for `Check()`.
|
|
|
|
// Save the value in `kRootRegister` to restore it later after the call. In
|
|
|
|
// some configurations `kRootRegister` is callee-saved for C++.
|
|
|
|
__ mov(ecx, kRootRegister);
|
|
|
|
__ LoadAddress(kRootRegister, ExternalReference::isolate_root(isolate()));
|
|
|
|
__ set_root_array_available(true);
|
2018-09-21 09:14:51 +02:00
|
|
|
__ set_abort_hard(true);
|
|
|
|
|
|
|
|
// Fail if the first parameter is 17.
|
|
|
|
__ mov(eax, 17);
|
|
|
|
__ cmp(eax, Operand(esp, 4)); // compare with 1st parameter.
|
|
|
|
__ Check(Condition::not_equal, AbortReason::kNoReason);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Restore the original value of `kRootRegister`.
|
|
|
|
__ mov(kRootRegister, ecx);
|
2018-09-21 09:14:51 +02:00
|
|
|
__ ret(0);
|
|
|
|
|
|
|
|
CodeDesc desc;
|
2023-03-30 12:11:08 +02:00
|
|
|
masm.GetCode(isolate(), &desc);
|
2019-03-12 09:01:49 +01:00
|
|
|
buffer->MakeExecutable();
|
2020-05-05 09:19:02 +02:00
|
|
|
auto f = GeneratedCode<void, int>::FromBuffer(isolate(), buffer->start());
|
2018-09-21 09:14:51 +02:00
|
|
|
|
|
|
|
f.Call(0);
|
|
|
|
f.Call(18);
|
2025-04-29 08:03:15 +02:00
|
|
|
ASSERT_DEATH_IF_SUPPORTED(
|
|
|
|
{ f.Call(17); }, v8_flags.debug_code ? "abort: no reason" : "");
|
2018-09-21 09:14:51 +02:00
|
|
|
}
|
|
|
|
|
2023-10-05 08:46:07 +02:00
|
|
|
TEST_F(MacroAssemblerTest, TestPCRelLea) {
|
|
|
|
auto buffer = AllocateAssemblerBuffer();
|
|
|
|
MacroAssembler masm(isolate(), AssemblerOptions{}, CodeObjectRequired::kNo,
|
|
|
|
buffer->CreateView());
|
|
|
|
__ set_root_array_available(false);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Initialize the root register, as we need it for `Check()`.
|
|
|
|
// Save the value in `kRootRegister` to restore it later after the call. In
|
|
|
|
// some configurations `kRootRegister` is callee-saved for C++.
|
|
|
|
__ mov(edi, kRootRegister);
|
|
|
|
__ LoadAddress(kRootRegister, ExternalReference::isolate_root(isolate()));
|
|
|
|
__ set_root_array_available(true);
|
2023-10-05 08:46:07 +02:00
|
|
|
__ set_abort_hard(true);
|
|
|
|
|
|
|
|
Label pt;
|
|
|
|
__ LoadLabelAddress(ecx, &pt);
|
|
|
|
__ mov(eax, 42);
|
|
|
|
__ call(ecx);
|
|
|
|
__ cmp(eax, 56);
|
|
|
|
__ Check(Condition::equal, AbortReason::kNoReason);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Restore the original value of `kRootRegister`.
|
|
|
|
__ mov(kRootRegister, edi);
|
2023-10-05 08:46:07 +02:00
|
|
|
__ ret(0);
|
|
|
|
__ bind(&pt);
|
|
|
|
__ mov(eax, 56);
|
|
|
|
__ ret(0);
|
|
|
|
|
|
|
|
CodeDesc desc;
|
|
|
|
masm.GetCode(isolate(), &desc);
|
|
|
|
buffer->MakeExecutable();
|
|
|
|
auto f = GeneratedCode<void, int>::FromBuffer(isolate(), buffer->start());
|
|
|
|
|
|
|
|
f.Call(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MacroAssemblerTest, TestDefinedPCRelLea) {
|
|
|
|
auto buffer = AllocateAssemblerBuffer();
|
|
|
|
MacroAssembler masm(isolate(), AssemblerOptions{}, CodeObjectRequired::kNo,
|
|
|
|
buffer->CreateView());
|
|
|
|
__ set_root_array_available(false);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Initialize the root register, as we need it for `Check()`.
|
|
|
|
// Save the value in `kRootRegister` to restore it later after the call. In
|
|
|
|
// some configurations `kRootRegister` is callee-saved for C++.
|
|
|
|
__ mov(edi, kRootRegister);
|
|
|
|
__ LoadAddress(kRootRegister, ExternalReference::isolate_root(isolate()));
|
|
|
|
__ set_root_array_available(true);
|
2023-10-05 08:46:07 +02:00
|
|
|
__ set_abort_hard(true);
|
|
|
|
|
|
|
|
Label pt, start;
|
|
|
|
__ jmp(&start);
|
|
|
|
__ bind(&pt);
|
|
|
|
__ mov(eax, 56);
|
|
|
|
__ ret(0);
|
|
|
|
__ bind(&start);
|
|
|
|
__ LoadLabelAddress(ecx, &pt);
|
|
|
|
__ mov(eax, 42);
|
|
|
|
__ call(ecx);
|
|
|
|
__ cmp(eax, 56);
|
|
|
|
__ Check(Condition::equal, AbortReason::kNoReason);
|
2024-08-14 20:41:00 +02:00
|
|
|
// Restore the original value of `kRootRegister`.
|
|
|
|
__ mov(kRootRegister, edi);
|
2023-10-05 08:46:07 +02:00
|
|
|
__ ret(0);
|
|
|
|
|
|
|
|
CodeDesc desc;
|
|
|
|
masm.GetCode(isolate(), &desc);
|
|
|
|
buffer->MakeExecutable();
|
|
|
|
auto f = GeneratedCode<void, int>::FromBuffer(isolate(), buffer->start());
|
|
|
|
|
|
|
|
f.Call(0);
|
|
|
|
}
|
|
|
|
|
2018-09-21 09:14:51 +02:00
|
|
|
#undef __
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|