ruby: Simplify RubyPort flow control and routing

This patch simplfies the retry logic in the RubyPort, avoiding
redundant attributes, and enforcing more stringent checks on the
interactions with the normal ports. The patch also simplifies the
routing done by the RubyPort, using the port identifiers instead of a
heavy-weight sender state.

The patch also fixes a bug in the sending of responses from PIO
ports. Previously these responses bypassed the queue in the queued
port, and ignored the return value, potentially leading to response
packets being lost.

Committed by: Nilay Vaish <nilay@cs.wisc.edu>
This commit is contained in:
Andreas Hansson 2014-02-23 19:16:16 -06:00
parent 6aafd5cb3f
commit 5755fff998
4 changed files with 73 additions and 93 deletions

View file

@ -78,8 +78,8 @@ class QueuedSlavePort : public SlavePort
* QueuePort constructor. * QueuePort constructor.
*/ */
QueuedSlavePort(const std::string& name, MemObject* owner, QueuedSlavePort(const std::string& name, MemObject* owner,
SlavePacketQueue &queue) : SlavePacketQueue &queue, PortID id = InvalidPortID) :
SlavePort(name, owner), queue(queue) SlavePort(name, owner, id), queue(queue)
{ } { }
virtual ~QueuedSlavePort() { } virtual ~QueuedSlavePort() { }

