gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
dma_device.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2012, 2015, 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  * Copyright (c) 2006 The Regents of The University of Michigan
15  * All rights reserved.
16  *
17  * Redistribution and use in source and binary forms, with or without
18  * modification, are permitted provided that the following conditions are
19  * met: redistributions of source code must retain the above copyright
20  * notice, this list of conditions and the following disclaimer;
21  * redistributions in binary form must reproduce the above copyright
22  * notice, this list of conditions and the following disclaimer in the
23  * documentation and/or other materials provided with the distribution;
24  * neither the name of the copyright holders nor the names of its
25  * contributors may be used to endorse or promote products derived from
26  * this software without specific prior written permission.
27  *
28  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
31  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
32  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
33  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
34  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
35  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
36  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
37  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
38  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
39  */
40 
41 #include "dev/dma_device.hh"
42 
43 #include <algorithm>
44 #include <cassert>
45 #include <cstring>
46 #include <utility>
47 
48 #include "base/logging.hh"
49 #include "base/trace.hh"
50 #include "debug/DMA.hh"
51 #include "debug/Drain.hh"
52 #include "sim/clocked_object.hh"
53 #include "sim/system.hh"
54 
56  uint32_t sid, uint32_t ssid)
57  : RequestPort(dev->name() + ".dma", dev),
58  device(dev), sys(s), requestorId(s->getRequestorId(dev)),
59  sendEvent([this]{ sendDma(); }, dev->name()),
60  defaultSid(sid), defaultSSid(ssid), cacheLineSize(s->cacheLineSize())
61 { }
62 
63 void
65 {
66  // Should always see a response with a sender state.
67  assert(pkt->isResponse());
68 
69  // Get the DMA sender state.
70  auto *state = dynamic_cast<DmaReqState*>(pkt->senderState);
71  assert(state);
72 
73  handleResp(state, pkt->getAddr(), pkt->req->getSize(), delay);
74 
75  delete pkt;
76 }
77 
78 void
80 {
81  DPRINTF(DMA, "Received response %s for addr: %#x size: %d nb: %d," \
82  " tot: %d sched %d\n",
83  MemCmd(state->cmd).toString(), addr, size,
84  state->numBytes, state->totBytes,
85  state->completionEvent ?
86  state->completionEvent->scheduled() : 0);
87 
88  // Update the number of bytes received based on the request rather
89  // than the packet as the latter could be rounded up to line sizes.
90  state->numBytes += size;
91  assert(state->totBytes >= state->numBytes);
92 
93  // If we have reached the total number of bytes for this DMA request,
94  // then signal the completion and delete the sate.
95  if (state->totBytes == state->numBytes) {
96  assert(pendingCount != 0);
97  pendingCount--;
98  if (state->completionEvent) {
99  delay += state->delay;
100  device->schedule(state->completionEvent, curTick() + delay);
101  }
102  delete state;
103  }
104 
105  // We might be drained at this point, if so signal the drain event.
106  if (pendingCount == 0)
107  signalDrainDone();
108 }
109 
110 PacketPtr
112 {
113  RequestPtr req = std::make_shared<Request>(
114  gen.addr(), gen.size(), flags, id);
115  req->setStreamId(sid);
116  req->setSubstreamId(ssid);
117  req->taskId(ContextSwitchTaskId::DMA);
118 
119  PacketPtr pkt = new Packet(req, cmd);
120 
121  if (data)
122  pkt->dataStatic(data + gen.complete());
123 
124  pkt->senderState = this;
125  return pkt;
126 }
127 
128 bool
130 {
131  // We shouldn't ever get a cacheable block in Modified state.
132  assert(pkt->req->isUncacheable() ||
133  !(pkt->cacheResponding() && !pkt->hasSharers()));
134 
135  handleRespPacket(pkt);
136 
137  return true;
138 }
139 
141  : PioDevice(p), dmaPort(this, sys, p.sid, p.ssid)
142 { }
143 
144 void
146 {
148  "DMA port of %s not connected to anything!", name());
149  PioDevice::init();
150 }
151 
154 {
155  if (pendingCount == 0) {
156  return DrainState::Drained;
157  } else {
158  DPRINTF(Drain, "DmaPort not drained\n");
159  return DrainState::Draining;
160  }
161 }
162 
163 void
165 {
166  assert(transmitList.size());
168 }
169 
170 void
172  uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay,
173  Request::Flags flag)
174 {
175  DPRINTF(DMA, "Starting DMA for addr: %#x size: %d sched: %d\n", addr, size,
176  event ? event->scheduled() : -1);
177 
178  // One DMA request sender state for every action, that is then
179  // split into many requests and packets based on the block size,
180  // i.e. cache line size.
181  transmitList.push_back(
182  new DmaReqState(cmd, addr, cacheLineSize, size,
183  data, flag, requestorId, sid, ssid, event, delay));
184  pendingCount++;
185 
186  // In zero time, also initiate the sending of the packets for the request
187  // we have just created. For atomic this involves actually completing all
188  // the requests.
189  sendDma();
190 }
191 
192 void
194  uint8_t *data, Tick delay, Request::Flags flag)
195 {
196  dmaAction(cmd, addr, size, event, data,
197  defaultSid, defaultSSid, delay, flag);
198 }
199 
200 void
202 {
203  // Send the next packet for the first DMA request on the transmit list,
204  // and schedule the following send if it is successful
205  DmaReqState *state = transmitList.front();
206 
207  PacketPtr pkt = inRetry ? inRetry : state->createPacket();
208  inRetry = nullptr;
209 
210  DPRINTF(DMA, "Trying to send %s addr %#x\n", pkt->cmdString(),
211  pkt->getAddr());
212 
213  // Check if this was the last packet now, since hypothetically the packet
214  // response may come immediately, and state may be deleted.
215  bool last = state->gen.last();
216  if (!sendTimingReq(pkt))
217  inRetry = pkt;
218  if (!inRetry) {
219  // If that was the last packet from this request, pop it from the list.
220  if (last)
221  transmitList.pop_front();
222  else
223  state->gen.next();
224  DPRINTF(DMA, "-- Done\n");
225  // If there is more to do, then do so.
226  if (!transmitList.empty()) {
227  // This should ultimately wait for as many cycles as the device
228  // needs to send the packet, but currently the port does not have
229  // any known width so simply wait a single cycle.
231  }
232  } else {
233  DPRINTF(DMA, "-- Failed, waiting for retry\n");
234  }
235 
236  DPRINTF(DMA, "TransmitList: %d, inRetry: %d\n",
237  transmitList.size(), inRetry ? 1 : 0);
238 }
239 
240 bool
242 {
243  PacketPtr pkt = state->createPacket();
244  DPRINTF(DMA, "Sending DMA for addr: %#x size: %d\n",
245  state->gen.addr(), state->gen.size());
246  Tick lat = sendAtomic(pkt);
247 
248  // Check if we're done, since handleResp may delete state.
249  bool done = !state->gen.next();
250  handleRespPacket(pkt, lat);
251  return done;
252 }
253 
254 bool
256 {
257  bool done = false;
258 
259  auto bd_it = memBackdoors.contains(state->gen.addr());
260  if (bd_it == memBackdoors.end()) {
261  // We don't have a backdoor for this address, so use a packet.
262 
263  PacketPtr pkt = state->createPacket();
264  DPRINTF(DMA, "Sending DMA for addr: %#x size: %d\n",
265  state->gen.addr(), state->gen.size());
266 
267  MemBackdoorPtr bd = nullptr;
268  Tick lat = sendAtomicBackdoor(pkt, bd);
269 
270  // If we got a backdoor, record it.
271  if (bd && memBackdoors.insert(bd->range(), bd) != memBackdoors.end()) {
272  // Invalidation callback which finds this backdoor and removes it.
273  auto callback = [this](const MemBackdoor &backdoor) {
274  for (auto it = memBackdoors.begin();
275  it != memBackdoors.end(); it++) {
276  if (it->second == &backdoor) {
277  memBackdoors.erase(it);
278  return;
279  }
280  }
281  panic("Got invalidation for unknown memory backdoor.");
282  };
283  bd->addInvalidationCallback(callback);
284  }
285 
286  // Check if we're done now, since handleResp may delete state.
287  done = !state->gen.next();
288  handleRespPacket(pkt, lat);
289  } else {
290  // We have a backdoor that can at least partially satisfy this request.
291  DPRINTF(DMA, "Handling DMA for addr: %#x size %d through backdoor\n",
292  state->gen.addr(), state->gen.size());
293 
294  const auto *bd = bd_it->second;
295  // Offset of this access into the backdoor.
296  const Addr offset = state->gen.addr() - bd->range().start();
297  // How many bytes we still need.
298  const Addr remaining = state->totBytes - state->gen.complete();
299  // How many bytes this backdoor can provide, starting from offset.
300  const Addr available = bd->range().size() - offset;
301 
302  // How many bytes we're going to handle through this backdoor.
303  const Addr handled = std::min(remaining, available);
304 
305  // If there's a buffer for data, read/write it.
306  if (state->data) {
307  uint8_t *bd_data = bd->ptr() + offset;
308  uint8_t *state_data = state->data + state->gen.complete();
309  if (MemCmd(state->cmd).isRead())
310  memcpy(state_data, bd_data, handled);
311  else
312  memcpy(bd_data, state_data, handled);
313  }
314 
315  // Advance the chunk generator past this region of memory.
316  state->gen.setNext(state->gen.addr() + handled);
317 
318  // Check if we're done now, since handleResp may delete state.
319  done = !state->gen.next();
320  handleResp(state, state->gen.addr(), handled);
321  }
322 
323  return done;
324 }
325 
326 void
328 {
329  // Some kind of selection between access methods. More work is going to
330  // have to be done to make switching actually work.
331  assert(transmitList.size());
332 
333  if (sys->isTimingMode()) {
334  // If we are either waiting for a retry or are still waiting after
335  // sending the last packet, then do not proceed.
336  if (inRetry || sendEvent.scheduled()) {
337  DPRINTF(DMA, "Can't send immediately, waiting to send\n");
338  return;
339  }
340 
342  } else if (sys->isAtomicMode()) {
343  const bool bypass = sys->bypassCaches();
344 
345  // Send everything there is to send in zero time.
346  while (!transmitList.empty()) {
347  DmaReqState *state = transmitList.front();
348  transmitList.pop_front();
349 
350  bool done = state->gen.done();
351  while (!done)
352  done = bypass ? sendAtomicBdReq(state) : sendAtomicReq(state);
353  }
354  } else {
355  panic("Unknown memory mode.");
356  }
357 }
358 
359 Port &
360 DmaDevice::getPort(const std::string &if_name, PortID idx)
361 {
362  if (if_name == "dma") {
363  return dmaPort;
364  }
365  return PioDevice::getPort(if_name, idx);
366 }
367 
368 DmaReadFifo::DmaReadFifo(DmaPort &_port, size_t size,
369  unsigned max_req_size,
370  unsigned max_pending,
371  Request::Flags flags)
372  : maxReqSize(max_req_size), fifoSize(size),
373  reqFlags(flags), port(_port), cacheLineSize(port.sys->cacheLineSize()),
374  buffer(size)
375 {
376  freeRequests.resize(max_pending);
377  for (auto &e : freeRequests)
378  e.reset(new DmaDoneEvent(this, max_req_size));
379 
380 }
381 
383 {
384  for (auto &p : pendingRequests) {
385  DmaDoneEvent *e(p.release());
386 
387  if (e->done()) {
388  delete e;
389  } else {
390  // We can't kill in-flight DMAs, so we'll just transfer
391  // ownership to the event queue so that they get freed
392  // when they are done.
393  e->kill();
394  }
395  }
396 }
397 
398 void
400 {
401  assert(pendingRequests.empty());
402 
406 }
407 
408 void
410 {
414 }
415 
416 bool
417 DmaReadFifo::tryGet(uint8_t *dst, size_t len)
418 {
419  if (buffer.size() >= len) {
420  buffer.read(dst, len);
421  resumeFill();
422  return true;
423  } else {
424  return false;
425  }
426 }
427 
428 void
429 DmaReadFifo::get(uint8_t *dst, size_t len)
430 {
431  panic_if(!tryGet(dst, len), "Buffer underrun in DmaReadFifo::get()");
432 }
433 
434 void
435 DmaReadFifo::startFill(Addr start, size_t size)
436 {
437  assert(atEndOfBlock());
438 
439  nextAddr = start;
440  endAddr = start + size;
441  resumeFill();
442 }
443 
444 void
446 {
447  // Prevent new DMA requests by setting the next address to the end
448  // address. Pending requests will still complete.
449  nextAddr = endAddr;
450 
451  // Flag in-flight accesses as canceled. This prevents their data
452  // from being written to the FIFO.
453  for (auto &p : pendingRequests)
454  p->cancel();
455 }
456 
457 void
459 {
460  // Don't try to fetch more data if we are draining. This ensures
461  // that the DMA engine settles down before we checkpoint it.
463  return;
464 
465  const bool old_eob(atEndOfBlock());
466 
467  if (port.sys->bypassCaches())
469  else
471 
472  if (!old_eob && atEndOfBlock())
473  onEndOfBlock();
474 }
475 
476 void
478 {
479  const size_t fifo_space = buffer.capacity() - buffer.size();
480  if (fifo_space >= cacheLineSize || buffer.capacity() < cacheLineSize) {
481  const size_t block_remaining = endAddr - nextAddr;
482  const size_t xfer_size = std::min(fifo_space, block_remaining);
483  std::vector<uint8_t> tmp_buffer(xfer_size);
484 
485  assert(pendingRequests.empty());
486  DPRINTF(DMA, "Direct bypass startAddr=%#x xfer_size=%#x " \
487  "fifo_space=%#x block_remaining=%#x\n",
488  nextAddr, xfer_size, fifo_space, block_remaining);
489 
490  port.dmaAction(MemCmd::ReadReq, nextAddr, xfer_size, nullptr,
491  tmp_buffer.data(), 0, reqFlags);
492 
493  buffer.write(tmp_buffer.begin(), xfer_size);
494  nextAddr += xfer_size;
495  }
496 }
497 
498 void
500 {
501  size_t size_pending(0);
502  for (auto &e : pendingRequests)
503  size_pending += e->requestSize();
504 
505  while (!freeRequests.empty() && !atEndOfBlock()) {
506  const size_t req_size(std::min(maxReqSize, endAddr - nextAddr));
507  if (buffer.size() + size_pending + req_size > fifoSize)
508  break;
509 
510  DmaDoneEventUPtr event(std::move(freeRequests.front()));
511  freeRequests.pop_front();
512  assert(event);
513 
514  event->reset(req_size);
515  port.dmaAction(MemCmd::ReadReq, nextAddr, req_size, event.get(),
516  event->data(), 0, reqFlags);
517  nextAddr += req_size;
518  size_pending += req_size;
519 
520  pendingRequests.emplace_back(std::move(event));
521  }
522 }
523 
524 void
526 {
527  const bool old_active(isActive());
528 
529  handlePending();
530  resumeFill();
531 
532  if (old_active && !isActive())
533  onIdle();
534 }
535 
536 void
538 {
539  while (!pendingRequests.empty() && pendingRequests.front()->done()) {
540  // Get the first finished pending request
541  DmaDoneEventUPtr event(std::move(pendingRequests.front()));
542  pendingRequests.pop_front();
543 
544  if (!event->canceled())
545  buffer.write(event->data(), event->requestSize());
546 
547  // Move the event to the list of free requests
548  freeRequests.emplace_back(std::move(event));
549  }
550 
551  if (pendingRequests.empty())
552  signalDrainDone();
553 }
554 
557 {
558  return pendingRequests.empty() ?
560 }
561 
562 
564  : parent(_parent), _data(max_size, 0)
565 {
566 }
567 
568 void
570 {
571  parent = nullptr;
572  setFlags(AutoDelete);
573 }
574 
575 void
577 {
578  _canceled = true;
579 }
580 
581 void
583 {
584  assert(size <= _data.size());
585  _done = false;
586  _canceled = false;
587  _requestSize = size;
588 }
589 
590 void
592 {
593  if (!parent)
594  return;
595 
596  assert(!_done);
597  _done = true;
598  parent->dmaDone();
599 }
ArmISA::sendEvent
void sendEvent(ThreadContext *tc)
Send an event (SEV) to a specific PE if there isn't already a pending event.
Definition: utility.cc:103
DmaReadFifo::tryGet
bool tryGet(uint8_t *dst, size_t len)
Try to read data from the FIFO.
Definition: dma_device.cc:417
DmaPort::DmaReqState::cmd
const Packet::Command cmd
Command for the request.
Definition: dma_device.hh:111
RequestPort::sendAtomicBackdoor
Tick sendAtomicBackdoor(PacketPtr pkt, MemBackdoorPtr &backdoor)
Send an atomic request packet like above, but also request a backdoor to the data being accessed.
Definition: port.hh:471
DmaReadFifo::nextAddr
Addr nextAddr
Definition: dma_device.hh:561
Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:462
DmaReadFifo::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: dma_device.cc:399
ChunkGenerator::complete
Addr complete() const
Number of bytes we have already chunked up.
Definition: chunk_generator.hh:129
DmaReadFifo::startFill
void startFill(Addr start, size_t size)
Start filling the FIFO.
Definition: dma_device.cc:435
Packet::isResponse
bool isResponse() const
Definition: packet.hh:561
MemBackdoor
Definition: backdoor.hh:38
system.hh
Packet::cacheResponding
bool cacheResponding() const
Definition: packet.hh:620
DmaReadFifo::DmaDoneEventUPtr
std::unique_ptr< DmaDoneEvent > DmaDoneEventUPtr
Definition: dma_device.hh:538
data
const char data[]
Definition: circlebuf.test.cc:47
Packet::hasSharers
bool hasSharers() const
Definition: packet.hh:647
System::isTimingMode
bool isTimingMode() const
Is the system in timing mode?
Definition: system.hh:264
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
Packet::getAddr
Addr getAddr() const
Definition: packet.hh:755
PioDevice::init
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: io_device.cc:56
UNSERIALIZE_CONTAINER
#define UNSERIALIZE_CONTAINER(member)
Definition: serialize.hh:650
Fifo::capacity
size_t capacity() const
Definition: circlebuf.hh:220
DmaReadFifo::freeRequests
std::deque< DmaDoneEventUPtr > freeRequests
Definition: dma_device.hh:565
Flags< FlagsType >
DmaPort::DmaReqState::data
uint8_t *const data
Pointer to a buffer for the data.
Definition: dma_device.hh:98
DmaPort::handleResp
void handleResp(DmaReqState *state, Addr addr, Addr size, Tick delay=0)
Definition: dma_device.cc:79
DmaPort::drain
DrainState drain() override
Draining is the process of clearing out the states of SimObjects.These are the SimObjects that are pa...
Definition: dma_device.cc:153
DmaPort::trySendTimingReq
void trySendTimingReq()
Take the first request on the transmit list and attempt to send a timing packet from it.
Definition: dma_device.cc:201
MemCmd::ReadReq
@ ReadReq
Definition: packet.hh:83
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
DmaReadFifo::port
DmaPort & port
Definition: dma_device.hh:509
DmaPort::DmaReqState::createPacket
PacketPtr createPacket()
Definition: dma_device.cc:111
PortID
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:243
DmaPort::DmaReqState::numBytes
Addr numBytes
Number of bytes that have been acked for this transaction.
Definition: dma_device.hh:89
DmaPort::DmaReqState::delay
const Tick delay
Amount to delay completion of dma by.
Definition: dma_device.hh:92
DmaPort::transmitList
std::deque< DmaReqState * > transmitList
Use a deque as we never do any insertion or removal in the middle.
Definition: dma_device.hh:158
RequestPtr
std::shared_ptr< Request > RequestPtr
Definition: request.hh:86
Packet::req
RequestPtr req
A pointer to the original request.
Definition: packet.hh:341
std::vector< uint8_t >
DmaPort::requestorId
const RequestorID requestorId
Id for all requests.
Definition: dma_device.hh:154
DmaPort::defaultSid
const uint32_t defaultSid
Default streamId.
Definition: dma_device.hh:170
DmaReadFifo::stopFill
void stopFill()
Stop the DMA engine.
Definition: dma_device.cc:445
ChunkGenerator::setNext
void setNext(Addr next)
Grow this chunk to cover additional bytes which are already handled.
Definition: chunk_generator.hh:155
DmaReadFifo::DmaDoneEvent
Definition: dma_device.hh:514
DmaPort::DmaReqState
Definition: dma_device.hh:79
ContextSwitchTaskId::DMA
@ DMA
Definition: request.hh:78
DmaReadFifo::get
T get()
Definition: dma_device.hh:416
DmaPort::sendAtomicBdReq
bool sendAtomicBdReq(DmaReqState *state)
Send the next packet from a DMA request in atomic mode, and request and/or use memory backdoors if po...
Definition: dma_device.cc:255
DmaReadFifo::handlePending
void handlePending()
Handle pending requests that have been flagged as done.
Definition: dma_device.cc:537
DmaPort::sendEvent
EventFunctionWrapper sendEvent
Event used to schedule a future sending from the transmit list.
Definition: dma_device.hh:161
DmaPort::device
ClockedObject *const device
The device that owns this port.
Definition: dma_device.hh:147
DmaReadFifo::endAddr
Addr endAddr
Definition: dma_device.hh:562
ClockedObject
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
Definition: clocked_object.hh:231
DmaPort::DmaReqState::flags
const Request::Flags flags
The flags to use for requests.
Definition: dma_device.hh:101
DmaReadFifo::reqFlags
const Request::Flags reqFlags
Request flags.
Definition: dma_device.hh:507
DmaReadFifo::buffer
Fifo< uint8_t > buffer
Definition: dma_device.hh:559
DmaReadFifo::~DmaReadFifo
~DmaReadFifo()
Definition: dma_device.cc:382
DrainState::Drained
@ Drained
Buffers drained, ready for serialization/handover.
DmaReadFifo::maxReqSize
const Addr maxReqSize
Maximum request size in bytes.
Definition: dma_device.hh:498
MemCmd::Command
Command
List of all commands associated with a packet.
Definition: packet.hh:80
DmaDevice::dmaPort
DmaPort dmaPort
Definition: dma_device.hh:203
DrainState
DrainState
Object drain/handover states.
Definition: drain.hh:71
Fifo::write
void write(InputIterator in, size_t len)
Definition: circlebuf.hh:231
DmaReadFifo::cacheLineSize
const int cacheLineSize
Definition: dma_device.hh:511
DmaReadFifo::onIdle
virtual void onIdle()
Last response received callback.
Definition: dma_device.hh:498
ChunkGenerator::addr
Addr addr() const
Return starting address of current chunk.
Definition: chunk_generator.hh:116
Fifo::read
void read(OutputIterator out, size_t len)
Definition: circlebuf.hh:227
cp
Definition: cprintf.cc:37
PioDevice
This device is the base class which all devices senstive to an address range inherit from.
Definition: io_device.hh:99
dma_device.hh
EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1016
Fifo::size
size_t size() const
Definition: circlebuf.hh:219
MemCmd::isRead
bool isRead() const
Definition: packet.hh:200
DmaReadFifo::resumeFillBypass
void resumeFillBypass()
Try to bypass DMA requests in non-caching mode.
Definition: dma_device.cc:477
RequestPort::sendTimingReq
bool sendTimingReq(PacketPtr pkt)
Attempt to send a timing request to the responder port by calling its corresponding receive function.
Definition: port.hh:492
Event
Definition: eventq.hh:248
DmaReadFifo::drain
DrainState drain() override
Draining is the process of clearing out the states of SimObjects.These are the SimObjects that are pa...
Definition: dma_device.cc:556
MemCmd::toString
const std::string & toString() const
Return the string to a cmd given by idx.
Definition: packet.hh:240
DmaPort::DmaReqState::ssid
const uint32_t ssid
Definition: dma_device.hh:108
System
Definition: system.hh:73
PioDevice::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: io_device.cc:64
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
MemCmd
Definition: packet.hh:72
Packet::cmdString
const std::string & cmdString() const
Return the string name of the cmd field (for debugging and tracing).
Definition: packet.hh:552
MipsISA::event
Bitfield< 10, 5 > event
Definition: pra_constants.hh:297
Port
Ports are used to interface objects to each other.
Definition: port.hh:56
Clocked::clockEdge
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...
Definition: clocked_object.hh:174
DmaDevice::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: dma_device.cc:360
DmaReadFifo::DmaDoneEvent::reset
void reset(size_t size)
Definition: dma_device.cc:582
System::bypassCaches
bool bypassCaches() const
Should caches be bypassed?
Definition: system.hh:274
Drainable::signalDrainDone
void signalDrainDone() const
Signal that an object is drained.
Definition: drain.hh:301
ChunkGenerator::next
bool next()
Advance generator to next chunk.
Definition: chunk_generator.hh:182
DmaReadFifo::DmaDoneEvent::kill
void kill()
Definition: dma_device.cc:569
DmaReadFifo
Buffered DMA engine helper class.
Definition: dma_device.hh:361
RequestPort
A RequestPort is a specialisation of a Port, which implements the default protocol for the three diff...
Definition: port.hh:74
DmaPort::handleRespPacket
void handleRespPacket(PacketPtr pkt, Tick delay=0)
Handle a response packet by updating the corresponding DMA request state to reflect the bytes receive...
Definition: dma_device.cc:64
DmaReadFifo::fifoSize
const size_t fifoSize
Maximum FIFO size in bytes.
Definition: dma_device.hh:505
DmaReadFifo::dmaDone
void dmaDone()
DMA request done, handle incoming data and issue new request.
Definition: dma_device.cc:525
DmaPort::cacheLineSize
const int cacheLineSize
Definition: dma_device.hh:175
ProbePoints::Packet
ProbePointArg< PacketInfo > Packet
Packet probe point.
Definition: mem.hh:103
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
DmaPort::sys
System *const sys
The system that device/port are in.
Definition: dma_device.hh:151
ChunkGenerator::done
bool done() const
Are we done? That is, did the last call to next() advance past the end of the region?
Definition: chunk_generator.hh:138
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
DmaReadFifo::DmaDoneEvent::DmaDoneEvent
DmaDoneEvent(DmaReadFifo *_parent, size_t max_size)
Definition: dma_device.cc:563
DmaReadFifo::resumeFill
void resumeFill()
Try to issue new DMA requests or bypass DMA requests.
Definition: dma_device.cc:458
DmaDevice::init
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: dma_device.cc:145
ArmISA::e
Bitfield< 9 > e
Definition: miscregs_types.hh:61
Drainable::drainState
DrainState drainState() const
Return the current drain state of an object.
Definition: drain.hh:320
DmaReadFifo::resumeFillTiming
void resumeFillTiming()
Try to issue new DMA requests during normal execution.
Definition: dma_device.cc:499
DmaPort::dmaAction
void dmaAction(Packet::Command cmd, Addr addr, int size, Event *event, uint8_t *data, Tick delay, Request::Flags flag=0)
Definition: dma_device.cc:193
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
SimObject::name
virtual const std::string name() const
Definition: sim_object.hh:182
DmaPort::DmaReqState::completionEvent
Event * completionEvent
Event to call on the device when this transaction (all packets) complete.
Definition: dma_device.hh:83
DmaPort::recvReqRetry
void recvReqRetry() override
Called by the peer if sendTimingReq was called on this peer (causing recvTimingReq to be called on th...
Definition: dma_device.cc:164
DmaPort::inRetry
PacketPtr inRetry
The packet (if any) waiting for a retry to send.
Definition: dma_device.hh:167
DmaReadFifo::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: dma_device.cc:409
DmaPort::recvTimingResp
bool recvTimingResp(PacketPtr pkt) override
Receive a timing response from the peer.
Definition: dma_device.cc:129
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
DmaPort::DmaReqState::sid
const uint32_t sid
Stream IDs.
Definition: dma_device.hh:107
DmaReadFifo::pendingRequests
std::deque< DmaDoneEventUPtr > pendingRequests
Definition: dma_device.hh:564
DmaReadFifo::onEndOfBlock
virtual void onEndOfBlock()
End of block callback.
Definition: dma_device.hh:485
DmaReadFifo::DmaDoneEvent::cancel
void cancel()
Definition: dma_device.cc:576
DmaDevice::DmaDevice
DmaDevice(const Params &p)
Definition: dma_device.cc:140
Packet::dataStatic
void dataStatic(T *p)
Set the data pointer to the following value that should not be freed.
Definition: packet.hh:1108
clocked_object.hh
SERIALIZE_CONTAINER
#define SERIALIZE_CONTAINER(member)
Definition: serialize.hh:642
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
DmaPort::pendingCount
uint32_t pendingCount
Number of outstanding packets the dma port has.
Definition: dma_device.hh:164
ArmISA::len
Bitfield< 18, 16 > len
Definition: miscregs_types.hh:439
DmaPort::memBackdoors
AddrRangeMap< MemBackdoorPtr, 1 > memBackdoors
Definition: dma_device.hh:61
Port::isConnected
bool isConnected() const
Is this port currently connected to a peer?
Definition: port.hh:128
DmaPort::sendDma
void sendDma()
For timing, attempt to send the first item on the transmit list, and if it is successful and there ar...
Definition: dma_device.cc:327
DmaReadFifo::atEndOfBlock
bool atEndOfBlock() const
Has the DMA engine sent out the last request for the active block?
Definition: dma_device.hh:460
ChunkGenerator::size
Addr size() const
Return size in bytes of current chunk.
Definition: chunk_generator.hh:122
logging.hh
Cycles
Cycles is a wrapper class for representing cycle counts, i.e.
Definition: types.hh:79
DmaReadFifo::size
size_t size() const
Get the amount of data stored in the FIFO.
Definition: dma_device.hh:424
DmaPort::DmaReqState::totBytes
const Addr totBytes
Total number of bytes that this transaction involves.
Definition: dma_device.hh:86
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
DmaReadFifo::isActive
bool isActive() const
Is the DMA engine active (i.e., are there still in-flight accesses)?
Definition: dma_device.hh:467
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
trace.hh
DmaPort::defaultSSid
const uint32_t defaultSSid
Default substreamId.
Definition: dma_device.hh:173
Packet::senderState
SenderState * senderState
This packet's sender state.
Definition: packet.hh:509
RequestPort::sendAtomic
Tick sendAtomic(PacketPtr pkt)
Send an atomic request packet, where the data is moved and the state is updated in zero time,...
Definition: port.hh:461
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
DmaPort::DmaPort
DmaPort(ClockedObject *dev, System *s, uint32_t sid=0, uint32_t ssid=0)
Definition: dma_device.cc:55
ArmISA::s
Bitfield< 4 > s
Definition: miscregs_types.hh:556
CheckpointIn
Definition: serialize.hh:68
ChunkGenerator::last
bool last() const
Is this the last chunk?
Definition: chunk_generator.hh:146
DmaReadFifo::DmaReadFifo
DmaReadFifo(DmaPort &port, size_t size, unsigned max_req_size, unsigned max_pending, Request::Flags flags=0)
Definition: dma_device.cc:368
System::isAtomicMode
bool isAtomicMode() const
Is the system in atomic mode?
Definition: system.hh:253
DmaPort
Definition: dma_device.hh:58
DmaPort::sendAtomicReq
bool sendAtomicReq(DmaReqState *state)
Send the next packet from a DMA request in atomic mode.
Definition: dma_device.cc:241
PowerISA::bd
Bitfield< 15, 2 > bd
Definition: types.hh:61
DmaDevice::Params
DmaDeviceParams Params
Definition: dma_device.hh:206
DrainState::Draining
@ Draining
Draining buffers pending serialization/handover.
DmaPort::DmaReqState::gen
ChunkGenerator gen
Object to track what chunks of bytes to send at a time.
Definition: dma_device.hh:95
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
DmaReadFifo::DmaDoneEvent::process
void process()
Definition: dma_device.cc:591
ArmISA::offset
Bitfield< 23, 0 > offset
Definition: types.hh:153

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