blob: f567b5c6ddc9b1d657da85ea1009ea75454a009c [file] [log] [blame]
Brian Carlstrom7940e442013-07-12 13:46:57 -07001/*
2 * Copyright (C) 2013 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
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_
18#define ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_
Brian Carlstrom7940e442013-07-12 13:46:57 -070019
20#include "mir_to_lir.h"
21
22#include "dex/compiler_internals.h"
23
24namespace art {
25
26/* Mark a temp register as dead. Does not affect allocation state. */
27inline void Mir2Lir::ClobberBody(RegisterInfo* p) {
28 if (p->is_temp) {
29 DCHECK(!(p->live && p->dirty)) << "Live & dirty temp in clobber";
30 p->live = false;
31 p->s_reg = INVALID_SREG;
32 p->def_start = NULL;
33 p->def_end = NULL;
34 if (p->pair) {
35 p->pair = false;
buzbee56c71782013-09-05 17:13:19 -070036 p = GetRegInfo(p->partner);
37 p->pair = false;
38 p->live = false;
39 p->s_reg = INVALID_SREG;
40 p->def_start = NULL;
41 p->def_end = NULL;
Brian Carlstrom7940e442013-07-12 13:46:57 -070042 }
43 }
44}
45
buzbee0d829482013-10-11 15:24:55 -070046inline LIR* Mir2Lir::RawLIR(DexOffset dalvik_offset, int opcode, int op0,
Brian Carlstrom7940e442013-07-12 13:46:57 -070047 int op1, int op2, int op3, int op4, LIR* target) {
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -070048 LIR* insn = static_cast<LIR*>(arena_->Alloc(sizeof(LIR), ArenaAllocator::kAllocLIR));
Brian Carlstrom7940e442013-07-12 13:46:57 -070049 insn->dalvik_offset = dalvik_offset;
50 insn->opcode = opcode;
51 insn->operands[0] = op0;
52 insn->operands[1] = op1;
53 insn->operands[2] = op2;
54 insn->operands[3] = op3;
55 insn->operands[4] = op4;
56 insn->target = target;
57 SetupResourceMasks(insn);
58 if ((opcode == kPseudoTargetLabel) || (opcode == kPseudoSafepointPC) ||
59 (opcode == kPseudoExportedPC)) {
60 // Always make labels scheduling barriers
buzbeeb48819d2013-09-14 16:15:25 -070061 DCHECK(!insn->flags.use_def_invalid);
62 insn->u.m.use_mask = insn->u.m.def_mask = ENCODE_ALL;
Brian Carlstrom7940e442013-07-12 13:46:57 -070063 }
64 return insn;
65}
66
67/*
68 * The following are building blocks to construct low-level IRs with 0 - 4
69 * operands.
70 */
71inline LIR* Mir2Lir::NewLIR0(int opcode) {
buzbee409fe942013-10-11 10:49:56 -070072 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & NO_OPERAND))
Brian Carlstrom7940e442013-07-12 13:46:57 -070073 << GetTargetInstName(opcode) << " " << opcode << " "
74 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
75 << current_dalvik_offset_;
76 LIR* insn = RawLIR(current_dalvik_offset_, opcode);
77 AppendLIR(insn);
78 return insn;
79}
80
81inline LIR* Mir2Lir::NewLIR1(int opcode, int dest) {
buzbee409fe942013-10-11 10:49:56 -070082 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_UNARY_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -070083 << GetTargetInstName(opcode) << " " << opcode << " "
84 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
85 << current_dalvik_offset_;
86 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest);
87 AppendLIR(insn);
88 return insn;
89}
90
91inline LIR* Mir2Lir::NewLIR2(int opcode, int dest, int src1) {
buzbee409fe942013-10-11 10:49:56 -070092 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_BINARY_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -070093 << GetTargetInstName(opcode) << " " << opcode << " "
94 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
95 << current_dalvik_offset_;
96 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1);
97 AppendLIR(insn);
98 return insn;
99}
100
101inline LIR* Mir2Lir::NewLIR3(int opcode, int dest, int src1, int src2) {
buzbee409fe942013-10-11 10:49:56 -0700102 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_TERTIARY_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -0700103 << GetTargetInstName(opcode) << " " << opcode << " "
104 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
105 << current_dalvik_offset_;
106 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2);
107 AppendLIR(insn);
108 return insn;
109}
110
111inline LIR* Mir2Lir::NewLIR4(int opcode, int dest, int src1, int src2, int info) {
buzbee409fe942013-10-11 10:49:56 -0700112 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_QUAD_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -0700113 << GetTargetInstName(opcode) << " " << opcode << " "
114 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
115 << current_dalvik_offset_;
116 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2, info);
117 AppendLIR(insn);
118 return insn;
119}
120
121inline LIR* Mir2Lir::NewLIR5(int opcode, int dest, int src1, int src2, int info1,
122 int info2) {
buzbee409fe942013-10-11 10:49:56 -0700123 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_QUIN_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -0700124 << GetTargetInstName(opcode) << " " << opcode << " "
125 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
126 << current_dalvik_offset_;
127 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2, info1, info2);
128 AppendLIR(insn);
129 return insn;
130}
131
132/*
133 * Mark the corresponding bit(s).
134 */
135inline void Mir2Lir::SetupRegMask(uint64_t* mask, int reg) {
136 *mask |= GetRegMaskCommon(reg);
137}
138
139/*
140 * Set up the proper fields in the resource mask
141 */
142inline void Mir2Lir::SetupResourceMasks(LIR* lir) {
143 int opcode = lir->opcode;
144
buzbee409fe942013-10-11 10:49:56 -0700145 if (IsPseudoLirOp(opcode)) {
146 if (opcode != kPseudoBarrier) {
147 lir->flags.fixup = kFixupLabel;
148 }
Brian Carlstrom7940e442013-07-12 13:46:57 -0700149 return;
150 }
151
152 uint64_t flags = GetTargetInstFlags(opcode);
153
154 if (flags & NEEDS_FIXUP) {
buzbeeb48819d2013-09-14 16:15:25 -0700155 // Note: target-specific setup may specialize the fixup kind.
156 lir->flags.fixup = kFixupLabel;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700157 }
158
159 /* Get the starting size of the instruction's template */
160 lir->flags.size = GetInsnSize(lir);
buzbeeb48819d2013-09-14 16:15:25 -0700161 estimated_native_code_size_ += lir->flags.size;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700162 /* Set up the mask for resources that are updated */
163 if (flags & (IS_LOAD | IS_STORE)) {
164 /* Default to heap - will catch specialized classes later */
165 SetMemRefType(lir, flags & IS_LOAD, kHeapRef);
166 }
167
168 /*
169 * Conservatively assume the branch here will call out a function that in
170 * turn will trash everything.
171 */
172 if (flags & IS_BRANCH) {
buzbeeb48819d2013-09-14 16:15:25 -0700173 lir->u.m.def_mask = lir->u.m.use_mask = ENCODE_ALL;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700174 return;
175 }
176
177 if (flags & REG_DEF0) {
buzbeeb48819d2013-09-14 16:15:25 -0700178 SetupRegMask(&lir->u.m.def_mask, lir->operands[0]);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700179 }
180
181 if (flags & REG_DEF1) {
buzbeeb48819d2013-09-14 16:15:25 -0700182 SetupRegMask(&lir->u.m.def_mask, lir->operands[1]);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700183 }
184
buzbeeb48819d2013-09-14 16:15:25 -0700185 if (flags & REG_USE0) {
186 SetupRegMask(&lir->u.m.use_mask, lir->operands[0]);
187 }
188
189 if (flags & REG_USE1) {
190 SetupRegMask(&lir->u.m.use_mask, lir->operands[1]);
191 }
192
193 if (flags & REG_USE2) {
194 SetupRegMask(&lir->u.m.use_mask, lir->operands[2]);
195 }
196
197 if (flags & REG_USE3) {
198 SetupRegMask(&lir->u.m.use_mask, lir->operands[3]);
199 }
Brian Carlstrom7940e442013-07-12 13:46:57 -0700200
buzbee17189ac2013-11-08 11:07:02 -0800201 if (flags & REG_USE4) {
202 SetupRegMask(&lir->u.m.use_mask, lir->operands[4]);
203 }
204
Brian Carlstrom7940e442013-07-12 13:46:57 -0700205 if (flags & SETS_CCODES) {
buzbeeb48819d2013-09-14 16:15:25 -0700206 lir->u.m.def_mask |= ENCODE_CCODE;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700207 }
208
209 if (flags & USES_CCODES) {
buzbeeb48819d2013-09-14 16:15:25 -0700210 lir->u.m.use_mask |= ENCODE_CCODE;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700211 }
212
213 // Handle target-specific actions
buzbeeb48819d2013-09-14 16:15:25 -0700214 SetupTargetResourceMasks(lir, flags);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700215}
216
buzbeebd663de2013-09-10 15:41:31 -0700217inline art::Mir2Lir::RegisterInfo* Mir2Lir::GetRegInfo(int reg) {
218 DCHECK(reginfo_map_.Get(reg) != NULL);
219 return reginfo_map_.Get(reg);
220}
221
Brian Carlstrom7940e442013-07-12 13:46:57 -0700222} // namespace art
223
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700224#endif // ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_