You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
256 lines
9.6 KiB
256 lines
9.6 KiB
4 months ago
|
/*
|
||
|
* Copyright (C) 2016 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.
|
||
|
*/
|
||
|
|
||
|
#include "cha_guard_optimization.h"
|
||
|
|
||
|
namespace art {
|
||
|
|
||
|
// Note we can only do CHA guard elimination/motion in a single pass, since
|
||
|
// if a guard is not removed, another guard might be removed due to
|
||
|
// the existence of the first guard. The first guard should not be further
|
||
|
// removed in another pass. For example, due to further optimizations,
|
||
|
// a receiver of a guard might turn out to be a parameter value, or defined at
|
||
|
// a different site, which makes the guard removable as a result. However
|
||
|
// it's not safe to remove the guard in another pass since another guard might
|
||
|
// have been removed due to the existence of this guard.
|
||
|
//
|
||
|
// As a consequence, we decided not to rely on other passes to remove them
|
||
|
// (such as GVN or instruction simplifier).
|
||
|
|
||
|
class CHAGuardVisitor : HGraphVisitor {
|
||
|
public:
|
||
|
explicit CHAGuardVisitor(HGraph* graph)
|
||
|
: HGraphVisitor(graph),
|
||
|
block_has_cha_guard_(GetGraph()->GetBlocks().size(),
|
||
|
0,
|
||
|
graph->GetAllocator()->Adapter(kArenaAllocCHA)),
|
||
|
instruction_iterator_(nullptr) {
|
||
|
number_of_guards_to_visit_ = GetGraph()->GetNumberOfCHAGuards();
|
||
|
DCHECK_NE(number_of_guards_to_visit_, 0u);
|
||
|
// Will recount number of guards during guard optimization.
|
||
|
GetGraph()->SetNumberOfCHAGuards(0);
|
||
|
}
|
||
|
|
||
|
void VisitShouldDeoptimizeFlag(HShouldDeoptimizeFlag* flag) override;
|
||
|
|
||
|
void VisitBasicBlock(HBasicBlock* block) override;
|
||
|
|
||
|
private:
|
||
|
void RemoveGuard(HShouldDeoptimizeFlag* flag);
|
||
|
// Return true if `flag` is removed.
|
||
|
bool OptimizeForParameter(HShouldDeoptimizeFlag* flag, HInstruction* receiver);
|
||
|
// Return true if `flag` is removed.
|
||
|
bool OptimizeWithDominatingGuard(HShouldDeoptimizeFlag* flag, HInstruction* receiver);
|
||
|
// Return true if `flag` is hoisted.
|
||
|
bool HoistGuard(HShouldDeoptimizeFlag* flag, HInstruction* receiver);
|
||
|
|
||
|
// Record if each block has any CHA guard. It's updated during the
|
||
|
// reverse post order visit. Use int instead of bool since ArenaVector
|
||
|
// does not support bool.
|
||
|
ArenaVector<int> block_has_cha_guard_;
|
||
|
|
||
|
// The iterator that's being used for this visitor. Need it to manually
|
||
|
// advance the iterator due to removing/moving more than one instruction.
|
||
|
HInstructionIterator* instruction_iterator_;
|
||
|
|
||
|
// Used to short-circuit the pass when there is no more guards left to visit.
|
||
|
uint32_t number_of_guards_to_visit_;
|
||
|
|
||
|
DISALLOW_COPY_AND_ASSIGN(CHAGuardVisitor);
|
||
|
};
|
||
|
|
||
|
void CHAGuardVisitor::VisitBasicBlock(HBasicBlock* block) {
|
||
|
if (number_of_guards_to_visit_ == 0) {
|
||
|
return;
|
||
|
}
|
||
|
// Skip phis, just iterate through instructions.
|
||
|
HInstructionIterator it(block->GetInstructions());
|
||
|
instruction_iterator_ = ⁢
|
||
|
for (; !it.Done(); it.Advance()) {
|
||
|
DCHECK(it.Current()->IsInBlock());
|
||
|
it.Current()->Accept(this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CHAGuardVisitor::RemoveGuard(HShouldDeoptimizeFlag* flag) {
|
||
|
HBasicBlock* block = flag->GetBlock();
|
||
|
HInstruction* compare = flag->GetNext();
|
||
|
DCHECK(compare->IsNotEqual());
|
||
|
HInstruction* deopt = compare->GetNext();
|
||
|
DCHECK(deopt->IsDeoptimize());
|
||
|
|
||
|
// Advance instruction iterator first before we remove the guard.
|
||
|
// We need to do it twice since we remove three instructions and the
|
||
|
// visitor is responsible for advancing it once.
|
||
|
instruction_iterator_->Advance();
|
||
|
instruction_iterator_->Advance();
|
||
|
block->RemoveInstruction(deopt);
|
||
|
block->RemoveInstruction(compare);
|
||
|
block->RemoveInstruction(flag);
|
||
|
}
|
||
|
|
||
|
bool CHAGuardVisitor::OptimizeForParameter(HShouldDeoptimizeFlag* flag,
|
||
|
HInstruction* receiver) {
|
||
|
// If some compiled code is invalidated by CHA due to class loading, the
|
||
|
// compiled code will not be entered anymore. So the very fact that the
|
||
|
// compiled code is invoked guarantees that a parameter receiver conforms
|
||
|
// to all the CHA devirtualization assumptions made by the compiled code,
|
||
|
// since all parameter receivers pre-exist any (potential) invalidation of
|
||
|
// the compiled code.
|
||
|
//
|
||
|
// TODO: allow more cases such as a phi whose inputs are all parameters.
|
||
|
if (receiver->IsParameterValue()) {
|
||
|
RemoveGuard(flag);
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool CHAGuardVisitor::OptimizeWithDominatingGuard(HShouldDeoptimizeFlag* flag,
|
||
|
HInstruction* receiver) {
|
||
|
// If there is another guard that dominates the current guard, and
|
||
|
// that guard is dominated by receiver's definition, then the current
|
||
|
// guard can be eliminated, since receiver must pre-exist that other
|
||
|
// guard, and passing that guard guarantees that receiver conforms to
|
||
|
// all the CHA devirtualization assumptions.
|
||
|
HBasicBlock* dominator = flag->GetBlock();
|
||
|
HBasicBlock* receiver_def_block = receiver->GetBlock();
|
||
|
|
||
|
// Complexity of the following algorithm:
|
||
|
// We potentially need to traverse the full dominator chain to receiver_def_block,
|
||
|
// plus a (partial) linear search within one block for each guard.
|
||
|
// So the worst case for each guard is bounded by the size of the
|
||
|
// biggest block plus the depth of the dominating tree.
|
||
|
|
||
|
while (dominator != receiver_def_block) {
|
||
|
if (block_has_cha_guard_[dominator->GetBlockId()] == 1) {
|
||
|
RemoveGuard(flag);
|
||
|
return true;
|
||
|
}
|
||
|
dominator = dominator->GetDominator();
|
||
|
}
|
||
|
|
||
|
// At this point dominator is the block where receiver is defined.
|
||
|
// We do a linear search within dominator to see if there is a guard after
|
||
|
// receiver's definition.
|
||
|
HInstruction* instruction;
|
||
|
if (dominator == flag->GetBlock()) {
|
||
|
// Flag and receiver are defined in the same block. Search backward from
|
||
|
// the current guard.
|
||
|
instruction = flag->GetPrevious();
|
||
|
} else {
|
||
|
// Search backward from the last instruction of that dominator.
|
||
|
instruction = dominator->GetLastInstruction();
|
||
|
}
|
||
|
while (instruction != receiver) {
|
||
|
if (instruction == nullptr) {
|
||
|
// receiver must be defined in this block, we didn't find it
|
||
|
// in the instruction list, so it must be a Phi.
|
||
|
DCHECK(receiver->IsPhi());
|
||
|
break;
|
||
|
}
|
||
|
if (instruction->IsShouldDeoptimizeFlag()) {
|
||
|
RemoveGuard(flag);
|
||
|
return true;
|
||
|
}
|
||
|
instruction = instruction->GetPrevious();
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool CHAGuardVisitor::HoistGuard(HShouldDeoptimizeFlag* flag,
|
||
|
HInstruction* receiver) {
|
||
|
// If receiver is loop invariant, we can hoist the guard out of the
|
||
|
// loop since passing a guard before entering the loop guarantees that
|
||
|
// receiver conforms to all the CHA devirtualization assumptions.
|
||
|
// We only hoist guards out of the inner loop since that offers most of the
|
||
|
// benefit and it might help remove other guards in the inner loop.
|
||
|
HBasicBlock* block = flag->GetBlock();
|
||
|
HLoopInformation* loop_info = block->GetLoopInformation();
|
||
|
if (loop_info != nullptr &&
|
||
|
!loop_info->IsIrreducible() &&
|
||
|
loop_info->IsDefinedOutOfTheLoop(receiver)) {
|
||
|
HInstruction* compare = flag->GetNext();
|
||
|
DCHECK(compare->IsNotEqual());
|
||
|
HInstruction* deopt = compare->GetNext();
|
||
|
DCHECK(deopt->IsDeoptimize());
|
||
|
|
||
|
// Advance instruction iterator first before we move the guard.
|
||
|
// We need to do it twice since we move three instructions and the
|
||
|
// visitor is responsible for advancing it once.
|
||
|
instruction_iterator_->Advance();
|
||
|
instruction_iterator_->Advance();
|
||
|
|
||
|
HBasicBlock* pre_header = loop_info->GetPreHeader();
|
||
|
flag->MoveBefore(pre_header->GetLastInstruction());
|
||
|
compare->MoveBefore(pre_header->GetLastInstruction());
|
||
|
|
||
|
block->RemoveInstruction(deopt);
|
||
|
HInstruction* suspend = loop_info->GetSuspendCheck();
|
||
|
// Need a new deoptimize instruction that copies the environment
|
||
|
// of the suspend instruction for the loop.
|
||
|
HDeoptimize* deoptimize = new (GetGraph()->GetAllocator()) HDeoptimize(
|
||
|
GetGraph()->GetAllocator(), compare, DeoptimizationKind::kCHA, suspend->GetDexPc());
|
||
|
pre_header->InsertInstructionBefore(deoptimize, pre_header->GetLastInstruction());
|
||
|
deoptimize->CopyEnvironmentFromWithLoopPhiAdjustment(
|
||
|
suspend->GetEnvironment(), loop_info->GetHeader());
|
||
|
block_has_cha_guard_[pre_header->GetBlockId()] = 1;
|
||
|
GetGraph()->IncrementNumberOfCHAGuards();
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
void CHAGuardVisitor::VisitShouldDeoptimizeFlag(HShouldDeoptimizeFlag* flag) {
|
||
|
number_of_guards_to_visit_--;
|
||
|
HInstruction* receiver = flag->InputAt(0);
|
||
|
// Don't need the receiver anymore.
|
||
|
flag->RemoveInputAt(0);
|
||
|
if (receiver->IsNullCheck()) {
|
||
|
receiver = receiver->InputAt(0);
|
||
|
}
|
||
|
|
||
|
if (OptimizeForParameter(flag, receiver)) {
|
||
|
DCHECK(!flag->IsInBlock());
|
||
|
return;
|
||
|
}
|
||
|
if (OptimizeWithDominatingGuard(flag, receiver)) {
|
||
|
DCHECK(!flag->IsInBlock());
|
||
|
return;
|
||
|
}
|
||
|
if (HoistGuard(flag, receiver)) {
|
||
|
DCHECK(flag->IsInBlock());
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Need to keep the CHA guard in place.
|
||
|
block_has_cha_guard_[flag->GetBlock()->GetBlockId()] = 1;
|
||
|
GetGraph()->IncrementNumberOfCHAGuards();
|
||
|
}
|
||
|
|
||
|
bool CHAGuardOptimization::Run() {
|
||
|
if (graph_->GetNumberOfCHAGuards() == 0) {
|
||
|
return false;
|
||
|
}
|
||
|
CHAGuardVisitor visitor(graph_);
|
||
|
for (HBasicBlock* block : graph_->GetReversePostOrder()) {
|
||
|
visitor.VisitBasicBlock(block);
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
} // namespace art
|