blob: 7ef39cfd66f57ef2169f98c4551961e4b14dcb9c [file] [log] [blame]
Logan Chien9347e0b2011-07-07 19:51:47 +08001/*
2 * Copyright 2011, The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "Disassembler.h"
18
19#include "Config.h"
20
Logan Chienf11da302011-07-10 18:37:49 +080021#include "DebugHelper.h"
Logan Chien9347e0b2011-07-07 19:51:47 +080022#include "ExecutionEngine/Compiler.h"
23
24#include "llvm/MC/MCAsmInfo.h"
25#include "llvm/MC/MCDisassembler.h"
26#include "llvm/MC/MCInst.h"
27#include "llvm/MC/MCInstPrinter.h"
28
29#include "llvm/Support/MemoryObject.h"
30#include "llvm/Support/raw_ostream.h"
31
32#include "llvm/Target/TargetData.h"
33#include "llvm/Target/TargetMachine.h"
34#include "llvm/Target/TargetOptions.h"
35#include "llvm/Target/TargetRegistry.h"
36#include "llvm/Target/TargetSelect.h"
37
38#include "llvm/LLVMContext.h"
39
40#if USE_DISASSEMBLER
41
42namespace {
43
44class BufferMemoryObject : public llvm::MemoryObject {
45private:
46 const uint8_t *mBytes;
47 uint64_t mLength;
48
49public:
50 BufferMemoryObject(const uint8_t *Bytes, uint64_t Length)
51 : mBytes(Bytes), mLength(Length) {
52 }
53
54 virtual uint64_t getBase() const { return 0; }
55 virtual uint64_t getExtent() const { return mLength; }
56
57 virtual int readByte(uint64_t Addr, uint8_t *Byte) const {
58 if (Addr > getExtent())
59 return -1;
60 *Byte = mBytes[Addr];
61 return 0;
62 }
63};
64
65} // namespace anonymous
66
67namespace bcc {
68
69void InitializeDisassembler() {
70#if defined(DEFAULT_ARM_CODEGEN) || defined(PROVIDE_ARM_CODEGEN)
71 LLVMInitializeARMDisassembler();
72 LLVMInitializeARMAsmPrinter();
73#endif
74
75#if defined(DEFAULT_X86_CODEGEN) || defined(PROVIDE_X86_CODEGEN)
76 LLVMInitializeX86Disassembler();
77 LLVMInitializeX86AsmPrinter();
78#endif
79
80#if defined(DEFAULT_X64_CODEGEN) || defined(PROVIDE_X64_CODEGEN)
81 LLVMInitializeX86Disassembler();
82 LLVMInitializeX86AsmPrinter();
83#endif
84}
85
86void Disassemble(char const *OutputFileName,
87 llvm::Target const *Target,
88 llvm::TargetMachine *TM,
89 std::string const &Name,
90 unsigned char const *Func,
91 size_t FuncSize) {
Logan Chien9347e0b2011-07-07 19:51:47 +080092
Logan Chien9347e0b2011-07-07 19:51:47 +080093 std::string ErrorInfo;
Logan Chien9347e0b2011-07-07 19:51:47 +080094
Logan Chienf11da302011-07-10 18:37:49 +080095 // Open the disassembler output file
96 llvm::raw_fd_ostream OS(OutputFileName, ErrorInfo,
97 llvm::raw_fd_ostream::F_Append);
98
99 if (!ErrorInfo.empty()) {
100 LOGE("Unable to open disassembler output file: %s\n", OutputFileName);
Logan Chien9347e0b2011-07-07 19:51:47 +0800101 return;
102 }
Logan Chien9347e0b2011-07-07 19:51:47 +0800103
Logan Chienf11da302011-07-10 18:37:49 +0800104 // Disassemble the given function
105 OS << "Disassembled code: " << Name << "\n";
Logan Chien9347e0b2011-07-07 19:51:47 +0800106
107 const llvm::MCAsmInfo *AsmInfo;
108 const llvm::MCDisassembler *Disassmbler;
109 llvm::MCInstPrinter *IP;
110
Shih-wei Liaobb5df5c2011-07-19 05:15:11 -0700111 AsmInfo = Target->createMCAsmInfo(Compiler::getTargetTriple());
Logan Chien9347e0b2011-07-07 19:51:47 +0800112 Disassmbler = Target->createMCDisassembler();
Shih-wei Liaobb5df5c2011-07-19 05:15:11 -0700113 IP = Target->createMCInstPrinter(AsmInfo->getAssemblerDialect(),
Logan Chien9347e0b2011-07-07 19:51:47 +0800114 *AsmInfo);
115
116 const BufferMemoryObject *BufferMObj = new BufferMemoryObject(Func, FuncSize);
117
118 uint64_t Size;
119 uint64_t Index;
120
121 for (Index = 0; Index < FuncSize; Index += Size) {
122 llvm::MCInst Inst;
123
124 if (Disassmbler->getInstruction(Inst, Size, *BufferMObj, Index,
125 /* REMOVED */ llvm::nulls())) {
Logan Chienf11da302011-07-10 18:37:49 +0800126 OS.indent(4);
127 OS.write("0x", 2);
128 OS.write_hex((uint32_t)Func + Index);
129 OS.write(": 0x", 4);
130 OS.write_hex(*(uint32_t *)(Func + Index));
131 IP->printInst(&Inst, OS);
132 OS << "\n";
Logan Chien9347e0b2011-07-07 19:51:47 +0800133 } else {
134 if (Size == 0)
135 Size = 1; // skip illegible bytes
136 }
137 }
138
Logan Chienf11da302011-07-10 18:37:49 +0800139 OS << "\n";
140
Logan Chien9347e0b2011-07-07 19:51:47 +0800141 delete BufferMObj;
142
143 delete AsmInfo;
144 delete Disassmbler;
145 delete IP;
146
Logan Chienf11da302011-07-10 18:37:49 +0800147 OS.close();
Logan Chien9347e0b2011-07-07 19:51:47 +0800148}
149
150} // namespace bcc
151
152#endif // USE_DISASSEMBLER