#include "Cache.h" #include "preprocessor/llvm_includes_start.h" #include #include #include #include #include #include #include "preprocessor/llvm_includes_end.h" #include "ExecutionEngine.h" #include "Utils.h" #include "BuildInfo.gen.h" namespace dev { namespace eth { namespace jit { namespace { llvm::MemoryBuffer* g_lastObject; ExecutionEngineListener* g_listener; static const size_t c_versionStampLength = 32; llvm::StringRef getLibVersionStamp() { static auto version = llvm::SmallString{}; if (version.empty()) { version = EVMJIT_VERSION_FULL; version.resize(c_versionStampLength); } return version; } } ObjectCache* Cache::getObjectCache(ExecutionEngineListener* _listener) { static ObjectCache objectCache; g_listener = _listener; return &objectCache; } std::unique_ptr Cache::getObject(std::string const& id) { if (g_listener) g_listener->stateChanged(ExecState::CacheLoad); DLOG(cache) << id << ": search\n"; if (!CHECK(!g_lastObject)) g_lastObject = nullptr; llvm::SmallString<256> cachePath; llvm::sys::path::system_temp_directory(false, cachePath); llvm::sys::path::append(cachePath, "evm_objs", id); if (auto r = llvm::MemoryBuffer::getFile(cachePath.str(), -1, false)) { auto& buf = r.get(); auto objVersionStamp = buf->getBufferSize() >= c_versionStampLength ? llvm::StringRef{buf->getBufferEnd() - c_versionStampLength, c_versionStampLength} : llvm::StringRef{}; if (objVersionStamp == getLibVersionStamp()) g_lastObject = llvm::MemoryBuffer::getMemBufferCopy(r.get()->getBuffer()); else DLOG(cache) << "Unmatched version: " << objVersionStamp.str() << ", expected " << getLibVersionStamp().str() << "\n"; } else if (r.getError() != std::make_error_code(std::errc::no_such_file_or_directory)) { DLOG(cache) << r.getError().message(); // TODO: Add warning log } if (g_lastObject) // if object found create fake module { DLOG(cache) << id << ": found\n"; auto&& context = llvm::getGlobalContext(); auto module = std::unique_ptr(new llvm::Module(id, context)); auto mainFuncType = llvm::FunctionType::get(llvm::Type::getVoidTy(context), {}, false); auto mainFunc = llvm::Function::Create(mainFuncType, llvm::Function::ExternalLinkage, id, module.get()); auto bb = llvm::BasicBlock::Create(context, {}, mainFunc); bb->getInstList().push_back(new llvm::UnreachableInst{context}); return module; } DLOG(cache) << id << ": not found\n"; return nullptr; } void ObjectCache::notifyObjectCompiled(llvm::Module const* _module, llvm::MemoryBuffer const* _object) { if (g_listener) g_listener->stateChanged(ExecState::CacheWrite); auto&& id = _module->getModuleIdentifier(); llvm::SmallString<256> cachePath; llvm::sys::path::system_temp_directory(false, cachePath); llvm::sys::path::append(cachePath, "evm_objs"); if (llvm::sys::fs::create_directory(cachePath.str())) DLOG(cache) << "Cannot create cache dir " << cachePath.str().str() << "\n"; llvm::sys::path::append(cachePath, id); DLOG(cache) << id << ": write\n"; std::string error; llvm::raw_fd_ostream cacheFile(cachePath.c_str(), error, llvm::sys::fs::F_None); cacheFile << _object->getBuffer() << getLibVersionStamp(); } llvm::MemoryBuffer* ObjectCache::getObject(llvm::Module const* _module) { DLOG(cache) << _module->getModuleIdentifier() << ": use\n"; auto o = g_lastObject; g_lastObject = nullptr; return o; } } } }