Coverage Report

Created: 2026-05-30 06:22

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonenext/arch/M680X/M680XDisassembler.c
Line
Count
Source
1
/* Capstone Disassembly Engine */
2
/* M680X Backend by Wolfgang Schwotzer <wolfgang.schwotzer@gmx.net> 2017 */
3
4
/* ======================================================================== */
5
/* ================================ INCLUDES ============================== */
6
/* ======================================================================== */
7
8
#include <stdlib.h>
9
#include <stdio.h>
10
#include <string.h>
11
12
#include "../../cs_priv.h"
13
#include "../../utils.h"
14
15
#include "../../MCInst.h"
16
#include "../../MCInstrDesc.h"
17
#include "../../MCRegisterInfo.h"
18
#include "M680XInstPrinter.h"
19
#include "M680XDisassembler.h"
20
#include "M680XDisassemblerInternals.h"
21
22
#ifdef CAPSTONE_HAS_M680X
23
24
#ifndef DECL_SPEC
25
#ifdef _MSC_VER
26
#define DECL_SPEC __cdecl
27
#else
28
#define DECL_SPEC
29
#endif // _MSC_VER
30
#endif // DECL_SPEC
31
32
/* ======================================================================== */
33
/* ============================ GENERAL DEFINES =========================== */
34
/* ======================================================================== */
35
36
/* ======================================================================== */
37
/* =============================== PROTOTYPES ============================= */
38
/* ======================================================================== */
39
40
typedef enum insn_hdlr_id {
41
  illgl_hid,
42
  rel8_hid,
43
  rel16_hid,
44
  imm8_hid,
45
  imm16_hid,
46
  imm32_hid,
47
  dir_hid,
48
  ext_hid,
49
  idxX_hid,
50
  idxY_hid,
51
  idx09_hid,
52
  inh_hid,
53
  rr09_hid,
54
  rbits_hid,
55
  bitmv_hid,
56
  tfm_hid,
57
  opidx_hid,
58
  opidxdr_hid,
59
  idxX0_hid,
60
  idxX16_hid,
61
  imm8rel_hid,
62
  idxS_hid,
63
  idxS16_hid,
64
  idxXp_hid,
65
  idxX0p_hid,
66
  idx12_hid,
67
  idx12s_hid,
68
  rr12_hid,
69
  loop_hid,
70
  index_hid,
71
  imm8i12x_hid,
72
  imm16i12x_hid,
73
  exti12x_hid,
74
  srt_hid,
75
  tny_hid,
76
  dirdir_hid,
77
  immdir_hid,
78
  HANDLER_ID_ENDING,
79
} insn_hdlr_id;
80
81
// Access modes for the first 4 operands. If there are more than
82
// four operands they use the same access mode as the 4th operand.
83
//
84
// u: unchanged
85
// r: (r)read access
86
// w: (w)write access
87
// m: (m)odify access (= read + write)
88
//
89
typedef enum e_access_mode {
90
91
  uuuu,
92
  rrrr,
93
  wwww,
94
  rwww,
95
  rrrm,
96
  rmmm,
97
  wrrr,
98
  mrrr,
99
  mwww,
100
  mmmm,
101
  mwrr,
102
  mmrr,
103
  wmmm,
104
  rruu,
105
  muuu,
106
  ACCESS_MODE_ENDING,
107
} e_access_mode;
108
109
// Access type values are compatible with enum cs_ac_type:
110
typedef cs_ac_type e_access;
111
0
#define UNCHANGED CS_AC_INVALID
112
202k
#define READ CS_AC_READ
113
285k
#define WRITE CS_AC_WRITE
114
340k
#define MODIFY CS_AC_READ_WRITE
115
116
/* Properties of one instruction in PAGE1 (without prefix) */
117
typedef struct inst_page1 {
118
  unsigned insn : 9; // A value of type m680x_insn
119
  unsigned handler_id1 : 6; // Type insn_hdlr_id, first instr. handler id
120
  unsigned handler_id2 : 6; // Type insn_hdlr_id, second instr. handler id
121
} inst_page1;
122
123
/* Properties of one instruction in any other PAGE X */
124
typedef struct inst_pageX {
125
  unsigned opcode : 8; // The opcode byte
126
  unsigned insn : 9; // A value of type m680x_insn
127
  unsigned handler_id1 : 6; // Type insn_hdlr_id, first instr. handler id
128
  unsigned handler_id2 : 6; // Type insn_hdlr_id, second instr. handler id
129
} inst_pageX;
130
131
typedef struct insn_props {
132
  unsigned group : 4;
133
  unsigned access_mode : 5; // A value of type e_access_mode
134
  unsigned reg0 : 5; // A value of type m680x_reg
135
  unsigned reg1 : 5; // A value of type m680x_reg
136
  bool cc_modified : 1;
137
  bool update_reg_access : 1;
138
} insn_props;
139
140
#include "m6800.inc"
141
#include "m6801.inc"
142
#include "hd6301.inc"
143
#include "m6811.inc"
144
#include "cpu12.inc"
145
#include "m6805.inc"
146
#include "m6808.inc"
147
#include "hcs08.inc"
148
#include "m6809.inc"
149
#include "hd6309.inc"
150
#include "rs08.inc"
151
#include "hcs12x.inc"
152
153
#include "insn_props.inc"
154
155
//////////////////////////////////////////////////////////////////////////////
156
157
// M680X instructions have 1 up to 8 bytes (CPU12: MOVW IDX2,IDX2).
158
// A reader is needed to read a byte or word from a given memory address.
159
// See also X86 reader(...)
160
static bool read_byte(const m680x_info *info, uint8_t *byte, uint16_t address)
161
506k
{
162
506k
  if (address < info->offset ||
163
506k
      (uint32_t)(address - info->offset) >= info->size)
164
    // out of code buffer range
165
801
    return false;
166
167
505k
  *byte = info->code[address - info->offset];
168
169
505k
  return true;
170
506k
}
171
172
static bool read_byte_sign_extended(const m680x_info *info, int16_t *word,
173
            uint16_t address)
174
32.1k
{
175
32.1k
  if (address < info->offset ||
176
32.1k
      (uint32_t)(address - info->offset) >= info->size)
177
    // out of code buffer range
178
0
    return false;
179
180
32.1k
  *word = (int16_t)info->code[address - info->offset];
181
182
32.1k
  if (*word & 0x80)
183
11.8k
    *word |= 0xFF00;
184
185
32.1k
  return true;
186
32.1k
}
187
188
static bool read_word(const m680x_info *info, uint16_t *word, uint16_t address)
189
35.3k
{
190
35.3k
  if (address < info->offset ||
191
35.3k
      (uint32_t)(address + 1 - info->offset) >= info->size)
192
    // out of code buffer range
193
6
    return false;
194
195
35.2k
  *word = (uint16_t)info->code[address - info->offset] << 8;
196
35.2k
  *word |= (uint16_t)info->code[address + 1 - info->offset];
197
198
35.2k
  return true;
199
35.3k
}
200
201
static bool read_sdword(const m680x_info *info, int32_t *sdword,
202
      uint16_t address)
203
106
{
204
106
  if (address < info->offset ||
205
106
      (uint32_t)(address + 3 - info->offset) >= info->size)
206
    // out of code buffer range
207
0
    return false;
208
209
106
  *sdword = (uint32_t)info->code[address - info->offset] << 24;
210
106
  *sdword |= (uint32_t)info->code[address + 1 - info->offset] << 16;
211
106
  *sdword |= (uint32_t)info->code[address + 2 - info->offset] << 8;
212
106
  *sdword |= (uint32_t)info->code[address + 3 - info->offset];
213
214
106
  return true;
215
106
}
216
217
// For PAGE2 and PAGE3 opcodes when using an array of inst_page1 most
218
// entries have M680X_INS_ILLGL. To avoid wasting memory an inst_pageX is
219
// used which contains the opcode. Using a binary search for the right opcode
220
// is much faster (= O(log n) ) in comparison to a linear search ( = O(n) ).
221
static int binary_search(const inst_pageX *const inst_pageX_table,
222
       size_t table_size, unsigned int opcode)
223
62.3k
{
224
  // As part of the algorithm last may get negative.
225
  // => signed integer has to be used.
226
62.3k
  int first = 0;
227
62.3k
  int last = (int)table_size - 1;
228
62.3k
  int middle = (first + last) / 2;
229
230
328k
  while (first <= last) {
231
300k
    if (inst_pageX_table[middle].opcode < opcode) {
232
94.8k
      first = middle + 1;
233
205k
    } else if (inst_pageX_table[middle].opcode == opcode) {
234
34.5k
      return middle; /* item found */
235
34.5k
    } else
236
170k
      last = middle - 1;
237
238
265k
    middle = (first + last) / 2;
239
265k
  }
240
241
27.7k
  if (first > last)
242
27.7k
    return -1; /* item not found */
243
244
0
  return -2;
245
27.7k
}
246
247
void M680X_get_insn_id(cs_struct *handle, cs_insn *insn, unsigned int id)
248
214k
{
249
214k
  const m680x_info *const info = (const m680x_info *)handle->printer_info;
250
214k
  const cpu_tables *cpu = info->cpu;
251
214k
  uint8_t insn_prefix = (id >> 8) & 0xff;
252
  // opcode is the first instruction byte without the prefix.
253
214k
  uint8_t opcode = id & 0xff;
254
214k
  int index;
255
214k
  int i;
256
257
214k
  insn->id = M680X_INS_ILLGL;
258
259
499k
  for (i = 0; i < ARR_SIZE(cpu->pageX_prefix); ++i) {
260
495k
    if (cpu->pageX_table_size[i] == 0 ||
261
301k
        (cpu->inst_pageX_table[i] == NULL))
262
194k
      break;
263
264
301k
    if (cpu->pageX_prefix[i] == insn_prefix) {
265
15.8k
      index = binary_search(cpu->inst_pageX_table[i],
266
15.8k
                cpu->pageX_table_size[i], opcode);
267
15.8k
      insn->id =
268
15.8k
        (index >= 0) ?
269
9.28k
          cpu->inst_pageX_table[i][index].insn :
270
15.8k
          M680X_INS_ILLGL;
271
15.8k
      return;
272
15.8k
    }
273
301k
  }
274
275
198k
  if (insn_prefix != 0)
276
0
    return;
277
278
198k
  insn->id = cpu->inst_page1_table[id].insn;
279
280
198k
  if (insn->id != M680X_INS_ILLGL)
281
181k
    return;
282
283
  // Check if opcode byte is present in an overlay table
284
24.4k
  for (i = 0; i < ARR_SIZE(cpu->overlay_table_size); ++i) {
285
23.9k
    if (cpu->overlay_table_size[i] == 0 ||
286
15.3k
        (cpu->inst_overlay_table[i] == NULL))
287
8.59k
      break;
288
289
15.3k
    if ((index = binary_search(cpu->inst_overlay_table[i],
290
15.3k
             cpu->overlay_table_size[i],
291
15.3k
             opcode)) >= 0) {
292
8.01k
      insn->id = cpu->inst_overlay_table[i][index].insn;
293
8.01k
      return;
294
8.01k
    }
295
15.3k
  }
296
17.1k
}
297
298
static void add_insn_group(cs_detail *detail, m680x_group_type group)
299
210k
{
300
210k
  if (detail != NULL && (group != M680X_GRP_INVALID) &&
301
50.9k
      (group != M680X_GRP_ENDING))
302
50.9k
    detail->groups[detail->groups_count++] = (uint8_t)group;
303
210k
}
304
305
static bool exists_reg_list(uint16_t *regs, uint8_t count, m680x_reg reg)
306
610k
{
307
610k
  uint8_t i;
308
309
1.01M
  for (i = 0; i < count; ++i) {
310
424k
    if (regs[i] == (uint16_t)reg)
311
16.2k
      return true;
312
424k
  }
313
314
593k
  return false;
315
610k
}
316
317
static void add_reg_to_rw_list(MCInst *MI, m680x_reg reg, e_access access)
318
397k
{
319
397k
  cs_detail *detail = MI->flat_insn->detail;
320
321
397k
  if (detail == NULL || (reg == M680X_REG_INVALID))
322
0
    return;
323
324
397k
  switch (access) {
325
212k
  case MODIFY:
326
212k
    if (!exists_reg_list(detail->regs_read, detail->regs_read_count,
327
212k
             reg))
328
208k
      detail->regs_read[detail->regs_read_count++] =
329
208k
        (uint16_t)reg;
330
331
    // intentionally fall through
332
333
272k
  case WRITE:
334
272k
    if (!exists_reg_list(detail->regs_write,
335
272k
             detail->regs_write_count, reg))
336
266k
      detail->regs_write[detail->regs_write_count++] =
337
266k
        (uint16_t)reg;
338
339
272k
    break;
340
341
125k
  case READ:
342
125k
    if (!exists_reg_list(detail->regs_read, detail->regs_read_count,
343
125k
             reg))
344
118k
      detail->regs_read[detail->regs_read_count++] =
345
118k
        (uint16_t)reg;
346
347
125k
    break;
348
349
0
  case UNCHANGED:
350
0
  default:
351
0
    break;
352
397k
  }
353
397k
}
354
355
static void update_am_reg_list(MCInst *MI, m680x_info *info, cs_m680x_op *op,
356
             e_access access)
