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],
 
  620             DPRINTF(RubySequencer, 
"AMO original data %s\n", 
data);
 
  625         } 
else if (
type != RubyRequestType_Store_Conditional || llscSuccess) {
 
  637         DPRINTF(RubySequencer, 
"hitCallback %s 0x%x using RubyTester\n",
 
  641         assert(testerSenderState);
 
  649         rs->m_cache_recorder->enqueueNextFetchRequest();
 
  652         rs->m_cache_recorder->enqueueNextFlushRequest();
 
  671         !pkt->
req->isHTMAbort()) {
 
  672         return RequestStatus_BufferFull;
 
  675     RubyRequestType primary_type = RubyRequestType_NULL;
 
  676     RubyRequestType secondary_type = RubyRequestType_NULL;
 
  690             DPRINTF(RubySequencer, 
"Issuing SC\n");
 
  691             primary_type = RubyRequestType_Store_Conditional;
 
  692 #if defined (PROTOCOL_MESI_Three_Level) || defined (PROTOCOL_MESI_Three_Level_HTM) 
  693             secondary_type = RubyRequestType_Store_Conditional;
 
  695             secondary_type = RubyRequestType_ST;
 
  698             DPRINTF(RubySequencer, 
"Issuing LL\n");
 
  700             primary_type = RubyRequestType_Load_Linked;
 
  701             secondary_type = RubyRequestType_LD;
 
  703     } 
else if (pkt->
req->isLockedRMW()) {
 
  711             DPRINTF(RubySequencer, 
"Issuing Locked RMW Write\n");
 
  712             primary_type = RubyRequestType_Locked_RMW_Write;
 
  714             DPRINTF(RubySequencer, 
"Issuing Locked RMW Read\n");
 
  716             primary_type = RubyRequestType_Locked_RMW_Read;
 
  718         secondary_type = RubyRequestType_ST;
 
  729             primary_type = secondary_type = RubyRequestType_ST;
 
  730         } 
else if (pkt->
isRead()) {
 
  732             if (pkt->
req->isHTMCmd()) {
 
  734             } 
else if (pkt->
req->isInstFetch()) {
 
  735                 primary_type = secondary_type = RubyRequestType_IFETCH;
 
  737                 if (pkt->
req->isReadModifyWrite()) {
 
  738                     primary_type = RubyRequestType_RMW_Read;
 
  739                     secondary_type = RubyRequestType_ST;
 
  741                     primary_type = secondary_type = RubyRequestType_LD;
 
  745           primary_type = secondary_type = RubyRequestType_FLUSH;
 
  747             panic(
"Unsupported ruby packet type\n");
 
  753         (primary_type != RubyRequestType_Locked_RMW_Write)) {
 
  757         return RequestStatus_Aliased;
 
  763     if (
status != RequestStatus_Ready && 
status != RequestStatus_Aliased)
 
  767     if (
status != RequestStatus_Aliased)
 
  771     return RequestStatus_Issued;
 
  785     if (pkt->
req->hasPC()) {
 
  786         pc = pkt->
req->getPC();
 
  791     std::shared_ptr<RubyRequest> msg =
 
  794                                       RubyAccessMode_Supervisor, pkt,
 
  795                                       PrefetchBit_No, proc_id, core_id);
 
  797     DPRINTFR(ProtocolTrace, 
"%15s %3s %10s%20s %6s>%-6s %#x %s\n",
 
  800             RubyRequestType_to_string(secondary_type));
 
  806         msg->m_htmFromTransaction = 
true;
 
  818 template <
class KEY, 
class VALUE>
 
  820 operator<<(std::ostream &out, 
const std::unordered_map<KEY, VALUE> &map)
 
  822     for (
const auto &table_entry : map) {
 
  823         out << 
"[ " << table_entry.first << 
" =";
 
  824         for (
const auto &seq_req : table_entry.second) {
 
  825             out << 
" " << RubyRequestType_to_string(seq_req.m_second_type);
 
  844     DPRINTF(RubyStats, 
"Recorded statistic: %s\n",
 
  845             SequencerRequestType_to_string(requestType));