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_runningGarnetStandalone =
p.garnet_standalone;
88 m_outstandReqHist.init(10);
89 m_latencyHist.init(10);
90 m_hitLatencyHist.init(10);
91 m_missLatencyHist.init(10);
93 for (
int i = 0;
i < RubyRequestType_NUM;
i++) {
95 m_typeLatencyHist[
i]->init(10);
98 m_hitTypeLatencyHist[
i]->init(10);
101 m_missTypeLatencyHist[
i]->init(10);
104 for (
int i = 0;
i < MachineType_NUM;
i++) {
106 m_hitMachLatencyHist[
i]->init(10);
109 m_missMachLatencyHist[
i]->init(10);
112 m_IssueToInitialDelayHist[
i]->init(10);
115 m_InitialToForwardDelayHist[
i]->init(10);
117 m_ForwardToFirstResponseDelayHist.push_back(
119 m_ForwardToFirstResponseDelayHist[
i]->init(10);
121 m_FirstResponseToCompletionDelayHist.push_back(
123 m_FirstResponseToCompletionDelayHist[
i]->init(10);
126 for (
int i = 0;
i < RubyRequestType_NUM;
i++) {
127 m_hitTypeMachLatencyHist.push_back(
129 m_missTypeMachLatencyHist.push_back(
132 for (
int j = 0;
j < MachineType_NUM;
j++) {
133 m_hitTypeMachLatencyHist[
i].push_back(
new statistics::Histogram());
134 m_hitTypeMachLatencyHist[
i][
j]->init(10);
136 m_missTypeMachLatencyHist[
i].push_back(
137 new statistics::Histogram());
138 m_missTypeMachLatencyHist[
i][
j]->init(10);
152 "%s must have a dcache object to support LLSC requests.",
name());
156 DPRINTF(LLSC,
"LLSC Monitor - inserting load linked - "
157 "addr=0x%lx - cpu=%u\n", claddr,
m_version);
170 DPRINTF(LLSC,
"LLSC Monitor - clearing due to store - "
171 "addr=0x%lx - cpu=%u\n", claddr,
m_version);
179 "%s must have a dcache object to support LLSC requests.",
name());
184 DPRINTF(LLSC,
"LLSC Monitor - clearing due to "
185 "store conditional - "
186 "addr=0x%lx - cpu=%u\n",
229 int total_outstanding = 0;
232 for (
const auto &seq_req : table_entry.second) {
236 panic(
"Possible Deadlock detected. Aborting!\n version: %d "
237 "request.paddr: 0x%x m_readRequestTable: %d current time: "
238 "%u issue_time: %d difference: %d\n",
m_version,
239 seq_req.pkt->getAddr(), table_entry.second.size(),
244 total_outstanding += table_entry.second.size();
261 for (
const auto& seq_req : table_entry.second) {
262 if (seq_req.functionalWrite(func_pkt))
276 for (
int i = 0;
i < RubyRequestType_NUM;
i++) {
280 for (
int j = 0;
j < MachineType_NUM;
j++) {
286 for (
int i = 0;
i < MachineType_NUM;
i++) {
303 RubyRequestType secondary_type)
315 seq_req_list.emplace_back(pkt, primary_type,
319 if (seq_req_list.size() > 1) {
320 return RequestStatus_Aliased;
325 return RequestStatus_Ready;
336 const MachineType respondingMach,
337 bool isExternalHit,
Cycles initialRequestTime,
338 Cycles forwardRequestTime,
346 Cycles total_lat = completion_time - issued_time;
348 if ((initialRequestTime != 0) && (initialRequestTime < issued_time)) {
357 DPRINTFR(ProtocolTrace,
"%15s %3s %10s%20s %6s>%-6s %s %d cycles\n",
368 if (respondingMach != MachineType_NUM) {
372 if ((issued_time <= initialRequestTime) &&
373 (initialRequestTime <= forwardRequestTime) &&
374 (forwardRequestTime <= firstResponseTime) &&
375 (firstResponseTime <= completion_time)) {
378 initialRequestTime - issued_time);
380 forwardRequestTime - initialRequestTime);
382 firstResponseTime - forwardRequestTime);
384 completion_time - firstResponseTime);
393 if (respondingMach != MachineType_NUM) {
409 const bool externalHit,
const MachineType mach,
410 const Cycles initialRequestTime,
411 const Cycles forwardRequestTime,
412 const Cycles firstResponseTime,
426 bool ruby_request =
true;
427 int aliased_stores = 0;
428 int aliased_loads = 0;
429 while (!seq_req_list.empty()) {
432 if (noCoales && !ruby_request) {
441 assert(seq_req.
m_type != RubyRequestType_LD);
442 assert(seq_req.
m_type != RubyRequestType_Load_Linked);
443 assert(seq_req.
m_type != RubyRequestType_IFETCH);
447 if ((seq_req.
m_type != RubyRequestType_LD) &&
448 (seq_req.
m_type != RubyRequestType_Load_Linked) &&
449 (seq_req.
m_type != RubyRequestType_IFETCH)) {
453 if (seq_req.
m_type != RubyRequestType_Store_Conditional) {
461 seq_req.
pkt->
req->setExtraData(success ? 1 : 0);
467 if (seq_req.
m_type == RubyRequestType_Locked_RMW_Read) {
475 }
else if (seq_req.
m_type == RubyRequestType_Locked_RMW_Write) {
481 initialRequestTime, forwardRequestTime,
488 initialRequestTime, forwardRequestTime,
489 firstResponseTime, !ruby_request);
490 ruby_request =
false;
493 assert(!ruby_request);
497 initialRequestTime, forwardRequestTime,
498 firstResponseTime, !ruby_request);
500 seq_req_list.pop_front();
504 if (seq_req_list.empty()) {
511 bool externalHit,
const MachineType mach,
512 Cycles initialRequestTime,
513 Cycles forwardRequestTime,
527 bool ruby_request =
true;
528 int aliased_loads = 0;
529 while (!seq_req_list.empty()) {
532 assert((seq_req.
m_type == RubyRequestType_LD) ||
533 (seq_req.
m_type == RubyRequestType_Load_Linked) ||
534 (seq_req.
m_type == RubyRequestType_IFETCH));
538 if ((seq_req.
m_type != RubyRequestType_LD) &&
539 (seq_req.
m_type != RubyRequestType_Load_Linked) &&
540 (seq_req.
m_type != RubyRequestType_IFETCH)) {
547 initialRequestTime, forwardRequestTime,
552 initialRequestTime, forwardRequestTime,
553 firstResponseTime, !ruby_request);
554 ruby_request =
false;
555 seq_req_list.pop_front();
559 if (seq_req_list.empty()) {
567 const MachineType mach,
const bool externalHit,
568 const Cycles initialRequestTime,
569 const Cycles forwardRequestTime,
570 const Cycles firstResponseTime,
571 const bool was_coalesced)
573 warn_once(
"Replacement policy updates recently became the responsibility "
574 "of SLICC state machines. Make sure to setMRU() near callbacks "
590 if (
type == RubyRequestType_Load_Linked) {
599 if ((
type == RubyRequestType_LD) ||
600 (
type == RubyRequestType_IFETCH) ||
601 (
type == RubyRequestType_RMW_Read) ||
602 (
type == RubyRequestType_Locked_RMW_Read) ||
603 (
type == RubyRequestType_Load_Linked)) {
607 }
else if (pkt->
req->isSwap()) {
613 data.setData(&overwrite_val[0],
616 }
else if (
type != RubyRequestType_Store_Conditional || llscSuccess) {
628 DPRINTF(RubySequencer,
"hitCallback %s 0x%x using RubyTester\n",
632 assert(testerSenderState);
640 rs->m_cache_recorder->enqueueNextFetchRequest();
643 rs->m_cache_recorder->enqueueNextFlushRequest();
662 !pkt->
req->isHTMAbort()) {
663 return RequestStatus_BufferFull;
666 RubyRequestType primary_type = RubyRequestType_NULL;
667 RubyRequestType secondary_type = RubyRequestType_NULL;
681 DPRINTF(RubySequencer,
"Issuing SC\n");
682 primary_type = RubyRequestType_Store_Conditional;
683 #if defined (PROTOCOL_MESI_Three_Level) || defined (PROTOCOL_MESI_Three_Level_HTM)
684 secondary_type = RubyRequestType_Store_Conditional;
686 secondary_type = RubyRequestType_ST;
689 DPRINTF(RubySequencer,
"Issuing LL\n");
691 primary_type = RubyRequestType_Load_Linked;
692 secondary_type = RubyRequestType_LD;
694 }
else if (pkt->
req->isLockedRMW()) {
702 DPRINTF(RubySequencer,
"Issuing Locked RMW Write\n");
703 primary_type = RubyRequestType_Locked_RMW_Write;
705 DPRINTF(RubySequencer,
"Issuing Locked RMW Read\n");
707 primary_type = RubyRequestType_Locked_RMW_Read;
709 secondary_type = RubyRequestType_ST;
720 primary_type = secondary_type = RubyRequestType_ST;
721 }
else if (pkt->
isRead()) {
723 if (pkt->
req->isHTMCmd()) {
725 }
else if (pkt->
req->isInstFetch()) {
726 primary_type = secondary_type = RubyRequestType_IFETCH;
728 bool storeCheck =
false;
731 uint32_t flags = pkt->
req->getFlags();
736 primary_type = RubyRequestType_RMW_Read;
737 secondary_type = RubyRequestType_ST;
739 primary_type = secondary_type = RubyRequestType_LD;
743 primary_type = secondary_type = RubyRequestType_FLUSH;
745 panic(
"Unsupported ruby packet type\n");
751 (primary_type != RubyRequestType_Locked_RMW_Write)) {
755 return RequestStatus_Aliased;
761 if (
status != RequestStatus_Ready &&
status != RequestStatus_Aliased)
765 if (
status != RequestStatus_Aliased)
769 return RequestStatus_Issued;
783 if (pkt->
req->hasPC()) {
784 pc = pkt->
req->getPC();
789 std::shared_ptr<RubyRequest> msg =
792 RubyAccessMode_Supervisor, pkt,
793 PrefetchBit_No, proc_id, core_id);
795 DPRINTFR(ProtocolTrace,
"%15s %3s %10s%20s %6s>%-6s %#x %s\n",
798 RubyRequestType_to_string(secondary_type));
804 msg->m_htmFromTransaction =
true;
816 template <
class KEY,
class VALUE>
818 operator<<(std::ostream &out,
const std::unordered_map<KEY, VALUE> &map)
820 for (
const auto &table_entry : map) {
821 out <<
"[ " << table_entry.first <<
" =";
822 for (
const auto &seq_req : table_entry.second) {
823 out <<
" " << RubyRequestType_to_string(seq_req.m_second_type);
842 DPRINTF(RubyStats,
"Recorded statistic: %s\n",
843 SequencerRequestType_to_string(requestType));