357
281k
{
358
281k
  if (MI->flat_insn->detail == NULL)
359
0
    return;
360
361
281k
  switch (op->type) {
362
122k
  case M680X_OP_REGISTER:
363
122k
    add_reg_to_rw_list(MI, op->reg, access);
364
122k
    break;
365
366
51.5k
  case M680X_OP_INDEXED:
367
51.5k
    add_reg_to_rw_list(MI, op->idx.base_reg, READ);
368
369
51.5k
    if (op->idx.base_reg == M680X_REG_X &&
370
22.4k
        info->cpu->reg_byte_size[M680X_REG_H])
371
7.81k
      add_reg_to_rw_list(MI, M680X_REG_H, READ);
372
373
51.5k
    if (op->idx.offset_reg != M680X_REG_INVALID)
374
4.15k
      add_reg_to_rw_list(MI, op->idx.offset_reg, READ);
375
376
51.5k
    if (op->idx.inc_dec) {
377
11.3k
      add_reg_to_rw_list(MI, op->idx.base_reg, WRITE);
378
379
11.3k
      if (op->idx.base_reg == M680X_REG_X &&
380
3.38k
          info->cpu->reg_byte_size[M680X_REG_H])
381
1.13k
        add_reg_to_rw_list(MI, M680X_REG_H, WRITE);
382
11.3k
    }
383
384
51.5k
    break;
385
386
107k
  default:
387
107k
    break;
388
281k
  }
389
281k
}
390
391
static const e_access g_access_mode_to_access[4][15] = {
392
  {
393
    UNCHANGED,
394
    READ,
395
    WRITE,
396
    READ,
397
    READ,
398
    READ,
399
    WRITE,
400
    MODIFY,
401
    MODIFY,
402
    MODIFY,
403
    MODIFY,
404
    MODIFY,
405
    WRITE,
406
    READ,
407
    MODIFY,
408
  },
409
  {
410
    UNCHANGED,
411
    READ,
412
    WRITE,
413
    WRITE,
414
    READ,
415
    MODIFY,
416
    READ,
417
    READ,
418
    WRITE,
419
    MODIFY,
420
    WRITE,
421
    MODIFY,
422
    MODIFY,
423
    READ,
424
    UNCHANGED,
425
  },
426
  {
427
    UNCHANGED,
428
    READ,
429
    WRITE,
430
    WRITE,
431
    READ,
432
    MODIFY,
433
    READ,
434
    READ,
435
    WRITE,
436
    MODIFY,
437
    READ,
438
    READ,
439
    MODIFY,
440
    UNCHANGED,
441
    UNCHANGED,
442
  },
443
  {
444
    UNCHANGED,
445
    READ,
446
    WRITE,
447
    WRITE,
448
    MODIFY,
449
    MODIFY,
450
    READ,
451
    READ,
452
    WRITE,
453
    MODIFY,
454
    READ,
455
    READ,
456
    MODIFY,
457
    UNCHANGED,
458
    UNCHANGED,
459
  },
460
};
461
462
static e_access get_access(int operator_index, e_access_mode access_mode)
463
598k
{
464
598k
  int idx = (operator_index > 3) ? 3 : operator_index;
465
466
598k
  return g_access_mode_to_access[idx][access_mode];
467
598k
}
468
469
static void build_regs_read_write_counts(MCInst *MI, m680x_info *info,
470
           e_access_mode access_mode)
471
191k
{
472
191k
  cs_m680x *m680x = &info->m680x;
473
191k
  int i;
474
475
191k
  if (MI->flat_insn->detail == NULL || (!m680x->op_count))
476
22.3k
    return;
477
478
450k
  for (i = 0; i < m680x->op_count; ++i) {
479
281k
    e_access access = get_access(i, access_mode);
480
281k
    update_am_reg_list(MI, info, &m680x->operands[i], access);
481
281k
  }
482
169k
}
483
484
static void add_operators_access(MCInst *MI, m680x_info *info,
485
         e_access_mode access_mode)
486
191k
{
487
191k
  cs_m680x *m680x = &info->m680x;
488
191k
  int offset = 0;
489
191k
  int i;
490
491
191k
  if (MI->flat_insn->detail == NULL || (!m680x->op_count) ||
492
169k
      (access_mode == uuuu))
493
43.2k
    return;
494
495
408k
  for (i = 0; i < m680x->op_count; ++i) {
496
260k
    e_access access;
497
498
    // Ugly fix: MULD has a register operand, an immediate operand
499
    // AND an implicitly changed register W
500
260k
    if (info->insn == M680X_INS_MULD && (i == 1))
501
201
      offset = 1;
502
503
260k
    access = get_access(i + offset, access_mode);
504
260k
    m680x->operands[i].access = access;
505
260k
  }
506
148k
}
507
508
typedef struct insn_to_changed_regs {
509
  m680x_insn insn;
510
  e_access_mode access_mode;
511
  m680x_reg regs[10];
512
} insn_to_changed_regs;
513
514
static void set_changed_regs_read_write_counts(MCInst *MI, m680x_info *info)
515
21.1k
{
516
  //TABLE
517
1.16M
#define EOL M680X_REG_INVALID
518
21.1k
  static const insn_to_changed_regs changed_regs[] = {
519
21.1k
    { M680X_INS_BSR, mmmm, { M680X_REG_S, EOL } },
520
21.1k
    { M680X_INS_CALL, mmmm, { M680X_REG_S, EOL } },
521
21.1k
    {
522
21.1k
      M680X_INS_CWAI,
523
21.1k
      mrrr,
524
21.1k
      { M680X_REG_S, M680X_REG_PC, M680X_REG_U, M680X_REG_Y,
525
21.1k
        M680X_REG_X, M680X_REG_DP, M680X_REG_D, M680X_REG_CC,
526
21.1k
        EOL },
527
21.1k
    },
528
21.1k
    { M680X_INS_DAA, mrrr, { M680X_REG_A, EOL } },
529
21.1k
    { M680X_INS_DIV,
530
21.1k
      mmrr,
531
21.1k
      { M680X_REG_A, M680X_REG_H, M680X_REG_X, EOL } },
532
21.1k
    { M680X_INS_EDIV,
533
21.1k
      mmrr,
534
21.1k
      { M680X_REG_D, M680X_REG_Y, M680X_REG_X, EOL } },
535
21.1k
    { M680X_INS_EDIVS,
536
21.1k
      mmrr,
537
21.1k
      { M680X_REG_D, M680X_REG_Y, M680X_REG_X, EOL } },
538
21.1k
    { M680X_INS_EMACS, mrrr, { M680X_REG_X, M680X_REG_Y, EOL } },
539
21.1k
    { M680X_INS_EMAXM, rrrr, { M680X_REG_D, EOL } },
540
21.1k
    { M680X_INS_EMINM, rrrr, { M680X_REG_D, EOL } },
541
21.1k
    { M680X_INS_EMUL, mmrr, { M680X_REG_D, M680X_REG_Y, EOL } },
542
21.1k
    { M680X_INS_EMULS, mmrr, { M680X_REG_D, M680X_REG_Y, EOL } },
543
21.1k
    { M680X_INS_ETBL, wmmm, { M680X_REG_A, M680X_REG_B, EOL } },
544
21.1k
    { M680X_INS_FDIV, mmmm, { M680X_REG_D, M680X_REG_X, EOL } },
545
21.1k
    { M680X_INS_IDIV, mmmm, { M680X_REG_D, M680X_REG_X, EOL } },
546
21.1k
    { M680X_INS_IDIVS, mmmm, { M680X_REG_D, M680X_REG_X, EOL } },
547
21.1k
    { M680X_INS_JSR, mmmm, { M680X_REG_S, EOL } },
548
21.1k
    { M680X_INS_LBSR, mmmm, { M680X_REG_S, EOL } },
549
21.1k
    { M680X_INS_MAXM, rrrr, { M680X_REG_A, EOL } },
550
21.1k
    { M680X_INS_MINM, rrrr, { M680X_REG_A, EOL } },
551
21.1k
    { M680X_INS_MEM,
552
21.1k
      mmrr,
553
21.1k
      { M680X_REG_X, M680X_REG_Y, M680X_REG_A, EOL } },
554
21.1k
    { M680X_INS_MUL, mmmm, { M680X_REG_A, M680X_REG_B, EOL } },
555
21.1k
    { M680X_INS_MULD, mwrr, { M680X_REG_D, M680X_REG_W, EOL } },
556
21.1k
    { M680X_INS_PSHA, rmmm, { M680X_REG_A, M680X_REG_S, EOL } },
557
21.1k
    { M680X_INS_PSHB, rmmm, { M680X_REG_B, M680X_REG_S, EOL } },
558
21.1k
    { M680X_INS_PSHC, rmmm, { M680X_REG_CC, M680X_REG_S, EOL } },
559
21.1k
    { M680X_INS_PSHD, rmmm, { M680X_REG_D, M680X_REG_S, EOL } },
560
21.1k
    { M680X_INS_PSHH, rmmm, { M680X_REG_H, M680X_REG_S, EOL } },
561
21.1k
    { M680X_INS_PSHX, rmmm, { M680X_REG_X, M680X_REG_S, EOL } },
562
21.1k
    { M680X_INS_PSHY, rmmm, { M680X_REG_Y, M680X_REG_S, EOL } },
563
21.1k
    { M680X_INS_PULA, wmmm, { M680X_REG_A, M680X_REG_S, EOL } },
564
21.1k
    { M680X_INS_PULB, wmmm, { M680X_REG_B, M680X_REG_S, EOL } },
565
21.1k
    { M680X_INS_PULC, wmmm, { M680X_REG_CC, M680X_REG_S, EOL } },
566
21.1k
    { M680X_INS_PULD, wmmm, { M680X_REG_D, M680X_REG_S, EOL } },
567
21.1k
    { M680X_INS_PULH, wmmm, { M680X_REG_H, M680X_REG_S, EOL } },
568
21.1k
    { M680X_INS_PULX, wmmm, { M680X_REG_X, M680X_REG_S, EOL } },
569
21.1k
    { M680X_INS_PULY, wmmm, { M680X_REG_Y, M680X_REG_S, EOL } },
570
21.1k
    { M680X_INS_REV,
571
21.1k
      mmrr,
572
21.1k
      { M680X_REG_A, M680X_REG_X, M680X_REG_Y, EOL } },
573
21.1k
    { M680X_INS_REVW,
574
21.1k
      mmmm,
575
21.1k
      { M680X_REG_A, M680X_REG_X, M680X_REG_Y, EOL } },
576
21.1k
    { M680X_INS_RTC, mwww, { M680X_REG_S, M680X_REG_PC, EOL } },
577
21.1k
    {
578
21.1k
      M680X_INS_RTI,
579
21.1k
      mwww,
580
21.1k
      { M680X_REG_S, M680X_REG_CC, M680X_REG_B, M680X_REG_A,
581
21.1k
        M680X_REG_DP, M680X_REG_X, M680X_REG_Y, M680X_REG_U,
582
21.1k
        M680X_REG_PC, EOL },
583
21.1k
    },
584
21.1k
    { M680X_INS_RTS, mwww, { M680X_REG_S, M680X_REG_PC, EOL } },
585
21.1k
    { M680X_INS_SEX, wrrr, { M680X_REG_A, M680X_REG_B, EOL } },
586
21.1k
    { M680X_INS_SEXW, rwww, { M680X_REG_W, M680X_REG_D, EOL } },
587
21.1k
    { M680X_INS_SWI,
588
21.1k
      mmrr,
589
21.1k
      { M680X_REG_S, M680X_REG_PC, M680X_REG_U, M680X_REG_Y,
590
21.1k
        M680X_REG_X, M680X_REG_DP, M680X_REG_A, M680X_REG_B,
591
21.1k
        M680X_REG_CC, EOL } },
592
21.1k
    {
593
21.1k
      M680X_INS_SWI2,
594
21.1k
      mmrr,
595
21.1k
      { M680X_REG_S, M680X_REG_PC, M680X_REG_U, M680X_REG_Y,
596
21.1k
        M680X_REG_X, M680X_REG_DP, M680X_REG_A, M680X_REG_B,
597
21.1k
        M680X_REG_CC, EOL },
598
21.1k
    },
599
21.1k
    {
600
21.1k
      M680X_INS_SWI3,
601
21.1k
      mmrr,
602
21.1k
      { M680X_REG_S, M680X_REG_PC, M680X_REG_U, M680X_REG_Y,
603
21.1k
        M680X_REG_X, M680X_REG_DP, M680X_REG_A, M680X_REG_B,
604
21.1k
        M680X_REG_CC, EOL },
605
21.1k
    },
606
21.1k
    { M680X_INS_TBL, wrrr, { M680X_REG_A, M680X_REG_B, EOL } },
607
21.1k
    { M680X_INS_WAI,
608
21.1k
      mrrr,
609
21.1k
      { M680X_REG_S, M680X_REG_PC, M680X_REG_X, M680X_REG_A,
610
21.1k
        M680X_REG_B, M680X_REG_CC, EOL } },
611
21.1k
    { M680X_INS_WAV,
612
21.1k
      rmmm,
613
21.1k
      { M680X_REG_A, M680X_REG_B, M680X_REG_X, M680X_REG_Y, EOL } },
614
21.1k
    { M680X_INS_WAVR,
615
21.1k
      rmmm,
616
21.1k
      { M680X_REG_A, M680X_REG_B, M680X_REG_X, M680X_REG_Y, EOL } },
617
21.1k
  };
618
619
21.1k
  int i, j;
620
621
21.1k
  if (MI->flat_insn->detail == NULL)
622
0
    return;
623
624
1.09M
  for (i = 0; i < ARR_SIZE(changed_regs); ++i) {
625
1.07M
    if (info->insn == changed_regs[i].insn) {
626
21.1k
      e_access_mode access_mode = changed_regs[i].access_mode;
627
628
82.2k
      for (j = 0; changed_regs[i].regs[j] != EOL; ++j) {
629
61.1k
        e_access access;
630
631
61.1k
        m680x_reg reg = changed_regs[i].regs[j];
632
633
61.1k
        if (!info->cpu->reg_byte_size[reg]) {
634
4.01k
          if (info->insn != M680X_INS_MUL)
635
3.78k
            continue;
636
637
          // Hack for M68HC05: MUL uses reg. A,X
638
231
          reg = M680X_REG_X;
639
231
        }
640
641
57.3k
        access = get_access(j, access_mode);
642
57.3k
        add_reg_to_rw_list(MI, reg, access);
643
57.3k
      }
644
21.1k
    }
645
1.07M
  }
646
647
21.1k
#undef EOL
648
21.1k
}
649
650
typedef struct insn_desc {
651
  uint32_t opcode;
652
  m680x_insn insn;
653
  insn_hdlr_id hid[2];
654
  uint16_t insn_size;
655
} insn_desc;
656
657
// If successful return the additional byte size needed for M6809
658
// indexed addressing mode (including the indexed addressing post_byte).
659
// On error return -1.
660
static int get_indexed09_post_byte_size(const m680x_info *info,
661
          uint16_t address)
