blob: 9a62255f5daf7ce4b200d0db02dabd63ccdd3bac [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) {
buzbeeba574512014-05-12 15:13:16 -070028 DCHECK(p->IsTemp());
buzbee082833c2014-05-17 23:16:26 -070029 if (p->SReg() != INVALID_SREG) {
buzbee091cc402014-03-31 10:14:40 -070030 DCHECK(!(p->IsLive() && p->IsDirty())) << "Live & dirty temp in clobber";
buzbee30adc732014-05-09 15:10:18 -070031 p->MarkDead();
buzbee091cc402014-03-31 10:14:40 -070032 if (p->IsWide()) {
33 p->SetIsWide(false);
34 if (p->GetReg() != p->Partner()) {
35 // Register pair - deal with the other half.
36 p = GetRegInfo(p->Partner());
37 p->SetIsWide(false);
buzbee30adc732014-05-09 15:10:18 -070038 p->MarkDead();
buzbee091cc402014-03-31 10:14:40 -070039 }
Brian Carlstrom7940e442013-07-12 13:46:57 -070040 }
41 }
42}
43
buzbee0d829482013-10-11 15:24:55 -070044inline LIR* Mir2Lir::RawLIR(DexOffset dalvik_offset, int opcode, int op0,
Brian Carlstrom7940e442013-07-12 13:46:57 -070045 int op1, int op2, int op3, int op4, LIR* target) {
Vladimir Marko83cc7ae2014-02-12 18:02:05 +000046 LIR* insn = static_cast<LIR*>(arena_->Alloc(sizeof(LIR), kArenaAllocLIR));
Brian Carlstrom7940e442013-07-12 13:46:57 -070047 insn->dalvik_offset = dalvik_offset;
48 insn->opcode = opcode;
49 insn->operands[0] = op0;
50 insn->operands[1] = op1;
51 insn->operands[2] = op2;
52 insn->operands[3] = op3;
53 insn->operands[4] = op4;
54 insn->target = target;
55 SetupResourceMasks(insn);
56 if ((opcode == kPseudoTargetLabel) || (opcode == kPseudoSafepointPC) ||
57 (opcode == kPseudoExportedPC)) {
58 // Always make labels scheduling barriers
buzbeeb48819d2013-09-14 16:15:25 -070059 DCHECK(!insn->flags.use_def_invalid);
Vladimir Marko8dea81c2014-06-06 14:50:36 +010060 insn->u.m.use_mask = insn->u.m.def_mask = &kEncodeAll;
Brian Carlstrom7940e442013-07-12 13:46:57 -070061 }
62 return insn;
63}
64
65/*
66 * The following are building blocks to construct low-level IRs with 0 - 4
67 * operands.
68 */
69inline LIR* Mir2Lir::NewLIR0(int opcode) {
buzbee409fe942013-10-11 10:49:56 -070070 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & NO_OPERAND))
Brian Carlstrom7940e442013-07-12 13:46:57 -070071 << GetTargetInstName(opcode) << " " << opcode << " "
72 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
73 << current_dalvik_offset_;
74 LIR* insn = RawLIR(current_dalvik_offset_, opcode);
75 AppendLIR(insn);
76 return insn;
77}
78
79inline LIR* Mir2Lir::NewLIR1(int opcode, int dest) {
buzbee409fe942013-10-11 10:49:56 -070080 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_UNARY_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -070081 << GetTargetInstName(opcode) << " " << opcode << " "
82 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
83 << current_dalvik_offset_;
84 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest);
85 AppendLIR(insn);
86 return insn;
87}
88
89inline LIR* Mir2Lir::NewLIR2(int opcode, int dest, int src1) {
buzbee409fe942013-10-11 10:49:56 -070090 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_BINARY_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -070091 << GetTargetInstName(opcode) << " " << opcode << " "
92 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
93 << current_dalvik_offset_;
94 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1);
95 AppendLIR(insn);
96 return insn;
97}
98
Razvan A Lupusoru614c2b42014-01-28 17:05:21 -080099inline LIR* Mir2Lir::NewLIR2NoDest(int opcode, int src, int info) {
100 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_UNARY_OP))
101 << GetTargetInstName(opcode) << " " << opcode << " "
102 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
103 << current_dalvik_offset_;
104 LIR* insn = RawLIR(current_dalvik_offset_, opcode, src, info);
105 AppendLIR(insn);
106 return insn;
107}
108
Brian Carlstrom7940e442013-07-12 13:46:57 -0700109inline LIR* Mir2Lir::NewLIR3(int opcode, int dest, int src1, int src2) {
buzbee409fe942013-10-11 10:49:56 -0700110 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_TERTIARY_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -0700111 << GetTargetInstName(opcode) << " " << opcode << " "
112 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
113 << current_dalvik_offset_;
114 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2);
115 AppendLIR(insn);
116 return insn;
117}
118
119inline LIR* Mir2Lir::NewLIR4(int opcode, int dest, int src1, int src2, int info) {
buzbee409fe942013-10-11 10:49:56 -0700120 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_QUAD_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -0700121 << GetTargetInstName(opcode) << " " << opcode << " "
122 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
123 << current_dalvik_offset_;
124 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2, info);
125 AppendLIR(insn);
126 return insn;
127}
128
129inline LIR* Mir2Lir::NewLIR5(int opcode, int dest, int src1, int src2, int info1,
130 int info2) {
buzbee409fe942013-10-11 10:49:56 -0700131 DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_QUIN_OP))
Brian Carlstrom7940e442013-07-12 13:46:57 -0700132 << GetTargetInstName(opcode) << " " << opcode << " "
133 << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " "
134 << current_dalvik_offset_;
135 LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2, info1, info2);
136 AppendLIR(insn);
137 return insn;
138}
139
140/*
141 * Mark the corresponding bit(s).
142 */
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100143inline void Mir2Lir::SetupRegMask(ResourceMask* mask, int reg) {
buzbee091cc402014-03-31 10:14:40 -0700144 DCHECK_EQ((reg & ~RegStorage::kRegValMask), 0);
145 DCHECK(reginfo_map_.Get(reg) != nullptr) << "No info for 0x" << reg;
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100146 *mask = mask->Union(reginfo_map_.Get(reg)->DefUseMask());
Brian Carlstrom7940e442013-07-12 13:46:57 -0700147}
148
149/*
150 * Set up the proper fields in the resource mask
151 */
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100152inline void Mir2Lir::SetupResourceMasks(LIR* lir) {
Brian Carlstrom7940e442013-07-12 13:46:57 -0700153 int opcode = lir->opcode;
154
buzbee409fe942013-10-11 10:49:56 -0700155 if (IsPseudoLirOp(opcode)) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100156 lir->u.m.use_mask = lir->u.m.def_mask = &kEncodeNone;
buzbee409fe942013-10-11 10:49:56 -0700157 if (opcode != kPseudoBarrier) {
158 lir->flags.fixup = kFixupLabel;
159 }
Brian Carlstrom7940e442013-07-12 13:46:57 -0700160 return;
161 }
162
163 uint64_t flags = GetTargetInstFlags(opcode);
164
165 if (flags & NEEDS_FIXUP) {
buzbeeb48819d2013-09-14 16:15:25 -0700166 // Note: target-specific setup may specialize the fixup kind.
167 lir->flags.fixup = kFixupLabel;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700168 }
169
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100170 /* Get the starting size of the instruction's template. */
Brian Carlstrom7940e442013-07-12 13:46:57 -0700171 lir->flags.size = GetInsnSize(lir);
buzbeeb48819d2013-09-14 16:15:25 -0700172 estimated_native_code_size_ += lir->flags.size;
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100173
174 /* Set up the mask for resources. */
175 ResourceMask use_mask;
176 ResourceMask def_mask;
177
178 if (flags & (IS_LOAD | IS_STORE)) {
179 /* Set memory reference type (defaults to heap, overridden by ScopedMemRefType). */
180 if (flags & IS_LOAD) {
181 use_mask.SetBit(mem_ref_type_);
182 } else {
183 /* Currently only loads can be marked as kMustNotAlias. */
184 DCHECK(mem_ref_type_ != ResourceMask::kMustNotAlias);
185 }
186 if (flags & IS_STORE) {
187 /* Literals cannot be written to. */
188 DCHECK(mem_ref_type_ != ResourceMask::kLiteral);
189 def_mask.SetBit(mem_ref_type_);
190 }
Brian Carlstrom7940e442013-07-12 13:46:57 -0700191 }
192
193 /*
194 * Conservatively assume the branch here will call out a function that in
195 * turn will trash everything.
196 */
197 if (flags & IS_BRANCH) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100198 lir->u.m.def_mask = lir->u.m.use_mask = &kEncodeAll;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700199 return;
200 }
201
202 if (flags & REG_DEF0) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100203 SetupRegMask(&def_mask, lir->operands[0]);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700204 }
205
206 if (flags & REG_DEF1) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100207 SetupRegMask(&def_mask, lir->operands[1]);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700208 }
209
Razvan A Lupusoru99ad7232014-02-25 17:41:08 -0800210 if (flags & REG_DEF2) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100211 SetupRegMask(&def_mask, lir->operands[2]);
Razvan A Lupusoru99ad7232014-02-25 17:41:08 -0800212 }
213
buzbeeb48819d2013-09-14 16:15:25 -0700214 if (flags & REG_USE0) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100215 SetupRegMask(&use_mask, lir->operands[0]);
buzbeeb48819d2013-09-14 16:15:25 -0700216 }
217
218 if (flags & REG_USE1) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100219 SetupRegMask(&use_mask, lir->operands[1]);
buzbeeb48819d2013-09-14 16:15:25 -0700220 }
221
222 if (flags & REG_USE2) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100223 SetupRegMask(&use_mask, lir->operands[2]);
buzbeeb48819d2013-09-14 16:15:25 -0700224 }
225
226 if (flags & REG_USE3) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100227 SetupRegMask(&use_mask, lir->operands[3]);
buzbeeb48819d2013-09-14 16:15:25 -0700228 }
Brian Carlstrom7940e442013-07-12 13:46:57 -0700229
buzbee17189ac2013-11-08 11:07:02 -0800230 if (flags & REG_USE4) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100231 SetupRegMask(&use_mask, lir->operands[4]);
buzbee17189ac2013-11-08 11:07:02 -0800232 }
233
Brian Carlstrom7940e442013-07-12 13:46:57 -0700234 if (flags & SETS_CCODES) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100235 def_mask.SetBit(ResourceMask::kCCode);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700236 }
237
238 if (flags & USES_CCODES) {
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100239 use_mask.SetBit(ResourceMask::kCCode);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700240 }
241
242 // Handle target-specific actions
Vladimir Marko8dea81c2014-06-06 14:50:36 +0100243 SetupTargetResourceMasks(lir, flags, &def_mask, &use_mask);
244
245 lir->u.m.use_mask = mask_cache_.GetMask(use_mask);
246 lir->u.m.def_mask = mask_cache_.GetMask(def_mask);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700247}
248
buzbee091cc402014-03-31 10:14:40 -0700249inline art::Mir2Lir::RegisterInfo* Mir2Lir::GetRegInfo(RegStorage reg) {
250 RegisterInfo* res = reg.IsPair() ? reginfo_map_.Get(reg.GetLowReg()) :
251 reginfo_map_.Get(reg.GetReg());
252 DCHECK(res != nullptr);
253 return res;
buzbeebd663de2013-09-10 15:41:31 -0700254}
255
Andreas Gampe4b537a82014-06-30 22:24:53 -0700256inline void Mir2Lir::CheckRegLocation(RegLocation rl) const {
257 if (kFailOnSizeError || kReportSizeError) {
258 CheckRegLocationImpl(rl, kFailOnSizeError, kReportSizeError);
259 }
260}
261
262inline void Mir2Lir::CheckRegStorage(RegStorage rs, WidenessCheck wide, RefCheck ref, FPCheck fp)
263 const {
264 if (kFailOnSizeError || kReportSizeError) {
265 CheckRegStorageImpl(rs, wide, ref, fp, kFailOnSizeError, kReportSizeError);
266 }
267}
268
Brian Carlstrom7940e442013-07-12 13:46:57 -0700269} // namespace art
270
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700271#endif // ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_