gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
fa_lru.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2018 Inria
3  * Copyright (c) 2013,2016-2018 ARM Limited
4  * All rights reserved.
5  *
6  * The license below extends only to copyright in the software and shall
7  * not be construed as granting a license to any other intellectual
8  * property including but not limited to intellectual property relating
9  * to a hardware implementation of the functionality of the software
10  * licensed hereunder. You may use the software subject to the license
11  * terms below provided that you ensure that this notice is replicated
12  * unmodified and in its entirety in all distributions of the software,
13  * modified or unmodified, in source code or in binary form.
14  *
15  * Copyright (c) 2003-2005 The Regents of The University of Michigan
16  * All rights reserved.
17  *
18  * Redistribution and use in source and binary forms, with or without
19  * modification, are permitted provided that the following conditions are
20  * met: redistributions of source code must retain the above copyright
21  * notice, this list of conditions and the following disclaimer;
22  * redistributions in binary form must reproduce the above copyright
23  * notice, this list of conditions and the following disclaimer in the
24  * documentation and/or other materials provided with the distribution;
25  * neither the name of the copyright holders nor the names of its
26  * contributors may be used to endorse or promote products derived from
27  * this software without specific prior written permission.
28  *
29  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
30  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
31  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
32  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
33  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
34  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
35  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
36  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
37  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
38  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
39  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40  */
41 
47 #include "mem/cache/tags/fa_lru.hh"
48 
49 #include <cassert>
50 #include <sstream>
51 
52 #include "base/intmath.hh"
53 #include "base/logging.hh"
54 #include "mem/cache/base.hh"
56 
57 std::string
59 {
60  return csprintf("%s inCachesMask: %#x", CacheBlk::print(), inCachesMask);
61 }
62 
64  : BaseTags(p),
65 
66  cacheTracking(p.min_tracked_cache_size, size, blkSize, this)
67 {
68  if (!isPowerOf2(blkSize))
69  fatal("cache block size (in bytes) `%d' must be a power of two",
70  blkSize);
71  if (!isPowerOf2(size))
72  fatal("Cache Size must be power of 2 for now");
73 
74  blks = new FALRUBlk[numBlocks];
75 }
76 
78 {
79  delete[] blks;
80 }
81 
82 void
84 {
85  head = &(blks[0]);
86  head->prev = nullptr;
87  head->next = &(blks[1]);
88  head->setPosition(0, 0);
89  head->data = &dataBlks[0];
90 
91  for (unsigned i = 1; i < numBlocks - 1; i++) {
92  blks[i].prev = &(blks[i-1]);
93  blks[i].next = &(blks[i+1]);
94  blks[i].setPosition(0, i);
95 
96  // Associate a data chunk to the block
97  blks[i].data = &dataBlks[blkSize*i];
98  }
99 
100  tail = &(blks[numBlocks - 1]);
101  tail->prev = &(blks[numBlocks - 2]);
102  tail->next = nullptr;
103  tail->setPosition(0, numBlocks - 1);
104  tail->data = &dataBlks[(numBlocks - 1) * blkSize];
105 
107 }
108 
109 void
111 {
112  // Erase block entry reference in the hash table
113  M5_VAR_USED auto num_erased =
114  tagHash.erase(std::make_pair(blk->getTag(), blk->isSecure()));
115 
116  // Sanity check; only one block reference should be erased
117  assert(num_erased == 1);
118 
119  // Invalidate block entry. Must be done after the hash is erased
121 
122  // Decrease the number of tags in use
123  stats.tagsInUse--;
124 
125  // Move the block to the tail to make it the next victim
126  moveToTail((FALRUBlk*)blk);
127 }
128 
129 CacheBlk*
130 FALRU::accessBlock(Addr addr, bool is_secure, Cycles &lat)
131 {
132  return accessBlock(addr, is_secure, lat, 0);
133 }
134 
135 CacheBlk*
136 FALRU::accessBlock(Addr addr, bool is_secure, Cycles &lat,
137  CachesMask *in_caches_mask)
138 {
139  CachesMask mask = 0;
140  FALRUBlk* blk = static_cast<FALRUBlk*>(findBlock(addr, is_secure));
141 
142  // If a cache hit
143  if (blk && blk->isValid()) {
144  mask = blk->inCachesMask;
145 
146  moveToHead(blk);
147  }
148 
149  if (in_caches_mask) {
150  *in_caches_mask = mask;
151  }
152 
154 
155  // The tag lookup latency is the same for a hit or a miss
156  lat = lookupLatency;
157 
158  return blk;
159 }
160 
161 CacheBlk*
162 FALRU::findBlock(Addr addr, bool is_secure) const
163 {
164  FALRUBlk* blk = nullptr;
165 
166  Addr tag = extractTag(addr);
167  auto iter = tagHash.find(std::make_pair(tag, is_secure));
168  if (iter != tagHash.end()) {
169  blk = (*iter).second;
170  }
171 
172  if (blk && blk->isValid()) {
173  assert(blk->getTag() == tag);
174  assert(blk->isSecure() == is_secure);
175  }
176 
177  return blk;
178 }
179 
181 FALRU::findBlockBySetAndWay(int set, int way) const
182 {
183  assert(set == 0);
184  return &blks[way];
185 }
186 
187 CacheBlk*
188 FALRU::findVictim(Addr addr, const bool is_secure, const std::size_t size,
189  std::vector<CacheBlk*>& evict_blks)
190 {
191  // The victim is always stored on the tail for the FALRU
192  FALRUBlk* victim = tail;
193 
194  // There is only one eviction for this replacement
195  evict_blks.push_back(victim);
196 
197  return victim;
198 }
199 
200 void
202 {
203  FALRUBlk* falruBlk = static_cast<FALRUBlk*>(blk);
204 
205  // Make sure block is not present in the cache
206  assert(falruBlk->inCachesMask == 0);
207 
208  // Do common block insertion functionality
209  BaseTags::insertBlock(pkt, blk);
210 
211  // Increment tag counter
212  stats.tagsInUse++;
213 
214  // New block is the MRU
215  moveToHead(falruBlk);
216 
217  // Insert new block in the hash table
218  tagHash[std::make_pair(blk->getTag(), blk->isSecure())] = falruBlk;
219 }
220 
221 void
222 FALRU::moveBlock(CacheBlk *src_blk, CacheBlk *dest_blk)
223 {
224  panic("Moving blocks in FALRU has not been implemented");
225 }
226 
227 void
229 {
230  // If block is not already head, do the moving
231  if (blk != head) {
233  // If block is tail, set previous block as new tail
234  if (blk == tail){
235  assert(blk->next == nullptr);
236  tail = blk->prev;
237  tail->next = nullptr;
238  // Inform block's surrounding blocks that it has been moved
239  } else {
240  blk->prev->next = blk->next;
241  blk->next->prev = blk->prev;
242  }
243 
244  // Swap pointers
245  blk->next = head;
246  blk->prev = nullptr;
247  head->prev = blk;
248  head = blk;
249 
251  }
252 }
253 
254 void
256 {
257  // If block is not already tail, do the moving
258  if (blk != tail) {
260  // If block is head, set next block as new head
261  if (blk == head){
262  assert(blk->prev == nullptr);
263  head = blk->next;
264  head->prev = nullptr;
265  // Inform block's surrounding blocks that it has been moved
266  } else {
267  blk->prev->next = blk->next;
268  blk->next->prev = blk->prev;
269  }
270 
271  // Swap pointers
272  blk->prev = tail;
273  blk->next = nullptr;
274  tail->next = blk;
275  tail = blk;
276 
278  }
279 }
280 
281 void
282 printSize(std::ostream &stream, size_t size)
283 {
284  static const char *SIZES[] = { "B", "kB", "MB", "GB", "TB", "ZB" };
285  int div = 0;
286  while (size >= 1024 && div < (sizeof SIZES / sizeof *SIZES)) {
287  div++;
288  size >>= 10;
289  }
290  stream << size << SIZES[div];
291 }
292 
293 FALRU::CacheTracking::CacheTracking(unsigned min_size, unsigned max_size,
294  unsigned block_size, Stats::Group *parent)
295  : Stats::Group(parent),
296  blkSize(block_size),
297  minTrackedSize(min_size),
298  numTrackedCaches(max_size > min_size ?
299  floorLog2(max_size) - floorLog2(min_size) : 0),
300  inAllCachesMask(mask(numTrackedCaches)),
301  boundaries(numTrackedCaches),
302  ADD_STAT(hits, UNIT_COUNT, "The number of hits in each cache size."),
303  ADD_STAT(misses, UNIT_COUNT, "The number of misses in each cache size."),
304  ADD_STAT(accesses, UNIT_COUNT,
305  "The number of accesses to the FA LRU cache.")
306 {
307  fatal_if(numTrackedCaches > sizeof(CachesMask) * 8,
308  "Not enough bits (%s) in type CachesMask type to keep "
309  "track of %d caches\n", sizeof(CachesMask),
311 
312  hits
313  .init(numTrackedCaches + 1);
314  misses
315  .init(numTrackedCaches + 1);
316 
317  for (unsigned i = 0; i < numTrackedCaches + 1; ++i) {
318  std::stringstream size_str;
319  printSize(size_str, minTrackedSize << i);
320  hits.subname(i, size_str.str());
321  hits.subdesc(i, "Hits in a " + size_str.str() + " cache");
322  misses.subname(i, size_str.str());
323  misses.subdesc(i, "Misses in a " + size_str.str() + " cache");
324  }
325 }
326 
327 void
329 {
330 #ifdef FALRU_DEBUG
331  const FALRUBlk* blk = head;
332  unsigned curr_size = 0;
333  unsigned tracked_cache_size = minTrackedSize;
334  CachesMask in_caches_mask = inAllCachesMask;
335  int j = 0;
336 
337  while (blk) {
338  panic_if(blk->inCachesMask != in_caches_mask, "Expected cache mask "
339  "%x found %x", blk->inCachesMask, in_caches_mask);
340 
341  curr_size += blkSize;
342  if (curr_size == tracked_cache_size && blk != tail) {
343  panic_if(boundaries[j] != blk, "Unexpected boundary for the %d-th "
344  "cache", j);
345  tracked_cache_size <<= 1;
346  // from this point, blocks fit only in the larger caches
347  in_caches_mask &= ~(1U << j);
348  ++j;
349  }
350  blk = blk->next;
351  }
352 #endif // FALRU_DEBUG
353 }
354 
355 void
357 {
358  // early exit if we are not tracking any extra caches
359  FALRUBlk* blk = numTrackedCaches ? head : nullptr;
360  unsigned curr_size = 0;
361  unsigned tracked_cache_size = minTrackedSize;
362  CachesMask in_caches_mask = inAllCachesMask;
363  int j = 0;
364 
365  while (blk) {
366  blk->inCachesMask = in_caches_mask;
367 
368  curr_size += blkSize;
369  if (curr_size == tracked_cache_size && blk != tail) {
370  boundaries[j] = blk;
371 
372  tracked_cache_size <<= 1;
373  // from this point, blocks fit only in the larger caches
374  in_caches_mask &= ~(1U << j);
375  ++j;
376  }
377  blk = blk->next;
378  }
379 }
380 
381 
382 void
384 {
385  // Get the mask of all caches, in which the block didn't fit
386  // before moving it to the head
387  CachesMask update_caches_mask = inAllCachesMask ^ blk->inCachesMask;
388 
389  for (int i = 0; i < numTrackedCaches; i++) {
390  CachesMask current_cache_mask = 1U << i;
391  if (current_cache_mask & update_caches_mask) {
392  // if the ith cache didn't fit the block (before it is moved to
393  // the head), move the ith boundary 1 block closer to the
394  // MRU
395  boundaries[i]->inCachesMask &= ~current_cache_mask;
396  boundaries[i] = boundaries[i]->prev;
397  } else if (boundaries[i] == blk) {
398  // Make sure the boundary doesn't point to the block
399  // we are about to move
400  boundaries[i] = blk->prev;
401  }
402  }
403 
404  // Make block reside in all caches
405  blk->inCachesMask = inAllCachesMask;
406 }
407 
408 void
410 {
411  CachesMask update_caches_mask = blk->inCachesMask;
412 
413  for (int i = 0; i < numTrackedCaches; i++) {
414  CachesMask current_cache_mask = 1U << i;
415  if (current_cache_mask & update_caches_mask) {
416  // if the ith cache fitted the block (before it is moved to
417  // the tail), move the ith boundary 1 block closer to the
418  // LRU
419  boundaries[i] = boundaries[i]->next;
420  if (boundaries[i] == blk) {
421  // Make sure the boundary doesn't point to the block
422  // we are about to move
423  boundaries[i] = blk->next;
424  }
425  boundaries[i]->inCachesMask |= current_cache_mask;
426  }
427  }
428 
429  // The block now fits only in the actual cache
430  blk->inCachesMask = 0;
431 }
432 
433 void
435 {
436  for (int i = 0; i < numTrackedCaches; i++) {
437  if (blk && ((1U << i) & blk->inCachesMask)) {
438  hits[i]++;
439  } else {
440  misses[i]++;
441  }
442  }
443 
444  // Record stats for the actual cache too
445  if (blk && blk->isValid()) {
446  hits[numTrackedCaches]++;
447  } else {
448  misses[numTrackedCaches]++;
449  }
450 
451  accesses++;
452 }
453 
fatal
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:183
FALRU::CacheTracking::numTrackedCaches
const int numTrackedCaches
The number of different size caches being tracked.
Definition: fa_lru.hh:344
ReplaceableEntry
A replaceable entry is a basic entry in a 2d table-like structure that needs to have replacement func...
Definition: replaceable_entry.hh:57
BaseTags::dataBlks
std::unique_ptr< uint8_t[]> dataBlks
The data blocks, 1 per cache block.
Definition: base.hh:100
FALRU::cacheTracking
CacheTracking cacheTracking
Definition: fa_lru.hh:369
FALRU::head
FALRUBlk * head
The MRU block.
Definition: fa_lru.hh:117
FALRUBlk::next
FALRUBlk * next
The next block in LRU order.
Definition: fa_lru.hh:89
FALRUBlk::print
std::string print() const override
Pretty-print inCachesMask and other CacheBlk information.
Definition: fa_lru.cc:58
FALRU::CacheTracking::minTrackedSize
const unsigned minTrackedSize
The smallest cache we are tracking.
Definition: fa_lru.hh:342
BaseTags::numBlocks
const unsigned numBlocks
the number of blocks in the cache
Definition: base.hh:97
base.hh
FALRU::CacheTracking::recordAccess
void recordAccess(FALRUBlk *blk)
Notify of a block access.
Definition: fa_lru.cc:434
FALRUBlk::inCachesMask
CachesMask inCachesMask
A bit mask of the caches that fit this block.
Definition: fa_lru.hh:92
FALRU::accessBlock
CacheBlk * accessBlock(Addr addr, bool is_secure, Cycles &lat, CachesMask *in_cache_mask)
Access block and update replacement data.
Definition: fa_lru.cc:136
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
fa_lru.hh
FALRU::CacheTracking::misses
Stats::Vector misses
Misses in each cache.
Definition: fa_lru.hh:361
FALRU::tail
FALRUBlk * tail
The LRU block.
Definition: fa_lru.hh:119
FALRU::moveBlock
void moveBlock(CacheBlk *src_blk, CacheBlk *dest_blk) override
Move a block's metadata to another location decided by the replacement policy.
Definition: fa_lru.cc:222
std::vector
STL vector class.
Definition: stl.hh:37
FALRU::insertBlock
void insertBlock(const PacketPtr pkt, CacheBlk *blk) override
Insert the new block into the cache and update replacement data.
Definition: fa_lru.cc:201
replaceable_entry.hh
FALRU::findBlock
CacheBlk * findBlock(Addr addr, bool is_secure) const override
Find the block in the cache, do not update the replacement data.
Definition: fa_lru.cc:162
BaseTags::invalidate
virtual void invalidate(CacheBlk *blk)
This function updates the tags when a block is invalidated.
Definition: base.hh:251
FALRU::FALRU
FALRU(const Params &p)
Construct and initialize this cache tagstore.
Definition: fa_lru.cc:63
CachesMask
uint32_t CachesMask
Definition: fa_lru.hh:70
ArmISA::j
Bitfield< 24 > j
Definition: miscregs_types.hh:54
FALRU::moveToTail
void moveToTail(FALRUBlk *blk)
Move a cache block to the LRU position.
Definition: fa_lru.cc:255
BaseTags::BaseTagStats::tagsInUse
Stats::Average tagsInUse
Per tick average of the number of tags that hold valid data.
Definition: base.hh:115
ADD_STAT
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Definition: group.hh:71
FALRU::CacheTracking::moveBlockToHead
void moveBlockToHead(FALRUBlk *blk)
Update boundaries as a block will be moved to the MRU.
Definition: fa_lru.cc:383
BaseTags
A common base class of Cache tagstore objects.
Definition: base.hh:70
FALRUBlk
A fully associative cache block.
Definition: fa_lru.hh:81
TaggedEntry::isSecure
bool isSecure() const
Check if this block holds data from the secure memory space.
Definition: tagged_entry.hh:61
BaseTags::insertBlock
virtual void insertBlock(const PacketPtr pkt, CacheBlk *blk)
Insert the new block into the cache and update stats.
Definition: base.cc:99
FALRU::tagsInit
void tagsInit() override
Initialize blocks as FALRUBlk instances.
Definition: fa_lru.cc:83
printSize
void printSize(std::ostream &stream, size_t size)
Definition: fa_lru.cc:282
ReplaceableEntry::setPosition
virtual void setPosition(const uint32_t set, const uint32_t way)
Set both the set and way.
Definition: replaceable_entry.hh:87
BaseTags::Params
BaseTagsParams Params
Definition: base.hh:158
FALRUBlk::prev
FALRUBlk * prev
The previous block in LRU order.
Definition: fa_lru.hh:87
FALRU::findVictim
CacheBlk * findVictim(Addr addr, const bool is_secure, const std::size_t size, std::vector< CacheBlk * > &evict_blks) override
Find replacement victim based on address.
Definition: fa_lru.cc:188
FALRU::~FALRU
~FALRU()
Definition: fa_lru.cc:77
BaseTags::blkSize
const unsigned blkSize
The block size of the cache.
Definition: base.hh:74
Stats::DataWrapVec::subdesc
Derived & subdesc(off_type index, const std::string &desc)
Set the subfield description for the given index and marks this stat to print at the end of simulatio...
Definition: statistics.hh:407
CacheBlk::print
std::string print() const override
Pretty-print tag, set and way, and interpret state bits to readable form including mapping to a MOESI...
Definition: cache_blk.hh:360
TaggedEntry::isValid
virtual bool isValid() const
Checks if the entry is valid.
Definition: tagged_entry.hh:54
UNIT_COUNT
#define UNIT_COUNT
Definition: units.hh:49
FALRU::CacheTracking::check
void check(const FALRUBlk *head, const FALRUBlk *tail) const
Check that the tracking mechanism is in consistent state.
Definition: fa_lru.cc:328
FALRU::CacheTracking::moveBlockToTail
void moveBlockToTail(FALRUBlk *blk)
Update boundaries as a block will be moved to the LRU.
Definition: fa_lru.cc:409
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
FALRU::tagHash
TagHash tagHash
The address hash table.
Definition: fa_lru.hh:134
Stats::VectorBase::init
Derived & init(size_type size)
Set this vector to have the given size.
Definition: statistics.hh:1028
FALRU::invalidate
void invalidate(CacheBlk *blk) override
Invalidate a cache block.
Definition: fa_lru.cc:110
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
BaseTags::stats
BaseTags::BaseTagStats stats
FALRU::CacheTracking::init
void init(FALRUBlk *head, FALRUBlk *tail)
Initialiaze cache blocks and the tracking mechanism.
Definition: fa_lru.cc:356
floorLog2
std::enable_if_t< std::is_integral< T >::value, int > floorLog2(T x)
Definition: intmath.hh:63
panic_if
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:197
FALRU::CacheTracking::CacheTracking
CacheTracking(unsigned min_size, unsigned max_size, unsigned block_size, Stats::Group *parent)
Definition: fa_lru.cc:293
CacheBlk::data
uint8_t * data
Contains a copy of the data in this block for easy access.
Definition: cache_blk.hh:100
FALRU::moveToHead
void moveToHead(FALRUBlk *blk)
Move a cache block to the MRU position.
Definition: fa_lru.cc:228
CacheBlk
A Basic Cache block.
Definition: cache_blk.hh:67
TaggedEntry::getTag
virtual Addr getTag() const
Get tag associated to this block.
Definition: tagged_entry.hh:68
FALRU::blks
FALRUBlk * blks
The cache blocks.
Definition: fa_lru.hh:114
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
Stats::Group
Statistics container.
Definition: group.hh:87
logging.hh
Cycles
Cycles is a wrapper class for representing cycle counts, i.e.
Definition: types.hh:79
Stats::DataWrapVec::subname
Derived & subname(off_type index, const std::string &name)
Set the subfield name for the given index, and marks this stat to print at the end of simulation.
Definition: statistics.hh:383
BaseTags::lookupLatency
const Cycles lookupLatency
The tag lookup latency of the cache.
Definition: base.hh:80
Stats
Definition: statistics.cc:53
FALRU::extractTag
Addr extractTag(Addr addr) const override
Generate the tag from the addres.
Definition: fa_lru.hh:238
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
intmath.hh
fatal_if
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Definition: logging.hh:219
isPowerOf2
bool isPowerOf2(const T &n)
Definition: intmath.hh:102
csprintf
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
BaseTags::size
const unsigned size
The size of the cache.
Definition: base.hh:78
FALRU::findBlockBySetAndWay
ReplaceableEntry * findBlockBySetAndWay(int set, int way) const override
Find a block given set and way.
Definition: fa_lru.cc:181
ArmISA::mask
Bitfield< 28, 24 > mask
Definition: miscregs_types.hh:711
FALRU::CacheTracking::hits
Stats::Vector hits
Hits in each cache.
Definition: fa_lru.hh:359
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171

Generated on Tue Mar 23 2021 19:41:27 for gem5 by doxygen 1.8.17