| //===-- TestX86GetControlFlowKind.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 "llvm/Support/TargetSelect.h" | 
 | #include "gtest/gtest.h" | 
 |  | 
 | #include "lldb/Core/Address.h" | 
 | #include "lldb/Core/Disassembler.h" | 
 | #include "lldb/Target/ExecutionContext.h" | 
 | #include "lldb/Utility/ArchSpec.h" | 
 |  | 
 | #include "Plugins/Disassembler/LLVMC/DisassemblerLLVMC.h" | 
 |  | 
 | using namespace lldb; | 
 | using namespace lldb_private; | 
 |  | 
 | namespace { | 
 | class TestGetControlFlowKindx86 : public testing::Test { | 
 | public: | 
 |   static void SetUpTestCase(); | 
 |   static void TearDownTestCase(); | 
 |  | 
 | protected: | 
 | }; | 
 |  | 
 | void TestGetControlFlowKindx86::SetUpTestCase() { | 
 |   llvm::InitializeAllTargets(); | 
 |   llvm::InitializeAllAsmPrinters(); | 
 |   llvm::InitializeAllTargetMCs(); | 
 |   llvm::InitializeAllDisassemblers(); | 
 |   DisassemblerLLVMC::Initialize(); | 
 | } | 
 |  | 
 | void TestGetControlFlowKindx86::TearDownTestCase() { | 
 |   DisassemblerLLVMC::Terminate(); | 
 | } | 
 | } // namespace | 
 |  | 
 | TEST_F(TestGetControlFlowKindx86, TestX86_64Instruction) { | 
 |   ArchSpec arch("x86_64-*-linux"); | 
 |  | 
 |   const unsigned num_of_instructions = 29; | 
 |   uint8_t data[] = { | 
 |       0x55,                               // other -- pushq %rbp | 
 |       0x48, 0x89, 0xe5,                   // other -- movq %rsp, %rbp | 
 |  | 
 |       0xe8, 0xfc, 0xfe, 0xff, 0xff,       // call -- callq 0x4004c0 | 
 |       0x41, 0xff, 0x14, 0xdc,             // call -- callq *(%r12,%rbx,8) | 
 |       0xff, 0x50, 0x18,                   // call -- callq *0x18(%rax) | 
 |       0xe8, 0x48, 0x0d, 0x00, 0x00,       // call -- callq 0x94fe0 | 
 |  | 
 |       0xc3,                               // return -- retq | 
 |  | 
 |       0xeb, 0xd3,                         // jump -- jmp 0x92dab | 
 |       0xe9, 0x22, 0xff, 0xff, 0xff,       // jump -- jmp 0x933ae | 
 |       0xff, 0xe0,                         // jump -- jmpq *%rax | 
 |       0xf2, 0xff, 0x25, 0x75, 0xe7, 0x39, 0x00, // jump -- repne jmpq *0x39e775 | 
 |  | 
 |       0x73, 0xc2,                         // cond jump -- jae 0x9515c | 
 |       0x74, 0x1f,                         // cond jump -- je 0x400626 | 
 |       0x75, 0xea,                         // cond jump -- jne 0x400610 | 
 |       0x76, 0x10,                         // cond jump -- jbe 0x94d10 | 
 |       0x77, 0x58,                         // cond jump -- ja 0x1208c8 | 
 |       0x7e, 0x67,                         // cond jump -- jle 0x92180 | 
 |       0x78, 0x0b,                         // cond jump -- js 0x92dc3 | 
 |       0x0f, 0x82, 0x17, 0x01, 0x00, 0x00, // cond jump -- jb 0x9c7b0 | 
 |       0x0f, 0x83, 0xa7, 0x00, 0x00, 0x00, // cond jump -- jae 0x895c8 | 
 |       0x0f, 0x84, 0x8c, 0x00, 0x00, 0x00, // cond jump -- je 0x941f0 | 
 |       0x0f, 0x85, 0x51, 0xff, 0xff, 0xff, // cond jump -- jne 0x8952c | 
 |       0x0f, 0x86, 0xa3, 0x02, 0x00, 0x00, // cond jump -- jbe 0x9ae10 | 
 |       0x0f, 0x87, 0xff, 0x00, 0x00, 0x00, // cond jump -- ja 0x9ab60 | 
 |       0x0f, 0x8e, 0x7e, 0x00, 0x00, 0x00, // cond jump -- jle 0x92dd8 | 
 |       0x0f, 0x86, 0xdf, 0x00, 0x00, 0x00, // cond jump -- jbe 0x921b0 | 
 |  | 
 |       0x0f, 0x05,                         // far call -- syscall | 
 |  | 
 |       0x0f, 0x07,                         // far return -- sysret | 
 |       0xcf,                               // far return -- interrupt ret | 
 |   }; | 
 |  | 
 |   InstructionControlFlowKind result[] = { | 
 |       eInstructionControlFlowKindOther, | 
 |       eInstructionControlFlowKindOther, | 
 |  | 
 |       eInstructionControlFlowKindCall, | 
 |       eInstructionControlFlowKindCall, | 
 |       eInstructionControlFlowKindCall, | 
 |       eInstructionControlFlowKindCall, | 
 |  | 
 |       eInstructionControlFlowKindReturn, | 
 |  | 
 |       eInstructionControlFlowKindJump, | 
 |       eInstructionControlFlowKindJump, | 
 |       eInstructionControlFlowKindJump, | 
 |       eInstructionControlFlowKindJump, | 
 |  | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |       eInstructionControlFlowKindCondJump, | 
 |  | 
 |       eInstructionControlFlowKindFarCall, | 
 |  | 
 |       eInstructionControlFlowKindFarReturn, | 
 |       eInstructionControlFlowKindFarReturn, | 
 |   }; | 
 |  | 
 |   DisassemblerSP disass_sp; | 
 |   Address start_addr(0x100); | 
 |   disass_sp = Disassembler::DisassembleBytes( | 
 |       arch, nullptr, nullptr, nullptr, nullptr, start_addr, &data, sizeof(data), | 
 |       num_of_instructions, false); | 
 |  | 
 |   // If we failed to get a disassembler, we can assume it is because | 
 |   // the llvm we linked against was not built with the i386 target, | 
 |   // and we should skip these tests without marking anything as failing. | 
 |   if (!disass_sp) | 
 |     return; | 
 |  | 
 |   const InstructionList inst_list(disass_sp->GetInstructionList()); | 
 |   EXPECT_EQ(num_of_instructions, inst_list.GetSize()); | 
 |  | 
 |   for (size_t i = 0; i < num_of_instructions; ++i) { | 
 |     InstructionSP inst_sp; | 
 |     inst_sp = inst_list.GetInstructionAtIndex(i); | 
 |     ExecutionContext exe_ctx(nullptr, nullptr, nullptr); | 
 |     InstructionControlFlowKind kind = inst_sp->GetControlFlowKind(&exe_ctx); | 
 |     EXPECT_EQ(kind, result[i]); | 
 |  | 
 |     // Also, test the DisassemblerLLVMC::MCDisasmInstance methods. | 
 |     if (kind == eInstructionControlFlowKindReturn) { | 
 |       EXPECT_FALSE(inst_sp->IsCall()); | 
 |     } | 
 |  | 
 |     if (kind == eInstructionControlFlowKindCall) { | 
 |       EXPECT_TRUE(inst_sp->IsCall()); | 
 |     } | 
 |  | 
 |     if (kind == eInstructionControlFlowKindCall || | 
 |         kind == eInstructionControlFlowKindJump || | 
 |         kind == eInstructionControlFlowKindCondJump || | 
 |         kind == eInstructionControlFlowKindReturn) { | 
 |       EXPECT_TRUE(inst_sp->DoesBranch()); | 
 |     } | 
 |   } | 
 | } |