#include "Optimizer.h" #include "preprocessor/llvm_includes_start.h" #include #include #include #include #include #include "preprocessor/llvm_includes_end.h" #include "Arith256.h" #include "Type.h" namespace dev { namespace eth { namespace jit { bool optimize(llvm::Module& _module) { auto pm = llvm::legacy::PassManager{}; //pm.add(llvm::createFunctionInliningPass(2, 2)); // Problem with APInt value bigger than 64bit pm.add(llvm::createCFGSimplificationPass()); pm.add(llvm::createInstructionCombiningPass()); pm.add(llvm::createAggressiveDCEPass()); pm.add(llvm::createLowerSwitchPass()); return pm.run(_module); } namespace { class LowerEVMPass : public llvm::BasicBlockPass { static char ID; bool m_mulFuncNeeded = false; public: LowerEVMPass(): llvm::BasicBlockPass(ID) {} virtual bool runOnBasicBlock(llvm::BasicBlock& _bb) override; virtual bool doFinalization(llvm::Module& _module) override; }; char LowerEVMPass::ID = 0; bool LowerEVMPass::runOnBasicBlock(llvm::BasicBlock& _bb) { auto modified = false; auto module = _bb.getParent()->getParent(); for (auto it = _bb.begin(); it != _bb.end(); ) { auto& inst = *it++; llvm::Function* func = nullptr; if (inst.getType() == Type::Word) { switch (inst.getOpcode()) { case llvm::Instruction::Mul: func = Arith256::getMulFunc(*module); break; case llvm::Instruction::UDiv: func = Arith256::getUDiv256Func(*module); break; case llvm::Instruction::URem: func = Arith256::getURem256Func(*module); break; case llvm::Instruction::SDiv: func = Arith256::getSDiv256Func(*module); break; case llvm::Instruction::SRem: func = Arith256::getSRem256Func(*module); break; } } if (func) { auto call = llvm::CallInst::Create(func, {inst.getOperand(0), inst.getOperand(1)}, "", &inst); inst.replaceAllUsesWith(call); inst.eraseFromParent(); modified = true; } } return modified; } bool LowerEVMPass::doFinalization(llvm::Module&) { return false; } } bool prepare(llvm::Module& _module) { auto pm = llvm::legacy::PassManager{}; pm.add(llvm::createDeadCodeEliminationPass()); pm.add(new LowerEVMPass{}); return pm.run(_module); } } } }