gem5  v20.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
xbar.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2011-2015, 2018-2019 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) 2006 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 
46 #include "mem/xbar.hh"
47 
48 #include "base/logging.hh"
49 #include "base/trace.hh"
50 #include "debug/AddrRanges.hh"
51 #include "debug/Drain.hh"
52 #include "debug/XBar.hh"
53 
54 BaseXBar::BaseXBar(const BaseXBarParams *p)
55  : ClockedObject(p),
56  frontendLatency(p->frontend_latency),
57  forwardLatency(p->forward_latency),
58  responseLatency(p->response_latency),
59  width(p->width),
60  gotAddrRanges(p->port_default_connection_count +
61  p->port_master_connection_count, false),
62  gotAllAddrRanges(false), defaultPortID(InvalidPortID),
63  useDefaultRange(p->use_default_range),
64 
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)")
70 {
71 }
72 
74 {
75  for (auto m: masterPorts)
76  delete m;
77 
78  for (auto s: slavePorts)
79  delete s;
80 }
81 
82 Port &
83 BaseXBar::getPort(const std::string &if_name, PortID idx)
84 {
85  if (if_name == "master" && idx < masterPorts.size()) {
86  // the master port index translates directly to the vector position
87  return *masterPorts[idx];
88  } else if (if_name == "default") {
89  return *masterPorts[defaultPortID];
90  } else if (if_name == "slave" && idx < slavePorts.size()) {
91  // the slave port index translates directly to the vector position
92  return *slavePorts[idx];
93  } else {
94  return ClockedObject::getPort(if_name, idx);
95  }
96 }
97 
98 void
100 {
101  // the crossbar will be called at a time that is not necessarily
102  // coinciding with its own clock, so start by determining how long
103  // until the next clock edge (could be zero)
104  Tick offset = clockEdge() - curTick();
105 
106  // the header delay depends on the path through the crossbar, and
107  // we therefore rely on the caller to provide the actual
108  // value
109  pkt->headerDelay += offset + header_delay;
110 
111  // note that we add the header delay to the existing value, and
112  // align it to the crossbar clock
113 
114  // do a quick sanity check to ensure the timings are not being
115  // ignored, note that this specific value may cause problems for
116  // slower interconnects
118  "Encountered header delay exceeding 1 us\n");
119 
120  if (pkt->hasData()) {
121  // the payloadDelay takes into account the relative time to
122  // deliver the payload of the packet, after the header delay,
123  // we take the maximum since the payload delay could already
124  // be longer than what this parcitular crossbar enforces.
125  pkt->payloadDelay = std::max<Tick>(pkt->payloadDelay,
126  divCeil(pkt->getSize(), width) *
127  clockPeriod());
128  }
129 
130  // the payload delay is not paying for the clock offset as that is
131  // already done using the header delay, and the payload delay is
132  // also used to determine how long the crossbar layer is busy and
133  // thus regulates throughput
134 }
135 
136 template <typename SrcType, typename DstType>
138  const std::string& _name) :
139  Stats::Group(&_xbar, _name.c_str()),
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 (%)")
144 {
145  occupancy
146  .flags(Stats::nozero);
147 
148  utilization
149  .precision(1)
150  .flags(Stats::nozero);
151 
152  utilization = 100 * occupancy / simTicks;
153 }
154 
155 template <typename SrcType, typename DstType>
157 {
158  // ensure the state is busy at this point, as the layer should
159  // transition from idle as soon as it has decided to forward the
160  // packet to prevent any follow-on calls to sendTiming seeing an
161  // unoccupied layer
162  assert(state == BUSY);
163 
164  // until should never be 0 as express snoops never occupy the layer
165  assert(until != 0);
166  xbar.schedule(releaseEvent, until);
167 
168  // account for the occupied ticks
169  occupancy += until - curTick();
170 
171  DPRINTF(BaseXBar, "The crossbar layer is now busy from tick %d to %d\n",
172  curTick(), until);
173 }
174 
175 template <typename SrcType, typename DstType>
176 bool
178 {
179  // if we are in the retry state, we will not see anything but the
180  // retrying port (or in the case of the snoop ports the snoop
181  // response port that mirrors the actual slave port) as we leave
182  // this state again in zero time if the peer does not immediately
183  // call the layer when receiving the retry
184 
185  // first we see if the layer is busy, next we check if the
186  // destination port is already engaged in a transaction waiting
187  // for a retry from the peer
188  if (state == BUSY || waitingForPeer != NULL) {
189  // the port should not be waiting already
190  assert(std::find(waitingForLayer.begin(), waitingForLayer.end(),
191  src_port) == waitingForLayer.end());
192 
193  // put the port at the end of the retry list waiting for the
194  // layer to be freed up (and in the case of a busy peer, for
195  // that transaction to go through, and then the layer to free
196  // up)
197  waitingForLayer.push_back(src_port);
198  return false;
199  }
200 
201  state = BUSY;
202 
203  return true;
204 }
205 
206 template <typename SrcType, typename DstType>
207 void
209 {
210  // we should have gone from idle or retry to busy in the tryTiming
211  // test
212  assert(state == BUSY);
213 
214  // occupy the layer accordingly
215  occupyLayer(busy_time);
216 }
217 
218 template <typename SrcType, typename DstType>
219 void
221  Tick busy_time)
222 {
223  // ensure no one got in between and tried to send something to
224  // this port
225  assert(waitingForPeer == NULL);
226 
227  // if the source port is the current retrying one or not, we have
228  // failed in forwarding and should track that we are now waiting
229  // for the peer to send a retry
230  waitingForPeer = src_port;
231 
232  // we should have gone from idle or retry to busy in the tryTiming
233  // test
234  assert(state == BUSY);
235 
236  // occupy the bus accordingly
237  occupyLayer(busy_time);
238 }
239 
240 template <typename SrcType, typename DstType>
241 void
243 {
244  // releasing the bus means we should now be idle
245  assert(state == BUSY);
246  assert(!releaseEvent.scheduled());
247 
248  // update the state
249  state = IDLE;
250 
251  // bus layer is now idle, so if someone is waiting we can retry
252  if (!waitingForLayer.empty()) {
253  // there is no point in sending a retry if someone is still
254  // waiting for the peer
255  if (waitingForPeer == NULL)
256  retryWaiting();
257  } else if (waitingForPeer == NULL && drainState() == DrainState::Draining) {
258  DPRINTF(Drain, "Crossbar done draining, signaling drain manager\n");
259  //If we weren't able to drain before, do it now.
260  signalDrainDone();
261  }
262 }
263 
264 template <typename SrcType, typename DstType>
265 void
267 {
268  // this should never be called with no one waiting
269  assert(!waitingForLayer.empty());
270 
271  // we always go to retrying from idle
272  assert(state == IDLE);
273 
274  // update the state
275  state = RETRY;
276 
277  // set the retrying port to the front of the retry list and pop it
278  // off the list
279  SrcType* retryingPort = waitingForLayer.front();
280  waitingForLayer.pop_front();
281 
282  // tell the port to retry, which in some cases ends up calling the
283  // layer again
284  sendRetry(retryingPort);
285 
286  // If the layer is still in the retry state, sendTiming wasn't
287  // called in zero time (e.g. the cache does this when a writeback
288  // is squashed)
289  if (state == RETRY) {
290  // update the state to busy and reset the retrying port, we
291  // have done our bit and sent the retry
292  state = BUSY;
293 
294  // occupy the crossbar layer until the next clock edge
295  occupyLayer(xbar.clockEdge());
296  }
297 }
298 
299 template <typename SrcType, typename DstType>
300 void
302 {
303  // we should never get a retry without having failed to forward
304  // something to this port
305  assert(waitingForPeer != NULL);
306 
307  // add the port where the failed packet originated to the front of
308  // the waiting ports for the layer, this allows us to call retry
309  // on the port immediately if the crossbar layer is idle
310  waitingForLayer.push_front(waitingForPeer);
311 
312  // we are no longer waiting for the peer
313  waitingForPeer = NULL;
314 
315  // if the layer is idle, retry this port straight away, if we
316  // are busy, then simply let the port wait for its turn
317  if (state == IDLE) {
318  retryWaiting();
319  } else {
320  assert(state == BUSY);
321  }
322 }
323 
324 PortID
326 {
327  // we should never see any address lookups before we've got the
328  // ranges of all connected slave modules
329  assert(gotAllAddrRanges);
330 
331  // Check the address map interval tree
332  auto i = portMap.contains(addr_range);
333  if (i != portMap.end()) {
334  return i->second;
335  }
336 
337  // Check if this matches the default range
338  if (useDefaultRange) {
339  if (addr_range.isSubset(defaultRange)) {
340  DPRINTF(AddrRanges, " found addr %s on default\n",
341  addr_range.to_string());
342  return defaultPortID;
343  }
344  } else if (defaultPortID != InvalidPortID) {
345  DPRINTF(AddrRanges, "Unable to find destination for %s, "
346  "will use default port\n", addr_range.to_string());
347  return defaultPortID;
348  }
349 
350  // we should use the range for the default port and it did not
351  // match, or the default port is not set
352  fatal("Unable to find destination for %s on %s\n", addr_range.to_string(),
353  name());
354 }
355 
357 void
359 {
360  DPRINTF(AddrRanges, "Received range change from slave port %s\n",
361  masterPorts[master_port_id]->getPeer());
362 
363  // remember that we got a range from this master port and thus the
364  // connected slave module
365  gotAddrRanges[master_port_id] = true;
366 
367  // update the global flag
368  if (!gotAllAddrRanges) {
369  // take a logical AND of all the ports and see if we got
370  // ranges from everyone
371  gotAllAddrRanges = true;
372  std::vector<bool>::const_iterator r = gotAddrRanges.begin();
373  while (gotAllAddrRanges && r != gotAddrRanges.end()) {
374  gotAllAddrRanges &= *r++;
375  }
376  if (gotAllAddrRanges)
377  DPRINTF(AddrRanges, "Got address ranges from all slaves\n");
378  }
379 
380  // note that we could get the range from the default port at any
381  // point in time, and we cannot assume that the default range is
382  // set before the other ones are, so we do additional checks once
383  // all ranges are provided
384  if (master_port_id == defaultPortID) {
385  // only update if we are indeed checking ranges for the
386  // default port since the port might not have a valid range
387  // otherwise
388  if (useDefaultRange) {
389  AddrRangeList ranges = masterPorts[master_port_id]->getAddrRanges();
390 
391  if (ranges.size() != 1)
392  fatal("Crossbar %s may only have a single default range",
393  name());
394 
395  defaultRange = ranges.front();
396  }
397  } else {
398  // the ports are allowed to update their address ranges
399  // dynamically, so remove any existing entries
400  if (gotAddrRanges[master_port_id]) {
401  for (auto p = portMap.begin(); p != portMap.end(); ) {
402  if (p->second == master_port_id)
403  // erasing invalidates the iterator, so advance it
404  // before the deletion takes place
405  portMap.erase(p++);
406  else
407  p++;
408  }
409  }
410 
411  AddrRangeList ranges = masterPorts[master_port_id]->getAddrRanges();
412 
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 "
419  "%s:\n\t%s\n\t%s\n",
420  name(),
421  r.to_string(),
422  masterPorts[master_port_id]->getPeer(),
423  masterPorts[conflict_id]->getPeer());
424  }
425  }
426  }
427 
428  // if we have received ranges from all our neighbouring slave
429  // modules, go ahead and tell our connected master modules in
430  // turn, this effectively assumes a tree structure of the system
431  if (gotAllAddrRanges) {
432  DPRINTF(AddrRanges, "Aggregating address ranges\n");
433  xbarRanges.clear();
434 
435  // start out with the default range
436  if (useDefaultRange) {
437  if (!gotAddrRanges[defaultPortID])
438  fatal("Crossbar %s uses default range, but none provided",
439  name());
440 
441  xbarRanges.push_back(defaultRange);
442  DPRINTF(AddrRanges, "-- Adding default %s\n",
443  defaultRange.to_string());
444  }
445 
446  // merge all interleaved ranges and add any range that is not
447  // a subset of the default range
448  std::vector<AddrRange> intlv_ranges;
449  for (const auto& r: portMap) {
450  // if the range is interleaved then save it for now
451  if (r.first.interleaved()) {
452  // if we already got interleaved ranges that are not
453  // part of the same range, then first do a merge
454  // before we add the new one
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());
459  AddrRange merged_range(intlv_ranges);
460  // next decide if we keep the merged range or not
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());
466  }
467  intlv_ranges.clear();
468  }
469  intlv_ranges.push_back(r.first);
470  } else {
471  // keep the current range if not a subset of the default
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());
477  }
478  }
479  }
480 
481  // if there is still interleaved ranges waiting to be merged,
482  // go ahead and do it
483  if (!intlv_ranges.empty()) {
484  DPRINTF(AddrRanges, "-- Merging range from %d ranges\n",
485  intlv_ranges.size());
486  AddrRange merged_range(intlv_ranges);
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());
491  }
492  }
493 
494  // also check that no range partially intersects with the
495  // default range, this has to be done after all ranges are set
496  // as there are no guarantees for when the default range is
497  // update with respect to the other ones
498  if (useDefaultRange) {
499  for (const auto& r: xbarRanges) {
500  // see if the new range is partially
501  // overlapping the default range
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());
507  }
508  }
509 
510  // tell all our neighbouring master ports that our address
511  // ranges have changed
512  for (const auto& s: slavePorts)
513  s->sendRangeChange();
514  }
515 }
516 
519 {
520  // we should never be asked without first having sent a range
521  // change, and the latter is only done once we have all the ranges
522  // of the connected devices
523  assert(gotAllAddrRanges);
524 
525  // at the moment, this never happens, as there are no cycles in
526  // the range queries and no devices on the master side of a crossbar
527  // (CPU, cache, bridge etc) actually care about the ranges of the
528  // ports they are connected to
529 
530  DPRINTF(AddrRanges, "Received address range request\n");
531 
532  return xbarRanges;
533 }
534 
535 void
537 {
539 
540  using namespace Stats;
541 
542  transDist
543  .init(MemCmd::NUM_MEM_CMDS)
544  .flags(nozero);
545 
546  // get the string representation of the commands
547  for (int i = 0; i < MemCmd::NUM_MEM_CMDS; i++) {
548  MemCmd cmd(i);
549  const std::string &cstr = cmd.toString();
550  transDist.subname(i, cstr);
551  }
552 
553  pktCount
554  .init(slavePorts.size(), masterPorts.size())
555  .flags(total | nozero | nonan);
556 
557  pktSize
558  .init(slavePorts.size(), masterPorts.size())
559  .flags(total | nozero | nonan);
560 
561  // both the packet count and total size are two-dimensional
562  // vectors, indexed by slave port id and master port id, thus the
563  // neighbouring master and slave, they do not differentiate what
564  // came from the master and was forwarded to the slave (requests
565  // and snoop responses) and what came from the slave and was
566  // forwarded to the master (responses and snoop requests)
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());
573  }
574  }
575 }
576 
577 template <typename SrcType, typename DstType>
580 {
581  //We should check that we're not "doing" anything, and that noone is
582  //waiting. We might be idle but have someone waiting if the device we
583  //contacted for a retry didn't actually retry.
584  if (state != IDLE) {
585  DPRINTF(Drain, "Crossbar not drained\n");
586  return DrainState::Draining;
587  } else {
588  return DrainState::Drained;
589  }
590 }
591 
#define DPRINTF(x,...)
Definition: trace.hh:225
Ports are used to interface objects to each other.
Definition: port.hh:56
Tick us
microsecond
Definition: core.cc:64
void recvRetry()
Handle a retry from a neighbouring module.
Definition: xbar.cc:301
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
Definition: packet.hh:70
const std::string & name()
Definition: trace.cc:50
Bitfield< 7 > i
const PortID InvalidPortID
Definition: types.hh:236
Bitfield< 0 > m
const FlagsType nonan
Don&#39;t print if this is NAN.
Definition: info.hh:59
const std::string & toString() const
Return the string to a cmd given by idx.
Definition: packet.hh:231
PortID defaultPortID
Port that handles requests that don&#39;t match any of the interfaces.
Definition: xbar.hh:377
bool isSubset(const AddrRange &r) const
Determine if this range is a subset of another range, i.e.
Definition: addr_range.hh:379
AddrRangeList getAddrRanges() const
Return the address ranges the crossbar is responsible for.
Definition: xbar.cc:518
Bitfield< 23, 0 > offset
Definition: types.hh:152
Tick clockPeriod() const
DrainState
Object drain/handover states.
Definition: drain.hh:71
STL vector class.
Definition: stl.hh:37
The base crossbar contains the common elements of the non-coherent and coherent crossbar.
Definition: xbar.hh:68
void regStats() override
Callback to set stat parameters.
Definition: xbar.cc:536
void succeededTiming(Tick busy_time)
Deal with a destination port accepting a packet by potentially removing the source port from the retr...
Definition: xbar.cc:208
unsigned getSize() const
Definition: packet.hh:730
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
Definition: addr_range.hh:68
Draining buffers pending serialization/handover.
Tick curTick()
The current simulated tick.
Definition: core.hh:44
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
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
Definition: packet.hh:360
Bitfield< 4 > s
virtual ~BaseXBar()
Definition: xbar.cc:73
const uint32_t width
the width of the xbar in bytes
Definition: xbar.hh:311
uint64_t Tick
Tick count type.
Definition: types.hh:61
BaseXBar(const BaseXBarParams *p)
Definition: xbar.cc:54
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
bool hasData() const
Definition: packet.hh:542
void calcPacketTiming(PacketPtr pkt, Tick header_delay)
Calculate the timing parameters for the packet.
Definition: xbar.cc:99
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
A function used to return the port associated with this object.
Definition: xbar.cc:83
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
Definition: packet.hh:378
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...
Definition: xbar.cc:220
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
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.
Definition: info.hh:49
Bitfield< 24 > j
std::vector< MasterPort * > masterPorts
Definition: xbar.hh:374
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Definition: group.hh:67
virtual const std::string name() const
Definition: sim_object.hh:129
std::vector< QueuedSlavePort * > slavePorts
The master and slave ports of the crossbar.
Definition: xbar.hh:373
void occupyLayer(Tick until)
Definition: xbar.cc:156
void releaseLayer()
Release the layer after being occupied and return to an idle state where we proceed to send a retry t...
Definition: xbar.cc:242
Group()=delete
DrainState drain() override
Drain according to the normal semantics, so that the crossbar can tell the layer to drain...
Definition: xbar.cc:579
PortID findPort(AddrRange addr_range)
Find which port connected to this crossbar (if any) should be given a packet with this address range...
Definition: xbar.cc:325
T divCeil(const T &a, const U &b)
Definition: intmath.hh:99
Layer(DstType &_port, BaseXBar &_xbar, const std::string &_name)
Create a layer and give it a name.
Definition: xbar.cc:137
Bitfield< 4 > width
bool tryTiming(SrcType *src_port)
Determine if the layer accepts a packet from a specific port.
Definition: xbar.cc:177
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:235
virtual void regStats()
Callback to set stat parameters.
Definition: group.cc:64
std::string to_string() const
Get a string representation of the range.
Definition: addr_range.hh:305
const FlagsType nozero
Don&#39;t print if this is zero.
Definition: info.hh:57
Bitfield< 0 > p
Running normally.
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:181
void retryWaiting()
Send a retry to the port at the head of waitingForLayer.
Definition: xbar.cc:266
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.
Definition: xbar.cc:358
Stats::Value simTicks
Definition: stat_control.cc:62

Generated on Thu May 28 2020 16:21:35 for gem5 by doxygen 1.8.13