View file

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2012 ARM Limited * Copyright (c) 2012-2013 ARM Limited
* All rights reserved. * All rights reserved.
* *
* The license below extends only to copyright in the software and shall * The license below extends only to copyright in the software and shall
@ -52,16 +52,17 @@ RubyPort::RubyPort(const Params *p)
: MemObject(p), m_version(p->version), m_controller(NULL), : MemObject(p), m_version(p->version), m_controller(NULL),
m_mandatory_q_ptr(NULL), m_mandatory_q_ptr(NULL),
pio_port(csprintf("%s-pio-port", name()), this), pio_port(csprintf("%s-pio-port", name()), this),
m_usingRubyTester(p->using_ruby_tester), m_request_cnt(0), m_usingRubyTester(p->using_ruby_tester),
drainManager(NULL), ruby_system(p->ruby_system), system(p->system), drainManager(NULL), ruby_system(p->ruby_system), system(p->system),
waitingOnSequencer(false), access_phys_mem(p->access_phys_mem) access_phys_mem(p->access_phys_mem)
{ {
assert(m_version != -1); assert(m_version != -1);
// create the slave ports based on the number of connected ports // create the slave ports based on the number of connected ports
for (size_t i = 0; i < p->port_slave_connection_count; ++i) { for (size_t i = 0; i < p->port_slave_connection_count; ++i) {
slave_ports.push_back(new M5Port(csprintf("%s-slave%d", name(), i), slave_ports.push_back(new M5Port(csprintf("%s-slave%d", name(), i),
this, ruby_system, access_phys_mem)); this, ruby_system,
access_phys_mem, i));
} }
// create the master ports based on the number of connected ports // create the master ports based on the number of connected ports
@ -119,16 +120,17 @@ RubyPort::getSlavePort(const std::string &if_name, PortID idx)
RubyPort::PioPort::PioPort(const std::string &_name, RubyPort::PioPort::PioPort(const std::string &_name,
RubyPort *_port) RubyPort *_port)
: QueuedMasterPort(_name, _port, queue), queue(*_port, *this) : QueuedMasterPort(_name, _port, queue), queue(*_port, *this),
ruby_port(_port)
{ {
DPRINTF(RubyPort, "creating master port on ruby sequencer %s\n", _name); DPRINTF(RubyPort, "creating master port on ruby sequencer %s\n", _name);
} }
RubyPort::M5Port::M5Port(const std::string &_name, RubyPort *_port, RubyPort::M5Port::M5Port(const std::string &_name, RubyPort *_port,
RubySystem *_system, bool _access_phys_mem) RubySystem *_system, bool _access_phys_mem, PortID id)
: QueuedSlavePort(_name, _port, queue), queue(*_port, *this), : QueuedSlavePort(_name, _port, queue, id), queue(*_port, *this),
ruby_port(_port), ruby_system(_system), ruby_port(_port), ruby_system(_system),
_onRetryList(false), access_phys_mem(_access_phys_mem) access_phys_mem(_access_phys_mem)
{ {
DPRINTF(RubyPort, "creating slave port on ruby sequencer %s\n", _name); DPRINTF(RubyPort, "creating slave port on ruby sequencer %s\n", _name);
} }
@ -140,22 +142,23 @@ RubyPort::M5Port::recvAtomic(PacketPtr pkt)
return 0; return 0;
} }
bool bool
RubyPort::PioPort::recvTimingResp(PacketPtr pkt) RubyPort::recvTimingResp(PacketPtr pkt, PortID master_port_id)
{ {
// got a response from a device
assert(pkt->isResponse());
// In FS mode, ruby memory will receive pio responses from devices // In FS mode, ruby memory will receive pio responses from devices
// and it must forward these responses back to the particular CPU. // and it must forward these responses back to the particular CPU.
DPRINTF(RubyPort, "Pio response for address %#x\n", pkt->getAddr()); DPRINTF(RubyPort, "Pio response for address %#x, going to %d\n",
pkt->getAddr(), pkt->getDest());
// First we must retrieve the request port from the sender State // Retrieve the port from the destination field
RubyPort::SenderState *senderState = assert(pkt->getDest() < slave_ports.size());
safe_cast<RubyPort::SenderState *>(pkt->popSenderState());
M5Port *port = senderState->port;
assert(port != NULL);
delete senderState;
port->sendTimingResp(pkt); // attempt to send the response in the next cycle
slave_ports[pkt->getDest()]->schedTimingResp(pkt, curTick() +
g_system_ptr->clockPeriod());
return true; return true;
} }
@ -164,16 +167,14 @@ bool
RubyPort::M5Port::recvTimingReq(PacketPtr pkt) RubyPort::M5Port::recvTimingReq(PacketPtr pkt)
{ {
DPRINTF(RubyPort, DPRINTF(RubyPort,
"Timing access caught for address %#x\n", pkt->getAddr()); "Timing access for address %#x on port %d\n", pkt->getAddr(),
id);
//dsm: based on SimpleTimingPort::recvTimingReq(pkt);
if (pkt->memInhibitAsserted()) if (pkt->memInhibitAsserted())
panic("RubyPort should never see an inhibited request\n"); panic("RubyPort should never see an inhibited request\n");
// Save the port in the sender state object to be used later to // Save the port id to be used later to route the response
// route the response pkt->setSrc(id);
pkt->pushSenderState(new SenderState(this));
// Check for pio requests and directly send them to the dedicated // Check for pio requests and directly send them to the dedicated
// pio port. // pio port.
@ -196,10 +197,11 @@ RubyPort::M5Port::recvTimingReq(PacketPtr pkt)
RequestStatus requestStatus = ruby_port->makeRequest(pkt); RequestStatus requestStatus = ruby_port->makeRequest(pkt);
// If the request successfully issued then we should return true. // If the request successfully issued then we should return true.
// Otherwise, we need to delete the senderStatus we just created and return // Otherwise, we need to tell the port to retry at a later point
// false. // and return false.
if (requestStatus == RequestStatus_Issued) { if (requestStatus == RequestStatus_Issued) {
DPRINTF(RubyPort, "Request %#x issued\n", pkt->getAddr()); DPRINTF(RubyPort, "Request %s 0x%x issued\n", pkt->cmdString(),
pkt->getAddr());
return true; return true;
} }
@ -215,9 +217,6 @@ RubyPort::M5Port::recvTimingReq(PacketPtr pkt)
"Request for address %#x did not issue because %s\n", "Request for address %#x did not issue because %s\n",
pkt->getAddr(), RequestStatus_to_string(requestStatus)); pkt->getAddr(), RequestStatus_to_string(requestStatus));
SenderState* senderState = safe_cast<SenderState*>(pkt->senderState);
pkt->senderState = senderState->predecessor;
delete senderState;
return false; return false;
} }
@ -284,23 +283,25 @@ RubyPort::M5Port::recvFunctional(PacketPtr pkt)
void void
RubyPort::ruby_hit_callback(PacketPtr pkt) RubyPort::ruby_hit_callback(PacketPtr pkt)
{ {
// Retrieve the request port from the sender State DPRINTF(RubyPort, "Hit callback for %s 0x%x\n", pkt->cmdString(),
RubyPort::SenderState *senderState = pkt->getAddr());
safe_cast<RubyPort::SenderState *>(pkt->senderState);
M5Port *port = senderState->port;
assert(port != NULL);
// pop the sender state from the packet // The packet was destined for memory and has not yet been turned
pkt->senderState = senderState->predecessor; // into a response
delete senderState; assert(system->isMemAddr(pkt->getAddr()));
assert(pkt->isRequest());
port->hitCallback(pkt); // As it has not yet been turned around, the source field tells us
// which port it came from.
assert(pkt->getSrc() < slave_ports.size());
slave_ports[pkt->getSrc()]->hitCallback(pkt);
// //
// If we had to stall the M5Ports, wake them up because the sequencer // If we had to stall the M5Ports, wake them up because the sequencer
// likely has free resources now. // likely has free resources now.
// //
if (waitingOnSequencer) { if (!retryList.empty()) {
// //
// Record the current list of ports to retry on a temporary list before // Record the current list of ports to retry on a temporary list before
// calling sendRetry on those ports. sendRetry will cause an // calling sendRetry on those ports. sendRetry will cause an
@ -308,17 +309,14 @@ RubyPort::ruby_hit_callback(PacketPtr pkt)
// list. Therefore we want to clear the retryList before calling // list. Therefore we want to clear the retryList before calling
// sendRetry. // sendRetry.
// //
std::list<M5Port*> curRetryList(retryList); std::vector<M5Port*> curRetryList(retryList);
retryList.clear(); retryList.clear();
waitingOnSequencer = false;
for (auto i = curRetryList.begin(); i != curRetryList.end(); ++i) {
for (std::list<M5Port*>::iterator i = curRetryList.begin();
i != curRetryList.end(); ++i) {
DPRINTF(RubyPort, DPRINTF(RubyPort,
"Sequencer may now be free. SendRetry to port %s\n", "Sequencer may now be free. SendRetry to port %s\n",
(*i)->name()); (*i)->name());
(*i)->onRetryList(false);
(*i)->sendRetry(); (*i)->sendRetry();
} }
} }
@ -471,7 +469,7 @@ RubyPort::M5Port::getAddrRanges() const
} }
bool bool
RubyPort::M5Port::isPhysMemAddress(Addr addr) RubyPort::M5Port::isPhysMemAddress(Addr addr) const
{ {
return ruby_port->system->isMemAddr(addr); return ruby_port->system->isMemAddr(addr);
} }

