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#5994 opcode_mix, part1: add categories to opcode_mix output #6512

Merged
merged 36 commits into from
Feb 14, 2024
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
dcd301e
i#5994 opcode_mix, part1: add categories to opcode_mix output
edeiana Dec 17, 2023
d9220f4
Caching category along with opcode to avoid calling decode on all
edeiana Dec 20, 2023
3880ea5
Moved logic that orders category names into the client opcode_mix.
edeiana Jan 15, 2024
76c9286
Added test for instruction categories.
edeiana Jan 17, 2024
0ea1e3b
Created a more meaningful structure than an std::pair
edeiana Jan 22, 2024
0fcdff1
To get the category names in opcode_mix, we now loop through all
edeiana Jan 22, 2024
bd7483a
Added check in CHECK_CATEGORY for the new api:
edeiana Jan 22, 2024
1e2238b
Added the new instr_get_category_name() API to changelist.
edeiana Jan 22, 2024
d2a239b
Fixed clang-format 9.0.0 issue. Reverted back.
edeiana Jan 23, 2024
b296c68
Fixed formatting. Line over 90 chars long.
edeiana Jan 23, 2024
5abee0e
Added tests for multi-category instruction (OP_fwait)
edeiana Jan 23, 2024
a461230
Print statements fixed, from fprintf to print, since
edeiana Jan 23, 2024
a179535
Merge branch 'master' into i5994-category-opcode_mix
edeiana Jan 30, 2024
4188a6f
Fixed bug in CHECK_CATEGORY when using strncmp().
edeiana Jan 30, 2024
65fb699
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 5, 2024
56bb7f2
Aligned definition, declaration, and uses
edeiana Feb 5, 2024
bef5272
clang-format issue fixed.
edeiana Feb 5, 2024
9cc5924
Temporary change to github actions
edeiana Feb 5, 2024
608ddab
Re-added tmate session
edeiana Feb 5, 2024
82ddcbd
Trying to add a tmate session, 3rd try.
edeiana Feb 5, 2024
ec5a634
Fixed opcode_categories test.
edeiana Feb 6, 2024
55786e7
Reverted github/workflows file to remove tmate.
edeiana Feb 6, 2024
e5245ab
Trigger tmate.
edeiana Feb 6, 2024
693dc03
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 7, 2024
b638e6f
DO NOT COMMIT
edeiana Feb 7, 2024
cc45a92
Added opcode_categories test in the righ place.
edeiana Feb 7, 2024
533aef8
Re-aligned github workflow for ci-x86 with master.
edeiana Feb 7, 2024
f441276
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 7, 2024
cef05df
Addressed PR feedback.
edeiana Feb 7, 2024
fb46013
Fixed CHECK_CATEGORY.
edeiana Feb 7, 2024
89f655b
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 9, 2024
04a42e7
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 9, 2024
33821bf
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 12, 2024
9573500
Moved get_category_names() as protected member
edeiana Feb 12, 2024
301666e
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 13, 2024
e2a8107
Merge branch 'master' into i5994-category-opcode_mix
edeiana Feb 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions api/docs/release.dox
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,8 @@ Further non-compatibility-affecting changes include:
- Added opportunity to run multiple drcachesim analysis tools simultaneously.
- Added support of loading separately-built analysis tools to drcachesim dynamically.
- Added instr_is_opnd_store_source().
- Added instr_get_category_name() API that returns the string version (as char*) of a
category.

**************************************************
<hr>
Expand Down
32 changes: 32 additions & 0 deletions clients/drcachesim/tests/offline-opcode_categories.templatex
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Hello world!
Opcode mix tool results:
133 : total executed instructions
34 : mov
17 : mov
17 : syscall
16 : cmp
16 : jnz
16 : sub
16 : lea
1 : and

