Jason Molenda | 0dfb84c | 2018-09-07 01:28:48 +0000 | [diff] [blame] | 1 | //===-- TestArmv7Disassembly.cpp ------------------------------------*- C++ |
| 2 | //-*-===// |
| 3 | |
| 4 | // |
| 5 | // The LLVM Compiler Infrastructure |
| 6 | // |
| 7 | // This file is distributed under the University of Illinois Open Source |
| 8 | // License. See LICENSE.TXT for details. |
| 9 | // |
| 10 | //===----------------------------------------------------------------------===// |
| 11 | |
| 12 | #include "gtest/gtest.h" |
| 13 | |
| 14 | #include "lldb/Core/Address.h" |
| 15 | #include "lldb/Core/Disassembler.h" |
| 16 | #include "lldb/Utility/ArchSpec.h" |
| 17 | #include "lldb/Target/ExecutionContext.h" |
| 18 | |
| 19 | #include "Plugins/Disassembler/llvm/DisassemblerLLVMC.h" |
| 20 | #include "llvm/Support/TargetSelect.h" |
| 21 | |
| 22 | using namespace lldb; |
| 23 | using namespace lldb_private; |
| 24 | |
| 25 | class TestArmv7Disassembly : public testing::Test { |
| 26 | public: |
| 27 | static void SetUpTestCase(); |
| 28 | static void TearDownTestCase(); |
| 29 | |
| 30 | // virtual void SetUp() override { } |
| 31 | // virtual void TearDown() override { } |
| 32 | |
| 33 | protected: |
| 34 | }; |
| 35 | |
| 36 | void TestArmv7Disassembly::SetUpTestCase() { |
| 37 | llvm::InitializeAllTargets(); |
| 38 | llvm::InitializeAllAsmPrinters(); |
| 39 | llvm::InitializeAllTargetMCs(); |
| 40 | llvm::InitializeAllDisassemblers(); |
| 41 | DisassemblerLLVMC::Initialize(); |
| 42 | } |
| 43 | |
| 44 | void TestArmv7Disassembly::TearDownTestCase() { |
| 45 | DisassemblerLLVMC::Terminate(); |
| 46 | } |
| 47 | |
| 48 | TEST_F(TestArmv7Disassembly, TestCortexFPDisass) { |
| 49 | ArchSpec arch("armv7em--"); |
| 50 | |
Raphael Isemann | b658f1d | 2018-09-11 12:45:22 +0000 | [diff] [blame] | 51 | const unsigned num_of_instructions = 3; |
Jason Molenda | 0dfb84c | 2018-09-07 01:28:48 +0000 | [diff] [blame] | 52 | uint8_t data[] = { |
| 53 | 0x00, 0xee, 0x10, 0x2a, // 0xee002a10 : vmov s0, r2 |
| 54 | 0xb8, 0xee, 0xc0, 0x0b, // 0xeeb80bc0 : vcvt.f64.s32 d0, s0 |
| 55 | 0xb6, 0xee, 0x00, 0x0a, // 0xeeb60a00 : vmov.f32 s0, #5.000000e-01 |
| 56 | }; |
| 57 | |
| 58 | // these can be disassembled by hand with llvm-mc, e.g. |
| 59 | // |
| 60 | // 0x00, 0xee, 0x10, 0x2a, // 0xee002a10 : vmov s0, r2 |
| 61 | // |
| 62 | // echo 0x00 0xee 0x10 0x2a | llvm-mc -arch thumb -disassemble -mattr=+fp-armv8 |
| 63 | // vmov s0, r2 |
| 64 | |
| 65 | DisassemblerSP disass_sp; |
| 66 | Address start_addr(0x100); |
| 67 | disass_sp = Disassembler::DisassembleBytes(arch, nullptr, nullptr, start_addr, |
| 68 | &data, sizeof (data), num_of_instructions, false); |
| 69 | |
Jason Molenda | 956ff0f | 2018-09-12 19:30:03 +0000 | [diff] [blame] | 70 | // If we failed to get a disassembler, we can assume it is because |
| 71 | // the llvm we linked against was not built with the ARM target, |
| 72 | // and we should skip these tests without marking anything as failing. |
| 73 | |
Jason Molenda | 0dfb84c | 2018-09-07 01:28:48 +0000 | [diff] [blame] | 74 | if (disass_sp) { |
| 75 | const InstructionList inst_list (disass_sp->GetInstructionList()); |
| 76 | EXPECT_EQ (num_of_instructions, inst_list.GetSize()); |
| 77 | |
| 78 | InstructionSP inst_sp; |
| 79 | const char *mnemonic; |
| 80 | ExecutionContext exe_ctx (nullptr, nullptr, nullptr); |
| 81 | inst_sp = inst_list.GetInstructionAtIndex (0); |
| 82 | mnemonic = inst_sp->GetMnemonic(&exe_ctx); |
| 83 | ASSERT_STREQ ("vmov", mnemonic); |
| 84 | |
| 85 | inst_sp = inst_list.GetInstructionAtIndex (1); |
| 86 | mnemonic = inst_sp->GetMnemonic(&exe_ctx); |
| 87 | ASSERT_STREQ ("vcvt.f64.s32", mnemonic); |
| 88 | |
| 89 | inst_sp = inst_list.GetInstructionAtIndex (2); |
| 90 | mnemonic = inst_sp->GetMnemonic(&exe_ctx); |
| 91 | ASSERT_STREQ ("vmov.f32", mnemonic); |
| 92 | } |
| 93 | } |