662
24.3k
{
663
24.3k
  uint8_t ir = 0;
664
24.3k
  uint8_t post_byte;
665
666
  // Read the indexed addressing post byte.
667
24.3k
  if (!read_byte(info, &post_byte, address))
668
74
    return -1;
669
670
  // Depending on the indexed addressing mode more bytes have to be read.
671
24.2k
  switch (post_byte & 0x9F) {
672
494
  case 0x87:
673
995
  case 0x8A:
674
1.68k
  case 0x8E:
675
2.37k
  case 0x8F:
676
2.68k
  case 0x90:
677
3.12k
  case 0x92:
678
3.38k
  case 0x97:
679
3.46k
  case 0x9A:
680
3.58k
  case 0x9E:
681
3.58k
    return -1; // illegal indexed post bytes
682
683
512
  case 0x88: // n8,R
684
909
  case 0x8C: // n8,PCR
685
1.07k
  case 0x98: // [n8,R]
686
1.56k
  case 0x9C: // [n8,PCR]
687
1.56k
    if (!read_byte(info, &ir, address + 1))
688
9
      return -1;
689
1.56k
    return 2;
690
691
470
  case 0x89: // n16,R
692
851
  case 0x8D: // n16,PCR
693
1.50k
  case 0x99: // [n16,R]
694
2.23k
  case 0x9D: // [n16,PCR]
695
2.23k
    if (!read_byte(info, &ir, address + 2))
696
16
      return -1;
697
2.21k
    return 3;
698
699
594
  case 0x9F: // [n]
700
594
    if ((post_byte & 0x60) != 0 ||
701
237
        !read_byte(info, &ir, address + 2))
702
361
      return -1;
703
233
    return 3;
704
24.2k
  }
705
706
  // Any other indexed post byte is valid and
707
  // no additional bytes have to be read.
708
16.2k
  return 1;
709
24.2k
}
710
711
// If successful return the additional byte size needed for CPU12
712
// indexed addressing mode (including the indexed addressing post_byte).
713
// On error return -1.
714
static int get_indexed12_post_byte_size(const m680x_info *info,
715
          uint16_t address, bool is_subset)
716
20.2k
{
717
20.2k
  uint8_t ir;
718
20.2k
  uint8_t post_byte;
719
720
  // Read the indexed addressing post byte.
721
20.2k
  if (!read_byte(info, &post_byte, address))
722
63
    return -1;
723
724
  // Depending on the indexed addressing mode more bytes have to be read.
725
20.1k
  if (!(post_byte & 0x20)) // n5,R
726
7.33k
    return 1;
727
728
12.8k
  switch (post_byte & 0xe7) {
729
937
  case 0xe0:
730
2.10k
  case 0xe1: // n9,R
731
2.10k
    if (is_subset)
732
84
      return -1;
733
734
2.01k
    if (!read_byte(info, &ir, address))
735
0
      return -1;
736
2.01k
    return 2;
737
738
1.29k
  case 0xe2: // n16,R
739
2.36k
  case 0xe3: // [n16,R]
740
2.36k
    if (is_subset)
741
330
      return -1;
742
743
2.03k
    if (!read_byte(info, &ir, address + 1))
744
11
      return -1;
745
2.02k
    return 3;
746
747
302
  case 0xe4: // A,R
748
847
  case 0xe5: // B,R
749
1.38k
  case 0xe6: // D,R
750
2.02k
  case 0xe7: // [D,R]
751
8.38k
  default: // n,-r n,+r n,r- n,r+
752
8.38k
    break;
753
12.8k
  }
754
755
8.38k
  return 1;
756
12.8k
}
757
758
// Check for M6809/HD6309 TFR/EXG instruction for valid register
759
static bool is_tfr09_reg_valid(const m680x_info *info, uint8_t reg_nibble)
760
3.67k
{
761
3.67k
  if (info->cpu->tfr_reg_valid != NULL)
762
1.60k
    return info->cpu->tfr_reg_valid[reg_nibble];
763
764
2.06k
  return true; // e.g. for the M6309 all registers are valid
765
3.67k
}
766
767
// Check for CPU12 TFR/EXG instruction for valid register
768
static bool is_exg_tfr12_post_byte_valid(const m680x_info *info,
769
           uint8_t post_byte)
770
880
{
771
880
  return !(post_byte & 0x08);
772
880
}
773
774
static bool is_tfm_reg_valid(const m680x_info *info, uint8_t reg_nibble)
775
1.60k
{
776
  // HD6809 TFM instruction: Only register X,Y,U,S,D is allowed
777
1.60k
  return reg_nibble <= 4;
778
1.60k
}
779
780
// If successful return the additional byte size needed for CPU12
781
// loop instructions DBEQ/DBNE/IBEQ/IBNE/TBEQ/TBNE (including the post byte).
782
// On error return -1.
783
static int get_loop_post_byte_size(const m680x_info *info, uint16_t address)
784
1.97k
{
785
1.97k
  uint8_t post_byte;
786
1.97k
  uint8_t rr;
787
788
1.97k
  if (!read_byte(info, &post_byte, address))
789
4
    return -1;
790
791
  // According to documentation bit 3 is don't care and not checked here.
792
1.97k
  if ((post_byte >= 0xc0) || ((post_byte & 0x07) == 2) ||
793
1.61k
      ((post_byte & 0x07) == 3))
794
451
    return -1;
795
796
1.52k
  if (!read_byte(info, &rr, address + 1))
797
2
    return -1;
798
799
1.52k
  return 2;
800
1.52k
}
801
802
// If successful return the additional byte size needed for HD6309
803
// bit move instructions BAND/BEOR/BIAND/BIEOR/BIOR/BOR/LDBT/STBT
804
// (including the post byte).
805
// On error return -1.
806
static int get_bitmv_post_byte_size(const m680x_info *info, uint16_t address)
807
1.29k
{
808
1.29k
  uint8_t post_byte;
809
1.29k
  uint8_t rr;
810
811
1.29k
  if (!read_byte(info, &post_byte, address))
812
3
    return -1;
813
814
1.28k
  if ((post_byte & 0xc0) == 0xc0)
815
194
    return -1; // Invalid register specified
816
1.09k
  else {
817
1.09k
    if (!read_byte(info, &rr, address + 1))
818
1
      return -1;
819
1.09k
  }
820
821
1.09k
  return 2;
822
1.28k
}
823
824
static bool is_sufficient_code_size(const m680x_info *info, uint16_t address,
825
            insn_desc *insn_description)
826
198k
{
827
198k
  int i;
828
198k
  bool retval = true;
829
198k
  uint16_t size = 0;
830
198k
  int sz;
831
832
581k
  for (i = 0; i < 2; i++) {
833
390k
    uint8_t ir = 0;
834
390k
    bool is_subset = false;
835
836
390k
    switch (insn_description->hid[i]) {
837
113
    case imm32_hid:
838
113
      if ((retval = read_byte(info, &ir, address + size + 3)))
839
106
        size += 4;
840
113
      break;
841
842
24.3k
    case ext_hid:
843
27.5k
    case imm16_hid:
844
28.2k
    case rel16_hid:
845
29.7k
    case imm8rel_hid:
846
31.5k
    case opidxdr_hid:
847
32.7k
    case idxX16_hid:
848
32.9k
    case idxS16_hid:
849
33.1k
    case dirdir_hid:
850
33.3k
    case immdir_hid:
851
33.3k
      if ((retval = read_byte(info, &ir, address + size + 1)))
852
33.0k
        size += 2;
853
33.3k
      break;
854
855
13.4k
    case rel8_hid:
856
43.8k
    case dir_hid:
857
46.6k
    case rbits_hid:
858
59.1k
    case imm8_hid:
859
62.4k
    case idxX_hid:
860
62.9k
    case idxXp_hid:
861
63.3k
    case idxY_hid:
862
63.4k
    case idxS_hid:
863
63.7k
    case index_hid:
864
63.7k
      if ((retval = read_byte(info, &ir, address + size)))
865
63.5k
        size++;
866
63.7k
      break;
867
868
0
    case illgl_hid:
869
229k
    case inh_hid:
870
234k
    case idxX0_hid:
871
234k
    case idxX0p_hid:
872
235k
    case opidx_hid:
873
238k
    case srt_hid:
874
241k
    case tny_hid:
875
241k
      retval = true;
876
241k
      break;
877
878
24.3k
    case idx09_hid:
879
24.3k
      sz = get_indexed09_post_byte_size(info, address + size);
880
24.3k
      if (sz >= 0)
881
20.2k
        size += sz;
882
4.04k
      else
883
4.04k
        retval = false;
884
24.3k
      break;
885
886
670
    case idx12s_hid:
887
670
      is_subset = true;
888
889
      // intentionally fall through
890
891
17.7k
    case idx12_hid:
892
17.7k
      sz = get_indexed12_post_byte_size(info, address + size,
893
17.7k
                is_subset);
894
17.7k
      if (sz >= 0)
895
17.3k
        size += sz;
896
484
      else
897
484
        retval = false;
898
17.7k
      break;
899
900
964
    case exti12x_hid:
901
1.44k
    case imm16i12x_hid:
902
1.44k
      sz = get_indexed12_post_byte_size(info, address + size,
903
1.44k
                false);
904
1.44k
      if (sz >= 0) {
905
1.44k
        size += sz;
906
1.44k
        if ((retval = read_byte(info, &ir,
907
1.44k
              address + size + 1)))
908
1.43k
          size += 2;
909
1.44k
      } else
910
1
        retval = false;
911
1.44k
      break;
912
913
1.02k
    case imm8i12x_hid:
914
1.02k
      sz = get_indexed12_post_byte_size(info, address + size,
915
1.02k
                false);
916
1.02k
      if (sz >= 0) {
917
1.02k
        size += sz;
918
1.02k
        if ((retval = read_byte(info, &ir,
919
1.02k
              address + size)))
920
1.01k
          size++;
921
1.02k
      } else
922
3
        retval = false;
923
1.02k
      break;
924
925
932
    case tfm_hid:
926
932
      if ((retval = read_byte(info, &ir, address + size))) {
927
930
        size++;
928
930
        retval = is_tfm_reg_valid(info,
929
930
                (ir >> 4) & 0x0F) &&
930
674
           is_tfm_reg_valid(info, ir & 0x0F);
931
930
      }
932
932
      break;
933
934
2.03k
    case rr09_hid:
935
2.03k
      if ((retval = read_byte(info, &ir, address + size))) {
936
2.03k
        size++;
937
2.03k
        retval = is_tfr09_reg_valid(info,
938
2.03k
                  (ir >> 4) & 0x0F) &&
939
1.63k
           is_tfr09_reg_valid(info, ir & 0x0F);
940
2.03k
      }
941
2.03k
      break;
942
943
883
    case rr12_hid:
944
883
      if ((retval = read_byte(info, &ir, address + size))) {
945
880
        size++;
946
880
        retval = is_exg_tfr12_post_byte_valid(info, ir);
947
880
      }
948
883
      break;
949
950
1.29k
    case bitmv_hid:
951
1.29k
      sz = get_bitmv_post_byte_size(info, address + size);
952
1.29k
      if (sz >= 0)
953
1.09k
        size += sz;
954
198
      else
955
198
        retval = false;
956
1.29k
      break;
957
958
1.97k
    case loop_hid:
959
1.97k
      sz = get_loop_post_byte_size(info, address + size);
960
1.97k
      if (sz >= 0)
961
1.52k
        size += sz;
962
457
      else
963
457
        retval = false;
964
1.97k
      break;
965
966
0
    default:
967
0
      CS_ASSERT(0 && "Unexpected instruction handler id");
968
0
      retval = false;
969
0
      break;
970
390k
    }
971
972
390k
    if (!retval)
973
7.29k
      return false;
974
390k
  }
975
976
191k
  insn_description->insn_size += size;
977
978
191k
  return retval;
979
198k
}
980
981
// Check for a valid M680X instruction AND for enough bytes in the code buffer
982
// Return an instruction description in insn_desc.
983
static bool decode_insn(const m680x_info *info, uint16_t address,
984
      insn_desc *insn_description)