4 : sets of categories
51 : move
33 : branch
derekbruening marked this conversation as resolved.
Show resolved Hide resolved
33 : math
16 : load
56 changes: 51 additions & 5 deletions clients/drcachesim/tools/opcode_mix.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -178,7 +178,7 @@ opcode_mix_t::parallel_shard_memref(void *shard_data, const memref_t &memref)
decode_pc = const_cast<app_pc>(memref.instr.encoding);
if (memref.instr.encoding_is_new) {
// The code may have changed: invalidate the cache.
shard->worker->opcode_cache.erase(trace_pc);
shard->worker->opcode_category_cache.erase(trace_pc);
}
} else {
// Legacy trace support where we need the binaries.
Expand Down Expand Up @@ -210,9 +210,11 @@ opcode_mix_t::parallel_shard_memref(void *shard_data, const memref_t &memref)
}
}
int opcode;
auto cached_opcode = shard->worker->opcode_cache.find(trace_pc);
if (cached_opcode != shard->worker->opcode_cache.end()) {
opcode = cached_opcode->second;
uint category;
auto cached_opcode_category = shard->worker->opcode_category_cache.find(trace_pc);
if (cached_opcode_category != shard->worker->opcode_category_cache.end()) {
opcode = cached_opcode_category->second.opcode;
category = cached_opcode_category->second.category;
} else {
instr_t instr;
instr_init(dcontext_.dcontext, &instr);
Expand All @@ -225,10 +227,13 @@ opcode_mix_t::parallel_shard_memref(void *shard_data, const memref_t &memref)
return false;
}
opcode = instr_get_opcode(&instr);
shard->worker->opcode_cache[trace_pc] = opcode;
category = instr_get_category(&instr);
shard->worker->opcode_category_cache[trace_pc] =
opcode_category_data_t(opcode, category);
instr_free(dcontext_.dcontext, &instr);
}
++shard->opcode_counts[opcode];
++shard->category_counts[category];
return true;
}

Expand All @@ -255,6 +260,33 @@ cmp_val(const std::pair<int, int64_t> &l, const std::pair<int, int64_t> &r)
return (l.second > r.second);
}

std::string
get_category_names(uint category)
{
std::string category_name;
if (category == DR_INSTR_CATEGORY_UNCATEGORIZED) {
category_name += instr_get_category_name(DR_INSTR_CATEGORY_UNCATEGORIZED);
return category_name;
}

uint max_mask = 0x80000000;
edeiana marked this conversation as resolved.
Show resolved Hide resolved
for (uint mask = 0x1; mask <= max_mask; mask <<= 1) {
if (TESTANY(mask, category)) {
category_name += " ";
category_name += instr_get_category_name(mask);
}

/*
* Guard against 32 bit overflow.
*/
if (mask == max_mask) {
break;
}
}

return category_name;
}

bool
opcode_mix_t::print_results()
{
Expand All @@ -267,6 +299,9 @@ opcode_mix_t::print_results()
for (const auto &keyvals : shard.second->opcode_counts) {
total.opcode_counts[keyvals.first] += keyvals.second;
}
for (const auto &keyvals : shard.second->category_counts) {
total.category_counts[keyvals.first] += keyvals.second;
}
}
}
std::cerr << TOOL_NAME << " results:\n";
Expand All @@ -278,6 +313,17 @@ opcode_mix_t::print_results()
std::cerr << std::setw(15) << keyvals.second << " : " << std::setw(9)
<< decode_opcode_name(keyvals.first) << "\n";
}
std::cerr << "\n";
std::cerr << std::setw(15) << total.category_counts.size()
<< " : sets of categories\n";
std::vector<std::pair<uint, int64_t>> sorted_category_counts(
total.category_counts.begin(), total.category_counts.end());
std::sort(sorted_category_counts.begin(), sorted_category_counts.end(), cmp_val);
for (const auto &keyvals : sorted_category_counts) {
std::cerr << std::setw(15) << keyvals.second << " : " << std::setw(9)
<< get_category_names(keyvals.first) << "\n";
}

return true;
}

Expand Down
25 changes: 24 additions & 1 deletion clients/drcachesim/tools/opcode_mix.h
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@
#include <memory>
#include <mutex>
#include <string>
#include <utility>
#include <unordered_map>

