48 #include "debug/LLSC.hh"
49 #include "debug/MemoryAccess.hh"
50 #include "debug/ProtocolTrace.hh"
51 #include "debug/RubySequencer.hh"
52 #include "debug/RubyStats.hh"
55 #include "mem/ruby/protocol/PrefetchBit.hh"
56 #include "mem/ruby/protocol/RubyAccessMode.hh"
69 :
RubyPort(
p), m_IncompleteTimes(MachineType_NUM),
70 deadlockCheckEvent([this]{
wakeup(); },
"Sequencer deadlock check")
72 m_outstanding_count = 0;
74 m_dataCache_ptr =
p.dcache;
75 m_max_outstanding_requests =
p.max_outstanding_requests;
76 m_deadlock_threshold =
p.deadlock_threshold;
79 assert(m_max_outstanding_requests > 0);
80 assert(m_deadlock_threshold > 0);
82 m_unaddressedTransactionCnt = 0;
84 m_runningGarnetStandalone =
p.garnet_standalone;
90 m_outstandReqHist.init(10);
91 m_latencyHist.init(10);
92 m_hitLatencyHist.init(10);
93 m_missLatencyHist.init(10);
95 for (
int i = 0;
i < RubyRequestType_NUM;
i++) {
97 m_typeLatencyHist[
i]->init(10);
100 m_hitTypeLatencyHist[
i]->init(10);
103 m_missTypeLatencyHist[
i]->init(10);
106 for (
int i = 0;
i < MachineType_NUM;
i++) {
108 m_hitMachLatencyHist[
i]->init(10);
111 m_missMachLatencyHist[
i]->init(10);
114 m_IssueToInitialDelayHist[
i]->init(10);
117 m_InitialToForwardDelayHist[
i]->init(10);
119 m_ForwardToFirstResponseDelayHist.push_back(
121 m_ForwardToFirstResponseDelayHist[
i]->init(10);
123 m_FirstResponseToCompletionDelayHist.push_back(
125 m_FirstResponseToCompletionDelayHist[
i]->init(10);
128 for (
int i = 0;
i < RubyRequestType_NUM;
i++) {
129 m_hitTypeMachLatencyHist.push_back(
131 m_missTypeMachLatencyHist.push_back(
134 for (
int j = 0;
j < MachineType_NUM;
j++) {
135 m_hitTypeMachLatencyHist[
i].push_back(
new statistics::Histogram());
136 m_hitTypeMachLatencyHist[
i][
j]->init(10);
138 m_missTypeMachLatencyHist[
i].push_back(
139 new statistics::Histogram());
140 m_missTypeMachLatencyHist[
i][
j]->init(10);
154 "%s must have a dcache object to support LLSC requests.",
name());
158 DPRINTF(LLSC,
"LLSC Monitor - inserting load linked - "
159 "addr=0x%lx - cpu=%u\n", claddr,
m_version);
172 DPRINTF(LLSC,
"LLSC Monitor - clearing due to store - "
173 "addr=0x%lx - cpu=%u\n", claddr,
m_version);
181 "%s must have a dcache object to support LLSC requests.",
name());
186 DPRINTF(LLSC,
"LLSC Monitor - clearing due to "
187 "store conditional - "
188 "addr=0x%lx - cpu=%u\n",
231 int total_outstanding = 0;
234 for (
const auto &seq_req : table_entry.second) {
238 panic(
"Possible Deadlock detected. Aborting!\n version: %d "
239 "request.paddr: 0x%x m_readRequestTable: %d current time: "
240 "%u issue_time: %d difference: %d\n",
m_version,
241 seq_req.pkt->getAddr(), table_entry.second.size(),
246 total_outstanding += table_entry.second.size();
263 for (
const auto& seq_req : table_entry.second) {
264 if (seq_req.functionalWrite(func_pkt))
278 for (
int i = 0;
i < RubyRequestType_NUM;
i++) {
282 for (
int j = 0;
j < MachineType_NUM;
j++) {
288 for (
int i = 0;
i < MachineType_NUM;
i++) {
305 RubyRequestType secondary_type)
314 assert(primary_type == secondary_type);
316 switch (primary_type) {
317 case RubyRequestType_TLBI_EXT_SYNC_COMP:
320 case RubyRequestType_TLBI:
321 case RubyRequestType_TLBI_SYNC:
326 [[maybe_unused]]
auto insert_data = \
327 m_UnaddressedRequestTable.emplace(
330 pkt, primary_type, secondary_type,
curCycle()));
333 assert(insert_data.second &&
334 "Another TLBI request with the same ID exists");
336 DPRINTF(RubySequencer,
"Inserting TLBI request %016x\n",
343 panic(
"Unexpected TLBI RubyRequestType");
346 return RequestStatus_Ready;
353 seq_req_list.emplace_back(pkt, primary_type,
357 if (seq_req_list.size() > 1) {
358 return RequestStatus_Aliased;
363 return RequestStatus_Ready;
374 const MachineType respondingMach,
375 bool isExternalHit,
Cycles initialRequestTime,
376 Cycles forwardRequestTime,
384 Cycles total_lat = completion_time - issued_time;
386 if ((initialRequestTime != 0) && (initialRequestTime < issued_time)) {
395 DPRINTFR(ProtocolTrace,
"%15s %3s %10s%20s %6s>%-6s %s %d cycles\n",
406 if (respondingMach != MachineType_NUM) {
410 if ((issued_time <= initialRequestTime) &&
411 (initialRequestTime <= forwardRequestTime) &&
412 (forwardRequestTime <= firstResponseTime) &&
413 (firstResponseTime <= completion_time)) {
416 initialRequestTime - issued_time);
418 forwardRequestTime - initialRequestTime);
420 firstResponseTime - forwardRequestTime);
422 completion_time - firstResponseTime);
431 if (respondingMach != MachineType_NUM) {
447 const bool externalHit,
const MachineType mach,
448 const Cycles initialRequestTime,
449 const Cycles forwardRequestTime,
450 const Cycles firstResponseTime,
464 bool ruby_request =
true;
465 int aliased_stores = 0;
466 int aliased_loads = 0;
467 while (!seq_req_list.empty()) {
470 if (noCoales && !ruby_request) {
479 assert(seq_req.
m_type != RubyRequestType_LD);
480 assert(seq_req.
m_type != RubyRequestType_Load_Linked);
481 assert(seq_req.
m_type != RubyRequestType_IFETCH);
485 if ((seq_req.
m_type != RubyRequestType_LD) &&
486 (seq_req.
m_type != RubyRequestType_Load_Linked) &&
487 (seq_req.
m_type != RubyRequestType_IFETCH)) {
491 if (seq_req.
m_type != RubyRequestType_Store_Conditional) {
499 seq_req.
pkt->
req->setExtraData(success ? 1 : 0);
505 if (seq_req.
m_type == RubyRequestType_Locked_RMW_Read) {
513 }
else if (seq_req.
m_type == RubyRequestType_Locked_RMW_Write) {
519 initialRequestTime, forwardRequestTime,
526 initialRequestTime, forwardRequestTime,
527 firstResponseTime, !ruby_request);
528 ruby_request =
false;
531 assert(!ruby_request);
535 initialRequestTime, forwardRequestTime,
536 firstResponseTime, !ruby_request);
538 seq_req_list.pop_front();
542 if (seq_req_list.empty()) {
549 bool externalHit,
const MachineType mach,
550 Cycles initialRequestTime,
551 Cycles forwardRequestTime,
565 bool ruby_request =
true;
566 int aliased_loads = 0;
567 while (!seq_req_list.empty()) {
570 assert((seq_req.
m_type == RubyRequestType_LD) ||
571 (seq_req.
m_type == RubyRequestType_Load_Linked) ||
572 (seq_req.
m_type == RubyRequestType_IFETCH));
576 if ((seq_req.
m_type != RubyRequestType_LD) &&
577 (seq_req.
m_type != RubyRequestType_Load_Linked) &&
578 (seq_req.
m_type != RubyRequestType_IFETCH)) {
585 initialRequestTime, forwardRequestTime,
590 initialRequestTime, forwardRequestTime,
591 firstResponseTime, !ruby_request);
592 ruby_request =
false;
593 seq_req_list.pop_front();
597 if (seq_req_list.empty()) {
605 const MachineType mach,
const bool externalHit,
606 const Cycles initialRequestTime,
607 const Cycles forwardRequestTime,
608 const Cycles firstResponseTime,
609 const bool was_coalesced)
611 warn_once(
"Replacement policy updates recently became the responsibility "
612 "of SLICC state machines. Make sure to setMRU() near callbacks "
628 if (
type == RubyRequestType_Load_Linked) {
637 if ((
type == RubyRequestType_LD) ||
638 (
type == RubyRequestType_IFETCH) ||
639 (
type == RubyRequestType_RMW_Read) ||
640 (
type == RubyRequestType_Locked_RMW_Read) ||
641 (
type == RubyRequestType_Load_Linked)) {
645 }
else if (pkt->
req->isSwap()) {
651 data.setData(&overwrite_val[0],
658 DPRINTF(RubySequencer,
"AMO original data %s\n",
data);
663 }
else if (
type != RubyRequestType_Store_Conditional || llscSuccess) {
675 DPRINTF(RubySequencer,
"hitCallback %s 0x%x using RubyTester\n",
679 assert(testerSenderState);
687 rs->m_cache_recorder->enqueueNextFetchRequest();
690 rs->m_cache_recorder->enqueueNextFlushRequest();
699 RubyRequestType reqType,
700 const MachineType mach,
701 const Cycles initialRequestTime,
702 const Cycles forwardRequestTime,
703 const Cycles firstResponseTime)
705 DPRINTF(RubySequencer,
"unaddressedCallback ID:%08x type:%d\n",
706 unaddressedReqId, reqType);
709 case RubyRequestType_TLBI_EXT_SYNC:
719 case RubyRequestType_TLBI:
720 case RubyRequestType_TLBI_SYNC:
731 assert(seq_req.
m_type == reqType);
743 panic(
"Unexpected TLBI RubyRequestType");
760 !pkt->
req->isHTMAbort()) {
761 return RequestStatus_BufferFull;
764 RubyRequestType primary_type = RubyRequestType_NULL;
765 RubyRequestType secondary_type = RubyRequestType_NULL;
779 DPRINTF(RubySequencer,
"Issuing SC\n");
780 primary_type = RubyRequestType_Store_Conditional;
781 #if defined (PROTOCOL_MESI_Three_Level) || defined (PROTOCOL_MESI_Three_Level_HTM)
782 secondary_type = RubyRequestType_Store_Conditional;
784 secondary_type = RubyRequestType_ST;
787 DPRINTF(RubySequencer,
"Issuing LL\n");
789 primary_type = RubyRequestType_Load_Linked;
790 secondary_type = RubyRequestType_LD;
792 }
else if (pkt->
req->isLockedRMW()) {
800 DPRINTF(RubySequencer,
"Issuing Locked RMW Write\n");
801 primary_type = RubyRequestType_Locked_RMW_Write;
803 DPRINTF(RubySequencer,
"Issuing Locked RMW Read\n");
805 primary_type = RubyRequestType_Locked_RMW_Read;
807 secondary_type = RubyRequestType_ST;
808 }
else if (pkt->
req->isTlbiCmd()) {
810 DPRINTF(RubySequencer,
"Issuing TLBI\n");
821 primary_type = secondary_type = RubyRequestType_ST;
822 }
else if (pkt->
isRead()) {
824 if (pkt->
req->isHTMCmd()) {
826 }
else if (pkt->
req->isInstFetch()) {
827 primary_type = secondary_type = RubyRequestType_IFETCH;
829 if (pkt->
req->isReadModifyWrite()) {
830 primary_type = RubyRequestType_RMW_Read;
831 secondary_type = RubyRequestType_ST;
833 primary_type = secondary_type = RubyRequestType_LD;
837 primary_type = secondary_type = RubyRequestType_FLUSH;
839 panic(
"Unsupported ruby packet type\n");
844 if (!pkt->
req->isMemMgmt() &&
846 (primary_type != RubyRequestType_Locked_RMW_Write)) {
850 return RequestStatus_Aliased;
856 if (
status != RequestStatus_Ready &&
status != RequestStatus_Aliased)
860 if (
status != RequestStatus_Aliased)
864 return RequestStatus_Issued;
878 if (pkt->
req->hasPC()) {
879 pc = pkt->
req->getPC();
884 std::shared_ptr<RubyRequest> msg;
885 if (pkt->
req->isMemMgmt()) {
886 msg = std::make_shared<RubyRequest>(
clockEdge(),
888 RubyAccessMode_Supervisor, pkt,
891 DPRINTFR(ProtocolTrace,
"%15s %3s %10s%20s %6s>%-6s %s\n",
893 RubyRequestType_to_string(secondary_type));
895 if (pkt->
req->isTlbiCmd()) {
896 msg->m_isTlbi =
true;
897 switch (secondary_type) {
898 case RubyRequestType_TLBI_EXT_SYNC_COMP:
899 msg->m_tlbiTransactionUid = pkt->
req->getExtraData();
901 case RubyRequestType_TLBI:
902 case RubyRequestType_TLBI_SYNC:
903 msg->m_tlbiTransactionUid = \
904 getCurrentUnaddressedTransactionID();
907 panic(
"Unexpected TLBI RubyRequestType");
909 DPRINTF(RubySequencer,
"Issuing TLBI %016x\n",
910 msg->m_tlbiTransactionUid);
915 RubyAccessMode_Supervisor, pkt,
916 PrefetchBit_No, proc_id, core_id);
918 DPRINTFR(ProtocolTrace,
"%15s %3s %10s%20s %6s>%-6s %#x %s\n",
921 RubyRequestType_to_string(secondary_type));
928 msg->m_htmFromTransaction =
true;
940 template <
class KEY,
class VALUE>
942 operator<<(std::ostream &out,
const std::unordered_map<KEY, VALUE> &map)
944 for (
const auto &table_entry : map) {
945 out <<
"[ " << table_entry.first <<
" =";
946 for (
const auto &seq_req : table_entry.second) {
947 out <<
" " << RubyRequestType_to_string(seq_req.m_second_type);
966 DPRINTF(RubyStats,
"Recorded statistic: %s\n",
967 SequencerRequestType_to_string(requestType));
983 uint64_t aligned_txid = \
986 if (aligned_txid > 0xFFFFFFFFull) {
995 uint64_t(
m_version & 0xFFFFFFFF) << 32) |