985
214k
{
986
214k
  const inst_pageX *inst_table = NULL;
987
214k
  const cpu_tables *cpu = info->cpu;
988
214k
  size_t table_size = 0;
989
214k
  uint16_t base_address = address;
990
214k
  uint8_t ir; // instruction register
991
214k
  int i;
992
214k
  int index;
993
994
214k
  if (!read_byte(info, &ir, address++))
995
0
    return false;
996
997
214k
  insn_description->insn = M680X_INS_ILLGL;
998
214k
  insn_description->opcode = ir;
999
1000
  // Check if a page prefix byte is present
1001
499k
  for (i = 0; i < ARR_SIZE(cpu->pageX_table_size); ++i) {
1002
495k
    if (cpu->pageX_table_size[i] == 0 ||
1003
301k
        (cpu->inst_pageX_table[i] == NULL))
1004
194k
      break;
1005
1006
301k
    if ((cpu->pageX_prefix[i] == ir)) {
1007
      // Get pageX instruction and handler id.
1008
      // Abort for illegal instr.
1009
15.8k
      inst_table = cpu->inst_pageX_table[i];
1010
15.8k
      table_size = cpu->pageX_table_size[i];
1011
1012
15.8k
      if (!read_byte(info, &ir, address++))
1013
18
        return false;
1014
1015
15.8k
      insn_description->opcode =
1016
15.8k
        (insn_description->opcode << 8) | ir;
1017
1018
15.8k
      if ((index = binary_search(inst_table, table_size,
1019
15.8k
               ir)) < 0)
1020
6.55k
        return false;
1021
1022
9.28k
      insn_description->hid[0] =
1023
9.28k
        inst_table[index].handler_id1;
1024
9.28k
      insn_description->hid[1] =
1025
9.28k
        inst_table[index].handler_id2;
1026
9.28k
      insn_description->insn = inst_table[index].insn;
1027
9.28k
      break;
1028
15.8k
    }
1029
301k
  }
1030
1031
207k
  if (insn_description->insn == M680X_INS_ILLGL) {
1032
    // Get page1 insn description
1033
198k
    insn_description->insn = cpu->inst_page1_table[ir].insn;
1034
198k
    insn_description->hid[0] =
1035
198k
      cpu->inst_page1_table[ir].handler_id1;
1036
198k
    insn_description->hid[1] =
1037
198k
      cpu->inst_page1_table[ir].handler_id2;
1038
198k
  }
1039
1040
207k
  if (insn_description->insn == M680X_INS_ILLGL) {
1041
    // Check if opcode byte is present in an overlay table
1042
24.4k
    for (i = 0; i < ARR_SIZE(cpu->overlay_table_size); ++i) {
1043
23.8k
      if (cpu->overlay_table_size[i] == 0 ||
1044
15.3k
          (cpu->inst_overlay_table[i] == NULL))
1045
8.57k
        break;
1046
1047
15.3k
      inst_table = cpu->inst_overlay_table[i];
1048
15.3k
      table_size = cpu->overlay_table_size[i];
1049
1050
15.3k
      if ((index = binary_search(inst_table, table_size,
1051
15.3k
               ir)) >= 0) {
1052
8.01k
        insn_description->hid[0] =
1053
8.01k
          inst_table[index].handler_id1;
1054
8.01k
        insn_description->hid[1] =
1055
8.01k
          inst_table[index].handler_id2;
1056
8.01k
        insn_description->insn = inst_table[index].insn;
1057
8.01k
        break;
1058
8.01k
      }
1059
15.3k
    }
1060
17.1k
  }
1061
1062
207k
  insn_description->insn_size = address - base_address;
1063
1064
207k
  return (insn_description->insn != M680X_INS_ILLGL) &&
1065
198k
         (insn_description->insn != M680X_INS_INVLD) &&
1066
198k
         is_sufficient_code_size(info, address, insn_description);
1067
214k
}
1068
1069
static void illegal_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1070
23.0k
{
1071
23.0k
  cs_m680x_op *op0 = &info->m680x.operands[info->m680x.op_count++];
1072
23.0k
  uint8_t temp8 = 0;
1073
1074
23.0k
  info->insn = M680X_INS_ILLGL;
1075
23.0k
  read_byte(info, &temp8, (*address)++);
1076
23.0k
  op0->imm = (int32_t)temp8 & 0xff;
1077
23.0k
  op0->type = M680X_OP_IMMEDIATE;
1078
23.0k
  op0->size = 1;
1079
23.0k
}
1080
1081
static void inherent_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1082
229k
{
1083
  // There is nothing to do here :-)
1084
229k
}
1085
1086
static void add_reg_operand(m680x_info *info, m680x_reg reg)
1087
122k
{
1088
122k
  cs_m680x *m680x = &info->m680x;
1089
122k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1090
1091
122k
  op->type = M680X_OP_REGISTER;
1092
122k
  op->reg = reg;
1093
122k
  op->size = info->cpu->reg_byte_size[reg];
1094
122k
}
1095
1096
static void set_operand_size(m680x_info *info, cs_m680x_op *op,
1097
           uint8_t default_size)
1098
129k
{
1099
129k
  cs_m680x *m680x = &info->m680x;
1100
1101
129k
  if (info->insn == M680X_INS_JMP || info->insn == M680X_INS_JSR)
1102
7.45k
    op->size = 0;
1103
121k
  else if (info->insn == M680X_INS_DIVD ||
1104
121k
     ((info->insn == M680X_INS_AIS ||
1105
121k
       info->insn == M680X_INS_AIX) &&
1106
930
      op->type != M680X_OP_REGISTER))
1107
1.42k
    op->size = 1;
1108
120k
  else if (info->insn == M680X_INS_DIVQ || info->insn == M680X_INS_MOVW)
1109
3.32k
    op->size = 2;
1110
117k
  else if (info->insn == M680X_INS_EMACS)
1111
90
    op->size = 4;
1112
117k
  else if ((m680x->op_count > 0) &&
1113
117k
     (m680x->operands[0].type == M680X_OP_REGISTER))
1114
69.1k
    op->size = m680x->operands[0].size;
1115
48.0k
  else
1116
48.0k
    op->size = default_size;
1117
129k
}
1118
1119
static const m680x_reg reg_s_reg_ids[] = {
1120
  M680X_REG_CC, M680X_REG_A, M680X_REG_B, M680X_REG_DP,
1121
  M680X_REG_X,  M680X_REG_Y, M680X_REG_U, M680X_REG_PC,
1122
};
1123
1124
static const m680x_reg reg_u_reg_ids[] = {
1125
  M680X_REG_CC, M680X_REG_A, M680X_REG_B, M680X_REG_DP,
1126
  M680X_REG_X,  M680X_REG_Y, M680X_REG_S, M680X_REG_PC,
1127
};
1128
1129
static void reg_bits_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1130
2.76k
{
1131
2.76k
  cs_m680x_op *op0 = &info->m680x.operands[0];
1132
2.76k
  uint8_t reg_bits = 0;
1133
2.76k
  uint16_t bit_index;
1134
2.76k
  const m680x_reg *reg_to_reg_ids = NULL;
1135
1136
2.76k
  read_byte(info, &reg_bits, (*address)++);
1137
1138
2.76k
  switch (op0->reg) {
1139
1.14k
  case M680X_REG_U:
1140
1.14k
    reg_to_reg_ids = &reg_u_reg_ids[0];
1141
1.14k
    break;
1142
1143
1.62k
  case M680X_REG_S:
1144
1.62k
    reg_to_reg_ids = &reg_s_reg_ids[0];
1145
1.62k
    break;
1146
1147
0
  default:
1148
0
    CS_ASSERT(0 && "Unexpected operand0 register");
1149
0
    break;
1150
2.76k
  }
1151
1152
2.76k
  if ((info->insn == M680X_INS_PULU || (info->insn == M680X_INS_PULS)) &&
1153
2.05k
      ((reg_bits & 0x80) != 0))
1154
    // PULS xxx,PC or PULU xxx,PC which is like return from
1155
    // subroutine (RTS)
1156
447
    add_insn_group(MI->flat_insn->detail, M680X_GRP_RET);
1157
1158
24.9k
  for (bit_index = 0; bit_index < 8; ++bit_index) {
1159
22.1k
    if (reg_bits & (1 << bit_index) && reg_to_reg_ids)
1160
12.1k
      add_reg_operand(info, reg_to_reg_ids[bit_index]);
1161
22.1k
  }
1162
2.76k
}
1163
1164
static const m680x_reg g_tfr_exg_reg_ids[] = {
1165
  /* 16-bit registers */
1166
  M680X_REG_D,
1167
  M680X_REG_X,
1168
  M680X_REG_Y,
1169
  M680X_REG_U,
1170
  M680X_REG_S,
1171
  M680X_REG_PC,
1172
  M680X_REG_W,
1173
  M680X_REG_V,
1174
  /* 8-bit registers */
1175
  M680X_REG_A,
1176
  M680X_REG_B,
1177
  M680X_REG_CC,
1178
  M680X_REG_DP,
1179
  M680X_REG_0,
1180
  M680X_REG_0,
1181
  M680X_REG_E,
1182
  M680X_REG_F,
1183
};
1184
1185
static void reg_reg09_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1186
1.13k
{
1187
1.13k
  uint8_t regs = 0;
1188
1189
1.13k
  read_byte(info, &regs, (*address)++);
1190
1191
1.13k
  add_reg_operand(info, g_tfr_exg_reg_ids[regs >> 4]);
1192
1.13k
  add_reg_operand(info, g_tfr_exg_reg_ids[regs & 0x0f]);
1193
1194
1.13k
  if ((regs & 0x0f) == 0x05) {
1195
    // EXG xxx,PC or TFR xxx,PC which is like a JMP
1196
148
    add_insn_group(MI->flat_insn->detail, M680X_GRP_JUMP);
1197
148
  }
1198
1.13k
}
1199
1200
static void reg_reg12_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1201
774
{
1202
774
  static const m680x_reg g_tfr_exg12_reg0_ids[] = {
1203
774
    M680X_REG_A, M680X_REG_B, M680X_REG_CC, M680X_REG_TMP3,
1204
774
    M680X_REG_D, M680X_REG_X, M680X_REG_Y,  M680X_REG_S,
1205
774
  };
1206
774
  static const m680x_reg g_tfr_exg12_reg1_ids[] = {
1207
774
    M680X_REG_A, M680X_REG_B, M680X_REG_CC, M680X_REG_TMP2,
1208
774
    M680X_REG_D, M680X_REG_X, M680X_REG_Y,  M680X_REG_S,
1209
774
  };
1210
774
  uint8_t regs = 0;
1211
1212
774
  read_byte(info, &regs, (*address)++);
1213
1214
  // The opcode of this instruction depends on
1215
  // the msb of its post byte.
1216
774
  if (regs & 0x80)
1217
657
    info->insn = M680X_INS_EXG;
1218
117
  else
1219
117
    info->insn = M680X_INS_TFR;
1220
1221
774
  add_reg_operand(info, g_tfr_exg12_reg0_ids[(regs >> 4) & 0x07]);
1222
774
  add_reg_operand(info, g_tfr_exg12_reg1_ids[regs & 0x07]);
1223
774
}
1224
1225
static void add_rel_operand(m680x_info *info, int16_t offset, uint16_t address)
1226
17.3k
{
1227
17.3k
  cs_m680x *m680x = &info->m680x;
1228
17.3k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1229
1230
17.3k
  op->type = M680X_OP_RELATIVE;
1231
17.3k
  op->size = 0;
1232
17.3k
  op->rel.offset = offset;
1233
17.3k
  op->rel.address = address;
1234
17.3k
}
1235
1236
static void relative8_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1237
16.5k
{
1238
16.5k
  int16_t offset = 0;
1239
1240
16.5k
  read_byte_sign_extended(info, &offset, (*address)++);
1241
16.5k
  add_rel_operand(info, offset, *address + offset);
1242
16.5k
  add_insn_group(MI->flat_insn->detail, M680X_GRP_BRAREL);
1243
1244
16.5k
  if ((info->insn != M680X_INS_BRA) && (info->insn != M680X_INS_BSR) &&
1245
14.5k
      (info->insn != M680X_INS_BRN))
1246
13.4k
    add_reg_to_rw_list(MI, M680X_REG_CC, READ);
1247
16.5k
}
1248
1249
static void relative16_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1250
741
{
1251
741
  uint16_t offset = 0;
1252
1253
741
  read_word(info, &offset, *address);
1254
741
  *address += 2;
1255
741
  add_rel_operand(info, (int16_t)offset, *address + offset);
1256
741
  add_insn_group(MI->flat_insn->detail, M680X_GRP_BRAREL);
1257
1258
741
  if ((info->insn != M680X_INS_LBRA) && (info->insn != M680X_INS_LBSR) &&
1259
263
      (info->insn != M680X_INS_LBRN))
1260
243
    add_reg_to_rw_list(MI, M680X_REG_CC, READ);
1261
741
}
1262
1263
static const m680x_reg g_rr5_to_reg_ids[] = {
1264
  M680X_REG_X,
1265
  M680X_REG_Y,
1266
  M680X_REG_U,
1267
  M680X_REG_S,
1268
};
1269
1270
static void add_indexed_operand(m680x_info *info, m680x_reg base_reg,
1271
        bool post_inc_dec, uint8_t inc_dec,
1272
        uint8_t offset_bits, uint16_t offset,
1273
        bool no_comma)