#include "dr_api.h" // Must be before trace_entry.h from analysis_tool.h.
Expand Down Expand Up @@ -82,8 +83,29 @@ class opcode_mix_t : public analysis_tool_t {
parallel_shard_error(void *shard_data) override;

protected:
struct opcode_category_data_t {
opcode_category_data_t()
: opcode(0)
edeiana marked this conversation as resolved.
Show resolved Hide resolved
, category(0)
edeiana marked this conversation as resolved.
Show resolved Hide resolved
{
}
opcode_category_data_t(int opcode, uint category)
: opcode(opcode)
, category(category)
{
}
int opcode;
/*
* The category field is a uint instead of a dr_instr_category_t because
* multiple category bits can be set when an instruction belongs to more
* than one category. We assume 32 bits (i.e., 32 categories) is enough
* to be future-proof.
*/
uint category;
};

struct worker_data_t {
std::unordered_map<app_pc, int> opcode_cache;
std::unordered_map<app_pc, opcode_category_data_t> opcode_category_cache;
edeiana marked this conversation as resolved.
Show resolved Hide resolved
};

struct shard_data_t {
Expand All @@ -103,6 +125,7 @@ class opcode_mix_t : public analysis_tool_t {
worker_data_t *worker;
int64_t instr_count;
std::unordered_map<int, int64_t> opcode_counts;
std::unordered_map<uint, int64_t> category_counts;
std::string error;
app_pc last_trace_module_start;
size_t last_trace_module_size;
edeiana marked this conversation as resolved.
Show resolved Hide resolved
Expand Down
9 changes: 9 additions & 0 deletions core/ir/instr_api.h
Original file line number Diff line number Diff line change
Expand Up @@ -738,6 +738,15 @@ DR_API
uint
instr_get_category(instr_t *instr);

DR_API
/**
* Assumes \p category is a DR_INSTR_CATEGORY_ constant.
* See #dr_instr_category_t.
* Returns \p category name in string format.
*/
const char *
instr_get_category_name(uint category);

/**
* Get the relative offset of \p instr in an encoded instruction list.
*
Expand Down
19 changes: 19 additions & 0 deletions core/ir/instr_shared.c
Original file line number Diff line number Diff line change
Expand Up @@ -470,6 +470,25 @@ instr_get_category(instr_t *instr)
/* in rest of file, directly de-reference for performance (PR 622253) */
#define instr_get_category inlined_instr_get_category

const char *
instr_get_category_name(dr_instr_category_t category)
edeiana marked this conversation as resolved.
Show resolved Hide resolved
edeiana marked this conversation as resolved.
Show resolved Hide resolved
{
switch (category) {
case DR_INSTR_CATEGORY_UNCATEGORIZED: return "uncategorized";
edeiana marked this conversation as resolved.
Show resolved Hide resolved
case DR_INSTR_CATEGORY_FP: return "fp";
case DR_INSTR_CATEGORY_LOAD: return "load";
case DR_INSTR_CATEGORY_STORE: return "store";
case DR_INSTR_CATEGORY_BRANCH: return "branch";
case DR_INSTR_CATEGORY_SIMD: return "simd";
case DR_INSTR_CATEGORY_STATE: return "state";
case DR_INSTR_CATEGORY_MOVE: return "move";
case DR_INSTR_CATEGORY_CONVERT: return "convert";
case DR_INSTR_CATEGORY_MATH: return "math";
case DR_INSTR_CATEGORY_OTHER: return "other";
default: return "";
}
}

static inline void
instr_being_modified(instr_t *instr, bool raw_bits_valid)
{
Expand Down
2 changes: 2 additions & 0 deletions suite/tests/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -4147,6 +4147,8 @@ if (BUILD_CLIENTS)
# TODO i#3544: Port tests to RISC-V 64
torunonly_drcacheoff(opcode_mix ${ci_shared_app} ""
"@-simulator_type@opcode_mix" "")
torunonly_drcacheoff(opcode_categories allasm_x86_64 ""
"@-simulator_type@opcode_mix" "")
# Ensure the tool works without the raw/ subdir.
set(tool.drcacheoff.opcode_mix_postcmd
"firstglob@${drraw2trace_path}@-indir@${dir_prefix}.*.dir")
Expand Down
55 changes: 42 additions & 13 deletions suite/tests/api/drdecode_x86.c
Original file line number Diff line number Diff line change
Expand Up @@ -34,14 +34,16 @@

#include "configure.h"
#include "dr_api.h"
#include "tools.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define GD GLOBAL_DCONTEXT

#define ASSERT(x) \
((void)((!(x)) ? (printf("ASSERT FAILURE: %s:%d: %s\n", __FILE__, __LINE__, #x), \
abort(), 0) \
#define ASSERT(x) \
((void)((!(x)) ? (print("ASSERT FAILURE: %s:%d: %s\n", __FILE__, __LINE__, #x), \
abort(), 0) \
: 0))

#define BUFFER_SIZE_BYTES(buf) sizeof(buf)
Expand Down Expand Up @@ -155,12 +157,20 @@ test_noalloc(void)
*/
}

#define CHECK_CATEGORY(dcontext, instr, pc, category) \
ASSERT(instr_encode(dcontext, instr, pc) - pc < BUFFER_SIZE_ELEMENTS(pc)); \
instr_reset(dcontext, instr); \
instr_set_operands_valid(instr, true); \
ASSERT(decode(dcontext, pc, instr) != NULL); \
ASSERT(instr_get_category(instr) == category); \
#define CHECK_CATEGORY(dcontext, instr, pc, categories, category_names) \
ASSERT(instr_encode(dcontext, instr, pc) - pc < BUFFER_SIZE_ELEMENTS(pc)); \
edeiana marked this conversation as resolved.
Show resolved Hide resolved
instr_reset(dcontext, instr); \
instr_set_operands_valid(instr, true); \
ASSERT(decode(dcontext, pc, instr) != NULL); \
for (int i = 0; i < BUFFER_SIZE_ELEMENTS(categories); ++i) { \
if (categories[i] == DR_INSTR_CATEGORY_UNCATEGORIZED) { \
ASSERT(instr_get_category(instr) == categories[i]); \
} else { \
ASSERT(TESTANY(categories[i], instr_get_category(instr))); \
} \
ASSERT(strncmp(instr_get_category_name(categories[i]), category_names[i], \
sizeof(category_names[i])) == 0); \
} \
instr_destroy(dcontext, instr);

static void
Expand All @@ -172,17 +182,36 @@ test_categories(void)
/* 55 OP_mov_ld */
instr = XINST_CREATE_load(GD, opnd_create_reg(DR_REG_XAX),
OPND_CREATE_MEMPTR(DR_REG_XAX, 42));
CHECK_CATEGORY(GD, instr, buf, DR_INSTR_CATEGORY_LOAD);
dr_instr_category_t categories_load[] = { DR_INSTR_CATEGORY_LOAD };
edeiana marked this conversation as resolved.
Show resolved Hide resolved
const char *category_names_load[] = { "load" };
CHECK_CATEGORY(GD, instr, buf, categories_load, category_names_load);

/* 14 OP_cmp */
instr =
XINST_CREATE_cmp(GD, opnd_create_reg(DR_REG_EAX), opnd_create_reg(DR_REG_EAX));
CHECK_CATEGORY(GD, instr, buf, DR_INSTR_CATEGORY_MATH);
dr_instr_category_t categories_cmp[] = { DR_INSTR_CATEGORY_MATH };
const char *category_names_cmp[] = { "math" };
CHECK_CATEGORY(GD, instr, buf, categories_cmp, category_names_cmp);

/* 46 OP_jmp */
instr_t *after_callee = INSTR_CREATE_label(GD);
instr = XINST_CREATE_jump(GD, opnd_create_instr(after_callee));
CHECK_CATEGORY(GD, instr, buf, DR_INSTR_CATEGORY_BRANCH);
dr_instr_category_t categories_jmp[] = { DR_INSTR_CATEGORY_BRANCH };
const char *category_names_jmp[] = { "branch" };
CHECK_CATEGORY(GD, instr, buf, categories_jmp, category_names_jmp);

/* OP_fwait */
instr = INSTR_CREATE_fwait(GD);
dr_instr_category_t categories_fwait[] = { DR_INSTR_CATEGORY_FP,
DR_INSTR_CATEGORY_STATE };
const char *category_names_fwait[] = { "fp", "state" };
CHECK_CATEGORY(GD, instr, buf, categories_fwait, category_names_fwait);

/* OP_in */
instr = INSTR_CREATE_in_1(GD);
dr_instr_category_t categories_in[] = { DR_INSTR_CATEGORY_UNCATEGORIZED };
const char *category_names_in[] = { "uncategorized" };
CHECK_CATEGORY(GD, instr, buf, categories_in, category_names_in);
}

static void
Expand Down Expand Up @@ -241,7 +270,7 @@ main()

test_store_source();

printf("done\n");
print("done\n");

return 0;
}