blob: 88d0e9b3add1d8b48f1bb171e94dae570064468d (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
|
// See LICENSE for license details.
#ifndef _RISCV_DISASM_H
#define _RISCV_DISASM_H
#include "decode.h"
#include <string>
#include <sstream>
#include <vector>
extern const char* xpr_name[NXPR];
extern const char* fpr_name[NFPR];
extern const char* vr_name[NVPR];
extern const char* csr_name(int which);
class arg_t
{
public:
virtual std::string to_string(insn_t val) const = 0;
virtual ~arg_t() {}
};
// Indicates that the next arg (only) is optional.
// If the result of converting the next arg to a string is ""
// then it will not be printed.
struct : public arg_t {
std::string to_string(insn_t insn) const { return ""; }
} opt;
class disasm_insn_t
{
public:
NOINLINE disasm_insn_t(const char* name, uint32_t match, uint32_t mask,
const std::vector<const arg_t*>& args)
: match(match), mask(mask), args(args), name(strdup(name)) {}
~disasm_insn_t() { free(const_cast<char *>(name)); }
bool operator == (insn_t insn) const
{
return (insn.bits() & mask) == match;
}
const char* get_name() const
{
return name;
}
std::string to_string(insn_t insn) const
{
std::stringstream s;
int len;
for (len = 0; name[len]; len++)
s << (name[len] == '_' ? '.' : name[len]);
if (args.size())
{
bool next_arg_optional = false;
s << std::string(std::max(1, 8 - len), ' ');
for (size_t i = 0; i < args.size(); i++) {
if (args[i] == &opt) {
next_arg_optional = true;
continue;
}
std::string argString = args[i]->to_string(insn);
if (next_arg_optional) {
next_arg_optional = false;
if (argString.empty()) continue;
}
if (i != 0) s << ", ";
s << argString;
}
}
return s.str();
}
uint32_t get_match() const { return match; }
uint32_t get_mask() const { return mask; }
private:
uint32_t match;
uint32_t mask;
std::vector<const arg_t*> args;
const char* name;
};
class disassembler_t
{
public:
disassembler_t(int xlen);
~disassembler_t();
std::string disassemble(insn_t insn) const;
const disasm_insn_t* lookup(insn_t insn) const;
void add_insn(disasm_insn_t* insn);
private:
static const int HASH_SIZE = 256;
std::vector<const disasm_insn_t*> chain[HASH_SIZE+1];
};
#endif
|