1274
11.7k
{
1275
11.7k
  cs_m680x *m680x = &info->m680x;
1276
11.7k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1277
1278
11.7k
  op->type = M680X_OP_INDEXED;
1279
11.7k
  set_operand_size(info, op, 1);
1280
11.7k
  op->idx.base_reg = base_reg;
1281
11.7k
  op->idx.offset_reg = M680X_REG_INVALID;
1282
11.7k
  op->idx.inc_dec = inc_dec;
1283
1284
11.7k
  if (inc_dec && post_inc_dec)
1285
1.58k
    op->idx.flags |= M680X_IDX_POST_INC_DEC;
1286
1287
11.7k
  if (offset_bits != M680X_OFFSET_NONE) {
1288
5.65k
    op->idx.offset = offset;
1289
5.65k
    op->idx.offset_addr = 0;
1290
5.65k
  }
1291
1292
11.7k
  op->idx.offset_bits = offset_bits;
1293
11.7k
  op->idx.flags |= (no_comma ? M680X_IDX_NO_COMMA : 0);
1294
11.7k
}
1295
1296
// M6800/1/2/3 indexed mode handler
1297
static void indexedX_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1298
3.35k
{
1299
3.35k
  uint8_t offset = 0;
1300
1301
3.35k
  read_byte(info, &offset, (*address)++);
1302
1303
3.35k
  add_indexed_operand(info, M680X_REG_X, false, 0, M680X_OFFSET_BITS_8,
1304
3.35k
          (uint16_t)offset, false);
1305
3.35k
}
1306
1307
static void indexedY_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1308
455
{
1309
455
  uint8_t offset = 0;
1310
1311
455
  read_byte(info, &offset, (*address)++);
1312
1313
455
  add_indexed_operand(info, M680X_REG_Y, false, 0, M680X_OFFSET_BITS_8,
1314
455
          (uint16_t)offset, false);
1315
455
}
1316
1317
// M6809/M6309 indexed mode handler
1318
static void indexed09_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1319
20.2k
{
1320
20.2k
  cs_m680x *m680x = &info->m680x;
1321
20.2k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1322
20.2k
  uint8_t post_byte = 0;
1323
20.2k
  uint16_t offset = 0;
1324
20.2k
  int16_t soffset = 0;
1325
1326
20.2k
  read_byte(info, &post_byte, (*address)++);
1327
1328
20.2k
  op->type = M680X_OP_INDEXED;
1329
20.2k
  set_operand_size(info, op, 1);
1330
20.2k
  op->idx.base_reg = g_rr5_to_reg_ids[(post_byte >> 5) & 0x03];
1331
20.2k
  op->idx.offset_reg = M680X_REG_INVALID;
1332
1333
20.2k
  if (!(post_byte & 0x80)) {
1334
    // n5,R
1335
10.0k
    if ((post_byte & 0x10) == 0x10)
1336
4.22k
      op->idx.offset = post_byte | 0xfff0;
1337
5.83k
    else
1338
5.83k
      op->idx.offset = post_byte & 0x0f;
1339
1340
10.0k
    op->idx.offset_addr = op->idx.offset + *address;
1341
10.0k
    op->idx.offset_bits = M680X_OFFSET_BITS_5;
1342
10.2k
  } else {
1343
10.2k
    if ((post_byte & 0x10) == 0x10)
1344
3.99k
      op->idx.flags |= M680X_IDX_INDIRECT;
1345
1346
    // indexed addressing
1347
10.2k
    switch (post_byte & 0x1f) {
1348
888
    case 0x00: // ,R+
1349
888
      op->idx.inc_dec = 1;
1350
888
      op->idx.flags |= M680X_IDX_POST_INC_DEC;
1351
888
      break;
1352
1353
273
    case 0x11: // [,R++]
1354
911
    case 0x01: // ,R++
1355
911
      op->idx.inc_dec = 2;
1356
911
      op->idx.flags |= M680X_IDX_POST_INC_DEC;
1357
911
      break;
1358
1359
388
    case 0x02: // ,-R
1360
388
      op->idx.inc_dec = -1;
1361
388
      break;
1362
1363
691
    case 0x13: // [,--R]
1364
1.19k
    case 0x03: // ,--R
1365
1.19k
      op->idx.inc_dec = -2;
1366
1.19k
      break;
1367
1368
142
    case 0x14: // [,R]
1369
703
    case 0x04: // ,R
1370
703
      break;
1371
1372
247
    case 0x15: // [B,R]
1373
672
    case 0x05: // B,R
1374
672
      op->idx.offset_reg = M680X_REG_B;
1375
672
      break;
1376
1377
130
    case 0x16: // [A,R]
1378
869
    case 0x06: // A,R
1379
869
      op->idx.offset_reg = M680X_REG_A;
1380
869
      break;
1381
1382
488
    case 0x1c: // [n8,PCR]
1383
881
    case 0x0c: // n8,PCR
1384
881
      op->idx.base_reg = M680X_REG_PC;
1385
881
      read_byte_sign_extended(info, &soffset, (*address)++);
1386
881
      op->idx.offset_addr = offset + *address;
1387
881
      op->idx.offset = soffset;
1388
881
      op->idx.offset_bits = M680X_OFFSET_BITS_8;
1389
881
      break;
1390
1391
167
    case 0x18: // [n8,R]
1392
679
    case 0x08: // n8,R
1393
679
      read_byte_sign_extended(info, &soffset, (*address)++);
1394
679
      op->idx.offset = soffset;
1395
679
      op->idx.offset_bits = M680X_OFFSET_BITS_8;
1396
679
      break;
1397
1398
728
    case 0x1d: // [n16,PCR]
1399
1.10k
    case 0x0d: // n16,PCR
1400
1.10k
      op->idx.base_reg = M680X_REG_PC;
1401
1.10k
      read_word(info, &offset, *address);
1402
1.10k
      *address += 2;
1403
1.10k
      op->idx.offset_addr = offset + *address;
1404
1.10k
      op->idx.offset = (int16_t)offset;
1405
1.10k
      op->idx.offset_bits = M680X_OFFSET_BITS_16;
1406
1.10k
      break;
1407
1408
648
    case 0x19: // [n16,R]
1409
1.10k
    case 0x09: // n16,R
1410
1.10k
      read_word(info, &offset, *address);
1411
1.10k
      *address += 2;
1412
1.10k
      op->idx.offset = (int16_t)offset;
1413
1.10k
      op->idx.offset_bits = M680X_OFFSET_BITS_16;
1414
1.10k
      break;
1415
1416
246
    case 0x1b: // [D,R]
1417
592
    case 0x0b: // D,R
1418
592
      op->idx.offset_reg = M680X_REG_D;
1419
592
      break;
1420
1421
233
    case 0x1f: // [n16]
1422
233
      op->type = M680X_OP_EXTENDED;
1423
233
      op->ext.indirect = true;
1424
233
      read_word(info, &op->ext.address, *address);
1425
233
      *address += 2;
1426
233
      break;
1427
1428
0
    default:
1429
0
      op->idx.base_reg = M680X_REG_INVALID;
1430
0
      break;
1431
10.2k
    }
1432
10.2k
  }
1433
1434
20.2k
  if (((info->insn == M680X_INS_LEAU) || (info->insn == M680X_INS_LEAS) ||
1435
18.7k
       (info->insn == M680X_INS_LEAX) ||
1436
17.8k
       (info->insn == M680X_INS_LEAY)) &&
1437
3.05k
      (m680x->operands[0].reg == M680X_REG_X ||
1438
2.15k
       (m680x->operands[0].reg == M680X_REG_Y)))
1439
    // Only LEAX and LEAY modify CC register
1440
1.55k
    add_reg_to_rw_list(MI, M680X_REG_CC, MODIFY);
1441
20.2k
}
1442
1443
static const m680x_reg g_idx12_to_reg_ids[4] = {
1444
  M680X_REG_X,
1445
  M680X_REG_Y,
1446
  M680X_REG_S,
1447
  M680X_REG_PC,
1448
};
1449
1450
static const m680x_reg g_or12_to_reg_ids[3] = { M680X_REG_A, M680X_REG_B,
1451
            M680X_REG_D };
