blob: f96816c3d947a23c844632d9a4a2c30eb9a73a77 [file] [log] [blame]
/*
* 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"
#include "gc_root.h"
#include "utils.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 size_t Mir2Lir::GetCacheOffset(uint32_t index) {
return sizeof(GcRoot<mirror::Object>) * index;
}
inline size_t Mir2Lir::GetCachePointerOffset(uint32_t index, size_t pointer_size) {
return pointer_size * index;
}
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_