Go to the documentation of this file.
   45 #include "debug/Config.hh" 
   46 #include "debug/Drain.hh" 
   47 #include "debug/Ruby.hh" 
   48 #include "mem/ruby/protocol/AccessPermission.hh" 
   56       m_controller(NULL), m_mandatory_q_ptr(NULL),
 
   58       pioRequestPort(
csprintf(
"%s.pio-request-port", 
name()), this),
 
   59       pioResponsePort(
csprintf(
"%s.pio-response-port", 
name()), this),
 
   60       memRequestPort(
csprintf(
"%s.mem-request-port", 
name()), this),
 
   61       memResponsePort(
csprintf(
"%s-mem-response-port", 
name()), this,
 
   62                    p.ruby_system->getAccessBackingStore(), -1,
 
   64       gotAddrRanges(
p.port_interrupt_out_port_connection_count),
 
   65       m_isCPUSequencer(
p.is_cpu_sequencer)
 
   70     for (
size_t i = 0; 
i < 
p.port_in_ports_connection_count; ++
i) {
 
   72             (
"%s.response_ports%d", 
name(), 
i), 
this,
 
   73             p.ruby_system->getAccessBackingStore(),
 
   74             i, 
p.no_retry_on_stall));
 
   78     for (
size_t i = 0; 
i < 
p.port_interrupt_out_port_connection_count; ++
i) {
 
   80                     "%s.request_ports%d", 
name(), 
i), 
this));
 
   90         response_port->sendRangeChange();
 
   96     if (if_name == 
"mem_request_port") {
 
   98     } 
else if (if_name == 
"pio_request_port") {
 
  100     } 
else if (if_name == 
"mem_response_port") {
 
  102     } 
else if (if_name == 
"pio_response_port") {
 
  104     } 
else if (if_name == 
"interrupt_out_port") {
 
  108             panic(
"%s: unknown %s index (%d)\n", __func__, if_name, idx);
 
  112     } 