1452
1453
// CPU12 indexed mode handler
1454
static void indexed12_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1455
19.7k
{
1456
19.7k
  cs_m680x *m680x = &info->m680x;
1457
19.7k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1458
19.7k
  uint8_t post_byte = 0;
1459
19.7k
  uint8_t offset8 = 0;
1460
1461
19.7k
  read_byte(info, &post_byte, (*address)++);
1462
1463
19.7k
  op->type = M680X_OP_INDEXED;
1464
19.7k
  set_operand_size(info, op, 1);
1465
19.7k
  op->idx.offset_reg = M680X_REG_INVALID;
1466
1467
19.7k
  if (!(post_byte & 0x20)) {
1468
    // n5,R      n5 is a 5-bit signed offset
1469
7.32k
    op->idx.base_reg = g_idx12_to_reg_ids[(post_byte >> 6) & 0x03];
1470
1471
7.32k
    if ((post_byte & 0x10) == 0x10)
1472
2.67k
      op->idx.offset = post_byte | 0xfff0;
1473
4.64k
    else
1474
4.64k
      op->idx.offset = post_byte & 0x0f;
1475
1476
7.32k
    op->idx.offset_addr = op->idx.offset + *address;
1477
7.32k
    op->idx.offset_bits = M680X_OFFSET_BITS_5;
1478
12.4k
  } else {
1479
12.4k
    if ((post_byte & 0xe0) == 0xe0)
1480
6.05k
      op->idx.base_reg =
1481
6.05k
        g_idx12_to_reg_ids[(post_byte >> 3) & 0x03];
1482
1483
12.4k
    switch (post_byte & 0xe7) {
1484
856
    case 0xe0:
1485
2.01k
    case 0xe1: // n9,R
1486
2.01k
      read_byte(info, &offset8, (*address)++);
1487
2.01k
      op->idx.offset = offset8;
1488
1489
2.01k
      if (post_byte & 0x01) // sign extension
1490
1.15k
        op->idx.offset |= 0xff00;
1491
1492
2.01k
      op->idx.offset_bits = M680X_OFFSET_BITS_9;
1493
1494
2.01k
      if (op->idx.base_reg == M680X_REG_PC)
1495
397
        op->idx.offset_addr = op->idx.offset + *address;
1496
1497
2.01k
      break;
1498
1499
1.06k
    case 0xe3: // [n16,R]
1500
1.06k
      op->idx.flags |= M680X_IDX_INDIRECT;
1501
1502
    // intentionally fall through
1503
2.02k
    case 0xe2: // n16,R
1504
2.02k
      read_word(info, (uint16_t *)&op->idx.offset, *address);
1505
2.02k
      (*address) += 2;
1506
2.02k
      op->idx.offset_bits = M680X_OFFSET_BITS_16;
1507
1508
2.02k
      if (op->idx.base_reg == M680X_REG_PC)
1509
173
        op->idx.offset_addr = op->idx.offset + *address;
1510
1511
2.02k
      break;
1512
1513
302
    case 0xe4: // A,R
1514
847
    case 0xe5: // B,R
1515
1.38k
    case 0xe6: // D,R
1516
1.38k
      op->idx.offset_reg =
1517
1.38k
        g_or12_to_reg_ids[post_byte & 0x03];
1518
1.38k
      break;
1519
1520
643
    case 0xe7: // [D,R]
1521
643
      op->idx.offset_reg = M680X_REG_D;
1522
643
      op->idx.flags |= M680X_IDX_INDIRECT;
1523
643
      break;
1524
1525
6.35k
    default: // n,-r n,+r n,r- n,r+
1526
      // PC is not allowed in this mode
1527
6.35k
      op->idx.base_reg =
1528
6.35k
        g_idx12_to_reg_ids[(post_byte >> 6) & 0x03];
1529
6.35k
      op->idx.inc_dec = post_byte & 0x0f;
1530
1531
6.35k
      if (op->idx.inc_dec & 0x08) // evtl. sign extend value
1532
3.30k
        op->idx.inc_dec |= 0xf0;
1533
1534
6.35k
      if (op->idx.inc_dec >= 0)
1535
3.05k
        op->idx.inc_dec++;
1536
1537
6.35k
      if (post_byte & 0x10)
1538
1.74k
        op->idx.flags |= M680X_IDX_POST_INC_DEC;
1539
1540
6.35k
      break;
1541
12.4k
    }
1542
12.4k
  }
1543
19.7k
}
1544
1545
static void index_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1546
319
{
1547
319
  cs_m680x *m680x = &info->m680x;
1548
319
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1549
1550
319
  op->type = M680X_OP_CONSTANT;
1551
319
  read_byte(info, &op->const_val, (*address)++);
1552
319
};
1553
1554
static void direct_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1555
33.8k
{
1556
33.8k
  cs_m680x *m680x = &info->m680x;
1557
33.8k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1558
1559
33.8k
  op->type = M680X_OP_DIRECT;
1560
33.8k
  set_operand_size(info, op, 1);
1561
33.8k
  read_byte(info, &op->direct_addr, (*address)++);
1562
33.8k
};
1563
1564
static void extended_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1565
24.1k
{
1566
24.1k
  cs_m680x *m680x = &info->m680x;
1567
24.1k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1568
1569
24.1k
  op->type = M680X_OP_EXTENDED;
1570
24.1k
  set_operand_size(info, op, 1);
1571
24.1k
  read_word(info, &op->ext.address, *address);
1572
24.1k
  *address += 2;
1573
24.1k
}
1574
1575
static void immediate_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1576
17.2k
{
1577
17.2k
  cs_m680x *m680x = &info->m680x;
1578
17.2k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1579
17.2k
  uint16_t word = 0;
1580
17.2k
  int16_t sword = 0;
1581
1582
17.2k
  op->type = M680X_OP_IMMEDIATE;
1583
17.2k
  set_operand_size(info, op, 1);
1584
1585
17.2k
  switch (op->size) {
1586
14.0k
  case 1:
1587
14.0k
    read_byte_sign_extended(info, &sword, *address);
1588
14.0k
    op->imm = sword;
1589
14.0k
    break;
1590
1591
3.14k
  case 2:
1592
3.14k
    read_word(info, &word, *address);
1593
3.14k
    op->imm = (int16_t)word;
1594
3.14k
    break;
1595
1596
106
  case 4:
1597
106
    read_sdword(info, &op->imm, *address);
1598
106
    break;
1599
1600
0
  default:
1601
0
    op->imm = 0;
1602
0
    CS_ASSERT(0 && "Unexpected immediate byte size");
1603
17.2k
  }
1604
1605
17.2k
  *address += op->size;
1606
17.2k
}
1607
1608
// handler for bit move instructions, e.g: BAND A,5,1,$40  Used by HD6309
1609
static void bit_move_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1610
1.09k
{
1611
1.09k
  static const m680x_reg m680x_reg[] = {
1612
1.09k
    M680X_REG_CC,
1613
1.09k
    M680X_REG_A,
1614
1.09k
    M680X_REG_B,
1615
1.09k
    M680X_REG_INVALID,
1616
1.09k
  };
1617
1618
1.09k
  uint8_t post_byte = 0;
1619
1.09k
  cs_m680x *m680x = &info->m680x;
1620
1.09k
  cs_m680x_op *op;
1621
1622
1.09k
  read_byte(info, &post_byte, *address);
1623
1.09k
  (*address)++;
1624
1625
  // operand[0] = register
1626
1.09k
  add_reg_operand(info, m680x_reg[post_byte >> 6]);
1627
1628
  // operand[1] = bit index in source operand
1629
1.09k
  op = &m680x->operands[m680x->op_count++];
1630
1.09k
  op->type = M680X_OP_CONSTANT;
1631
1.09k
  op->const_val = (post_byte >> 3) & 0x07;
1632
1633
  // operand[2] = bit index in destination operand
1634
1.09k
  op = &m680x->operands[m680x->op_count++];
1635
1.09k
  op->type = M680X_OP_CONSTANT;
1636
1.09k
  op->const_val = post_byte & 0x07;
1637
1638
1.09k
  direct_hdlr(MI, info, address);
1639
1.09k
}
1640
1641
// handler for TFM instruction, e.g: TFM X+,Y+  Used by HD6309
1642
static void tfm_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1643
417
{
1644
417
  static const uint8_t inc_dec_r0[] = {
1645
417
    1,
1646
417
    -1,
1647
417
    1,
1648
417
    0,
1649
417
  };
1650
417
  static const uint8_t inc_dec_r1[] = {
1651
417
    1,
1652
417
    -1,
1653
417
    0,
1654
417
    1,
1655
417
  };
1656
417
  uint8_t regs = 0;
1657
417
  uint8_t index = (MI->Opcode & 0xff) - 0x38;
1658
1659
417
  read_byte(info, &regs, *address);
1660
1661
417
  add_indexed_operand(info, g_tfr_exg_reg_ids[regs >> 4], true,
1662
417
          inc_dec_r0[index], M680X_OFFSET_NONE, 0, true);
1663
417
  add_indexed_operand(info, g_tfr_exg_reg_ids[regs & 0x0f], true,
1664
417
          inc_dec_r1[index], M680X_OFFSET_NONE, 0, true);
1665
1666
417
  add_reg_to_rw_list(MI, M680X_REG_W, READ | WRITE);
1667
417
}
1668
1669
static void opidx_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1670
1.12k
{
1671
1.12k
  cs_m680x *m680x = &info->m680x;
1672
1.12k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1673
1674
  // bit index is coded in Opcode
1675
1.12k
  op->type = M680X_OP_CONSTANT;
1676
1.12k
  op->const_val = (MI->Opcode & 0x0e) >> 1;
1677
1.12k
}
1678
1679
// handler for bit test and branch instruction. Used by M6805.
1680
// The bit index is part of the opcode.
1681
// Example: BRSET 3,<$40,LOOP
1682
static void opidx_dir_rel_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1683
1.76k
{
1684
1.76k
  cs_m680x *m680x = &info->m680x;
1685
1.76k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1686
1687
  // bit index is coded in Opcode
1688
1.76k
  op->type = M680X_OP_CONSTANT;
1689
1.76k
  op->const_val = (MI->Opcode & 0x0e) >> 1;
1690
1.76k
  direct_hdlr(MI, info, address);
1691
1.76k
  relative8_hdlr(MI, info, address);
1692
1693
1.76k
  add_reg_to_rw_list(MI, M680X_REG_CC, MODIFY);
1694
1.76k
}
1695
1696
static void indexedX0_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1697
4.51k
{
1698
4.51k
  add_indexed_operand(info, M680X_REG_X, false, 0, M680X_OFFSET_NONE, 0,
1699
4.51k
          false);
1700
4.51k
}
1701
1702
static void indexedX16_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1703
1.15k
{
1704
1.15k
  uint16_t offset = 0;
1705
1706
1.15k
  read_word(info, &offset, *address);
1707
1.15k
  *address += 2;
1708
1.15k
  add_indexed_operand(info, M680X_REG_X, false, 0, M680X_OFFSET_BITS_16,
1709
1.15k
          offset, false);
1710
1.15k
}
1711
1712
static void imm_rel_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1713
1.46k
{
1714
1.46k
  immediate_hdlr(MI, info, address);
1715
1.46k
  relative8_hdlr(MI, info, address);
1716
1.46k
}
1717
1718
static void indexedS_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1719
70
{
1720
70
  uint8_t offset = 0;
1721
1722
70
  read_byte(info, &offset, (*address)++);
1723
1724
70
  add_indexed_operand(info, M680X_REG_S, false, 0, M680X_OFFSET_BITS_8,
1725
70
          (uint16_t)offset, false);
1726
70
}
1727
1728
static void indexedS16_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1729
196
{
1730
196
  uint16_t offset = 0;
1731
1732
196
  read_word(info, &offset, *address);
1733
196
  *address += 2;
1734
1735
196
  add_indexed_operand(info, M680X_REG_S, false, 0, M680X_OFFSET_BITS_16,
1736
196
          offset, false);
1737
196
}
1738
1739
static void indexedX0p_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1740
703
{
1741
703
  add_indexed_operand(info, M680X_REG_X, true, 1, M680X_OFFSET_NONE, 0,
1742
703
          true);
1743
703
}
1744
1745
static void indexedXp_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1746
427
{
1747
427
  uint8_t offset = 0;
1748
1749
427
  read_byte(info, &offset, (*address)++);
1750
1751
427
  add_indexed_operand(info, M680X_REG_X, true, 1, M680X_OFFSET_BITS_8,
1752
427
          (uint16_t)offset, false);
1753
427
}
1754
1755
static void imm_idx12_x_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1756
1.49k
{
1757
1.49k
  cs_m680x *m680x = &info->m680x;
1758
1.49k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1759
1760
1.49k
  indexed12_hdlr(MI, info, address);
1761
1.49k
  op->type = M680X_OP_IMMEDIATE;
1762
1763
1.49k
  if (info->insn == M680X_INS_MOVW) {
1764
473
    uint16_t imm16 = 0;
1765
1766
473
    read_word(info, &imm16, *address);
1767
473
    op->imm = (int16_t)imm16;
1768
473
    op->size = 2;
1769
1.01k
  } else {
1770
1.01k
    uint8_t imm8 = 0;
1771
1772
1.01k
    read_byte(info, &imm8, *address);
1773
1.01k
    op->imm = (int8_t)imm8;
1774
1.01k
    op->size = 1;
1775
1.01k
  }
1776
1777
1.49k
  set_operand_size(info, op, 1);
1778
1.49k
}
1779
1780
static void ext_idx12_x_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1781
960
{
1782
960
  cs_m680x *m680x = &info->m680x;
1783
960
  cs_m680x_op *op0 = &m680x->operands[m680x->op_count++];
1784
960
  uint16_t imm16 = 0;
1785
1786
960
  indexed12_hdlr(MI, info, address);
1787
960
  read_word(info, &imm16, *address);
1788
960
  op0->type = M680X_OP_EXTENDED;
1789
960
  op0->ext.address = (int16_t)imm16;
1790
960
  set_operand_size(info, op0, 1);
1791
960
}
1792
1793
// handler for CPU12 DBEQ/DNBE/IBEQ/IBNE/TBEQ/TBNE instructions.
1794
// Example: DBNE X,$1000
1795
static void loop_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1796
1.52k
{
1797
1.52k
  static const m680x_reg index_to_reg_id[] = {
1798
1.52k
    M680X_REG_A, M680X_REG_B, M680X_REG_INVALID, M680X_REG_INVALID,
1799
1.52k
    M680X_REG_D, M680X_REG_X, M680X_REG_Y,       M680X_REG_S,
1800
1.52k
  };
1801
1.52k
  static const m680x_insn index_to_insn_id[] = {
1802
1.52k
    M680X_INS_DBEQ, M680X_INS_DBNE, M680X_INS_TBEQ,  M680X_INS_TBNE,
1803
1.52k
    M680X_INS_IBEQ, M680X_INS_IBNE, M680X_INS_ILLGL, M680X_INS_ILLGL
1804
1.52k
  };
1805
1.52k
  cs_m680x *m680x = &info->m680x;
1806
1.52k
  uint8_t post_byte = 0;
1807
1.52k
  uint8_t rel = 0;
1808
1.52k
  cs_m680x_op *op;
1809
1810
1.52k
  read_byte(info, &post_byte, (*address)++);
1811
1812
1.52k
  info->insn = index_to_insn_id[(post_byte >> 5) & 0x07];
1813
1814
1.52k
  if (info->insn == M680X_INS_ILLGL) {
1815
0
    illegal_hdlr(MI, info, address);
1816
0
  };
1817
1818
1.52k
  read_byte(info, &rel, (*address)++);
1819
1820
1.52k
  add_reg_operand(info, index_to_reg_id[post_byte & 0x07]);
1821
1822
1.52k
  op = &m680x->operands[m680x->op_count++];
1823
1824
1.52k
  op->type = M680X_OP_RELATIVE;
1825
1826
1.52k
  op->rel.offset = (post_byte & 0x10) ? (int16_t)(0xff00 | rel) : rel;
1827
1828
1.52k
  op->rel.address = *address + op->rel.offset;
1829
1830
1.52k
  add_insn_group(MI->flat_insn->detail, M680X_GRP_BRAREL);
1831
1.52k
}
1832
1833
// handler for RS08 specific TNY instruction
1834
// The operand address is embedded in the the least 4 significant bits of the opcode
1835
static void tny_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1836
2.90k
{
1837
2.90k
  cs_m680x *m680x = &info->m680x;
1838
2.90k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1839
1840
2.90k
  op->type = M680X_OP_DIRECT;
1841
2.90k
  op->direct_addr = (uint8_t)(MI->Opcode & 0x0F);
1842
2.90k
  op->size = 1;
1843
2.90k
}
1844
1845
// handler for RS08 specific SRT instruction
1846
// The operand address is embedded in the the least 5 significant bits of the opcode
1847
static void srt_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1848
2.60k
{
1849
2.60k
  cs_m680x *m680x = &info->m680x;
1850
2.60k
  cs_m680x_op *op = &m680x->operands[m680x->op_count++];
1851
1852
2.60k
  op->type = M680X_OP_DIRECT;
1853
2.60k
  op->direct_addr = (uint8_t)(MI->Opcode & 0x1F);
1854
2.60k
  op->size = 1;
1855
2.60k
}
1856
1857
static void dirdir_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1858
210
{
1859
210
  direct_hdlr(MI, info, address);
1860
210
  direct_hdlr(MI, info, address);
1861
210
}
1862
1863
static void immdir_hdlr(MCInst *MI, m680x_info *info, uint16_t *address)
1864
217
{
1865
217
  immediate_hdlr(MI, info, address);
1866
217
  direct_hdlr(MI, info, address);
1867
217
}
1868
1869
static void (*const g_insn_handler[])(MCInst *, m680x_info *, uint16_t *) = {
1870
  illegal_hdlr,   relative8_hdlr,   relative16_hdlr,
1871
  immediate_hdlr, // 8-bit
1872
  immediate_hdlr, // 16-bit
1873
  immediate_hdlr, // 32-bit
1874
  direct_hdlr,    extended_hdlr,    indexedX_hdlr,   indexedY_hdlr,
1875
  indexed09_hdlr,   inherent_hdlr,    reg_reg09_hdlr,  reg_bits_hdlr,
1876
  bit_move_hdlr,    tfm_hdlr,     opidx_hdlr,      opidx_dir_rel_hdlr,
1877
  indexedX0_hdlr,   indexedX16_hdlr,  imm_rel_hdlr,    indexedS_hdlr,
1878
  indexedS16_hdlr,  indexedXp_hdlr,   indexedX0p_hdlr, indexed12_hdlr,
1879
  indexed12_hdlr, // subset of indexed12
1880
  reg_reg12_hdlr,   loop_hdlr,      index_hdlr,      imm_idx12_x_hdlr,
1881
  imm_idx12_x_hdlr, ext_idx12_x_hdlr, srt_hdlr,      tny_hdlr,
1882
  dirdir_hdlr,    immdir_hdlr
1883
}; /* handler function pointers */
1884
1885
/* Disasemble one instruction at address and store in str_buff */
1886
static unsigned int m680x_disassemble(MCInst *MI, m680x_info *info,
1887
              uint16_t address)
