深入理解 Gem5 之五

Gem5 中的 Cache

CacheBlock

replaceable Entry

ReplaceableEntry 类描述了 Cache 中具有可替换功能的基本项。除表示该项在 Cache的位置(特定组和路)外,还包含了指向 ReplacementData 智能指针。注意,在使用 ReplaceableEntry 类之前,必须由 replacement_policy 来将 ReplacementData 实例化。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
 /* The replacement data needed by replacement policies. Each replacement policy
* should have its own implementation of replacement data.
*/
struct ReplacementData {};

class ReplaceableEntry {
protected:
// Set & way to which this entry belongs.
uint32_t _set;
uint32_t _way;
public:
/* Replacement data associated to this entry.
* It must be instantiated by the replacement policy before being used. */
std::shared_ptr<replacement_policy::ReplacementData> replacementData;
}

Tagged Entry

TaggedEntry 是包含有 tag 的条目。每个 tag 都附有一个安全 bit,用于标识它是否位于安全地址空间内。TaggedEntry 的内容仅在 valid 位有效时才有意义。

1
2
3
4
5
6
7
8
9
10
class TaggedEntry : public ReplaceableEntry {
// Valid bit. The contents of this entry are only valid if it is set.
bool _valid;

// Secure bit. Marks whether entry's address in the secure memory space.
bool _secure;

/// The entry's tag.
Addr _tag;
};

CacheBlk

CacheBlk 类包含了与一致性、预取状态以及指向其数据指针等相关的信息。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class CacheBlk : public TaggedEntry {
public:
// Contains a copy of the data in this block for easy access.
uint8_t *data = nullptr;

/**
* Cache block's enum listing the supported coherence bits.
*/
enum CoherenceBits : unsigned {
/** write permission */
WritableBit = 0x02,
/**
* Read permission. Note that a block can be valid but not readable
* if there is an outstanding write upgrade miss.
*/
ReadableBit = 0x04,
/** dirty (modified) */
DirtyBit = 0x08,

/**
* Helper enum value that includes all other bits. Whenever a new
* bits is added, this should be updated.
*/
AllBits = 0x0E,
};
protected:
/**
* Represents that the indicated thread context has a "lock" on
* the block, in the LL/SC sense.
*/
class Lock {};

/** The current coherence status of this block. */
unsigned coherence;
}

Queue

QueueEntry

QueueEntry 类描述了 Queue 中的基本表项。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
class QueueEntry : public Packet::SenderState, public Named {
protected:
/** Tick when ready to issue */
Tick readyTime;
/** True if the entry is uncacheable */
bool _isUncacheable;

public:
/** True if the entry has been sent downstream. */
bool inService;
/** Order number assigned to disambiguate writes and misses. */
Counter order;
/** Block aligned address. */
Addr blkAddr;
/** Block size of the cache. */
unsigned blkSize;
/** True if the entry targets the secure memory space. */
bool isSecure;

/* Since multiple references to the same
* address can arrive while a packet is not serviced, each packet is
* stored in a target containing its availability, order and other info,
* and the queue entry stores these similar targets in a list. */
class Target {
public:
const Tick recvTime; //!< Time when request was received (for stats)
const Tick readyTime; //!< Time when request is ready to be serviced
const Counter order; //!< Global order (for memory consistency mgmt)
const PacketPtr pkt; //!< Pending request packet.
};
};

Queue

gem5 中实现了一个高级别的队列接口模板类 Queue,规定队列中只允许放入从 QueueEntry 派生的子类。文章后面我们会了解到,在 Queue 类的基础上,gem5 又分别为 Cache 实现了 MSHR 队列和 Cache 的 Write buffer。可以看出,Queue 在 gem5 的 Cache 系统中显得尤为重要。

该模板类分别继承父类 DrainableNamed 类,它们都在之前的博文中有过介绍。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
template<class Entry>
class Queue : public Drainable, public Named {
protected:
/** Local label (for functional print requests) */
const std::string label;

/* The total number of entries in this queue. */
const int numEntries;

/** The number of entries to hold as a temporary overflow space. */
const int numReserve;
/** Actual storage. */
std::vector<Entry> entries;
/** Holds pointers to all allocated entries. */
typename Entry::List allocatedList;
/** Holds pointers to entries that haven't been sent downstream. */
typename Entry::List readyList;
/** Holds non allocated entries. */
typename Entry::List freeList;
/** The number of entries that are in service. */
int _numInService;
/** The number of currently allocated entries. */
int allocated;
}

