blob: 6057e169e3474b20a3151d177d94ab0c85dc450e [file] [log] [blame]
Sean Callananee5dfd42010-02-01 08:49:35 +00001//===-EDInst.cpp - LLVM Enhanced Disassembler -----------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the Enhanced Disassembly library's instruction class.
11// The instruction is responsible for vending the string representation,
12// individual tokens, and operands for a single instruction.
13//
14//===----------------------------------------------------------------------===//
15
Sean Callananee5dfd42010-02-01 08:49:35 +000016#include "EDInst.h"
Chris Lattner847da552010-07-20 18:25:19 +000017#include "EDDisassembler.h"
Sean Callananee5dfd42010-02-01 08:49:35 +000018#include "EDOperand.h"
19#include "EDToken.h"
20
Sean Callanan9899f702010-04-13 21:21:57 +000021#include "llvm/MC/EDInstInfo.h"
Sean Callananee5dfd42010-02-01 08:49:35 +000022#include "llvm/MC/MCInst.h"
23
24using namespace llvm;
25
26EDInst::EDInst(llvm::MCInst *inst,
27 uint64_t byteSize,
28 EDDisassembler &disassembler,
Sean Callanan9899f702010-04-13 21:21:57 +000029 const llvm::EDInstInfo *info) :
Sean Callananee5dfd42010-02-01 08:49:35 +000030 Disassembler(disassembler),
31 Inst(inst),
32 ThisInstInfo(info),
33 ByteSize(byteSize),
34 BranchTarget(-1),
35 MoveSource(-1),
36 MoveTarget(-1) {
Sean Callanan4285b292010-04-09 00:11:15 +000037 OperandOrder = ThisInstInfo->operandOrders[Disassembler.llvmSyntaxVariant()];
Sean Callananee5dfd42010-02-01 08:49:35 +000038}
39
40EDInst::~EDInst() {
41 unsigned int index;
42 unsigned int numOperands = Operands.size();
43
44 for (index = 0; index < numOperands; ++index)
45 delete Operands[index];
46
47 unsigned int numTokens = Tokens.size();
48
49 for (index = 0; index < numTokens; ++index)
50 delete Tokens[index];
51
52 delete Inst;
53}
54
55uint64_t EDInst::byteSize() {
56 return ByteSize;
57}
58
59int EDInst::stringify() {
60 if (StringifyResult.valid())
61 return StringifyResult.result();
62
63 if (Disassembler.printInst(String, *Inst))
64 return StringifyResult.setResult(-1);
Sean Callanan38a92882010-09-23 02:14:12 +000065
66 String.push_back('\n');
Sean Callananee5dfd42010-02-01 08:49:35 +000067
68 return StringifyResult.setResult(0);
69}
70
71int EDInst::getString(const char*& str) {
72 if (stringify())
73 return -1;
74
75 str = String.c_str();
76
77 return 0;
78}
79
80unsigned EDInst::instID() {
81 return Inst->getOpcode();
82}
83
84bool EDInst::isBranch() {
85 if (ThisInstInfo)
Sean Callanan972bf8d2010-05-11 01:27:08 +000086 return
87 ThisInstInfo->instructionType == kInstructionTypeBranch ||
88 ThisInstInfo->instructionType == kInstructionTypeCall;
Sean Callananee5dfd42010-02-01 08:49:35 +000089 else
90 return false;
91}
92
93bool EDInst::isMove() {
94 if (ThisInstInfo)
Sean Callanan8f993b82010-04-08 00:48:21 +000095 return ThisInstInfo->instructionType == kInstructionTypeMove;
Sean Callananee5dfd42010-02-01 08:49:35 +000096 else
97 return false;
98}
99
100int EDInst::parseOperands() {
101 if (ParseResult.valid())
Sean Callanan8f993b82010-04-08 00:48:21 +0000102 return ParseResult.result();
Sean Callananee5dfd42010-02-01 08:49:35 +0000103
104 if (!ThisInstInfo)
105 return ParseResult.setResult(-1);
106
107 unsigned int opIndex;
108 unsigned int mcOpIndex = 0;
109
110 for (opIndex = 0; opIndex < ThisInstInfo->numOperands; ++opIndex) {
111 if (isBranch() &&
112 (ThisInstInfo->operandFlags[opIndex] & kOperandFlagTarget)) {
113 BranchTarget = opIndex;
114 }
115 else if (isMove()) {
116 if (ThisInstInfo->operandFlags[opIndex] & kOperandFlagSource)
117 MoveSource = opIndex;
118 else if (ThisInstInfo->operandFlags[opIndex] & kOperandFlagTarget)
119 MoveTarget = opIndex;
120 }
121
122 EDOperand *operand = new EDOperand(Disassembler, *this, opIndex, mcOpIndex);
123
124 Operands.push_back(operand);
125 }
126
127 return ParseResult.setResult(0);
128}
129
130int EDInst::branchTargetID() {
131 if (parseOperands())
132 return -1;
133 return BranchTarget;
134}
135
136int EDInst::moveSourceID() {
137 if (parseOperands())
138 return -1;
139 return MoveSource;
140}
141
142int EDInst::moveTargetID() {
143 if (parseOperands())
144 return -1;
145 return MoveTarget;
146}
147
148int EDInst::numOperands() {
149 if (parseOperands())
150 return -1;
151 return Operands.size();
152}
153
154int EDInst::getOperand(EDOperand *&operand, unsigned int index) {
155 if (parseOperands())
156 return -1;
157
158 if (index >= Operands.size())
159 return -1;
160
161 operand = Operands[index];
162 return 0;
163}
164
165int EDInst::tokenize() {
166 if (TokenizeResult.valid())
167 return TokenizeResult.result();
Sean Callanan5c3a2452011-04-07 01:56:01 +0000168
169 if (ThisInstInfo == NULL)
170 return TokenizeResult.setResult(-1);
Sean Callananee5dfd42010-02-01 08:49:35 +0000171
172 if (stringify())
173 return TokenizeResult.setResult(-1);
174
175 return TokenizeResult.setResult(EDToken::tokenize(Tokens,
176 String,
177 OperandOrder,
178 Disassembler));
179
180}
181
182int EDInst::numTokens() {
183 if (tokenize())
184 return -1;
185 return Tokens.size();
186}
187
188int EDInst::getToken(EDToken *&token, unsigned int index) {
189 if (tokenize())
190 return -1;
191 token = Tokens[index];
192 return 0;
193}
194
195#ifdef __BLOCKS__
196int EDInst::visitTokens(EDTokenVisitor_t visitor) {
197 if (tokenize())
198 return -1;
199
200 tokvec_t::iterator iter;
201
202 for (iter = Tokens.begin(); iter != Tokens.end(); ++iter) {
203 int ret = visitor(*iter);
204 if (ret == 1)
205 return 0;
206 if (ret != 0)
207 return -1;
208 }
209
210 return 0;
211}
212#endif