1888
214k
{
1889
214k
  cs_m680x *m680x = &info->m680x;
1890
214k
  cs_detail *detail = MI->flat_insn->detail;
1891
214k
  uint16_t base_address = address;
1892
214k
  insn_desc insn_description;
1893
214k
  e_access_mode access_mode;
1894
1895
214k
  if (detail != NULL) {
1896
214k
    memset(detail, 0,
1897
214k
           offsetof(cs_detail, m680x) + sizeof(cs_m680x));
1898
214k
  }
1899
1900
214k
  memset(&insn_description, 0, sizeof(insn_description));
1901
214k
  memset(m680x, 0, sizeof(*m680x));
1902
214k
  info->insn_size = 1;
1903
1904
214k
  if (decode_insn(info, address, &insn_description)) {
1905
191k
    m680x_reg reg;
1906
1907
191k
    if (insn_description.opcode > 0xff)
1908
8.09k
      address += 2; // 8-bit opcode + page prefix
1909
183k
    else
1910
183k
      address++; // 8-bit opcode only
1911
1912
191k
    info->insn = insn_description.insn;
1913
1914
191k
    MCInst_setOpcode(MI, insn_description.opcode);
1915
1916
191k
    reg = g_insn_props[info->insn].reg0;
1917
1918
191k
    if (reg != M680X_REG_INVALID) {
1919
101k
      if (reg == M680X_REG_HX &&
1920
1.03k
          (!info->cpu->reg_byte_size[reg]))
1921
204
        reg = M680X_REG_X;
1922
1923
101k
      add_reg_operand(info, reg);
1924
      // First (or second) operand is a register which is
1925
      // part of the mnemonic
1926
101k
      m680x->flags |= M680X_FIRST_OP_IN_MNEM;
1927
101k
      reg = g_insn_props[info->insn].reg1;
1928
1929
101k
      if (reg != M680X_REG_INVALID) {
1930
2.46k
        if (reg == M680X_REG_HX &&
1931
572
            (!info->cpu->reg_byte_size[reg]))
1932
355
          reg = M680X_REG_X;
1933
1934
2.46k
        add_reg_operand(info, reg);
1935
2.46k
        m680x->flags |= M680X_SECOND_OP_IN_MNEM;
1936
2.46k
      }
1937
101k
    }
1938
1939
    // Call addressing mode specific instruction handler
1940
191k
    (g_insn_handler[insn_description.hid[0]])(MI, info, &address);
1941
191k
    (g_insn_handler[insn_description.hid[1]])(MI, info, &address);
1942
1943
191k
    add_insn_group(detail, g_insn_props[info->insn].group);
1944
1945
191k
    if (g_insn_props[info->insn].cc_modified &&
1946
125k
        (info->cpu->insn_cc_not_modified[0] != info->insn) &&
1947
125k
        (info->cpu->insn_cc_not_modified[1] != info->insn))
1948
124k
      add_reg_to_rw_list(MI, M680X_REG_CC, MODIFY);
1949
1950
191k
    access_mode = g_insn_props[info->insn].access_mode;
1951
1952
    // Fix for M6805 BSET/BCLR. It has a different operand order
1953
    // in comparison to the M6811
1954
191k
    if ((info->cpu->insn_cc_not_modified[0] == info->insn) ||
1955
190k
        (info->cpu->insn_cc_not_modified[1] == info->insn))
1956
1.12k
      access_mode = rmmm;
1957
1958
191k
    build_regs_read_write_counts(MI, info, access_mode);
1959
191k
    add_operators_access(MI, info, access_mode);
1960
1961
191k
    if (g_insn_props[info->insn].update_reg_access)
1962
21.1k
      set_changed_regs_read_write_counts(MI, info);
1963
1964
191k
    info->insn_size = (uint8_t)insn_description.insn_size;
1965
1966
191k
    return info->insn_size;
1967
191k
  } else
1968
23.0k
    MCInst_setOpcode(MI, insn_description.opcode);
1969
1970
  // Illegal instruction
1971
23.0k
  address = base_address;
1972
23.0k
  illegal_hdlr(MI, info, &address);
1973
23.0k
  return 1;
1974
214k
}
1975
1976
// Tables to get the byte size of a register on the CPU
1977
// based on an enum m680x_reg value defined in m680x.h
1978
// Invalid registers return 0.
1979
static const uint8_t g_m6800_reg_byte_size[23] = {
1980
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
1981
  0, 1, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 2, 0, 2, 0, 0, 0, 2, 0, 0, 0
1982
};
1983
1984
static const uint8_t g_m6805_reg_byte_size[23] = {
1985
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
1986
  0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 2, 0, 0, 0, 2, 0, 0, 0
1987
};
1988
1989
static const uint8_t g_m6808_reg_byte_size[23] = {
1990
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
1991
  0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 2, 1, 1, 0, 2, 0, 0, 0, 2, 0, 0, 0
1992
};
1993
1994
static const uint8_t g_m6801_reg_byte_size[23] = {
1995
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
1996
  0, 1, 1, 0, 0, 0, 2, 0, 1, 0, 0, 0, 0, 2, 0, 2, 0, 0, 0, 2, 0, 0, 0
1997
};
1998
1999
static const uint8_t g_m6811_reg_byte_size[23] = {
2000
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
2001
  0, 1, 1, 0, 0, 0, 2, 0, 1, 0, 0, 0, 0, 2, 2, 2, 0, 0, 0, 2, 0, 0, 0
2002
};
2003
2004
static const uint8_t g_cpu12_reg_byte_size[23] = {
2005
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
2006
  0, 1, 1, 0, 0, 0, 2, 0, 1, 0, 0, 0, 0, 2, 2, 2, 0, 0, 0, 2, 0, 2, 2
2007
};
2008
2009
static const uint8_t g_m6809_reg_byte_size[23] = {
2010
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
2011
  0, 1, 1, 0, 0, 0, 2, 0, 1, 1, 0, 0, 0, 2, 2, 2, 2, 0, 0, 2, 0, 0, 0
2012
};
2013
2014
static const uint8_t g_hd6309_reg_byte_size[23] = {
2015
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
2016
  0, 1, 1, 1, 1, 1, 2, 2, 1, 1, 1, 0, 0, 2, 2, 2, 2, 2, 4, 2, 0, 0, 0
2017
};
2018
2019
static const uint8_t g_rs08_reg_byte_size[23] = {
2020
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
2021
  0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 2, 1, 1, 0, 2, 0, 0, 0, 2, 2, 0, 0
2022
};
2023
2024
static const uint8_t g_hcs12x_reg_byte_size[23] = {
2025
  // A  B  E  F  0  D  W  CC DP MD HX H  X  Y  S  U  V  Q  PC SPC T2 T3
2026
  0, 1, 1, 0, 0, 0, 2, 0, 1, 0, 0, 0, 0, 2, 2, 2, 0, 0, 0, 2, 0, 2, 2
2027
};
2028
2029
// Table to check for a valid register nibble on the M6809 CPU
2030
// used for TFR and EXG instruction.
2031
static const bool m6809_tfr_reg_valid[16] = {
2032
  true, true, true, true, true,  true,  false, false,
2033
  true, true, true, true, false, false, false, false,
2034
};
2035
2036
static const cpu_tables g_cpu_tables[] = {
2037
  { // M680X_CPU_TYPE_INVALID
2038
    NULL,
2039
    { NULL, NULL },
2040
    { 0, 0 },
2041
    { 0x00, 0x00, 0x00 },
2042
    { NULL, NULL, NULL },
2043
    { 0, 0, 0 },
2044
    NULL,
2045
    NULL,
2046
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2047
  { // M680X_CPU_TYPE_6301
2048
    &g_m6800_inst_page1_table[0],
2049
    { &g_m6801_inst_overlay_table[0], &g_hd6301_inst_overlay_table[0] },
2050
    { ARR_SIZE(g_m6801_inst_overlay_table),
2051
      ARR_SIZE(g_hd6301_inst_overlay_table) },
2052
    { 0x00, 0x00, 0x00 },
2053
    { NULL, NULL, NULL },
2054
    { 0, 0, 0 },
2055
    &g_m6801_reg_byte_size[0],
2056
    NULL,
2057
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2058
  { // M680X_CPU_TYPE_6309
2059
    &g_m6809_inst_page1_table[0],
2060
    { &g_hd6309_inst_overlay_table[0], NULL },
2061
    { ARR_SIZE(g_hd6309_inst_overlay_table), 0 },
2062
    { 0x10, 0x11, 0x00 },
2063
    { &g_hd6309_inst_page2_table[0], &g_hd6309_inst_page3_table[0],
2064
      NULL },
2065
    { ARR_SIZE(g_hd6309_inst_page2_table),
2066
      ARR_SIZE(g_hd6309_inst_page3_table), 0 },
2067
    &g_hd6309_reg_byte_size[0],
2068
    NULL,
2069
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2070
  { // M680X_CPU_TYPE_6800
2071
    &g_m6800_inst_page1_table[0],
2072
    { NULL, NULL },
2073
    { 0, 0 },
2074
    { 0x00, 0x00, 0x00 },
2075
    { NULL, NULL, NULL },
2076
    { 0, 0, 0 },
2077
    &g_m6800_reg_byte_size[0],
2078
    NULL,
2079
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2080
  { // M680X_CPU_TYPE_6801
2081
    &g_m6800_inst_page1_table[0],
2082
    { &g_m6801_inst_overlay_table[0], NULL },
2083
    { ARR_SIZE(g_m6801_inst_overlay_table), 0 },
2084
    { 0x00, 0x00, 0x00 },
2085
    { NULL, NULL, NULL },
2086
    { 0, 0, 0 },
2087
    &g_m6801_reg_byte_size[0],
2088
    NULL,
2089
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2090
  { // M680X_CPU_TYPE_6805
2091
    &g_m6805_inst_page1_table[0],
2092
    { NULL, NULL },
2093
    { 0, 0 },
2094
    { 0x00, 0x00, 0x00 },
2095
    { NULL, NULL, NULL },
2096
    { 0, 0, 0 },
2097
    &g_m6805_reg_byte_size[0],
2098
    NULL,
2099
    { M680X_INS_BCLR, M680X_INS_BSET } },
2100
  { // M680X_CPU_TYPE_6808
2101
    &g_m6805_inst_page1_table[0],
2102
    { &g_m6808_inst_overlay_table[0], NULL },
2103
    { ARR_SIZE(g_m6808_inst_overlay_table), 0 },
2104
    { 0x9E, 0x00, 0x00 },
2105
    { &g_m6808_inst_page2_table[0], NULL, NULL },
2106
    { ARR_SIZE(g_m6808_inst_page2_table), 0, 0 },
2107
    &g_m6808_reg_byte_size[0],
2108
    NULL,
2109
    { M680X_INS_BCLR, M680X_INS_BSET } },
2110
  { // M680X_CPU_TYPE_6809
2111
    &g_m6809_inst_page1_table[0],
2112
    { NULL, NULL },
2113
    { 0, 0 },
2114
    { 0x10, 0x11, 0x00 },
2115
    { &g_m6809_inst_page2_table[0], &g_m6809_inst_page3_table[0], NULL },
2116
    { ARR_SIZE(g_m6809_inst_page2_table),
2117
      ARR_SIZE(g_m6809_inst_page3_table), 0 },
2118
    &g_m6809_reg_byte_size[0],
2119
    &m6809_tfr_reg_valid[0],
2120
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2121
  { // M680X_CPU_TYPE_6811
2122
    &g_m6800_inst_page1_table[0],
2123
    { &g_m6801_inst_overlay_table[0], &g_m6811_inst_overlay_table[0] },
2124
    { ARR_SIZE(g_m6801_inst_overlay_table),
2125
      ARR_SIZE(g_m6811_inst_overlay_table) },
2126
    { 0x18, 0x1A, 0xCD },
2127
    { &g_m6811_inst_page2_table[0], &g_m6811_inst_page3_table[0],
2128
      &g_m6811_inst_page4_table[0] },
2129
    { ARR_SIZE(g_m6811_inst_page2_table),
2130
      ARR_SIZE(g_m6811_inst_page3_table),
2131
      ARR_SIZE(g_m6811_inst_page4_table) },
2132
    &g_m6811_reg_byte_size[0],
2133
    NULL,
2134
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2135
  { // M680X_CPU_TYPE_CPU12
2136
    &g_cpu12_inst_page1_table[0],
2137
    { NULL, NULL },
2138
    { 0, 0 },
2139
    { 0x18, 0x00, 0x00 },
2140
    { &g_cpu12_inst_page2_table[0], NULL, NULL },
2141
    { ARR_SIZE(g_cpu12_inst_page2_table), 0, 0 },
2142
    &g_cpu12_reg_byte_size[0],
2143
    NULL,
2144
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2145
  { // M680X_CPU_TYPE_HCS08
2146
    &g_m6805_inst_page1_table[0],
2147
    { &g_m6808_inst_overlay_table[0], &g_hcs08_inst_overlay_table[0] },
2148
    { ARR_SIZE(g_m6808_inst_overlay_table),
2149
      ARR_SIZE(g_hcs08_inst_overlay_table) },
2150
    { 0x9E, 0x00, 0x00 },
2151
    { &g_hcs08_inst_page2_table[0], NULL, NULL },
2152
    { ARR_SIZE(g_hcs08_inst_page2_table), 0, 0 },
2153
    &g_m6808_reg_byte_size[0],
2154
    NULL,
2155
    { M680X_INS_BCLR, M680X_INS_BSET } },
2156
  { // M680X_CPU_TYPE_RS08
2157
    &g_rs08_inst_page1_table[0],
2158
    { NULL, NULL },
2159
    { 0, 0 },
2160
    { 0x00, 0x00, 0x00 },
2161
    { NULL, NULL, NULL },
2162
    { 0, 0, 0 },
2163
    &g_rs08_reg_byte_size[0],
2164
    NULL,
2165
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2166
  { // M680X_CPU_TYPE_HCS12X
2167
    &g_cpu12_inst_page1_table[0],
2168
    { NULL, NULL },
2169
    { 0, 0 },
2170
    { 0x18, 0x00, 0x00 },
2171
    { &g_hcs12x_inst_page2_table[0], NULL, NULL },
2172
    { ARR_SIZE(g_hcs12x_inst_page2_table), 0, 0 },
2173
    &g_hcs12x_reg_byte_size[0],
2174
    NULL,
2175
    { M680X_INS_INVLD, M680X_INS_INVLD } },
2176
};
2177
2178
static bool m680x_setup_internals(m680x_info *info, e_cpu_type cpu_type,
2179
          uint16_t address, const uint8_t *code,
2180
          uint16_t code_len)
2181
214k
{
2182
214k
  if (cpu_type == M680X_CPU_TYPE_INVALID) {
2183
0
    return false;
2184
0
  }
2185
2186
214k
  info->code = code;
2187
214k
  info->size = code_len;
2188
214k
  info->offset = address;
2189
214k
  info->cpu_type = cpu_type;
2190
2191
214k
  info->cpu = &g_cpu_tables[info->cpu_type];
2192
2193
214k
  return true;
2194
214k
}
2195
2196
bool M680X_getInstruction(csh ud, const uint8_t *code, size_t code_len,
2197
        MCInst *MI, uint16_t *size, uint64_t address,
2198
        void *inst_info)
2199
214k
{
2200
214k
  unsigned int insn_size = 0;
2201
214k
  e_cpu_type cpu_type = M680X_CPU_TYPE_INVALID; // No default CPU type
2202
214k
  cs_struct *handle = (cs_struct *)ud;
2203
214k
  m680x_info *info = (m680x_info *)handle->printer_info;
2204
2205
214k
  MCInst_clear(MI);
2206
2207
214k
  if (handle->mode & CS_MODE_M680X_6800)
2208
2.70k
    cpu_type = M680X_CPU_TYPE_6800;
2209
2210
211k
  else if (handle->mode & CS_MODE_M680X_6801)
2211
718
    cpu_type = M680X_CPU_TYPE_6801;
2212
2213
211k
  else if (handle->mode & CS_MODE_M680X_6805)
2214
3.28k
    cpu_type = M680X_CPU_TYPE_6805;
2215
2216
207k
  else if (handle->mode & CS_MODE_M680X_6808)
2217
12.3k
    cpu_type = M680X_CPU_TYPE_6808;
2218
2219
195k
  else if (handle->mode & CS_MODE_M680X_HCS08)
2220
5.75k
    cpu_type = M680X_CPU_TYPE_HCS08;
2221
2222
189k
  else if (handle->mode & CS_MODE_M680X_6809)
2223
20.3k
    cpu_type = M680X_CPU_TYPE_6809;
2224
2225
169k
  else if (handle->mode & CS_MODE_M680X_6301)
2226
802
    cpu_type = M680X_CPU_TYPE_6301;
2227
2228
168k
  else if (handle->mode & CS_MODE_M680X_6309)
2229
74.4k
    cpu_type = M680X_CPU_TYPE_6309;
2230
2231
94.0k
  else if (handle->mode & CS_MODE_M680X_6811)
2232
4.67k
    cpu_type = M680X_CPU_TYPE_6811;
2233
2234
89.4k
  else if (handle->mode & CS_MODE_M680X_CPU12)
2235
54.4k
    cpu_type = M680X_CPU_TYPE_CPU12;
2236
2237
34.9k
  else if (handle->mode & CS_MODE_M680X_RS08)
2238
7.66k
    cpu_type = M680X_CPU_TYPE_RS08;
2239
2240
27.2k
  else if (handle->mode & CS_MODE_M680X_HCS12X)
2241
27.2k
    cpu_type = M680X_CPU_TYPE_HCS12X;
2242
2243
214k
  if (cpu_type != M680X_CPU_TYPE_INVALID &&
2244
214k
      m680x_setup_internals(info, cpu_type, (uint16_t)address, code,
2245
214k
          (uint16_t)code_len))
2246
214k
    insn_size = m680x_disassemble(MI, info, (uint16_t)address);
2247
2248
214k
  if (insn_size == 0) {
2249
0
    *size = 1;
2250
0
    return false;
2251
0
  }
2252
2253
  // Make sure we always stay within range
2254
214k
  if (insn_size > code_len) {
2255
11
    *size = (uint16_t)code_len;
2256
11
    return false;
2257
11
  } else
2258
214k
    *size = (uint16_t)insn_size;
2259
2260
214k
  return true;
2261
214k
}
2262
2263
cs_err M680X_disassembler_init(cs_struct *ud)
2264
1.30k
{
2265
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_m6800_reg_byte_size)) {
2266
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_m6800_reg_byte_size));
2267
2268
0
    return CS_ERR_MODE;
2269
0
  }
2270
2271
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_m6801_reg_byte_size)) {
2272
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_m6801_reg_byte_size));
2273
2274
0
    return CS_ERR_MODE;
2275
0
  }
2276
2277
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_m6805_reg_byte_size)) {
2278
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_m6805_reg_byte_size));
2279
2280
0
    return CS_ERR_MODE;
