Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

i#3068: API for accessing cache model metrics #5020

Merged
merged 10 commits into from
Aug 5, 2021
41 changes: 41 additions & 0 deletions clients/drcachesim/simulator/cache_simulator.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -594,6 +594,47 @@ cache_simulator_t::print_results()
return true;
}

// All valid metrics are returned as a positive number.
// Negative return value is an error and is of type stats_error_t.
int_least64_t
cache_simulator_t::get_cache_metric(metric_name_t metric, unsigned level, unsigned core,
cache_split_t split) const
{
caching_device_t *curr_cache;

if (core >= knobs_.num_cores) {
return STATS_ERROR_WRONG_CORE_NUMBER;
}

if (split == cache_split_t::DATA) {
curr_cache = l1_dcaches_[core];
} else {
curr_cache = l1_icaches_[core];
}

for (size_t i = 1; i < level; i++) {
caching_device_t *parent = curr_cache->get_parent();

if (parent == NULL) {
return STATS_ERROR_WRONG_CACHE_LEVEL;
}
curr_cache = parent;
}
caching_device_stats_t *stats = curr_cache->get_stats();

if (stats == NULL) {
return STATS_ERROR_NO_CACHE_STATS;
}

return stats->get_metric(metric);
}

const cache_simulator_knobs_t &
cache_simulator_t::get_knobs() const
{
return knobs_;
}