else if (if_name == 
"in_ports") {
 
  116             panic(
"%s: unknown %s index (%d)\n", __func__, if_name, idx);
 
  129       reqQueue(*_port, *this), snoopRespQueue(*_port, *this)
 
  138     DPRINTF(
RubyPort, 
"Created response pioport on sequencer %s\n", _name);
 
  144       reqQueue(*_port, *this), snoopRespQueue(*_port, *this)
 
  146     DPRINTF(
RubyPort, 
"Created request memport on ruby sequencer %s\n", _name);
 
  151                                      bool _access_backing_store, 
PortID id,
 
  152                                      bool _no_retry_on_stall)
 
  154       access_backing_store(_access_backing_store),
 
  155       no_retry_on_stall(_no_retry_on_stall)
 
  183     assert(port != NULL);
 
  205         for (
auto it = 
l.begin(); it != 
l.end(); ++it) {
 
  206             if (it->contains(pkt->
getAddr())) {
 
  209                 M5_VAR_USED 
bool success =
 
  216     panic(
"Should never reach here!\n");
 
  225         panic(
"Ruby supports atomic accesses only in noncaching mode\n");
 
  230         for (
auto it = 
l.begin(); it != 
l.end(); ++it) {
 
  231             if (it->contains(pkt->
getAddr())) {
 
  236     panic(
"Could not find address in Ruby PIO address ranges!\n");
 
  247         panic(
"RubyPort should never see request with the " 
  248               "cacheResponding flag set\n");
 
  252     if (pkt->
req->isCacheMaintenance()) {
 
  253         warn_once(
"Cache maintenance operations are not supported in Ruby.\n");
 
  255         schedTimingResp(pkt, 
curTick());
 
  261         if (!isPhysMemAddress(pkt)) {
 
  262             assert(!pkt->
req->isHTMCmd());
 
  265                     "pio address\n", pkt->
getAddr());
 
  284     RequestStatus requestStatus = ruby_port->
makeRequest(pkt);
 
  289     if (requestStatus == RequestStatus_Issued) {
 
  301                 "Request %s for address %#x did not issue because %s\n",
 
  303                 RequestStatus_to_string(requestStatus));
 
  317         panic(
"Ruby supports atomic accesses only in noncaching mode\n");
 
  323         if (!isPhysMemAddress(pkt)) {
 
  326                     "pio address\n", pkt->
getAddr());
 
  346                     pkt->
getAddr(), MachineType_Directory);
 
  349         rs->m_abstract_controls[
id.getType()][
id.getNum()];
 
  351     if (access_backing_store)
 
  352         rs->getPhysMem()->access(pkt);
 
  366     if (!no_retry_on_stall && !ruby_port->
onRetryList(
this)) {
 
  381     if (!isPhysMemAddress(pkt)) {
 
  383         assert(rp->pioRequestPort.isConnected());
 
  384         rp->pioRequestPort.sendFunctional(pkt);
 
  391     if (access_backing_store) {
 
  396         rs->getPhysMem()->functionalAccess(pkt);
 
  398         bool accessSucceeded = 
false;
 
  403             accessSucceeded = 
rs->functionalRead(pkt);
 
  405             accessSucceeded = 
rs->functionalWrite(pkt);
 
  413             fatal(
"Ruby functional %s failed for address %#x\n",
 
  429                 accessSucceeded ? 
"successful":
"failed");
 
  448     assert(port != NULL);
 
  473         for (
auto i = curRetryList.begin(); 
i != curRetryList.end(); ++
i) {
 
  475                     "Sequencer may now be free. SendRetry to port %s\n",
 
  477             (*i)->sendRetryReq();
 
  488         DPRINTF(Drain, 
"Drain count: %u\n", drainCount);
 
  489         if (drainCount == 0) {
 
  490             DPRINTF(Drain, 
"RubyPort done draining, signaling drain done\n");
 
  509         DPRINTF(Drain, 
"RubyPort not drained\n");
 
  523     bool accessPhysMem = access_backing_store;
 
  527             if (pkt->
req->getExtraData() != 0) {
 
  537                 accessPhysMem = 
false;
 
  551         accessPhysMem = 
false;
 
  554     if (pkt->
req->isKernel()) {
 
  555         accessPhysMem = 
false;
 
  556         needsResponse = 
true;
 
  570             rs->getPhysMem()->access(pkt);
 
  572             panic(
"Packet is in neither device nor system memory!");
 
  574     } 
else if (needsResponse) {
 
  584         schedTimingResp(pkt, 
curTick());
 
  600         ranges.splice(ranges.begin(),
 
  603     for (M5_VAR_USED 
const auto &
r : ranges)
 
  623     auto request = std::make_shared<Request>(
 
  633         if ((*p)->isSnooping()) {
 
  635             (*p)->sendTimingSnoopReq(&pkt);
 
  646         r.pioResponsePort.sendRangeChange();
 
  655         if (port->trySatisfyFunctional(func_pkt)) {
 
  
#define fatal(...)
This implements a cprintf based fatal() function.
Tick recvAtomic(PacketPtr pkt)
Receive an atomic request packet from the peer.
virtual bool isDeadlockEventScheduled() const =0
void ruby_hit_callback(PacketPtr pkt)
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
bool cacheResponding() const
static uint32_t getBlockSizeBytes()
Tick recvAtomic(PacketPtr pkt)
Addr makeLineAddress(Addr addr)
void convertLlToRead()
When ruby is in use, Ruby will monitor the cache line and the phys memory should treat LL ops as norm...
bool recvTimingResp(PacketPtr pkt)
Receive a timing response from the peer.
virtual MessageBuffer * getMandatoryQueue() const =0
Tick recvAtomic(PacketPtr pkt)
Receive an atomic request packet from the peer.
bool recvTimingReq(PacketPtr pkt)
Receive a timing request from the peer.
uint64_t Tick
Tick count type.
MachineID mapAddressToMachine(Addr addr, MachineType mtype) const
Map an address to the correct MachineID.
RubyPort(const Params &p)
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
DrainState drain() override
Draining is the process of clearing out the states of SimObjects.These are the SimObjects that are pa...
RequestPtr req
A pointer to the original request.
RequestorID requestorId() const
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
void recvFunctional(PacketPtr pkt)
Receive a functional request packet from the peer.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
AddrRangeList getAddrRanges() const
Get a list of the non-overlapping address ranges the owner is responsible for.
bool onRetryList(MemResponsePort *port)
void schedTimingResp(PacketPtr pkt, Tick when)
Schedule the sending of a timing response.
void convertScToWrite()
It has been determined that the SC packet should successfully update memory.
MemRequestPort(const std::string &_name, RubyPort *_port)
virtual RequestStatus makeRequest(PacketPtr pkt)=0
void schedTimingReq(PacketPtr pkt, Tick when)
Schedule the sending of a timing request.
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
MessageBuffer * m_mandatory_q_ptr
@ Drained
Buffers drained, ready for serialization/handover.
AbstractController * m_controller
DrainState
Object drain/handover states.
std::vector< MemResponsePort * > response_ports
void access(PacketPtr pkt)
Perform an untimed memory access and update all the state (e.g.
The QueuedRequestPort combines two queues, a request queue and a snoop response queue,...
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
RubyTester::SenderState SenderState
const std::string & cmdString() const
Return the string name of the cmd field (for debugging and tracing).
Ports are used to interface objects to each other.
bool needsResponse() const
MemResponsePort(const std::string &_name, RubyPort *_port, bool _access_backing_store, PortID id, bool _no_retry_on_stall)
bool suppressFuncError() const
bool bypassCaches() const
Should caches be bypassed?
void signalDrainDone() const
Signal that an object is drained.
A queued port is a port that has an infinite queue for outgoing packets and thus decouples the module...
MemRequestPort memRequestPort
MemResponsePort memResponsePort
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
const std::string name() const
Return port name (for DPRINTF).
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
const std::string & name()
bool htmTransactionFailedInCache() const
Returns whether or not this packet/request has returned from the cache hierarchy in a failed transact...
@ funcRequestorId
This requestor id is used for functional requests that don't come from a particular device.
virtual int functionalWrite(Packet *func_pkt)
PioResponsePort(const std::string &_name, RubyPort *_port)
DrainState drainState() const
Return the current drain state of an object.
PioRequestPort pioRequestPort
virtual const std::string name() const
void addToRetryList(MemResponsePort *port)
bool recvTimingResp(PacketPtr pkt)
Receive a timing response from the peer.
MemCmd cmd
The command field of the packet.
void pushSenderState(SenderState *sender_state)
Push a new sender state to the packet and make the current sender state the predecessor of the new on...
bool isDeviceMemAddr(PacketPtr pkt) const
Similar to isMemAddr but for devices.
void recvRangeChange()
Called to receive an address range change from the peer response port.
Cycles ticksToCycles(Tick t) const
std::vector< MemResponsePort * > retryList
std::vector< PioRequestPort * > request_ports
virtual void descheduleDeadlockEvent()=0
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
SenderState * popSenderState()
Pop the top of the state stack and return a pointer to it.
Addr getOffset(Addr addr)
void hitCallback(PacketPtr pkt)
bool isConnected() const
Is this port currently connected to a peer?
PioRequestPort(const std::string &_name, RubyPort *_port)
PioResponsePort pioResponsePort
virtual int outstandingCount() const =0
AbstractMemory * getDeviceMemory(RequestorID _id) const
Return a pointer to the device memory.
Tick curTick()
The universal simulation clock.
Tick sendAtomic(PacketPtr pkt)
Send an atomic request packet, where the data is moved and the state is updated in zero time,...
bool isPhysMemAddress(PacketPtr pkt) const
RubySystem * m_ruby_system
bool isMemAddr(Addr addr) const
Check if a physical address is within a range of a memory that is part of the global address map.
void setFunctionalResponseStatus(bool success)
std::string csprintf(const char *format, const Args &...args)
std::vector< MemResponsePort * >::iterator CpuPortIter
Vector of M5 Ports attached to this Ruby port.
void ruby_eviction_callback(Addr address)
@ Draining
Draining buffers pending serialization/handover.
bool recvTimingReq(PacketPtr pkt)
Receive a timing request from the peer.
#define panic(...)
This implements a cprintf based panic() function.
Generated on Tue Jun 22 2021 15:28:30 for gem5 by  doxygen 1.8.17