blob: cb448c883f4474f72c20375a7f9d23d218aa8030 [file] [log] [blame]
/*
* Copyright (C) 2014 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 "gvn.h"
#include "side_effects_analysis.h"
namespace art {
/**
* A node in the collision list of a ValueSet. Encodes the instruction,
* the hash code, and the next node in the collision list.
*/
class ValueSetNode : public ArenaObject<kArenaAllocMisc> {
public:
ValueSetNode(HInstruction* instruction, size_t hash_code, ValueSetNode* next)
: instruction_(instruction), hash_code_(hash_code), next_(next) {}
size_t GetHashCode() const { return hash_code_; }
HInstruction* GetInstruction() const { return instruction_; }
ValueSetNode* GetNext() const { return next_; }
void SetNext(ValueSetNode* node) { next_ = node; }
private:
HInstruction* const instruction_;
const size_t hash_code_;
ValueSetNode* next_;
DISALLOW_COPY_AND_ASSIGN(ValueSetNode);
};
/**
* A ValueSet holds instructions that can replace other instructions. It is updated
* through the `Add` method, and the `Kill` method. The `Kill` method removes
* instructions that are affected by the given side effect.
*
* The `Lookup` method returns an equivalent instruction to the given instruction
* if there is one in the set. In GVN, we would say those instructions have the
* same "number".
*/
class ValueSet : public ArenaObject<kArenaAllocMisc> {
public:
explicit ValueSet(ArenaAllocator* allocator)
: allocator_(allocator), number_of_entries_(0), collisions_(nullptr) {
for (size_t i = 0; i < kDefaultNumberOfEntries; ++i) {
table_[i] = nullptr;
}
}
// Adds an instruction in the set.
void Add(HInstruction* instruction) {
DCHECK(Lookup(instruction) == nullptr);
size_t hash_code = instruction->ComputeHashCode();
size_t index = hash_code % kDefaultNumberOfEntries;
if (table_[index] == nullptr) {
table_[index] = instruction;
} else {
collisions_ = new (allocator_) ValueSetNode(instruction, hash_code, collisions_);
}
++number_of_entries_;
}
// If in the set, returns an equivalent instruction to the given instruction. Returns
// null otherwise.
HInstruction* Lookup(HInstruction* instruction) const {
size_t hash_code = instruction->ComputeHashCode();
size_t index = hash_code % kDefaultNumberOfEntries;
HInstruction* existing = table_[index];
if (existing != nullptr && existing->Equals(instruction)) {
return existing;
}
for (ValueSetNode* node = collisions_; node != nullptr; node = node->GetNext()) {
if (node->GetHashCode() == hash_code) {
existing = node->GetInstruction();
if (existing->Equals(instruction)) {
return existing;
}
}
}
return nullptr;
}
// Returns whether `instruction` is in the set.
HInstruction* IdentityLookup(HInstruction* instruction) const {
size_t hash_code = instruction->ComputeHashCode();
size_t index = hash_code % kDefaultNumberOfEntries;
HInstruction* existing = table_[index];
if (existing != nullptr && existing == instruction) {
return existing;
}
for (ValueSetNode* node = collisions_; node != nullptr; node = node->GetNext()) {
if (node->GetHashCode() == hash_code) {
existing = node->GetInstruction();
if (existing == instruction) {
return existing;
}
}
}
return nullptr;
}
// Removes all instructions in the set that are affected by the given side effects.
void Kill(SideEffects side_effects) {
for (size_t i = 0; i < kDefaultNumberOfEntries; ++i) {
HInstruction* instruction = table_[i];
if (instruction != nullptr && instruction->GetSideEffects().DependsOn(side_effects)) {
table_[i] = nullptr;
--number_of_entries_;
}
}
for (ValueSetNode* current = collisions_, *previous = nullptr;
current != nullptr;
current = current->GetNext()) {
HInstruction* instruction = current->GetInstruction();
if (instruction->GetSideEffects().DependsOn(side_effects)) {
if (previous == nullptr) {
collisions_ = current->GetNext();
} else {
previous->SetNext(current->GetNext());
}
--number_of_entries_;
} else {
previous = current;
}
}
}
// Returns a copy of this set.
ValueSet* Copy() const {
ValueSet* copy = new (allocator_) ValueSet(allocator_);
for (size_t i = 0; i < kDefaultNumberOfEntries; ++i) {
copy->table_[i] = table_[i];
}
// Note that the order will be inverted in the copy. This is fine, as the order is not
// relevant for a ValueSet.
for (ValueSetNode* node = collisions_; node != nullptr; node = node->GetNext()) {
copy->collisions_ = new (allocator_) ValueSetNode(
node->GetInstruction(), node->GetHashCode(), copy->collisions_);
}
copy->number_of_entries_ = number_of_entries_;
return copy;
}
void Clear() {
number_of_entries_ = 0;
collisions_ = nullptr;
for (size_t i = 0; i < kDefaultNumberOfEntries; ++i) {
table_[i] = nullptr;
}
}
// Update this `ValueSet` by intersecting with instructions in `other`.
void IntersectionWith(ValueSet* other) {
if (IsEmpty()) {
return;
} else if (other->IsEmpty()) {
Clear();
} else {
for (size_t i = 0; i < kDefaultNumberOfEntries; ++i) {
if (table_[i] != nullptr && other->IdentityLookup(table_[i]) == nullptr) {
--number_of_entries_;
table_[i] = nullptr;
}
}
for (ValueSetNode* current = collisions_, *previous = nullptr;
current != nullptr;
current = current->GetNext()) {
if (other->IdentityLookup(current->GetInstruction()) == nullptr) {
if (previous == nullptr) {
collisions_ = current->GetNext();
} else {
previous->SetNext(current->GetNext());
}
--number_of_entries_;
} else {
previous = current;
}
}
}
}
bool IsEmpty() const { return number_of_entries_ == 0; }
size_t GetNumberOfEntries() const { return number_of_entries_; }
private:
static constexpr size_t kDefaultNumberOfEntries = 8;
ArenaAllocator* const allocator_;
// The number of entries in the set.
size_t number_of_entries_;
// The internal implementation of the set. It uses a combination of a hash code based
// fixed-size list, and a linked list to handle hash code collisions.
// TODO: Tune the fixed size list original size, and support growing it.
ValueSetNode* collisions_;
HInstruction* table_[kDefaultNumberOfEntries];
DISALLOW_COPY_AND_ASSIGN(ValueSet);
};
/**
* Optimization phase that removes redundant instruction.
*/
class GlobalValueNumberer : public ValueObject {
public:
GlobalValueNumberer(ArenaAllocator* allocator,
HGraph* graph,
const SideEffectsAnalysis& side_effects)
: graph_(graph),
allocator_(allocator),
side_effects_(side_effects),
sets_(allocator, graph->GetBlocks().Size(), nullptr) {}
void Run();
private:
// Per-block GVN. Will also update the ValueSet of the dominated and
// successor blocks.
void VisitBasicBlock(HBasicBlock* block);
HGraph* graph_;
ArenaAllocator* const allocator_;
const SideEffectsAnalysis& side_effects_;
// ValueSet for blocks. Initially null, but for an individual block they
// are allocated and populated by the dominator, and updated by all blocks
// in the path from the dominator to the block.
GrowableArray<ValueSet*> sets_;
DISALLOW_COPY_AND_ASSIGN(GlobalValueNumberer);
};
void GlobalValueNumberer::Run() {
DCHECK(side_effects_.HasRun());
sets_.Put(graph_->GetEntryBlock()->GetBlockId(), new (allocator_) ValueSet(allocator_));
// Use the reverse post order to ensure the non back-edge predecessors of a block are
// visited before the block itself.
for (HReversePostOrderIterator it(*graph_); !it.Done(); it.Advance()) {
VisitBasicBlock(it.Current());
}
}
void GlobalValueNumberer::VisitBasicBlock(HBasicBlock* block) {
ValueSet* set = nullptr;
const GrowableArray<HBasicBlock*>& predecessors = block->GetPredecessors();
if (predecessors.Size() == 0 || predecessors.Get(0)->IsEntryBlock()) {
// The entry block should only accumulate constant instructions, and
// the builder puts constants only in the entry block.
// Therefore, there is no need to propagate the value set to the next block.
set = new (allocator_) ValueSet(allocator_);
} else {
HBasicBlock* dominator = block->GetDominator();
set = sets_.Get(dominator->GetBlockId());
if (dominator->GetSuccessors().Size() != 1 || dominator->GetSuccessors().Get(0) != block) {
// We have to copy if the dominator has other successors, or `block` is not a successor
// of the dominator.
set = set->Copy();
}
if (!set->IsEmpty()) {
if (block->IsLoopHeader()) {
DCHECK_EQ(block->GetDominator(), block->GetLoopInformation()->GetPreHeader());
set->Kill(side_effects_.GetLoopEffects(block));
} else if (predecessors.Size() > 1) {
for (size_t i = 0, e = predecessors.Size(); i < e; ++i) {
set->IntersectionWith(sets_.Get(predecessors.Get(i)->GetBlockId()));
if (set->IsEmpty()) {
break;
}
}
}
}
}
sets_.Put(block->GetBlockId(), set);
HInstruction* current = block->GetFirstInstruction();
while (current != nullptr) {
set->Kill(current->GetSideEffects());
// Save the next instruction in case `current` is removed from the graph.
HInstruction* next = current->GetNext();
if (current->CanBeMoved()) {
HInstruction* existing = set->Lookup(current);
if (existing != nullptr) {
current->ReplaceWith(existing);
current->GetBlock()->RemoveInstruction(current);
} else {
set->Add(current);
}
}
current = next;
}
}
void GVNOptimization::Run() {
GlobalValueNumberer gvn(graph_->GetArena(), graph_, side_effects_);
gvn.Run();
}
} // namespace art