| /* |
| * Copyright (C) 2013 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_ |
| #define ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_ |
| |
| #include "mir_to_lir.h" |
| |
| #include "base/logging.h" |
| #include "dex/compiler_ir.h" |
| |
| namespace art { |
| |
| /* Mark a temp register as dead. Does not affect allocation state. */ |
| inline void Mir2Lir::ClobberBody(RegisterInfo* p) { |
| DCHECK(p->IsTemp()); |
| if (p->SReg() != INVALID_SREG) { |
| DCHECK(!(p->IsLive() && p->IsDirty())) << "Live & dirty temp in clobber"; |
| p->MarkDead(); |
| if (p->IsWide()) { |
| p->SetIsWide(false); |
| if (p->GetReg().NotExactlyEquals(p->Partner())) { |
| // Register pair - deal with the other half. |
| p = GetRegInfo(p->Partner()); |
| p->SetIsWide(false); |
| p->MarkDead(); |
| } |
| } |
| } |
| } |
| |
| inline LIR* Mir2Lir::RawLIR(DexOffset dalvik_offset, int opcode, int op0, |
| int op1, int op2, int op3, int op4, LIR* target) { |
| LIR* insn = static_cast<LIR*>(arena_->Alloc(sizeof(LIR), kArenaAllocLIR)); |
| insn->dalvik_offset = dalvik_offset; |
| insn->opcode = opcode; |
| insn->operands[0] = op0; |
| insn->operands[1] = op1; |
| insn->operands[2] = op2; |
| insn->operands[3] = op3; |
| insn->operands[4] = op4; |
| insn->target = target; |
| SetupResourceMasks(insn); |
| if ((opcode == kPseudoTargetLabel) || (opcode == kPseudoSafepointPC) || |
| (opcode == kPseudoExportedPC)) { |
| // Always make labels scheduling barriers |
| DCHECK(!insn->flags.use_def_invalid); |
| insn->u.m.use_mask = insn->u.m.def_mask = &kEncodeAll; |
| } |
| return insn; |
| } |
| |
| /* |
| * The following are building blocks to construct low-level IRs with 0 - 4 |
| * operands. |
| */ |
| inline LIR* Mir2Lir::NewLIR0(int opcode) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & NO_OPERAND)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| inline LIR* Mir2Lir::NewLIR1(int opcode, int dest) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_UNARY_OP)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| inline LIR* Mir2Lir::NewLIR2(int opcode, int dest, int src1) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_BINARY_OP)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| inline LIR* Mir2Lir::NewLIR2NoDest(int opcode, int src, int info) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_BINARY_OP)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode, src, info); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| inline LIR* Mir2Lir::NewLIR3(int opcode, int dest, int src1, int src2) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_TERTIARY_OP)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| inline LIR* Mir2Lir::NewLIR4(int opcode, int dest, int src1, int src2, int info) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_QUAD_OP)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2, info); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| inline LIR* Mir2Lir::NewLIR5(int opcode, int dest, int src1, int src2, int info1, |
| int info2) { |
| DCHECK(IsPseudoLirOp(opcode) || (GetTargetInstFlags(opcode) & IS_QUIN_OP)) |
| << GetTargetInstName(opcode) << " " << opcode << " " |
| << PrettyMethod(cu_->method_idx, *cu_->dex_file) << " " |
| << current_dalvik_offset_; |
| LIR* insn = RawLIR(current_dalvik_offset_, opcode, dest, src1, src2, info1, info2); |
| AppendLIR(insn); |
| return insn; |
| } |
| |
| /* |
| * Mark the corresponding bit(s). |
| */ |
| inline void Mir2Lir::SetupRegMask(ResourceMask* mask, int reg) { |
| DCHECK_EQ((reg & ~RegStorage::kRegValMask), 0); |
| DCHECK_LT(static_cast<size_t>(reg), reginfo_map_.size()); |
| DCHECK(reginfo_map_[reg] != nullptr) << "No info for 0x" << reg; |
| *mask = mask->Union(reginfo_map_[reg]->DefUseMask()); |
| } |
| |
| /* |
| * Clear the corresponding bit(s). |
| */ |
| inline void Mir2Lir::ClearRegMask(ResourceMask* mask, int reg) { |
| DCHECK_EQ((reg & ~RegStorage::kRegValMask), 0); |
| DCHECK_LT(static_cast<size_t>(reg), reginfo_map_.size()); |
| DCHECK(reginfo_map_[reg] != nullptr) << "No info for 0x" << reg; |
| *mask = mask->ClearBits(reginfo_map_[reg]->DefUseMask()); |
| } |
| |
| /* |
| * Set up the proper fields in the resource mask |
| */ |
| inline void Mir2Lir::SetupResourceMasks(LIR* lir) { |
| int opcode = lir->opcode; |
| |
| if (IsPseudoLirOp(opcode)) { |
| lir->u.m.use_mask = lir->u.m.def_mask = &kEncodeNone; |
| if (opcode != kPseudoBarrier) { |
| lir->flags.fixup = kFixupLabel; |
| } |
| return; |
| } |
| |
| uint64_t flags = GetTargetInstFlags(opcode); |
| |
| if (flags & NEEDS_FIXUP) { |
| // Note: target-specific setup may specialize the fixup kind. |
| lir->flags.fixup = kFixupLabel; |
| } |
| |
| /* Get the starting size of the instruction's template. */ |
| lir->flags.size = GetInsnSize(lir); |
| estimated_native_code_size_ += lir->flags.size; |
| |
| /* Set up the mask for resources. */ |
| ResourceMask use_mask; |
| ResourceMask def_mask; |
| |
| if (flags & (IS_LOAD | IS_STORE)) { |
| /* Set memory reference type (defaults to heap, overridden by ScopedMemRefType). */ |
| if (flags & IS_LOAD) { |
| use_mask.SetBit(mem_ref_type_); |
| } else { |
| /* Currently only loads can be marked as kMustNotAlias. */ |
| DCHECK(mem_ref_type_ != ResourceMask::kMustNotAlias); |
| } |
| if (flags & IS_STORE) { |
| /* Literals cannot be written to. */ |
| DCHECK(mem_ref_type_ != ResourceMask::kLiteral); |
| def_mask.SetBit(mem_ref_type_); |
| } |
| } |
| |
| /* |
| * Conservatively assume the branch here will call out a function that in |
| * turn will trash everything. |
| */ |
| if (flags & IS_BRANCH) { |
| lir->u.m.def_mask = lir->u.m.use_mask = &kEncodeAll; |
| return; |
| } |
| |
| if (flags & REG_DEF0) { |
| SetupRegMask(&def_mask, lir->operands[0]); |
| } |
| |
| if (flags & REG_DEF1) { |
| SetupRegMask(&def_mask, lir->operands[1]); |
| } |
| |
| if (flags & REG_DEF2) { |
| SetupRegMask(&def_mask, lir->operands[2]); |
| } |
| |
| if (flags & REG_USE0) { |
| SetupRegMask(&use_mask, lir->operands[0]); |
| } |
| |
| if (flags & REG_USE1) { |
| SetupRegMask(&use_mask, lir->operands[1]); |
| } |
| |
| if (flags & REG_USE2) { |
| SetupRegMask(&use_mask, lir->operands[2]); |
| } |
| |
| if (flags & REG_USE3) { |
| SetupRegMask(&use_mask, lir->operands[3]); |
| } |
| |
| if (flags & REG_USE4) { |
| SetupRegMask(&use_mask, lir->operands[4]); |
| } |
| |
| if (flags & SETS_CCODES) { |
| def_mask.SetBit(ResourceMask::kCCode); |
| } |
| |
| if (flags & USES_CCODES) { |
| use_mask.SetBit(ResourceMask::kCCode); |
| } |
| |
| // Handle target-specific actions |
| SetupTargetResourceMasks(lir, flags, &use_mask, &def_mask); |
| |
| lir->u.m.use_mask = mask_cache_.GetMask(use_mask); |
| lir->u.m.def_mask = mask_cache_.GetMask(def_mask); |
| } |
| |
| inline art::Mir2Lir::RegisterInfo* Mir2Lir::GetRegInfo(RegStorage reg) { |
| RegisterInfo* res = reg.IsPair() ? reginfo_map_[reg.GetLowReg()] : reginfo_map_[reg.GetReg()]; |
| DCHECK(res != nullptr); |
| return res; |
| } |
| |
| inline void Mir2Lir::CheckRegLocation(RegLocation rl) const { |
| if (kFailOnSizeError || kReportSizeError) { |
| CheckRegLocationImpl(rl, kFailOnSizeError, kReportSizeError); |
| } |
| } |
| |
| inline void Mir2Lir::CheckRegStorage(RegStorage rs, WidenessCheck wide, RefCheck ref, FPCheck fp) |
| const { |
| if (kFailOnSizeError || kReportSizeError) { |
| CheckRegStorageImpl(rs, wide, ref, fp, kFailOnSizeError, kReportSizeError); |
| } |
| } |
| |
| inline Mir2Lir::ShortyIterator::ShortyIterator(const char* shorty, bool is_static) |
| : cur_(shorty + 1), pending_this_(!is_static), initialized_(false) { |
| DCHECK(shorty != nullptr); |
| DCHECK_NE(*shorty, 0); |
| } |
| |
| inline bool Mir2Lir::ShortyIterator::Next() { |
| if (!initialized_) { |
| initialized_ = true; |
| } else if (pending_this_) { |
| pending_this_ = false; |
| } else if (*cur_ != 0) { |
| cur_++; |
| } |
| |
| return *cur_ != 0 || pending_this_; |
| } |
| |
| } // namespace art |
| |
| #endif // ART_COMPILER_DEX_QUICK_MIR_TO_LIR_INL_H_ |