gem5  v20.0.0.3
mem_ctrl.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2017-2019 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  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 #include "mem_ctrl.hh"
39 
40 #include "turnaround_policy.hh"
41 
42 namespace QoS {
43 
44 MemCtrl::MemCtrl(const QoSMemCtrlParams * p)
45  : AbstractMemory(p),
46  policy(p->qos_policy),
47  turnPolicy(p->qos_turnaround_policy),
48  queuePolicy(QueuePolicy::create(p)),
49  _numPriorities(p->qos_priorities),
50  qosPriorityEscalation(p->qos_priority_escalation),
51  qosSyncroScheduler(p->qos_syncro_scheduler),
52  totalReadQueueSize(0), totalWriteQueueSize(0),
53  busState(READ), busStateNext(READ),
54  stats(*this)
55 {
56  // Set the priority policy
57  if (policy) {
58  policy->setMemCtrl(this);
59  }
60 
61  // Set the queue priority policy
62  if (queuePolicy) {
63  queuePolicy->setMemCtrl(this);
64  }
65 
66  // Set the bus turnaround policy
67  if (turnPolicy) {
68  turnPolicy->setMemCtrl(this);
69  }
70 
74 }
75 
77 {}
78 
79 void
81 {
83 }
84 
85 void
86 MemCtrl::logRequest(BusState dir, MasterID m_id, uint8_t qos,
87  Addr addr, uint64_t entries)
88 {
89  // If needed, initialize all counters and statistics
90  // for this master
91  addMaster(m_id);
92 
93  DPRINTF(QOS,
94  "QoSMemCtrl::logRequest MASTER %s [id %d] address %d"
95  " prio %d this master q packets %d"
96  " - queue size %d - requested entries %d\n",
97  masters[m_id], m_id, addr, qos, packetPriorities[m_id][qos],
98  (dir == READ) ? readQueueSizes[qos]: writeQueueSizes[qos],
99  entries);
100 
101  if (dir == READ) {
102  readQueueSizes[qos] += entries;
103  totalReadQueueSize += entries;
104  } else if (dir == WRITE) {
105  writeQueueSizes[qos] += entries;
106  totalWriteQueueSize += entries;
107  }
108 
109  packetPriorities[m_id][qos] += entries;
110  for (auto j = 0; j < entries; ++j) {
111  requestTimes[m_id][addr].push_back(curTick());
112  }
113 
114  // Record statistics
115  stats.avgPriority[m_id].sample(qos);
116 
117  // Compute avg priority distance
118 
119  for (uint8_t i = 0; i < packetPriorities[m_id].size(); ++i) {
120  uint8_t distance =
121  (abs(int(qos) - int(i))) * packetPriorities[m_id][i];
122 
123  if (distance > 0) {
124  stats.avgPriorityDistance[m_id].sample(distance);
125  DPRINTF(QOS,
126  "QoSMemCtrl::logRequest MASTER %s [id %d]"
127  " registering priority distance %d for priority %d"
128  " (packets %d)\n",
129  masters[m_id], m_id, distance, i,
130  packetPriorities[m_id][i]);
131  }
132  }
133 
134  DPRINTF(QOS,
135  "QoSMemCtrl::logRequest MASTER %s [id %d] prio %d "
136  "this master q packets %d - new queue size %d\n",
137  masters[m_id], m_id, qos, packetPriorities[m_id][qos],
138  (dir == READ) ? readQueueSizes[qos]: writeQueueSizes[qos]);
139 
140 }
141 
142 void
143 MemCtrl::logResponse(BusState dir, MasterID m_id, uint8_t qos,
144  Addr addr, uint64_t entries, double delay)
145 {
146  panic_if(!hasMaster(m_id),
147  "Logging response with invalid master\n");
148 
149  DPRINTF(QOS,
150  "QoSMemCtrl::logResponse MASTER %s [id %d] address %d prio"
151  " %d this master q packets %d"
152  " - queue size %d - requested entries %d\n",
153  masters[m_id], m_id, addr, qos, packetPriorities[m_id][qos],
154  (dir == READ) ? readQueueSizes[qos]: writeQueueSizes[qos],
155  entries);
156 
157  if (dir == READ) {
158  readQueueSizes[qos] -= entries;
159  totalReadQueueSize -= entries;
160  } else if (dir == WRITE) {
161  writeQueueSizes[qos] -= entries;
162  totalWriteQueueSize -= entries;
163  }
164 
165  panic_if(packetPriorities[m_id][qos] == 0,
166  "QoSMemCtrl::logResponse master %s negative packets for priority"
167  " %d", masters[m_id], qos);
168 
169  packetPriorities[m_id][qos] -= entries;
170 
171  for (auto j = 0; j < entries; ++j) {
172  auto it = requestTimes[m_id].find(addr);
173  panic_if(it == requestTimes[m_id].end(),
174  "QoSMemCtrl::logResponse master %s unmatched response for"
175  " address %d received", masters[m_id], addr);
176 
177  // Load request time
178  uint64_t requestTime = it->second.front();
179 
180  // Remove request entry
181  it->second.pop_front();
182 
183  // Remove whole address entry if last one
184  if (it->second.empty()) {
185  requestTimes[m_id].erase(it);
186  }
187  // Compute latency
188  double latency = (double) (curTick() + delay - requestTime)
190 
191  if (latency > 0) {
192  // Record per-priority latency stats
193  if (stats.priorityMaxLatency[qos].value() < latency) {
194  stats.priorityMaxLatency[qos] = latency;
195  }
196 
197  if (stats.priorityMinLatency[qos].value() > latency
198  || stats.priorityMinLatency[qos].value() == 0) {
199  stats.priorityMinLatency[qos] = latency;
200  }
201  }
202  }
203 
204  DPRINTF(QOS,
205  "QoSMemCtrl::logResponse MASTER %s [id %d] prio %d "
206  "this master q packets %d - new queue size %d\n",
207  masters[m_id], m_id, qos, packetPriorities[m_id][qos],
208  (dir == READ) ? readQueueSizes[qos]: writeQueueSizes[qos]);
209 }
210 
211 uint8_t
213 {
214  if (policy) {
215  return policy->schedule(m_id, data);
216  } else {
217  DPRINTF(QOS,
218  "QoSScheduler::schedule master ID [%d] "
219  "data received [%d], but QoS scheduler not initialized\n",
220  m_id,data);
221  return 0;
222  }
223 }
224 
225 uint8_t
227 {
228  assert(pkt->req);
229 
230  if (policy) {
231  return schedule(pkt->req->masterId(), pkt->getSize());
232  } else {
233  DPRINTF(QOS, "QoSScheduler::schedule Packet received [Qv %d], "
234  "but QoS scheduler not initialized\n",
235  pkt->qosValue());
236  return pkt->qosValue();
237  }
238 }
239 
242 {
243  auto bus_state = getBusState();
244 
245  if (turnPolicy) {
246  DPRINTF(QOS,
247  "QoSMemoryTurnaround::selectBusState running policy %s\n",
248  turnPolicy->name());
249 
250  bus_state = turnPolicy->selectBusState();
251  } else {
252  DPRINTF(QOS,
253  "QoSMemoryTurnaround::selectBusState running "
254  "default bus direction selection policy\n");
255 
256  if ((!getTotalReadQueueSize() && bus_state == MemCtrl::READ) ||
257  (!getTotalWriteQueueSize() && bus_state == MemCtrl::WRITE)) {
258  // READ/WRITE turnaround
259  bus_state = (bus_state == MemCtrl::READ) ? MemCtrl::WRITE :
261 
262  }
263  }
264 
265  return bus_state;
266 }
267 
268 void
270 {
271  if (!hasMaster(m_id)) {
272  masters.emplace(m_id, _system->getMasterName(m_id));
273  packetPriorities[m_id].resize(numPriorities(), 0);
274 
275  DPRINTF(QOS,
276  "QoSMemCtrl::addMaster registering"
277  " Master %s [id %d]\n",
278  masters[m_id], m_id);
279  }
280 }
281 
283  : Stats::Group(&mc),
284  memCtrl(mc),
285 
286  ADD_STAT(avgPriority,
287  "Average QoS priority value for accepted requests"),
288  ADD_STAT(avgPriorityDistance,
289  "Average QoS priority distance between assigned and "
290  "queued values"),
291 
292  ADD_STAT(priorityMinLatency,
293  "per QoS priority minimum request to response latency (s)"),
294  ADD_STAT(priorityMaxLatency,
295  "per QoS priority maximum request to response latency (s)"),
296  ADD_STAT(numReadWriteTurnArounds,
297  "Number of turnarounds from READ to WRITE"),
298  ADD_STAT(numWriteReadTurnArounds,
299  "Number of turnarounds from WRITE to READ"),
300  ADD_STAT(numStayReadState,
301  "Number of times bus staying in READ state"),
302  ADD_STAT(numStayWriteState,
303  "Number of times bus staying in WRITE state")
304 {
305 }
306 
307 void
309 {
311 
312  using namespace Stats;
313 
315  const auto max_masters = system->maxMasters();
316  const auto num_priorities = memCtrl.numPriorities();
317 
318  // Initializes per master statistics
320  .init(max_masters)
321  .flags(nozero | nonan)
322  .precision(2)
323  ;
324 
326  .init(max_masters)
327  .flags(nozero | nonan)
328  ;
329 
331  .init(num_priorities)
332  .precision(12)
333  ;
334 
336  .init(num_priorities)
337  .precision(12)
338  ;
339 
340  for (int i = 0; i < max_masters; i++) {
341  const std::string master = system->getMasterName(i);
342  avgPriority.subname(i, master);
343  avgPriorityDistance.subname(i, master);
344  }
345 
346  for (int j = 0; j < num_priorities; ++j) {
349  }
350 }
351 
352 void
354 {
355  if (busStateNext != busState) {
356  if (busState == READ) {
357  stats.numWriteReadTurnArounds++;
358  } else if (busState == WRITE) {
359  stats.numReadWriteTurnArounds++;
360  }
361  } else {
362  if (busState == READ) {
363  stats.numStayReadState++;
364  } else if (busState == WRITE) {
365  stats.numStayWriteState++;
366  }
367  }
368 }
369 
370 } // namespace QoS
QoS::MemCtrl::MemCtrlStats stats
void logResponse(BusState dir, MasterID m_id, uint8_t qos, Addr addr, uint64_t entries, double delay)
Called upon receiving a response, updates statistics and updates queues status.
Definition: mem_ctrl.cc:143
#define DPRINTF(x,...)
Definition: trace.hh:225
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:376
BusState busStateNext
bus state for next request event triggered
Definition: mem_ctrl.hh:122
Bitfield< 7 > i
std::string getMasterName(MasterID master_id)
Get the name of an object for a given request id.
Definition: system.cc:541
const std::unique_ptr< QueuePolicy > queuePolicy
QoS Queue Policy: selects packet among same-priority queue.
Definition: mem_ctrl.hh:73
uint64_t totalReadQueueSize
Total read request packets queue length in #packets.
Definition: mem_ctrl.hh:110
const FlagsType nonan
Don&#39;t print if this is NAN.
Definition: info.hh:59
Stats::Vector priorityMinLatency
per-priority minimum latency
Definition: mem_ctrl.hh:141
void addMaster(const MasterID m_id)
Initializes dynamically counters and statistics for a given Master.
Definition: mem_ctrl.cc:269
ip6_addr_t addr
Definition: inet.hh:330
void value(VCounter &vec) const
Definition: statistics.hh:1075
Definition: system.hh:72
double s
These variables equal the number of ticks in the unit of time they&#39;re named after in a double...
Definition: core.cc:49
std::vector< Tick > serviceTick
Vector of QoS priorities/last service time.
Definition: mem_ctrl.hh:101
uint8_t numPriorities() const
Gets the total number of priority levels in the QoS memory controller.
Definition: mem_ctrl.hh:348
Derived & flags(Flags _flags)
Set the flags and marks this stat to print at the end of simulation.
Definition: statistics.hh:333
const std::unique_ptr< TurnaroundPolicy > turnPolicy
QoS Bus Turnaround Policy: selects the bus direction (READ/WRITE)
Definition: mem_ctrl.hh:70
Derived & init(size_type size)
Set this vector to have the given size.
Definition: statistics.hh:1149
std::unordered_map< MasterID, const std::string > masters
Hash of master ID - master name.
Definition: mem_ctrl.hh:88
RequestPtr req
A pointer to the original request.
Definition: packet.hh:321
MemCtrl(const QoSMemCtrlParams *)
QoS Memory base class.
Definition: mem_ctrl.cc:44
unsigned getSize() const
Definition: packet.hh:730
QoS Queue Policy.
Definition: q_policy.hh:59
Tick curTick()
The current simulated tick.
Definition: core.hh:44
Stats::VectorStandardDeviation avgPriorityDistance
per-master average QoS distance between assigned and queued values
Definition: mem_ctrl.hh:138
MemCtrlStats(MemCtrl &mc)
Definition: mem_ctrl.cc:282
std::vector< uint64_t > readQueueSizes
Read request packets queue length in #packets, per QoS priority.
Definition: mem_ctrl.hh:104
uint8_t qosValue() const
QoS Value getter Returns 0 if QoS value was never set (constructor default).
Definition: packet.hh:695
BusState
Bus Direction.
Definition: mem_ctrl.hh:63
BusState busState
Bus state used to control the read/write switching and drive the scheduling of the next request...
Definition: mem_ctrl.hh:119
void recordTurnaroundStats()
Record statistics on turnarounds based on busStateNext and busState values.
Definition: mem_ctrl.cc:353
uint64_t getTotalWriteQueueSize() const
Gets the total combined WRITE queues size.
Definition: mem_ctrl.hh:332
uint64_t totalWriteQueueSize
Total write request packets queue length in #packets.
Definition: mem_ctrl.hh:113
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
bool hasMaster(MasterID m_id) const
hasMaster returns true if the selected master(ID) has been registered in the memory controller...
Definition: mem_ctrl.hh:297
uint16_t MasterID
Definition: request.hh:84
Derived & precision(int _precision)
Set the precision and marks this stat to print at the end of simulation.
Definition: statistics.hh:321
const uint8_t _numPriorities
Number of configured QoS priorities.
Definition: mem_ctrl.hh:76
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
void regStats() override
Callback to set stat parameters.
Definition: mem_ctrl.cc:308
MasterID maxMasters()
Get the number of masters registered in the system.
Definition: system.hh:373
Bitfield< 24 > j
Definition: mem_ctrl.cc:42
const MemCtrl & memCtrl
Definition: mem_ctrl.hh:130
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Definition: group.hh:67
const std::unique_ptr< Policy > policy
QoS Policy, assigns QoS priority to the incoming packets.
Definition: mem_ctrl.hh:67
Stats::VectorStandardDeviation avgPriority
per-master average QoS priority
Definition: mem_ctrl.hh:133
System * system() const
read the system pointer Implemented for completeness with the setter
Group()=delete
uint64_t getTotalReadQueueSize() const
Gets the total combined READ queues size.
Definition: mem_ctrl.hh:325
void logRequest(BusState dir, MasterID m_id, uint8_t qos, Addr addr, uint64_t entries)
Called upon receiving a request or updates statistics and updates queues status.
Definition: mem_ctrl.cc:86
System * _system
Pointer to the System object.
std::unordered_map< MasterID, std::unordered_map< uint64_t, std::deque< uint64_t > > > requestTimes
Hash of masters - address of request - queue of times of request.
Definition: mem_ctrl.hh:95
virtual void init()
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: sim_object.cc:73
Stats::Vector priorityMaxLatency
per-priority maximum latency
Definition: mem_ctrl.hh:143
BusState getBusState() const
Gets the current bus state.
Definition: mem_ctrl.hh:278
An abstract memory represents a contiguous block of physical memory, with an associated address range...
void init() override
Initializes this object.
Definition: mem_ctrl.cc:80
virtual ~MemCtrl()
Definition: mem_ctrl.cc:76
virtual void regStats()
Callback to set stat parameters.
Definition: group.cc:64
std::unordered_map< MasterID, std::vector< uint64_t > > packetPriorities
Hash of masters - number of packets queued per priority.
Definition: mem_ctrl.hh:91
VectorStandardDeviation & init(size_type size)
Initialize storage for this distribution.
Definition: statistics.hh:2747
std::vector< uint64_t > writeQueueSizes
Write request packets queue length in #packets, per QoS priority.
Definition: mem_ctrl.hh:107
const FlagsType nozero
Don&#39;t print if this is zero.
Definition: info.hh:57
Bitfield< 0 > p
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:181
std::vector< Info * > stats
Definition: group.hh:212
const char data[]
The QoS::MemCtrl is a base class for Memory objects which support QoS - it provides access to a set o...
Definition: mem_ctrl.hh:59
uint8_t schedule(MasterID m_id, uint64_t data)
Definition: mem_ctrl.cc:212
const std::string to_string(sc_enc enc)
Definition: sc_fxdefs.cc:60
BusState selectNextBusState()
Returns next bus direction (READ or WRITE) based on configured policy.
Definition: mem_ctrl.cc:241

Generated on Fri Jul 3 2020 15:53:03 for gem5 by doxygen 1.8.13