veriblock-pop-cpp
C++11 Libraries for leveraging VeriBlock Proof-Of-Proof blockchain technology.
small_lfru_cache.hpp
1// Copyright (c) 2019-2022 Xenios SEZC
2// https://www.veriblock.org
3// Distributed under the MIT software license, see the accompanying
4// file LICENSE or http://www.opensource.org/licenses/mit-license.php.
5
6#ifndef VERIBLOCK_POP_CPP_SMALL_LRU_CACHE_HPP
7#define VERIBLOCK_POP_CPP_SMALL_LRU_CACHE_HPP
8
9#include <algorithm>
10#include <array>
11#include <functional>
12#include <memory>
13#include <veriblock/pop/time.hpp>
14
15namespace altintegration {
16namespace cache {
17
24template <typename Key,
25 typename Value,
26 size_t Size,
27 size_t TimeWindow = 10 * 60, // 10 min
28 typename = typename std::enable_if<(Size <= 50)>::type>
31 struct Item {
32 size_t frequency = 0;
33 size_t lastAccessed = 0;
34 Key key;
35 std::shared_ptr<Value> value = nullptr;
36
37 void clear() {
38 frequency = 0;
39 lastAccessed = 0;
40 value.reset();
41 }
42 };
43
44 std::shared_ptr<Value> getOrDefault(
45 Key key, std::function<std::shared_ptr<Value>()> factory) {
46 std::shared_ptr<Value> value;
47 if (!get(key, value)) {
48 value = factory();
49 insert(key, value);
50 }
51
52 return value;
53 }
54
55 void clear() {
56 size_ = 0;
57 for (auto& i : container_) {
58 i.clear();
59 }
60 }
61
62 void insert(Key key, std::shared_ptr<Value> value) {
63 auto current = currentTimestamp4();
64
65 if (size_ < Size) {
66 // cache can accept new elements
67 auto& it = container_[size_++];
68 it.frequency = 0;
69 it.key = key;
70 it.value = std::move(value);
71 it.lastAccessed = current;
72 return;
73 }
74
75 // it's time to evict someone
76 auto begin = container_.begin();
77 auto end = container_.end();
78 // iterator to least frequency item
79 auto min = begin;
80 // iterator to least recently used item
81 auto leastRecent = begin;
82 for (auto it = begin; it != end; ++it) {
83 if (it->frequency < min->frequency) {
84 min = it;
85 }
86
87 if (it->lastAccessed < leastRecent->lastAccessed ||
88 (it->lastAccessed == leastRecent->lastAccessed &&
89 it->frequency < leastRecent->frequency)) {
90 leastRecent = it;
91 }
92 }
93
94 // by default, use LFU policy and evict element with min frequency
95 auto evict = min;
96 if (leastRecent->lastAccessed <= current - TimeWindow) {
97 // if least recently element is not used in TimeWindow sec, evict it
98 evict = leastRecent;
99 }
100
101 // do evict
102 evict->key = key;
103 evict->frequency = 0;
104 evict->value = std::move(value);
105 evict->lastAccessed = current;
106 }
107
108 bool get(const Key& key, std::shared_ptr<Value>& out) {
109 // search for key
110 auto begin = container_.begin();
111 auto end = container_.begin() + size_;
112 auto it = std::find_if(
113 begin, end, [&key](const Item& i) { return i.key == key; });
114 if (it == end) {
115 // key not found
116 return false;
117 }
118
119 it->frequency++;
120 it->lastAccessed = currentTimestamp4();
121
122 // return result
123 out = it->value;
124
125 return true;
126 }
127
128 private:
129 std::array<Item, Size> container_;
130
131 // number of elements in a cache
132 size_t size_ = 0;
133};
134
135} // namespace cache
136} // namespace altintegration
137
138#endif // VERIBLOCK_POP_CPP_SMALL_LRU_CACHE_HPP
Defines logging helpers.
Definition: block.hpp:14
uint32_t currentTimestamp4()
Get current time as 4 bytes. If mock time is set, returns mock time.