-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdecode.c
85 lines (69 loc) · 1.58 KB
/
decode.c
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
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <string.h>
#include <inttypes.h>
#include "dat.h"
extern int verbose;
#define dprintf if(verbose)printf
void
printb(unsigned int v)
{
unsigned int i, s;
s = 1<<((sizeof(v)<<3)-1);
for (i = s; i; i>>=1)
dprintf("%d", v & i || 0 );
dprintf("\n");
}
int
decode_register(uint64_t r, char *name, Field *fields)
{
Field *f;
unsigned int i;
int v;
dprintf("%s 0x%.16" PRIx64 "\n", name, r);
printb(r);
for (i = 0; fields[i].name != NULL; i++) {
f = &fields[i];
v = (r>>f->bit) & MASK(f->size);
printf("%s\t%d", f->name, v);
if (f->decode != NULL) {
printf("\t");
f->decode(v);
}
printf("\n");
}
return 0;
}
Register registers[] = {
{ "Status", status, decode_register },
{ "Cause", cause, decode_register },
{ "nestedexc", nestedexc, decode_register },
{ "PRId", prid, decode_register },
{ "EntryHi", entryhi, decode_register },
{ "Config", config, decode_register },
{ "Config1", config1, decode_register },
{ "Config2", config2, decode_register },
{ "Config3", config3, decode_register },
{ "Config5", config5, decode_register },
{ "Ebase", ebase, decode_register },
};
int
decode(char *name, char *value)
{
Register *r;
uint64_t v;
unsigned int i;
v = strtoll(value, NULL, 0);
if (v == 0) {
fprintf(stderr, "error: strtoll: invalid value %s\n", value);
return -1;
}
for (i = 0; i < nelem(registers); i++) {
r = ®isters[i];
if(strcasecmp(r->name, name) == 0)
return r->decode(v, r->name, r->fields);
}
fprintf(stderr, "error: unknown register %s\n", name);
return -1;
}