#pragma once #include #include #include "BasicBlock.h" namespace dev { namespace eth { namespace jit { class Compiler { public: struct Options { /// Optimize stack operations between basic blocks bool optimizeStack; /// Rewrite switch instructions to sequences of branches bool rewriteSwitchToBranches; /// Dump CFG as a .dot file for graphviz bool dumpCFG; Options(): optimizeStack(true), rewriteSwitchToBranches(true), dumpCFG(false) {} }; using ProgramCounter = uint64_t; Compiler(Options const& _options); std::unique_ptr compile(bytes const& _bytecode); private: void createBasicBlocks(bytes const& _bytecode); void compileBasicBlock(BasicBlock& _basicBlock, bytes const& _bytecode, class RuntimeManager& _runtimeManager, class Arith256& _arith, class Memory& _memory, class Ext& _ext, class GasMeter& _gasMeter, llvm::BasicBlock* _nextBasicBlock); void removeDeadBlocks(); /// Dumps basic block graph in graphviz format to a file, if option dumpCFG is enabled. void dumpCFGifRequired(std::string const& _dotfilePath); /// Dumps basic block graph in graphviz format to a stream. void dumpCFGtoStream(std::ostream& _out); /// Dumps all basic blocks to stderr. Useful in a debugging session. void dump(); /// Compiler options Options const& m_options; /// Helper class for generating IR llvm::IRBuilder<> m_builder; /// Maps a program counter pc to a basic block that starts at pc (if any). std::map basicBlocks = {}; /// Maps a pc at which there is a JUMP or JUMPI to the target block of the jump. std::map m_directJumpTargets = {}; /// A list of possible blocks to which there may be indirect jumps. std::vector m_indirectJumpTargets = {}; /// Stop basic block - terminates execution with STOP code (0) llvm::BasicBlock* m_stopBB = nullptr; /// Block with a jump table. std::unique_ptr m_jumpTableBlock = nullptr; /// Default destination for indirect jumps. std::unique_ptr m_badJumpBlock = nullptr; /// Main program function llvm::Function* m_mainFunc = nullptr; }; } } }