gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
cache_blk.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2012-2018 ARM Limited
3  * All rights reserved.
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Copyright (c) 2020 Inria
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 
46 #ifndef __MEM_CACHE_CACHE_BLK_HH__
47 #define __MEM_CACHE_CACHE_BLK_HH__
48 
49 #include <cassert>
50 #include <cstdint>
51 #include <iosfwd>
52 #include <list>
53 #include <string>
54 
55 #include "base/printable.hh"
56 #include "base/types.hh"
58 #include "mem/packet.hh"
59 #include "mem/request.hh"
60 #include "sim/core.hh"
61 
67 class CacheBlk : public TaggedEntry
68 {
69  public:
74  enum CoherenceBits : unsigned
75  {
77  WritableBit = 0x02,
82  ReadableBit = 0x04,
84  DirtyBit = 0x08,
85 
90  AllBits = 0x0E,
91  };
92 
100  uint8_t *data;
101 
107 
108  protected:
113  class Lock {
114  public:
115  ContextID contextId; // locking context
116  Addr lowAddr; // low address of lock range
117  Addr highAddr; // high address of lock range
118 
119  // check for matching execution context, and an address that
120  // is within the lock
121  bool matches(const RequestPtr &req) const
122  {
123  Addr req_low = req->getPaddr();
124  Addr req_high = req_low + req->getSize() -1;
125  return (contextId == req->contextId()) &&
126  (req_low >= lowAddr) && (req_high <= highAddr);
127  }
128 
129  // check if a request is intersecting and thus invalidating the lock
130  bool intersects(const RequestPtr &req) const
131  {
132  Addr req_low = req->getPaddr();
133  Addr req_high = req_low + req->getSize() - 1;
134 
135  return (req_low <= highAddr) && (req_high >= lowAddr);
136  }
137 
138  Lock(const RequestPtr &req)
139  : contextId(req->contextId()),
140  lowAddr(req->getPaddr()),
141  highAddr(lowAddr + req->getSize() - 1)
142  {
143  }
144  };
145 
149 
150  public:
152  {
153  invalidate();
154  }
155 
156  CacheBlk(const CacheBlk&) = delete;
157  CacheBlk& operator=(const CacheBlk&) = delete;
158  CacheBlk(const CacheBlk&&) = delete;
167  virtual CacheBlk&
169  {
170  // Copying an entry into a valid one would imply in skipping all
171  // replacement steps, so it cannot be allowed
172  assert(!isValid());
173  assert(other.isValid());
174 
175  insert(other.getTag(), other.isSecure());
176 
177  if (other.wasPrefetched()) {
178  setPrefetched();
179  }
180  setCoherenceBits(other.coherence);
181  setTaskId(other.getTaskId());
183  setRefCount(other.getRefCount());
184  setSrcRequestorId(other.getSrcRequestorId());
185  std::swap(lockList, other.lockList);
186 
187  other.invalidate();
188 
189  return *this;
190  }
191  virtual ~CacheBlk() {};
192 
196  virtual void invalidate() override
197  {
199 
200  clearPrefetched();
202 
205  setRefCount(0);
207  lockList.clear();
208  }
209 
215  void
217  {
218  assert(isValid());
219  coherence |= bits;
220  }
221 
227  void clearCoherenceBits(unsigned bits) { coherence &= ~bits; }
228 
234  bool
235  isSet(unsigned bits) const
236  {
237  return isValid() && (coherence & bits);
238  }
239 
245  bool wasPrefetched() const { return _prefetched; }
246 
251  void clearPrefetched() { _prefetched = false; }
252 
254  void setPrefetched() { _prefetched = true; }
255 
262  {
263  assert(whenReady != MaxTick);
264  return whenReady;
265  }
266 
274  void setWhenReady(const Tick tick)
275  {
276  assert(tick >= _tickInserted);
277  whenReady = tick;
278  }
279 
281  uint32_t getTaskId() const { return _taskId; }
282 
284  uint32_t getSrcRequestorId() const { return _srcRequestorId; }
285 
287  unsigned getRefCount() const { return _refCount; }
288 
291 
297  Tick
298  getAge() const
299  {
300  assert(_tickInserted <= curTick());
301  return curTick() - _tickInserted;
302  }
303 
315  void insert(const Addr tag, const bool is_secure,
316  const int src_requestor_ID, const uint32_t task_ID);
317  using TaggedEntry::insert;
318 
324  {
325  assert(pkt->isLLSC());
326  auto l = lockList.begin();
327  while (l != lockList.end()) {
328  if (l->intersects(pkt->req))
329  l = lockList.erase(l);
330  else
331  ++l;
332  }
333 
334  lockList.emplace_front(pkt->req);
335  }
336 
341  void clearLoadLocks(const RequestPtr &req)
342  {
343  auto l = lockList.begin();
344  while (l != lockList.end()) {
345  if (l->intersects(req) && l->contextId != req->contextId()) {
346  l = lockList.erase(l);
347  } else {
348  ++l;
349  }
350  }
351  }
352 
359  std::string
360  print() const override
361  {
386  unsigned state =
387  isSet(WritableBit) << 2 | isSet(DirtyBit) << 1 | isValid();
388  char s = '?';
389  switch (state) {
390  case 0b111: s = 'M'; break;
391  case 0b011: s = 'O'; break;
392  case 0b101: s = 'E'; break;
393  case 0b001: s = 'S'; break;
394  case 0b000: s = 'I'; break;
395  default: s = 'T'; break; // @TODO add other types
396  }
397  return csprintf("state: %x (%c) writable: %d readable: %d "
398  "dirty: %d | %s", coherence, s, isSet(WritableBit),
400  }
401 
408  {
409  assert(pkt->isWrite());
410 
411  // common case
412  if (!pkt->isLLSC() && lockList.empty())
413  return true;
414 
415  const RequestPtr &req = pkt->req;
416 
417  if (pkt->isLLSC()) {
418  // it's a store conditional... have to check for matching
419  // load locked.
420  bool success = false;
421 
422  auto l = lockList.begin();
423  while (!success && l != lockList.end()) {
424  if (l->matches(pkt->req)) {
425  // it's a store conditional, and as far as the
426  // memory system can tell, the requesting
427  // context's lock is still valid.
428  success = true;
429  lockList.erase(l);
430  } else {
431  ++l;
432  }
433  }
434 
435  req->setExtraData(success ? 1 : 0);
436  // clear any intersected locks from other contexts (our LL
437  // should already have cleared them)
438  clearLoadLocks(req);
439  return success;
440  } else {
441  // a normal write, if there is any lock not from this
442  // context we clear the list, thus for a private cache we
443  // never clear locks on normal writes
444  clearLoadLocks(req);
445  return true;
446  }
447  }
448 
449  protected:
451  unsigned coherence;
452 
453  // The following setters have been marked as protected because their
454  // respective variables should only be modified at 2 moments:
455  // invalidation and insertion. Because of that, they shall only be
456  // called by the functions that perform those actions.
457 
459  void setTaskId(const uint32_t task_id) { _taskId = task_id; }
460 
462  void setSrcRequestorId(const uint32_t id) { _srcRequestorId = id; }
463 
465  void setRefCount(const unsigned count) { _refCount = count; }
466 
469 
470  private:
472  uint32_t _taskId;
473 
476 
478  unsigned _refCount;
479 
485 
488 };
489 
495 class TempCacheBlk final : public CacheBlk
496 {
497  private:
502 
503  public:
508  TempCacheBlk(unsigned size) : CacheBlk()
509  {
510  data = new uint8_t[size];
511  }
512  TempCacheBlk(const TempCacheBlk&) = delete;
513  TempCacheBlk& operator=(const TempCacheBlk&) = delete;
514  ~TempCacheBlk() { delete [] data; };
515 
519  void invalidate() override {
521 
522  _addr = MaxAddr;
523  }
524 
525  void
526  insert(const Addr addr, const bool is_secure) override
527  {
528  CacheBlk::insert(addr, is_secure);
529  _addr = addr;
530  }
531 
537  Addr getAddr() const
538  {
539  return _addr;
540  }
541 };
542 
550 {
552  public:
555  void print(std::ostream &o, int verbosity = 0,
556  const std::string &prefix = "") const;
557 };
558 
559 #endif //__MEM_CACHE_CACHE_BLK_HH__
tagged_entry.hh
TaggedEntry::print
std::string print() const override
Prints relevant information about this entry.
Definition: tagged_entry.hh:108
CacheBlk::Lock::highAddr
Addr highAddr
Definition: cache_blk.hh:117
CacheBlk::Lock::intersects
bool intersects(const RequestPtr &req) const
Definition: cache_blk.hh:130
TaggedEntry::insert
virtual void insert(const Addr tag, const bool is_secure)
Insert the block by assigning it a tag and marking it valid.
Definition: tagged_entry.hh:90
CacheBlk::clearLoadLocks
void clearLoadLocks(const RequestPtr &req)
Clear the any load lock that intersect the request, and is from a different context.
Definition: cache_blk.hh:341
CacheBlk::setRefCount
void setRefCount(const unsigned count)
Set the number of references to this block since insertion.
Definition: cache_blk.hh:465
CacheBlk::clearCoherenceBits
void clearCoherenceBits(unsigned bits)
Clear the corresponding coherence bits.
Definition: cache_blk.hh:227
CacheBlk::CacheBlk
CacheBlk()
Definition: cache_blk.hh:151
CacheBlk::getWhenReady
Tick getWhenReady() const
Get tick at which block's data will be available for access.
Definition: cache_blk.hh:261
TaggedEntry
Copyright (c) 2020 Inria All rights reserved.
Definition: tagged_entry.hh:43
CacheBlk::clearPrefetched
void clearPrefetched()
Clear the prefetching bit.
Definition: cache_blk.hh:251
CacheBlk::Lock::matches
bool matches(const RequestPtr &req) const
Definition: cache_blk.hh:121
Request::invldRequestorId
@ invldRequestorId
Invalid requestor id for assertion checking only.
Definition: request.hh:259
ContextID
int ContextID
Globally unique thread context ID.
Definition: types.hh:237
CacheBlk::coherence
unsigned coherence
The current coherence status of this block.
Definition: cache_blk.hh:451
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
CacheBlkPrintWrapper::~CacheBlkPrintWrapper
virtual ~CacheBlkPrintWrapper()
Definition: cache_blk.hh:554
RequestPtr
std::shared_ptr< Request > RequestPtr
Definition: request.hh:86
Packet::req
RequestPtr req
A pointer to the original request.
Definition: packet.hh:341
CacheBlk::Lock::Lock
Lock(const RequestPtr &req)
Definition: cache_blk.hh:138
Packet::isLLSC
bool isLLSC() const
Definition: packet.hh:583
X86ISA::count
count
Definition: misc.hh:703
MaxAddr
const Addr MaxAddr
Definition: types.hh:172
ContextSwitchTaskId::Unknown
@ Unknown
Definition: request.hh:79
request.hh
printable.hh
CacheBlk::getSrcRequestorId
uint32_t getSrcRequestorId() const
Get the requestor id associated to this block.
Definition: cache_blk.hh:284
packet.hh
TempCacheBlk::insert
void insert(const Addr addr, const bool is_secure) override
Insert the block by assigning it a tag and marking it valid.
Definition: cache_blk.hh:526
CacheBlk::lockList
std::list< Lock > lockList
List of thread contexts that have performed a load-locked (LL) on the block since the last store.
Definition: cache_blk.hh:148
CacheBlk::invalidate
virtual void invalidate() override
Invalidate the block and clear all state.
Definition: cache_blk.hh:196
TempCacheBlk::~TempCacheBlk
~TempCacheBlk()
Definition: cache_blk.hh:514
CacheBlk::_refCount
unsigned _refCount
Number of references to this block since it was brought in.
Definition: cache_blk.hh:478
CacheBlk::operator=
virtual CacheBlk & operator=(CacheBlk &&other)
Move assignment operator.
Definition: cache_blk.hh:168
CacheBlk::setTickInserted
void setTickInserted()
Set the current tick as this block's insertion tick.
Definition: cache_blk.hh:468
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
Printable
Abstract base class for objects which support being printed to a stream for debugging.
Definition: printable.hh:44
CacheBlk::_prefetched
bool _prefetched
Whether this block is an unaccessed hardware prefetch.
Definition: cache_blk.hh:487
CacheBlkPrintWrapper
Simple class to provide virtual print() method on cache blocks without allocating a vtable pointer fo...
Definition: cache_blk.hh:549
CacheBlk::checkWrite
bool checkWrite(PacketPtr pkt)
Handle interaction of load-locked operations and stores.
Definition: cache_blk.hh:407
TempCacheBlk::TempCacheBlk
TempCacheBlk(unsigned size)
Creates a temporary cache block, with its own storage.
Definition: cache_blk.hh:508
CacheBlk::Lock::lowAddr
Addr lowAddr
Definition: cache_blk.hh:116
CacheBlk::getTaskId
uint32_t getTaskId() const
Get the task id associated to this block.
Definition: cache_blk.hh:281
CacheBlk::WritableBit
@ WritableBit
write permission
Definition: cache_blk.hh:77
core.hh
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
CacheBlk::setPrefetched
void setPrefetched()
Marks this blocks as a recently prefetched block.
Definition: cache_blk.hh:254
CacheBlk::wasPrefetched
bool wasPrefetched() const
Check if this block was the result of a hardware prefetch, yet to be touched.
Definition: cache_blk.hh:245
CacheBlk::Lock::contextId
ContextID contextId
Definition: cache_blk.hh:115
CacheBlk::AllBits
@ AllBits
Helper enum value that includes all other bits.
Definition: cache_blk.hh:90
TempCacheBlk::_addr
Addr _addr
Copy of the block's address, used to regenerate tempBlock's address.
Definition: cache_blk.hh:501
CacheBlk::isSet
bool isSet(unsigned bits) const
Checks the given coherence bits are set.
Definition: cache_blk.hh:235
CacheBlk::setCoherenceBits
void setCoherenceBits(unsigned bits)
Sets the corresponding coherence bits.
Definition: cache_blk.hh:216
CacheBlk::setWhenReady
void setWhenReady(const Tick tick)
Set tick at which block's data will be available for access.
Definition: cache_blk.hh:274
CacheBlk::getAge
Tick getAge() const
Get the block's age, that is, the number of ticks since its insertion.
Definition: cache_blk.hh:298
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
CacheBlkPrintWrapper::print
void print(std::ostream &o, int verbosity=0, const std::string &prefix="") const
Definition: cache_blk.cc:69
CacheBlk::setSrcRequestorId
void setSrcRequestorId(const uint32_t id)
Set the source requestor id.
Definition: cache_blk.hh:462
CacheBlk::CoherenceBits
CoherenceBits
Cache block's enum listing the supported coherence bits.
Definition: cache_blk.hh:74
CacheBlk::data
uint8_t * data
Contains a copy of the data in this block for easy access.
Definition: cache_blk.hh:100
CacheBlk::insert
void insert(const Addr tag, const bool is_secure, const int src_requestor_ID, const uint32_t task_ID)
Set member variables when a block insertion occurs.
Definition: cache_blk.cc:47
TaggedEntry::invalidate
virtual void invalidate()
Invalidate the block.
Definition: tagged_entry.hh:100
CacheBlk
A Basic Cache block.
Definition: cache_blk.hh:67
types.hh
CacheBlk::increaseRefCount
void increaseRefCount()
Get the number of references to this block since insertion.
Definition: cache_blk.hh:290
CacheBlk::DirtyBit
@ DirtyBit
dirty (modified)
Definition: cache_blk.hh:84
TempCacheBlk::operator=
TempCacheBlk & operator=(const TempCacheBlk &)=delete
CacheBlk::trackLoadLocked
void trackLoadLocked(PacketPtr pkt)
Track the fact that a local locked was issued to the block.
Definition: cache_blk.hh:323
CacheBlkPrintWrapper::blk
CacheBlk * blk
Definition: cache_blk.hh:551
CacheBlk::whenReady
Tick whenReady
Which curTick() will this block be accessible.
Definition: cache_blk.hh:106
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
CacheBlk::setTaskId
void setTaskId(const uint32_t task_id)
Set the task id value.
Definition: cache_blk.hh:459
CacheBlk::_taskId
uint32_t _taskId
Task Id associated with this block.
Definition: cache_blk.hh:472
CacheBlk::getRefCount
unsigned getRefCount() const
Get the number of references to this block since insertion.
Definition: cache_blk.hh:287
Packet::isWrite
bool isWrite() const
Definition: packet.hh:558
CacheBlk::_tickInserted
Tick _tickInserted
Tick on which the block was inserted in the cache.
Definition: cache_blk.hh:484
CacheBlk::operator=
CacheBlk & operator=(const CacheBlk &)=delete
CacheBlkPrintWrapper::CacheBlkPrintWrapper
CacheBlkPrintWrapper(CacheBlk *_blk)
Definition: cache_blk.hh:553
bits
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Definition: bitfield.hh:73
TempCacheBlk::invalidate
void invalidate() override
Invalidate the block and clear all state.
Definition: cache_blk.hh:519
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
std::list
STL list class.
Definition: stl.hh:51
TempCacheBlk
Special instance of CacheBlk for use with tempBlk that deals with its block address regeneration.
Definition: cache_blk.hh:495
CacheBlk::_srcRequestorId
int _srcRequestorId
holds the source requestor ID for this block.
Definition: cache_blk.hh:475
ArmISA::s
Bitfield< 4 > s
Definition: miscregs_types.hh:556
CacheBlk::ReadableBit
@ ReadableBit
Read permission.
Definition: cache_blk.hh:82
MipsISA::l
Bitfield< 5 > l
Definition: pra_constants.hh:320
CacheBlk::~CacheBlk
virtual ~CacheBlk()
Definition: cache_blk.hh:191
MaxTick
const Tick MaxTick
Definition: types.hh:61
csprintf
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
ArmISA::id
Bitfield< 33 > id
Definition: miscregs_types.hh:247
CacheBlk::Lock
Represents that the indicated thread context has a "lock" on the block, in the LL/SC sense.
Definition: cache_blk.hh:113
TempCacheBlk::getAddr
Addr getAddr() const
Get block's address.
Definition: cache_blk.hh:537

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