#include "Cache.h" #include #include #include #include #include #include #include #include #include "ExecutionEngine.h" namespace dev { namespace eth { namespace jit { //#define LOG(...) std::cerr << "CACHE " #define LOG(...) std::ostream(nullptr) namespace { llvm::MemoryBuffer* g_lastObject; ExecutionEngineListener* g_listener; } 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); assert(!g_lastObject); llvm::SmallString<256> cachePath; llvm::sys::path::system_temp_directory(false, cachePath); llvm::sys::path::append(cachePath, "evm_objs", id); #if defined(__GNUC__) && !defined(NDEBUG) llvm::sys::fs::file_status st; auto err = llvm::sys::fs::status(cachePath.str(), st); if (err) return nullptr; auto mtime = st.getLastModificationTime().toEpochTime(); std::tm tm; strptime(__DATE__ __TIME__, " %b %d %Y %H:%M:%S", &tm); auto btime = (uint64_t)std::mktime(&tm); if (btime > mtime) return nullptr; #endif if (auto r = llvm::MemoryBuffer::getFile(cachePath.str(), -1, false)) g_lastObject = llvm::MemoryBuffer::getMemBufferCopy(r.get()->getBuffer()); else if (r.getError() != std::make_error_code(std::errc::no_such_file_or_directory)) std::cerr << r.getError().message(); // TODO: Add log if (g_lastObject) // if object found create fake module { auto module = std::unique_ptr(new llvm::Module(id, llvm::getGlobalContext())); auto mainFuncType = llvm::FunctionType::get(llvm::IntegerType::get(llvm::getGlobalContext(), 32), {}, false); llvm::Function::Create(mainFuncType, llvm::Function::ExternalLinkage, id, module.get()); } 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())) return; // TODO: Add log llvm::sys::path::append(cachePath, id); std::string error; llvm::raw_fd_ostream cacheFile(cachePath.c_str(), error, llvm::sys::fs::F_None); cacheFile << _object->getBuffer(); } llvm::MemoryBuffer* ObjectCache::getObject(llvm::Module const*) { auto o = g_lastObject; g_lastObject = nullptr; return o; } } } }