gem5  v20.0.0.3
dramsim2.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 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/dramsim2.hh"
39 
40 #include "DRAMSim2/Callback.h"
41 #include "base/callback.hh"
42 #include "base/trace.hh"
43 #include "debug/DRAMSim2.hh"
44 #include "debug/Drain.hh"
45 #include "sim/system.hh"
46 
48  AbstractMemory(p),
49  port(name() + ".port", *this),
50  wrapper(p->deviceConfigFile, p->systemConfigFile, p->filePath,
51  p->traceFile, p->range.size() / 1024 / 1024, p->enableDebug),
52  retryReq(false), retryResp(false), startTick(0),
53  nbrOutstandingReads(0), nbrOutstandingWrites(0),
54  sendResponseEvent([this]{ sendResponse(); }, name()),
55  tickEvent([this]{ tick(); }, name())
56 {
58  "Instantiated DRAMSim2 with clock %d ns and queue size %d\n",
60 
61  DRAMSim::TransactionCompleteCB* read_cb =
62  new DRAMSim::Callback<DRAMSim2, void, unsigned, uint64_t, uint64_t>(
63  this, &DRAMSim2::readComplete);
64  DRAMSim::TransactionCompleteCB* write_cb =
65  new DRAMSim::Callback<DRAMSim2, void, unsigned, uint64_t, uint64_t>(
67  wrapper.setCallbacks(read_cb, write_cb);
68 
69  // Register a callback to compensate for the destructor not
70  // being called. The callback prints the DRAMSim2 stats.
74 }
75 
76 void
78 {
80 
81  if (!port.isConnected()) {
82  fatal("DRAMSim2 %s is unconnected!\n", name());
83  } else {
85  }
86 
87  if (system()->cacheLineSize() != wrapper.burstSize())
88  fatal("DRAMSim2 burst size %d does not match cache line size %d\n",
90 }
91 
92 void
94 {
95  startTick = curTick();
96 
97  // kick off the clock ticks
99 }
100 
101 void
103 {
104  assert(!retryResp);
105  assert(!responseQueue.empty());
106 
107  DPRINTF(DRAMSim2, "Attempting to send response\n");
108 
109  bool success = port.sendTimingResp(responseQueue.front());
110  if (success) {
111  responseQueue.pop_front();
112 
113  DPRINTF(DRAMSim2, "Have %d read, %d write, %d responses outstanding\n",
115  responseQueue.size());
116 
117  if (!responseQueue.empty() && !sendResponseEvent.scheduled())
119 
120  if (nbrOutstanding() == 0)
121  signalDrainDone();
122  } else {
123  retryResp = true;
124 
125  DPRINTF(DRAMSim2, "Waiting for response retry\n");
126 
127  assert(!sendResponseEvent.scheduled());
128  }
129 }
130 
131 unsigned int
133 {
135 }
136 
137 void
139 {
140  wrapper.tick();
141 
142  // is the connected port waiting for a retry, if so check the
143  // state and send a retry if conditions have changed
144  if (retryReq && nbrOutstanding() < wrapper.queueSize()) {
145  retryReq = false;
146  port.sendRetryReq();
147  }
148 
150 }
151 
152 Tick
154 {
155  access(pkt);
156 
157  // 50 ns is just an arbitrary value at this point
158  return pkt->cacheResponding() ? 0 : 50000;
159 }
160 
161 void
163 {
164  pkt->pushLabel(name());
165 
166  functionalAccess(pkt);
167 
168  // potentially update the packets in our response queue as well
169  for (auto i = responseQueue.begin(); i != responseQueue.end(); ++i)
170  pkt->trySatisfyFunctional(*i);
171 
172  pkt->popLabel();
173 }
174 
175 bool
177 {
178  // if a cache is responding, sink the packet without further action
179  if (pkt->cacheResponding()) {
180  pendingDelete.reset(pkt);
181  return true;
182  }
183 
184  // we should not get a new request after committing to retry the
185  // current one, but unfortunately the CPU violates this rule, so
186  // simply ignore it for now
187  if (retryReq)
188  return false;
189 
190  // if we cannot accept we need to send a retry once progress can
191  // be made
192  bool can_accept = nbrOutstanding() < wrapper.queueSize();
193 
194  // keep track of the transaction
195  if (pkt->isRead()) {
196  if (can_accept) {
197  outstandingReads[pkt->getAddr()].push(pkt);
198 
199  // we count a transaction as outstanding until it has left the
200  // queue in the controller, and the response has been sent
201  // back, note that this will differ for reads and writes
203  }
204  } else if (pkt->isWrite()) {
205  if (can_accept) {
206  outstandingWrites[pkt->getAddr()].push(pkt);
207 
209 
210  // perform the access for writes
211  accessAndRespond(pkt);
212  }
213  } else {
214  // keep it simple and just respond if necessary
215  accessAndRespond(pkt);
216  return true;
217  }
218 
219  if (can_accept) {
220  // we should never have a situation when we think there is space,
221  // and there isn't
222  assert(wrapper.canAccept());
223 
224  DPRINTF(DRAMSim2, "Enqueueing address %lld\n", pkt->getAddr());
225 
226  // @todo what about the granularity here, implicit assumption that
227  // a transaction matches the burst size of the memory (which we
228  // cannot determine without parsing the ini file ourselves)
229  wrapper.enqueue(pkt->isWrite(), pkt->getAddr());
230 
231  return true;
232  } else {
233  retryReq = true;
234  return false;
235  }
236 }
237 
238 void
240 {
241  DPRINTF(DRAMSim2, "Retrying\n");
242 
243  assert(retryResp);
244  retryResp = false;
245  sendResponse();
246 }
247 
248 void
250 {
251  DPRINTF(DRAMSim2, "Access for address %lld\n", pkt->getAddr());
252 
253  bool needsResponse = pkt->needsResponse();
254 
255  // do the actual memory access which also turns the packet into a
256  // response
257  access(pkt);
258 
259  // turn packet around to go back to requester if response expected
260  if (needsResponse) {
261  // access already turned the packet into a response
262  assert(pkt->isResponse());
263  // Here we pay for xbar additional delay and to process the payload
264  // of the packet.
265  Tick time = curTick() + pkt->headerDelay + pkt->payloadDelay;
266  // Reset the timings of the packet
267  pkt->headerDelay = pkt->payloadDelay = 0;
268 
269  DPRINTF(DRAMSim2, "Queuing response for address %lld\n",
270  pkt->getAddr());
271 
272  // queue it to be sent back
273  responseQueue.push_back(pkt);
274 
275  // if we are not already waiting for a retry, or are scheduled
276  // to send a response, schedule an event
279  } else {
280  // queue the packet for deletion
281  pendingDelete.reset(pkt);
282  }
283 }
284 
285 void DRAMSim2::readComplete(unsigned id, uint64_t addr, uint64_t cycle)
286 {
287  assert(cycle == divCeil(curTick() - startTick,
289 
290  DPRINTF(DRAMSim2, "Read to address %lld complete\n", addr);
291 
292  // get the outstanding reads for the address in question
293  auto p = outstandingReads.find(addr);
294  assert(p != outstandingReads.end());
295 
296  // first in first out, which is not necessarily true, but it is
297  // the best we can do at this point
298  PacketPtr pkt = p->second.front();
299  p->second.pop();
300 
301  if (p->second.empty())
302  outstandingReads.erase(p);
303 
304  // no need to check for drain here as the next call will add a
305  // response to the response queue straight away
306  assert(nbrOutstandingReads != 0);
308 
309  // perform the actual memory access
310  accessAndRespond(pkt);
311 }
312 
313 void DRAMSim2::writeComplete(unsigned id, uint64_t addr, uint64_t cycle)
314 {
315  assert(cycle == divCeil(curTick() - startTick,
317 
318  DPRINTF(DRAMSim2, "Write to address %lld complete\n", addr);
319 
320  // get the outstanding reads for the address in question
321  auto p = outstandingWrites.find(addr);
322  assert(p != outstandingWrites.end());
323 
324  // we have already responded, and this is only to keep track of
325  // what is outstanding
326  p->second.pop();
327  if (p->second.empty())
328  outstandingWrites.erase(p);
329 
330  assert(nbrOutstandingWrites != 0);
332 
333  if (nbrOutstanding() == 0)
334  signalDrainDone();
335 }
336 
337 Port &
338 DRAMSim2::getPort(const std::string &if_name, PortID idx)
339 {
340  if (if_name != "port") {
341  return AbstractMemory::getPort(if_name, idx);
342  } else {
343  return port;
344  }
345 }
346 
349 {
350  // check our outstanding reads and writes and if any they need to
351  // drain
353 }
354 
355 DRAMSim2::MemoryPort::MemoryPort(const std::string& _name,
356  DRAMSim2& _memory)
357  : SlavePort(_name, &_memory), memory(_memory)
358 { }
359 
362 {
363  AddrRangeList ranges;
364  ranges.push_back(memory.getAddrRange());
365  return ranges;
366 }
367 
368 Tick
370 {
371  return memory.recvAtomic(pkt);
372 }
373 
374 void
376 {
377  memory.recvFunctional(pkt);
378 }
379 
380 bool
382 {
383  // pass it to the memory controller
384  return memory.recvTimingReq(pkt);
385 }
386 
387 void
389 {
390  memory.recvRespRetry();
391 }
392 
393 DRAMSim2*
394 DRAMSim2Params::create()
395 {
396  return new DRAMSim2(this);
397 }
MemoryPort(const std::string &_name, DRAMSim2 &_memory)
Definition: dramsim2.cc:355
#define DPRINTF(x,...)
Definition: trace.hh:225
void functionalAccess(PacketPtr pkt)
Perform an untimed memory read or write without changing anything but the memory itself.
Tick recvAtomic(PacketPtr pkt)
Definition: dramsim2.cc:153
Ports are used to interface objects to each other.
Definition: port.hh:56
Wrapper class to avoid having DRAMSim2 names like ClockDomain etc clashing with the normal gem5 world...
void printStats()
Print the stats gathered in DRAMsim2.
EventFunctionWrapper sendResponseEvent
Event to schedule sending of responses.
Definition: dramsim2.hh:149
void recvFunctional(PacketPtr pkt)
Definition: dramsim2.cc:162
void sendRangeChange() const
Called by the owner to send a range change.
Definition: port.hh:282
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
Generic callback class.
Definition: callback.hh:39
const std::string & name()
Definition: trace.cc:50
Bitfield< 7 > i
DrainState drain() override
Notify an object that it needs to drain its state.
Definition: dramsim2.cc:348
void recvRespRetry()
Definition: dramsim2.cc:239
unsigned int nbrOutstanding() const
Definition: dramsim2.cc:132
void enqueue(bool is_write, uint64_t addr)
Enqueue a packet.
void recvFunctional(PacketPtr pkt)
Receive a functional request packet from the peer.
Definition: dramsim2.cc:375
Tick startTick
Definition: stat_control.cc:69
bool retryResp
Are we waiting for a retry for sending a response.
Definition: dramsim2.hh:102
ip6_addr_t addr
Definition: inet.hh:330
bool cacheResponding() const
Definition: packet.hh:585
std::unique_ptr< Packet > pendingDelete
Upstream caches need this packet until true is returned, so hold it for deletion until a subsequent c...
Definition: dramsim2.hh:165
DRAMSim2.
bool isConnected() const
Is this port currently connected to a peer?
Definition: port.hh:124
A SlavePort is a specialisation of a port.
Definition: port.hh:254
void readComplete(unsigned id, uint64_t addr, uint64_t cycle)
Read completion callback.
Definition: dramsim2.cc:285
void pushLabel(const std::string &lbl)
Push label for PrintReq (safe to call unconditionally).
Definition: packet.hh:1320
bool isWrite() const
Definition: packet.hh:523
std::unordered_map< Addr, std::queue< PacketPtr > > outstandingWrites
Definition: dramsim2.hh:116
bool isRead() const
Definition: packet.hh:522
DrainState
Object drain/handover states.
Definition: drain.hh:71
unsigned int nbrOutstandingWrites
Definition: dramsim2.hh:124
void recvRespRetry()
Called by the peer if sendTimingResp was called on this protocol (causing recvTimingResp to be called...
Definition: dramsim2.cc:388
void tick()
Progress the controller one clock cycle.
Definition: dramsim2.cc:138
void startup() override
startup() is the final initialization call before simulation.
Definition: dramsim2.cc:93
bool sendTimingResp(PacketPtr pkt)
Attempt to send a timing response to the master port by calling its corresponding receive function...
Definition: port.hh:353
Draining buffers pending serialization/handover.
Tick curTick()
The current simulated tick.
Definition: core.hh:44
bool recvTimingReq(PacketPtr pkt)
Definition: dramsim2.cc:176
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
Definition: sim_object.cc:123
bool needsResponse() const
Definition: packet.hh:536
std::deque< PacketPtr > responseQueue
Queue to hold response packets until we can send them back.
Definition: dramsim2.hh:131
AbstractMemoryParams Params
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
Definition: packet.hh:360
unsigned int burstSize() const
Get the burst size in bytes used by DRAMSim2.
uint64_t Tick
Tick count type.
Definition: types.hh:61
bool isResponse() const
Definition: packet.hh:526
bool trySatisfyFunctional(PacketPtr other)
Check a functional request against a memory value stored in another packet (i.e.
Definition: packet.hh:1258
void popLabel()
Pop label for PrintReq (safe to call unconditionally).
Definition: packet.hh:1330
void writeComplete(unsigned id, uint64_t addr, uint64_t cycle)
Write completion callback.
Definition: dramsim2.cc:313
void access(PacketPtr pkt)
Perform an untimed memory access and update all the state (e.g.
Addr getAddr() const
Definition: packet.hh:720
void registerExitCallback(Callback *callback)
Register an exit callback.
Definition: core.cc:140
void accessAndRespond(PacketPtr pkt)
When a packet is ready, use the "access()" method in AbstractMemory to actually create the response p...
Definition: dramsim2.cc:249
void schedule(Event &event, Tick when)
Definition: eventq.hh:934
DRAMSim2Wrapper wrapper
The actual DRAMSim2 wrapper.
Definition: dramsim2.hh:92
unsigned int queueSize() const
Get the transaction queue size used by DRAMSim2.
Tick startTick
Keep track of when the wrapper is started.
Definition: dramsim2.hh:107
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
Definition: packet.hh:378
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
Tick clockEdge(Cycles cycles=Cycles(0)) const
Determine the tick when a cycle begins, by default the current one, but the argument also enables the...
bool canAccept() const
Determine if the controller can accept a new packet or not.
void setCallbacks(DRAMSim::TransactionCompleteCB *read_callback, DRAMSim::TransactionCompleteCB *write_callback)
Set the callbacks to use for read and write completion.
AddrRangeList getAddrRanges() const
Get a list of the non-overlapping address ranges the owner is responsible for.
Definition: dramsim2.cc:361
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:459
virtual const std::string name() const
Definition: sim_object.hh:129
std::unordered_map< Addr, std::queue< PacketPtr > > outstandingReads
Keep track of what packets are outstanding per address, and do so separately for reads and writes...
Definition: dramsim2.hh:115
Tick recvAtomic(PacketPtr pkt)
Receive an atomic request packet from the peer.
Definition: dramsim2.cc:369
bool retryReq
Is the connected port waiting for a retry from us.
Definition: dramsim2.hh:97
System * system() const
read the system pointer Implemented for completeness with the setter
DRAMSim2(const Params *p)
Definition: dramsim2.cc:47
Cycles cycle
void sendRetryReq()
Send a retry to the master port that previously attempted a sendTimingReq to this slave port and fail...
Definition: port.hh:376
void sendResponse()
Definition: dramsim2.cc:102
EventFunctionWrapper tickEvent
Event to schedule clock ticks.
Definition: dramsim2.hh:159
MemoryPort port
Definition: dramsim2.hh:87
void signalDrainDone() const
Signal that an object is drained.
Definition: drain.hh:289
T divCeil(const T &a, const U &b)
Definition: intmath.hh:99
virtual void init()
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: sim_object.cc:73
Definition: mem.h:38
An abstract memory represents a contiguous block of physical memory, with an associated address range...
Tick ns
nanosecond
Definition: core.cc:65
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:235
Helper template class to turn a simple class member function into a callback.
Definition: callback.hh:62
void tick()
Progress the memory controller one cycle.
double clockPeriod() const
Get the internal clock period used by DRAMSim2, specified in ns.
unsigned int nbrOutstandingReads
Count the number of outstanding transactions so that we can block any further requests until there is...
Definition: dramsim2.hh:123
Bitfield< 0 > p
Running normally.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: dramsim2.cc:338
bool recvTimingReq(PacketPtr pkt)
Receive a timing request from the peer.
Definition: dramsim2.cc:381
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: dramsim2.cc:77
unsigned int cacheLineSize() const
Get the cache line size of the system.
Definition: system.hh:180

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