-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathreg.cpp
executable file
·308 lines (267 loc) · 9.51 KB
/
reg.cpp
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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
#include "stm8.hpp"
#include <fpro.h>
#include <diskio.hpp>
#include <entry.hpp>
//--------------------------------------------------------------------------
static const char *register_names[] =
{
"a", "x", "y", "cc", "sp",
"xl", "xh", "yl", "yh",
"ds", "cs"
};
//--------------------------------------------------------------------------
static uchar retcode0[] = { 0x80 }; // iret 80
static uchar retcode1[] = { 0x81 }; // ret 81
static bytes_t retcodes[] =
{
{ sizeof(retcode0), retcode0 },
{ sizeof(retcode1), retcode1 },
{ 0, NULL }
};
//-----------------------------------------------------------------------
// STMicroelectronics - Assembler - rel. 4.10
// We support Motorola format
//-----------------------------------------------------------------------
static const char *stm8_header[] =
{
"stm8/",
"",
NULL
};
static asm_t stasm =
{
ASH_HEXF4| // $1234
ASD_DECF0| // 1234
ASB_BINF2| // %1010
ASO_OCTF6| // ~1234
AS_NOXRF| // Disable xrefs during the output file generation
AS_ONEDUP, // one array definition per line
0,
"STMicroelectronics - Assembler8",
0,
stm8_header, // header lines
NULL, // no bad instructions
"org", // org
"end", // end
";", // comment string
'\"', // string delimiter
'\'', // char delimiter
"'\"", // special symbols in char and string constants
"dc.b", // ascii string directive
"dc.b", // byte directive
"dc.w", // word directive
"dc.l", // double words
NULL, // qwords
NULL, // oword (16 bytes)
NULL, // float (4 bytes)
NULL, // double (8 bytes)
NULL, // tbyte (10/12 bytes)
NULL, // packed decimal real
"skip#s( )#d, #v", // arrays (#h,#d,#v,#s(...) ONLY BYTE ARRAYS!!!
"ds.b %s", // uninited arrays
"equ", // equ
NULL, // 'seg' prefix (example: push seg seg001)
NULL, // Pointer to checkarg_preline() function.
NULL, // char *(*checkarg_atomprefix)(char *operand,void *res); // if !NULL, is called before each atom
NULL, // const char **checkarg_operations;
NULL, // translation to use in character and string constants.
"*", // current IP (instruction pointer)
NULL, // func_header
NULL, // func_footer
"public", // "public" name keyword
NULL, // "weak" name keyword
"extern", // "extrn" name keyword
// .extern directive requires an explicit object size
NULL, // "comm" (communal variable)
NULL, // get_type_name
NULL, // "align" keyword
'{', '}', // lbrace, rbrace
NULL, // mod
"and", // and
"or", // or
"xor", // xor
NULL, // not
"shl", // shl
"shr", // shr
NULL, // sizeof
AS2_BRACE,
};
static asm_t *asms[] = { &stasm, NULL };
//--------------------------------------------------------------------------
ea_t memstart;
//static char device[MAXSTR] = "";
//static const char cfgname[] = "stm8.cfg";
char device[MAXSTR] = "";
static size_t numports;
static ioport_t *ports = NULL;
#include <iocommon.cpp>
//
//static void load_symbols(void)
//{
// free_ioports(ports, numports);
// ports = read_ioports(&numports, cfgname, device, NULL);
//}
//
//----------------------------------------------------------------------
const ioport_t *find_sym(ea_t address)
{
const ioport_t *port = find_ioport(ports, numports, address);
return port;
}
//----------------------------------------------------------------------
static void create_words(void)
{
for ( size_t i=0; i < numports; i++ )
{
ea_t ea = ports[i].address;
if ( isTail(get_flags_novalue(ea)) )
do_unknown(ea, DOUNK_SIMPLE);
doWord(ea, 2);
}
}
//--------------------------------------------------------------------------
const char * idaapi set_idp_options(const char *keyword,int /*value_type*/,const void * /*value*/)
{
if ( keyword != NULL ) return IDPOPT_BADKEY;
char cfgfile[QMAXFILE];
get_cfg_filename(cfgfile, sizeof(cfgfile));
if ( choose_ioport_device(cfgfile, device, sizeof(device), NULL) )
set_device_name(device, IORESP_PORT|IORESP_INT);
return IDPOPT_OK;
}
//--------------------------------------------------------------------------
netnode helper;
static int notify(processor_t::idp_notify msgid, ...)
{
int retcode = 1;
va_list va;
va_start(va, msgid);
// A well behaving processor module should call invoke_callbacks()
// in his notify() function. If this function returns 0, then
// the processor module should process the notification itself
// Otherwise the code should be returned to the caller:
int code = invoke_callbacks(HT_IDP, msgid, va);
if ( code ) return code;
switch(msgid)
{
case processor_t::init:
helper.create("$ stm8");
inf.mf = 1;
break;
case processor_t::newfile: // new file loaded
{
char cfgfile[QMAXFILE];
get_cfg_filename(cfgfile, sizeof(cfgfile));
if ( choose_ioport_device(cfgfile, device, sizeof(device), NULL) )
set_device_name(device, IORESP_ALL);
create_words();
}
break;
case processor_t::oldfile: // old file loaded
{
char buf[MAXSTR];
if ( helper.supval(-1, buf, sizeof(buf)) > 0 )
set_device_name(buf, IORESP_NONE);
}
break;
case processor_t::is_jump_func:
{
const func_t *pfn = va_arg(va, const func_t *);
ea_t *jump_target = va_arg(va, ea_t *);
retcode = is_jump_func(pfn, jump_target);
}
break;
case processor_t::is_sane_insn:
retcode = is_sane_insn(va_arg(va, int));
break;
case processor_t::may_be_func:
// can a function start here?
// arg: none, the instruction is in 'cmd'
// returns: probability 0..100
// 'cmd' structure is filled upon the entrace
// the idp module is allowed to modify 'cmd'
retcode = may_be_func();
break;
case processor_t::get_autocmt:
{
char *buf = va_arg(va, char *);
size_t bufsize = va_arg(va, size_t);
if ( qsnprintf(buf, bufsize, "%s", insn_auto_cmts[cmd.itype]) )
retcode = 2;
}
break;
}
va_end(va);
return retcode;
}
//-----------------------------------------------------------------------
static char const *shnames[] = { "stm8", NULL };
static char const *lnames[] = {
"SGS-Thomson STM8",
NULL
};
//-----------------------------------------------------------------------
// Processor Definition
//-----------------------------------------------------------------------
processor_t LPH =
{
IDP_INTERFACE_VERSION, // version
0x8000 | 1234, // id
PRN_HEX|PR_RNAMESOK,
8, // 8 bits in a byte for code segments
8, // 8 bits in a byte for other segments
(char const **)shnames,
(char const **)lnames,
asms,
notify,
header,
footer,
segstart,
segend,
NULL, // generate "assume" directives
ana, // analyze instruction
emu, // emulate instruction
out, // generate text representation of instruction
outop, // generate ... operand
intel_data, // generate ... data directive
NULL, // compare operands
NULL, // can have type
qnumber(register_names), // Number of registers
(char const **)register_names, // Register names
NULL, // get abstract register
0, // Number of register files
NULL, // Register file names
NULL, // Register descriptions
NULL, // Pointer to CPU registers
ds, // first
cs, // last
2, // size of a segment register
cs, ds,
NULL, // No known code start sequences
retcodes,
STM8_null,
STM8_last,
Instructions,
NULL, // int (*is_far_jump)(int icode);
NULL, // Translation function for offsets
0, // int tbyte_size; -- doesn't exist
NULL, // int (*realcvt)(void *m, ushort *e, ushort swt);
{ 0, 7, 15, 0 }, // char real_width[4];
// number of symbols after decimal point
// 2byte float (0-does not exist)
// normal float
// normal double
// long double
is_switch, // bool idaapi is_switch(switch_info_ex_t *si);
NULL, // long (*gen_map_file)(FILE *fp);
NULL, // ulong (*extract_address)(ulong ea,const char *string,int x);
NULL, // int (*is_sp_based)(op_t &x);
NULL, // int (*create_func_frame)(func_t *pfn);
NULL, // int (*get_frame_retsize(func_t *pfn)
NULL, // void (*gen_stkvar_def)(char *buf,const member_t *mptr,long v);
gen_spcdef, // Generate text representation of an item in a special segment
STM8_ret, // Icode of return instruction. It is ok to give any of possible return instructions
set_idp_options, // const char *(*set_idp_options)(const char *keyword,int value_type,const void *value);
is_align_insn, // int (*is_align_insn)(ulong ea);
NULL, // mvm_t *mvm;
};