#include "BasicBlock.h" #include #include #include "Type.h" namespace dev { namespace eth { namespace jit { const char* BasicBlock::NamePrefix = "Instr."; BasicBlock::BasicBlock(ProgramCounter _beginInstIdx, ProgramCounter _endInstIdx, llvm::Function* _mainFunc) : m_beginInstIdx(_beginInstIdx), m_endInstIdx(_endInstIdx), m_llvmBB(llvm::BasicBlock::Create(_mainFunc->getContext(), {NamePrefix, std::to_string(_beginInstIdx)}, _mainFunc)), m_stack(m_llvmBB) {} BasicBlock::BasicBlock(std::string _name, llvm::Function* _mainFunc) : m_beginInstIdx(0), m_endInstIdx(0), m_llvmBB(llvm::BasicBlock::Create(_mainFunc->getContext(), _name, _mainFunc)), m_stack(m_llvmBB) {} void BasicBlock::LocalStack::push(llvm::Value* _value) { assert(_value->getType() == Type::i256); m_backend.push_back(_value); } llvm::Value* BasicBlock::LocalStack::pop() { auto top = get(0); m_backend.pop_back(); return top; } llvm::Value* BasicBlock::LocalStack::get(size_t _index) { if (_index >= m_backend.size()) { // Create PHI node for missing values auto nMissingVals = _index - m_backend.size() + 1; m_backend.insert(m_backend.begin(), nMissingVals, nullptr); for (decltype(nMissingVals) i = 0; i < nMissingVals; ++i) { m_backend[i] = m_llvmBB->empty() ? llvm::PHINode::Create(Type::i256, 0, {}, m_llvmBB) : llvm::PHINode::Create(Type::i256, 0, {}, m_llvmBB->getFirstNonPHI()); m_numRequiredStackItems += 1; } } return *(m_backend.rbegin() + _index); } void BasicBlock::LocalStack::dup(size_t _index) { m_backend.push_back(get(_index)); } void BasicBlock::LocalStack::swap(size_t _index) { assert(_index != 0); get(_index); // Create PHI nodes std::swap(*m_backend.rbegin(), *(m_backend.rbegin() + _index)); } } } }