GitOrigin-RevId: 64b7825c34
release-1.7
@@ -24,7 +24,7 @@ | |||||
#include "megbrain/comp_node.h" | #include "megbrain/comp_node.h" | ||||
#include "megbrain/serialization/extern_c_opr.h" | #include "megbrain/serialization/extern_c_opr.h" | ||||
#include "megbrain/version.h" | #include "megbrain/version.h" | ||||
#include "mge/algo_cache/file_cache.h" | |||||
#include "megbrain/utils/infile_persistent_cache.h" | |||||
#include "mge/common.h" | #include "mge/common.h" | ||||
#if MGB_ENABLE_TENSOR_RT | #if MGB_ENABLE_TENSOR_RT | ||||
#include "megbrain/tensorrt/tensorrt_engine_cache.h" | #include "megbrain/tensorrt/tensorrt_engine_cache.h" | ||||
@@ -170,8 +170,8 @@ void lite::set_persistent_cache(const std::string& cache_path, bool always_sync) | |||||
"it now may cause unknow error!!"); | "it now may cause unknow error!!"); | ||||
} | } | ||||
cache_control.config_algo_times++; | cache_control.config_algo_times++; | ||||
mgb::PersistentCache::set_impl( | |||||
std::make_shared<InFilePersistentCache>(cache_path.c_str(), always_sync)); | |||||
mgb::PersistentCache::set_impl(std::make_shared<mgb::InFilePersistentCache>( | |||||
cache_path.c_str(), always_sync)); | |||||
} | } | ||||
void lite::dump_persistent_cache(const std::string& cache_path) { | void lite::dump_persistent_cache(const std::string& cache_path) { | ||||
@@ -179,7 +179,7 @@ void lite::dump_persistent_cache(const std::string& cache_path) { | |||||
LITE_ASSERT( | LITE_ASSERT( | ||||
cache_control.cache_type == "file", | cache_control.cache_type == "file", | ||||
"now cache type not correct, it can't be dumped."); | "now cache type not correct, it can't be dumped."); | ||||
static_cast<InFilePersistentCache&>(mgb::PersistentCache::inst()) | |||||
static_cast<mgb::InFilePersistentCache&>(mgb::PersistentCache::inst()) | |||||
.dump_cache(cache_path.c_str()); | .dump_cache(cache_path.c_str()); | ||||
} | } | ||||
@@ -1,258 +0,0 @@ | |||||
/** | |||||
* \file lite/src/mge/algo_cache/file_cache.cpp | |||||
* MegEngine is Licensed under the Apache License, Version 2.0 (the "License") | |||||
* | |||||
* Copyright (c) 2014-2021 Megvii Inc. All rights reserved. | |||||
* | |||||
* Unless required by applicable law or agreed to in writing, | |||||
* software distributed under the License is distributed on an | |||||
* "AS IS" BASIS, WITHOUT ARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |||||
*/ | |||||
#include "lite_build_config.h" | |||||
#if LITE_BUILD_WITH_MGE | |||||
#include "../common.h" | |||||
#include "file_cache.h" | |||||
using namespace lite; | |||||
//////////////////////// InFilePersistentCache::InputMemory /////////////// | |||||
class InFilePersistentCache::InputMemory { | |||||
const uint8_t* m_ptr; | |||||
size_t m_offset = 0; | |||||
size_t m_size; | |||||
public: | |||||
InputMemory(const uint8_t* bin, size_t size) : m_ptr{bin}, m_size{size} {} | |||||
template <typename T> | |||||
void read(T& val) { | |||||
static_assert( | |||||
std::is_trivially_copyable<T>::value, | |||||
"only support trivially copyable type"); | |||||
LITE_ASSERT(m_offset + sizeof(T) <= m_size); | |||||
memcpy(&val, m_ptr, sizeof(T)); | |||||
m_offset += sizeof(T); | |||||
m_ptr += sizeof(T); | |||||
} | |||||
template <typename T> | |||||
void read(T* buf, size_t size) { | |||||
static_assert( | |||||
std::is_trivially_copyable<T>::value && sizeof(T) == 1, | |||||
"only support read bytes"); | |||||
LITE_ASSERT(m_offset + size <= m_size); | |||||
memcpy(buf, m_ptr, size); | |||||
m_offset += size; | |||||
m_ptr += size; | |||||
} | |||||
}; | |||||
//////////////////////// InFilePersistentCache::InputFile /////////////// | |||||
class InFilePersistentCache::InputFile { | |||||
FILE* m_fp; | |||||
public: | |||||
InputFile(const char* path) : m_fp{fopen(path, "rb")} { | |||||
LITE_ASSERT(m_fp, "failed to open %s: %s", path, strerror(errno)); | |||||
} | |||||
~InputFile() { | |||||
if (m_fp) { | |||||
fclose(m_fp); | |||||
} | |||||
} | |||||
template <typename T> | |||||
void read(T& val) { | |||||
static_assert( | |||||
std::is_trivially_copyable<T>::value, | |||||
"only support trivially copyable type"); | |||||
auto ret = fread(&val, sizeof(T), 1, m_fp); | |||||
LITE_ASSERT(ret == 1); | |||||
} | |||||
template <typename T> | |||||
void read(T* buf, size_t size) { | |||||
static_assert( | |||||
std::is_trivially_copyable<T>::value && sizeof(T) == 1, | |||||
"only support read bytes"); | |||||
auto ret = fread(buf, size, 1, m_fp); | |||||
LITE_ASSERT(ret == 1); | |||||
} | |||||
}; | |||||
//////////////////////// InFilePersistentCache::OutputFile /////////////// | |||||
class InFilePersistentCache::OutputFile { | |||||
FILE* m_fp; | |||||
public: | |||||
OutputFile(const char* path) : m_fp{fopen(path, "wb")} { | |||||
LITE_ASSERT(m_fp, "failed to open %s: %s", path, strerror(errno)); | |||||
} | |||||
~OutputFile() { | |||||
if (m_fp) { | |||||
fclose(m_fp); | |||||
} | |||||
} | |||||
template <typename T> | |||||
void write(T val) { | |||||
auto ret = fwrite(&val, sizeof(T), 1, m_fp); | |||||
LITE_ASSERT(ret == 1); | |||||
} | |||||
template <typename T> | |||||
void write(const T* buf, size_t size) { | |||||
static_assert(sizeof(T) == 1, "only support write bytes"); | |||||
auto ret = fwrite(buf, size, 1, m_fp); | |||||
LITE_ASSERT(ret == 1); | |||||
} | |||||
void flush() { fflush(m_fp); } | |||||
void set_head() { fseek(m_fp, 0, SEEK_SET); } | |||||
}; | |||||
//////////////////////// InFilePersistentCache::BlobStorage /////////////// | |||||
template <typename Input> | |||||
InFilePersistentCache::BlobStorage& InFilePersistentCache::BlobStorage::init_from_input( | |||||
Input& inp) { | |||||
uint32_t data_size; | |||||
inp.read(data_size); | |||||
size = data_size; | |||||
data_refhold = std::make_unique<uint8_t[]>(size); | |||||
inp.read(data_refhold.get(), size); | |||||
ptr = data_refhold.get(); | |||||
return *this; | |||||
} | |||||
void InFilePersistentCache::BlobStorage::write_to_file(OutputFile& out_file) const { | |||||
uint32_t u_size = size; | |||||
out_file.write(u_size); | |||||
out_file.write(data_refhold.get(), u_size); | |||||
} | |||||
InFilePersistentCache::BlobStorage& InFilePersistentCache::BlobStorage::init_data_ref( | |||||
const Blob& b) { | |||||
data_refhold = std::make_unique<uint8_t[]>(b.size + 1); | |||||
memcpy(data_refhold.get(), b.ptr, b.size); | |||||
data_refhold.get()[b.size] = 0; // for C-string safety | |||||
ptr = data_refhold.get(); | |||||
size = b.size; | |||||
return *this; | |||||
} | |||||
//////////////////////// InFilePersistentCache ////////////////////// | |||||
template <typename Input> | |||||
void InFilePersistentCache::read_cache(Input& inp) { | |||||
uint32_t nr_category; | |||||
inp.read(nr_category); | |||||
char category_buf[256]; | |||||
for (uint32_t i = 0; i < nr_category; i++) { | |||||
uint32_t category_size; | |||||
inp.read(category_size); | |||||
inp.read(category_buf, category_size); | |||||
category_buf[category_size] = '\0'; | |||||
std::string category(category_buf); | |||||
mgb_log_debug("load new category: %s", category_buf); | |||||
// read bobs | |||||
uint32_t nr_bobs; | |||||
inp.read(nr_bobs); | |||||
for (uint32_t j = 0; j < nr_bobs; j++) { | |||||
BlobStorage key_storage; | |||||
key_storage.init_from_input(inp).init_hash(); | |||||
mgb_log_debug("read key: %zu", key_storage.hash); | |||||
m_cache[category][std::move(key_storage)].init_from_input(inp); | |||||
} | |||||
} | |||||
} | |||||
InFilePersistentCache::InFilePersistentCache(const char* path, bool always_open) { | |||||
if (!access(path, F_OK)) { | |||||
mgb_log_debug("use fastrun cache: %s", path); | |||||
InputFile inp(path); | |||||
read_cache<InputFile>(inp); | |||||
} | |||||
if (always_open) { | |||||
m_always_open_file = std::make_shared<OutputFile>(path); | |||||
} | |||||
} | |||||
InFilePersistentCache::InFilePersistentCache(const uint8_t* bin, size_t size) { | |||||
LITE_ASSERT(bin); | |||||
InputMemory inp(bin, size); | |||||
read_cache<InputMemory>(inp); | |||||
} | |||||
void InFilePersistentCache::dump_cache(const char* path) { | |||||
OutputFile out_file(path); | |||||
dump_cache(&out_file); | |||||
} | |||||
void InFilePersistentCache::dump_cache(OutputFile* out_file) { | |||||
uint32_t nr_category = m_cache.size(); | |||||
out_file->write(nr_category); | |||||
for (const auto& cached_category : m_cache) { | |||||
uint32_t category_size = cached_category.first.size(); | |||||
out_file->write(category_size); | |||||
out_file->write(cached_category.first.data(), category_size); | |||||
mgb_log_debug("write new category: %s", cached_category.first.c_str()); | |||||
uint32_t nr_bobs = cached_category.second.size(); | |||||
out_file->write(nr_bobs); | |||||
for (const auto& item : cached_category.second) { | |||||
mgb_log_debug("dump key: %zu", item.first.hash); | |||||
item.first.write_to_file(*out_file); | |||||
item.second.write_to_file(*out_file); | |||||
} | |||||
} | |||||
} | |||||
mgb::Maybe<InFilePersistentCache::Blob> InFilePersistentCache::get( | |||||
const std::string& category, const Blob& key) { | |||||
decltype(m_cache.begin()) iter0; | |||||
{ | |||||
MGB_LOCK_GUARD(m_mtx); | |||||
iter0 = m_cache.find(category); | |||||
if (iter0 == m_cache.end()) | |||||
return mgb::None; | |||||
} | |||||
BlobStorage key_storage; | |||||
key_storage.Blob::operator=(key); | |||||
key_storage.init_hash(); | |||||
MGB_LOCK_GUARD(m_mtx); | |||||
auto iter1 = iter0->second.find(key_storage); | |||||
if (iter1 == iter0->second.end()) | |||||
return mgb::None; | |||||
return iter1->second; | |||||
} | |||||
void InFilePersistentCache::put( | |||||
const std::string& category, const Blob& key, const Blob& value) { | |||||
BlobStorage key_storage; | |||||
key_storage.init_data_ref(key).init_hash(); | |||||
MGB_LOCK_GUARD(m_mtx); | |||||
auto size0 = m_cache.size(); | |||||
m_cache[category][std::move(key_storage)].init_data_ref(value); | |||||
if (m_cache.size() > size0) { | |||||
mgb_log_debug("new cache category: %s", category.c_str()); | |||||
} | |||||
if (m_always_open_file) { | |||||
m_always_open_file->set_head(); | |||||
dump_cache(m_always_open_file.get()); | |||||
m_always_open_file->flush(); | |||||
} | |||||
} | |||||
#endif | |||||
// vim: syntax=cpp.doxygen foldmethod=marker foldmarker=f{{{,f}}} |
@@ -1,87 +0,0 @@ | |||||
/** | |||||
* \file lite/src/mge/algo_cache/file_cache.h | |||||
* MegEngine is Licensed under the Apache License, Version 2.0 (the "License") | |||||
* | |||||
* Copyright (c) 2014-2021 Megvii Inc. All rights reserved. | |||||
* | |||||
* Unless required by applicable law or agreed to in writing, | |||||
* software distributed under the License is distributed on an | |||||
* "AS IS" BASIS, WITHOUT ARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |||||
*/ | |||||
#pragma once | |||||
#include "lite_build_config.h" | |||||
#if LITE_BUILD_WITH_MGE | |||||
#include "megbrain/utils/persistent_cache.h" | |||||
namespace lite { | |||||
/** | |||||
* dump format: | |||||
* | |||||
* all integers in local endian (effectively little endian as I can see) | |||||
* | |||||
* dump format: | |||||
* <nr_category|uint32_t><category_size|uint32_t><category|uint8_t*> | |||||
* <nr_bob|uint32_t>[<key_size|uint32_t><key|uint8_t*><data_size| | |||||
* uint32_t><data|uint8_t*>]* | |||||
*/ | |||||
//! TODO: fix one thread set cache when other threads is using old cache | |||||
class InFilePersistentCache final : public mgb::PersistentCache { | |||||
class InputFile; | |||||
class InputMemory; | |||||
class OutputFile; | |||||
struct BlobStorage : public Blob { | |||||
std::unique_ptr<uint8_t[]> data_refhold; | |||||
size_t hash = 0; | |||||
template <typename Input> | |||||
BlobStorage& init_from_input(Input& inp); | |||||
void write_to_file(OutputFile& out_file) const; | |||||
BlobStorage& init_data_ref(const Blob& b); | |||||
BlobStorage& init_hash() { | |||||
hash = mgb::XXHash{}.update(ptr, size).digest(); | |||||
return *this; | |||||
} | |||||
bool operator==(const BlobStorage& rhs) const { | |||||
return size == rhs.size && !memcmp(ptr, rhs.ptr, size); | |||||
} | |||||
struct Hash { | |||||
size_t operator()(const BlobStorage& b) const { return b.hash; } | |||||
}; | |||||
}; | |||||
std::unordered_map< | |||||
std::string, | |||||
std::unordered_map<BlobStorage, BlobStorage, BlobStorage::Hash>> | |||||
m_cache; | |||||
LITE_MUTEX m_mtx; | |||||
std::shared_ptr<OutputFile> m_always_open_file; | |||||
template <typename Input> | |||||
void read_cache(Input& inp); | |||||
public: | |||||
InFilePersistentCache() = default; | |||||
InFilePersistentCache(const char* path, bool always_open = false); | |||||
InFilePersistentCache(const uint8_t* bin, size_t size); | |||||
/** | |||||
* \warning You should invoke \c dump_cache mannually to save the cache | |||||
* file. | |||||
*/ | |||||
void dump_cache(const char* path); | |||||
void dump_cache(OutputFile* out_file); | |||||
mgb::Maybe<Blob> get(const std::string& category, const Blob& key) override; | |||||
void put(const std::string& category, const Blob& key, const Blob& value) override; | |||||
}; | |||||
} // namespace lite | |||||
#endif | |||||
// vim: syntax=cpp.doxygen foldmethod=marker foldmarker=f{{{,f}}} |
@@ -112,6 +112,9 @@ public: | |||||
auto ret = fwrite(buf, size, 1, m_fp); | auto ret = fwrite(buf, size, 1, m_fp); | ||||
mgb_assert(ret == 1); | mgb_assert(ret == 1); | ||||
} | } | ||||
void flush() { fflush(m_fp); } | |||||
void set_head() { fseek(m_fp, 0, SEEK_SET); } | |||||
}; | }; | ||||
//////////////////////// InFilePersistentCache::BlobStorage /////////////// | //////////////////////// InFilePersistentCache::BlobStorage /////////////// | ||||
@@ -172,12 +175,15 @@ void InFilePersistentCache::read_cache(Input& inp) { | |||||
} | } | ||||
} | } | ||||
InFilePersistentCache::InFilePersistentCache(const char* path) { | |||||
InFilePersistentCache::InFilePersistentCache(const char* path, bool always_open) { | |||||
if (!access(path, F_OK)) { | if (!access(path, F_OK)) { | ||||
mgb_log_debug("use fastrun cache: %s", path); | mgb_log_debug("use fastrun cache: %s", path); | ||||
InputFile inp(path); | InputFile inp(path); | ||||
read_cache<InputFile>(inp); | read_cache<InputFile>(inp); | ||||
} | } | ||||
if (always_open) { | |||||
m_always_open_file = std::make_shared<OutputFile>(path); | |||||
} | |||||
} | } | ||||
InFilePersistentCache::InFilePersistentCache(const uint8_t* bin, size_t size) { | InFilePersistentCache::InFilePersistentCache(const uint8_t* bin, size_t size) { | ||||
@@ -188,25 +194,28 @@ InFilePersistentCache::InFilePersistentCache(const uint8_t* bin, size_t size) { | |||||
void InFilePersistentCache::dump_cache(const char* path) { | void InFilePersistentCache::dump_cache(const char* path) { | ||||
OutputFile out_file(path); | OutputFile out_file(path); | ||||
dump_cache(&out_file); | |||||
} | |||||
void InFilePersistentCache::dump_cache(OutputFile* out_file) { | |||||
uint32_t nr_category = m_cache.size(); | uint32_t nr_category = m_cache.size(); | ||||
out_file.write(nr_category); | |||||
out_file->write(nr_category); | |||||
for (const auto& cached_category : m_cache) { | for (const auto& cached_category : m_cache) { | ||||
uint32_t category_size = cached_category.first.size(); | uint32_t category_size = cached_category.first.size(); | ||||
out_file.write(category_size); | |||||
out_file.write(cached_category.first.data(), category_size); | |||||
out_file->write(category_size); | |||||
out_file->write(cached_category.first.data(), category_size); | |||||
mgb_log_debug("write new category: %s", cached_category.first.c_str()); | mgb_log_debug("write new category: %s", cached_category.first.c_str()); | ||||
uint32_t nr_bobs = cached_category.second.size(); | uint32_t nr_bobs = cached_category.second.size(); | ||||
out_file.write(nr_bobs); | |||||
out_file->write(nr_bobs); | |||||
for (const auto& item : cached_category.second) { | for (const auto& item : cached_category.second) { | ||||
mgb_log_debug("dump key: %zu", item.first.hash); | mgb_log_debug("dump key: %zu", item.first.hash); | ||||
item.first.write_to_file(out_file); | |||||
item.second.write_to_file(out_file); | |||||
item.first.write_to_file(*out_file); | |||||
item.second.write_to_file(*out_file); | |||||
} | } | ||||
} | } | ||||
} | } | ||||
Maybe<InFilePersistentCache::Blob> InFilePersistentCache::get( | Maybe<InFilePersistentCache::Blob> InFilePersistentCache::get( | ||||
const std::string& category, const Blob& key) { | const std::string& category, const Blob& key) { | ||||
decltype(m_cache.begin()) iter0; | decltype(m_cache.begin()) iter0; | ||||
@@ -240,6 +249,11 @@ void InFilePersistentCache::put( | |||||
if (m_cache.size() > size0) { | if (m_cache.size() > size0) { | ||||
mgb_log_debug("new cache category: %s", category.c_str()); | mgb_log_debug("new cache category: %s", category.c_str()); | ||||
} | } | ||||
if (m_always_open_file) { | |||||
m_always_open_file->set_head(); | |||||
dump_cache(m_always_open_file.get()); | |||||
m_always_open_file->flush(); | |||||
} | |||||
} | } | ||||
// vim: syntax=cpp.doxygen foldmethod=marker foldmarker=f{{{,f}}} | // vim: syntax=cpp.doxygen foldmethod=marker foldmarker=f{{{,f}}} |
@@ -54,13 +54,14 @@ class InFilePersistentCache final : public PersistentCache { | |||||
std::unordered_map<BlobStorage, BlobStorage, BlobStorage::Hash>> | std::unordered_map<BlobStorage, BlobStorage, BlobStorage::Hash>> | ||||
m_cache; | m_cache; | ||||
MGB_MUTEX m_mtx; | MGB_MUTEX m_mtx; | ||||
std::shared_ptr<OutputFile> m_always_open_file; | |||||
template <typename Input> | template <typename Input> | ||||
void read_cache(Input& inp); | void read_cache(Input& inp); | ||||
public: | public: | ||||
InFilePersistentCache() = default; | InFilePersistentCache() = default; | ||||
InFilePersistentCache(const char* path); | |||||
InFilePersistentCache(const char* path, bool always_open = false); | |||||
InFilePersistentCache(const uint8_t* bin, size_t size); | InFilePersistentCache(const uint8_t* bin, size_t size); | ||||
/** | /** | ||||
@@ -68,6 +69,7 @@ public: | |||||
* file. | * file. | ||||
*/ | */ | ||||
void dump_cache(const char* path); | void dump_cache(const char* path); | ||||
void dump_cache(OutputFile* out_file); | |||||
Maybe<Blob> get(const std::string& category, const Blob& key) override; | Maybe<Blob> get(const std::string& category, const Blob& key) override; | ||||
void put(const std::string& category, const Blob& key, const Blob& value) override; | void put(const std::string& category, const Blob& key, const Blob& value) override; | ||||