gem5 v24.1.0.1
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
controller.cc
Go to the documentation of this file.
1/*
2 * Copyright (c) 2023 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 * Redistribution and use in source and binary forms, with or without
15 * modification, are permitted provided that the following conditions are
16 * met: redistributions of source code must retain the above copyright
17 * notice, this list of conditions and the following disclaimer;
18 * redistributions in binary form must reproduce the above copyright
19 * notice, this list of conditions and the following disclaimer in the
20 * documentation and/or other materials provided with the distribution;
21 * neither the name of the copyright holders nor the names of its
22 * contributors may be used to endorse or promote products derived from
23 * this software without specific prior written permission.
24 *
25 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36 */
37
39
40#include "debug/TLM.hh"
41#include "mem/ruby/protocol/CHI/CHIDataMsg.hh"
42#include "mem/ruby/protocol/CHI/CHIRequestMsg.hh"
43#include "mem/ruby/protocol/CHI/CHIResponseMsg.hh"
45
46namespace gem5 {
47
48namespace tlm::chi {
49
50using namespace ruby::CHI;
51
53 : CHIGenericController(p)
54{
55}
56
57bool
59{
60 panic("recvRequestMsg");
61 return true;
62}
63
64bool
66{
67 ARM::CHI::Phase phase;
68 ARM::CHI::Payload *payload = ARM::CHI::Payload::new_payload();
69
70 payload->address = msg->m_addr;
71 payload->ns = msg->m_ns;
72 phase.channel = ARM::CHI::CHANNEL_SNP;
73 phase.snp_opcode = ruby_to_tlm::snpOpcode(msg->m_type);
74 phase.txn_id = msg->m_txnId % 1024;
75
76 bw(payload, &phase);
77
78 // payload->unref(); ??
79 return true;
80}
81
82void
84{
85 ARM::CHI::Phase phase;
86 ARM::CHI::Payload *payload = ARM::CHI::Payload::new_payload();
87 phase.channel = ARM::CHI::CHANNEL_RSP;
88 phase.rsp_opcode = ARM::CHI::RSP_OPCODE_PCRD_GRANT;
89 phase.pcrd_type = 0; // TODO: set this one depending on allow retry
90
91 bw(payload, &phase);
92
93 payload->unref(); // TODO: check this
94}
95
96bool
98{
99
100 if (msg->m_type == CHIResponseType_PCrdGrant) {
101 // P-credit grant does not refer to a specific transaction id
102 pCreditGrant(msg);
103 return true;
104 }
105
106 auto txn_id = msg->m_txnId;
107 if (auto it = pendingTransactions.find(txn_id);
108 it != pendingTransactions.end()) {
109
110 auto& transaction = it->second;
111 if (transaction->handle(msg))
112 pendingTransactions.erase(it);
113 } else {
114 panic("recvResponseMsg");
115 }
116 return true;
117}
118
119bool
121{
122 auto txn_id = msg->m_txnId;
123 if (auto it = pendingTransactions.find(txn_id);
124 it != pendingTransactions.end()) {
125 auto& transaction = it->second;
126 if (transaction->handle(msg))
127 pendingTransactions.erase(it);
128 } else {
129 panic("Not able to find transaction");
130 }
131 return true;
132}
133
134bool
136{
137 const auto opcode = ruby_to_tlm::rspOpcode(msg->m_type);
138 phase.channel = ARM::CHI::CHANNEL_RSP;
139 phase.rsp_opcode = opcode;
140 phase.resp = ruby_to_tlm::rspResp(msg->m_type);
141 phase.txn_id = msg->m_txnId % 1024;
142
144 return opcode != ARM::CHI::RSP_OPCODE_RETRY_ACK;
145}
146
147bool
149{
150 dataMsgCnt++;
151
152 for (auto byte = 0; byte < controller->cacheLineSize; byte++) {
153 if (msg->m_bitMask.test(byte))
154 payload->data[byte] = msg->m_dataBlk.getByte(byte);
155 }
156
157 // ARM::CHI::Phase phase;
158 phase.channel = ARM::CHI::CHANNEL_DAT;
159 phase.dat_opcode = ruby_to_tlm::datOpcode(msg->m_type);
160 phase.resp = ruby_to_tlm::datResp(msg->m_type);
161 phase.txn_id = msg->m_txnId % 1024;
162 phase.data_id = dataId(msg->m_addr + msg->m_bitMask.firstBitSet(true));
163
164 // This is a hack, we should fix it on the ruby side
165 if (forward(msg)) {
166 controller->bw(payload, &phase);
167 }
168
169 if (dataMsgCnt == controller->dataMsgsPerLine) {
170 if (phase.exp_comp_ack == false) {
171 // The client is not sending a CompAck but ruby is
172 // expecting it so we send it anyway
173 controller->sendCompAck(*payload, phase);
174 }
175 return true;
176 } else {
177 return false;
178 }
179}
180
181bool
183{
185 phase.dbid = msg->m_dbid % 1024;
186 return Transaction::handle(msg);
187}
188
189bool
191{
192 if (payload->size == 6) {
193 return true;
194 } else {
195 if (msg->m_bitMask.test(payload->address - msg->m_addr)) {
196 return true;
197 } else {
198 // This is not the packet holding the original address
199 // Do not forward it back (just drop it)
200 return false;
201 }
202 }
203}
204
205bool
207{
208 const auto opcode = ruby_to_tlm::rspOpcode(msg->m_type);
209 assert(opcode == ARM::CHI::RSP_OPCODE_COMP ||
210 opcode == ARM::CHI::RSP_OPCODE_RETRY_ACK);
211
212 return Transaction::handle(msg);
213}
214
215bool
217{
218 const auto opcode = ruby_to_tlm::rspOpcode(msg->m_type);
219 if (opcode == ARM::CHI::RSP_OPCODE_COMP_DBID_RESP) {
220 recvComp = true;
221 recvDBID = true;
222 }
223 if (opcode == ARM::CHI::RSP_OPCODE_COMP) {
224 recvComp = true;
225 }
226 if (opcode == ARM::CHI::RSP_OPCODE_DBID_RESP) {
227 recvDBID = true;
228 }
229
230 phase.dbid = msg->m_dbid % 1024;
232
233 return recvComp && recvDBID;
234}
235
236void
237CacheController::sendCompAck(ARM::CHI::Payload &payload,
238 ARM::CHI::Phase &phase)
239{
240 auto res_msg = std::make_shared<CHIResponseMsg>(
242
243 res_msg->m_addr = ruby::makeLineAddress(payload.address, cacheLineBits);
244 res_msg->m_type = CHIResponseType_CompAck;
245 res_msg->m_Destination.add(mapAddressToDownstreamMachine(payload.address));
246 res_msg->m_responder = getMachineID();
247
248 res_msg->m_txnId = phase.txn_id;
249
250 sendResponseMsg(res_msg);
251}
252
253void
254CacheController::sendMsg(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase)
255{
256 switch (phase.channel) {
257 case ARM::CHI::CHANNEL_REQ:
258 sendRequestMsg(payload, phase);
259 break;
260 case ARM::CHI::CHANNEL_DAT:
261 sendDataMsg(payload, phase);
262 break;
263 case ARM::CHI::CHANNEL_RSP:
264 sendResponseMsg(payload, phase);
265 break;
266 default:
267 panic("Unimplemented channel: %d", phase.channel);
268 }
269}
270
271Addr
272CacheController::reqAddr(ARM::CHI::Payload &payload,
273 ARM::CHI::Phase &phase) const
274{
275 switch (phase.req_opcode) {
276 case ARM::CHI::REQ_OPCODE_WRITE_NO_SNP_PTL:
277 return ruby::makeLineAddress(payload.address, cacheLineBits) +
278 ctz64(payload.byte_enable);
279 default:
280 return payload.address;
281 }
282}
283
284Addr
285CacheController::reqSize(ARM::CHI::Payload &payload,
286 ARM::CHI::Phase &phase) const
287{
288 switch (phase.req_opcode) {
289 case ARM::CHI::REQ_OPCODE_WRITE_NO_SNP_PTL:
290 assert(transactionSize(payload.size) >= popCount(payload.byte_enable));
291 return popCount(payload.byte_enable);
292 default:
293 return transactionSize(payload.size);
294 }
295}
296
297void
298CacheController::sendRequestMsg(ARM::CHI::Payload &payload,
299 ARM::CHI::Phase &phase)
300{
301 auto req_msg = std::make_shared<CHIRequestMsg>(
303
304 req_msg->m_addr = ruby::makeLineAddress(payload.address, cacheLineBits);
305 req_msg->m_accAddr = reqAddr(payload, phase);
306 req_msg->m_accSize = reqSize(payload, phase);
307 req_msg->m_requestor = getMachineID();
308 req_msg->m_fwdRequestor = getMachineID();
309 req_msg->m_dataToFwdRequestor = false;
310 req_msg->m_type = tlm_to_ruby::reqOpcode(phase.req_opcode);
311 req_msg->m_isSeqReqValid = false;
312 req_msg->m_is_local_pf = false;
313 req_msg->m_is_remote_pf = false;
314 req_msg->m_allowRetry = phase.allow_retry;
315 req_msg->m_Destination.add(mapAddressToDownstreamMachine(payload.address));
316
317 req_msg->m_txnId = phase.txn_id + (payload.lpid * 1024);
318 req_msg->m_ns = payload.ns;
319
320 sendRequestMsg(req_msg);
321
322 pendingTransactions[req_msg->m_txnId] = Transaction::gen(
323 this, payload, phase);
324}
325
326void
327CacheController::sendDataMsg(ARM::CHI::Payload &payload,
328 ARM::CHI::Phase &phase)
329{
330 auto data_msg = std::make_shared<CHIDataMsg>(
332
333 data_msg->m_addr = ruby::makeLineAddress(payload.address, cacheLineBits);
334 data_msg->m_responder = getMachineID();
335 data_msg->m_type = tlm_to_ruby::datOpcode(phase.dat_opcode, phase.resp);
336 data_msg->m_Destination.add(
337 mapAddressToDownstreamMachine(payload.address));
338 data_msg->m_txnId = phase.txn_id + (payload.lpid * 1024);
339 data_msg->m_dataBlk.setData(payload.data, 0, cacheLineSize);
340
341 std::vector<bool> byte_enabled(cacheLineSize, false);
342
343 const Addr data_id_offset = phase.data_id * 16;
344 for (int bit = data_id_offset; bit < data_id_offset + 32; bit++) {
345 if (bits(payload.byte_enable, bit)) {
346 byte_enabled[bit] = true;
347 }
348 }
349
350 data_msg->m_bitMask = ruby::WriteMask(byte_enabled.size(), byte_enabled);
351
352 sendDataMsg(data_msg);
353}
354
355void
356CacheController::sendResponseMsg(ARM::CHI::Payload &payload,
357 ARM::CHI::Phase &phase)
358{
359 auto res_msg = std::make_shared<CHIResponseMsg>(
361
362 res_msg->m_addr = ruby::makeLineAddress(payload.address, cacheLineBits);
363 res_msg->m_responder = getMachineID();
364 res_msg->m_type = tlm_to_ruby::rspOpcode(phase.rsp_opcode, phase.resp);
365 res_msg->m_Destination.add(mapAddressToDownstreamMachine(payload.address));
366 res_msg->m_txnId = phase.txn_id + (payload.lpid * 1024);
367
368 sendResponseMsg(res_msg);
369}
370
372 ARM::CHI::Payload &_payload,
373 ARM::CHI::Phase &_phase)
374 : controller(_controller), payload(&_payload), phase(_phase)
375{
376 payload->ref();
377}
378
380{
381 payload->unref();
382}
383
384std::unique_ptr<CacheController::Transaction>
386 ARM::CHI::Payload &payload,
387 ARM::CHI::Phase &phase)
388{
389 switch (phase.req_opcode) {
390 case ARM::CHI::REQ_OPCODE_READ_SHARED:
391 case ARM::CHI::REQ_OPCODE_READ_CLEAN:
392 case ARM::CHI::REQ_OPCODE_READ_ONCE:
393 case ARM::CHI::REQ_OPCODE_READ_NO_SNP:
394 case ARM::CHI::REQ_OPCODE_READ_UNIQUE:
395 case ARM::CHI::REQ_OPCODE_READ_NOT_SHARED_DIRTY:
396 case ARM::CHI::REQ_OPCODE_READ_PREFER_UNIQUE:
397 case ARM::CHI::REQ_OPCODE_MAKE_READ_UNIQUE:
398 return std::make_unique<ReadTransaction>(controller, payload, phase);
399 case ARM::CHI::REQ_OPCODE_WRITE_NO_SNP_PTL:
400 case ARM::CHI::REQ_OPCODE_WRITE_NO_SNP_FULL:
401 case ARM::CHI::REQ_OPCODE_WRITE_UNIQUE_ZERO:
402 case ARM::CHI::REQ_OPCODE_WRITE_UNIQUE_FULL:
403 case ARM::CHI::REQ_OPCODE_WRITE_BACK_FULL:
404 // This is a write transaction for now. Will
405 // this still be the case once WriteEvictOrEvict will also be supported
406 // in ruby
407 case ARM::CHI::REQ_OPCODE_WRITE_EVICT_OR_EVICT:
408 return std::make_unique<WriteTransaction>(
409 controller, payload, phase);
410 case ARM::CHI::REQ_OPCODE_CLEAN_UNIQUE:
411 case ARM::CHI::REQ_OPCODE_MAKE_UNIQUE:
412 case ARM::CHI::REQ_OPCODE_EVICT:
413 case ARM::CHI::REQ_OPCODE_STASH_ONCE_SEP_SHARED:
414 case ARM::CHI::REQ_OPCODE_STASH_ONCE_SEP_UNIQUE:
415 return std::make_unique<DatalessTransaction>(
416 controller, payload, phase);
417 default:
418 panic("Impossible to generate transaction object");
419 }
420}
421
422} // namespace tlm::chi
423
424} // namespace gem5
ClockedObjectParams Params
Parameters of ClockedObject.
MachineID mapAddressToDownstreamMachine(Addr addr, MachineType mtype=MachineType_NUM) const
Maps an address to the correct dowstream MachineID (i.e.
The tlm::chi::CacheController is a ruby CacheController which acts as a bridge between the AMBA TLM 2...
Definition controller.hh:92
bool recvRequestMsg(const CHIRequestMsg *msg) override
Definition controller.cc:58
CacheController(const Params &p)
Definition controller.cc:52
void sendResponseMsg(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase)
Addr reqSize(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase) const
bool recvDataMsg(const CHIDataMsg *msg) override
void pCreditGrant(const CHIResponseMsg *msg)
Definition controller.cc:83
bool recvSnoopMsg(const CHIRequestMsg *msg) override
Definition controller.cc:65
void sendCompAck(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase)
void sendMsg(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase)
bool recvResponseMsg(const CHIResponseMsg *msg) override
Definition controller.cc:97
void sendDataMsg(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase)
void sendRequestMsg(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase)
Addr reqAddr(ARM::CHI::Payload &payload, ARM::CHI::Phase &phase) const
std::function< void(ARM::CHI::Payload *payload, ARM::CHI::Phase *phase)> bw
Set this to send data upstream.
Definition controller.hh:98
std::unordered_map< uint16_t, std::unique_ptr< Transaction > > pendingTransactions
STL vector class.
Definition stl.hh:37
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Definition bitfield.hh:79
constexpr int popCount(uint64_t val)
Returns the number of set ones in the provided value.
Definition bitfield.hh:415
constexpr int ctz64(uint64_t value)
Count trailing zeros in a 64-bit value.
Definition bitfield.hh:487
#define panic(...)
This implements a cprintf based panic() function.
Definition logging.hh:188
Bitfield< 24, 21 > opcode
Definition types.hh:92
Bitfield< 0 > p
Addr makeLineAddress(Addr addr, int cacheLineBits)
Definition Address.cc:61
uint8_t snpOpcode(CHIRequestType snp)
Definition utils.cc:392
uint8_t rspOpcode(CHIResponseType rsp)
Definition utils.cc:375
Resp rspResp(CHIResponseType rsp)
Definition utils.cc:455
Resp datResp(CHIDataType dat)
Definition utils.cc:411
uint8_t datOpcode(CHIDataType dat)
Definition utils.cc:334
CHIDataType datOpcode(DatOpcode dat, Resp resp)
Definition utils.cc:294
CHIResponseType rspOpcode(RspOpcode opc, Resp resp)
Definition utils.cc:315
CHIRequestType reqOpcode(ReqOpcode req)
Definition utils.cc:247
Addr transactionSize(Size sz)
Definition utils.cc:477
uint8_t dataId(Addr address)
Definition utils.cc:483
Copyright (c) 2024 Arm Limited All rights reserved.
Definition binary32.hh:36
Tick curTick()
The universal simulation clock.
Definition cur_tick.hh:46
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition types.hh:147
bool handle(const CHIResponseMsg *msg) override
bool handle(const CHIDataMsg *msg) override
Transaction(CacheController *parent, ARM::CHI::Payload &_payload, ARM::CHI::Phase &_phase)
static std::unique_ptr< Transaction > gen(CacheController *parent, ARM::CHI::Payload &_payload, ARM::CHI::Phase &_phase)
virtual bool handle(const CHIDataMsg *msg)
bool handle(const CHIResponseMsg *msg) override

Generated on Mon Jan 13 2025 04:28:40 for gem5 by doxygen 1.9.8