View file

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2012 ARM Limited * Copyright (c) 2012-2013 ARM Limited
* All rights reserved. * All rights reserved.
* *
* The license below extends only to copyright in the software and shall * The license below extends only to copyright in the software and shall
@ -65,54 +65,37 @@ class RubyPort : public MemObject
SlavePacketQueue queue; SlavePacketQueue queue;
RubyPort *ruby_port; RubyPort *ruby_port;
RubySystem* ruby_system; RubySystem* ruby_system;
bool _onRetryList;
bool access_phys_mem; bool access_phys_mem;
public: public:
M5Port(const std::string &_name, RubyPort *_port, M5Port(const std::string &_name, RubyPort *_port,
RubySystem*_system, bool _access_phys_mem); RubySystem*_system, bool _access_phys_mem, PortID id);
void hitCallback(PacketPtr pkt); void hitCallback(PacketPtr pkt);
void evictionCallback(const Address& address); void evictionCallback(const Address& address);
bool onRetryList()
{ return _onRetryList; }
void onRetryList(bool newVal)
{ _onRetryList = newVal; }
protected: protected:
virtual bool recvTimingReq(PacketPtr pkt); bool recvTimingReq(PacketPtr pkt);
virtual Tick recvAtomic(PacketPtr pkt); Tick recvAtomic(PacketPtr pkt);
virtual void recvFunctional(PacketPtr pkt); void recvFunctional(PacketPtr pkt);
virtual AddrRangeList getAddrRanges() const; AddrRangeList getAddrRanges() const;
private: private:
bool isPhysMemAddress(Addr addr); bool isPhysMemAddress(Addr addr) const;
}; };
friend class M5Port;
class PioPort : public QueuedMasterPort class PioPort : public QueuedMasterPort
{ {
private: private:
MasterPacketQueue queue; MasterPacketQueue queue;
RubyPort *ruby_port;
public: public:
PioPort(const std::string &_name, RubyPort *_port); PioPort(const std::string &_name, RubyPort *_port);
protected: protected:
virtual bool recvTimingResp(PacketPtr pkt); bool recvTimingResp(PacketPtr pkt)
}; { return ruby_port->recvTimingResp(pkt, id); }
friend class PioPort;
struct SenderState : public Packet::SenderState
{
M5Port* port;
SenderState(M5Port* _port) : port(_port)
{}
}; };
typedef RubyPortParams Params; typedef RubyPortParams Params;
@ -145,6 +128,16 @@ class RubyPort : public MemObject
void testDrainComplete(); void testDrainComplete();
void ruby_eviction_callback(const Address& address); void ruby_eviction_callback(const Address& address);
/**
* Called by the PIO port when receiving a timing response.
*
* @param pkt Response packet
* @param master_port_id Port id of the PIO port
*
* @return Whether successfully sent
*/
bool recvTimingResp(PacketPtr pkt, PortID master_port_id);
uint32_t m_version; uint32_t m_version;
AbstractController* m_controller; AbstractController* m_controller;
MessageBuffer* m_mandatory_q_ptr; MessageBuffer* m_mandatory_q_ptr;
@ -154,18 +147,13 @@ class RubyPort : public MemObject
private: private:
void addToRetryList(M5Port * port) void addToRetryList(M5Port * port)
{ {
if (!port->onRetryList()) { assert(std::find(retryList.begin(), retryList.end(), port) ==
port->onRetryList(true); retryList.end());
retryList.push_back(port); retryList.push_back(port);
waitingOnSequencer = true;
}
} }
unsigned int getChildDrainCount(DrainManager *dm); unsigned int getChildDrainCount(DrainManager *dm);
uint16_t m_port_id;
uint64_t m_request_cnt;
/** Vector of M5 Ports attached to this Ruby port. */ /** Vector of M5 Ports attached to this Ruby port. */
typedef std::vector<M5Port*>::iterator CpuPortIter; typedef std::vector<M5Port*>::iterator CpuPortIter;
std::vector<M5Port*> slave_ports; std::vector<M5Port*> slave_ports;
@ -180,9 +168,8 @@ class RubyPort : public MemObject
// Based on similar code in the M5 bus. Stores pointers to those ports // Based on similar code in the M5 bus. Stores pointers to those ports
// that should be called when the Sequencer becomes available after a stall. // that should be called when the Sequencer becomes available after a stall.
// //
std::list<M5Port*> retryList; std::vector<M5Port*> retryList;
bool waitingOnSequencer;
bool access_phys_mem; bool access_phys_mem;
}; };

View file

@ -553,15 +553,10 @@ Sequencer::hitCallback(SequencerRequest* srequest, DataBlock& data,
// If using the RubyTester, update the RubyTester sender state's // If using the RubyTester, update the RubyTester sender state's
// subBlock with the recieved data. The tester will later access // subBlock with the recieved data. The tester will later access
// this state. // this state.
// Note: RubyPort will access it's sender state before the
// RubyTester.
if (m_usingRubyTester) { if (m_usingRubyTester) {
RubyPort::SenderState *reqSenderState =
safe_cast<RubyPort::SenderState*>(pkt->senderState);
// @todo This is a dangerous assumption on nothing else
// modifying the senderState
RubyTester::SenderState* testerSenderState = RubyTester::SenderState* testerSenderState =
safe_cast<RubyTester::SenderState*>(reqSenderState->predecessor); pkt->findNextSenderState<RubyTester::SenderState>();
assert(testerSenderState);
testerSenderState->subBlock.mergeFrom(data); testerSenderState->subBlock.mergeFrom(data);
} }