#pragma once #include #include "CompilerHelper.h" namespace dev { namespace eth { namespace jit { class LazyFunction { public: using Creator = std::function; LazyFunction(Creator _creator) : m_creator(_creator) {} llvm::Value* call(llvm::IRBuilder<>& _builder, std::initializer_list const& _args, llvm::Twine const& _name = {}); private: llvm::Function* m_func = nullptr; Creator m_creator; }; class Array : public CompilerHelper { public: Array(llvm::IRBuilder<>& _builder, char const* _name); Array(llvm::IRBuilder<>& _builder, llvm::Value* _array); void push(llvm::Value* _value) { m_pushFunc.call(m_builder, {m_array, _value}); } void set(llvm::Value* _index, llvm::Value* _value) { m_setFunc.call(m_builder, {m_array, _index, _value}); } llvm::Value* get(llvm::Value* _index) { return m_getFunc.call(m_builder, {m_array, _index}); } void pop(llvm::Value* _count); llvm::Value* size(llvm::Value* _array = nullptr); void free() { m_freeFunc.call(m_builder, {m_array}); } void extend(llvm::Value* _arrayPtr, llvm::Value* _size); llvm::Value* getPtr(llvm::Value* _arrayPtr, llvm::Value* _index) { return m_getPtrFunc.call(m_builder, {_arrayPtr, _index}); } llvm::Value* getPointerTo() const { return m_array; } static llvm::Type* getType(); private: llvm::Value* m_array = nullptr; llvm::Function* createArrayPushFunc(); llvm::Function* createArraySetFunc(); llvm::Function* createArrayGetFunc(); llvm::Function* createGetPtrFunc(); llvm::Function* createFreeFunc(); llvm::Function* createExtendFunc(); llvm::Function* getReallocFunc(); LazyFunction m_pushFunc = {[this](){ return createArrayPushFunc(); }}; // TODO: If works on MSVC, remove form initialization list LazyFunction m_setFunc; LazyFunction m_getPtrFunc = {[this](){ return createGetPtrFunc(); }}; LazyFunction m_getFunc; LazyFunction m_freeFunc; LazyFunction m_extendFunc = {[this](){ return createExtendFunc(); }}; LazyFunction m_reallocFunc = {[this](){ return getReallocFunc(); }}; }; } } }