gem5  v22.1.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
dma_device.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2012-2013, 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) 2004-2005 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 #ifndef __DEV_DMA_DEVICE_HH__
42 #define __DEV_DMA_DEVICE_HH__
43 
44 #include <deque>
45 #include <memory>
46 
47 #include "base/addr_range_map.hh"
48 #include "base/chunk_generator.hh"
49 #include "base/circlebuf.hh"
50 #include "dev/io_device.hh"
51 #include "mem/backdoor.hh"
52 #include "params/DmaDevice.hh"
53 #include "sim/drain.hh"
54 #include "sim/system.hh"
55 
56 namespace gem5
57 {
58 
59 class ClockedObject;
60 
61 class DmaPort : public RequestPort, public Drainable
62 {
63  private:
65 
71  void trySendTimingReq();
72 
80  void sendDma();
81 
83  {
87 
89  const Addr totBytes;
90 
93 
95  const Tick delay;
96 
99 
101  uint8_t *const data = nullptr;
102 
105 
108 
110  const uint32_t sid;
111  const uint32_t ssid;
112 
115 
117  uint8_t *_data, Request::Flags _flags, RequestorID _id,
118  uint32_t _sid, uint32_t _ssid, Event *ce, Tick _delay)
119  : completionEvent(ce), totBytes(tb), delay(_delay),
120  gen(addr, tb, chunk_sz), data(_data), flags(_flags), id(_id),
121  sid(_sid), ssid(_ssid), cmd(_cmd)
122  {}
123 
125  };
126 
128  bool sendAtomicReq(DmaReqState *state);
133  bool sendAtomicBdReq(DmaReqState *state);
134 
145  void handleRespPacket(PacketPtr pkt, Tick delay=0);
146  void handleResp(DmaReqState *state, Addr addr, Addr size, Tick delay=0);
147 
148  public:
151 
154  System *const sys;
155 
158 
159  protected:
162 
165 
167  uint32_t pendingCount = 0;
168 
170  PacketPtr inRetry = nullptr;
171 
173  const uint32_t defaultSid;
174 
176  const uint32_t defaultSSid;
177 
178  const int cacheLineSize;
179 
180  protected:
181 
182  bool recvTimingResp(PacketPtr pkt) override;
183  void recvReqRetry() override;
184 
185  public:
186 
187  DmaPort(ClockedObject *dev, System *s, uint32_t sid=0, uint32_t ssid=0);
188 
189  void
190  dmaAction(Packet::Command cmd, Addr addr, int size, Event *event,
191  uint8_t *data, Tick delay, Request::Flags flag=0);
192 
193  void
194  dmaAction(Packet::Command cmd, Addr addr, int size, Event *event,
195  uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay,
196  Request::Flags flag=0);
197 
198  bool dmaPending() const { return pendingCount > 0; }
199 
200  DrainState drain() override;
201 };
202 
203 class DmaDevice : public PioDevice
204 {
205  protected:
207 
208  public:
209  typedef DmaDeviceParams Params;
210  DmaDevice(const Params &p);
211  virtual ~DmaDevice() = default;
212 
213  void
214  dmaWrite(Addr addr, int size, Event *event, uint8_t *data,
215  uint32_t sid, uint32_t ssid, Tick delay=0)
216  {
218  sid, ssid, delay);
219  }
220 
221  void
222  dmaWrite(Addr addr, int size, Event *event, uint8_t *data, Tick delay=0)
223  {
224  dmaPort.dmaAction(MemCmd::WriteReq, addr, size, event, data, delay);
225  }
226 
227  void
228  dmaRead(Addr addr, int size, Event *event, uint8_t *data,
229  uint32_t sid, uint32_t ssid, Tick delay=0)
230  {
232  sid, ssid, delay);
233  }
234 
235  void
236  dmaRead(Addr addr, int size, Event *event, uint8_t *data, Tick delay=0)
237  {
238  dmaPort.dmaAction(MemCmd::ReadReq, addr, size, event, data, delay);
239  }
240 
241  bool dmaPending() const { return dmaPort.dmaPending(); }
242 
243  void init() override;
244 
245  unsigned int cacheBlockSize() const { return sys->cacheLineSize(); }
246 
247  Port &getPort(const std::string &if_name,
248  PortID idx=InvalidPortID) override;
249 
250 };
251 
260 class DmaCallback : public Drainable
261 {
262  public:
263  virtual const std::string name() const { return "DmaCallback"; }
264 
272  DrainState
273  drain() override
274  {
276  }
277 
278  protected:
279  int count = 0;
280 
281  virtual ~DmaCallback() = default;
282 
286  virtual void process() = 0;
287 
288  private:
294  void
296  {
297  if (--count == 0) {
298  process();
299  // Need to notify DrainManager that this object is finished
300  // draining, even though it is immediately deleted.
301  signalDrainDone();
302  delete this;
303  }
304  }
305 
306  public:
307 
312  Event *
314  {
315  ++count;
316  return new EventFunctionWrapper([this]{ chunkComplete(); }, name(),
317  true);
318  }
319 };
320 
364 class DmaReadFifo : public Drainable, public Serializable
365 {
366  public:
367  DmaReadFifo(DmaPort &port, size_t size,
368  unsigned max_req_size,
369  unsigned max_pending,
371 
372  ~DmaReadFifo();
373 
374  public: // Serializable
375  void serialize(CheckpointOut &cp) const override;
376  void unserialize(CheckpointIn &cp) override;
377 
378  public: // Drainable
379  DrainState drain() override;
380 
381  public: // FIFO access
398  bool tryGet(uint8_t *dst, size_t len);
399 
400  template<typename T>
401  bool
402  tryGet(T &value)
403  {
404  return tryGet(static_cast<T *>(&value), sizeof(T));
405  };
406 
415  void get(uint8_t *dst, size_t len);
416 
417  template<typename T>
418  T
419  get()
420  {
421  T value;
422  get(static_cast<uint8_t *>(&value), sizeof(T));
423  return value;
424  };
425 
427  size_t size() const { return buffer.size(); }
429  void flush() { buffer.flush(); }
430 
432  public: // FIFO fill control
447  void startFill(Addr start, size_t size);
448 
457  void stopFill();
458 
463  bool atEndOfBlock() const { return nextAddr == endAddr; }
464 
469  bool
470  isActive() const
471  {
472  return !(pendingRequests.empty() && atEndOfBlock());
473  }
474 
476  protected: // Callbacks
488  virtual void onEndOfBlock() {};
489 
501  virtual void onIdle() {};
502 
504  private: // Configuration
508  const size_t fifoSize;
511 
513 
514  const int cacheLineSize;
515 
516  private:
517  class DmaDoneEvent : public Event
518  {
519  public:
520  DmaDoneEvent(DmaReadFifo *_parent, size_t max_size);
521 
522  void kill();
523  void cancel();
524  bool canceled() const { return _canceled; }
525  void reset(size_t size);
526  void process();
527 
528  bool done() const { return _done; }
529  size_t requestSize() const { return _requestSize; }
530  const uint8_t *data() const { return _data.data(); }
531  uint8_t *data() { return _data.data(); }
532 
533  private:
535  bool _done = false;
536  bool _canceled = false;
537  size_t _requestSize;
539  };
540 
541  typedef std::unique_ptr<DmaDoneEvent> DmaDoneEventUPtr;
542 
547  void dmaDone();
548 
550  void handlePending();
551 
553  void resumeFill();
554 
556  void resumeFillTiming();
557 
559  void resumeFillBypass();
560 
561  private: // Internal state
563 
566 
569 };
570 
571 } // namespace gem5
572 
573 #endif // __DEV_DMA_DEVICE_HH__
Declaration and inline definition of ChunkGenerator object.
const char data[]
The AddrRangeMap uses an STL map to implement an interval tree for address decoding.
This class takes an arbitrary memory region (address/length pair) and generates a series of appropria...
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
DMA callback class.
Definition: dma_device.hh:261
void chunkComplete()
Called by DMA engine completion event on each chunk completion.
Definition: dma_device.hh:295
virtual ~DmaCallback()=default
DrainState drain() override
DmaPort ensures that all oustanding DMA accesses have completed before it finishes draining.
Definition: dma_device.hh:273
Event * getChunkEvent()
Request a chunk event.
Definition: dma_device.hh:313
virtual const std::string name() const
Definition: dma_device.hh:263
virtual void process()=0
Callback function invoked on completion of all chunks.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: dma_device.cc:363
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: dma_device.cc:148
void dmaWrite(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
Definition: dma_device.hh:214
bool dmaPending() const
Definition: dma_device.hh:241
void dmaRead(Addr addr, int size, Event *event, uint8_t *data, Tick delay=0)
Definition: dma_device.hh:236
unsigned int cacheBlockSize() const
Definition: dma_device.hh:245
void dmaRead(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
Definition: dma_device.hh:228
virtual ~DmaDevice()=default
void dmaWrite(Addr addr, int size, Event *event, uint8_t *data, Tick delay=0)
Definition: dma_device.hh:222
DmaDevice(const Params &p)
Definition: dma_device.cc:143
DmaDeviceParams Params
Definition: dma_device.hh:209
bool dmaPending() const
Definition: dma_device.hh:198
ClockedObject *const device
The device that owns this port.
Definition: dma_device.hh:150
EventFunctionWrapper sendEvent
Event used to schedule a future sending from the transmit list.
Definition: dma_device.hh:164
System *const sys
The system that device/port are in.
Definition: dma_device.hh:154
void dmaAction(Packet::Command cmd, Addr addr, int size, Event *event, uint8_t *data, Tick delay, Request::Flags flag=0)
Definition: dma_device.cc:196
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:258
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:167
DmaPort(ClockedObject *dev, System *s, uint32_t sid=0, uint32_t ssid=0)
Definition: dma_device.cc:58
void trySendTimingReq()
Take the first request on the transmit list and attempt to send a timing packet from it.
Definition: dma_device.cc:204
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:156
PacketPtr inRetry
The packet (if any) waiting for a retry to send.
Definition: dma_device.hh:170
uint32_t pendingCount
Number of outstanding packets the dma port has.
Definition: dma_device.hh:167
std::deque< DmaReqState * > transmitList
Use a deque as we never do any insertion or removal in the middle.
Definition: dma_device.hh:161
AddrRangeMap< MemBackdoorPtr, 1 > memBackdoors
Definition: dma_device.hh:64
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:330
const uint32_t defaultSid
Default streamId.
Definition: dma_device.hh:173
void handleResp(DmaReqState *state, Addr addr, Addr size, Tick delay=0)
Definition: dma_device.cc:82
const RequestorID requestorId
Id for all requests.
Definition: dma_device.hh:157
bool recvTimingResp(PacketPtr pkt) override
Receive a timing response from the peer.
Definition: dma_device.cc:132
bool sendAtomicReq(DmaReqState *state)
Send the next packet from a DMA request in atomic mode.
Definition: dma_device.cc:244
const uint32_t defaultSSid
Default substreamId.
Definition: dma_device.hh:176
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:67
const int cacheLineSize
Definition: dma_device.hh:178
std::vector< uint8_t > _data
Definition: dma_device.hh:538
DmaDoneEvent(DmaReadFifo *_parent, size_t max_size)
Definition: dma_device.cc:566
const uint8_t * data() const
Definition: dma_device.hh:530
Buffered DMA engine helper class.
Definition: dma_device.hh:365
Fifo< uint8_t > buffer
Definition: dma_device.hh:562
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: dma_device.cc:402
size_t size() const
Get the amount of data stored in the FIFO.
Definition: dma_device.hh:427
bool tryGet(T &value)
Definition: dma_device.hh:402
virtual void onIdle()
Last response received callback.
Definition: dma_device.hh:501
bool tryGet(uint8_t *dst, size_t len)
Try to read data from the FIFO.
Definition: dma_device.cc:420
void startFill(Addr start, size_t size)
Start filling the FIFO.
Definition: dma_device.cc:438
DmaReadFifo(DmaPort &port, size_t size, unsigned max_req_size, unsigned max_pending, Request::Flags flags=0)
Definition: dma_device.cc:371
const Request::Flags reqFlags
Request flags.
Definition: dma_device.hh:510
virtual void onEndOfBlock()
End of block callback.
Definition: dma_device.hh:488
void resumeFill()
Try to issue new DMA requests or bypass DMA requests.
Definition: dma_device.cc:461
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:559
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: dma_device.cc:412
std::deque< DmaDoneEventUPtr > freeRequests
Definition: dma_device.hh:568
void flush()
Flush the FIFO.
Definition: dma_device.hh:429
void dmaDone()
DMA request done, handle incoming data and issue new request.
Definition: dma_device.cc:528
bool isActive() const
Is the DMA engine active (i.e., are there still in-flight accesses)?
Definition: dma_device.hh:470
const int cacheLineSize
Definition: dma_device.hh:514
void stopFill()
Stop the DMA engine.
Definition: dma_device.cc:448
void resumeFillBypass()
Try to bypass DMA requests in non-caching mode.
Definition: dma_device.cc:480
bool atEndOfBlock() const
Has the DMA engine sent out the last request for the active block?
Definition: dma_device.hh:463
const Addr maxReqSize
Maximum request size in bytes.
Definition: dma_device.hh:501
const size_t fifoSize
Maximum FIFO size in bytes.
Definition: dma_device.hh:508
void handlePending()
Handle pending requests that have been flagged as done.
Definition: dma_device.cc:540
void resumeFillTiming()
Try to issue new DMA requests during normal execution.
Definition: dma_device.cc:502
std::unique_ptr< DmaDoneEvent > DmaDoneEventUPtr
Definition: dma_device.hh:541
std::deque< DmaDoneEventUPtr > pendingRequests
Definition: dma_device.hh:567
Interface for objects that might require draining before checkpointing.
Definition: drain.hh:235
void flush()
Definition: circlebuf.hh:225
size_t size() const
Definition: circlebuf.hh:222
Command
List of all commands associated with a packet.
Definition: packet.hh:84
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:294
This device is the base class which all devices senstive to an address range inherit from.
Definition: io_device.hh:103
System * sys
Definition: io_device.hh:105
Ports are used to interface objects to each other.
Definition: port.hh:62
A RequestPort is a specialisation of a Port, which implements the default protocol for the three diff...
Definition: port.hh:79
Basic support for object serialization.
Definition: serialize.hh:170
unsigned int cacheLineSize() const
Get the cache line size of the system.
Definition: system.hh:311
STL deque class.
Definition: stl.hh:44
void signalDrainDone() const
Signal that an object is drained.
Definition: drain.hh:305
DrainState
Object drain/handover states.
Definition: drain.hh:75
@ Draining
Draining buffers pending serialization/handover.
@ Drained
Buffers drained, ready for serialization/handover.
atomic_var_t state
Definition: helpers.cc:188
uint16_t len
Definition: helpers.cc:62
uint8_t flags
Definition: helpers.cc:66
Bitfield< 10, 5 > event
Bitfield< 29, 28 > ce
Bitfield< 27 > tb
Definition: dt_constants.hh:77
Bitfield< 1 > s
Definition: pagetable.hh:64
Bitfield< 54 > p
Definition: pagetable.hh:70
Bitfield< 3 > addr
Definition: types.hh:84
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
const PortID InvalidPortID
Definition: types.hh:246
std::ostream CheckpointOut
Definition: serialize.hh:66
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:147
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:245
uint64_t Tick
Tick count type.
Definition: types.hh:58
uint16_t RequestorID
Definition: request.hh:95
const Packet::Command cmd
Command for the request.
Definition: dma_device.hh:114
const Addr totBytes
Total number of bytes that this transaction involves.
Definition: dma_device.hh:89
DmaReqState(Packet::Command _cmd, Addr addr, Addr chunk_sz, Addr tb, uint8_t *_data, Request::Flags _flags, RequestorID _id, uint32_t _sid, uint32_t _ssid, Event *ce, Tick _delay)
Definition: dma_device.hh:116
Event * completionEvent
Event to call on the device when this transaction (all packets) complete.
Definition: dma_device.hh:86
Addr numBytes
Number of bytes that have been acked for this transaction.
Definition: dma_device.hh:92
const uint32_t sid
Stream IDs.
Definition: dma_device.hh:110
const RequestorID id
The requestor ID to use for requests.
Definition: dma_device.hh:107
const Tick delay
Amount to delay completion of dma by.
Definition: dma_device.hh:95
ChunkGenerator gen
Object to track what chunks of bytes to send at a time.
Definition: dma_device.hh:98
uint8_t *const data
Pointer to a buffer for the data.
Definition: dma_device.hh:101
const Request::Flags flags
The flags to use for requests.
Definition: dma_device.hh:104
A virtual base opaque structure used to hold state associated with the packet (e.g....
Definition: packet.hh:468

Generated on Wed Dec 21 2022 10:22:33 for gem5 by doxygen 1.9.1