50 #include "debug/AddrRanges.hh" 51 #include "debug/Drain.hh" 52 #include "debug/XBar.hh" 56 frontendLatency(p->frontend_latency),
57 forwardLatency(p->forward_latency),
58 responseLatency(p->response_latency),
60 gotAddrRanges(p->port_default_connection_count +
61 p->port_master_connection_count, false),
63 useDefaultRange(p->use_default_range),
65 transDist(this,
"trans_dist",
"Transaction distribution"),
66 pktCount(this,
"pkt_count",
67 "Packet count per connected master and slave (bytes)"),
68 pktSize(this,
"pkt_size",
69 "Cumulative packet size per connected master and slave (bytes)")
85 if (if_name ==
"master" && idx <
masterPorts.size()) {
88 }
else if (if_name ==
"default") {
90 }
else if (if_name ==
"slave" && idx <
slavePorts.size()) {
118 "Encountered header delay exceeding 1 us\n");
136 template <
typename SrcType,
typename DstType>
138 const std::string& _name) :
140 port(_port), xbar(_xbar), _name(xbar.
name() +
"." + _name), state(IDLE),
141 waitingForPeer(NULL), releaseEvent([this]{ releaseLayer(); },
name()),
142 ADD_STAT(occupancy,
"Layer occupancy (ticks)"),
143 ADD_STAT(utilization,
"Layer utilization (%)")
152 utilization = 100 * occupancy /
simTicks;
155 template <
typename SrcType,
typename DstType>
162 assert(state == BUSY);
166 xbar.schedule(releaseEvent, until);
169 occupancy += until -
curTick();
171 DPRINTF(
BaseXBar,
"The crossbar layer is now busy from tick %d to %d\n",
175 template <
typename SrcType,
typename DstType>
188 if (state == BUSY || waitingForPeer != NULL) {
190 assert(std::find(waitingForLayer.begin(), waitingForLayer.end(),
191 src_port) == waitingForLayer.end());
197 waitingForLayer.push_back(src_port);
206 template <
typename SrcType,
typename DstType>
212 assert(state == BUSY);
215 occupyLayer(busy_time);
218 template <
typename SrcType,
typename DstType>
225 assert(waitingForPeer == NULL);
230 waitingForPeer = src_port;
234 assert(state == BUSY);
237 occupyLayer(busy_time);
240 template <
typename SrcType,
typename DstType>
245 assert(state == BUSY);
246 assert(!releaseEvent.scheduled());
252 if (!waitingForLayer.empty()) {
255 if (waitingForPeer == NULL)
258 DPRINTF(Drain,
"Crossbar done draining, signaling drain manager\n");
264 template <
typename SrcType,
typename DstType>
269 assert(!waitingForLayer.empty());
272 assert(state == IDLE);
279 SrcType* retryingPort = waitingForLayer.front();
280 waitingForLayer.pop_front();
284 sendRetry(retryingPort);
289 if (state == RETRY) {
295 occupyLayer(xbar.clockEdge());
299 template <
typename SrcType,
typename DstType>
305 assert(waitingForPeer != NULL);
310 waitingForLayer.push_front(waitingForPeer);
313 waitingForPeer = NULL;
320 assert(state == BUSY);
329 assert(gotAllAddrRanges);
332 auto i = portMap.contains(addr_range);
333 if (
i != portMap.end()) {
338 if (useDefaultRange) {
339 if (addr_range.
isSubset(defaultRange)) {
340 DPRINTF(AddrRanges,
" found addr %s on default\n",
342 return defaultPortID;
345 DPRINTF(AddrRanges,
"Unable to find destination for %s, " 346 "will use default port\n", addr_range.
to_string());
347 return defaultPortID;
352 fatal(
"Unable to find destination for %s on %s\n", addr_range.
to_string(),
360 DPRINTF(AddrRanges,
"Received range change from slave port %s\n",
361 masterPorts[master_port_id]->getPeer());
365 gotAddrRanges[master_port_id] =
true;
368 if (!gotAllAddrRanges) {
371 gotAllAddrRanges =
true;
373 while (gotAllAddrRanges && r != gotAddrRanges.end()) {
374 gotAllAddrRanges &= *r++;
376 if (gotAllAddrRanges)
377 DPRINTF(AddrRanges,
"Got address ranges from all slaves\n");
384 if (master_port_id == defaultPortID) {
388 if (useDefaultRange) {
389 AddrRangeList ranges = masterPorts[master_port_id]->getAddrRanges();
391 if (ranges.size() != 1)
392 fatal(
"Crossbar %s may only have a single default range",
395 defaultRange = ranges.front();
400 if (gotAddrRanges[master_port_id]) {
401 for (
auto p = portMap.begin();
p != portMap.end(); ) {
402 if (
p->second == master_port_id)
411 AddrRangeList ranges = masterPorts[master_port_id]->getAddrRanges();
413 for (
const auto&
r: ranges) {
414 DPRINTF(AddrRanges,
"Adding range %s for id %d\n",
415 r.to_string(), master_port_id);
416 if (portMap.insert(
r, master_port_id) == portMap.end()) {
417 PortID conflict_id = portMap.intersects(
r)->second;
418 fatal(
"%s has two ports responding within range " 422 masterPorts[master_port_id]->getPeer(),
423 masterPorts[conflict_id]->getPeer());
431 if (gotAllAddrRanges) {
432 DPRINTF(AddrRanges,
"Aggregating address ranges\n");
436 if (useDefaultRange) {
437 if (!gotAddrRanges[defaultPortID])
438 fatal(
"Crossbar %s uses default range, but none provided",
441 xbarRanges.push_back(defaultRange);
442 DPRINTF(AddrRanges,
"-- Adding default %s\n",
443 defaultRange.to_string());
449 for (
const auto&
r: portMap) {
451 if (
r.first.interleaved()) {
455 if (!intlv_ranges.empty() &&
456 !intlv_ranges.back().mergesWith(
r.first)) {
457 DPRINTF(AddrRanges,
"-- Merging range from %d ranges\n",
458 intlv_ranges.size());
461 if (!(useDefaultRange &&
462 merged_range.isSubset(defaultRange))) {
463 xbarRanges.push_back(merged_range);
464 DPRINTF(AddrRanges,
"-- Adding merged range %s\n",
465 merged_range.to_string());
467 intlv_ranges.clear();
469 intlv_ranges.push_back(
r.first);
472 if (!(useDefaultRange &&
473 r.first.isSubset(defaultRange))) {
474 xbarRanges.push_back(
r.first);
475 DPRINTF(AddrRanges,
"-- Adding range %s\n",
476 r.first.to_string());
483 if (!intlv_ranges.empty()) {
484 DPRINTF(AddrRanges,
"-- Merging range from %d ranges\n",
485 intlv_ranges.size());
487 if (!(useDefaultRange && merged_range.isSubset(defaultRange))) {
488 xbarRanges.push_back(merged_range);
489 DPRINTF(AddrRanges,
"-- Adding merged range %s\n",
490 merged_range.to_string());
498 if (useDefaultRange) {
499 for (
const auto&
r: xbarRanges) {
502 if (
r.intersects(defaultRange) &&
503 !
r.isSubset(defaultRange))
504 fatal(
"Range %s intersects the " \
505 "default range of %s but is not a " \
506 "subset\n",
r.to_string(),
name());
512 for (
const auto&
s: slavePorts)
513 s->sendRangeChange();
523 assert(gotAllAddrRanges);
530 DPRINTF(AddrRanges,
"Received address range request\n");
540 using namespace Stats;
549 const std::string &cstr = cmd.
toString();
550 transDist.subname(
i, cstr);
554 .init(slavePorts.size(), masterPorts.size())
558 .init(slavePorts.size(), masterPorts.size())
567 for (
int i = 0;
i < slavePorts.size();
i++) {
568 pktCount.subname(
i, slavePorts[
i]->getPeer().
name());
569 pktSize.subname(
i, slavePorts[
i]->getPeer().
name());
570 for (
int j = 0;
j < masterPorts.size();
j++) {
571 pktCount.ysubname(
j, masterPorts[
j]->getPeer().
name());
572 pktSize.ysubname(
j, masterPorts[
j]->getPeer().
name());
577 template <
typename SrcType,
typename DstType>
585 DPRINTF(Drain,
"Crossbar not drained\n");
Ports are used to interface objects to each other.
void recvRetry()
Handle a retry from a neighbouring module.
#define fatal(...)
This implements a cprintf based fatal() function.
const std::string & name()
const PortID InvalidPortID
const FlagsType nonan
Don't print if this is NAN.
const std::string & toString() const
Return the string to a cmd given by idx.
PortID defaultPortID
Port that handles requests that don't match any of the interfaces.
bool isSubset(const AddrRange &r) const
Determine if this range is a subset of another range, i.e.
AddrRangeList getAddrRanges() const
Return the address ranges the crossbar is responsible for.
DrainState
Object drain/handover states.
The base crossbar contains the common elements of the non-coherent and coherent crossbar.
void regStats() override
Callback to set stat parameters.
void succeededTiming(Tick busy_time)
Deal with a destination port accepting a packet by potentially removing the source port from the retr...
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
Draining buffers pending serialization/handover.
Tick curTick()
The current simulated tick.
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
const uint32_t width
the width of the xbar in bytes
uint64_t Tick
Tick count type.
BaseXBar(const BaseXBarParams *p)
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
void calcPacketTiming(PacketPtr pkt, Tick header_delay)
Calculate the timing parameters for the packet.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
A function used to return the port associated with this object.
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
void failedTiming(SrcType *src_port, Tick busy_time)
Deal with a destination port not accepting a packet by potentially adding the source port to the retr...
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Tick clockEdge(Cycles cycles=Cycles(0)) const
Determine the tick when a cycle begins, by default the current one, but the argument also enables the...
const FlagsType total
Print the total.
std::vector< MasterPort * > masterPorts
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
virtual const std::string name() const
std::vector< QueuedSlavePort * > slavePorts
The master and slave ports of the crossbar.
void occupyLayer(Tick until)
void releaseLayer()
Release the layer after being occupied and return to an idle state where we proceed to send a retry t...
DrainState drain() override
Drain according to the normal semantics, so that the crossbar can tell the layer to drain...
PortID findPort(AddrRange addr_range)
Find which port connected to this crossbar (if any) should be given a packet with this address range...
T divCeil(const T &a, const U &b)
Layer(DstType &_port, BaseXBar &_xbar, const std::string &_name)
Create a layer and give it a name.
bool tryTiming(SrcType *src_port)
Determine if the layer accepts a packet from a specific port.
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
virtual void regStats()
Callback to set stat parameters.
std::string to_string() const
Get a string representation of the range.
const FlagsType nozero
Don't print if this is zero.
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
void retryWaiting()
Send a retry to the port at the head of waitingForLayer.
Declaration of an abstract crossbar base class.
virtual void recvRangeChange(PortID master_port_id)
Function called by the port when the crossbar is recieving a range change.