gem5 [DEVELOP-FOR-25.0]
Loading...
Searching...
No Matches
noncoherent_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
45
47
48#include "base/logging.hh"
49#include "base/trace.hh"
50#include "debug/NoncoherentXBar.hh"
51#include "debug/XBar.hh"
52
53namespace gem5
54{
55
56NoncoherentXBar::NoncoherentXBar(const NoncoherentXBarParams &p)
57 : BaseXBar(p)
58{
59 // create the ports based on the size of the memory-side port and
60 // CPU-side port vector ports, and the presence of the default port,
61 // the ports are enumerated starting from zero
62 for (int i = 0; i < p.port_mem_side_ports_connection_count; ++i) {
63 std::string portName = csprintf("%s.mem_side_port[%d]", name(), i);
64 RequestPort* bp = new NoncoherentXBarRequestPort(portName, *this, i);
65 memSidePorts.push_back(bp);
66 reqLayers.push_back(new ReqLayer(*bp, *this,
67 csprintf("reqLayer%d", i)));
68 }
69
70 // see if we have a default CPU-side-port device connected and if so add
71 // our corresponding memory-side port
72 if (p.port_default_connection_count) {
73 defaultPortID = memSidePorts.size();
74 std::string portName = name() + ".default";
75 RequestPort* bp = new NoncoherentXBarRequestPort(portName, *this,
76 defaultPortID);
77 memSidePorts.push_back(bp);
78 reqLayers.push_back(new ReqLayer(*bp, *this, csprintf("reqLayer%d",
79 defaultPortID)));
80 }
81
82 // create the CPU-side ports, once again starting at zero
83 for (int i = 0; i < p.port_cpu_side_ports_connection_count; ++i) {
84 std::string portName = csprintf("%s.cpu_side_ports[%d]", name(), i);
86 *this, i);
87 cpuSidePorts.push_back(bp);
88 respLayers.push_back(new RespLayer(*bp, *this,
89 csprintf("respLayer%d", i)));
90 }
91}
92
94{
95 for (auto l: reqLayers)
96 delete l;
97 for (auto l: respLayers)
98 delete l;
99}
100
101bool
103{
104 // determine the source port based on the id
105 ResponsePort *src_port = cpuSidePorts[cpu_side_port_id];
106
107 // we should never see express snoops on a non-coherent crossbar
108 assert(!pkt->isExpressSnoop());
109
110 // determine the destination port
111 PortID mem_side_port_id = findPort(pkt);
112
113 // test if the layer should be considered occupied for the current
114 // port
115 if (!reqLayers[mem_side_port_id]->tryTiming(src_port)) {
116 DPRINTF(NoncoherentXBar, "recvTimingReq: src %s %s 0x%x BUSY\n",
117 src_port->name(), pkt->cmdString(), pkt->getAddr());
118 return false;
119 }
120
121 DPRINTF(NoncoherentXBar, "recvTimingReq: src %s %s 0x%x\n",
122 src_port->name(), pkt->cmdString(), pkt->getAddr());
123
124 // store size and command as they might be modified when
125 // forwarding the packet
126 unsigned int pkt_size = pkt->hasData() ? pkt->getSize() : 0;
127 unsigned int pkt_cmd = pkt->cmdToIndex();
128
129 // store the old header delay so we can restore it if needed
130 Tick old_header_delay = pkt->headerDelay;
131
132 // a request sees the frontend and forward latency
133 Tick xbar_delay = (frontendLatency + forwardLatency) * clockPeriod();
134
135 // set the packet header and payload delay
136 calcPacketTiming(pkt, xbar_delay);
137
138 // determine how long to be crossbar layer is busy
139 Tick packetFinishTime = clockEdge(Cycles(1)) + pkt->payloadDelay;
140
141 // before forwarding the packet (and possibly altering it),
142 // remember if we are expecting a response
143 const bool expect_response = pkt->needsResponse() &&
144 !pkt->cacheResponding();
145
146 // since it is a normal request, attempt to send the packet
147 bool success = memSidePorts[mem_side_port_id]->sendTimingReq(pkt);
148
149 if (!success) {
150 DPRINTF(NoncoherentXBar, "recvTimingReq: src %s %s 0x%x RETRY\n",
151 src_port->name(), pkt->cmdString(), pkt->getAddr());
152
153 // restore the header delay as it is additive
154 pkt->headerDelay = old_header_delay;
155
156 // occupy until the header is sent
157 reqLayers[mem_side_port_id]->failedTiming(src_port,
158 clockEdge(Cycles(1)));
159
160 return false;
161 }
162
163 // remember where to route the response to
164 if (expect_response) {
165 assert(routeTo.find(pkt->req) == routeTo.end());
166 routeTo[pkt->req] = cpu_side_port_id;
167 }
168
169 reqLayers[mem_side_port_id]->succeededTiming(packetFinishTime);
170
171 // stats updates
172 pktCount[cpu_side_port_id][mem_side_port_id]++;
173 pktSize[cpu_side_port_id][mem_side_port_id] += pkt_size;
174 transDist[pkt_cmd]++;
175
176 return true;
177}
178
179bool
181{
182 // determine the source port based on the id
183 RequestPort *src_port = memSidePorts[mem_side_port_id];
184
185 // determine the destination
186 const auto route_lookup = routeTo.find(pkt->req);
187 assert(route_lookup != routeTo.end());
188 const PortID cpu_side_port_id = route_lookup->second;
189 assert(cpu_side_port_id != InvalidPortID);
190 assert(cpu_side_port_id < respLayers.size());
191
192 // test if the layer should be considered occupied for the current
193 // port
194 if (!respLayers[cpu_side_port_id]->tryTiming(src_port)) {
195 DPRINTF(NoncoherentXBar, "recvTimingResp: src %s %s 0x%x BUSY\n",
196 src_port->name(), pkt->cmdString(), pkt->getAddr());
197 return false;
198 }
199
200 DPRINTF(NoncoherentXBar, "recvTimingResp: src %s %s 0x%x\n",
201 src_port->name(), pkt->cmdString(), pkt->getAddr());
202
203 // store size and command as they might be modified when
204 // forwarding the packet
205 unsigned int pkt_size = pkt->hasData() ? pkt->getSize() : 0;
206 unsigned int pkt_cmd = pkt->cmdToIndex();
207
208 // a response sees the response latency
209 Tick xbar_delay = responseLatency * clockPeriod();
210
211 // set the packet header and payload delay
212 calcPacketTiming(pkt, xbar_delay);
213
214 // determine how long to be crossbar layer is busy
215 Tick packetFinishTime = clockEdge(Cycles(1)) + pkt->payloadDelay;
216
217 // send the packet through the destination CPU-side port, and pay for
218 // any outstanding latency
219 Tick latency = pkt->headerDelay;
220 pkt->headerDelay = 0;
221 cpuSidePorts[cpu_side_port_id]->schedTimingResp(pkt,
222 curTick() + latency);
223
224 // remove the request from the routing table
225 routeTo.erase(route_lookup);
226
227 respLayers[cpu_side_port_id]->succeededTiming(packetFinishTime);
228
229 // stats updates
230 pktCount[cpu_side_port_id][mem_side_port_id]++;
231 pktSize[cpu_side_port_id][mem_side_port_id] += pkt_size;
232 transDist[pkt_cmd]++;
233
234 return true;
235}
236
237void
239{
240 // responses never block on forwarding them, so the retry will
241 // always be coming from a port to which we tried to forward a
242 // request
243 reqLayers[mem_side_port_id]->recvRetry();
244}
245
246Tick
248 MemBackdoorPtr *backdoor)
249{
250 DPRINTF(NoncoherentXBar, "recvAtomic: packet src %s addr 0x%x cmd %s\n",
251 cpuSidePorts[cpu_side_port_id]->name(), pkt->getAddr(),
252 pkt->cmdString());
253
254 unsigned int pkt_size = pkt->hasData() ? pkt->getSize() : 0;
255 unsigned int pkt_cmd = pkt->cmdToIndex();
256
257 // determine the destination port
258 PortID mem_side_port_id = findPort(pkt);
259
260 // stats updates for the request
261 pktCount[cpu_side_port_id][mem_side_port_id]++;
262 pktSize[cpu_side_port_id][mem_side_port_id] += pkt_size;
263 transDist[pkt_cmd]++;
264
265 // forward the request to the appropriate destination
266 auto mem_side_port = memSidePorts[mem_side_port_id];
267 Tick response_latency = backdoor ?
268 mem_side_port->sendAtomicBackdoor(pkt, *backdoor) :
269 mem_side_port->sendAtomic(pkt);
270
271 // add the response data
272 if (pkt->isResponse()) {
273 pkt_size = pkt->hasData() ? pkt->getSize() : 0;
274 pkt_cmd = pkt->cmdToIndex();
275
276 // stats updates
277 pktCount[cpu_side_port_id][mem_side_port_id]++;
278 pktSize[cpu_side_port_id][mem_side_port_id] += pkt_size;
279 transDist[pkt_cmd]++;
280 }
281
282 // @todo: Not setting first-word time
283 pkt->payloadDelay = response_latency;
284 return response_latency;
285}
286
287void
289 MemBackdoorPtr &backdoor)
290{
291 PortID dest_id = findPort(req.range());
292 memSidePorts[dest_id]->sendMemBackdoorReq(req, backdoor);
293}
294
295void
297{
298 if (!pkt->isPrint()) {
299 // don't do DPRINTFs on PrintReq as it clutters up the output
301 "recvFunctional: packet src %s addr 0x%x cmd %s\n",
302 cpuSidePorts[cpu_side_port_id]->name(), pkt->getAddr(),
303 pkt->cmdString());
304 }
305
306 // since our CPU-side ports are queued ports we need to check them as well
307 for (const auto& p : cpuSidePorts) {
308 // if we find a response that has the data, then the
309 // downstream caches/memories may be out of date, so simply stop
310 // here
311 if (p->trySatisfyFunctional(pkt)) {
312 if (pkt->needsResponse())
313 pkt->makeResponse();
314 return;
315 }
316 }
317
318 // determine the destination port
319 PortID dest_id = findPort(pkt);
320
321 // forward the request to the appropriate destination
322 memSidePorts[dest_id]->sendFunctional(pkt);
323}
324
325} // namespace gem5
#define DPRINTF(x,...)
Definition trace.hh:209
std::vector< RequestPort * > memSidePorts
Definition xbar.hh:387
const Cycles frontendLatency
Cycles of front-end pipeline including the delay to accept the request and to decode the address.
Definition xbar.hh:311
PortID findPort(AddrRange addr_range, PacketPtr pkt=nullptr)
Find which port connected to this crossbar (if any) should be given a packet with this address range.
Definition xbar.cc:334
statistics::Vector transDist
Stats for transaction distribution and data passing through the crossbar.
Definition xbar.hh:409
std::unordered_map< RequestPtr, PortID > routeTo
Remember where request packets came from so that we can route responses to the appropriate port.
Definition xbar.hh:327
statistics::Vector2d pktCount
Definition xbar.hh:410
BaseXBar(const BaseXBarParams &p)
Definition xbar.cc:60
std::vector< QueuedResponsePort * > cpuSidePorts
The memory-side ports and CPU-side ports of the crossbar.
Definition xbar.hh:386
const Cycles forwardLatency
Definition xbar.hh:312
statistics::Vector2d pktSize
Definition xbar.hh:411
void calcPacketTiming(PacketPtr pkt, Tick header_delay)
Calculate the timing parameters for the packet.
Definition xbar.cc:108
const Cycles responseLatency
Definition xbar.hh:313
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...
Tick clockPeriod() const
Cycles is a wrapper class for representing cycle counts, i.e.
Definition types.hh:79
const AddrRange & range() const
Definition backdoor.hh:140
virtual std::string name() const
Definition named.hh:60
Declaration of the crossbar memory-side port type, one will be instantiated for each of the CPU-side ...
Declaration of the non-coherent crossbar CPU-side port type, one will be instantiated for each of the...
NoncoherentXBar(const NoncoherentXBarParams &p)
std::vector< RespLayer * > respLayers
Tick recvAtomicBackdoor(PacketPtr pkt, PortID cpu_side_port_id, MemBackdoorPtr *backdoor=nullptr)
void recvReqRetry(PortID mem_side_port_id)
virtual bool recvTimingReq(PacketPtr pkt, PortID cpu_side_port_id)
virtual bool recvTimingResp(PacketPtr pkt, PortID mem_side_port_id)
void recvMemBackdoorReq(const MemBackdoorReq &req, MemBackdoorPtr &backdoor)
void recvFunctional(PacketPtr pkt, PortID cpu_side_port_id)
std::vector< ReqLayer * > reqLayers
Declare the layers of this crossbar, one vector for requests and one for responses.
Addr getAddr() const
Definition packet.hh:807
const std::string & cmdString() const
Return the string name of the cmd field (for debugging and tracing).
Definition packet.hh:588
bool isResponse() const
Definition packet.hh:598
bool needsResponse() const
Definition packet.hh:608
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
Definition packet.hh:449
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
Definition packet.hh:1062
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
Definition packet.hh:431
int cmdToIndex() const
Return the index of this command.
Definition packet.hh:591
bool hasData() const
Definition packet.hh:614
RequestPtr req
A pointer to the original request.
Definition packet.hh:377
bool isPrint() const
Definition packet.hh:623
unsigned getSize() const
Definition packet.hh:817
bool isExpressSnoop() const
Definition packet.hh:702
bool cacheResponding() const
Definition packet.hh:659
const std::string name() const
Return port name (for DPRINTF).
Definition port.hh:111
A queued port is a port that has an infinite queue for outgoing packets and thus decouples the module...
Definition qport.hh:62
A RequestPort is a specialisation of a Port, which implements the default protocol for the three diff...
Definition port.hh:136
A ResponsePort is a specialization of a port.
Definition port.hh:349
Bitfield< 7 > i
Definition misc_types.hh:67
Bitfield< 5 > l
Bitfield< 0 > p
Copyright (c) 2024 Arm Limited All rights reserved.
Definition binary32.hh:36
const PortID InvalidPortID
Definition types.hh:246
Tick curTick()
The universal simulation clock.
Definition cur_tick.hh:46
MemBackdoor * MemBackdoorPtr
Definition backdoor.hh:127
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition types.hh:245
uint64_t Tick
Tick count type.
Definition types.hh:58
Packet * PacketPtr
std::string csprintf(const char *format, const Args &...args)
Definition cprintf.hh:161
Declaration of a non-coherent crossbar.

Generated on Mon May 26 2025 09:19:12 for gem5 by doxygen 1.13.2