46#include "debug/DRAM.hh" 
   47#include "debug/DRAMPower.hh" 
   48#include "debug/DRAMState.hh" 
   59std::pair<MemPacketQueue::iterator, Tick>
 
   65    bool filled_earliest_banks = 
false;
 
   67    bool hidden_bank_prep = 
false;
 
   74    bool found_hidden_bank = 
false;
 
   78    bool found_prepped_pkt = 
false;
 
   82    bool found_earliest_pkt = 
false;
 
   85    auto selected_pkt_it = queue.end();
 
   87    for (
auto i = queue.begin(); 
i != queue.end() ; ++
i) {
 
   96            DPRINTF(DRAM, 
"%s checking DRAM packet in bank %d, row %d\n",
 
  104                        "%s bank %d - Rank %d available\n", __func__,
 
  112                    if (col_allowed_at <= min_col_at) {
 
  117                        DPRINTF(DRAM, 
"%s Seamless buffer hit\n", __func__);
 
  119                        selected_col_at = col_allowed_at;
 
  122                    } 
else if (!found_hidden_bank && !found_prepped_pkt) {
 
  128                        selected_col_at = col_allowed_at;
 
  129                        found_prepped_pkt = 
true;
 
  130                        DPRINTF(DRAM, 
"%s Prepped row buffer hit\n", __func__);
 
  132                } 
else if (!found_earliest_pkt) {
 
  135                    if (!filled_earliest_banks) {
 
  137                        std::tie(earliest_banks, hidden_bank_prep) =
 
  139                        filled_earliest_banks = 
true;
 
  147                        found_earliest_pkt = 
true;
 
  148                        found_hidden_bank = hidden_bank_prep;
 
  154                        if (hidden_bank_prep || !found_prepped_pkt) {
 
  156                            selected_col_at = col_allowed_at;
 
  161                DPRINTF(DRAM, 
"%s bank %d - Rank %d not available\n", __func__,
 
  167    if (selected_pkt_it == queue.end()) {
 
  168        DPRINTF(DRAM, 
"%s no available DRAM ranks found\n", __func__);
 
  171    return std::make_pair(selected_pkt_it, selected_col_at);
 
 
  176                       Tick act_tick, uint32_t row)
 
  188    DPRINTF(DRAM, 
"Activate at tick %d\n", act_at);
 
  191    assert(bank_ref.
openRow == Bank::NO_ROW);
 
  203    DPRINTF(DRAM, 
"Activate bank %d, rank %d at tick %lld, now got " 
  204            "%d active\n", bank_ref.
bank, rank_ref.
rank, act_at,
 
  228            rank_ref.
banks[
i].actAllowedAt = std::max(act_at + 
tRRD_L,
 
  229                                             rank_ref.
banks[
i].actAllowedAt);
 
  234            rank_ref.
banks[
i].actAllowedAt = std::max(act_at + 
tRRD,
 
  235                                             rank_ref.
banks[
i].actAllowedAt);
 
  245            panic(
"Got %d activates in window %d (%llu - %llu) which " 
  256        rank_ref.
actTicks.push_front(act_at);
 
  263            DPRINTF(DRAM, 
"Enforcing tXAW with X = %d, next activate " 
  268                rank_ref.
banks[j].actAllowedAt =
 
  270                             rank_ref.
banks[j].actAllowedAt);
 
 
  285                             bool auto_or_preall, 
bool trace)
 
  288    assert(bank.
openRow != Bank::NO_ROW);
 
  296    Tick pre_at = pre_tick;
 
  297    if (auto_or_preall) {
 
  307            rank_ref.
banks[
i].preAllowedAt = std::max(pre_at + 
tPPD,
 
  308                                             rank_ref.
banks[
i].preAllowedAt);
 
  312    Tick pre_done_at = pre_at + 
tRP;
 
  319    DPRINTF(DRAM, 
"Precharging bank %d, rank %d at tick %lld, now got " 
  320            "%d active\n", bank.
bank, rank_ref.
rank, pre_at,
 
 
  350    DPRINTF(DRAM, 
"Timing access to addr %#x, rank/bank/row %d %d %d\n",
 
  379        if (bank_ref.
openRow != Bank::NO_ROW) {
 
  393    const Tick col_allowed_at = mem_pkt->
isRead() ?
 
  398    Tick cmd_at = std::max({col_allowed_at, next_burst_at, 
curTick()});
 
  424    DPRINTF(DRAM, 
"Schedule RD/WR burst at tick %d\n", cmd_at);
 
  441            if (mem_pkt->
rank == j) {
 
  449                    dly_to_rd_cmd = mem_pkt->
isRead() ?
 
  451                    dly_to_wr_cmd = mem_pkt->
isRead() ?
 
  457                    dly_to_rd_cmd = mem_pkt->
isRead() ? burst_gap :
 
  469            ranks[j]->banks[
i].rdAllowedAt = std::max(cmd_at + dly_to_rd_cmd,
 
  470                                             ranks[j]->banks[
i].rdAllowedAt);
 
  471            ranks[j]->banks[
i].wrAllowedAt = std::max(cmd_at + dly_to_wr_cmd,
 
  472                                             ranks[j]->banks[
i].wrAllowedAt);
 
  491    bool auto_precharge = 
pageMgmt == enums::close ||
 
  496    if (!auto_precharge &&
 
  497        (
pageMgmt == enums::open_adaptive ||
 
  498         pageMgmt == enums::close_adaptive)) {
 
  507        bool got_more_hits = 
false;
 
  508        bool got_bank_conflict = 
false;
 
  510        for (uint8_t 
i = 0; 
i < 
ctrl->numPriorities(); ++
i) {
 
  511            auto p = queue[
i].begin();
 
  520            while (!got_more_hits && 
p != queue[
i].end()) {
 
  528                if (mem_pkt != (*
p)) {
 
  529                    bool same_rank_bank = (mem_pkt->
rank == (*p)->rank) &&
 
  530                                          (mem_pkt->
bank == (*p)->bank);
 
  532                    bool same_row = mem_pkt->
row == (*p)->row;
 
  533                    got_more_hits |= same_rank_bank && same_row;
 
  534                    got_bank_conflict |= same_rank_bank && !same_row;
 
  547        auto_precharge = !got_more_hits &&
 
  548            (got_bank_conflict || 
pageMgmt == enums::close_adaptive);
 
  552    std::string mem_cmd = mem_pkt->
isRead() ? 
"RD" : 
"WR";
 
  555    MemCommand::cmds command = (mem_cmd == 
"RD") ? MemCommand::RD :
 
  565    if (auto_precharge) {
 
  571        DPRINTF(DRAM, 
"Auto-precharged bank: %d\n", mem_pkt->
bankId);
 
  610            stats.writeRowHits++;
 
  616    return std::make_pair(cmd_at, cmd_at + burst_gap);
 
 
  629                                             n->banks[
i].rdAllowedAt);
 
  631                                             n->banks[
i].wrAllowedAt);
 
 
  663    DPRINTF(DRAM, 
"Setting up DRAM Interface\n");
 
  674        DPRINTF(DRAM, 
"Creating DRAM rank %d \n", 
i);
 
  676        ranks.push_back(rank);
 
  685    DPRINTF(DRAM, 
"Memory capacity %lld (%lld) bytes\n", capacity,
 
  689    if (deviceCapacity != capacity / (1024 * 1024))
 
  690        warn(
"DRAM device capacity (%d Mbytes) does not match the " 
  691             "address range assigned (%d Mbytes)\n", deviceCapacity,
 
  692             capacity / (1024 * 1024));
 
  694    DPRINTF(DRAM, 
"Row buffer size %d bytes with %d bursts per row buffer\n",
 
  701        fatal(
"tREFI (%d) must be larger than tRP (%d) and tRFC (%d)\n",
 
  709            fatal(
"banks per rank (%d) must be equal to or larger than " 
  710                  "banks groups per rank (%d)\n",
 
  715            fatal(
"Banks per rank (%d) must be evenly divisible by bank " 
  716                  "groups per rank (%d) for equal banks per bank group\n",
 
  721            fatal(
"tCCD_L (%d) should be larger than the minimum bus delay " 
  722                  "(%d) when bank groups per rank (%d) is greater than 1\n",
 
  727            fatal(
"tCCD_L_WR (%d) should be larger than the minimum bus delay " 
  728                  " (%d) when bank groups per rank (%d) is greater than 1\n",
 
  734            fatal(
"tRRD_L (%d) should be larger than tRRD (%d) when " 
  735                  "bank groups per rank (%d) is greater than 1\n",
 
 
  748    if (
range.interleaved()) {
 
  751                fatal(
"Channel interleaving of %s doesn't match RoRaBaChCo " 
  752                      "address map\n", 
name());
 
  767            if (
system()->cacheLineSize() > 
range.granularity()) {
 
  768                fatal(
"Channel interleaving of %s must be at least as large " 
  769                      "as the cache line size\n", 
name());
 
  774                fatal(
"Channel interleaving of %s must be at most as large " 
  775                      "as the row-buffer size\n", 
name());
 
 
  786    if (
system()->isTimingMode()) {
 
 
  801        if (!
r->inRefIdleState()) {
 
  804                DPRINTF(DRAMState, 
"Rank %d is not available\n", 
r->rank);
 
  814            if ((
r->pwrState == 
PWR_SREF) && 
r->inLowPowerState) {
 
  815                DPRINTF(DRAMState, 
"Rank %d is in self-refresh\n", 
r->rank);
 
  819                if (
r->forceSelfRefreshExit()) {
 
  820                    DPRINTF(DRAMState, 
"rank %d was in self refresh and" 
  821                           " should wake up\n", 
r->rank);
 
  823                    r->scheduleWakeUpEvent(
tXS);
 
 
  836                       unsigned size, 
bool is_read, uint8_t pseudo_channel)
 
  901        panic(
"Unknown address mapping policy chosen!");
 
  906    assert(row < Bank::NO_ROW);
 
  908    DPRINTF(DRAM, 
"Address: %#x Rank %d Bank %d Row %d\n",
 
  909            pkt_addr, rank, bank, row);
 
  916    return new MemPacket(pkt, is_read, 
true, pseudo_channel, rank, bank, row,
 
  917                   bank_id, pkt_addr, 
size);
 
 
  924        ++
ranks[rank]->readEntries;
 
  926        ++
ranks[rank]->writeEntries;
 
 
  939    DPRINTF(DRAM, 
"number of read entries for rank %d is %d\n",
 
  963        DPRINTF(DRAMState, 
"Rank %d sleep at tick %d; current power state is " 
 
  997            DPRINTF(DRAM,
"Rank%d: Forcing self-refresh wakeup in drain\n",
 
  999            r->scheduleWakeUpEvent(
tXS);
 
 
 1008    bool all_ranks_drained = 
true;
 
 1013        all_ranks_drained = 
r->inPwrIdleState() && 
r->inRefIdleState() &&
 
 1016    return all_ranks_drained;
 
 
 1029                      Tick min_col_at)
 const 
 1035    bool found_seamless_bank = 
false;
 
 1039    bool hidden_bank_prep = 
false;
 
 1044    for (
const auto& 
p : queue) {
 
 1047        if (
p->isDram() && 
ranks[
p->rank]->inRefIdleState())
 
 1048            got_waiting[
p->bankId] = 
true;
 
 1059            if (got_waiting[bank_id]) {
 
 1061                assert(
ranks[
i]->inRefIdleState());
 
 1065                Tick act_at = 
ranks[
i]->banks[j].openRow == Bank::NO_ROW ?
 
 1071                const Tick tRCD = 
ctrl->inReadBusState(
false, 
this) ?
 
 1073                const Tick hidden_act_max =
 
 1074                            std::max(min_col_at - tRCD, 
curTick());
 
 1077                const Tick col_allowed_at = 
ctrl->inReadBusState(
false,
 
 1079                                              ranks[
i]->banks[j].rdAllowedAt :
 
 1080                                              ranks[
i]->banks[j].wrAllowedAt;
 
 1081                Tick col_at = std::max(col_allowed_at, act_at + tRCD);
 
 1085                bool new_seamless_bank = col_at <= min_col_at;
 
 1090                if (new_seamless_bank ||
 
 1091                    (!found_seamless_bank && act_at <= min_act_at)) {
 
 1097                    if (!found_seamless_bank &&
 
 1098                        (new_seamless_bank || act_at < min_act_at)) {
 
 1099                        std::fill(bank_mask.begin(), bank_mask.end(), 0);
 
 1102                    found_seamless_bank |= new_seamless_bank;
 
 1105                    hidden_bank_prep = act_at <= hidden_act_max;
 
 1109                    min_act_at = act_at;
 
 1115    return std::make_pair(bank_mask, hidden_bank_prep);
 
 
 1128      activateEvent([
this]{ processActivateEvent(); }, 
name()),
 
 1129      prechargeEvent([
this]{ processPrechargeEvent(); }, 
name()),
 
 1130      refreshEvent([
this]{ processRefreshEvent(); }, 
name()),
 
 1131      powerEvent([
this]{ processPowerEvent(); }, 
name()),
 
 1132      wakeUpEvent([
this]{ processWakeUpEvent(); }, 
name()),
 
 1135    for (
int b = 0; 
b < _p.banks_per_rank; 
b++) {
 
 1140        if (_p.bank_groups_per_rank > 0) {
 
 1148            banks[
b].bankgr = 
b % _p.bank_groups_per_rank;
 
 1151            banks[
b].bankgr = 
b;
 
 
 1184    bool no_queued_cmds = (
dram.ctrl->inReadBusState(
true, &(this->
dram))
 
 1186                          (
dram.ctrl->inWriteBusState(
true, &(this->
dram))
 
 1188    return no_queued_cmds;
 
 
 1197        DPRINTF(DRAM, 
"Refresh drain done, now precharging\n");
 
 
 1213    auto next_iter = 
cmdList.begin();
 
 1215    for ( ; next_iter != 
cmdList.end() ; ++next_iter) {
 
 1221                                      dram.timeStampOffset);
 
 
 1259            dram.enableDRAMPowerdown) {
 
 1264            DPRINTF(DRAMState, 
"Rank %d sleep at tick %d\n",
 
 
 1301        DPRINTF(DRAM, 
"Refresh due\n");
 
 1311            && (
dram.ctrl->requestEventScheduled(
dram.pseudoChannel))) {
 
 1314            DPRINTF(DRAM, 
"Refresh awaiting draining\n");
 
 1326            DPRINTF(DRAM, 
"Wake Up for refresh\n");
 
 1341            DPRINTF(DRAM, 
"Precharging all\n");
 
 1350                pre_at = std::max(
b.preAllowedAt, pre_at);
 
 1355            Tick act_allowed_at = pre_at + 
dram.tRP;
 
 1358                if (
b.openRow != Bank::NO_ROW) {
 
 1359                    dram.prechargeBank(*
this, 
b, pre_at, 
true, 
false);
 
 1361                    b.actAllowedAt = std::max(
b.actAllowedAt, act_allowed_at);
 
 1362                    b.preAllowedAt = std::max(
b.preAllowedAt, pre_at);
 
 1375            DPRINTF(DRAM, 
"All banks already precharged, starting refresh\n");
 
 1385                   dram.ctrl->respondEventScheduled(
dram.pseudoChannel));
 
 1410            b.actAllowedAt = ref_done_at;
 
 1428            fatal(
"Refresh was delayed so long we cannot catch up\n");
 
 1456                DPRINTF(DRAMState, 
"Rank %d sleeping after refresh and was in " 
 1457                        "power state %d before refreshing\n", 
rank,
 
 1467                DPRINTF(DRAMState, 
"Rank %d sleeping after refresh but was NOT" 
 1468                        " in a low power state before refreshing\n", 
rank);
 
 1486        DPRINTF(DRAMState, 
"Refresh done at %llu and next refresh" 
 
 1498        DPRINTF(DRAMState, 
"Scheduling power event at %llu to state %d\n",
 
 1506        panic(
"Scheduled power event at %llu to state %d, " 
 1507              "with scheduled event at %llu to %d\n", 
tick, pwr_state,
 
 
 1533    } 
else if (pwr_state == 
PWR_REF) {
 
 1543    } 
else if (pwr_state == 
PWR_SREF) {
 
 
 1568    DPRINTF(DRAMState, 
"Scheduling wake-up for rank %d at tick %d\n",
 
 1569            rank, wake_up_tick);
 
 1588        b.wrAllowedAt = std::max(wake_up_tick + exit_delay, 
b.wrAllowedAt);
 
 1589        b.rdAllowedAt = std::max(wake_up_tick + exit_delay, 
b.rdAllowedAt);
 
 1590        b.preAllowedAt = std::max(wake_up_tick + exit_delay, 
b.preAllowedAt);
 
 1591        b.actAllowedAt = std::max(wake_up_tick + exit_delay, 
b.actAllowedAt);
 
 1600        cmdList.push_back(
Command(MemCommand::PUP_ACT, 0, wake_up_tick));
 
 1605        cmdList.push_back(
Command(MemCommand::PUP_PRE, 0, wake_up_tick));
 
 
 1642    stats.pwrStateTime[prev_state] += duration;
 
 1647        stats.totalIdleTime += duration;
 
 1662        DPRINTF(DRAMState, 
"Was refreshing for %llu ticks\n", duration);
 
 1666            DPRINTF(DRAMState, 
"Switching to power down state after refreshing" 
 1671        if (!(
dram.ctrl->requestEventScheduled(
dram.pseudoChannel))) {
 
 1672            DPRINTF(DRAM, 
"Scheduling next request after refreshing" 
 1673                           " rank %d, PC %d \n", 
rank, 
dram.pseudoChannel);
 
 1686        DPRINTF(DRAMState, 
"All banks precharged\n");
 
 1735           dram.enableDRAMPowerdown) {
 
 1736            DPRINTF(DRAMState, 
"Rank %d bypassing refresh and transitioning " 
 1737                    "to self refresh at %11u tick\n", 
rank, 
curTick());
 
 1749            DPRINTF(DRAMState, 
"Refreshing\n");
 
 
 1778                                    dram.timeStampOffset);
 
 1781    Data::MemoryPowerModel::Energy energy = 
power.powerlib.getEnergy();
 
 1785    stats.actEnergy += energy.act_energy * 
dram.devicesPerRank;
 
 1786    stats.preEnergy += energy.pre_energy * 
dram.devicesPerRank;
 
 1787    stats.readEnergy += energy.read_energy * 
dram.devicesPerRank;
 
 1788    stats.writeEnergy += energy.write_energy * 
dram.devicesPerRank;
 
 1789    stats.refreshEnergy += energy.ref_energy * 
dram.devicesPerRank;
 
 1790    stats.actBackEnergy += energy.act_stdby_energy * 
dram.devicesPerRank;
 
 1791    stats.preBackEnergy += energy.pre_stdby_energy * 
dram.devicesPerRank;
 
 1792    stats.actPowerDownEnergy += energy.f_act_pd_energy * 
dram.devicesPerRank;
 
 1793    stats.prePowerDownEnergy += energy.f_pre_pd_energy * 
dram.devicesPerRank;
 
 1794    stats.selfRefreshEnergy += energy.sref_energy * 
dram.devicesPerRank;
 
 1797    stats.totalEnergy += energy.window_energy * 
dram.devicesPerRank;
 
 1804    stats.averagePower = (
stats.totalEnergy.value() /
 
 
 1812    DPRINTF(DRAM,
"Computing stats due to a dump callback\n");
 
 
 1828                                    dram.timeStampOffset);
 
 
 1835           (
dram.ctrl->inWriteBusState(
true, &(this->
dram))
 
 
 1850             "Number of DRAM read bursts"),
 
 1852             "Number of DRAM write bursts"),
 
 1855             "Per bank write bursts"),
 
 1857             "Per bank write bursts"),
 
 1860             "Total ticks spent queuing"),
 
 1862             "Total ticks spent in databus transfers"),
 
 1864             "Total ticks spent from burst creation until serviced " 
 1869             "Average queueing delay per DRAM burst"),
 
 1872             "Average bus latency per DRAM burst"),
 
 1875             "Average memory access latency per DRAM burst"),
 
 1878             "Number of row buffer hits during reads"),
 
 1880             "Number of row buffer hits during writes"),
 
 1882             "Row buffer hit rate for reads"),
 
 1884             "Row buffer hit rate for writes"),
 
 1887             "Bytes accessed per row activation"),
 
 1889            "Total bytes read"),
 
 1891            "Total bytes written"),
 
 1895             "Average DRAM read bandwidth in MiBytes/s"),
 
 1898             "Average DRAM write bandwidth in MiBytes/s"),
 
 1901             "Theoretical peak bandwidth in MiByte/s"),
 
 1904             "Data bus utilization in percentage"),
 
 1906             "Data bus utilization in percentage for reads"),
 
 1908             "Data bus utilization in percentage for writes"),
 
 1911             "Row buffer hit rate, read and write combined")
 
 
 1932        .init(
dram.maxAccessesPerRow ?
 
 1933              dram.maxAccessesPerRow : 
dram.rowBufferSize)
 
 1954              dram.bytesPerBurst() / 1000000;
 
 
 1969             "Energy for activate commands per rank (pJ)"),
 
 1971             "Energy for precharge commands per rank (pJ)"),
 
 1973             "Energy for read commands per rank (pJ)"),
 
 1975             "Energy for write commands per rank (pJ)"),
 
 1977             "Energy for refresh commands per rank (pJ)"),
 
 1979             "Energy for active background per rank (pJ)"),
 
 1981             "Energy for precharge background per rank (pJ)"),
 
 1983             "Energy for active power-down per rank (pJ)"),
 
 1985             "Energy for precharge power-down per rank (pJ)"),
 
 1987             "Energy for self refresh per rank (pJ)"),
 
 1990             "Total energy per rank (pJ)"),
 
 1992             "Core power per rank (mW)"),
 
 1995             "Total Idle time Per DRAM Rank"),
 
 1997             "Time in different power states")
 
 
 2011        .subname(3, 
"PRE_PDN")
 
 2013        .subname(5, 
"ACT_PDN");
 
 
 2029    rank.computeStats();
 
 
DRAMPower is a standalone tool which calculates the power consumed by a DRAM in the system.
virtual std::string name() const
uint64_t size() const
Get the memory size.
System * system() const
read the system pointer Implemented for completeness with the setter
Rank class includes a vector of banks.
void computeStats()
Computes stats just prior to dump event.
void processWriteDoneEvent()
void powerDownSleep(PowerState pwr_state, Tick tick)
Schedule a transition to power-down (sleep)
void checkDrainDone()
Let the rank check if it was waiting for requests to drain to allow it to transition states.
DRAMInterface & dram
A reference to the parent DRAMInterface instance.
EventFunctionWrapper refreshEvent
void resetStats()
Reset stats on a stats event.
DRAMPower power
One DRAMPower instance per rank.
uint8_t outstandingEvents
Number of ACT, RD, and WR events currently scheduled Incremented when a refresh event is started as w...
bool inRefIdleState() const
Check if there is no refresh and no preparation of refresh ongoing i.e.
PowerState pwrState
Current power state.
void processRefreshEvent()
EventFunctionWrapper activateEvent
std::deque< Tick > actTicks
List to keep track of activate ticks.
void processActivateEvent()
bool inLowPowerState
rank is in or transitioning to power-down or self-refresh
EventFunctionWrapper prechargeEvent
uint32_t readEntries
Track number of packets in read queue going to this rank.
RefreshState refreshState
current refresh state
Tick pwrStateTick
Track when we transitioned to the current power state.
PowerState pwrStatePostRefresh
Previous low-power state, which will be re-entered after refresh.
PowerState pwrStateTrans
Since we are taking decisions out of order, we need to keep track of what power transition is happeni...
void processPrechargeEvent()
EventFunctionWrapper powerEvent
unsigned int numBanksActive
To track number of banks which are currently active for this rank.
void startup(Tick ref_tick)
Kick off accounting for power and refresh states and schedule initial refresh.
std::vector< Command > cmdList
List of commands issued, to be sent to DRAMPpower at refresh and stats dump.
std::vector< Bank > banks
Vector of Banks.
Tick lastBurstTick
Track when we issued the last read/write burst.
void scheduleWakeUpEvent(Tick exit_delay)
schedule and event to wake-up from power-down or self-refresh and update bank timing parameters
Tick wakeUpAllowedAt
delay low-power exit until this requirement is met
Tick refreshDueAt
Keep track of when a refresh is due.
void processWakeUpEvent()
void schedulePowerEvent(PowerState pwr_state, Tick tick)
Schedule a power state transition in the future, and potentially override an already scheduled transi...
uint8_t rank
Current Rank index.
EventFunctionWrapper wakeUpEvent
void suspend()
Stop the refresh events.
void updatePowerStats()
Function to update Power Stats.
EventFunctionWrapper writeDoneEvent
bool isQueueEmpty() const
Check if the command queue of current rank is idle.
Rank(const DRAMInterfaceParams &_p, int _rank, DRAMInterface &_dram)
bool forceSelfRefreshExit() const
Trigger a self-refresh exit if there are entries enqueued Exit if there are any read entries regardle...
uint32_t writeEntries
Track number of packets in write queue going to this rank.
void flushCmdList()
Push command out of cmdList queue that are scheduled at or before curTick() to DRAMPower library All ...
std::vector< Rank * > ranks
Vector of dram ranks.
const Tick clkResyncDelay
void checkRefreshState(uint8_t rank) override
Check the refresh state to determine if refresh needs to be kicked back into action after a read resp...
void respondEvent(uint8_t rank) override
Complete response process for DRAM when read burst is complete This will update the counters and chec...
const uint32_t maxAccessesPerRow
Max column accesses (read and write) per row, before forefully closing it.
Tick lastStatsResetTick
The time when stats were last reset used to calculate average power.
const Tick rdToWrDlySameBG
bool allRanksDrained() const override
Return true once refresh is complete for all ranks and there are no additional commands enqueued.
void addRankToRankDelay(Tick cmd_at) override
Add rank to rank delay to bus timing to all DRAM banks in alli ranks when access to an alternate inte...
std::pair< Tick, Tick > doBurstAccess(MemPacket *mem_pkt, Tick next_burst_at, const std::vector< MemPacketQueue > &queue) override
Actually do the burst - figure out the latency it will take to service the req based on bank state,...
void drainRanks() override
Iterate through dram ranks to exit self-refresh in order to drain.
const bool burstInterleave
bool isBusy(bool read_queue_empty, bool all_writes_nvm) override
This function checks if ranks are actively refreshing and therefore busy.
void prechargeBank(Rank &rank_ref, Bank &bank_ref, Tick pre_tick, bool auto_or_preall=false, bool trace=true)
Precharge a given bank and also update when the precharge is done.
const uint32_t bankGroupsPerRank
DRAM specific device characteristics.
void activateBank(Rank &rank_ref, Bank &bank_ref, Tick act_tick, uint32_t row)
Keep track of when row activations happen, in order to enforce the maximum number of activations in t...
std::pair< MemPacketQueue::iterator, Tick > chooseNextFRFCFS(MemPacketQueue &queue, Tick min_col_at) const override
For FR-FCFS policy, find first DRAM command that can issue.
const uint32_t activationLimit
PowerState
The power state captures the different operational states of the DRAM and interacts with the bus read...
const Tick wrToRdDlySameBG
MemPacket * decodePacket(const PacketPtr pkt, Addr pkt_addr, unsigned int size, bool is_read, uint8_t pseudo_channel=0) override
Address decoder to figure out physical mapping onto ranks, banks, and rows.
const Tick tRL
DRAM specific timing requirements.
void startup() override
Iterate through dram ranks and instantiate per rank startup routine.
bool enableDRAMPowerdown
Enable or disable DRAM powerdown states.
Tick writeToReadDelay() const override
std::pair< std::vector< uint32_t >, bool > minBankPrep(const MemPacketQueue &queue, Tick min_col_at) const
Find which are the earliest banks ready to issue an activate for the enqueued requests.
static bool sortTime(const Command &cmd, const Command &cmd_next)
Function for sorting Command structures based on timeStamp.
void init() override
Initialize the DRAM interface and verify parameters.
enums::PageManage pageMgmt
DRAMInterface(const DRAMInterfaceParams &_p)
void suspend() override
Iterate through DRAM ranks and suspend them.
void setupRank(const uint8_t rank, const bool is_read) override
Setup the rank based on packet received.
const uint8_t twoCycleActivate
bool burstReady(MemPacket *pkt) const override
Check if a burst operation can be issued to the DRAM.
A basic class to track the bank state, i.e.
const uint32_t burstsPerStripe
enums::AddrMap addrMapping
Memory controller configuration initialized based on parameter values.
const uint32_t devicesPerRank
MemCtrl * ctrl
A pointer to the parent memory controller instance.
const uint32_t ranksPerChannel
Addr getCtrlAddr(Addr addr)
Get an address in a dense range which starts from 0.
uint8_t pseudoChannel
pseudo channel number used for HBM modeling
const uint64_t deviceSize
Tick rankToRankDelay() const
const uint32_t burstSize
General device and channel characteristics The rowsPerBank is determined based on the capacity,...
const uint32_t banksPerRank
unsigned int maxCommandsPerWindow
Number of commands that can issue in the defined controller command window, used to verify command ba...
const uint32_t burstsPerRowBuffer
GEM5_CLASS_VAR_USED const Tick tCK
General timing requirements.
const uint32_t rowBufferSize
MemInterface(const Params &_p)
Tick readToWriteDelay() const
A memory packet stores packets along with the timestamp of when the packet entered the queue,...
Tick readyTime
When will request leave the controller.
const uint8_t pseudoChannel
pseudo channel num
const uint16_t bankId
Bank id is calculated considering banks in all the ranks eg: 2 ranks each with 8 banks,...
Addr addr
The starting address of the packet.
bool isDram() const
Return true if its a DRAM access.
bool isRead() const
Return true if its a read packet (interface compatibility with Packet)
const Tick entryTime
When did request enter the controller.
const uint8_t rank
Will be populated by address decoder.
DRAMInterface declaration.
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
static constexpr int ceilLog2(const T &n)
static constexpr bool isPowerOf2(const T &n)
static constexpr T divCeil(const T &a, const U &b)
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
constexpr void replaceBits(T &val, unsigned first, unsigned last, B bit_val)
A convenience function to replace bits first to last of val with bit_val in place.
@ Draining
Draining buffers pending serialization/handover.
@ Drained
Buffers drained, ready for serialization/handover.
void deschedule(Event &event)
bool scheduled() const
Determine if the current event is scheduled.
void schedule(Event &event, Tick when)
EventManager(EventManager &em)
Event manger manages events in the event queue.
void reschedule(Event &event, Tick when, bool always=false)
Tick when() const
Get the time that the event is scheduled.
#define panic(...)
This implements a cprintf based panic() function.
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
#define fatal(...)
This implements a cprintf based fatal() function.
virtual void init()
init() is called after all C++ SimObjects have been created and all ports are connected.
virtual void regStats()
Callback to set stat parameters.
virtual void preDumpStats()
Callback before stats are dumped.
virtual void resetStats()
Callback to reset stats.
std::deque< MemPacket * > MemPacketQueue
Tick Frequency
The simulated frequency of curTick(). (In ticks per second)
const FlagsType nozero
Don't print if this is zero.
Copyright (c) 2024 Arm Limited All rights reserved.
Tick curTick()
The universal simulation clock.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
uint64_t Tick
Tick count type.
std::string csprintf(const char *format, const Args &...args)
statistics::Formula & simSeconds
Simple structure to hold the values needed to keep track of commands for DRAMPower.
Data::MemCommand::cmds type
statistics::Formula avgQLat
statistics::Formula avgRdBW
statistics::Scalar dramBytesWritten
statistics::Formula writeRowHitRate
statistics::Formula pageHitRate
statistics::Scalar totBusLat
statistics::Vector perBankRdBursts
DRAM per bank stats.
statistics::Scalar writeRowHits
statistics::Formula peakBW
statistics::Formula busUtilRead
statistics::Formula avgWrBW
void resetStats() override
Callback to reset stats.
statistics::Formula avgMemAccLat
statistics::Histogram bytesPerActivate
void regStats() override
Callback to set stat parameters.
statistics::Formula busUtilWrite
statistics::Scalar readBursts
total number of DRAM bursts serviced
statistics::Scalar readRowHits
statistics::Vector perBankWrBursts
DRAMStats(DRAMInterface &dram)
statistics::Scalar dramBytesRead
statistics::Scalar totQLat
statistics::Formula avgBusLat
statistics::Formula readRowHitRate
statistics::Scalar writeBursts
statistics::Scalar totMemAccLat
statistics::Formula busUtil
void resetStats() override
Callback to reset stats.
statistics::Scalar actBackEnergy
void preDumpStats() override
Callback before stats are dumped.
statistics::Scalar selfRefreshEnergy
statistics::Scalar actEnergy
statistics::Scalar totalEnergy
statistics::Scalar preEnergy
statistics::Vector pwrStateTime
Track time spent in each power state.
statistics::Scalar writeEnergy
void regStats() override
Callback to set stat parameters.
statistics::Scalar actPowerDownEnergy
statistics::Scalar preBackEnergy
statistics::Scalar readEnergy
statistics::Scalar averagePower
statistics::Scalar prePowerDownEnergy
statistics::Scalar refreshEnergy
statistics::Scalar totalIdleTime
Stat to track total DRAM idle time.
RankStats(DRAMInterface &dram, Rank &rank)