gem5  [DEVELOP-FOR-23.0]
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
cfi_mem.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2010-2013, 2015, 2021 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) 2001-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 #include "mem/cfi_mem.hh"
42 
43 #include <cmath>
44 
45 #include "base/intmath.hh"
46 #include "base/random.hh"
47 #include "base/trace.hh"
48 #include "debug/CFI.hh"
49 #include "debug/Drain.hh"
50 
51 namespace gem5
52 {
53 
54 namespace memory
55 {
56 
57 bool
59 {
60  return locked[blockIdx(block_address)];
61 }
62 
63 void
65 {
66  locked[blockIdx(block_address)] = true;
67 }
68 
69 void
71 {
72  locked[blockIdx(block_address)] = false;
73 }
74 
75 void
77 {
78  SERIALIZE_CONTAINER(locked);
79 }
80 
81 void
83 {
84  UNSERIALIZE_CONTAINER(locked);
85 }
86 
87 uint32_t
89 {
90  return block_address / blockSize;
91 }
92 
93 void
94 CfiMemory::ProgramBuffer::setup(ssize_t buffer_size)
95 {
97  if (buffer_size > MAX_BUFFER_SIZE) {
98  buffer_size = MAX_BUFFER_SIZE;
99  }
100 
101  buffer.resize(buffer_size);
102  std::fill(buffer.begin(), buffer.end(), 0);
103  bytesWritten = 0;
104 }
105 
106 bool
108  void *data_ptr, ssize_t size)
109 {
110  if (bytesWritten >= buffer.size())
111  return true;
112 
113  if (bytesWritten == 0) {
114  blockPointer = flash_address;
115  }
116 
117  const Addr offset = flash_address - blockPointer;
118 
119  if (flash_address < blockPointer || offset >= MAX_BUFFER_SIZE)
120  return true;
121 
122  std::memcpy(buffer.data() + offset, data_ptr, size);
123  bytesWritten += size;
124 
125  return false;
126 }
127 
128 bool
130 {
131  if (parent.blocks.isLocked(blockPointer)) {
132  return false;
133  } else {
134  std::memcpy(parent.toHostAddr(parent.start() + blockPointer),
135  buffer.data(), bytesWritten);
136  return true;
137  }
138 }
139 
140 void
142 {
143  SERIALIZE_CONTAINER(buffer);
144  SERIALIZE_SCALAR(bytesWritten);
145  SERIALIZE_SCALAR(blockPointer);
146 }
147 
148 void
150 {
151  UNSERIALIZE_CONTAINER(buffer);
152  UNSERIALIZE_SCALAR(bytesWritten);
153  UNSERIALIZE_SCALAR(blockPointer);
154 }
155 
156 CfiMemory::CfiMemory(const CfiMemoryParams &p)
157  : AbstractMemory(p),
158  port(name() + ".port", *this), latency(p.latency),
160  retryReq(false), retryResp(false),
161  releaseEvent([this]{ release(); }, name()),
162  dequeueEvent([this]{ dequeue(); }, name()),
163  numberOfChips(2),
164  vendorID(p.vendor_id),
165  deviceID(p.device_id),
166  bankWidth(p.bank_width),
167  readState(CfiCommand::READ_ARRAY), writeState(CfiCommand::NO_CMD),
168  statusRegister(STATUS_READY),
169  blocks(*this, size() / p.blk_size, p.blk_size),
170  programBuffer(*this),
171  cfiQueryTable{
172  0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
173  0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
174  /* Query-unique ASCII string */
175  'Q', 'R', 'Y',
176  /* Primary Algorithm Command Set and Control = Intel/Sharp */
177  0x01, 0x00,
178  /* Address for Primary Algorithm extended Query */
179  0x31, 0x00,
180  /* Alternative Algorithm Command Set and Control Interface */
181  0x00, 0x00,
182  /* Address for Alternative Algorithm extended Query */
183  0x00, 0x00,
184  /* Vcc Minimum Program/Erase or Write voltage ([7:4].[3-0]V) */
185  0x45,
186  /* Vcc Maximum Program/Erase or Write voltage ([7:4].[3-0]V) */
187  0x55,
188  /* Vpp Minimum Program/Erase voltage (0 = No Vpp pin) */
189  0x00,
190  /* Vpp Minimum Program/Erase voltage (0 = No Vpp pin) */
191  0x00,
192  /* Typical timeout per single byte/word/D-word program: (2^N us) */
193  0x01,
194  /* Typical timeout for maximum-size multi-byte program: (2^N us) */
195  0x01,
196  /* Typical timeout per individual block erase: (2^N ms) */
197  0x01,
198  /* Typical timeout for full chip erase: (2^N ms) */
199  0x00,
200  /* Maximum timeout for byte/word/D-word program (2^N typical) */
201  0x00,
202  /* Maximum timeout for multi-byte program (2^N typical) */
203  0x00,
204  /* Maximum timeout per individual block erase (2^N typical) */
205  0x00,
206  /* Maximum timeout for chip erase (2^N typical) */
207  0x00,
208  /* Device Size in number of bytes (2^N) */
209  static_cast<uint8_t>(log2(size())),
210  /* Flash Device Interface Code description */
211  0x05, 0x00,
212  /* Maximum number of bytes in multi-byte program (2^N) */
213  static_cast<uint8_t>(bits(log2i(ProgramBuffer::MAX_BUFFER_SIZE), 7, 0)),
214  static_cast<uint8_t>(bits(log2i(ProgramBuffer::MAX_BUFFER_SIZE), 15, 8)),
215  /* Number of Erase Block Regions within device */
216  0x01,
217  /* Erase Block Region Information */
218  static_cast<uint8_t>(bits(blocks.number(), 7, 0)),
219  static_cast<uint8_t>(bits(blocks.number(), 15, 8)),
220  static_cast<uint8_t>(bits(blocks.size(), 7, 0)),
221  static_cast<uint8_t>(bits(blocks.size(), 15, 8)),
222  0x00,0x00,0x00,0x00,//empty Block region 2 info
223  0x00,0x00,0x00,0x00,//empty Block region 3 info
224  0x00,0x00,0x00,0x00//empty Block region 4 info
225  }
226 {}
227 
228 void
230 {
232 
233  // allow unconnected memories as this is used in several ruby
234  // systems at the moment
235  if (port.isConnected()) {
237  }
238 }
239 
240 Tick
242 {
243  panic_if(pkt->cacheResponding(), "Should not see packets where cache "
244  "is responding");
245 
246  cfiAccess(pkt);
247  return getLatency();
248 }
249 
250 Tick
252 {
253  Tick latency = recvAtomic(pkt);
254 
255  if (backdoor.ptr())
256  _backdoor = &backdoor;
257  return latency;
258 }
259 
260 void
262 {
263  pkt->pushLabel(name());
264 
265  functionalAccess(pkt);
266 
267  bool done = false;
268  auto p = packetQueue.begin();
269  // potentially update the packets in our packet queue as well
270  while (!done && p != packetQueue.end()) {
271  done = pkt->trySatisfyFunctional(p->pkt);
272  ++p;
273  }
274 
275  pkt->popLabel();
276 }
277 
278 void
280  MemBackdoorPtr &_backdoor)
281 {
282  if (backdoor.ptr())
283  _backdoor = &backdoor;
284 }
285 
286 bool
288 {
289  panic_if(pkt->cacheResponding(), "Should not see packets where cache "
290  "is responding");
291 
292  panic_if(!(pkt->isRead() || pkt->isWrite()),
293  "Should only see read and writes at memory controller, "
294  "saw %s to %#llx\n", pkt->cmdString(), pkt->getAddr());
295 
296  // we should not get a new request after committing to retry the
297  // current one, but unfortunately the CPU violates this rule, so
298  // simply ignore it for now
299  if (retryReq)
300  return false;
301 
302  // if we are busy with a read or write, remember that we have to
303  // retry
304  if (isBusy) {
305  retryReq = true;
306  return false;
307  }
308 
309  // technically the packet only reaches us after the header delay,
310  // and since this is a memory controller we also need to
311  // deserialise the payload before performing any write operation
312  Tick receive_delay = pkt->headerDelay + pkt->payloadDelay;
313  pkt->headerDelay = pkt->payloadDelay = 0;
314 
315  // update the release time according to the bandwidth limit, and
316  // do so with respect to the time it takes to finish this request
317  // rather than long term as it is the short term data rate that is
318  // limited for any real memory
319 
320  // calculate an appropriate tick to release to not exceed
321  // the bandwidth limit
322  Tick duration = pkt->getSize() * bandwidth;
323 
324  // only consider ourselves busy if there is any need to wait
325  // to avoid extra events being scheduled for (infinitely) fast
326  // memories
327  if (duration != 0) {
328  schedule(releaseEvent, curTick() + duration);
329  isBusy = true;
330  }
331 
332  // go ahead and deal with the packet and put the response in the
333  // queue if there is one
334  bool needs_response = pkt->needsResponse();
335  recvAtomic(pkt);
336  // turn packet around to go back to requester if response expected
337  if (needs_response) {
338  // recvAtomic() should already have turned packet into
339  // atomic response
340  assert(pkt->isResponse());
341 
342  Tick when_to_send = curTick() + receive_delay + getLatency();
343 
344  // typically this should be added at the end, so start the
345  // insertion sort with the last element, also make sure not to
346  // re-order in front of some existing packet with the same
347  // address, the latter is important as this memory effectively
348  // hands out exclusive copies (shared is not asserted)
349  auto i = packetQueue.end();
350  --i;
351  while (i != packetQueue.begin() && when_to_send < i->tick &&
352  !i->pkt->matchAddr(pkt)) {
353  --i;
354  }
355 
356  // emplace inserts the element before the position pointed to by
357  // the iterator, so advance it one step
358  packetQueue.emplace(++i, pkt, when_to_send);
359 
360  if (!retryResp && !dequeueEvent.scheduled())
361  schedule(dequeueEvent, packetQueue.back().tick);
362  } else {
363  pendingDelete.reset(pkt);
364  }
365 
366  return true;
367 }
368 
369 void
371 {
372  assert(isBusy);
373  isBusy = false;
374  if (retryReq) {
375  retryReq = false;
376  port.sendRetryReq();
377  }
378 }
379 
380 void
382 {
383  assert(!packetQueue.empty());
384  DeferredPacket deferred_pkt = packetQueue.front();
385 
386  retryResp = !port.sendTimingResp(deferred_pkt.pkt);
387 
388  if (!retryResp) {
389  packetQueue.pop_front();
390 
391  // if the queue is not empty, schedule the next dequeue event,
392  // otherwise signal that we are drained if we were asked to do so
393  if (!packetQueue.empty()) {
394  // if there were packets that got in-between then we
395  // already have an event scheduled, so use re-schedule
397  std::max(packetQueue.front().tick, curTick()), true);
398  } else if (drainState() == DrainState::Draining) {
399  DPRINTF(Drain, "Draining of CfiMemory complete\n");
400  signalDrainDone();
401  }
402  }
403 }
404 
405 Tick
407 {
408  return latency +
410 }
411 
412 void
414 {
415  assert(retryResp);
416 
417  dequeue();
418 }
419 
420 Port &
421 CfiMemory::getPort(const std::string &if_name, PortID idx)
422 {
423  if (if_name != "port") {
424  return AbstractMemory::getPort(if_name, idx);
425  } else {
426  return port;
427  }
428 }
429 
432 {
433  if (!packetQueue.empty()) {
434  DPRINTF(Drain, "CfiMemory Queue has requests, waiting to drain\n");
435  return DrainState::Draining;
436  } else {
437  return DrainState::Drained;
438  }
439 }
440 
441 void
443 {
446 
448 
451 }
452 
453 void
455 {
458 
460 
463 }
464 
465 CfiMemory::MemoryPort::MemoryPort(const std::string& _name,
466  CfiMemory& _memory)
467  : ResponsePort(_name), mem(_memory)
468 { }
469 
472 {
473  AddrRangeList ranges;
474  ranges.push_back(mem.getAddrRange());
475  return ranges;
476 }
477 
478 Tick
480 {
481  return mem.recvAtomic(pkt);
482 }
483 
484 Tick
486  PacketPtr pkt, MemBackdoorPtr &_backdoor)
487 {
488  return mem.recvAtomicBackdoor(pkt, _backdoor);
489 }
490 
491 void
493 {
494  mem.recvFunctional(pkt);
495 }
496 
497 void
499  MemBackdoorPtr &_backdoor)
500 {
501  mem.recvMemBackdoorReq(req, _backdoor);
502 }
503 
504 bool
506 {
507  return mem.recvTimingReq(pkt);
508 }
509 
510 void
512 {
513  mem.recvRespRetry();
514 }
515 
516 void
518 {
519  if (pkt->isWrite()) {
520  write(pkt);
521  } else {
522  read(pkt);
523  }
524 }
525 
526 void
528 {
529  DPRINTF(CFI, "write, address: %#x, val: %#x\n", pkt->getAddr(),
530  pkt->getUintX(ByteOrder::little));
531 
532  const Addr flash_address = pkt->getAddr() - start();
533 
534  const uint16_t value = pkt->getUintX(ByteOrder::little) & 0xffff;
535  const auto new_cmd = static_cast<CfiCommand>(value & 0xff);
536 
537  switch (writeState) {
538  case CfiCommand::NO_CMD:
539  handleCommand(new_cmd);
540  break;
541 
543  if (new_cmd == CfiCommand::BLOCK_ERASE_CONFIRM) {
544  // Erasing the block
545  // Check if block is locked
546  if (blocks.isLocked(flash_address)) {
548  } else {
549  blocks.erase(pkt);
550  }
551  } else {
553  }
556  break;
557 
559  if (new_cmd == CfiCommand::LOCK_BLOCK) {
560 
561  // Lock the addressed block
562  blocks.lock(flash_address);
564 
565  } else if (new_cmd == CfiCommand::UNLOCK_BLOCK) {
566 
567  // Unlock the addressed block
568  blocks.unlock(flash_address);
570 
571  } else {
573  }
574 
576  break;
577 
581 
582  if (blocks.isLocked(flash_address)) {
584  } else {
586  return;
587  }
588  break;
589 
591  // Buffer size in bytes
592  auto buffer_size = (value + 1) * sizeof(uint32_t);
593 
594  // Clearing the program buffer
595  programBuffer.setup(buffer_size);
596 
599  break;
600  }
601 
603  // Write to the buffer and check if a writeback is needed
604  // (if the buffer is full)
606  flash_address, pkt->getPtr<void>(), pkt->getSize());
607 
608  if (writeback) {
609  if (new_cmd == CfiCommand::BUFFERED_PROGRAM_CONFIRM) {
610  auto success = programBuffer.writeback();
611  if (!success)
613 
615  } else {
617  }
619  }
620  break;
621  }
622 
623  default:
624  panic("Invalid Write State\n");
625  return;
626  }
627 
628  pkt->makeResponse();
629 }
630 
631 void
633 {
634  const Addr flash_address = pkt->getAddr() - start();
635  uint64_t value = 0;
636 
637  switch (readState) {
639  value = statusRegister;
640  break;
642  value = readDeviceID(flash_address);
643  break;
645  value = cfiQuery(flash_address);
646  break;
649  return;
650  default:
651  panic("Invalid Read State\n");
652  return;
653  }
654 
655  if (numberOfChips == 2) {
656  value |= (value << 16);
657  }
658 
659  pkt->setUintX(value, ByteOrder::little);
660  pkt->makeResponse();
661 
662  DPRINTF(CFI, "read, address: %#x, val: %#x\n", pkt->getAddr(),
663  pkt->getUintX(ByteOrder::little));
664 
665 }
666 
667 uint64_t
668 CfiMemory::readDeviceID(Addr flash_address) const
669 {
670  switch ((flash_address & 0xff) / bankWidth) {
671  case 0x00: // vendor ID
672  return vendorID;
673  case 0x01: // device ID
674  return deviceID;
675  case 0x02: // lock bit
676  return blocks.isLocked(flash_address);
677  default:
678  // Unsupported entries
679  warn("Invalid Device Identifier code: %d\n", flash_address & 0xff);
680  return 0;
681  }
682 }
683 
684 void
686 {
687  switch (new_cmd) {
689  DPRINTF(CFI, "CFI Command: Read Array\n");
691  break;
693  DPRINTF(CFI, "CFI Command: Read Device Identifier\n");
695  break;
697  DPRINTF(CFI, "CFI Command: CFI Query\n");
699  break;
701  DPRINTF(CFI, "CFI Command: Read Status Register\n");
703  break;
705  DPRINTF(CFI, "CFI Command: Clear Status Register\n");
707  break;
709  DPRINTF(CFI, "CFI Command: Buffered Program Confirm\n");
710  break;
712  DPRINTF(CFI, "CFI Command: Erase Block Setup\n");
715  break;
717  DPRINTF(CFI, "CFI Command: Lock Block Setup\n");
719  break;
721  DPRINTF(CFI, "CFI Command: Word Program\n");
724  break;
726  DPRINTF(CFI, "CFI Command: Buffered Program Setup\n");
729  break;
731  //because of how u-boot works and reset the flash
732  //we have to ignore the AMD RESET explicitly
733  // (see the function __flash_cmd_reset in drivers/mtd/cfi_flash.c)
734  break;
735  default:
736  panic("Don't know what to do with %#x\n",
737  static_cast<uint16_t>(new_cmd));
738  }
739 
740 }
741 
742 uint64_t
743 CfiMemory::cfiQuery(Addr flash_address)
744 {
745  flash_address /= bankWidth;
746 
747  panic_if(flash_address >= sizeof(cfiQueryTable),
748  "Acessing invalid entry in CFI query table (addr=%#x)",
749  flash_address);
750 
751  return cfiQueryTable[flash_address];
752 }
753 
754 void
756 {
757  auto host_address = parent.toHostAddr(pkt->getAddr());
758  std::memset(host_address, 0xff, blockSize);
759 }
760 
761 } // namespace memory
762 } // namespace gem5
gem5::memory::CfiMemory::pendingDelete
std::unique_ptr< Packet > pendingDelete
Upstream caches need this packet until true is returned, so hold it for deletion until a subsequent c...
Definition: cfi_mem.hh:330
gem5::curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:46
gem5::memory::CfiMemory::isBusy
bool isBusy
Track the state of the memory as either idle or busy, no need for an enum with only two states.
Definition: cfi_mem.hh:289
gem5::PortID
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:245
gem5::Packet::cmdString
const std::string & cmdString() const
Return the string name of the cmd field (for debugging and tracing).
Definition: packet.hh:588
gem5::memory::CfiMemory::drain
DrainState drain() override
Draining is the process of clearing out the states of SimObjects.These are the SimObjects that are pa...
Definition: cfi_mem.cc:431
gem5::MipsISA::fill
fill
Definition: pra_constants.hh:57
gem5::SimObject::getPort
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
UNSERIALIZE_OBJ
#define UNSERIALIZE_OBJ(obj)
Definition: serialize.hh:655
gem5::memory::CfiMemory::CfiCommand::WORD_PROGRAM
@ WORD_PROGRAM
warn
#define warn(...)
Definition: logging.hh:256
gem5::memory::CfiMemory::programBuffer
ProgramBuffer programBuffer
Definition: cfi_mem.hh:346
gem5::memory::CfiMemory::writeState
CfiCommand writeState
Definition: cfi_mem.hh:340
gem5::memory::CfiMemory::readState
CfiCommand readState
Previous command (issued in the previous write cycle)
Definition: cfi_mem.hh:339
gem5::memory::CfiMemory::write
void write(PacketPtr pkt)
Write request to the CFI Memory.
Definition: cfi_mem.cc:527
gem5::memory::CfiMemory::DeferredPacket
A deferred packet stores a packet along with its scheduled transmission time.
Definition: cfi_mem.hh:226
gem5::Packet::getUintX
uint64_t getUintX(ByteOrder endian) const
Get the data in the packet byte swapped from the specified endianness and zero-extended to 64 bits.
Definition: packet.cc:352
gem5::memory::CfiMemory::read
void read(PacketPtr pkt)
Read request to the CFI Memory.
Definition: cfi_mem.cc:632
gem5::memory::CfiMemory::dequeue
void dequeue()
Dequeue a packet from our internal packet queue and move it to the port where it will be sent as soon...
Definition: cfi_mem.cc:381
gem5::ResponsePort::sendRetryReq
void sendRetryReq()
Send a retry to the request port that previously attempted a sendTimingReq to this response port and ...
Definition: port.hh:473
gem5::memory::CfiMemory::recvMemBackdoorReq
void recvMemBackdoorReq(const MemBackdoorReq &req, MemBackdoorPtr &_backdoor)
Definition: cfi_mem.cc:279
gem5::memory::CfiMemory::BlockData::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: cfi_mem.cc:82
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:575
gem5::memory::CfiMemory::recvAtomicBackdoor
Tick recvAtomicBackdoor(PacketPtr pkt, MemBackdoorPtr &_backdoor)
Definition: cfi_mem.cc:251
gem5::Drainable::drainState
DrainState drainState() const
Return the current drain state of an object.
Definition: drain.hh:324
gem5::memory::CfiMemory::numberOfChips
const uint8_t numberOfChips
Definition: cfi_mem.hh:332
memory
Definition: mem.h:38
UNSERIALIZE_CONTAINER
#define UNSERIALIZE_CONTAINER(member)
Definition: serialize.hh:634
gem5::ArmISA::writeback
Bitfield< 21 > writeback
Definition: types.hh:126
gem5::memory::CfiMemory::MemoryPort::recvTimingReq
bool recvTimingReq(PacketPtr pkt) override
Receive a timing request from the peer.
Definition: cfi_mem.cc:505
gem5::memory::CfiMemory::ProgramBuffer::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: cfi_mem.cc:149
gem5::memory::CfiMemory::releaseEvent
EventFunctionWrapper releaseEvent
Definition: cfi_mem.hh:309
gem5::CheckpointIn
Definition: serialize.hh:68
gem5::memory::CfiMemory::recvRespRetry
void recvRespRetry()
Definition: cfi_mem.cc:413
gem5::memory::CfiMemory::getLatency
Tick getLatency() const
Detemine the latency.
Definition: cfi_mem.cc:406
gem5::memory::CfiMemory::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: cfi_mem.cc:421
gem5::SimObject::init
virtual void init()
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: sim_object.cc:73
random.hh
gem5::memory::CfiMemory::CfiCommand::READ_ARRAY
@ READ_ARRAY
gem5::Packet::cacheResponding
bool cacheResponding() const
Definition: packet.hh:659
gem5::Packet::isWrite
bool isWrite() const
Definition: packet.hh:594
gem5::memory::CfiMemory::MemoryPort::recvAtomicBackdoor
Tick recvAtomicBackdoor(PacketPtr pkt, MemBackdoorPtr &_backdoor) override
Receive an atomic request packet from the peer, and optionally provide a backdoor to the data being a...
Definition: cfi_mem.cc:485
gem5::memory::CfiMemory::MemoryPort::recvFunctional
void recvFunctional(PacketPtr pkt) override
Receive a functional request packet from the peer.
Definition: cfi_mem.cc:492
gem5::log2i
static constexpr int log2i(int value)
Calculate the log2 of a power of 2 integer.
Definition: intmath.hh:295
gem5::EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1012
gem5::memory::CfiMemory::port
MemoryPort port
Definition: cfi_mem.hh:258
gem5::memory::CfiMemory::CfiCommand::LOCK_BLOCK_SETUP
@ LOCK_BLOCK_SETUP
gem5::memory::CfiMemory::CfiCommand::UNLOCK_BLOCK
@ UNLOCK_BLOCK
gem5::ArmISA::i
Bitfield< 7 > i
Definition: misc_types.hh:67
gem5::Packet::headerDelay
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
Definition: packet.hh:431
gem5::memory::CfiMemory::cfiQuery
uint64_t cfiQuery(Addr addr)
Return the selected entry in the CFI table.
Definition: cfi_mem.cc:743
gem5::memory::CfiMemory::init
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: cfi_mem.cc:229
gem5::memory::CfiMemory::CfiCommand::BUFFER_SIZE_READ
@ BUFFER_SIZE_READ
This is not a real command, but it is used by the internal model only to represent the 2nd write cycl...
gem5::memory::CfiMemory::bandwidth
const double bandwidth
Bandwidth in ticks per byte.
Definition: cfi_mem.hh:283
gem5::Random::random
std::enable_if_t< std::is_integral_v< T >, T > random()
Use the SFINAE idiom to choose an implementation based on whether the type is integral or floating po...
Definition: random.hh:90
gem5::memory::CfiMemory::BlockData::blockIdx
uint32_t blockIdx(Addr block_address) const
Definition: cfi_mem.cc:88
SERIALIZE_ENUM
#define SERIALIZE_ENUM(scalar)
Definition: serialize.hh:591
gem5::Packet::payloadDelay
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
Definition: packet.hh:449
gem5::memory::CfiMemory::STATUS_LOCK_ERROR
static const uint8_t STATUS_LOCK_ERROR
Definition: cfi_mem.hh:99
gem5::memory::CfiMemory::CfiCommand::NO_CMD
@ NO_CMD
gem5::DrainState
DrainState
Object drain/handover states.
Definition: drain.hh:74
gem5::memory::CfiMemory::CfiCommand::CLEAR_STATUS_REG
@ CLEAR_STATUS_REG
gem5::Packet::isRead
bool isRead() const
Definition: packet.hh:593
gem5::memory::AbstractMemory::start
Addr start() const
Get the start address.
Definition: abstract_mem.hh:315
gem5::memory::CfiMemory::ProgramBuffer::setup
void setup(ssize_t buffer_size)
Start buffering.
Definition: cfi_mem.cc:94
gem5::memory::AbstractMemory::functionalAccess
void functionalAccess(PacketPtr pkt)
Perform an untimed memory read or write without changing anything but the memory itself.
Definition: abstract_mem.cc:482
gem5::Named::name
virtual std::string name() const
Definition: named.hh:47
gem5::VegaISA::p
Bitfield< 54 > p
Definition: pagetable.hh:70
gem5::memory::CfiMemory::CfiCommand::LOCK_BLOCK
@ LOCK_BLOCK
gem5::memory::CfiMemory::CfiCommand::BUFFERED_PROGRAM_SETUP
@ BUFFERED_PROGRAM_SETUP
gem5::memory::CfiMemory::STATUS_ERASE_ERROR
static const uint8_t STATUS_ERASE_ERROR
Possible in the status register.
Definition: cfi_mem.hh:98
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:210
SERIALIZE_OBJ
#define SERIALIZE_OBJ(obj)
Definition: serialize.hh:648
gem5::memory::CfiMemory::MemoryPort::getAddrRanges
AddrRangeList getAddrRanges() const override
Get a list of the non-overlapping address ranges the owner is responsible for.
Definition: cfi_mem.cc:471
gem5::Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:294
gem5::memory::AbstractMemory
An abstract memory represents a contiguous block of physical memory, with an associated address range...
Definition: abstract_mem.hh:110
gem5::Tick
uint64_t Tick
Tick count type.
Definition: types.hh:58
gem5::memory::CfiMemory::MemoryPort::recvMemBackdoorReq
void recvMemBackdoorReq(const MemBackdoorReq &req, MemBackdoorPtr &_backdoor) override
Receive a request for a back door to a range of memory.
Definition: cfi_mem.cc:498
gem5::memory::CfiMemory::latency_var
const Tick latency_var
Fudge factor added to the latency.
Definition: cfi_mem.hh:269
gem5::memory::CfiMemory::ProgramBuffer::writeback
bool writeback()
Definition: cfi_mem.cc:129
gem5::memory::CfiMemory::STATUS_READY
static const uint8_t STATUS_READY
Definition: cfi_mem.hh:100
gem5::memory::CfiMemory::MemoryPort::MemoryPort
MemoryPort(const std::string &_name, CfiMemory &_memory)
Definition: cfi_mem.cc:465
gem5::memory::CfiMemory
CfiMemory: This is modelling a flash memory adhering to the Common Flash Interface (CFI):
Definition: cfi_mem.hh:71
gem5::ResponsePort::sendTimingResp
bool sendTimingResp(PacketPtr pkt)
Attempt to send a timing response to the request port by calling its corresponding receive function.
Definition: port.hh:438
gem5::ArmISA::offset
Bitfield< 23, 0 > offset
Definition: types.hh:144
gem5::memory::CfiMemory::ProgramBuffer::write
bool write(Addr flash_address, void *data_ptr, ssize_t size)
Write data into the buffer.
Definition: cfi_mem.cc:107
gem5::memory::CfiMemory::packetQueue
std::list< DeferredPacket > packetQueue
Internal (unbounded) storage to mimic the delay caused by the actual memory access.
Definition: cfi_mem.hh:276
gem5::memory::CfiMemory::blocks
BlockData blocks
Definition: cfi_mem.hh:344
gem5::EventManager::reschedule
void reschedule(Event &event, Tick when, bool always=false)
Definition: eventq.hh:1030
gem5::memory::CfiMemory::CfiCommand::BUFFERED_PROGRAM_CONFIRM
@ BUFFERED_PROGRAM_CONFIRM
gem5::memory::CfiMemory::retryReq
bool retryReq
Remember if we have to retry an outstanding request that arrived while we were busy.
Definition: cfi_mem.hh:295
gem5::memory::CfiMemory::cfiAccess
void cfiAccess(PacketPtr pkt)
Make a read/write access to the CFI Memory.
Definition: cfi_mem.cc:517
gem5::Port::isConnected
bool isConnected() const
Is this port currently connected to a peer?
Definition: port.hh:133
gem5::bits
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Definition: bitfield.hh:76
gem5::memory::CfiMemory::CfiCommand::READ_DEVICE_ID
@ READ_DEVICE_ID
gem5::DrainState::Drained
@ Drained
Buffers drained, ready for serialization/handover.
gem5::memory::CfiMemory::CfiCommand::BLOCK_ERASE_CONFIRM
@ BLOCK_ERASE_CONFIRM
gem5::Packet::pushLabel
void pushLabel(const std::string &lbl)
Push label for PrintReq (safe to call unconditionally).
Definition: packet.hh:1461
gem5::memory::CfiMemory::BlockData::erase
void erase(PacketPtr pkt)
Erase a single block.
Definition: cfi_mem.cc:755
gem5::memory::CfiMemory::CfiMemory
CfiMemory(const CfiMemoryParams &p)
Definition: cfi_mem.cc:156
gem5::memory::CfiMemory::dequeueEvent
EventFunctionWrapper dequeueEvent
Definition: cfi_mem.hh:317
gem5::Packet::popLabel
void popLabel()
Pop label for PrintReq (safe to call unconditionally).
Definition: packet.hh:1471
gem5::Packet::needsResponse
bool needsResponse() const
Definition: packet.hh:608
gem5::memory::CfiMemory::CfiCommand::READ_CFI_QUERY
@ READ_CFI_QUERY
gem5::Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:147
gem5::memory::CfiMemory::ProgramBuffer::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: cfi_mem.cc:141
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:568
gem5::MemBackdoor
Definition: backdoor.hh:41
gem5::ResponsePort::sendRangeChange
void sendRangeChange() const
Called by the owner to send a range change.
Definition: port.hh:364
gem5::memory::CfiMemory::recvTimingReq
bool recvTimingReq(PacketPtr pkt)
Definition: cfi_mem.cc:287
gem5::memory::CfiMemory::STATUS_PROGRAM_LOCK_BIT
static const uint8_t STATUS_PROGRAM_LOCK_BIT
Definition: cfi_mem.hh:101
gem5::Drainable::signalDrainDone
void signalDrainDone() const
Signal that an object is drained.
Definition: drain.hh:305
gem5::memory::AbstractMemory::size
uint64_t size() const
Get the memory size.
Definition: abstract_mem.hh:308
gem5::ResponsePort
A ResponsePort is a specialization of a port.
Definition: port.hh:331
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:214
gem5::memory::CfiMemory::readDeviceID
uint64_t readDeviceID(Addr flash_address) const
Helper function to read the device identifier after the read state machine is put in the CfiCommand::...
Definition: cfi_mem.cc:668
gem5::memory::CfiMemory::BlockData::locked
std::vector< bool > locked
Definition: cfi_mem.hh:161
gem5::memory::CfiMemory::BlockData::unlock
void unlock(Addr block_address)
Unlock the block pointed by the block_address parameter.
Definition: cfi_mem.cc:70
gem5::memory::CfiMemory::recvAtomic
Tick recvAtomic(PacketPtr pkt)
Definition: cfi_mem.cc:241
gem5::MemBackdoor::ptr
uint8_t * ptr() const
Definition: backdoor.hh:62
gem5::memory::CfiMemory::release
void release()
Release the memory after being busy and send a retry if a request was rejected in the meanwhile.
Definition: cfi_mem.cc:370
gem5::memory::CfiMemory::MemoryPort::recvRespRetry
void recvRespRetry() override
Called by the peer if sendTimingResp was called on this protocol (causing recvTimingResp to be called...
Definition: cfi_mem.cc:511
gem5::Port
Ports are used to interface objects to each other.
Definition: port.hh:61
gem5::memory::CfiMemory::MemoryPort::recvAtomic
Tick recvAtomic(PacketPtr pkt) override
Receive an atomic request packet from the peer.
Definition: cfi_mem.cc:479
gem5::memory::AbstractMemory::backdoor
MemBackdoor backdoor
Definition: abstract_mem.hh:121
SERIALIZE_CONTAINER
#define SERIALIZE_CONTAINER(member)
Definition: serialize.hh:626
gem5::memory::CfiMemory::CfiCommand::ERASE_BLOCK_SETUP
@ ERASE_BLOCK_SETUP
gem5::Packet::makeResponse
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
Definition: packet.hh:1062
gem5::memory::CfiMemory::CfiCommand::READ_STATUS_REG
@ READ_STATUS_REG
gem5::Clocked::tick
Tick tick
Definition: clocked_object.hh:68
gem5::memory::CfiMemory::deviceID
const uint16_t deviceID
Definition: cfi_mem.hh:335
gem5::memory::CfiMemory::vendorID
const uint16_t vendorID
Definition: cfi_mem.hh:334
gem5::Packet::trySatisfyFunctional
bool trySatisfyFunctional(PacketPtr other)
Check a functional request against a memory value stored in another packet (i.e.
Definition: packet.hh:1399
gem5::memory::CfiMemory::cfiQueryTable
uint8_t cfiQueryTable[61]
Definition: cfi_mem.hh:348
cfi_mem.hh
gem5::memory::CfiMemory::BlockData::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: cfi_mem.cc:76
gem5::CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:66
mem
bool_vector8 mem[]
Definition: reset_stim.h:43
gem5::memory::CfiMemory::BlockData::isLocked
bool isLocked(Addr block_address) const
Return true if the block pointed by the block_address parameter is locked.
Definition: cfi_mem.cc:58
UNSERIALIZE_ENUM
#define UNSERIALIZE_ENUM(scalar)
Definition: serialize.hh:598
gem5::memory::CfiMemory::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: cfi_mem.cc:454
trace.hh
std::list< AddrRange >
gem5::Packet::getAddr
Addr getAddr() const
Definition: packet.hh:807
intmath.hh
gem5::memory::CfiMemory::latency
const Tick latency
Latency from that a request is accepted until the response is ready to be sent.
Definition: cfi_mem.hh:264
gem5::memory::CfiMemory::DeferredPacket::pkt
const PacketPtr pkt
Definition: cfi_mem.hh:232
gem5::memory::AbstractMemory::access
void access(PacketPtr pkt)
Perform an untimed memory access and update all the state (e.g.
Definition: abstract_mem.cc:380
gem5
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
Definition: gpu_translation_state.hh:37
gem5::MemBackdoorReq
Definition: backdoor.hh:129
gem5::Packet::setUintX
void setUintX(uint64_t w, ByteOrder endian)
Set the value in the word w after truncating it to the length of the packet and then byteswapping it ...
Definition: packet.cc:361
gem5::random_mt
Random random_mt
Definition: random.cc:99
gem5::memory::CfiMemory::CfiCommand
CfiCommand
Definition: cfi_mem.hh:74
gem5::memory::CfiMemory::statusRegister
uint8_t statusRegister
Definition: cfi_mem.hh:342
gem5::memory::CfiMemory::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: cfi_mem.cc:442
gem5::memory::CfiMemory::CfiCommand::AMD_RESET
@ AMD_RESET
gem5::memory::CfiMemory::recvFunctional
void recvFunctional(PacketPtr pkt)
Definition: cfi_mem.cc:261
gem5::Packet::isResponse
bool isResponse() const
Definition: packet.hh:598
gem5::Packet::getSize
unsigned getSize() const
Definition: packet.hh:817
gem5::DrainState::Draining
@ Draining
Draining buffers pending serialization/handover.
gem5::Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:458
gem5::memory::CfiMemory::bankWidth
const uint16_t bankWidth
Definition: cfi_mem.hh:336
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:188
gem5::memory::CfiMemory::BlockData::lock
void lock(Addr block_address)
Lock the block pointed by the block_address parameter.
Definition: cfi_mem.cc:64
gem5::Packet::getPtr
T * getPtr()
get a pointer to the data ptr.
Definition: packet.hh:1225
gem5::memory::CfiMemory::handleCommand
void handleCommand(CfiCommand command)
Service a new command issued to the flash device.
Definition: cfi_mem.cc:685
gem5::memory::CfiMemory::retryResp
bool retryResp
Remember if we failed to send a response and are awaiting a retry.
Definition: cfi_mem.hh:301

Generated on Sun Jul 30 2023 01:56:58 for gem5 by doxygen 1.8.17