#pragma once #include #include "Common.h" #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, std::string const& _id); 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); llvm::BasicBlock* getJumpTableBlock(); llvm::BasicBlock* getBadJumpBlock(); 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 m_basicBlocks; /// Stop basic block - terminates execution with STOP code (0) llvm::BasicBlock* m_stopBB = nullptr; /// Block with a jump table. std::unique_ptr m_jumpTableBlock; /// Destination for invalid jumps std::unique_ptr m_badJumpBlock; /// Main program function llvm::Function* m_mainFunc = nullptr; }; } } }