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