MSHR

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
class MSHR : public QueueEntry, public Printable {
private:

/** Flag set by downstream caches */
bool downstreamPending;

/**
* Here we use one flag to track both if:
*
* 1. We are going to become owner or not, i.e., we will get the
* block in an ownership state (Owned or Modified) with BlkDirty
* set. This determines whether or not we are going to become the
* responder and ordering point for future requests that we snoop.
*
* 2. We know that we are going to get a writable block, i.e. we
* will get the block in writable state (Exclusive or Modified
* state) with BlkWritable set. That determines whether additional
* targets with needsWritable set will be able to be satisfied, or
* if not should be put on the deferred list to possibly wait for
* another request that does give us writable access.
*
* Condition 2 is actually just a shortcut that saves us from
* possibly building a deferred target list and calling
* promoteWritable() every time we get a writable block. Condition
* 1, tracking ownership, is what is important. However, we never
* receive ownership without marking the block dirty, and
* consequently use pendingModified to track both ownership and
* writability rather than having separate pendingDirty and
* pendingWritable flags.
*/
bool pendingModified;

/** Did we snoop an invalidate while waiting for data? */
bool postInvalidate;

/** Did we snoop a read while waiting for data? */
bool postDowngrade;

public:

/** Track if we sent this as a whole line write or not */
bool wasWholeLineWrite;

/** True if the entry is just a simple forward from an upper level */
bool isForward;

class Target : public QueueEntry::Target
{
public:

enum Source
{
FromCPU,
FromSnoop,
FromPrefetcher
};

const Source source; //!< Request from cpu, memory, or prefetcher?

/**
* We use this flag to track whether we have cleared the
* downstreamPending flag for the MSHR of the cache above
* where this packet originates from and guard noninitial
* attempts to clear it.
*
* The flag markedPending needs to be updated when the
* TargetList is in service which can be:
* 1) during the Target instantiation if the MSHR is in
* service and the target is not deferred,
* 2) when the MSHR becomes in service if the target is not
* deferred,
* 3) or when the TargetList is promoted (deferredTargets ->
* targets).
*/
bool markedPending;

const bool allocOnFill; //!< Should the response servicing this
//!< target list allocate in the cache?
};
class TargetList : public std::list<Target>, public Named {
public:
bool needsWritable;
bool hasUpgrade;
/** Set when the response should allocate on fill */
bool allocOnFill;
/**
* Determine whether there was at least one non-snooping
* target coming from another cache.
*/
bool hasFromCache;
private:
/** Address of the cache block for this list of targets. */
Addr blkAddr;

/** Size of the cache block. */
Addr blkSize;

/** Indicates whether we can merge incoming write requests */
bool canMergeWrites;

// NOTE: std::vector<bool> might not meet satisfy the
// ForwardIterator requirement and therefore cannot be used
// for writesBitmap.
/**
* Track which bytes are written by requests in this target
* list.
*/
std::vector<char> writesBitmap;
};
/** A list of MSHRs. */
typedef std::list<MSHR *> List;
/** MSHR list iterator. */
typedef List::iterator Iterator;

/**
* Pointer to this MSHR on the ready list.
* @sa MissQueue, MSHRQueue::readyList
*/
Iterator readyIter;

/**
* Pointer to this MSHR on the allocated list.
* @sa MissQueue, MSHRQueue::allocatedList
*/
Iterator allocIter;

/** List of all requests that match the address */
TargetList targets;

TargetList deferredTargets;
};

WriteBuffer

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
class WriteQueueEntry : public QueueEntry, public Printable {
public:
class TargetList : public std::list<Target> {
public:
TargetList() {}
void add(PacketPtr pkt, Tick readyTime, Counter order);
bool trySatisfyFunctional(PacketPtr pkt);
void print(std::ostream &os, int verbosity,
const std::string &prefix) const;
};
/** A list of write queue entriess. */
typedef std::list<WriteQueueEntry *> List;
/** WriteQueueEntry list iterator. */
typedef List::iterator Iterator;
bool sendPacket(BaseCache &cache) override;

private:
/** Pointer to this entry on the ready list.
* @sa MissQueue, WriteQueue::readyList
*/
Iterator readyIter;

/** Pointer to this entry on the allocated list.
* @sa MissQueue, WriteQueue::allocatedList
*/
Iterator allocIter;

/** List of all requests that match the address */
TargetList targets;
};

深入理解 Gem5 之五
https://dingfen.github.io/2022/06/18/2022-6-18-gem5-6/
作者
Bill Ding
发布于
2022年6月18日
更新于
2024年4月9日
许可协议