2021-01-03 19:49:22 +00:00
|
|
|
// Copyright 2021 Dolphin Emulator Project
|
2021-07-05 01:22:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2021-01-03 19:49:22 +00:00
|
|
|
|
2022-08-06 04:10:17 +00:00
|
|
|
#include <bit>
|
2021-01-03 19:49:22 +00:00
|
|
|
#include <cstddef>
|
2023-02-22 20:55:12 +00:00
|
|
|
#include <random>
|
2021-01-03 19:49:22 +00:00
|
|
|
#include <type_traits>
|
|
|
|
|
|
|
|
#include "Common/Arm64Emitter.h"
|
|
|
|
#include "Common/Assert.h"
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
using namespace Arm64Gen;
|
|
|
|
|
|
|
|
class TestMovI2R : public ARM64CodeBlock
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
TestMovI2R() { AllocCodeSpace(4096); }
|
|
|
|
|
|
|
|
void Check32(u32 value)
|
|
|
|
{
|
|
|
|
ResetCodePtr();
|
|
|
|
|
|
|
|
const u8* fn = GetCodePtr();
|
2021-06-19 17:05:35 +00:00
|
|
|
{
|
|
|
|
const Common::ScopedJITPageWriteAndNoExecute enable_jit_page_writes;
|
|
|
|
MOVI2R(ARM64Reg::W0, value);
|
|
|
|
RET();
|
|
|
|
}
|
2021-01-03 19:49:22 +00:00
|
|
|
|
|
|
|
FlushIcacheSection(const_cast<u8*>(fn), const_cast<u8*>(GetCodePtr()));
|
|
|
|
|
2022-08-06 04:10:17 +00:00
|
|
|
const u64 result = std::bit_cast<u64 (*)()>(fn)();
|
2021-01-03 19:49:22 +00:00
|
|
|
EXPECT_EQ(value, result);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Check64(u64 value)
|
|
|
|
{
|
|
|
|
ResetCodePtr();
|
|
|
|
|
|
|
|
const u8* fn = GetCodePtr();
|
2021-06-19 17:05:35 +00:00
|
|
|
{
|
|
|
|
const Common::ScopedJITPageWriteAndNoExecute enable_jit_page_writes;
|
|
|
|
MOVI2R(ARM64Reg::X0, value);
|
|
|
|
RET();
|
|
|
|
}
|
2021-01-03 19:49:22 +00:00
|
|
|
|
|
|
|
FlushIcacheSection(const_cast<u8*>(fn), const_cast<u8*>(GetCodePtr()));
|
|
|
|
|
2022-08-06 04:10:17 +00:00
|
|
|
const u64 result = std::bit_cast<u64 (*)()>(fn)();
|
2021-01-03 19:49:22 +00:00
|
|
|
EXPECT_EQ(value, result);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST(JitArm64, MovI2R_32BitValues)
|
|
|
|
{
|
2023-02-22 20:55:12 +00:00
|
|
|
std::default_random_engine engine(0);
|
|
|
|
std::uniform_int_distribution<u32> dist;
|
2021-01-03 19:49:22 +00:00
|
|
|
TestMovI2R test;
|
2021-01-31 13:13:10 +00:00
|
|
|
for (u64 i = 0; i < 0x100000; i++)
|
2021-01-03 19:49:22 +00:00
|
|
|
{
|
2023-02-22 20:55:12 +00:00
|
|
|
const u32 value = dist(engine);
|
2021-01-31 13:13:10 +00:00
|
|
|
test.Check32(value);
|
|
|
|
test.Check64(value);
|
2021-01-03 19:49:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(JitArm64, MovI2R_Rand)
|
|
|
|
{
|
2023-02-22 20:55:12 +00:00
|
|
|
std::default_random_engine engine(0);
|
|
|
|
std::uniform_int_distribution<u64> dist;
|
2021-01-03 19:49:22 +00:00
|
|
|
TestMovI2R test;
|
|
|
|
for (u64 i = 0; i < 0x100000; i++)
|
|
|
|
{
|
2023-02-22 20:55:12 +00:00
|
|
|
test.Check64(dist(engine));
|
2021-01-03 19:49:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-03 16:48:14 +00:00
|
|
|
// Construct and test every 64-bit logical immediate
|
|
|
|
TEST(JitArm64, MovI2R_LogImm)
|
|
|
|
{
|
|
|
|
TestMovI2R test;
|
|
|
|
|
|
|
|
for (unsigned size = 2; size <= 64; size *= 2)
|
|
|
|
{
|
|
|
|
for (unsigned length = 1; length < size; ++length)
|
|
|
|
{
|
|
|
|
u64 imm = ~u64{0} >> (64 - length);
|
|
|
|
for (unsigned e = size; e < 64; e *= 2)
|
|
|
|
{
|
|
|
|
imm |= imm << e;
|
|
|
|
}
|
|
|
|
for (unsigned rotation = 0; rotation < size; ++rotation)
|
|
|
|
{
|
|
|
|
test.Check64(imm);
|
2023-12-16 12:27:13 +00:00
|
|
|
EXPECT_EQ(static_cast<bool>(LogicalImm(imm, GPRSize::B64)), true);
|
2022-07-03 16:48:14 +00:00
|
|
|
|
|
|
|
if (size < 64)
|
|
|
|
{
|
|
|
|
test.Check32(imm);
|
2023-12-16 12:27:13 +00:00
|
|
|
EXPECT_EQ(static_cast<bool>(LogicalImm(static_cast<u32>(imm), GPRSize::B32)), true);
|
2022-07-03 16:48:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
imm = (imm >> 63) | (imm << 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-03 19:49:22 +00:00
|
|
|
TEST(JitArm64, MovI2R_ADP)
|
|
|
|
{
|
|
|
|
TestMovI2R test;
|
2022-08-06 04:10:17 +00:00
|
|
|
const u64 base = std::bit_cast<u64>(test.GetCodePtr());
|
2021-02-13 10:54:46 +00:00
|
|
|
|
|
|
|
// Test offsets around 0
|
2021-01-03 19:49:22 +00:00
|
|
|
for (s64 i = -0x20000; i < 0x20000; i++)
|
|
|
|
{
|
|
|
|
const u64 offset = static_cast<u64>(i);
|
|
|
|
test.Check64(base + offset);
|
|
|
|
}
|
2021-02-13 10:54:46 +00:00
|
|
|
|
|
|
|
// Test offsets around the maximum
|
|
|
|
for (const s64 i : {-0x200000ll, 0x200000ll})
|
|
|
|
{
|
|
|
|
for (s64 j = -4; j < 4; j++)
|
|
|
|
{
|
|
|
|
const u64 offset = static_cast<u64>(i + j);
|
|
|
|
test.Check64(base + offset);
|
|
|
|
}
|
|
|
|
}
|
2021-01-03 19:49:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(JitArm64, MovI2R_ADRP)
|
|
|
|
{
|
|
|
|
TestMovI2R test;
|
2022-08-06 04:10:17 +00:00
|
|
|
const u64 base = std::bit_cast<u64>(test.GetCodePtr()) & ~0xFFF;
|
2021-02-13 10:54:46 +00:00
|
|
|
|
|
|
|
// Test offsets around 0
|
2021-01-03 19:49:22 +00:00
|
|
|
for (s64 i = -0x20000; i < 0x20000; i++)
|
|
|
|
{
|
|
|
|
const u64 offset = static_cast<u64>(i) << 12;
|
|
|
|
test.Check64(base + offset);
|
|
|
|
}
|
2021-02-13 10:54:46 +00:00
|
|
|
|
|
|
|
// Test offsets around the maximum
|
|
|
|
for (const s64 i : {-0x100000000ll, -0x80000000ll, 0x80000000ll, 0x100000000ll})
|
|
|
|
{
|
|
|
|
for (s64 j = -4; j < 4; j++)
|
|
|
|
{
|
|
|
|
const u64 offset = static_cast<u64>(i + (j << 12));
|
|
|
|
test.Check64(base + offset);
|
|
|
|
}
|
|
|
|
}
|
2021-01-03 19:49:22 +00:00
|
|
|
}
|