|
@ -29,6 +29,8 @@ namespace eth |
|
|
namespace jit |
|
|
namespace jit |
|
|
{ |
|
|
{ |
|
|
|
|
|
|
|
|
|
|
|
static const auto c_destIdxLabel = "destIdx"; |
|
|
|
|
|
|
|
|
Compiler::Compiler(Options const& _options): |
|
|
Compiler::Compiler(Options const& _options): |
|
|
m_options(_options), |
|
|
m_options(_options), |
|
|
m_builder(llvm::getGlobalContext()) |
|
|
m_builder(llvm::getGlobalContext()) |
|
@ -93,48 +95,41 @@ std::vector<BasicBlock> Compiler::createBasicBlocks(code_iterator _codeBegin, co |
|
|
return blocks; |
|
|
return blocks; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void Compiler::fillJumpTable() |
|
|
void Compiler::resolveJumps() |
|
|
{ |
|
|
{ |
|
|
for (auto it = m_mainFunc->begin(); it != m_mainFunc->end(); ++it) |
|
|
// Iterate through all EVM instructions blocks (skip first 4 - special blocks).
|
|
|
|
|
|
for (auto it = std::next(m_mainFunc->begin(), 4); it != m_mainFunc->end(); ++it) |
|
|
{ |
|
|
{ |
|
|
|
|
|
auto jumpTable = llvm::cast<llvm::SwitchInst>(m_jumpTableBB->getTerminator()); |
|
|
|
|
|
auto jumpTableInput = llvm::cast<llvm::PHINode>(m_jumpTableBB->begin()); |
|
|
|
|
|
auto nextBlock = it->getNextNode() != m_mainFunc->end() ? it->getNextNode() : m_stopBB; |
|
|
auto term = it->getTerminator(); |
|
|
auto term = it->getTerminator(); |
|
|
|
|
|
|
|
|
if (!term) |
|
|
if (!term) |
|
|
{ |
|
|
{ |
|
|
// Block may have no terminator if the next instruction is a jump destination.
|
|
|
// Block may have no terminator if the next instruction is a jump destination.
|
|
|
auto next = it->getNextNode(); |
|
|
llvm::IRBuilder<>{it}.CreateBr(nextBlock); |
|
|
if (next == m_mainFunc->end()) |
|
|
|
|
|
next = m_stopBB; |
|
|
|
|
|
llvm::IRBuilder<>{it}.CreateBr(next); |
|
|
|
|
|
} |
|
|
} |
|
|
else if (auto jump = llvm::dyn_cast<llvm::BranchInst>(term)) |
|
|
else if (auto jump = llvm::dyn_cast<llvm::BranchInst>(term)) |
|
|
{ |
|
|
{ |
|
|
if (jump->isConditional()) |
|
|
// Resolve jump
|
|
|
|
|
|
if (jump->getSuccessor(0) == m_jumpTableBB) |
|
|
{ |
|
|
{ |
|
|
if (!jump->getSuccessor(1)) |
|
|
auto destIdx = llvm::cast<llvm::ValueAsMetadata>(jump->getMetadata(c_destIdxLabel)->getOperand(0))->getValue(); |
|
|
|
|
|
if (auto constant = llvm::dyn_cast<llvm::ConstantInt>(destIdx)) |
|
|
{ |
|
|
{ |
|
|
auto next = it->getNextNode(); |
|
|
// If destination index is a constant do direct jump to the destination block.
|
|
|
if (next == m_mainFunc->end()) |
|
|
auto bb = jumpTable->findCaseValue(constant).getCaseSuccessor(); |
|
|
next = m_stopBB; |
|
|
jump->setSuccessor(0, bb); |
|
|
jump->setSuccessor(1, next); |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
else |
|
|
|
|
|
jumpTableInput->addIncoming(destIdx, it); // Fill up PHI node
|
|
|
|
|
|
|
|
|
|
|
|
if (jump->isConditional()) |
|
|
|
|
|
jump->setSuccessor(1, nextBlock); // Set next block for conditional jumps
|
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
assert(m_jumpTableBB); |
|
|
|
|
|
if (llvm::pred_empty(m_jumpTableBB)) |
|
|
|
|
|
{ |
|
|
|
|
|
m_jumpTableBB->eraseFromParent(); // remove if unused
|
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// TODO: Extend this function as `resolveJumps()` and fill gaps in branch instructions.
|
|
|
|
|
|
auto target = llvm::cast<llvm::PHINode>(m_jumpTableBB->begin()); |
|
|
|
|
|
for (auto pred: llvm::predecessors(m_jumpTableBB)) |
|
|
|
|
|
{ |
|
|
|
|
|
auto targetMd = llvm::cast<llvm::LocalAsMetadata>(pred->getTerminator()->getMetadata("target")->getOperand(0)); |
|
|
|
|
|
target->addIncoming(targetMd->getValue(), pred); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
std::unique_ptr<llvm::Module> Compiler::compile(code_iterator _begin, code_iterator _end, std::string const& _id) |
|
|
std::unique_ptr<llvm::Module> Compiler::compile(code_iterator _begin, code_iterator _end, std::string const& _id) |
|
@ -187,7 +182,7 @@ std::unique_ptr<llvm::Module> Compiler::compile(code_iterator _begin, code_itera |
|
|
m_builder.CreateCondBr(normalFlow, firstBB, m_abortBB, Type::expectTrue); |
|
|
m_builder.CreateCondBr(normalFlow, firstBB, m_abortBB, Type::expectTrue); |
|
|
|
|
|
|
|
|
for (auto& block: blocks) |
|
|
for (auto& block: blocks) |
|
|
compileBasicBlock(block, runtimeManager, arith, memory, ext, gasMeter, stack, jumpTable); |
|
|
compileBasicBlock(block, runtimeManager, arith, memory, ext, gasMeter, stack); |
|
|
|
|
|
|
|
|
// Code for special blocks:
|
|
|
// Code for special blocks:
|
|
|
m_builder.SetInsertPoint(m_stopBB); |
|
|
m_builder.SetInsertPoint(m_stopBB); |
|
@ -196,15 +191,14 @@ std::unique_ptr<llvm::Module> Compiler::compile(code_iterator _begin, code_itera |
|
|
m_builder.SetInsertPoint(m_abortBB); |
|
|
m_builder.SetInsertPoint(m_abortBB); |
|
|
runtimeManager.exit(ReturnCode::OutOfGas); |
|
|
runtimeManager.exit(ReturnCode::OutOfGas); |
|
|
|
|
|
|
|
|
fillJumpTable(); |
|
|
resolveJumps(); |
|
|
|
|
|
|
|
|
return module; |
|
|
return module; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void Compiler::compileBasicBlock(BasicBlock& _basicBlock, RuntimeManager& _runtimeManager, |
|
|
void Compiler::compileBasicBlock(BasicBlock& _basicBlock, RuntimeManager& _runtimeManager, |
|
|
Arith256& _arith, Memory& _memory, Ext& _ext, GasMeter& _gasMeter, Stack& _globalStack, |
|
|
Arith256& _arith, Memory& _memory, Ext& _ext, GasMeter& _gasMeter, Stack& _globalStack) |
|
|
llvm::SwitchInst& jumpTable) |
|
|
|
|
|
{ |
|
|
{ |
|
|
m_builder.SetInsertPoint(_basicBlock.llvm()); |
|
|
m_builder.SetInsertPoint(_basicBlock.llvm()); |
|
|
LocalStack stack{_globalStack}; |
|
|
LocalStack stack{_globalStack}; |
|
@ -574,24 +568,16 @@ void Compiler::compileBasicBlock(BasicBlock& _basicBlock, RuntimeManager& _runti |
|
|
case Instruction::JUMP: |
|
|
case Instruction::JUMP: |
|
|
case Instruction::JUMPI: |
|
|
case Instruction::JUMPI: |
|
|
{ |
|
|
{ |
|
|
auto jumpBlock = m_jumpTableBB; |
|
|
auto destIdx = llvm::MDNode::get(m_builder.getContext(), llvm::ValueAsMetadata::get(stack.pop())); |
|
|
auto target = stack.pop(); |
|
|
|
|
|
|
|
|
// Create branch instruction, initially to jump table.
|
|
|
|
|
|
// Destination will be optimized with direct jump during jump resolving if destination index is a constant.
|
|
|
auto jumpInst = (inst == Instruction::JUMP) ? |
|
|
auto jumpInst = (inst == Instruction::JUMP) ? |
|
|
m_builder.CreateBr(jumpBlock) : |
|
|
m_builder.CreateBr(m_jumpTableBB) : |
|
|
m_builder.CreateCondBr(m_builder.CreateICmpNE(stack.pop(), Constant::get(0), "jump.check"), jumpBlock, nullptr); |
|
|
m_builder.CreateCondBr(m_builder.CreateICmpNE(stack.pop(), Constant::get(0), "jump.check"), m_jumpTableBB, nullptr); |
|
|
|
|
|
|
|
|
if (auto constant = llvm::dyn_cast<llvm::ConstantInt>(target)) |
|
|
// Attach medatada to branch instruction with information about destination index.
|
|
|
{ |
|
|
jumpInst->setMetadata(c_destIdxLabel, destIdx); |
|
|
// If target index is a constant do direct jump to the target block.
|
|
|
|
|
|
auto bb = jumpTable.findCaseValue(constant).getCaseSuccessor(); |
|
|
|
|
|
jumpInst->setSuccessor(0, bb); |
|
|
|
|
|
} |
|
|
|
|
|
else |
|
|
|
|
|
{ |
|
|
|
|
|
// Attach medatada to branch instruction with information about target index.
|
|
|
|
|
|
auto targetMd = llvm::MDNode::get(jumpInst->getContext(), llvm::LocalAsMetadata::get(target)); |
|
|
|
|
|
jumpInst->setMetadata("target", targetMd); |
|
|
|
|
|
} |
|
|
|
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|