gem5  v20.0.0.3
RubySystem.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2019 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  * Copyright (c) 1999-2011 Mark D. Hill and David A. Wood
15  * All rights reserved.
16  *
17  * Redistribution and use in source and binary forms, with or without
18  * modification, are permitted provided that the following conditions are
19  * met: redistributions of source code must retain the above copyright
20  * notice, this list of conditions and the following disclaimer;
21  * redistributions in binary form must reproduce the above copyright
22  * notice, this list of conditions and the following disclaimer in the
23  * documentation and/or other materials provided with the distribution;
24  * neither the name of the copyright holders nor the names of its
25  * contributors may be used to endorse or promote products derived from
26  * this software without specific prior written permission.
27  *
28  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
31  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
32  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
33  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
34  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
35  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
36  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
37  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
38  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
39  */
40 
42 
43 #include <fcntl.h>
44 #include <zlib.h>
45 
46 #include <cstdio>
47 #include <list>
48 
49 #include "base/intmath.hh"
50 #include "base/statistics.hh"
51 #include "debug/RubyCacheTrace.hh"
52 #include "debug/RubySystem.hh"
57 #include "mem/simple_mem.hh"
58 #include "sim/eventq.hh"
59 #include "sim/simulate.hh"
60 
61 using namespace std;
62 
67 bool RubySystem::m_warmup_enabled = false;
68 // To look forward to allowing multiple RubySystem instances, track the number
69 // of RubySystems that need to be warmed up on checkpoint restore.
72 
74  : ClockedObject(p), m_access_backing_store(p->access_backing_store),
75  m_cache_recorder(NULL)
76 {
77  m_randomization = p->randomization;
78 
79  m_block_size_bytes = p->block_size_bytes;
82  m_memory_size_bits = p->memory_size_bits;
83 
84  // Resize to the size of different machine types
85  m_abstract_controls.resize(MachineType_NUM);
86 
87  // Collate the statistics before they are printed.
89  // Create the profiler
90  m_profiler = new Profiler(p, this);
91  m_phys_mem = p->phys_mem;
92 }
93 
94 void
96 {
97  m_network = network_ptr;
98 }
99 
100 void
102 {
103  m_abs_cntrl_vec.push_back(cntrl);
104 
105  MachineID id = cntrl->getMachineID();
106  m_abstract_controls[id.getType()][id.getNum()] = cntrl;
107 }
108 
110 {
111  delete m_network;
112  delete m_profiler;
113 }
114 
115 void
116 RubySystem::makeCacheRecorder(uint8_t *uncompressed_trace,
117  uint64_t cache_trace_size,
118  uint64_t block_size_bytes)
119 {
120  vector<Sequencer*> sequencer_map;
121  Sequencer* sequencer_ptr = NULL;
122 
123  for (int cntrl = 0; cntrl < m_abs_cntrl_vec.size(); cntrl++) {
124  sequencer_map.push_back(m_abs_cntrl_vec[cntrl]->getCPUSequencer());
125  if (sequencer_ptr == NULL) {
126  sequencer_ptr = sequencer_map[cntrl];
127  }
128  }
129 
130  assert(sequencer_ptr != NULL);
131 
132  for (int cntrl = 0; cntrl < m_abs_cntrl_vec.size(); cntrl++) {
133  if (sequencer_map[cntrl] == NULL) {
134  sequencer_map[cntrl] = sequencer_ptr;
135  }
136  }
137 
138  // Remove the old CacheRecorder if it's still hanging about.
139  if (m_cache_recorder != NULL) {
140  delete m_cache_recorder;
141  }
142 
143  // Create the CacheRecorder and record the cache trace
144  m_cache_recorder = new CacheRecorder(uncompressed_trace, cache_trace_size,
145  sequencer_map, block_size_bytes);
146 }
147 
148 void
150 {
151  m_cooldown_enabled = true;
152 
153  // Make the trace so we know what to write back.
154  DPRINTF(RubyCacheTrace, "Recording Cache Trace\n");
156  for (int cntrl = 0; cntrl < m_abs_cntrl_vec.size(); cntrl++) {
157  m_abs_cntrl_vec[cntrl]->recordCacheTrace(cntrl, m_cache_recorder);
158  }
159  DPRINTF(RubyCacheTrace, "Cache Trace Complete\n");
160 
161  // save the current tick value
162  Tick curtick_original = curTick();
163  DPRINTF(RubyCacheTrace, "Recording current tick %ld\n", curtick_original);
164 
165  // Deschedule all prior events on the event queue, but record the tick they
166  // were scheduled at so they can be restored correctly later.
167  list<pair<Event*, Tick> > original_events;
168  while (!eventq->empty()) {
169  Event *curr_head = eventq->getHead();
170  if (curr_head->isAutoDelete()) {
171  DPRINTF(RubyCacheTrace, "Event %s auto-deletes when descheduled,"
172  " not recording\n", curr_head->name());
173  } else {
174  original_events.push_back(make_pair(curr_head, curr_head->when()));
175  }
176  eventq->deschedule(curr_head);
177  }
178 
179  // Schedule an event to start cache cooldown
180  DPRINTF(RubyCacheTrace, "Starting cache flush\n");
182  simulate();
183  DPRINTF(RubyCacheTrace, "Cache flush complete\n");
184 
185  // Deschedule any events left on the event queue.
186  while (!eventq->empty()) {
188  }
189 
190  // Restore curTick
191  setCurTick(curtick_original);
192 
193  // Restore all events that were originally on the event queue. This is
194  // done after setting curTick back to its original value so that events do
195  // not seem to be scheduled in the past.
196  while (!original_events.empty()) {
197  pair<Event*, Tick> event = original_events.back();
198  eventq->schedule(event.first, event.second);
199  original_events.pop_back();
200  }
201 
202  // No longer flushing back to memory.
203  m_cooldown_enabled = false;
204 
205  // There are several issues with continuing simulation after calling
206  // memWriteback() at the moment, that stem from taking events off the
207  // queue, simulating again, and then putting them back on, whilst
208  // pretending that no time has passed. One is that some events will have
209  // been deleted, so can't be put back. Another is that any object
210  // recording the tick something happens may end up storing a tick in the
211  // future. A simple warning here alerts the user that things may not work
212  // as expected.
213  warn_once("Ruby memory writeback is experimental. Continuing simulation "
214  "afterwards may not always work as intended.");
215 
216  // Keep the cache recorder around so that we can dump the trace if a
217  // checkpoint is immediately taken.
218 }
219 
220 void
221 RubySystem::writeCompressedTrace(uint8_t *raw_data, string filename,
222  uint64_t uncompressed_trace_size)
223 {
224  // Create the checkpoint file for the memory
225  string thefile = CheckpointIn::dir() + "/" + filename.c_str();
226 
227  int fd = creat(thefile.c_str(), 0664);
228  if (fd < 0) {
229  perror("creat");
230  fatal("Can't open memory trace file '%s'\n", filename);
231  }
232 
233  gzFile compressedMemory = gzdopen(fd, "wb");
234  if (compressedMemory == NULL)
235  fatal("Insufficient memory to allocate compression state for %s\n",
236  filename);
237 
238  if (gzwrite(compressedMemory, raw_data, uncompressed_trace_size) !=
239  uncompressed_trace_size) {
240  fatal("Write failed on memory trace file '%s'\n", filename);
241  }
242 
243  if (gzclose(compressedMemory)) {
244  fatal("Close failed on memory trace file '%s'\n", filename);
245  }
246  delete[] raw_data;
247 }
248 
249 void
251 {
252  // Store the cache-block size, so we are able to restore on systems with a
253  // different cache-block size. CacheRecorder depends on the correct
254  // cache-block size upon unserializing.
255  uint64_t block_size_bytes = getBlockSizeBytes();
256  SERIALIZE_SCALAR(block_size_bytes);
257 
258  // Check that there's a valid trace to use. If not, then memory won't be
259  // up-to-date and the simulation will probably fail when restoring from the
260  // checkpoint.
261  if (m_cache_recorder == NULL) {
262  fatal("Call memWriteback() before serialize() to create ruby trace");
263  }
264 
265  // Aggregate the trace entries together into a single array
266  uint8_t *raw_data = new uint8_t[4096];
267  uint64_t cache_trace_size = m_cache_recorder->aggregateRecords(&raw_data,
268  4096);
269  string cache_trace_file = name() + ".cache.gz";
270  writeCompressedTrace(raw_data, cache_trace_file, cache_trace_size);
271 
272  SERIALIZE_SCALAR(cache_trace_file);
273  SERIALIZE_SCALAR(cache_trace_size);
274 }
275 
276 void
278 {
279  // Delete the cache recorder if it was created in memWriteback()
280  // to checkpoint the current cache state.
281  if (m_cache_recorder) {
282  delete m_cache_recorder;
283  m_cache_recorder = NULL;
284  }
285 }
286 
287 void
288 RubySystem::readCompressedTrace(string filename, uint8_t *&raw_data,
289  uint64_t &uncompressed_trace_size)
290 {
291  // Read the trace file
292  gzFile compressedTrace;
293 
294  // trace file
295  int fd = open(filename.c_str(), O_RDONLY);
296  if (fd < 0) {
297  perror("open");
298  fatal("Unable to open trace file %s", filename);
299  }
300 
301  compressedTrace = gzdopen(fd, "rb");
302  if (compressedTrace == NULL) {
303  fatal("Insufficient memory to allocate compression state for %s\n",
304  filename);
305  }
306 
307  raw_data = new uint8_t[uncompressed_trace_size];
308  if (gzread(compressedTrace, raw_data, uncompressed_trace_size) <
309  uncompressed_trace_size) {
310  fatal("Unable to read complete trace from file %s\n", filename);
311  }
312 
313  if (gzclose(compressedTrace)) {
314  fatal("Failed to close cache trace file '%s'\n", filename);
315  }
316 }
317 
318 void
320 {
321  uint8_t *uncompressed_trace = NULL;
322 
323  // This value should be set to the checkpoint-system's block-size.
324  // Optional, as checkpoints without it can be run if the
325  // checkpoint-system's block-size == current block-size.
326  uint64_t block_size_bytes = getBlockSizeBytes();
327  UNSERIALIZE_OPT_SCALAR(block_size_bytes);
328 
329  string cache_trace_file;
330  uint64_t cache_trace_size = 0;
331 
332  UNSERIALIZE_SCALAR(cache_trace_file);
333  UNSERIALIZE_SCALAR(cache_trace_size);
334  cache_trace_file = cp.getCptDir() + "/" + cache_trace_file;
335 
336  readCompressedTrace(cache_trace_file, uncompressed_trace,
337  cache_trace_size);
338  m_warmup_enabled = true;
340 
341  // Create the cache recorder that will hang around until startup.
342  makeCacheRecorder(uncompressed_trace, cache_trace_size, block_size_bytes);
343 }
344 
345 void
347 {
348 
349  // Ruby restores state from a checkpoint by resetting the clock to 0 and
350  // playing the requests that can possibly re-generate the cache state.
351  // The clock value is set to the actual checkpointed value once all the
352  // requests have been executed.
353  //
354  // This way of restoring state is pretty finicky. For example, if a
355  // Ruby component reads time before the state has been restored, it would
356  // cache this value and hence its clock would not be reset to 0, when
357  // Ruby resets the global clock. This can potentially result in a
358  // deadlock.
359  //
360  // The solution is that no Ruby component should read time before the
361  // simulation starts. And then one also needs to hope that the time
362  // Ruby finishes restoring the state is less than the time when the
363  // state was checkpointed.
364 
365  if (m_warmup_enabled) {
366  DPRINTF(RubyCacheTrace, "Starting ruby cache warmup\n");
367  // save the current tick value
368  Tick curtick_original = curTick();
369  // save the event queue head
370  Event* eventq_head = eventq->replaceHead(NULL);
371  // set curTick to 0 and reset Ruby System's clock
372  setCurTick(0);
373  resetClock();
374 
375  // Schedule an event to start cache warmup
377  simulate();
378 
379  delete m_cache_recorder;
380  m_cache_recorder = NULL;
382  if (m_systems_to_warmup == 0) {
383  m_warmup_enabled = false;
384  }
385 
386  // Restore eventq head
387  eventq->replaceHead(eventq_head);
388  // Restore curTick and Ruby System's clock
389  setCurTick(curtick_original);
390  resetClock();
391  }
392 
393  resetStats();
394 }
395 
396 void
398 {
399  if (getWarmupEnabled()) {
401  } else if (getCooldownEnabled()) {
403  }
404 }
405 
406 void
408 {
410 }
411 
412 bool
414 {
415  Addr address(pkt->getAddr());
416  Addr line_address = makeLineAddress(address);
417 
418  AccessPermission access_perm = AccessPermission_NotPresent;
419  int num_controllers = m_abs_cntrl_vec.size();
420 
421  DPRINTF(RubySystem, "Functional Read request for %#x\n", address);
422 
423  unsigned int num_ro = 0;
424  unsigned int num_rw = 0;
425  unsigned int num_busy = 0;
426  unsigned int num_maybe_stale = 0;
427  unsigned int num_backing_store = 0;
428  unsigned int num_invalid = 0;
429 
430  AbstractController *ctrl_ro = nullptr;
431  AbstractController *ctrl_rw = nullptr;
432  AbstractController *ctrl_backing_store = nullptr;
433 
434  // In this loop we count the number of controllers that have the given
435  // address in read only, read write and busy states.
436  for (unsigned int i = 0; i < num_controllers; ++i) {
437  access_perm = m_abs_cntrl_vec[i]-> getAccessPermission(line_address);
438  if (access_perm == AccessPermission_Read_Only){
439  num_ro++;
440  if (ctrl_ro == nullptr) ctrl_ro = m_abs_cntrl_vec[i];
441  }
442  else if (access_perm == AccessPermission_Read_Write){
443  num_rw++;
444  if (ctrl_rw == nullptr) ctrl_rw = m_abs_cntrl_vec[i];
445  }
446  else if (access_perm == AccessPermission_Busy)
447  num_busy++;
448  else if (access_perm == AccessPermission_Maybe_Stale)
449  num_maybe_stale++;
450  else if (access_perm == AccessPermission_Backing_Store) {
451  // See RubySlicc_Exports.sm for details, but Backing_Store is meant
452  // to represent blocks in memory *for Broadcast/Snooping protocols*,
453  // where memory has no idea whether it has an exclusive copy of data
454  // or not.
455  num_backing_store++;
456  if (ctrl_backing_store == nullptr)
457  ctrl_backing_store = m_abs_cntrl_vec[i];
458  }
459  else if (access_perm == AccessPermission_Invalid ||
460  access_perm == AccessPermission_NotPresent)
461  num_invalid++;
462  }
463 
464  // This if case is meant to capture what happens in a Broadcast/Snoop
465  // protocol where the block does not exist in the cache hierarchy. You
466  // only want to read from the Backing_Store memory if there is no copy in
467  // the cache hierarchy, otherwise you want to try to read the RO or RW
468  // copies existing in the cache hierarchy (covered by the else statement).
469  // The reason is because the Backing_Store memory could easily be stale, if
470  // there are copies floating around the cache hierarchy, so you want to read
471  // it only if it's not in the cache hierarchy at all.
472  if (num_invalid == (num_controllers - 1) && num_backing_store == 1) {
473  DPRINTF(RubySystem, "only copy in Backing_Store memory, read from it\n");
474  ctrl_backing_store->functionalRead(line_address, pkt);
475  return true;
476  } else if (num_ro > 0 || num_rw >= 1) {
477  if (num_rw > 1) {
478  // We iterate over the vector of abstract controllers, and return
479  // the first copy found. If we have more than one cache with block
480  // in writable permission, the first one found would be returned.
481  warn("More than one Abstract Controller with RW permission for "
482  "addr: %#x on cacheline: %#x.", address, line_address);
483  }
484  // In Broadcast/Snoop protocols, this covers if you know the block
485  // exists somewhere in the caching hierarchy, then you want to read any
486  // valid RO or RW block. In directory protocols, same thing, you want
487  // to read any valid readable copy of the block.
488  DPRINTF(RubySystem, "num_maybe_stale=%d, num_busy = %d, num_ro = %d, "
489  "num_rw = %d\n",
490  num_maybe_stale, num_busy, num_ro, num_rw);
491  // Use the copy from the controller with read/write permission (if
492  // any), otherwise use get the first read only found
493  if (ctrl_rw) {
494  ctrl_rw->functionalRead(line_address, pkt);
495  } else {
496  assert(ctrl_ro);
497  ctrl_ro->functionalRead(line_address, pkt);
498  }
499  return true;
500  } else if ((num_busy + num_maybe_stale) > 0) {
501  // No controller has a valid copy of the block, but a transient or
502  // stale state indicates a valid copy should be in transit in the
503  // network or in a message buffer waiting to be handled
504  DPRINTF(RubySystem, "Controllers functionalRead lookup "
505  "(num_maybe_stale=%d, num_busy = %d)\n",
506  num_maybe_stale, num_busy);
507  for (unsigned int i = 0; i < num_controllers;++i) {
508  if (m_abs_cntrl_vec[i]->functionalReadBuffers(pkt))
509  return true;
510  }
511  DPRINTF(RubySystem, "Network functionalRead lookup "
512  "(num_maybe_stale=%d, num_busy = %d)\n",
513  num_maybe_stale, num_busy);
514  if (m_network->functionalRead(pkt))
515  return true;
516  }
517 
518  return false;
519 }
520 
521 // The function searches through all the buffers that exist in different
522 // cache, directory and memory controllers, and in the network components
523 // and writes the data portion of those that hold the address specified
524 // in the packet.
525 bool
527 {
528  Addr addr(pkt->getAddr());
529  Addr line_addr = makeLineAddress(addr);
530  AccessPermission access_perm = AccessPermission_NotPresent;
531  int num_controllers = m_abs_cntrl_vec.size();
532 
533  DPRINTF(RubySystem, "Functional Write request for %#x\n", addr);
534 
535  uint32_t M5_VAR_USED num_functional_writes = 0;
536 
537  for (unsigned int i = 0; i < num_controllers;++i) {
538  num_functional_writes +=
539  m_abs_cntrl_vec[i]->functionalWriteBuffers(pkt);
540 
541  access_perm = m_abs_cntrl_vec[i]->getAccessPermission(line_addr);
542  if (access_perm != AccessPermission_Invalid &&
543  access_perm != AccessPermission_NotPresent) {
544  num_functional_writes +=
545  m_abs_cntrl_vec[i]->functionalWrite(line_addr, pkt);
546  }
547 
548  // Also updates requests pending in any sequencer associated
549  // with the controller
550  if (m_abs_cntrl_vec[i]->getCPUSequencer()) {
551  num_functional_writes +=
552  m_abs_cntrl_vec[i]->getCPUSequencer()->functionalWrite(pkt);
553  }
554  if (m_abs_cntrl_vec[i]->getDMASequencer()) {
555  num_functional_writes +=
556  m_abs_cntrl_vec[i]->getDMASequencer()->functionalWrite(pkt);
557  }
558  }
559 
560  num_functional_writes += m_network->functionalWrite(pkt);
561  DPRINTF(RubySystem, "Messages written = %u\n", num_functional_writes);
562 
563  return true;
564 }
565 
566 RubySystem *
567 RubySystemParams::create()
568 {
569  return new RubySystem(this);
570 }
EventQueue * eventq
A pointer to this object&#39;s event queue.
Definition: eventq.hh:909
void enqueueNextFetchRequest()
Function for fetching warming up the memory and the caches.
#define DPRINTF(x,...)
Definition: trace.hh:225
Network * m_network
Definition: RubySystem.hh:133
static void readCompressedTrace(std::string filename, uint8_t *&raw_data, uint64_t &uncompressed_trace_size)
Definition: RubySystem.cc:288
static void writeCompressedTrace(uint8_t *raw_data, std::string file, uint64_t uncompressed_trace_size)
Definition: RubySystem.cc:221
GlobalSimLoopExitEvent * simulate(Tick num_cycles)
Simulate for num_cycles additional cycles.
Definition: simulate.cc:80
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: RubySystem.cc:250
bool isAutoDelete() const
Definition: eventq.hh:492
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:171
SimpleMemory * m_phys_mem
Definition: RubySystem.hh:130
bool empty() const
Returns true if no events are queued.
Definition: eventq.hh:823
Bitfield< 7 > i
static bool m_randomization
Definition: RubySystem.hh:122
STL pair class.
Definition: stl.hh:58
void makeCacheRecorder(uint8_t *uncompressed_trace, uint64_t cache_trace_size, uint64_t block_size_bytes)
Definition: RubySystem.cc:116
CacheRecorder * m_cache_recorder
Definition: RubySystem.hh:139
static uint32_t m_block_size_bytes
Definition: RubySystem.hh:123
RubySystem(const Params *p)
Definition: RubySystem.cc:73
ip6_addr_t addr
Definition: inet.hh:330
uint64_t aggregateRecords(uint8_t **data, uint64_t size)
bool functionalRead(Packet *ptr)
Definition: RubySystem.cc:413
static bool m_warmup_enabled
Definition: RubySystem.hh:127
virtual bool functionalRead(Packet *pkt)
Definition: Network.hh:118
void schedule(Event *event, Tick when, bool global=false)
Schedule the given event on this queue.
Definition: eventq_impl.hh:38
void deschedule(Event *event)
Deschedule the specified event.
Definition: eventq_impl.hh:65
Overload hash function for BasicBlockRange type.
Definition: vec_reg.hh:587
virtual void functionalRead(const Addr &addr, PacketPtr)=0
Profiler * m_profiler
Definition: RubySystem.hh:138
Definition: cprintf.cc:40
bool functionalWrite(Packet *ptr)
Definition: RubySystem.cc:526
void drainResume() override
Resume execution after a successful drain.
Definition: RubySystem.cc:277
Declaration of Statistics objects.
void enqueueRubyEvent(Tick tick)
Definition: RubySystem.hh:97
SimpleMemory declaration.
STL vector class.
Definition: stl.hh:37
std::enable_if< std::is_integral< T >::value, int >::type floorLog2(T x)
Definition: intmath.hh:57
static bool getWarmupEnabled()
Definition: RubySystem.hh:62
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:770
void resetClock() const
Reset the object&#39;s clock using the current global tick value.
RubySystemParams Params
Definition: RubySystem.hh:53
Tick curTick()
The current simulated tick.
Definition: core.hh:44
void enqueueNextFlushRequest()
Function for flushing the memory contents of the caches to the main memory.
virtual const std::string name() const
Definition: eventq.cc:82
void registerNetwork(Network *)
Definition: RubySystem.cc:95
MachineID getMachineID() const
uint64_t Tick
Tick count type.
Definition: types.hh:61
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
#define UNSERIALIZE_OPT_SCALAR(scalar)
Definition: serialize.hh:777
std::vector< std::map< uint32_t, AbstractController * > > m_abstract_controls
Definition: RubySystem.hh:140
Addr getAddr() const
Definition: packet.hh:720
Cycles curCycle() const
Determine the current cycle, corresponding to a tick aligned to a clock edge.
bool isPowerOf2(const T &n)
Definition: intmath.hh:90
void startup() override
startup() is the final initialization call before simulation.
Definition: RubySystem.cc:346
STL list class.
Definition: stl.hh:51
void setCurTick(Tick newVal)
Definition: eventq.hh:992
void memWriteback() override
Write back dirty buffers to memory using functional writes.
Definition: RubySystem.cc:149
static bool getCooldownEnabled()
Definition: RubySystem.hh:63
static uint32_t m_memory_size_bits
Definition: RubySystem.hh:125
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
Addr makeLineAddress(Addr addr)
Definition: Address.cc:54
Bitfield< 10, 5 > event
#define warn_once(...)
Definition: logging.hh:212
void resetStats() override
Callback to reset stats.
Definition: RubySystem.cc:407
static uint32_t m_block_size_bits
Definition: RubySystem.hh:124
std::vector< AbstractController * > m_abs_cntrl_vec
Definition: RubySystem.hh:134
void registerDumpCallback(Callback *cb)
Register a callback that should be called whenever statistics are about to be dumped.
Definition: statistics.cc:589
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:763
static std::string dir()
Get the current checkout directory name.
Definition: serialize.cc:263
Event * replaceHead(Event *s)
function for replacing the head of the event queue, so that a different set of events can run without...
Definition: eventq.cc:354
Cycles m_start_cycle
Definition: RubySystem.hh:135
virtual const std::string name() const
Definition: sim_object.hh:129
virtual uint32_t functionalWrite(Packet *pkt)
Definition: Network.hh:120
std::ostream CheckpointOut
Definition: serialize.hh:63
Definition: eventq.hh:245
const std::string getCptDir()
Definition: serialize.hh:85
void processRubyEvent()
Definition: RubySystem.cc:397
#define warn(...)
Definition: logging.hh:208
Event * getHead() const
Definition: eventq.hh:783
static bool m_cooldown_enabled
Definition: RubySystem.hh:129
Bitfield< 14, 12 > fd
Definition: types.hh:158
Bitfield< 0 > p
Tick when() const
Get the time that the event is scheduled.
Definition: eventq.hh:499
static unsigned m_systems_to_warmup
Definition: RubySystem.hh:128
static uint32_t getBlockSizeBytes()
Definition: RubySystem.hh:59
void registerAbstractController(AbstractController *)
Definition: RubySystem.cc:101
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: RubySystem.cc:319

Generated on Fri Jul 3 2020 15:53:04 for gem5 by doxygen 1.8.13