/*
* 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.
*/
#ifndef ART_COMPILER_OPTIMIZING_PRETTY_PRINTER_H_
#define ART_COMPILER_OPTIMIZING_PRETTY_PRINTER_H_
#include "android-base/stringprintf.h"
#include "nodes.h"
namespace art {
class HPrettyPrinter : public HGraphVisitor {
public:
explicit HPrettyPrinter(HGraph* graph) : HGraphVisitor(graph) { }
void PrintPreInstruction(HInstruction* instruction) {
PrintString(" ");
PrintInt(instruction->GetId());
PrintString(": ");
}
void VisitInstruction(HInstruction* instruction) override {
PrintPreInstruction(instruction);
PrintString(instruction->DebugName());
PrintPostInstruction(instruction);
}
void PrintPostInstruction(HInstruction* instruction) {
HConstInputsRef inputs = instruction->GetInputs();
if (!inputs.empty()) {
PrintString("(");
bool first = true;
for (const HInstruction* input : inputs) {
if (first) {
first = false;
} else {
PrintString(", ");
}
PrintInt(input->GetId());
}
PrintString(")");
}
if (instruction->HasUses()) {
PrintString(" [");
bool first = true;
for (const HUseListNode<HInstruction*>& use : instruction->GetUses()) {
if (first) {
first = false;
} else {
PrintString(", ");
}
PrintInt(use.GetUser()->GetId());
}
PrintString("]");
}
PrintNewLine();
}
void VisitBasicBlock(HBasicBlock* block) override {
PrintString("BasicBlock ");
PrintInt(block->GetBlockId());
const ArenaVector<HBasicBlock*>& predecessors = block->GetPredecessors();
if (!predecessors.empty()) {
PrintString(", pred: ");
for (size_t i = 0; i < predecessors.size() -1; i++) {
PrintInt(predecessors[i]->GetBlockId());
PrintString(", ");
}
PrintInt(predecessors.back()->GetBlockId());
}
const ArenaVector<HBasicBlock*>& successors = block->GetSuccessors();
if (!successors.empty()) {
PrintString(", succ: ");
for (size_t i = 0; i < successors.size() - 1; i++) {
PrintInt(successors[i]->GetBlockId());
PrintString(", ");
}
PrintInt(successors.back()->GetBlockId());
}
PrintNewLine();
HGraphVisitor::VisitBasicBlock(block);
}
virtual void PrintNewLine() = 0;
virtual void PrintInt(int value) = 0;
virtual void PrintString(const char* value) = 0;
private:
DISALLOW_COPY_AND_ASSIGN(HPrettyPrinter);
};
class StringPrettyPrinter : public HPrettyPrinter {
public:
explicit StringPrettyPrinter(HGraph* graph)
: HPrettyPrinter(graph), str_(""), current_block_(nullptr) { }
void PrintInt(int value) override {
str_ += android::base::StringPrintf("%d", value);
}
void PrintString(const char* value) override {
str_ += value;
}
void PrintNewLine() override {
str_ += '\n';
}
void Clear() { str_.clear(); }
std::string str() const { return str_; }
void VisitBasicBlock(HBasicBlock* block) override {
current_block_ = block;
HPrettyPrinter::VisitBasicBlock(block);
}
void VisitGoto(HGoto* gota) override {
PrintString(" ");
PrintInt(gota->GetId());
PrintString(": Goto ");
PrintInt(current_block_->GetSuccessors()[0]->GetBlockId());
PrintNewLine();
}
private:
std::string str_;
HBasicBlock* current_block_;
DISALLOW_COPY_AND_ASSIGN(StringPrettyPrinter);
};
} // namespace art
#endif // ART_COMPILER_OPTIMIZING_PRETTY_PRINTER_H_