/*
* 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_SEA_IR_DEBUG_DOT_GEN_H_
#define ART_COMPILER_SEA_IR_DEBUG_DOT_GEN_H_
#include "safe_map.h"
#include "base/stringprintf.h"
#include "file_output_stream.h"
#include "os.h"
#include "sea_ir/ir/sea.h"
#include "sea_ir/types/type_inference.h"
namespace sea_ir {
class DotConversionOptions {
public:
DotConversionOptions(): save_use_edges_(false) { }
bool WillSaveUseEdges() const {
return save_use_edges_;
}
private:
bool save_use_edges_;
};
class DotGenerationVisitor: public IRVisitor {
public:
explicit DotGenerationVisitor(const DotConversionOptions* const options,
art::SafeMap<int, const Type*>* types): graph_(), types_(types), options_(options) { }
virtual void Initialize(SeaGraph* graph);
// Saves the ssa def->use edges corresponding to @instruction.
void ToDotSSAEdges(InstructionNode* instruction);
void ToDotSSAEdges(PhiInstructionNode* instruction);
void Visit(SeaGraph* graph) {
dot_text_ += "digraph seaOfNodes {\ncompound=true\n";
}
void Visit(SignatureNode* parameter);
// Appends to @result a dot language formatted string representing the node and
// (by convention) outgoing edges, so that the composition of theToDot() of all nodes
// builds a complete dot graph (without prolog and epilog though).
void Visit(Region* region);
void Visit(InstructionNode* instruction);
void Visit(PhiInstructionNode* phi);
void Visit(UnnamedConstInstructionNode* instruction);
void Visit(ConstInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(ReturnInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(IfNeInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(MoveResultInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(InvokeStaticInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(AddIntInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(GotoInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
void Visit(IfEqzInstructionNode* instruction) {
Visit(reinterpret_cast<InstructionNode*>(instruction));
}
std::string GetResult() const {
return dot_text_;
}
private:
std::string dot_text_;
SeaGraph* graph_;
art::SafeMap<int, const Type*>* types_;
const DotConversionOptions* const options_;
};
// Stores options for turning a SEA IR graph to a .dot file.
class DotConversion {
public:
DotConversion(): options_() { }
// Saves to @filename the .dot representation of @graph with the options @options.
void DumpSea(SeaGraph* graph, std::string filename,
art::SafeMap<int, const Type*>* types) const {
LOG(INFO) << "Starting to write SEA string to file " << filename << std::endl;
DotGenerationVisitor dgv = DotGenerationVisitor(&options_, types);
graph->Accept(&dgv);
// TODO: UniquePtr to close file properly. Switch to BufferedOutputStream.
art::File* file = art::OS::CreateEmptyFile(filename.c_str());
art::FileOutputStream fos(file);
std::string graph_as_string = dgv.GetResult();
graph_as_string += "}";
fos.WriteFully(graph_as_string.c_str(), graph_as_string.size());
LOG(INFO) << "Written SEA string to file.";
}
private:
DotConversionOptions options_;
};
} // namespace sea_ir
#endif // ART_COMPILER_SEA_IR_DEBUG_DOT_GEN_H_