2014-02-02 09:33:57 +01:00
|
|
|
/**
|
|
|
|
|
******************************************************************************
|
|
|
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
|
|
|
******************************************************************************
|
|
|
|
|
* Copyright 2014 Ben Vanik. All rights reserved. *
|
|
|
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
|
|
|
******************************************************************************
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <alloy/compiler/passes/control_flow_analysis_pass.h>
|
|
|
|
|
|
|
|
|
|
#include <alloy/backend/backend.h>
|
|
|
|
|
#include <alloy/compiler/compiler.h>
|
|
|
|
|
#include <alloy/runtime/runtime.h>
|
|
|
|
|
|
|
|
|
|
#pragma warning(push)
|
|
|
|
|
#pragma warning(disable : 4244)
|
|
|
|
|
#pragma warning(disable : 4267)
|
|
|
|
|
#include <llvm/ADT/BitVector.h>
|
|
|
|
|
#pragma warning(pop)
|
|
|
|
|
|
|
|
|
|
using namespace alloy;
|
|
|
|
|
using namespace alloy::backend;
|
|
|
|
|
using namespace alloy::compiler;
|
|
|
|
|
using namespace alloy::compiler::passes;
|
|
|
|
|
using namespace alloy::frontend;
|
|
|
|
|
using namespace alloy::hir;
|
|
|
|
|
using namespace alloy::runtime;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ControlFlowAnalysisPass::ControlFlowAnalysisPass() :
|
|
|
|
|
CompilerPass() {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ControlFlowAnalysisPass::~ControlFlowAnalysisPass() {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ControlFlowAnalysisPass::Run(HIRBuilder* builder) {
|
|
|
|
|
// TODO(benvanik): reset edges for all blocks? Needed to be re-runnable.
|
|
|
|
|
|
|
|
|
|
// Add edges.
|
|
|
|
|
auto block = builder->first_block();
|
|
|
|
|
while (block) {
|
2014-02-11 06:16:38 +01:00
|
|
|
auto instr = block->instr_tail;
|
2014-02-02 09:33:57 +01:00
|
|
|
while (instr) {
|
|
|
|
|
if (instr->opcode->flags & OPCODE_FLAG_BRANCH) {
|
|
|
|
|
if (instr->opcode == &OPCODE_BRANCH_info) {
|
|
|
|
|
auto label = instr->src1.label;
|
|
|
|
|
builder->AddEdge(block, label->block, Edge::UNCONDITIONAL);
|
2014-02-11 06:16:38 +01:00
|
|
|
break;
|
2014-02-02 09:33:57 +01:00
|
|
|
} else if (instr->opcode == &OPCODE_BRANCH_TRUE_info ||
|
|
|
|
|
instr->opcode == &OPCODE_BRANCH_FALSE_info) {
|
|
|
|
|
auto label = instr->src2.label;
|
|
|
|
|
builder->AddEdge(block, label->block, 0);
|
2014-02-11 06:16:38 +01:00
|
|
|
break;
|
2014-02-02 09:33:57 +01:00
|
|
|
}
|
|
|
|
|
}
|
2014-02-11 06:16:38 +01:00
|
|
|
instr = instr->prev;
|
2014-02-02 09:33:57 +01:00
|
|
|
}
|
|
|
|
|
block = block->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Mark dominators.
|
|
|
|
|
block = builder->first_block();
|
|
|
|
|
while (block) {
|
|
|
|
|
if (block->incoming_edge_head &&
|
|
|
|
|
!block->incoming_edge_head->incoming_next) {
|
|
|
|
|
block->incoming_edge_head->flags |= Edge::DOMINATES;
|
|
|
|
|
}
|
|
|
|
|
block = block->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|