2281
0
  }
2282
2283
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_m6808_reg_byte_size)) {
2284
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_m6808_reg_byte_size));
2285
2286
0
    return CS_ERR_MODE;
2287
0
  }
2288
2289
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_m6811_reg_byte_size)) {
2290
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_m6811_reg_byte_size));
2291
2292
0
    return CS_ERR_MODE;
2293
0
  }
2294
2295
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_cpu12_reg_byte_size)) {
2296
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_cpu12_reg_byte_size));
2297
2298
0
    return CS_ERR_MODE;
2299
0
  }
2300
2301
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_m6809_reg_byte_size)) {
2302
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_m6809_reg_byte_size));
2303
2304
0
    return CS_ERR_MODE;
2305
0
  }
2306
2307
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_rs08_reg_byte_size)) {
2308
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_rs08_reg_byte_size));
2309
2310
0
    return CS_ERR_MODE;
2311
0
  }
2312
2313
1.30k
  if (M680X_REG_ENDING != ARR_SIZE(g_hcs12x_reg_byte_size)) {
2314
0
    CS_ASSERT(M680X_REG_ENDING == ARR_SIZE(g_hcs12x_reg_byte_size));
2315
2316
0
    return CS_ERR_MODE;
2317
0
  }
2318
2319
1.30k
  if (M680X_INS_ENDING != ARR_SIZE(g_insn_props)) {
2320
0
    CS_ASSERT(M680X_INS_ENDING == ARR_SIZE(g_insn_props));
2321
2322
0
    return CS_ERR_MODE;
2323
0
  }
2324
2325
1.30k
  if (M680X_CPU_TYPE_ENDING != ARR_SIZE(g_cpu_tables)) {
2326
0
    CS_ASSERT(M680X_CPU_TYPE_ENDING == ARR_SIZE(g_cpu_tables));
2327
2328
0
    return CS_ERR_MODE;
2329
0
  }
2330
2331
1.30k
  if (HANDLER_ID_ENDING != ARR_SIZE(g_insn_handler)) {
2332
0
    CS_ASSERT(HANDLER_ID_ENDING == ARR_SIZE(g_insn_handler));
2333
2334
0
    return CS_ERR_MODE;
2335
0
  }
2336
2337
1.30k
  if (ACCESS_MODE_ENDING != MATRIX_SIZE(g_access_mode_to_access)) {
2338
0
    CS_ASSERT(ACCESS_MODE_ENDING ==
2339
0
        MATRIX_SIZE(g_access_mode_to_access));
2340
2341
0
    return CS_ERR_MODE;
2342
0
  }
2343
2344
1.30k
  return CS_ERR_OK;
2345
1.30k
}
2346
2347
#ifndef CAPSTONE_DIET
2348
void M680X_reg_access(const cs_insn *insn, cs_regs regs_read,
2349
          uint8_t *regs_read_count, cs_regs regs_write,
2350
          uint8_t *regs_write_count)
2351
0
{
2352
0
  if (insn->detail == NULL) {
2353
0
    *regs_read_count = 0;
2354
0
    *regs_write_count = 0;
2355
0
  } else {
2356
0
    *regs_read_count = insn->detail->regs_read_count;
2357
0
    *regs_write_count = insn->detail->regs_write_count;
2358
2359
0
    memcpy(regs_read, insn->detail->regs_read,
2360
0
           *regs_read_count * sizeof(insn->detail->regs_read[0]));
2361
0
    memcpy(regs_write, insn->detail->regs_write,
2362
0
           *regs_write_count * sizeof(insn->detail->regs_write[0]));
2363
0
  }
2364
0
}
2365
#endif
2366
2367
#endif