diff --git a/base/CMakeLists.txt b/base/CMakeLists.txt index 291d6d3189..ba5d50f8d0 100644 --- a/base/CMakeLists.txt +++ b/base/CMakeLists.txt @@ -23,6 +23,7 @@ set( dfa_helpers.hpp exception.cpp exception.hpp + fifo_cache.hpp get_time.hpp gmtime.cpp gmtime.hpp diff --git a/base/fifo_cache.hpp b/base/fifo_cache.hpp new file mode 100644 index 0000000000..da4665d57f --- /dev/null +++ b/base/fifo_cache.hpp @@ -0,0 +1,56 @@ +#pragma once + +#include "base/assert.hpp" + +#include +#include +#include +#include + +template +class FifoCache +{ + template friend class FifoCacheTest; + +public: + using Loader = std::function; + + /// \param capacity maximum size of the cache in number of items. + /// \param loader Function which is called if it's necessary to load a new item for the cache. + FifoCache(size_t capacity, Loader const & loader) : m_capacity(capacity), m_loader(loader) {} + + /// \brief Loads value, if it's necessary, by |key| with |m_loader|, puts it to cache and + /// returns the reference to the value to |m_map|. + Value const & GetValue(const Key & key) + { + auto const it = m_map.find(key); + if (it != m_map.cend()) + return it->second; + + if (Size() >= m_capacity) + Evict(); + + m_list.push_front(key); + + auto & v = m_map[key]; + m_loader(key, v); + return (m_map[key] = v); + } + +private: + void Evict() + { + auto const i = --m_list.end(); + m_map.erase(*i); + m_list.erase(i); + } + + size_t Size() const { return m_map.size(); } + + std::unordered_map m_map; + // @TODO(bykoianko) Consider using a structure like boost/circular_buffer instead of list + // but with handling overwriting shift. We need it to know to remove overwritten key from |m_map|. + std::list m_list; + size_t m_capacity; + Loader m_loader; +}; diff --git a/xcode/base/base.xcodeproj/project.pbxproj b/xcode/base/base.xcodeproj/project.pbxproj index 3b98cba339..ab21dc7d78 100644 --- a/xcode/base/base.xcodeproj/project.pbxproj +++ b/xcode/base/base.xcodeproj/project.pbxproj @@ -56,6 +56,7 @@ 3D74EF151F8B902C0081202C /* bwt.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 3D74EF0F1F8B902C0081202C /* bwt.hpp */; }; 3D7815731F3A145F0068B6AC /* task_loop.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 3D7815711F3A145F0068B6AC /* task_loop.hpp */; }; 3D78157B1F3D89EC0068B6AC /* waiter.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 3D78157A1F3D89EC0068B6AC /* waiter.hpp */; }; + 564BB445206E89ED00BDD211 /* fifo_cache.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 564BB444206E89ED00BDD211 /* fifo_cache.hpp */; }; 56B1A0741E69DE4D00395022 /* random.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56B1A0711E69DE4D00395022 /* random.cpp */; }; 56B1A0751E69DE4D00395022 /* random.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56B1A0721E69DE4D00395022 /* random.hpp */; }; 56B1A0761E69DE4D00395022 /* small_set.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56B1A0731E69DE4D00395022 /* small_set.hpp */; }; @@ -188,6 +189,7 @@ 3D74EF0F1F8B902C0081202C /* bwt.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = bwt.hpp; sourceTree = ""; }; 3D7815711F3A145F0068B6AC /* task_loop.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = task_loop.hpp; sourceTree = ""; }; 3D78157A1F3D89EC0068B6AC /* waiter.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = waiter.hpp; sourceTree = ""; }; + 564BB444206E89ED00BDD211 /* fifo_cache.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = fifo_cache.hpp; sourceTree = ""; }; 56B1A0711E69DE4D00395022 /* random.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = random.cpp; sourceTree = ""; }; 56B1A0721E69DE4D00395022 /* random.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = random.hpp; sourceTree = ""; }; 56B1A0731E69DE4D00395022 /* small_set.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = small_set.hpp; sourceTree = ""; }; @@ -350,6 +352,7 @@ 675341791A3F57BF00A0A8C3 /* base */ = { isa = PBXGroup; children = ( + 564BB444206E89ED00BDD211 /* fifo_cache.hpp */, 675341851A3F57E400A0A8C3 /* array_adapters.hpp */, 675341861A3F57E400A0A8C3 /* assert.hpp */, 675341871A3F57E400A0A8C3 /* base.cpp */, @@ -526,6 +529,7 @@ 6753420B1A3F57E400A0A8C3 /* threaded_container.hpp in Headers */, 672DD4C21E0425600078E13C /* condition.hpp in Headers */, 672DD4C41E0425600078E13C /* newtype.hpp in Headers */, + 564BB445206E89ED00BDD211 /* fifo_cache.hpp in Headers */, 672DD4C31E0425600078E13C /* mem_trie.hpp in Headers */, 672DD4C61E0425600078E13C /* range_iterator.hpp in Headers */, F6F8E3C91EF846CE00F2DE8F /* worker_thread.hpp in Headers */,