Files
llvm/lldb/unittests/Target/DynamicRegisterInfoTest.cpp
Michał Górny 214054f78a [lldb] Move DynamicRegisterInfo to public Target library
Move DynamicRegisterInfo from the internal lldbPluginProcessUtility
library to the public lldbTarget library.  This is a prerequisite
towards ABI plugin changes that are going to pass DynamicRegisterInfo
parameters.

Differential Revision: https://reviews.llvm.org/D110942
2021-10-05 12:40:55 +02:00

170 lines
5.4 KiB
C++

//===-- DynamicRegisterInfoTest.cpp ---------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "lldb/Target/DynamicRegisterInfo.h"
#include "lldb/Utility/ArchSpec.h"
using namespace lldb_private;
static std::vector<uint32_t> regs_to_vector(uint32_t *regs) {
std::vector<uint32_t> ret;
if (regs) {
while (*regs != LLDB_INVALID_REGNUM)
ret.push_back(*regs++);
}
return ret;
}
class DynamicRegisterInfoTest : public ::testing::Test {
protected:
DynamicRegisterInfo info;
uint32_t next_regnum = 0;
ConstString group{"group"};
uint32_t AddTestRegister(const char *name, uint32_t byte_size,
std::vector<uint32_t> value_regs = {},
std::vector<uint32_t> invalidate_regs = {}) {
struct RegisterInfo new_reg {
name, nullptr, byte_size, LLDB_INVALID_INDEX32, lldb::eEncodingUint,
lldb::eFormatUnsigned,
{LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM,
next_regnum, next_regnum},
nullptr, nullptr, nullptr, 0
};
if (!value_regs.empty()) {
value_regs.push_back(LLDB_INVALID_REGNUM);
new_reg.value_regs = value_regs.data();
}
if (!invalidate_regs.empty()) {
invalidate_regs.push_back(LLDB_INVALID_REGNUM);
new_reg.invalidate_regs = invalidate_regs.data();
}
info.AddRegister(new_reg, group);
return next_regnum++;
}
void AssertRegisterInfo(uint32_t reg_num, const char *reg_name,
uint32_t byte_offset,
std::vector<uint32_t> value_regs = {},
std::vector<uint32_t> invalidate_regs = {}) {
const RegisterInfo *reg = info.GetRegisterInfoAtIndex(reg_num);
EXPECT_NE(reg, nullptr);
if (!reg)
return;
EXPECT_STREQ(reg->name, reg_name);
EXPECT_EQ(reg->byte_offset, byte_offset);
EXPECT_THAT(regs_to_vector(reg->value_regs), value_regs);
EXPECT_THAT(regs_to_vector(reg->invalidate_regs), invalidate_regs);
}
};
#define ASSERT_REG(reg, ...) { \
SCOPED_TRACE("at register " #reg); \
AssertRegisterInfo(reg, #reg, __VA_ARGS__); \
}
TEST_F(DynamicRegisterInfoTest, finalize_regs) {
// Add regular registers
uint32_t b1 = AddTestRegister("b1", 8);
uint32_t b2 = AddTestRegister("b2", 8);
// Add a few sub-registers
uint32_t s1 = AddTestRegister("s1", 4, {b1});
uint32_t s2 = AddTestRegister("s2", 4, {b2});
// Add a register with invalidate_regs
uint32_t i1 = AddTestRegister("i1", 8, {}, {b1});
// Add a register with indirect invalidate regs to be expanded
// TODO: why is it done conditionally to value_regs?
uint32_t i2 = AddTestRegister("i2", 4, {b2}, {i1});
info.Finalize(lldb_private::ArchSpec());
ASSERT_REG(b1, 0);
ASSERT_REG(b2, 8);
ASSERT_REG(s1, 0, {b1});
ASSERT_REG(s2, 8, {b2});
ASSERT_REG(i1, 16, {}, {b1});
ASSERT_REG(i2, 8, {b2}, {b1, i1});
}
TEST_F(DynamicRegisterInfoTest, no_finalize_regs) {
// Add regular registers
uint32_t b1 = AddTestRegister("b1", 8);
uint32_t b2 = AddTestRegister("b2", 8);
// Add a few sub-registers
uint32_t s1 = AddTestRegister("s1", 4, {b1});
uint32_t s2 = AddTestRegister("s2", 4, {b2});
// Add a register with invalidate_regs
uint32_t i1 = AddTestRegister("i1", 8, {}, {b1});
// Add a register with indirect invalidate regs to be expanded
// TODO: why is it done conditionally to value_regs?
uint32_t i2 = AddTestRegister("i2", 4, {b2}, {i1});
ASSERT_REG(b1, LLDB_INVALID_INDEX32);
ASSERT_REG(b2, LLDB_INVALID_INDEX32);
ASSERT_REG(s1, LLDB_INVALID_INDEX32);
ASSERT_REG(s2, LLDB_INVALID_INDEX32);
ASSERT_REG(i1, LLDB_INVALID_INDEX32);
ASSERT_REG(i2, LLDB_INVALID_INDEX32);
}
TEST_F(DynamicRegisterInfoTest, add_supplementary_register) {
// Add a base register
uint32_t rax = AddTestRegister("rax", 8);
// Register numbers
uint32_t eax = 1;
uint32_t ax = 2;
uint32_t al = 3;
ConstString group{"supplementary registers"};
uint32_t value_regs[2] = {rax, LLDB_INVALID_REGNUM};
struct RegisterInfo eax_reg {
"eax", nullptr, 4, LLDB_INVALID_INDEX32, lldb::eEncodingUint,
lldb::eFormatUnsigned,
{LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, eax,
eax},
value_regs, nullptr, nullptr, 0
};
info.AddSupplementaryRegister(eax_reg, group);
struct RegisterInfo ax_reg {
"ax", nullptr, 2, LLDB_INVALID_INDEX32, lldb::eEncodingUint,
lldb::eFormatUnsigned,
{LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, ax, ax},
value_regs, nullptr, nullptr, 0
};
info.AddSupplementaryRegister(ax_reg, group);
struct RegisterInfo al_reg {
"al", nullptr, 1, LLDB_INVALID_INDEX32, lldb::eEncodingUint,
lldb::eFormatUnsigned,
{LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, al, al},
value_regs, nullptr, nullptr, 0
};
info.AddSupplementaryRegister(al_reg, group);
info.Finalize(lldb_private::ArchSpec());
ASSERT_REG(rax, 0, {}, {eax, ax, al});
ASSERT_REG(eax, 0, {rax}, {rax, ax, al});
ASSERT_REG(ax, 0, {rax}, {rax, eax, al});
ASSERT_REG(al, 0, {rax}, {rax, eax, ax});
}