cache_t *
cache_simulator_t::create_cache(const std::string &policy)
{
Expand Down
21 changes: 21 additions & 0 deletions clients/drcachesim/simulator/cache_simulator.h
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,20 @@
#include "cache_stats.h"
#include "cache.h"
#include "snoop_filter.h"
#include <limits.h>

enum class cache_split_t { DATA, INSTRUCTION };

// Error codes returned when passing wrong parameters to the
// get_cache_metric function.
typedef enum {
// Core number is larger then congifured number of cores.
STATS_ERROR_WRONG_CORE_NUMBER = INT_MIN,
// Cache level is larger then configured number of levels.
STATS_ERROR_WRONG_CACHE_LEVEL,
// Given cache doesn't support counting statistics.
STATS_ERROR_NO_CACHE_STATS,
} stats_error_t;

class cache_simulator_t : public simulator_t {
public:
Expand All @@ -60,12 +74,19 @@ class cache_simulator_t : public simulator_t {
bool
print_results() override;

int_least64_t
get_cache_metric(metric_name_t metric, unsigned level, unsigned core = 0,
derekbruening marked this conversation as resolved.
Show resolved Hide resolved
cache_split_t split = cache_split_t::DATA) const;

// Exposed to make it easy to test
bool
check_warmed_up();
uint64_t
remaining_sim_refs() const;

const cache_simulator_knobs_t &
get_knobs() const;

protected:
// Create a cache_t object with a specific replacement policy.
virtual cache_t *
Expand Down
3 changes: 3 additions & 0 deletions clients/drcachesim/simulator/cache_stats.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,9 @@ cache_stats_t::cache_stats_t(const std::string &miss_file, bool warmup_enabled,
, num_prefetch_hits_(0)
, num_prefetch_misses_(0)
{
stats_map_.emplace(metric_name_t::FLUSHES, num_flushes_);
stats_map_.emplace(metric_name_t::PREFETCH_HITS, num_prefetch_hits_);
stats_map_.emplace(metric_name_t::PREFETCH_MISSES, num_prefetch_misses_);
}

void
Expand Down
9 changes: 9 additions & 0 deletions clients/drcachesim/simulator/caching_device_stats.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,15 @@ caching_device_stats_t::caching_device_stats_t(const std::string &miss_file,
} else
dump_misses_ = true;
}

stats_map_.emplace(metric_name_t::HITS, num_hits_);
stats_map_.emplace(metric_name_t::MISSES, num_misses_);
stats_map_.emplace(metric_name_t::HITS_AT_RESET, num_hits_at_reset_);
stats_map_.emplace(metric_name_t::MISSES_AT_RESET, num_misses_at_reset_);
stats_map_.emplace(metric_name_t::CHILD_HITS_AT_RESET, num_child_hits_at_reset_);
stats_map_.emplace(metric_name_t::CHILD_HITS, num_child_hits_);
stats_map_.emplace(metric_name_t::INCLUSIVE_INVALIDATES, num_inclusive_invalidates_);
stats_map_.emplace(metric_name_t::COHERENCE_INVALIDATES, num_coherence_invalidates_);
}

caching_device_stats_t::~caching_device_stats_t()
Expand Down
30 changes: 30 additions & 0 deletions clients/drcachesim/simulator/caching_device_stats.h
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@

#include "caching_device_block.h"
#include <string>
#include <map>
#include <stdint.h>
#ifdef HAS_ZLIB
# include <zlib.h>
Expand All @@ -49,6 +50,20 @@ enum invalidation_type_t {
INVALIDATION_COHERENCE,
};

enum class metric_name_t {
HITS,
MISSES,
HITS_AT_RESET,
MISSES_AT_RESET,
CHILD_HITS,
CHILD_HITS_AT_RESET,
INCLUSIVE_INVALIDATES,
COHERENCE_INVALIDATES,
PREFETCH_HITS,
PREFETCH_MISSES,
FLUSHES
};

class caching_device_stats_t {
public:
explicit caching_device_stats_t(const std::string &miss_file,
Expand Down Expand Up @@ -81,6 +96,17 @@ class caching_device_stats_t {
virtual void
invalidate(invalidation_type_t invalidation_type);

int_least64_t
get_metric(metric_name_t metric) const
{
if (stats_map_.find(metric) != stats_map_.end()) {
return stats_map_.at(metric);
} else {
ERRMSG("Wrong metric name.\n");
return 0;
}
}

protected:
bool success_;

Expand Down Expand Up @@ -115,6 +141,10 @@ class caching_device_stats_t {
// Print out write invalidations if cache is coherent.
bool is_coherent_;

// References to the properties with statistics are held in the map with the
// statistic name as the key. Sample map element: {HITS, num_hits_}
std::map<metric_name_t, int_least64_t &> stats_map_;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add some kind of test. Maybe in drcachesim_unit_tests or a new similar test? Xref #4842.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have added unit test in the drcachesim_unit_tests. I couldn't test Invalidates metric properly because of the bug/issue that I described here: #5031

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK, best to put a XXX i#5031: or TODO i#5031: comment in the test about adding that metric.

// We provide a feature of dumping misses to a file.
bool dump_misses_;
#ifdef HAS_ZLIB
Expand Down
72 changes: 72 additions & 0 deletions clients/drcachesim/tests/drcachesim_unit_tests.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
// Unit tests for drcachesim
#include <iostream>
#include <cstdlib>
#include <assert.h>
#include "simulator/cache_simulator.h"
#include "../common/memref.h"

Expand Down Expand Up @@ -135,9 +136,80 @@ unit_test_sim_refs()
}
}

void
unit_test_metrics_API()
{
cache_simulator_knobs_t knobs = make_test_knobs();
cache_simulator_t cache_sim(knobs);

memref_t ref;
ref.data.type = TRACE_TYPE_WRITE;
ref.data.addr = 0;
ref.data.size = 8;

// Currently invalidates are not counted properly in the configuration of
// cache_simulator_t with cache_simulator_knobs_t.
// TODO i#5031: Test invalidates metric when the issue is solved.
for (int i = 0; i < 4; i++) {
if (!cache_sim.process_memref(ref)) {
std::cerr << "drcachesim unit_test_metrics_API failed: "
<< cache_sim.get_error_string() << "\n";
exit(1);
}
}
assert(cache_sim.get_cache_metric(metric_name_t::MISSES, 1, 0, cache_split_t::DATA) ==
1);
assert(cache_sim.get_cache_metric(metric_name_t::HITS, 1, 0, cache_split_t::DATA) ==
3);

ref.data.type = TRACE_TYPE_INSTR;

for (int i = 0; i < 4; i++) {
if (!cache_sim.process_memref(ref)) {
std::cerr << "drcachesim unit_test_metrics_API failed: "
<< cache_sim.get_error_string() << "\n";
exit(1);
}
}
assert(cache_sim.get_cache_metric(metric_name_t::MISSES, 1, 0,
cache_split_t::INSTRUCTION) == 1);
assert(cache_sim.get_cache_metric(metric_name_t::HITS, 1, 0,
cache_split_t::INSTRUCTION) == 3);

assert(cache_sim.get_cache_metric(metric_name_t::MISSES, 2) == 1);
assert(cache_sim.get_cache_metric(metric_name_t::HITS, 2) == 1);

ref.data.type = TRACE_TYPE_PREFETCH;
ref.data.addr += 64;

for (int i = 0; i < 4; i++) {
if (!cache_sim.process_memref(ref)) {
std::cerr << "drcachesim unit_test_metrics_API failed: "
<< cache_sim.get_error_string() << "\n";
exit(1);
}
}
assert(cache_sim.get_cache_metric(metric_name_t::PREFETCH_MISSES, 1, 0,
cache_split_t::DATA) == 1);
assert(cache_sim.get_cache_metric(metric_name_t::PREFETCH_HITS, 1, 0,
cache_split_t::DATA) == 3);

ref.data.type = TRACE_TYPE_DATA_FLUSH;

for (int i = 0; i < 4; i++) {
if (!cache_sim.process_memref(ref)) {
std::cerr << "drcachesim unit_test_metrics_API failed: "
<< cache_sim.get_error_string() << "\n";
exit(1);
}
}
assert(cache_sim.get_cache_metric(metric_name_t::FLUSHES, 2) == 4);
}

int
main(int argc, const char *argv[])
{
unit_test_metrics_API();
unit_test_warmup_fraction();
unit_test_warmup_refs();
unit_test_sim_refs();
Expand Down