Coverage Report

Created: 2025-08-26 06:30

/src/capstonenext/arch/M68K/M68KDisassembler.c
Line
Count
Source (jump to first uncovered line)
1
/* ======================================================================== */
2
/* ========================= LICENSING & COPYRIGHT ======================== */
3
/* ======================================================================== */
4
/*
5
 *                                  MUSASHI
6
 *                                Version 3.4
7
 *
8
 * A portable Motorola M680x0 processor emulation engine.
9
 * Copyright 1998-2001 Karl Stenerud.  All rights reserved.
10
 *
11
 * Permission is hereby granted, free of charge, to any person obtaining a copy
12
 * of this software and associated documentation files (the "Software"), to deal
13
 * in the Software without restriction, including without limitation the rights
14
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
15
 * copies of the Software, and to permit persons to whom the Software is
16
 * furnished to do so, subject to the following conditions:
17
 *
18
 * The above copyright notice and this permission notice shall be included in
19
 * all copies or substantial portions of the Software.
20
21
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
24
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
26
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
27
 * THE SOFTWARE.
28
 */
29
30
/* The code below is based on MUSASHI but has been heavily modified for Capstone by
31
 * Daniel Collin <daniel@collin.com> 2015-2019 */
32
33
/* ======================================================================== */
34
/* ================================ INCLUDES ============================== */
35
/* ======================================================================== */
36
37
#include <stdlib.h>
38
#include <stdio.h>
39
#include <string.h>
40
41
#include "../../cs_priv.h"
42
#include "../../utils.h"
43
44
#include "../../MathExtras.h"
45
#include "../../MCInst.h"
46
#include "../../MCInstrDesc.h"
47
#include "../../MCRegisterInfo.h"
48
#include "M68KInstPrinter.h"
49
#include "M68KDisassembler.h"
50
51
/* ======================================================================== */
52
/* ============================ GENERAL DEFINES =========================== */
53
/* ======================================================================== */
54
55
/* Bit Isolation Functions */
56
3.00k
#define BIT_0(A)  ((A) & 0x00000001)
57
#define BIT_1(A)  ((A) & 0x00000002)
58
#define BIT_2(A)  ((A) & 0x00000004)
59
0
#define BIT_3(A)  ((A) & 0x00000008)
60
#define BIT_4(A)  ((A) & 0x00000010)
61
1.97k
#define BIT_5(A)  ((A) & 0x00000020)
62
7.80k
#define BIT_6(A)  ((A) & 0x00000040)
63
7.80k
#define BIT_7(A)  ((A) & 0x00000080)
64
18.9k
#define BIT_8(A)  ((A) & 0x00000100)
65
#define BIT_9(A)  ((A) & 0x00000200)
66
376
#define BIT_A(A)  ((A) & 0x00000400)
67
19.6k
#define BIT_B(A)  ((A) & 0x00000800)
68
#define BIT_C(A)  ((A) & 0x00001000)
69
#define BIT_D(A)  ((A) & 0x00002000)
70
#define BIT_E(A)  ((A) & 0x00004000)
71
21.1k
#define BIT_F(A)  ((A) & 0x00008000)
72
#define BIT_10(A) ((A) & 0x00010000)
73
#define BIT_11(A) ((A) & 0x00020000)
74
#define BIT_12(A) ((A) & 0x00040000)
75
#define BIT_13(A) ((A) & 0x00080000)
76
#define BIT_14(A) ((A) & 0x00100000)
77
#define BIT_15(A) ((A) & 0x00200000)
78
#define BIT_16(A) ((A) & 0x00400000)
79
#define BIT_17(A) ((A) & 0x00800000)
80
#define BIT_18(A) ((A) & 0x01000000)
81
#define BIT_19(A) ((A) & 0x02000000)
82
#define BIT_1A(A) ((A) & 0x04000000)
83
#define BIT_1B(A) ((A) & 0x08000000)
84
#define BIT_1C(A) ((A) & 0x10000000)
85
#define BIT_1D(A) ((A) & 0x20000000)
86
#define BIT_1E(A) ((A) & 0x40000000)
87
901
#define BIT_1F(A) ((A) & 0x80000000)
88
89
/* These are the CPU types understood by this disassembler */
90
75.4k
#define TYPE_68000 1
91
0
#define TYPE_68010 2
92
0
#define TYPE_68020 4
93
0
#define TYPE_68030 8
94
172k
#define TYPE_68040 16
95
96
#define M68000_ONLY   TYPE_68000
97
98
#define M68010_ONLY   TYPE_68010
99
#define M68010_LESS   (TYPE_68000 | TYPE_68010)
100
#define M68010_PLUS   (TYPE_68010 | TYPE_68020 | TYPE_68030 | TYPE_68040)
101
102
#define M68020_ONLY   TYPE_68020
103
#define M68020_LESS   (TYPE_68010 | TYPE_68020)
104
#define M68020_PLUS   (TYPE_68020 | TYPE_68030 | TYPE_68040)
105
106
#define M68030_ONLY   TYPE_68030
107
#define M68030_LESS   (TYPE_68010 | TYPE_68020 | TYPE_68030)
108
#define M68030_PLUS   (TYPE_68030 | TYPE_68040)
109
110
#define M68040_PLUS   TYPE_68040
111
112
enum {
113
  M68K_CPU_TYPE_INVALID,
114
  M68K_CPU_TYPE_68000,
115
  M68K_CPU_TYPE_68010,
116
  M68K_CPU_TYPE_68EC020,
117
  M68K_CPU_TYPE_68020,
118
  M68K_CPU_TYPE_68030,  /* Supported by disassembler ONLY */
119
  M68K_CPU_TYPE_68040   /* Supported by disassembler ONLY */
120
};
121
122
/* Extension word formats */
123
11.1k
#define EXT_8BIT_DISPLACEMENT(A)          ((A)&0xff)
124
18.9k
#define EXT_FULL(A)                       BIT_8(A)
125
#define EXT_EFFECTIVE_ZERO(A)             (((A)&0xe4) == 0xc4 || ((A)&0xe2) == 0xc0)
126
7.80k
#define EXT_BASE_REGISTER_PRESENT(A)      (!BIT_7(A))
127
7.80k
#define EXT_INDEX_REGISTER_PRESENT(A)     (!BIT_6(A))
128
15.9k
#define EXT_INDEX_REGISTER(A)             (((A)>>12)&7)
129
#define EXT_INDEX_PRE_POST(A)             (EXT_INDEX_PRESENT(A) && (A)&3)
130
#define EXT_INDEX_PRE(A)                  (EXT_INDEX_PRESENT(A) && ((A)&7) < 4 && ((A)&7) != 0)
131
#define EXT_INDEX_POST(A)                 (EXT_INDEX_PRESENT(A) && ((A)&7) > 4)
132
25.5k
#define EXT_INDEX_SCALE(A)                (((A)>>9)&3)
133
15.9k
#define EXT_INDEX_LONG(A)                 BIT_B(A)
134
15.9k
#define EXT_INDEX_AR(A)                   BIT_F(A)
135
7.80k
#define EXT_BASE_DISPLACEMENT_PRESENT(A)  (((A)&0x30) > 0x10)
136
#define EXT_BASE_DISPLACEMENT_WORD(A)     (((A)&0x30) == 0x20)
137
4.41k
#define EXT_BASE_DISPLACEMENT_LONG(A)     (((A)&0x30) == 0x30)
138
7.80k
#define EXT_OUTER_DISPLACEMENT_PRESENT(A) (((A)&3) > 1 && ((A)&0x47) < 0x44)
139
#define EXT_OUTER_DISPLACEMENT_WORD(A)    (((A)&3) == 2 && ((A)&0x47) < 0x44)
140
2.42k
#define EXT_OUTER_DISPLACEMENT_LONG(A)    (((A)&3) == 3 && ((A)&0x47) < 0x44)
141
142
#define IS_BITSET(val,b) ((val) & (1 << (b)))
143
13.6k
#define BITFIELD_MASK(sb,eb)  (((1 << ((sb) + 1))-1) & (~((1 << (eb))-1)))
144
13.6k
#define BITFIELD(val,sb,eb) ((BITFIELD_MASK(sb,eb) & (val)) >> (eb))
145
146
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
147
148
static unsigned int m68k_read_disassembler_16(const m68k_info *info, const uint64_t addr)
149
591k
{
150
591k
  const uint16_t v0 = info->code[addr + 0];
151
591k
  const uint16_t v1 = info->code[addr + 1];
152
591k
  return (v0 << 8) | v1;
153
591k
}
154
155
static unsigned int m68k_read_disassembler_32(const m68k_info *info, const uint64_t addr)
156
260k
{
157
260k
  const uint32_t v0 = info->code[addr + 0];
158
260k
  const uint32_t v1 = info->code[addr + 1];
159
260k
  const uint32_t v2 = info->code[addr + 2];
160
260k
  const uint32_t v3 = info->code[addr + 3];
161
260k
  return (v0 << 24) | (v1 << 16) | (v2 << 8) | v3;
162
260k
}
163
164
static uint64_t m68k_read_disassembler_64(const m68k_info *info, const uint64_t addr)
165
95
{
166
95
  const uint64_t v0 = info->code[addr + 0];
167
95
  const uint64_t v1 = info->code[addr + 1];
168
95
  const uint64_t v2 = info->code[addr + 2];
169
95
  const uint64_t v3 = info->code[addr + 3];
170
95
  const uint64_t v4 = info->code[addr + 4];
171
95
  const uint64_t v5 = info->code[addr + 5];
172
95
  const uint64_t v6 = info->code[addr + 6];
173
95
  const uint64_t v7 = info->code[addr + 7];
174
95
  return (v0 << 56) | (v1 << 48) | (v2 << 40) | (v3 << 32) | (v4 << 24) | (v5 << 16) | (v6 << 8) | v7;
175
95
}
176
177
static unsigned int m68k_read_safe_16(const m68k_info *info, const uint64_t address)
178
592k
{
179
592k
  const uint64_t addr = (address - info->baseAddress) & info->address_mask;
180
592k
  if (info->code_len < addr + 2) {
181
920
    return 0xaaaa;
182
920
  }
183
591k
  return m68k_read_disassembler_16(info, addr);
184
592k
}
185
186
static unsigned int m68k_read_safe_32(const m68k_info *info, const uint64_t address)
187
263k
{
188
263k
  const uint64_t addr = (address - info->baseAddress) & info->address_mask;
189
263k
  if (info->code_len < addr + 4) {
190
2.79k
    return 0xaaaaaaaa;
191
2.79k
  }
192
260k
  return m68k_read_disassembler_32(info, addr);
193
263k
}
194
195
static uint64_t m68k_read_safe_64(const m68k_info *info, const uint64_t address)
196
101
{
197
101
  const uint64_t addr = (address - info->baseAddress) & info->address_mask;
198
101
  if (info->code_len < addr + 8) {
199
6
    return 0xaaaaaaaaaaaaaaaaLL;
200
6
  }
201
95
  return m68k_read_disassembler_64(info, addr);
202
101
}
203
204
/* ======================================================================== */
205
/* =============================== PROTOTYPES ============================= */
206
/* ======================================================================== */
207
208
/* make signed integers 100% portably */
209
static int make_int_8(int value);
210
static int make_int_16(int value);
211
212
/* Stuff to build the opcode handler jump table */
213
static void d68000_invalid(m68k_info *info);
214
static int instruction_is_valid(m68k_info *info, const unsigned int word_check);
215
216
typedef struct {
217
  void (*instruction)(m68k_info *info);   /* handler function */
218
  uint16_t word2_mask;                  /* mask the 2nd word */
219
  uint16_t word2_match;                 /* what to match after masking */
220
} instruction_struct;
221
222
/* ======================================================================== */
223
/* ================================= DATA ================================= */
224
/* ======================================================================== */
225
226
static const instruction_struct g_instruction_table[0x10000];
227
228
/* used by ops like asr, ror, addq, etc */
229
static const uint32_t g_3bit_qdata_table[8] = {8, 1, 2, 3, 4, 5, 6, 7};
230
231
static const uint32_t g_5bit_data_table[32] = {
232
  32,  1,  2,  3,  4,  5,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15,
233
  16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31
234
};
235
236
static const m68k_insn s_branch_lut[] = {
237
  M68K_INS_INVALID, M68K_INS_INVALID, M68K_INS_BHI, M68K_INS_BLS,
238
  M68K_INS_BCC, M68K_INS_BCS, M68K_INS_BNE, M68K_INS_BEQ,
239
  M68K_INS_BVC, M68K_INS_BVS, M68K_INS_BPL, M68K_INS_BMI,
240
  M68K_INS_BGE, M68K_INS_BLT, M68K_INS_BGT, M68K_INS_BLE,
241
};
242
243
static const m68k_insn s_dbcc_lut[] = {
244
  M68K_INS_DBT, M68K_INS_DBF, M68K_INS_DBHI, M68K_INS_DBLS,
245
  M68K_INS_DBCC, M68K_INS_DBCS, M68K_INS_DBNE, M68K_INS_DBEQ,
246
  M68K_INS_DBVC, M68K_INS_DBVS, M68K_INS_DBPL, M68K_INS_DBMI,
247
  M68K_INS_DBGE, M68K_INS_DBLT, M68K_INS_DBGT, M68K_INS_DBLE,
248
};
249
250
static const m68k_insn s_scc_lut[] = {
251
  M68K_INS_ST, M68K_INS_SF, M68K_INS_SHI, M68K_INS_SLS,
252
  M68K_INS_SCC, M68K_INS_SCS, M68K_INS_SNE, M68K_INS_SEQ,
253
  M68K_INS_SVC, M68K_INS_SVS, M68K_INS_SPL, M68K_INS_SMI,
254
  M68K_INS_SGE, M68K_INS_SLT, M68K_INS_SGT, M68K_INS_SLE,
255
};
256
257
static const m68k_insn s_trap_lut[] = {
258
  M68K_INS_TRAPT, M68K_INS_TRAPF, M68K_INS_TRAPHI, M68K_INS_TRAPLS,
259
  M68K_INS_TRAPCC, M68K_INS_TRAPCS, M68K_INS_TRAPNE, M68K_INS_TRAPEQ,
260
  M68K_INS_TRAPVC, M68K_INS_TRAPVS, M68K_INS_TRAPPL, M68K_INS_TRAPMI,
261
  M68K_INS_TRAPGE, M68K_INS_TRAPLT, M68K_INS_TRAPGT, M68K_INS_TRAPLE,
262
};
263
264
/* ======================================================================== */
265
/* =========================== UTILITY FUNCTIONS ========================== */
266
/* ======================================================================== */
267
268
#define LIMIT_CPU_TYPES(info, ALLOWED_CPU_TYPES)  \
269
51.5k
  do {           \
270
51.5k
    if (!(info->type & ALLOWED_CPU_TYPES)) { \
271
15.3k
      d68000_invalid(info);   \
272
15.3k
      return;       \
273
15.3k
    }          \
274
51.5k
  } while (0)
275
276
16.5k
static unsigned int peek_imm_8(const m68k_info *info)  { return (m68k_read_safe_16((info), (info)->pc)&0xff); }
277
576k
static unsigned int peek_imm_16(const m68k_info *info) { return m68k_read_safe_16((info), (info)->pc); }
278
263k
static unsigned int peek_imm_32(const m68k_info *info) { return m68k_read_safe_32((info), (info)->pc); }
279
101
static unsigned long long peek_imm_64(const m68k_info *info) { return m68k_read_safe_64((info), (info)->pc); }
280
281
16.5k
static unsigned int read_imm_8(m68k_info *info)  { const unsigned int value = peek_imm_8(info);  (info)->pc+=2; return value & 0xff; }
282
327k
static unsigned int read_imm_16(m68k_info *info) { const unsigned int value = peek_imm_16(info); (info)->pc+=2; return value & 0xffff; }
283
12.9k
static unsigned int read_imm_32(m68k_info *info) { const unsigned int value = peek_imm_32(info); (info)->pc+=4; return value & 0xffffffff; }
284
101
static unsigned long long read_imm_64(m68k_info *info) { const unsigned long long value = peek_imm_64(info); (info)->pc+=8; return value & 0xffffffffffffffff; }
285
286
/* Fake a split interface */
287
#define get_ea_mode_str_8(instruction) get_ea_mode_str(instruction, 0)
288
#define get_ea_mode_str_16(instruction) get_ea_mode_str(instruction, 1)
289
#define get_ea_mode_str_32(instruction) get_ea_mode_str(instruction, 2)
290
291
#define get_imm_str_s8() get_imm_str_s(0)
292
#define get_imm_str_s16() get_imm_str_s(1)
293
#define get_imm_str_s32() get_imm_str_s(2)
294
295
#define get_imm_str_u8() get_imm_str_u(0)
296
#define get_imm_str_u16() get_imm_str_u(1)
297
#define get_imm_str_u32() get_imm_str_u(2)
298
299
300
/* 100% portable signed int generators */
301
static int make_int_8(int value)
302
11.6k
{
303
11.6k
  return (value & 0x80) ? value | ~0xff : value & 0xff;
304
11.6k
}
305
306
static int make_int_16(int value)
307
4.47k
{
308
4.47k
  return (value & 0x8000) ? value | ~0xffff : value & 0xffff;
309
4.47k
}
310
311
static void get_with_index_address_mode(m68k_info *info, cs_m68k_op* op, uint32_t instruction, uint32_t size, bool is_pc)
312
18.9k
{
313
18.9k
  uint32_t extension = read_imm_16(info);
314
315
18.9k
  op->address_mode = M68K_AM_AREGI_INDEX_BASE_DISP;
316
317
18.9k
  if (EXT_FULL(extension)) {
318
7.80k
    uint32_t preindex;
319
7.80k
    uint32_t postindex;
320
321
7.80k
    op->mem.base_reg = M68K_REG_INVALID;
322
7.80k
    op->mem.index_reg = M68K_REG_INVALID;
323
324
7.80k
    op->mem.in_disp = EXT_BASE_DISPLACEMENT_PRESENT(extension) ? (EXT_BASE_DISPLACEMENT_LONG(extension) ? read_imm_32(info) : read_imm_16(info)) : 0;
325
7.80k
    op->mem.out_disp = EXT_OUTER_DISPLACEMENT_PRESENT(extension) ? (EXT_OUTER_DISPLACEMENT_LONG(extension) ? read_imm_32(info) : read_imm_16(info)) : 0;
326
327
7.80k
    if (EXT_BASE_REGISTER_PRESENT(extension)) {
328
3.60k
      if (is_pc) {
329
438
        op->mem.base_reg = M68K_REG_PC;
330
3.17k
      } else {
331
3.17k
        op->mem.base_reg = M68K_REG_A0 + (instruction & 7);
332
3.17k
      }
333
3.60k
    }
334
335
7.80k
    if (EXT_INDEX_REGISTER_PRESENT(extension)) {
336
4.86k
      if (EXT_INDEX_AR(extension)) {
337
2.07k
        op->mem.index_reg = M68K_REG_A0 + EXT_INDEX_REGISTER(extension);
338
2.78k
      } else {
339
2.78k
        op->mem.index_reg = M68K_REG_D0 + EXT_INDEX_REGISTER(extension);
340
2.78k
      }
341
342
4.86k
      op->mem.index_size = EXT_INDEX_LONG(extension) ? 1 : 0;
343
344
4.86k
      if (EXT_INDEX_SCALE(extension)) {
345
3.02k
        op->mem.scale = 1 << EXT_INDEX_SCALE(extension);
346
3.02k
      }
347
4.86k
    }
348
349
7.80k
    preindex = (extension & 7) > 0 && (extension & 7) < 4;
350
7.80k
    postindex = (extension & 7) > 4;
351
352
7.80k
    if (preindex) {
353
2.91k
      op->address_mode = is_pc ? M68K_AM_PC_MEMI_PRE_INDEX : M68K_AM_MEMI_PRE_INDEX;
354
4.88k
    } else if (postindex) {
355
2.96k
      op->address_mode = is_pc ? M68K_AM_PC_MEMI_POST_INDEX : M68K_AM_MEMI_POST_INDEX;
356
2.96k
    }
357
358
7.80k
    return;
359
7.80k
  }
360
361
11.1k
  op->mem.index_reg = (EXT_INDEX_AR(extension) ? M68K_REG_A0 : M68K_REG_D0) + EXT_INDEX_REGISTER(extension);
362
11.1k
  op->mem.index_size = EXT_INDEX_LONG(extension) ? 1 : 0;
363
364
11.1k
  if (EXT_8BIT_DISPLACEMENT(extension) == 0) {
365
1.22k
    if (is_pc) {
366
92
      op->mem.base_reg = M68K_REG_PC;
367
92
      op->address_mode = M68K_AM_PCI_INDEX_BASE_DISP;
368
1.12k
    } else {
369
1.12k
      op->mem.base_reg = M68K_REG_A0 + (instruction & 7);
370
1.12k
    }
371
9.88k
  } else {
372
9.88k
    if (is_pc) {
373
801
      op->mem.base_reg = M68K_REG_PC;
374
801
      op->address_mode = M68K_AM_PCI_INDEX_8_BIT_DISP;
375
9.08k
    } else {
376
9.08k
      op->mem.base_reg = M68K_REG_A0 + (instruction & 7);
377
9.08k
      op->address_mode = M68K_AM_AREGI_INDEX_8_BIT_DISP;
378
9.08k
    }
379
380
9.88k
    op->mem.disp = (int8_t)(extension & 0xff);
381
9.88k
  }
382
383
11.1k
  if (EXT_INDEX_SCALE(extension)) {
384
6.53k
    op->mem.scale = 1 << EXT_INDEX_SCALE(extension);
385
6.53k
  }
386
11.1k
}
387
388
/* Make string of effective address mode */
389
static void get_ea_mode_op(m68k_info *info, cs_m68k_op* op, uint32_t instruction, uint32_t size)
390
172k
{
391
  // default to memory
392
393
172k
  op->type = M68K_OP_MEM;
394
395
172k
  switch (instruction & 0x3f) {
396
52.8k
    case 0x00: case 0x01: case 0x02: case 0x03: case 0x04: case 0x05: case 0x06: case 0x07:
397
      /* data register direct */
398
52.8k
      op->address_mode = M68K_AM_REG_DIRECT_DATA;
399
52.8k
      op->reg = M68K_REG_D0 + (instruction & 7);
400
52.8k
      op->type = M68K_OP_REG;
401
52.8k
      break;
402
403
8.66k
    case 0x08: case 0x09: case 0x0a: case 0x0b: case 0x0c: case 0x0d: case 0x0e: case 0x0f:
404
      /* address register direct */
405
8.66k
      op->address_mode = M68K_AM_REG_DIRECT_ADDR;
406
8.66k
      op->reg = M68K_REG_A0 + (instruction & 7);
407
8.66k
      op->type = M68K_OP_REG;
408
8.66k
      break;
409
410
17.5k
    case 0x10: case 0x11: case 0x12: case 0x13: case 0x14: case 0x15: case 0x16: case 0x17:
411
      /* address register indirect */
412
17.5k
      op->address_mode = M68K_AM_REGI_ADDR;
413
17.5k
      op->reg = M68K_REG_A0 + (instruction & 7);
414
17.5k
      break;
415
416
21.9k
    case 0x18: case 0x19: case 0x1a: case 0x1b: case 0x1c: case 0x1d: case 0x1e: case 0x1f:
417
      /* address register indirect with postincrement */
418
21.9k
      op->address_mode = M68K_AM_REGI_ADDR_POST_INC;
419
21.9k
      op->reg = M68K_REG_A0 + (instruction & 7);
420
21.9k
      break;
421
422
30.7k
    case 0x20: case 0x21: case 0x22: case 0x23: case 0x24: case 0x25: case 0x26: case 0x27:
423
      /* address register indirect with predecrement */
424
30.7k
      op->address_mode = M68K_AM_REGI_ADDR_PRE_DEC;
425
30.7k
      op->reg = M68K_REG_A0 + (instruction & 7);
426
30.7k
      break;
427
428
14.7k
    case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f:
429
      /* address register indirect with displacement*/
430
14.7k
      op->address_mode = M68K_AM_REGI_ADDR_DISP;
431
14.7k
      op->mem.base_reg = M68K_REG_A0 + (instruction & 7);
432
14.7k
      op->mem.disp = (int16_t)read_imm_16(info);
433
14.7k
      break;
434
435
16.9k
    case 0x30: case 0x31: case 0x32: case 0x33: case 0x34: case 0x35: case 0x36: case 0x37:
436
      /* address register indirect with index */
437
16.9k
      get_with_index_address_mode(info, op, instruction, size, false);
438
16.9k
      break;
439
440
2.02k
    case 0x38:
441
      /* absolute short address */
442
2.02k
      op->address_mode = M68K_AM_ABSOLUTE_DATA_SHORT;
443
2.02k
      op->imm = read_imm_16(info);
444
2.02k
      break;
445
446
967
    case 0x39:
447
      /* absolute long address */
448
967
      op->address_mode = M68K_AM_ABSOLUTE_DATA_LONG;
449
967
      op->imm = read_imm_32(info);
450
967
      break;
451
452
1.51k
    case 0x3a:
453
      /* program counter with displacement */
454
1.51k
      op->address_mode = M68K_AM_PCI_DISP;
455
1.51k
      op->mem.disp = (int16_t)read_imm_16(info);
456
1.51k
      break;
457
458
1.96k
    case 0x3b:
459
      /* program counter with index */
460
1.96k
      get_with_index_address_mode(info, op, instruction, size, true);
461
1.96k
      break;
462
463
2.01k
    case 0x3c:
464
2.01k
      op->address_mode = M68K_AM_IMMEDIATE;
465
2.01k
      op->type = M68K_OP_IMM;
466
467
2.01k
      if (size == 1)
468
476
        op->imm = read_imm_8(info);
469
1.54k
      else if (size == 2)
470
795
        op->imm = read_imm_16(info);
471
748
      else if (size == 4)
472
647
        op->imm = read_imm_32(info);
473
101
      else
474
101
        op->imm = read_imm_64(info);
475
476
2.01k
      break;
477
478
118
    default:
479
118
      break;
480
172k
  }
481
172k
}
482
483
static void set_insn_group(m68k_info *info, m68k_group_type group)
484
40.7k
{
485
40.7k
  info->groups[info->groups_count++] = (uint8_t)group;
486
40.7k
}
487
488
static cs_m68k* build_init_op(m68k_info *info, int opcode, int count, int size)
489
239k
{
490
239k
  cs_m68k* ext;
491
492
239k
  MCInst_setOpcode(info->inst, opcode);
493
494
239k
  ext = &info->extension;
495
496
239k
  ext->op_count = (uint8_t)count;
497
239k
  ext->op_size.type = M68K_SIZE_TYPE_CPU;
498
239k
  ext->op_size.cpu_size = size;
499
500
239k
  return ext;
501
239k
}
502
503
static void build_re_gen_1(m68k_info *info, bool isDreg, int opcode, uint8_t size)
504
21.6k
{
505
21.6k
  cs_m68k_op* op0;
506
21.6k
  cs_m68k_op* op1;
507
21.6k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
508
509
21.6k
  op0 = &ext->operands[0];
510
21.6k
  op1 = &ext->operands[1];
511
512
21.6k
  if (isDreg) {
513
21.6k
    op0->address_mode = M68K_AM_REG_DIRECT_DATA;
514
21.6k
    op0->reg = M68K_REG_D0 + ((info->ir >> 9 ) & 7);
515
21.6k
  } else {
516
0
    op0->address_mode = M68K_AM_REG_DIRECT_ADDR;
517
0
    op0->reg = M68K_REG_A0 + ((info->ir >> 9 ) & 7);
518
0
  }
519
520
21.6k
  get_ea_mode_op(info, op1, info->ir, size);
521
21.6k
}
522
523
static void build_re_1(m68k_info *info, int opcode, uint8_t size)
524
21.6k
{
525
21.6k
  build_re_gen_1(info, true, opcode, size);
526
21.6k
}
527
528
static void build_er_gen_1(m68k_info *info, bool isDreg, int opcode, uint8_t size)
529
19.7k
{
530
19.7k
  cs_m68k_op* op0;
531
19.7k
  cs_m68k_op* op1;
532
19.7k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
533
534
19.7k
  op0 = &ext->operands[0];
535
19.7k
  op1 = &ext->operands[1];
536
537
19.7k
  get_ea_mode_op(info, op0, info->ir, size);
538
539
19.7k
  if (isDreg) {
540
19.7k
    op1->address_mode = M68K_AM_REG_DIRECT_DATA;
541
19.7k
    op1->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
542
19.7k
  } else {
543
0
    op1->address_mode = M68K_AM_REG_DIRECT_ADDR;
544
0
    op1->reg = M68K_REG_A0 + ((info->ir >> 9) & 7);
545
0
  }
546
19.7k
}
547
548
static void build_rr(m68k_info *info, int opcode, uint8_t size, int imm)
549
4.79k
{
550
4.79k
  cs_m68k_op* op0;
551
4.79k
  cs_m68k_op* op1;
552
4.79k
  cs_m68k_op* op2;
553
4.79k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
554
555
4.79k
  op0 = &ext->operands[0];
556
4.79k
  op1 = &ext->operands[1];
557
4.79k
  op2 = &ext->operands[2];
558
559
4.79k
  op0->address_mode = M68K_AM_REG_DIRECT_DATA;
560
4.79k
  op0->reg = M68K_REG_D0 + (info->ir & 7);
561
562
4.79k
  op1->address_mode = M68K_AM_REG_DIRECT_DATA;
563
4.79k
  op1->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
564
565
4.79k
  if (imm > 0) {
566
1.60k
    ext->op_count = 3;
567
1.60k
    op2->type = M68K_OP_IMM;
568
1.60k
    op2->address_mode = M68K_AM_IMMEDIATE;
569
1.60k
    op2->imm = imm;
570
1.60k
  }
571
4.79k
}
572
573
static void build_r(m68k_info *info, int opcode, uint8_t size)
574
5.85k
{
575
5.85k
  cs_m68k_op* op0;
576
5.85k
  cs_m68k_op* op1;
577
5.85k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
578
579
5.85k
  op0 = &ext->operands[0];
580
5.85k
  op1 = &ext->operands[1];
581
582
5.85k
  op0->address_mode = M68K_AM_REG_DIRECT_DATA;
583
5.85k
  op0->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
584
585
5.85k
  op1->address_mode = M68K_AM_REG_DIRECT_DATA;
586
5.85k
  op1->reg = M68K_REG_D0 + (info->ir & 7);
587
5.85k
}
588
589
static void build_imm_ea(m68k_info *info, int opcode, uint8_t size, int imm)
590
21.0k
{
591
21.0k
  cs_m68k_op* op0;
592
21.0k
  cs_m68k_op* op1;
593
21.0k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
594
595
21.0k
  op0 = &ext->operands[0];
596
21.0k
  op1 = &ext->operands[1];
597
598
21.0k
  op0->type = M68K_OP_IMM;
599
21.0k
  op0->address_mode = M68K_AM_IMMEDIATE;
600
21.0k
  op0->imm = imm & info->address_mask;
601
602
21.0k
  get_ea_mode_op(info, op1, info->ir, size);
603
21.0k
}
604
605
static void build_3bit_d(m68k_info *info, int opcode, int size)
606
7.56k
{
607
7.56k
  cs_m68k_op* op0;
608
7.56k
  cs_m68k_op* op1;
609
7.56k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
610
611
7.56k
  op0 = &ext->operands[0];
612
7.56k
  op1 = &ext->operands[1];
613
614
7.56k
  op0->type = M68K_OP_IMM;
615
7.56k
  op0->address_mode = M68K_AM_IMMEDIATE;
616
7.56k
  op0->imm = g_3bit_qdata_table[(info->ir >> 9) & 7];
617
618
7.56k
  op1->address_mode = M68K_AM_REG_DIRECT_DATA;
619
7.56k
  op1->reg = M68K_REG_D0 + (info->ir & 7);
620
7.56k
}
621
622
static void build_3bit_ea(m68k_info *info, int opcode, int size)
623
7.86k
{
624
7.86k
  cs_m68k_op* op0;
625
7.86k
  cs_m68k_op* op1;
626
7.86k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
627
628
7.86k
  op0 = &ext->operands[0];
629
7.86k
  op1 = &ext->operands[1];
630
631
7.86k
  op0->type = M68K_OP_IMM;
632
7.86k
  op0->address_mode = M68K_AM_IMMEDIATE;
633
7.86k
  op0->imm = g_3bit_qdata_table[(info->ir >> 9) & 7];
634
635
7.86k
  get_ea_mode_op(info, op1, info->ir, size);
636
7.86k
}
637
638
static void build_mm(m68k_info *info, int opcode, uint8_t size, int imm)
639
4.49k
{
640
4.49k
  cs_m68k_op* op0;
641
4.49k
  cs_m68k_op* op1;
642
4.49k
  cs_m68k_op* op2;
643
4.49k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
644
645
4.49k
  op0 = &ext->operands[0];
646
4.49k
  op1 = &ext->operands[1];
647
4.49k
  op2 = &ext->operands[2];
648
649
4.49k
  op0->address_mode = M68K_AM_REGI_ADDR_PRE_DEC;
650
4.49k
  op0->reg = M68K_REG_A0 + (info->ir & 7);
651
652
4.49k
  op1->address_mode = M68K_AM_REGI_ADDR_PRE_DEC;
653
4.49k
  op1->reg = M68K_REG_A0 + ((info->ir >> 9) & 7);
654
655
4.49k
  if (imm > 0) {
656
1.73k
    ext->op_count = 3;
657
1.73k
    op2->type = M68K_OP_IMM;
658
1.73k
    op2->address_mode = M68K_AM_IMMEDIATE;
659
1.73k
    op2->imm = imm;
660
1.73k
  }
661
4.49k
}
662
663
static void build_ea(m68k_info *info, int opcode, uint8_t size)
664
14.6k
{
665
14.6k
  cs_m68k* ext = build_init_op(info, opcode, 1, size);
666
14.6k
  get_ea_mode_op(info, &ext->operands[0], info->ir, size);
667
14.6k
}
668
669
static void build_ea_a(m68k_info *info, int opcode, uint8_t size)
670
12.2k
{
671
12.2k
  cs_m68k_op* op0;
672
12.2k
  cs_m68k_op* op1;
673
12.2k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
674
675
12.2k
  op0 = &ext->operands[0];
676
12.2k
  op1 = &ext->operands[1];
677
678
12.2k
  get_ea_mode_op(info, op0, info->ir, size);
679
680
12.2k
  op1->address_mode = M68K_AM_REG_DIRECT_ADDR;
681
12.2k
  op1->reg = M68K_REG_A0 + ((info->ir >> 9) & 7);
682
12.2k
}
683
684
static void build_ea_ea(m68k_info *info, int opcode, int size)
685
27.1k
{
686
27.1k
  cs_m68k_op* op0;
687
27.1k
  cs_m68k_op* op1;
688
27.1k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
689
690
27.1k
  op0 = &ext->operands[0];
691
27.1k
  op1 = &ext->operands[1];
692
693
27.1k
  get_ea_mode_op(info, op0, info->ir, size);
694
27.1k
  get_ea_mode_op(info, op1, (((info->ir>>9) & 7) | ((info->ir>>3) & 0x38)), size);
695
27.1k
}
696
697
static void build_pi_pi(m68k_info *info, int opcode, int size)
698
595
{
699
595
  cs_m68k_op* op0;
700
595
  cs_m68k_op* op1;
701
595
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
702
703
595
  op0 = &ext->operands[0];
704
595
  op1 = &ext->operands[1];
705
706
595
  op0->address_mode = M68K_AM_REGI_ADDR_POST_INC;
707
595
  op0->reg = M68K_REG_A0 + (info->ir & 7);
708
709
595
  op1->address_mode = M68K_AM_REGI_ADDR_POST_INC;
710
595
  op1->reg = M68K_REG_A0 + ((info->ir >> 9) & 7);
711
595
}
712
713
static void build_imm_special_reg(m68k_info *info, int opcode, int imm, int size, m68k_reg reg)
714
1.52k
{
715
1.52k
  cs_m68k_op* op0;
716
1.52k
  cs_m68k_op* op1;
717
1.52k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
718
719
1.52k
  op0 = &ext->operands[0];
720
1.52k
  op1 = &ext->operands[1];
721
722
1.52k
  op0->type = M68K_OP_IMM;
723
1.52k
  op0->address_mode = M68K_AM_IMMEDIATE;
724
1.52k
  op0->imm = imm;
725
726
1.52k
  op1->address_mode = M68K_AM_NONE;
727
1.52k
  op1->reg = reg;
728
1.52k
}
729
730
static void build_relative_branch(m68k_info *info, int opcode, int size, int displacement)
731
13.9k
{
732
13.9k
  cs_m68k_op* op;
733
13.9k
  cs_m68k* ext = build_init_op(info, opcode, 1, size);
734
735
13.9k
  op = &ext->operands[0];
736
737
13.9k
  op->type = M68K_OP_BR_DISP;
738
13.9k
  op->address_mode = M68K_AM_BRANCH_DISPLACEMENT;
739
13.9k
  op->br_disp.disp = displacement;
740
13.9k
  op->br_disp.disp_size = size;
741
742
13.9k
  set_insn_group(info, M68K_GRP_JUMP);
743
13.9k
  set_insn_group(info, M68K_GRP_BRANCH_RELATIVE);
744
13.9k
}
745
746
static void build_absolute_jump_with_immediate(m68k_info *info, int opcode, int size, int immediate)
747
3.18k
{
748
3.18k
  cs_m68k_op* op;
749
3.18k
  cs_m68k* ext = build_init_op(info, opcode, 1, size);
750
751
3.18k
  op = &ext->operands[0];
752
753
3.18k
  op->type = M68K_OP_IMM;
754
3.18k
  op->address_mode = M68K_AM_IMMEDIATE;
755
3.18k
  op->imm = immediate;
756
757
3.18k
  set_insn_group(info, M68K_GRP_JUMP);
758
3.18k
}
759
760
static void build_bcc(m68k_info *info, int size, int displacement)
761
9.31k
{
762
9.31k
  build_relative_branch(info, s_branch_lut[(info->ir >> 8) & 0xf], size, displacement);
763
9.31k
}
764
765
static void build_trap(m68k_info *info, int size, int immediate)
766
699
{
767
699
  build_absolute_jump_with_immediate(info, s_trap_lut[(info->ir >> 8) & 0xf], size, immediate);
768
699
}
769
770
static void build_dbxx(m68k_info *info, int opcode, int size, int displacement)
771
288
{
772
288
  cs_m68k_op* op0;
773
288
  cs_m68k_op* op1;
774
288
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
775
776
288
  op0 = &ext->operands[0];
777
288
  op1 = &ext->operands[1];
778
779
288
  op0->address_mode = M68K_AM_REG_DIRECT_DATA;
780
288
  op0->reg = M68K_REG_D0 + (info->ir & 7);
781
782
288
  op1->type = M68K_OP_BR_DISP;
783
288
  op1->address_mode = M68K_AM_BRANCH_DISPLACEMENT;
784
288
  op1->br_disp.disp = displacement;
785
288
  op1->br_disp.disp_size = M68K_OP_BR_DISP_SIZE_LONG;
786
787
288
  set_insn_group(info, M68K_GRP_JUMP);
788
288
  set_insn_group(info, M68K_GRP_BRANCH_RELATIVE);
789
288
}
790
791
static void build_dbcc(m68k_info *info, int size, int displacement)
792
159
{
793
159
  build_dbxx(info, s_dbcc_lut[(info->ir >> 8) & 0xf], size, displacement);
794
159
}
795
796
static void build_d_d_ea(m68k_info *info, int opcode, int size)
797
180
{
798
180
  cs_m68k_op* op0;
799
180
  cs_m68k_op* op1;
800
180
  cs_m68k_op* op2;
801
180
  uint32_t extension = read_imm_16(info);
802
180
  cs_m68k* ext = build_init_op(info, opcode, 3, size);
803
804
180
  op0 = &ext->operands[0];
805
180
  op1 = &ext->operands[1];
806
180
  op2 = &ext->operands[2];
807
808
180
  op0->address_mode = M68K_AM_REG_DIRECT_DATA;
809
180
  op0->reg = M68K_REG_D0 + (extension & 7);
810
811
180
  op1->address_mode = M68K_AM_REG_DIRECT_DATA;
812
180
  op1->reg = M68K_REG_D0 + ((extension >> 6) & 7);
813
814
180
  get_ea_mode_op(info, op2, info->ir, size);
815
180
}
816
817
static void build_bitfield_ins(m68k_info *info, int opcode, int has_d_arg)
818
1.97k
{
819
1.97k
  uint8_t offset;
820
1.97k
  uint8_t width;
821
1.97k
  cs_m68k_op* op_ea;
822
1.97k
  cs_m68k_op* op1;
823
1.97k
  cs_m68k* ext = build_init_op(info, opcode, 1, 0);
824
1.97k
  uint32_t extension = read_imm_16(info);
825
826
1.97k
  op_ea = &ext->operands[0];
827
1.97k
  op1 = &ext->operands[1];
828
829
1.97k
  if (BIT_B(extension))
830
1.41k
    offset = (extension >> 6) & 7;
831
556
  else
832
556
    offset = (extension >> 6) & 31;
833
834
1.97k
  if (BIT_5(extension))
835
915
    width = extension & 7;
836
1.05k
  else
837
1.05k
    width = (uint8_t)g_5bit_data_table[extension & 31];
838
839
1.97k
  if (has_d_arg) {
840
888
    ext->op_count = 2;
841
888
    op1->address_mode = M68K_AM_REG_DIRECT_DATA;
842
888
    op1->reg = M68K_REG_D0 + ((extension >> 12) & 7);
843
888
  }
844
845
1.97k
  get_ea_mode_op(info, op_ea, info->ir, 1);
846
847
1.97k
  op_ea->mem.bitfield = 1;
848
1.97k
  op_ea->mem.width = width;
849
1.97k
  op_ea->mem.offset = offset;
850
1.97k
}
851
852
static void build_d(m68k_info *info, int opcode, int size)
853
361
{
854
361
  cs_m68k* ext = build_init_op(info, opcode, 1, size);
855
361
  cs_m68k_op* op;
856
857
361
  op = &ext->operands[0];
858
859
361
  op->address_mode = M68K_AM_REG_DIRECT_DATA;
860
361
  op->reg = M68K_REG_D0 + (info->ir & 7);
861
361
}
862
863
static uint16_t reverse_bits(uint32_t v)
864
846
{
865
846
  uint32_t r = v; // r will be reversed bits of v; first get LSB of v
866
846
  uint32_t s = 16 - 1; // extra shift needed at end
867
868
11.1k
  for (v >>= 1; v; v >>= 1) {
869
10.2k
    r <<= 1;
870
10.2k
    r |= v & 1;
871
10.2k
    s--;
872
10.2k
  }
873
874
846
  r <<= s; // shift when v's highest bits are zero
875
846
  return r;
876
846
}
877
878
static uint8_t reverse_bits_8(uint32_t v)
879
693
{
880
693
  uint32_t r = v; // r will be reversed bits of v; first get LSB of v
881
693
  uint32_t s = 8 - 1; // extra shift needed at end
882
883
2.19k
  for (v >>= 1; v; v >>= 1) {
884
1.50k
    r <<= 1;
885
1.50k
    r |= v & 1;
886
1.50k
    s--;
887
1.50k
  }
888
889
693
  r <<= s; // shift when v's highest bits are zero
890
693
  return r;
891
693
}
892
893
894
static void build_movem_re(m68k_info *info, int opcode, int size)
895
1.76k
{
896
1.76k
  cs_m68k_op* op0;
897
1.76k
  cs_m68k_op* op1;
898
1.76k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
899
900
1.76k
  op0 = &ext->operands[0];
901
1.76k
  op1 = &ext->operands[1];
902
903
1.76k
  op0->type = M68K_OP_REG_BITS;
904
1.76k
  op0->register_bits = read_imm_16(info);
905
906
1.76k
  get_ea_mode_op(info, op1, info->ir, size);
907
908
1.76k
  if (op1->address_mode == M68K_AM_REGI_ADDR_PRE_DEC)
909
846
    op0->register_bits = reverse_bits(op0->register_bits);
910
1.76k
}
911
912
static void build_movem_er(m68k_info *info, int opcode, int size)
913
1.28k
{
914
1.28k
  cs_m68k_op* op0;
915
1.28k
  cs_m68k_op* op1;
916
1.28k
  cs_m68k* ext = build_init_op(info, opcode, 2, size);
917
918
1.28k
  op0 = &ext->operands[0];
919
1.28k
  op1 = &ext->operands[1];
920
921
1.28k
  op1->type = M68K_OP_REG_BITS;
922
1.28k
  op1->register_bits = read_imm_16(info);
923
924
1.28k
  get_ea_mode_op(info, op0, info->ir, size);
925
1.28k
}
926
927
static void build_imm(m68k_info *info, int opcode, int data)
928
35.1k
{
929
35.1k
  cs_m68k_op* op;
930
35.1k
  cs_m68k* ext = build_init_op(info, opcode, 1, 0);
931
932
35.1k
  MCInst_setOpcode(info->inst, opcode);
933
934
35.1k
  op = &ext->operands[0];
935
936
35.1k
  op->type = M68K_OP_IMM;
937
35.1k
  op->address_mode = M68K_AM_IMMEDIATE;
938
35.1k
  op->imm = data;
939
35.1k
}
940
941
static void build_illegal(m68k_info *info, int data)
942
237
{
943
237
  build_imm(info, M68K_INS_ILLEGAL, data);
944
237
}
945
946
static void build_invalid(m68k_info *info, int data)
947
34.8k
{
948
34.8k
  build_imm(info, M68K_INS_INVALID, data);
949
34.8k
}
950
951
static void build_cas2(m68k_info *info, int size)
952
1.90k
{
953
1.90k
  uint32_t word3;
954
1.90k
  uint32_t extension;
955
1.90k
  cs_m68k_op* op0;
956
1.90k
  cs_m68k_op* op1;
957
1.90k
  cs_m68k_op* op2;
958
1.90k
  cs_m68k* ext = build_init_op(info, M68K_INS_CAS2, 3, size);
959
1.90k
  int reg_0, reg_1;
960
961
  /* cas2 is the only 3 words instruction, word2 and word3 have the same motif bits to check */
962
1.90k
  word3 = peek_imm_32(info) & 0xffff;
963
1.90k
  if (!instruction_is_valid(info, word3))
964
999
    return;
965
966
901
  op0 = &ext->operands[0];
967
901
  op1 = &ext->operands[1];
968
901
  op2 = &ext->operands[2];
969
970
901
  extension = read_imm_32(info);
971
972
901
  op0->address_mode = M68K_AM_NONE;
973
901
  op0->type = M68K_OP_REG_PAIR;
974
901
  op0->reg_pair.reg_0 = ((extension >> 16) & 7) + M68K_REG_D0;
975
901
  op0->reg_pair.reg_1 = (extension & 7) + M68K_REG_D0;
976
977
901
  op1->address_mode = M68K_AM_NONE;
978
901
  op1->type = M68K_OP_REG_PAIR;
979
901
  op1->reg_pair.reg_0 = ((extension >> 22) & 7) + M68K_REG_D0;
980
901
  op1->reg_pair.reg_1 = ((extension >> 6) & 7) + M68K_REG_D0;
981
982
901
  reg_0 = (extension >> 28) & 7;
983
901
  reg_1 = (extension >> 12) & 7;
984
985
901
  op2->address_mode = M68K_AM_NONE;
986
901
  op2->type = M68K_OP_REG_PAIR;
987
901
  op2->reg_pair.reg_0 = reg_0 + (BIT_1F(extension) ? 8 : 0) + M68K_REG_D0;
988
901
  op2->reg_pair.reg_1 = reg_1 + (BIT_F(extension) ? 8 : 0) + M68K_REG_D0;
989
901
}
990
991
static void build_chk2_cmp2(m68k_info *info, int size)
992
843
{
993
843
  cs_m68k_op* op0;
994
843
  cs_m68k_op* op1;
995
843
  cs_m68k* ext = build_init_op(info, M68K_INS_CHK2, 2, size);
996
997
843
  uint32_t extension = read_imm_16(info);
998
999
843
  if (BIT_B(extension))
1000
214
    MCInst_setOpcode(info->inst, M68K_INS_CHK2);
1001
629
  else
1002
629
    MCInst_setOpcode(info->inst, M68K_INS_CMP2);
1003
1004
843
  op0 = &ext->operands[0];
1005
843
  op1 = &ext->operands[1];
1006
1007
843
  get_ea_mode_op(info, op0, info->ir, size);
1008
1009
843
  op1->address_mode = M68K_AM_NONE;
1010
843
  op1->type = M68K_OP_REG;
1011
843
  op1->reg = (BIT_F(extension) ? M68K_REG_A0 : M68K_REG_D0) + ((extension >> 12) & 7);
1012
843
}
1013
1014
static void build_move16(m68k_info *info, int data[2], int modes[2])
1015
710
{
1016
710
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVE16, 2, 0);
1017
710
  int i;
1018
1019
2.13k
  for (i = 0; i < 2; ++i) {
1020
1.42k
    cs_m68k_op* op = &ext->operands[i];
1021
1.42k
    const int d = data[i];
1022
1.42k
    const int m = modes[i];
1023
1024
1.42k
    op->type = M68K_OP_MEM;
1025
1026
1.42k
    if (m == M68K_AM_REGI_ADDR_POST_INC || m == M68K_AM_REG_DIRECT_ADDR) {
1027
760
      op->address_mode = m;
1028
760
      op->reg = M68K_REG_A0 + d;
1029
760
    } else {
1030
660
      op->address_mode = m;
1031
660
      op->imm = d;
1032
660
    }
1033
1.42k
  }
1034
710
}
1035
1036
static void build_link(m68k_info *info, int disp, int size)
1037
287
{
1038
287
  cs_m68k_op* op0;
1039
287
  cs_m68k_op* op1;
1040
287
  cs_m68k* ext = build_init_op(info, M68K_INS_LINK, 2, size);
1041
1042
287
  op0 = &ext->operands[0];
1043
287
  op1 = &ext->operands[1];
1044
1045
287
  op0->address_mode = M68K_AM_NONE;
1046
287
  op0->reg = M68K_REG_A0 + (info->ir & 7);
1047
1048
287
  op1->address_mode = M68K_AM_IMMEDIATE;
1049
287
  op1->type = M68K_OP_IMM;
1050
287
  op1->imm = disp;
1051
287
}
1052
1053
static void build_cpush_cinv(m68k_info *info, int op_offset)
1054
646
{
1055
646
  cs_m68k_op* op0;
1056
646
  cs_m68k_op* op1;
1057
646
  cs_m68k* ext = build_init_op(info, M68K_INS_INVALID, 2, 0);
1058
1059
646
  switch ((info->ir >> 3) & 3) { // scope
1060
    // Invalid
1061
114
    case 0:
1062
114
      d68000_invalid(info);
1063
114
      return;
1064
      // Line
1065
80
    case 1:
1066
80
      MCInst_setOpcode(info->inst, op_offset + 0);
1067
80
      break;
1068
      // Page
1069
365
    case 2:
1070
365
      MCInst_setOpcode(info->inst, op_offset + 1);
1071
365
      break;
1072
      // All
1073
87
    case 3:
1074
87
      ext->op_count = 1;
1075
87
      MCInst_setOpcode(info->inst, op_offset + 2);
1076
87
      break;
1077
646
  }
1078
1079
532
  op0 = &ext->operands[0];
1080
532
  op1 = &ext->operands[1];
1081
1082
532
  op0->address_mode = M68K_AM_IMMEDIATE;
1083
532
  op0->type = M68K_OP_IMM;
1084
532
  op0->imm = (info->ir >> 6) & 3;
1085
1086
532
  op1->type = M68K_OP_MEM;
1087
532
  op1->address_mode = M68K_AM_REG_DIRECT_ADDR;
1088
532
  op1->imm = M68K_REG_A0 + (info->ir & 7);
1089
532
}
1090
1091
static void build_movep_re(m68k_info *info, int size)
1092
591
{
1093
591
  cs_m68k_op* op0;
1094
591
  cs_m68k_op* op1;
1095
591
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVEP, 2, size);
1096
1097
591
  op0 = &ext->operands[0];
1098
591
  op1 = &ext->operands[1];
1099
1100
591
  op0->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
1101
1102
591
  op1->address_mode = M68K_AM_REGI_ADDR_DISP;
1103
591
  op1->type = M68K_OP_MEM;
1104
591
  op1->mem.base_reg = M68K_REG_A0 + (info->ir & 7);
1105
591
  op1->mem.disp = (int16_t)read_imm_16(info);
1106
591
}
1107
1108
static void build_movep_er(m68k_info *info, int size)
1109
1.41k
{
1110
1.41k
  cs_m68k_op* op0;
1111
1.41k
  cs_m68k_op* op1;
1112
1.41k
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVEP, 2, size);
1113
1114
1.41k
  op0 = &ext->operands[0];
1115
1.41k
  op1 = &ext->operands[1];
1116
1117
1.41k
  op0->address_mode = M68K_AM_REGI_ADDR_DISP;
1118
1.41k
  op0->type = M68K_OP_MEM;
1119
1.41k
  op0->mem.base_reg = M68K_REG_A0 + (info->ir & 7);
1120
1.41k
  op0->mem.disp = (int16_t)read_imm_16(info);
1121
1122
1.41k
  op1->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
1123
1.41k
}
1124
1125
static void build_moves(m68k_info *info, int size)
1126
458
{
1127
458
  cs_m68k_op* op0;
1128
458
  cs_m68k_op* op1;
1129
458
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVES, 2, size);
1130
458
  uint32_t extension = read_imm_16(info);
1131
1132
458
  op0 = &ext->operands[0];
1133
458
  op1 = &ext->operands[1];
1134
1135
458
  if (BIT_B(extension)) {
1136
182
    op0->reg = (BIT_F(extension) ? M68K_REG_A0 : M68K_REG_D0) + ((extension >> 12) & 7);
1137
182
    get_ea_mode_op(info, op1, info->ir, size);
1138
276
  } else {
1139
276
    get_ea_mode_op(info, op0, info->ir, size);
1140
276
    op1->reg = (BIT_F(extension) ? M68K_REG_A0 : M68K_REG_D0) + ((extension >> 12) & 7);
1141
276
  }
1142
458
}
1143
1144
static void build_er_1(m68k_info *info, int opcode, uint8_t size)
1145
19.7k
{
1146
19.7k
  build_er_gen_1(info, true, opcode, size);
1147
19.7k
}
1148
1149
/* ======================================================================== */
1150
/* ========================= INSTRUCTION HANDLERS ========================= */
1151
/* ======================================================================== */
1152
/* Instruction handler function names follow this convention:
1153
 *
1154
 * d68000_NAME_EXTENSIONS(void)
1155
 * where NAME is the name of the opcode it handles and EXTENSIONS are any
1156
 * extensions for special instances of that opcode.
1157
 *
1158
 * Examples:
1159
 *   d68000_add_er_8(): add opcode, from effective address to register,
1160
 *                      size = byte
1161
 *
1162
 *   d68000_asr_s_8(): arithmetic shift right, static count, size = byte
1163
 *
1164
 *
1165
 * Common extensions:
1166
 * 8   : size = byte
1167
 * 16  : size = word
1168
 * 32  : size = long
1169
 * rr  : register to register
1170
 * mm  : memory to memory
1171
 * r   : register
1172
 * s   : static
1173
 * er  : effective address -> register
1174
 * re  : register -> effective address
1175
 * ea  : using effective address mode of operation
1176
 * d   : data register direct
1177
 * a   : address register direct
1178
 * ai  : address register indirect
1179
 * pi  : address register indirect with postincrement
1180
 * pd  : address register indirect with predecrement
1181
 * di  : address register indirect with displacement
1182
 * ix  : address register indirect with index
1183
 * aw  : absolute word
1184
 * al  : absolute long
1185
 */
1186
1187
1188
static void d68000_invalid(m68k_info *info)
1189
17.4k
{
1190
17.4k
  build_invalid(info, info->ir);
1191
17.4k
}
1192
1193
static void d68000_illegal(m68k_info *info)
1194
237
{
1195
237
  build_illegal(info, info->ir);
1196
237
}
1197
1198
static void d68000_1010(m68k_info *info)
1199
9.05k
{
1200
9.05k
  build_invalid(info, info->ir);
1201
9.05k
}
1202
1203
static void d68000_1111(m68k_info *info)
1204
8.43k
{
1205
8.43k
  build_invalid(info, info->ir);
1206
8.43k
}
1207
1208
static void d68000_abcd_rr(m68k_info *info)
1209
377
{
1210
377
  build_rr(info, M68K_INS_ABCD, 1, 0);
1211
377
}
1212
1213
static void d68000_abcd_mm(m68k_info *info)
1214
536
{
1215
536
  build_mm(info, M68K_INS_ABCD, 1, 0);
1216
536
}
1217
1218
static void d68000_add_er_8(m68k_info *info)
1219
515
{
1220
515
  build_er_1(info, M68K_INS_ADD, 1);
1221
515
}
1222
1223
static void d68000_add_er_16(m68k_info *info)
1224
334
{
1225
334
  build_er_1(info, M68K_INS_ADD, 2);
1226
334
}
1227
1228
static void d68000_add_er_32(m68k_info *info)
1229
334
{
1230
334
  build_er_1(info, M68K_INS_ADD, 4);
1231
334
}
1232
1233
static void d68000_add_re_8(m68k_info *info)
1234
553
{
1235
553
  build_re_1(info, M68K_INS_ADD, 1);
1236
553
}
1237
1238
static void d68000_add_re_16(m68k_info *info)
1239
519
{
1240
519
  build_re_1(info, M68K_INS_ADD, 2);
1241
519
}
1242
1243
static void d68000_add_re_32(m68k_info *info)
1244
191
{
1245
191
  build_re_1(info, M68K_INS_ADD, 4);
1246
191
}
1247
1248
static void d68000_adda_16(m68k_info *info)
1249
2.71k
{
1250
2.71k
  build_ea_a(info, M68K_INS_ADDA, 2);
1251
2.71k
}
1252
1253
static void d68000_adda_32(m68k_info *info)
1254
2.08k
{
1255
2.08k
  build_ea_a(info, M68K_INS_ADDA, 4);
1256
2.08k
}
1257
1258
static void d68000_addi_8(m68k_info *info)
1259
396
{
1260
396
  build_imm_ea(info, M68K_INS_ADDI, 1, read_imm_8(info));
1261
396
}
1262
1263
static void d68000_addi_16(m68k_info *info)
1264
324
{
1265
324
  build_imm_ea(info, M68K_INS_ADDI, 2, read_imm_16(info));
1266
324
}
1267
1268
static void d68000_addi_32(m68k_info *info)
1269
240
{
1270
240
  build_imm_ea(info, M68K_INS_ADDI, 4, read_imm_32(info));
1271
240
}
1272
1273
static void d68000_addq_8(m68k_info *info)
1274
784
{
1275
784
  build_3bit_ea(info, M68K_INS_ADDQ, 1);
1276
784
}
1277
1278
static void d68000_addq_16(m68k_info *info)
1279
1.77k
{
1280
1.77k
  build_3bit_ea(info, M68K_INS_ADDQ, 2);
1281
1.77k
}
1282
1283
static void d68000_addq_32(m68k_info *info)
1284
776
{
1285
776
  build_3bit_ea(info, M68K_INS_ADDQ, 4);
1286
776
}
1287
1288
static void d68000_addx_rr_8(m68k_info *info)
1289
417
{
1290
417
  build_rr(info, M68K_INS_ADDX, 1, 0);
1291
417
}
1292
1293
static void d68000_addx_rr_16(m68k_info *info)
1294
382
{
1295
382
  build_rr(info, M68K_INS_ADDX, 2, 0);
1296
382
}
1297
1298
static void d68000_addx_rr_32(m68k_info *info)
1299
237
{
1300
237
  build_rr(info, M68K_INS_ADDX, 4, 0);
1301
237
}
1302
1303
static void d68000_addx_mm_8(m68k_info *info)
1304
450
{
1305
450
  build_mm(info, M68K_INS_ADDX, 1, 0);
1306
450
}
1307
1308
static void d68000_addx_mm_16(m68k_info *info)
1309
347
{
1310
347
  build_mm(info, M68K_INS_ADDX, 2, 0);
1311
347
}
1312
1313
static void d68000_addx_mm_32(m68k_info *info)
1314
253
{
1315
253
  build_mm(info, M68K_INS_ADDX, 4, 0);
1316
253
}
1317
1318
static void d68000_and_er_8(m68k_info *info)
1319
682
{
1320
682
  build_er_1(info, M68K_INS_AND, 1);
1321
682
}
1322
1323
static void d68000_and_er_16(m68k_info *info)
1324
516
{
1325
516
  build_er_1(info, M68K_INS_AND, 2);
1326
516
}
1327
1328
static void d68000_and_er_32(m68k_info *info)
1329
417
{
1330
417
  build_er_1(info, M68K_INS_AND, 4);
1331
417
}
1332
1333
static void d68000_and_re_8(m68k_info *info)
1334
323
{
1335
323
  build_re_1(info, M68K_INS_AND, 1);
1336
323
}
1337
1338
static void d68000_and_re_16(m68k_info *info)
1339
420
{
1340
420
  build_re_1(info, M68K_INS_AND, 2);
1341
420
}
1342
1343
static void d68000_and_re_32(m68k_info *info)
1344
319
{
1345
319
  build_re_1(info, M68K_INS_AND, 4);
1346
319
}
1347
1348
static void d68000_andi_8(m68k_info *info)
1349
367
{
1350
367
  build_imm_ea(info, M68K_INS_ANDI, 1, read_imm_8(info));
1351
367
}
1352
1353
static void d68000_andi_16(m68k_info *info)
1354
298
{
1355
298
  build_imm_ea(info, M68K_INS_ANDI, 2, read_imm_16(info));
1356
298
}
1357
1358
static void d68000_andi_32(m68k_info *info)
1359
248
{
1360
248
  build_imm_ea(info, M68K_INS_ANDI, 4, read_imm_32(info));
1361
248
}
1362
1363
static void d68000_andi_to_ccr(m68k_info *info)
1364
81
{
1365
81
  build_imm_special_reg(info, M68K_INS_ANDI, read_imm_8(info), 1, M68K_REG_CCR);
1366
81
}
1367
1368
static void d68000_andi_to_sr(m68k_info *info)
1369
442
{
1370
442
  build_imm_special_reg(info, M68K_INS_ANDI, read_imm_16(info), 2, M68K_REG_SR);
1371
442
}
1372
1373
static void d68000_asr_s_8(m68k_info *info)
1374
521
{
1375
521
  build_3bit_d(info, M68K_INS_ASR, 1);
1376
521
}
1377
1378
static void d68000_asr_s_16(m68k_info *info)
1379
280
{
1380
280
  build_3bit_d(info, M68K_INS_ASR, 2);
1381
280
}
1382
1383
static void d68000_asr_s_32(m68k_info *info)
1384
343
{
1385
343
  build_3bit_d(info, M68K_INS_ASR, 4);
1386
343
}
1387
1388
static void d68000_asr_r_8(m68k_info *info)
1389
332
{
1390
332
  build_r(info, M68K_INS_ASR, 1);
1391
332
}
1392
1393
static void d68000_asr_r_16(m68k_info *info)
1394
148
{
1395
148
  build_r(info, M68K_INS_ASR, 2);
1396
148
}
1397
1398
static void d68000_asr_r_32(m68k_info *info)
1399
259
{
1400
259
  build_r(info, M68K_INS_ASR, 4);
1401
259
}
1402
1403
static void d68000_asr_ea(m68k_info *info)
1404
432
{
1405
432
  build_ea(info, M68K_INS_ASR, 2);
1406
432
}
1407
1408
static void d68000_asl_s_8(m68k_info *info)
1409
368
{
1410
368
  build_3bit_d(info, M68K_INS_ASL, 1);
1411
368
}
1412
1413
static void d68000_asl_s_16(m68k_info *info)
1414
161
{
1415
161
  build_3bit_d(info, M68K_INS_ASL, 2);
1416
161
}
1417
1418
static void d68000_asl_s_32(m68k_info *info)
1419
257
{
1420
257
  build_3bit_d(info, M68K_INS_ASL, 4);
1421
257
}
1422
1423
static void d68000_asl_r_8(m68k_info *info)
1424
364
{
1425
364
  build_r(info, M68K_INS_ASL, 1);
1426
364
}
1427
1428
static void d68000_asl_r_16(m68k_info *info)
1429
335
{
1430
335
  build_r(info, M68K_INS_ASL, 2);
1431
335
}
1432
1433
static void d68000_asl_r_32(m68k_info *info)
1434
596
{
1435
596
  build_r(info, M68K_INS_ASL, 4);
1436
596
}
1437
1438
static void d68000_asl_ea(m68k_info *info)
1439
989
{
1440
989
  build_ea(info, M68K_INS_ASL, 2);
1441
989
}
1442
1443
static void d68000_bcc_8(m68k_info *info)
1444
8.54k
{
1445
8.54k
  build_bcc(info, 1, make_int_8(info->ir));
1446
8.54k
}
1447
1448
static void d68000_bcc_16(m68k_info *info)
1449
659
{
1450
659
  build_bcc(info, 2, make_int_16(read_imm_16(info)));
1451
659
}
1452
1453
static void d68020_bcc_32(m68k_info *info)
1454
217
{
1455
217
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1456
111
  build_bcc(info, 4, read_imm_32(info));
1457
111
}
1458
1459
static void d68000_bchg_r(m68k_info *info)
1460
2.15k
{
1461
2.15k
  build_re_1(info, M68K_INS_BCHG, 1);
1462
2.15k
}
1463
1464
static void d68000_bchg_s(m68k_info *info)
1465
326
{
1466
326
  build_imm_ea(info, M68K_INS_BCHG, 1, read_imm_8(info));
1467
326
}
1468
1469
static void d68000_bclr_r(m68k_info *info)
1470
1.18k
{
1471
1.18k
  build_re_1(info, M68K_INS_BCLR, 1);
1472
1.18k
}
1473
1474
static void d68000_bclr_s(m68k_info *info)
1475
97
{
1476
97
  build_imm_ea(info, M68K_INS_BCLR, 1, read_imm_8(info));
1477
97
}
1478
1479
static void d68010_bkpt(m68k_info *info)
1480
1.60k
{
1481
1.60k
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1482
1.13k
  build_absolute_jump_with_immediate(info, M68K_INS_BKPT, 0, info->ir & 7);
1483
1.13k
}
1484
1485
static void d68020_bfchg(m68k_info *info)
1486
366
{
1487
366
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1488
279
  build_bitfield_ins(info, M68K_INS_BFCHG, false);
1489
279
}
1490
1491
1492
static void d68020_bfclr(m68k_info *info)
1493
897
{
1494
897
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1495
209
  build_bitfield_ins(info, M68K_INS_BFCLR, false);
1496
209
}
1497
1498
static void d68020_bfexts(m68k_info *info)
1499
417
{
1500
417
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1501
335
  build_bitfield_ins(info, M68K_INS_BFEXTS, true);
1502
335
}
1503
1504
static void d68020_bfextu(m68k_info *info)
1505
442
{
1506
442
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1507
240
  build_bitfield_ins(info, M68K_INS_BFEXTU, true);
1508
240
}
1509
1510
static void d68020_bfffo(m68k_info *info)
1511
597
{
1512
597
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1513
239
  build_bitfield_ins(info, M68K_INS_BFFFO, true);
1514
239
}
1515
1516
static void d68020_bfins(m68k_info *info)
1517
189
{
1518
189
  cs_m68k* ext = &info->extension;
1519
189
  cs_m68k_op temp;
1520
1521
189
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1522
74
  build_bitfield_ins(info, M68K_INS_BFINS, true);
1523
1524
  // a bit hacky but we need to flip the args on only this instruction
1525
1526
74
  temp = ext->operands[0];
1527
74
  ext->operands[0] = ext->operands[1];
1528
74
  ext->operands[1] = temp;
1529
74
}
1530
1531
static void d68020_bfset(m68k_info *info)
1532
148
{
1533
148
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1534
69
  build_bitfield_ins(info, M68K_INS_BFSET, false);
1535
69
}
1536
1537
static void d68020_bftst(m68k_info *info)
1538
525
{
1539
525
  build_bitfield_ins(info, M68K_INS_BFTST, false);
1540
525
}
1541
1542
static void d68000_bra_8(m68k_info *info)
1543
1.73k
{
1544
1.73k
  build_relative_branch(info, M68K_INS_BRA, 1, make_int_8(info->ir));
1545
1.73k
}
1546
1547
static void d68000_bra_16(m68k_info *info)
1548
317
{
1549
317
  build_relative_branch(info, M68K_INS_BRA, 2, make_int_16(read_imm_16(info)));
1550
317
}
1551
1552
static void d68020_bra_32(m68k_info *info)
1553
113
{
1554
113
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1555
46
  build_relative_branch(info, M68K_INS_BRA, 4, read_imm_32(info));
1556
46
}
1557
1558
static void d68000_bset_r(m68k_info *info)
1559
3.13k
{
1560
3.13k
  build_re_1(info, M68K_INS_BSET, 1);
1561
3.13k
}
1562
1563
static void d68000_bset_s(m68k_info *info)
1564
222
{
1565
222
  build_imm_ea(info, M68K_INS_BSET, 1, read_imm_8(info));
1566
222
}
1567
1568
static void d68000_bsr_8(m68k_info *info)
1569
1.40k
{
1570
1.40k
  build_relative_branch(info, M68K_INS_BSR, 1, make_int_8(info->ir));
1571
1.40k
}
1572
1573
static void d68000_bsr_16(m68k_info *info)
1574
1.02k
{
1575
1.02k
  build_relative_branch(info, M68K_INS_BSR, 2, make_int_16(read_imm_16(info)));
1576
1.02k
}
1577
1578
static void d68020_bsr_32(m68k_info *info)
1579
184
{
1580
184
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1581
95
  build_relative_branch(info, M68K_INS_BSR, 4, read_imm_32(info));
1582
95
}
1583
1584
static void d68000_btst_r(m68k_info *info)
1585
2.97k
{
1586
2.97k
  build_re_1(info, M68K_INS_BTST, 4);
1587
2.97k
}
1588
1589
static void d68000_btst_s(m68k_info *info)
1590
118
{
1591
118
  build_imm_ea(info, M68K_INS_BTST, 1, read_imm_8(info));
1592
118
}
1593
1594
static void d68020_callm(m68k_info *info)
1595
52
{
1596
52
  LIMIT_CPU_TYPES(info, M68020_ONLY);
1597
0
  build_imm_ea(info, M68K_INS_CALLM, 0, read_imm_8(info));
1598
0
}
1599
1600
static void d68020_cas_8(m68k_info *info)
1601
105
{
1602
105
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1603
27
  build_d_d_ea(info, M68K_INS_CAS, 1);
1604
27
}
1605
1606
static void d68020_cas_16(m68k_info *info)
1607
164
{
1608
164
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1609
86
  build_d_d_ea(info, M68K_INS_CAS, 2);
1610
86
}
1611
1612
static void d68020_cas_32(m68k_info *info)
1613
157
{
1614
157
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1615
67
  build_d_d_ea(info, M68K_INS_CAS, 4);
1616
67
}
1617
1618
static void d68020_cas2_16(m68k_info *info)
1619
970
{
1620
970
  build_cas2(info, 2);
1621
970
}
1622
1623
static void d68020_cas2_32(m68k_info *info)
1624
930
{
1625
930
  build_cas2(info, 4);
1626
930
}
1627
1628
static void d68000_chk_16(m68k_info *info)
1629
281
{
1630
281
  build_er_1(info, M68K_INS_CHK, 2);
1631
281
}
1632
1633
static void d68020_chk_32(m68k_info *info)
1634
839
{
1635
839
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1636
554
  build_er_1(info, M68K_INS_CHK, 4);
1637
554
}
1638
1639
static void d68020_chk2_cmp2_8(m68k_info *info)
1640
799
{
1641
799
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1642
578
  build_chk2_cmp2(info, 1);
1643
578
}
1644
1645
static void d68020_chk2_cmp2_16(m68k_info *info)
1646
148
{
1647
148
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1648
60
  build_chk2_cmp2(info, 2);
1649
60
}
1650
1651
static void d68020_chk2_cmp2_32(m68k_info *info)
1652
278
{
1653
278
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1654
205
  build_chk2_cmp2(info, 4);
1655
205
}
1656
1657
static void d68040_cinv(m68k_info *info)
1658
269
{
1659
269
  LIMIT_CPU_TYPES(info, M68040_PLUS);
1660
171
  build_cpush_cinv(info, M68K_INS_CINVL);
1661
171
}
1662
1663
static void d68000_clr_8(m68k_info *info)
1664
286
{
1665
286
  build_ea(info, M68K_INS_CLR, 1);
1666
286
}
1667
1668
static void d68000_clr_16(m68k_info *info)
1669
475
{
1670
475
  build_ea(info, M68K_INS_CLR, 2);
1671
475
}
1672
1673
static void d68000_clr_32(m68k_info *info)
1674
348
{
1675
348
  build_ea(info, M68K_INS_CLR, 4);
1676
348
}
1677
1678
static void d68000_cmp_8(m68k_info *info)
1679
1.14k
{
1680
1.14k
  build_er_1(info, M68K_INS_CMP, 1);
1681
1.14k
}
1682
1683
static void d68000_cmp_16(m68k_info *info)
1684
858
{
1685
858
  build_er_1(info, M68K_INS_CMP, 2);
1686
858
}
1687
1688
static void d68000_cmp_32(m68k_info *info)
1689
1.34k
{
1690
1.34k
  build_er_1(info, M68K_INS_CMP, 4);
1691
1.34k
}
1692
1693
static void d68000_cmpa_16(m68k_info *info)
1694
558
{
1695
558
  build_ea_a(info, M68K_INS_CMPA, 2);
1696
558
}
1697
1698
static void d68000_cmpa_32(m68k_info *info)
1699
414
{
1700
414
  build_ea_a(info, M68K_INS_CMPA, 4);
1701
414
}
1702
1703
static void d68000_cmpi_8(m68k_info *info)
1704
233
{
1705
233
  build_imm_ea(info, M68K_INS_CMPI, 1, read_imm_8(info));
1706
233
}
1707
1708
static void d68020_cmpi_pcdi_8(m68k_info *info)
1709
179
{
1710
179
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1711
106
  build_imm_ea(info, M68K_INS_CMPI, 1, read_imm_8(info));
1712
106
}
1713
1714
static void d68020_cmpi_pcix_8(m68k_info *info)
1715
281
{
1716
281
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1717
78
  build_imm_ea(info, M68K_INS_CMPI, 1, read_imm_8(info));
1718
78
}
1719
1720
static void d68000_cmpi_16(m68k_info *info)
1721
214
{
1722
214
  build_imm_ea(info, M68K_INS_CMPI, 2, read_imm_16(info));
1723
214
}
1724
1725
static void d68020_cmpi_pcdi_16(m68k_info *info)
1726
209
{
1727
209
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1728
139
  build_imm_ea(info, M68K_INS_CMPI, 2, read_imm_16(info));
1729
139
}
1730
1731
static void d68020_cmpi_pcix_16(m68k_info *info)
1732
166
{
1733
166
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1734
131
  build_imm_ea(info, M68K_INS_CMPI, 2, read_imm_16(info));
1735
131
}
1736
1737
static void d68000_cmpi_32(m68k_info *info)
1738
111
{
1739
111
  build_imm_ea(info, M68K_INS_CMPI, 4, read_imm_32(info));
1740
111
}
1741
1742
static void d68020_cmpi_pcdi_32(m68k_info *info)
1743
53
{
1744
53
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1745
35
  build_imm_ea(info, M68K_INS_CMPI, 4, read_imm_32(info));
1746
35
}
1747
1748
static void d68020_cmpi_pcix_32(m68k_info *info)
1749
131
{
1750
131
  LIMIT_CPU_TYPES(info, M68010_PLUS);
1751
63
  build_imm_ea(info, M68K_INS_CMPI, 4, read_imm_32(info));
1752
63
}
1753
1754
static void d68000_cmpm_8(m68k_info *info)
1755
95
{
1756
95
  build_pi_pi(info, M68K_INS_CMPM, 1);
1757
95
}
1758
1759
static void d68000_cmpm_16(m68k_info *info)
1760
268
{
1761
268
  build_pi_pi(info, M68K_INS_CMPM, 2);
1762
268
}
1763
1764
static void d68000_cmpm_32(m68k_info *info)
1765
232
{
1766
232
  build_pi_pi(info, M68K_INS_CMPM, 4);
1767
232
}
1768
1769
static void make_cpbcc_operand(cs_m68k_op* op, int size, int displacement)
1770
3.70k
{
1771
3.70k
  op->address_mode = M68K_AM_BRANCH_DISPLACEMENT;
1772
3.70k
  op->type = M68K_OP_BR_DISP;
1773
3.70k
  op->br_disp.disp = displacement;
1774
3.70k
  op->br_disp.disp_size = size;
1775
3.70k
}
1776
1777
static void d68020_cpbcc_16(m68k_info *info)
1778
1.97k
{
1779
1.97k
  cs_m68k_op* op0;
1780
1.97k
  cs_m68k* ext;
1781
1.97k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1782
1783
  // FNOP is a special case of FBF
1784
1.47k
  if (info->ir == 0xf280 && peek_imm_16(info) == 0) {
1785
73
    MCInst_setOpcode(info->inst, M68K_INS_FNOP);
1786
73
    info->pc += 2;
1787
73
    return;
1788
73
  }
1789
1790
  // these are all in row with the extension so just doing a add here is fine
1791
1.40k
  info->inst->Opcode += (info->ir & 0x2f);
1792
1793
1.40k
  ext = build_init_op(info, M68K_INS_FBF, 1, 2);
1794
1.40k
  op0 = &ext->operands[0];
1795
1796
1.40k
  make_cpbcc_operand(op0, M68K_OP_BR_DISP_SIZE_WORD, make_int_16(read_imm_16(info)));
1797
1798
1.40k
  set_insn_group(info, M68K_GRP_JUMP);
1799
1.40k
  set_insn_group(info, M68K_GRP_BRANCH_RELATIVE);
1800
1.40k
}
1801
1802
static void d68020_cpbcc_32(m68k_info *info)
1803
2.30k
{
1804
2.30k
  cs_m68k* ext;
1805
2.30k
  cs_m68k_op* op0;
1806
1807
2.30k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1808
1809
  // these are all in row with the extension so just doing a add here is fine
1810
1.51k
  info->inst->Opcode += (info->ir & 0x2f);
1811
1812
1.51k
  ext = build_init_op(info, M68K_INS_FBF, 1, 4);
1813
1.51k
  op0 = &ext->operands[0];
1814
1815
1.51k
  make_cpbcc_operand(op0, M68K_OP_BR_DISP_SIZE_LONG, read_imm_32(info));
1816
1817
1.51k
  set_insn_group(info, M68K_GRP_JUMP);
1818
1.51k
  set_insn_group(info, M68K_GRP_BRANCH_RELATIVE);
1819
1.51k
}
1820
1821
static void d68020_cpdbcc(m68k_info *info)
1822
1.38k
{
1823
1.38k
  cs_m68k* ext;
1824
1.38k
  cs_m68k_op* op0;
1825
1.38k
  cs_m68k_op* op1;
1826
1.38k
  uint32_t ext1, ext2;
1827
1828
1.38k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1829
1830
785
  ext1 = read_imm_16(info);
1831
785
  ext2 = read_imm_16(info);
1832
1833
  // these are all in row with the extension so just doing a add here is fine
1834
785
  info->inst->Opcode += (ext1 & 0x2f);
1835
1836
785
  ext = build_init_op(info, M68K_INS_FDBF, 2, 0);
1837
785
  op0 = &ext->operands[0];
1838
785
  op1 = &ext->operands[1];
1839
1840
785
  op0->reg = M68K_REG_D0 + (info->ir & 7);
1841
1842
785
  make_cpbcc_operand(op1, M68K_OP_BR_DISP_SIZE_WORD, make_int_16(ext2) + 2);
1843
1844
785
  set_insn_group(info, M68K_GRP_JUMP);
1845
785
  set_insn_group(info, M68K_GRP_BRANCH_RELATIVE);
1846
785
}
1847
1848
static void fmove_fpcr(m68k_info *info, uint32_t extension)
1849
1.02k
{
1850
1.02k
  cs_m68k_op* special;
1851
1.02k
  cs_m68k_op* op_ea;
1852
1853
1.02k
  int regsel = (extension >> 10) & 0x7;
1854
1.02k
  int dir = (extension >> 13) & 0x1;
1855
1856
1.02k
  cs_m68k* ext = build_init_op(info, M68K_INS_FMOVE, 2, 4);
1857
1858
1.02k
  special = &ext->operands[0];
1859
1.02k
  op_ea = &ext->operands[1];
1860
1861
1.02k
  if (!dir) {
1862
311
    cs_m68k_op* t = special;
1863
311
    special = op_ea;
1864
311
    op_ea = t;
1865
311
  }
1866
1867
1.02k
  get_ea_mode_op(info, op_ea, info->ir, 4);
1868
1869
1.02k
  if (regsel & 4)
1870
341
    special->reg = M68K_REG_FPCR;
1871
682
  else if (regsel & 2)
1872
140
    special->reg = M68K_REG_FPSR;
1873
542
  else if (regsel & 1)
1874
289
    special->reg = M68K_REG_FPIAR;
1875
1.02k
}
1876
1877
static void fmovem(m68k_info *info, uint32_t extension)
1878
2.93k
{
1879
2.93k
  cs_m68k_op* op_reglist;
1880
2.93k
  cs_m68k_op* op_ea;
1881
2.93k
  int dir = (extension >> 13) & 0x1;
1882
2.93k
  int mode = (extension >> 11) & 0x3;
1883
2.93k
  uint32_t reglist = extension & 0xff;
1884
2.93k
  cs_m68k* ext = build_init_op(info, M68K_INS_FMOVEM, 2, 0);
1885
1886
2.93k
  op_reglist = &ext->operands[0];
1887
2.93k
  op_ea = &ext->operands[1];
1888
1889
  // flip args around
1890
1891
2.93k
  if (!dir) {
1892
575
    cs_m68k_op* t = op_reglist;
1893
575
    op_reglist = op_ea;
1894
575
    op_ea = t;
1895
575
  }
1896
1897
2.93k
  get_ea_mode_op(info, op_ea, info->ir, 0);
1898
1899
2.93k
  switch (mode) {
1900
262
    case 1 : // Dynamic list in dn register
1901
262
      op_reglist->reg = M68K_REG_D0 + ((reglist >> 4) & 7);
1902
262
      break;
1903
1904
1.40k
    case 0 :
1905
1.40k
      op_reglist->address_mode = M68K_AM_NONE;
1906
1.40k
      op_reglist->type = M68K_OP_REG_BITS;
1907
1.40k
      op_reglist->register_bits = reglist << 16;
1908
1.40k
      break;
1909
1910
693
    case 2 : // Static list
1911
693
      op_reglist->address_mode = M68K_AM_NONE;
1912
693
      op_reglist->type = M68K_OP_REG_BITS;
1913
693
      op_reglist->register_bits = ((uint32_t)reverse_bits_8(reglist)) << 16;
1914
693
      break;
1915
2.93k
  }
1916
2.93k
}
1917
1918
static void d68020_cpgen(m68k_info *info)
1919
13.8k
{
1920
13.8k
  cs_m68k *ext;
1921
13.8k
  cs_m68k_op* op0;
1922
13.8k
  cs_m68k_op* op1;
1923
13.8k
  bool supports_single_op;
1924
13.8k
  uint32_t next;
1925
13.8k
  int rm, src, dst, opmode;
1926
1927
1928
13.8k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
1929
1930
12.4k
  supports_single_op = true;
1931
1932
12.4k
  next = read_imm_16(info);
1933
1934
12.4k
  rm = (next >> 14) & 0x1;
1935
12.4k
  src = (next >> 10) & 0x7;
1936
12.4k
  dst = (next >> 7) & 0x7;
1937
12.4k
  opmode = next & 0x3f;
1938
1939
  // special handling for fmovecr
1940
1941
12.4k
  if (BITFIELD(info->ir, 5, 0) == 0 && BITFIELD(next, 15, 10) == 0x17) {
1942
86
    ext = build_init_op(info, M68K_INS_FMOVECR, 2, 0);
1943
1944
86
    op0 = &ext->operands[0];
1945
86
    op1 = &ext->operands[1];
1946
1947
86
    op0->address_mode = M68K_AM_IMMEDIATE;
1948
86
    op0->type = M68K_OP_IMM;
1949
86
    op0->imm = next & 0x3f;
1950
1951
86
    op1->reg = M68K_REG_FP0 + ((next >> 7) & 7);
1952
1953
86
    return;
1954
86
  }
1955
1956
  // deal with extended move stuff
1957
1958
12.3k
  switch ((next >> 13) & 0x7) {
1959
    // fmovem fpcr
1960
311
    case 0x4: // FMOVEM ea, FPCR
1961
1.02k
    case 0x5: // FMOVEM FPCR, ea
1962
1.02k
      fmove_fpcr(info, next);
1963
1.02k
      return;
1964
1965
    // fmovem list
1966
575
    case 0x6:
1967
2.93k
    case 0x7:
1968
2.93k
      fmovem(info, next);
1969
2.93k
      return;
1970
12.3k
  }
1971
1972
  // See comment bellow on why this is being done
1973
1974
8.43k
  if ((next >> 6) & 1)
1975
4.11k
    opmode &= ~4;
1976
1977
  // special handling of some instructions here
1978
1979
8.43k
  switch (opmode) {
1980
318
    case 0x00: MCInst_setOpcode(info->inst, M68K_INS_FMOVE); supports_single_op = false; break;
1981
621
    case 0x01: MCInst_setOpcode(info->inst, M68K_INS_FINT); break;
1982
87
    case 0x02: MCInst_setOpcode(info->inst, M68K_INS_FSINH); break;
1983
148
    case 0x03: MCInst_setOpcode(info->inst, M68K_INS_FINTRZ); break;
1984
81
    case 0x04: MCInst_setOpcode(info->inst, M68K_INS_FSQRT); break;
1985
77
    case 0x06: MCInst_setOpcode(info->inst, M68K_INS_FLOGNP1); break;
1986
151
    case 0x08: MCInst_setOpcode(info->inst, M68K_INS_FETOXM1); break;
1987
85
    case 0x09: MCInst_setOpcode(info->inst, M68K_INS_FATANH); break;
1988
91
    case 0x0a: MCInst_setOpcode(info->inst, M68K_INS_FATAN); break;
1989
79
    case 0x0c: MCInst_setOpcode(info->inst, M68K_INS_FASIN); break;
1990
73
    case 0x0d: MCInst_setOpcode(info->inst, M68K_INS_FATANH); break;
1991
81
    case 0x0e: MCInst_setOpcode(info->inst, M68K_INS_FSIN); break;
1992
99
    case 0x0f: MCInst_setOpcode(info->inst, M68K_INS_FTAN); break;
1993
250
    case 0x10: MCInst_setOpcode(info->inst, M68K_INS_FETOX); break;
1994
573
    case 0x11: MCInst_setOpcode(info->inst, M68K_INS_FTWOTOX); break;
1995
492
    case 0x12: MCInst_setOpcode(info->inst, M68K_INS_FTENTOX); break;
1996
61
    case 0x14: MCInst_setOpcode(info->inst, M68K_INS_FLOGN); break;
1997
204
    case 0x15: MCInst_setOpcode(info->inst, M68K_INS_FLOG10); break;
1998
101
    case 0x16: MCInst_setOpcode(info->inst, M68K_INS_FLOG2); break;
1999
84
    case 0x18: MCInst_setOpcode(info->inst, M68K_INS_FABS); break;
2000
719
    case 0x19: MCInst_setOpcode(info->inst, M68K_INS_FCOSH); break;
2001
103
    case 0x1a: MCInst_setOpcode(info->inst, M68K_INS_FNEG); break;
2002
82
    case 0x1c: MCInst_setOpcode(info->inst, M68K_INS_FACOS); break;
2003
59
    case 0x1d: MCInst_setOpcode(info->inst, M68K_INS_FCOS); break;
2004
82
    case 0x1e: MCInst_setOpcode(info->inst, M68K_INS_FGETEXP); break;
2005
74
    case 0x1f: MCInst_setOpcode(info->inst, M68K_INS_FGETMAN); break;
2006
389
    case 0x20: MCInst_setOpcode(info->inst, M68K_INS_FDIV); supports_single_op = false; break;
2007
624
    case 0x21: MCInst_setOpcode(info->inst, M68K_INS_FMOD); supports_single_op = false; break;
2008
300
    case 0x22: MCInst_setOpcode(info->inst, M68K_INS_FADD); supports_single_op = false; break;
2009
72
    case 0x23: MCInst_setOpcode(info->inst, M68K_INS_FMUL); supports_single_op = false; break;
2010
246
    case 0x24: MCInst_setOpcode(info->inst, M68K_INS_FSGLDIV); supports_single_op = false; break;
2011
81
    case 0x25: MCInst_setOpcode(info->inst, M68K_INS_FREM); break;
2012
135
    case 0x26: MCInst_setOpcode(info->inst, M68K_INS_FSCALE); break;
2013
100
    case 0x27: MCInst_setOpcode(info->inst, M68K_INS_FSGLMUL); break;
2014
93
    case 0x28: MCInst_setOpcode(info->inst, M68K_INS_FSUB); supports_single_op = false; break;
2015
158
    case 0x38: MCInst_setOpcode(info->inst, M68K_INS_FCMP); supports_single_op = false; break;
2016
579
    case 0x3a: MCInst_setOpcode(info->inst, M68K_INS_FTST); break;
2017
781
    default:
2018
781
      break;
2019
8.43k
  }
2020
2021
  // Some trickery here! It's not documented but if bit 6 is set this is a s/d opcode and then
2022
  // if bit 2 is set it's a d. As we already have set our opcode in the code above we can just
2023
  // offset it as the following 2 op codes (if s/d is supported) will always be directly after it
2024
2025
8.43k
  if ((next >> 6) & 1) {
2026
4.11k
    if ((next >> 2) & 1)
2027
1.56k
      info->inst->Opcode += 2;
2028
2.54k
    else
2029
2.54k
      info->inst->Opcode += 1;
2030
4.11k
  }
2031
2032
8.43k
  ext = &info->extension;
2033
2034
8.43k
  ext->op_count = 2;
2035
8.43k
  ext->op_size.type = M68K_SIZE_TYPE_CPU;
2036
8.43k
  ext->op_size.cpu_size = 0;
2037
2038
  // Special case - adjust direction of fmove
2039
8.43k
  if ((opmode == 0x00) && ((next >> 13) & 0x1) != 0) {
2040
41
    op0 = &ext->operands[1];
2041
41
    op1 = &ext->operands[0];
2042
8.39k
  } else {
2043
8.39k
    op0 = &ext->operands[0];
2044
8.39k
    op1 = &ext->operands[1];
2045
8.39k
  }
2046
2047
8.43k
  if (rm == 0 && supports_single_op && src == dst) {
2048
566
    ext->op_count = 1;
2049
566
    op0->reg = M68K_REG_FP0 + dst;
2050
566
    return;
2051
566
  }
2052
2053
7.86k
  if (rm == 1) {
2054
4.19k
    switch (src) {
2055
749
      case 0x00 :
2056
749
        ext->op_size.cpu_size = M68K_CPU_SIZE_LONG;
2057
749
        get_ea_mode_op(info, op0, info->ir, 4);
2058
749
        break;
2059
2060
886
      case 0x06 :
2061
886
        ext->op_size.cpu_size = M68K_CPU_SIZE_BYTE;
2062
886
        get_ea_mode_op(info, op0, info->ir, 1);
2063
886
        break;
2064
2065
551
      case 0x04 :
2066
551
        ext->op_size.cpu_size = M68K_CPU_SIZE_WORD;
2067
551
        get_ea_mode_op(info, op0, info->ir, 2);
2068
551
        break;
2069
2070
1.40k
      case 0x01 :
2071
1.40k
        ext->op_size.type = M68K_SIZE_TYPE_FPU;
2072
1.40k
        ext->op_size.fpu_size = M68K_FPU_SIZE_SINGLE;
2073
1.40k
        get_ea_mode_op(info, op0, info->ir, 4);
2074
1.40k
        op0->simm = BitsToFloat(op0->imm);
2075
1.40k
        op0->type = M68K_OP_FP_SINGLE;
2076
1.40k
        break;
2077
2078
345
      case 0x05:
2079
345
        ext->op_size.type = M68K_SIZE_TYPE_FPU;
2080
345
        ext->op_size.fpu_size = M68K_FPU_SIZE_DOUBLE;
2081
345
        get_ea_mode_op(info, op0, info->ir, 8);
2082
345
        op0->type = M68K_OP_FP_DOUBLE;
2083
345
        break;
2084
2085
265
      default :
2086
265
        ext->op_size.type = M68K_SIZE_TYPE_FPU;
2087
265
        ext->op_size.fpu_size = M68K_FPU_SIZE_EXTENDED;
2088
265
        break;
2089
4.19k
    }
2090
4.19k
  } else {
2091
3.67k
    op0->reg = M68K_REG_FP0 + src;
2092
3.67k
  }
2093
2094
7.86k
  op1->reg = M68K_REG_FP0 + dst;
2095
7.86k
}
2096
2097
static void d68020_cprestore(m68k_info *info)
2098
1.00k
{
2099
1.00k
  cs_m68k* ext;
2100
1.00k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2101
2102
714
  ext = build_init_op(info, M68K_INS_FRESTORE, 1, 0);
2103
714
  get_ea_mode_op(info, &ext->operands[0], info->ir, 1);
2104
714
}
2105
2106
static void d68020_cpsave(m68k_info *info)
2107
1.07k
{
2108
1.07k
  cs_m68k* ext;
2109
2110
1.07k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2111
2112
756
  ext = build_init_op(info, M68K_INS_FSAVE, 1, 0);
2113
756
  get_ea_mode_op(info, &ext->operands[0], info->ir, 1);
2114
756
}
2115
2116
static void d68020_cpscc(m68k_info *info)
2117
1.20k
{
2118
1.20k
  cs_m68k* ext;
2119
2120
1.20k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2121
737
  ext = build_init_op(info, M68K_INS_FSF, 1, 1);
2122
2123
  // these are all in row with the extension so just doing a add here is fine
2124
737
  info->inst->Opcode += (read_imm_16(info) & 0x2f);
2125
2126
737
  get_ea_mode_op(info, &ext->operands[0], info->ir, 1);
2127
737
}
2128
2129
static void d68020_cptrapcc_0(m68k_info *info)
2130
314
{
2131
314
  uint32_t extension1;
2132
314
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2133
2134
230
  extension1 = read_imm_16(info);
2135
2136
230
  build_init_op(info, M68K_INS_FTRAPF, 0, 0);
2137
2138
  // these are all in row with the extension so just doing a add here is fine
2139
230
  info->inst->Opcode += (extension1 & 0x2f);
2140
230
}
2141
2142
static void d68020_cptrapcc_16(m68k_info *info)
2143
296
{
2144
296
  uint32_t extension1, extension2;
2145
296
  cs_m68k_op* op0;
2146
296
  cs_m68k* ext;
2147
2148
296
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2149
2150
86
  extension1 = read_imm_16(info);
2151
86
  extension2 = read_imm_16(info);
2152
2153
86
  ext = build_init_op(info, M68K_INS_FTRAPF, 1, 2);
2154
2155
  // these are all in row with the extension so just doing a add here is fine
2156
86
  info->inst->Opcode += (extension1 & 0x2f);
2157
2158
86
  op0 = &ext->operands[0];
2159
2160
86
  op0->address_mode = M68K_AM_IMMEDIATE;
2161
86
  op0->type = M68K_OP_IMM;
2162
86
  op0->imm = extension2;
2163
86
}
2164
2165
static void d68020_cptrapcc_32(m68k_info *info)
2166
171
{
2167
171
  uint32_t extension1, extension2;
2168
171
  cs_m68k* ext;
2169
171
  cs_m68k_op* op0;
2170
2171
171
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2172
2173
102
  extension1 = read_imm_16(info);
2174
102
  extension2 = read_imm_32(info);
2175
2176
102
  ext = build_init_op(info, M68K_INS_FTRAPF, 1, 2);
2177
2178
  // these are all in row with the extension so just doing a add here is fine
2179
102
  info->inst->Opcode += (extension1 & 0x2f);
2180
2181
102
  op0 = &ext->operands[0];
2182
2183
102
  op0->address_mode = M68K_AM_IMMEDIATE;
2184
102
  op0->type = M68K_OP_IMM;
2185
102
  op0->imm = extension2;
2186
102
}
2187
2188
static void d68040_cpush(m68k_info *info)
2189
698
{
2190
698
  LIMIT_CPU_TYPES(info, M68040_PLUS);
2191
475
  build_cpush_cinv(info, M68K_INS_CPUSHL);
2192
475
}
2193
2194
static void d68000_dbra(m68k_info *info)
2195
129
{
2196
129
  build_dbxx(info, M68K_INS_DBRA, 0, make_int_16(read_imm_16(info)));
2197
129
}
2198
2199
static void d68000_dbcc(m68k_info *info)
2200
159
{
2201
159
  build_dbcc(info, 0, make_int_16(read_imm_16(info)));
2202
159
}
2203
2204
static void d68000_divs(m68k_info *info)
2205
1.08k
{
2206
1.08k
  build_er_1(info, M68K_INS_DIVS, 2);
2207
1.08k
}
2208
2209
static void d68000_divu(m68k_info *info)
2210
657
{
2211
657
  build_er_1(info, M68K_INS_DIVU, 2);
2212
657
}
2213
2214
static void d68020_divl(m68k_info *info)
2215
149
{
2216
149
  uint32_t extension, insn_signed;
2217
149
  cs_m68k* ext;
2218
149
  cs_m68k_op* op0;
2219
149
  cs_m68k_op* op1;
2220
149
  uint32_t reg_0, reg_1;
2221
2222
149
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2223
2224
115
  extension = read_imm_16(info);
2225
115
  insn_signed = 0;
2226
2227
115
  if (BIT_B((extension)))
2228
85
    insn_signed = 1;
2229
2230
115
  ext = build_init_op(info, insn_signed ? M68K_INS_DIVS : M68K_INS_DIVU, 2, 4);
2231
2232
115
  op0 = &ext->operands[0];
2233
115
  op1 = &ext->operands[1];
2234
2235
115
  get_ea_mode_op(info, op0, info->ir, 4);
2236
2237
115
  reg_0 = extension & 7;
2238
115
  reg_1 = (extension >> 12) & 7;
2239
2240
115
  op1->address_mode = M68K_AM_NONE;
2241
115
  op1->type = M68K_OP_REG_PAIR;
2242
115
  op1->reg_pair.reg_0 = reg_0 + M68K_REG_D0;
2243
115
  op1->reg_pair.reg_1 = reg_1 + M68K_REG_D0;
2244
2245
115
  if ((reg_0 == reg_1) || !BIT_A(extension)) {
2246
89
    op1->type = M68K_OP_REG;
2247
89
    op1->reg = M68K_REG_D0 + reg_1;
2248
89
  }
2249
115
}
2250
2251
static void d68000_eor_8(m68k_info *info)
2252
557
{
2253
557
  build_re_1(info, M68K_INS_EOR, 1);
2254
557
}
2255
2256
static void d68000_eor_16(m68k_info *info)
2257
690
{
2258
690
  build_re_1(info, M68K_INS_EOR, 2);
2259
690
}
2260
2261
static void d68000_eor_32(m68k_info *info)
2262
1.92k
{
2263
1.92k
  build_re_1(info, M68K_INS_EOR, 4);
2264
1.92k
}
2265
2266
static void d68000_eori_8(m68k_info *info)
2267
150
{
2268
150
  build_imm_ea(info, M68K_INS_EORI, 1, read_imm_8(info));
2269
150
}
2270
2271
static void d68000_eori_16(m68k_info *info)
2272
264
{
2273
264
  build_imm_ea(info, M68K_INS_EORI, 2, read_imm_16(info));
2274
264
}
2275
2276
static void d68000_eori_32(m68k_info *info)
2277
739
{
2278
739
  build_imm_ea(info, M68K_INS_EORI, 4, read_imm_32(info));
2279
739
}
2280
2281
static void d68000_eori_to_ccr(m68k_info *info)
2282
79
{
2283
79
  build_imm_special_reg(info, M68K_INS_EORI, read_imm_8(info), 1, M68K_REG_CCR);
2284
79
}
2285
2286
static void d68000_eori_to_sr(m68k_info *info)
2287
104
{
2288
104
  build_imm_special_reg(info, M68K_INS_EORI, read_imm_16(info), 2, M68K_REG_SR);
2289
104
}
2290
2291
static void d68000_exg_dd(m68k_info *info)
2292
97
{
2293
97
  build_r(info, M68K_INS_EXG, 4);
2294
97
}
2295
2296
static void d68000_exg_aa(m68k_info *info)
2297
115
{
2298
115
  cs_m68k_op* op0;
2299
115
  cs_m68k_op* op1;
2300
115
  cs_m68k* ext = build_init_op(info, M68K_INS_EXG, 2, 4);
2301
2302
115
  op0 = &ext->operands[0];
2303
115
  op1 = &ext->operands[1];
2304
2305
115
  op0->address_mode = M68K_AM_NONE;
2306
115
  op0->reg = M68K_REG_A0 + ((info->ir >> 9) & 7);
2307
2308
115
  op1->address_mode = M68K_AM_NONE;
2309
115
  op1->reg = M68K_REG_A0 + (info->ir & 7);
2310
115
}
2311
2312
static void d68000_exg_da(m68k_info *info)
2313
117
{
2314
117
  cs_m68k_op* op0;
2315
117
  cs_m68k_op* op1;
2316
117
  cs_m68k* ext = build_init_op(info, M68K_INS_EXG, 2, 4);
2317
2318
117
  op0 = &ext->operands[0];
2319
117
  op1 = &ext->operands[1];
2320
2321
117
  op0->address_mode = M68K_AM_NONE;
2322
117
  op0->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
2323
2324
117
  op1->address_mode = M68K_AM_NONE;
2325
117
  op1->reg = M68K_REG_A0 + (info->ir & 7);
2326
117
}
2327
2328
static void d68000_ext_16(m68k_info *info)
2329
41
{
2330
41
  build_d(info, M68K_INS_EXT, 2);
2331
41
}
2332
2333
static void d68000_ext_32(m68k_info *info)
2334
77
{
2335
77
  build_d(info, M68K_INS_EXT, 4);
2336
77
}
2337
2338
static void d68020_extb_32(m68k_info *info)
2339
397
{
2340
397
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2341
200
  build_d(info, M68K_INS_EXTB, 4);
2342
200
}
2343
2344
static void d68000_jmp(m68k_info *info)
2345
269
{
2346
269
  cs_m68k* ext = build_init_op(info, M68K_INS_JMP, 1, 0);
2347
269
  set_insn_group(info, M68K_GRP_JUMP);
2348
269
  get_ea_mode_op(info, &ext->operands[0], info->ir, 4);
2349
269
}
2350
2351
static void d68000_jsr(m68k_info *info)
2352
175
{
2353
175
  cs_m68k* ext = build_init_op(info, M68K_INS_JSR, 1, 0);
2354
175
  set_insn_group(info, M68K_GRP_JUMP);
2355
175
  get_ea_mode_op(info, &ext->operands[0], info->ir, 4);
2356
175
}
2357
2358
static void d68000_lea(m68k_info *info)
2359
535
{
2360
535
  build_ea_a(info, M68K_INS_LEA, 4);
2361
535
}
2362
2363
static void d68000_link_16(m68k_info *info)
2364
211
{
2365
211
  build_link(info, read_imm_16(info), 2);
2366
211
}
2367
2368
static void d68020_link_32(m68k_info *info)
2369
111
{
2370
111
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2371
76
  build_link(info, read_imm_32(info), 4);
2372
76
}
2373
2374
static void d68000_lsr_s_8(m68k_info *info)
2375
823
{
2376
823
  build_3bit_d(info, M68K_INS_LSR, 1);
2377
823
}
2378
2379
static void d68000_lsr_s_16(m68k_info *info)
2380
456
{
2381
456
  build_3bit_d(info, M68K_INS_LSR, 2);
2382
456
}
2383
2384
static void d68000_lsr_s_32(m68k_info *info)
2385
220
{
2386
220
  build_3bit_d(info, M68K_INS_LSR, 4);
2387
220
}
2388
2389
static void d68000_lsr_r_8(m68k_info *info)
2390
432
{
2391
432
  build_r(info, M68K_INS_LSR, 1);
2392
432
}
2393
2394
static void d68000_lsr_r_16(m68k_info *info)
2395
229
{
2396
229
  build_r(info, M68K_INS_LSR, 2);
2397
229
}
2398
2399
static void d68000_lsr_r_32(m68k_info *info)
2400
255
{
2401
255
  build_r(info, M68K_INS_LSR, 4);
2402
255
}
2403
2404
static void d68000_lsr_ea(m68k_info *info)
2405
1.96k
{
2406
1.96k
  build_ea(info, M68K_INS_LSR, 2);
2407
1.96k
}
2408
2409
static void d68000_lsl_s_8(m68k_info *info)
2410
591
{
2411
591
  build_3bit_d(info, M68K_INS_LSL, 1);
2412
591
}
2413
2414
static void d68000_lsl_s_16(m68k_info *info)
2415
148
{
2416
148
  build_3bit_d(info, M68K_INS_LSL, 2);
2417
148
}
2418
2419
static void d68000_lsl_s_32(m68k_info *info)
2420
69
{
2421
69
  build_3bit_d(info, M68K_INS_LSL, 4);
2422
69
}
2423
2424
static void d68000_lsl_r_8(m68k_info *info)
2425
219
{
2426
219
  build_r(info, M68K_INS_LSL, 1);
2427
219
}
2428
2429
static void d68000_lsl_r_16(m68k_info *info)
2430
105
{
2431
105
  build_r(info, M68K_INS_LSL, 2);
2432
105
}
2433
2434
static void d68000_lsl_r_32(m68k_info *info)
2435
95
{
2436
95
  build_r(info, M68K_INS_LSL, 4);
2437
95
}
2438
2439
static void d68000_lsl_ea(m68k_info *info)
2440
354
{
2441
354
  build_ea(info, M68K_INS_LSL, 2);
2442
354
}
2443
2444
static void d68000_move_8(m68k_info *info)
2445
7.73k
{
2446
7.73k
  build_ea_ea(info, M68K_INS_MOVE, 1);
2447
7.73k
}
2448
2449
static void d68000_move_16(m68k_info *info)
2450
8.12k
{
2451
8.12k
  build_ea_ea(info, M68K_INS_MOVE, 2);
2452
8.12k
}
2453
2454
static void d68000_move_32(m68k_info *info)
2455
11.3k
{
2456
11.3k
  build_ea_ea(info, M68K_INS_MOVE, 4);
2457
11.3k
}
2458
2459
static void d68000_movea_16(m68k_info *info)
2460
1.55k
{
2461
1.55k
  build_ea_a(info, M68K_INS_MOVEA, 2);
2462
1.55k
}
2463
2464
static void d68000_movea_32(m68k_info *info)
2465
3.29k
{
2466
3.29k
  build_ea_a(info, M68K_INS_MOVEA, 4);
2467
3.29k
}
2468
2469
static void d68000_move_to_ccr(m68k_info *info)
2470
434
{
2471
434
  cs_m68k_op* op0;
2472
434
  cs_m68k_op* op1;
2473
434
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVE, 2, 2);
2474
2475
434
  op0 = &ext->operands[0];
2476
434
  op1 = &ext->operands[1];
2477
2478
434
  get_ea_mode_op(info, op0, info->ir, 1);
2479
2480
434
  op1->address_mode = M68K_AM_NONE;
2481
434
  op1->reg = M68K_REG_CCR;
2482
434
}
2483
2484
static void d68010_move_fr_ccr(m68k_info *info)
2485
903
{
2486
903
  cs_m68k_op* op0;
2487
903
  cs_m68k_op* op1;
2488
903
  cs_m68k* ext;
2489
2490
903
  LIMIT_CPU_TYPES(info, M68010_PLUS);
2491
2492
329
  ext = build_init_op(info, M68K_INS_MOVE, 2, 2);
2493
2494
329
  op0 = &ext->operands[0];
2495
329
  op1 = &ext->operands[1];
2496
2497
329
  op0->address_mode = M68K_AM_NONE;
2498
329
  op0->reg = M68K_REG_CCR;
2499
2500
329
  get_ea_mode_op(info, op1, info->ir, 1);
2501
329
}
2502
2503
static void d68000_move_fr_sr(m68k_info *info)
2504
675
{
2505
675
  cs_m68k_op* op0;
2506
675
  cs_m68k_op* op1;
2507
675
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVE, 2, 2);
2508
2509
675
  op0 = &ext->operands[0];
2510
675
  op1 = &ext->operands[1];
2511
2512
675
  op0->address_mode = M68K_AM_NONE;
2513
675
  op0->reg = M68K_REG_SR;
2514
2515
675
  get_ea_mode_op(info, op1, info->ir, 2);
2516
675
}
2517
2518
static void d68000_move_to_sr(m68k_info *info)
2519
299
{
2520
299
  cs_m68k_op* op0;
2521
299
  cs_m68k_op* op1;
2522
299
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVE, 2, 2);
2523
2524
299
  op0 = &ext->operands[0];
2525
299
  op1 = &ext->operands[1];
2526
2527
299
  get_ea_mode_op(info, op0, info->ir, 2);
2528
2529
299
  op1->address_mode = M68K_AM_NONE;
2530
299
  op1->reg = M68K_REG_SR;
2531
299
}
2532
2533
static void d68000_move_fr_usp(m68k_info *info)
2534
258
{
2535
258
  cs_m68k_op* op0;
2536
258
  cs_m68k_op* op1;
2537
258
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVE, 2, 0);
2538
2539
258
  op0 = &ext->operands[0];
2540
258
  op1 = &ext->operands[1];
2541
2542
258
  op0->address_mode = M68K_AM_NONE;
2543
258
  op0->reg = M68K_REG_USP;
2544
2545
258
  op1->address_mode = M68K_AM_NONE;
2546
258
  op1->reg = M68K_REG_A0 + (info->ir & 7);
2547
258
}
2548
2549
static void d68000_move_to_usp(m68k_info *info)
2550
107
{
2551
107
  cs_m68k_op* op0;
2552
107
  cs_m68k_op* op1;
2553
107
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVE, 2, 0);
2554
2555
107
  op0 = &ext->operands[0];
2556
107
  op1 = &ext->operands[1];
2557
2558
107
  op0->address_mode = M68K_AM_NONE;
2559
107
  op0->reg = M68K_REG_A0 + (info->ir & 7);
2560
2561
107
  op1->address_mode = M68K_AM_NONE;
2562
107
  op1->reg = M68K_REG_USP;
2563
107
}
2564
2565
static void d68010_movec(m68k_info *info)
2566
3.08k
{
2567
3.08k
  uint32_t extension;
2568
3.08k
  m68k_reg reg;
2569
3.08k
  cs_m68k* ext;
2570
3.08k
  cs_m68k_op* op0;
2571
3.08k
  cs_m68k_op* op1;
2572
2573
2574
3.08k
  LIMIT_CPU_TYPES(info, M68010_PLUS);
2575
2576
3.00k
  extension = read_imm_16(info);
2577
3.00k
  reg = M68K_REG_INVALID;
2578
2579
3.00k
  ext = build_init_op(info, M68K_INS_MOVEC, 2, 0);
2580
2581
3.00k
  op0 = &ext->operands[0];
2582
3.00k
  op1 = &ext->operands[1];
2583
2584
3.00k
  switch (extension & 0xfff) {
2585
27
    case 0x000: reg = M68K_REG_SFC; break;
2586
52
    case 0x001: reg = M68K_REG_DFC; break;
2587
40
    case 0x800: reg = M68K_REG_USP; break;
2588
104
    case 0x801: reg = M68K_REG_VBR; break;
2589
66
    case 0x002: reg = M68K_REG_CACR; break;
2590
78
    case 0x802: reg = M68K_REG_CAAR; break;
2591
73
    case 0x803: reg = M68K_REG_MSP; break;
2592
84
    case 0x804: reg = M68K_REG_ISP; break;
2593
22
    case 0x003: reg = M68K_REG_TC; break;
2594
113
    case 0x004: reg = M68K_REG_ITT0; break;
2595
205
    case 0x005: reg = M68K_REG_ITT1; break;
2596
86
    case 0x006: reg = M68K_REG_DTT0; break;
2597
352
    case 0x007: reg = M68K_REG_DTT1; break;
2598
79
    case 0x805: reg = M68K_REG_MMUSR; break;
2599
304
    case 0x806: reg = M68K_REG_URP; break;
2600
243
    case 0x807: reg = M68K_REG_SRP; break;
2601
3.00k
  }
2602
2603
3.00k
  if (BIT_0(info->ir)) {
2604
1.27k
    op0->reg = (BIT_F(extension) ? M68K_REG_A0 : M68K_REG_D0) + ((extension >> 12) & 7);
2605
1.27k
    op1->reg = reg;
2606
1.73k
  } else {
2607
1.73k
    op0->reg = reg;
2608
1.73k
    op1->reg = (BIT_F(extension) ? M68K_REG_A0 : M68K_REG_D0) + ((extension >> 12) & 7);
2609
1.73k
  }
2610
3.00k
}
2611
2612
static void d68000_movem_pd_16(m68k_info *info)
2613
747
{
2614
747
  build_movem_re(info, M68K_INS_MOVEM, 2);
2615
747
}
2616
2617
static void d68000_movem_pd_32(m68k_info *info)
2618
99
{
2619
99
  build_movem_re(info, M68K_INS_MOVEM, 4);
2620
99
}
2621
2622
static void d68000_movem_er_16(m68k_info *info)
2623
704
{
2624
704
  build_movem_er(info, M68K_INS_MOVEM, 2);
2625
704
}
2626
2627
static void d68000_movem_er_32(m68k_info *info)
2628
581
{
2629
581
  build_movem_er(info, M68K_INS_MOVEM, 4);
2630
581
}
2631
2632
static void d68000_movem_re_16(m68k_info *info)
2633
267
{
2634
267
  build_movem_re(info, M68K_INS_MOVEM, 2);
2635
267
}
2636
2637
static void d68000_movem_re_32(m68k_info *info)
2638
656
{
2639
656
  build_movem_re(info, M68K_INS_MOVEM, 4);
2640
656
}
2641
2642
static void d68000_movep_re_16(m68k_info *info)
2643
346
{
2644
346
  build_movep_re(info, 2);
2645
346
}
2646
2647
static void d68000_movep_re_32(m68k_info *info)
2648
245
{
2649
245
  build_movep_re(info, 4);
2650
245
}
2651
2652
static void d68000_movep_er_16(m68k_info *info)
2653
540
{
2654
540
  build_movep_er(info, 2);
2655
540
}
2656
2657
static void d68000_movep_er_32(m68k_info *info)
2658
872
{
2659
872
  build_movep_er(info, 4);
2660
872
}
2661
2662
static void d68010_moves_8(m68k_info *info)
2663
238
{
2664
238
  LIMIT_CPU_TYPES(info, M68010_PLUS);
2665
196
  build_moves(info, 1);
2666
196
}
2667
2668
static void d68010_moves_16(m68k_info *info)
2669
84
{
2670
  //uint32_t extension;
2671
84
  LIMIT_CPU_TYPES(info, M68010_PLUS);
2672
47
  build_moves(info, 2);
2673
47
}
2674
2675
static void d68010_moves_32(m68k_info *info)
2676
306
{
2677
306
  LIMIT_CPU_TYPES(info, M68010_PLUS);
2678
215
  build_moves(info, 4);
2679
215
}
2680
2681
static void d68000_moveq(m68k_info *info)
2682
7.31k
{
2683
7.31k
  cs_m68k_op* op0;
2684
7.31k
  cs_m68k_op* op1;
2685
2686
7.31k
  cs_m68k* ext = build_init_op(info, M68K_INS_MOVEQ, 2, 0);
2687
2688
7.31k
  op0 = &ext->operands[0];
2689
7.31k
  op1 = &ext->operands[1];
2690
2691
7.31k
  op0->type = M68K_OP_IMM;
2692
7.31k
  op0->address_mode = M68K_AM_IMMEDIATE;
2693
7.31k
  op0->imm = (info->ir & 0xff);
2694
2695
7.31k
  op1->address_mode = M68K_AM_REG_DIRECT_DATA;
2696
7.31k
  op1->reg = M68K_REG_D0 + ((info->ir >> 9) & 7);
2697
7.31k
}
2698
2699
static void d68040_move16_pi_pi(m68k_info *info)
2700
156
{
2701
156
  int data[] = { info->ir & 7, (read_imm_16(info) >> 12) & 7 };
2702
156
  int modes[] = { M68K_AM_REGI_ADDR_POST_INC, M68K_AM_REGI_ADDR_POST_INC };
2703
2704
156
  LIMIT_CPU_TYPES(info, M68040_PLUS);
2705
2706
50
  build_move16(info, data, modes);
2707
50
}
2708
2709
static void d68040_move16_pi_al(m68k_info *info)
2710
347
{
2711
347
  int data[] = { info->ir & 7, read_imm_32(info) };
2712
347
  int modes[] = { M68K_AM_REGI_ADDR_POST_INC, M68K_AM_ABSOLUTE_DATA_LONG };
2713
2714
347
  LIMIT_CPU_TYPES(info, M68040_PLUS);
2715
2716
104
  build_move16(info, data, modes);
2717
104
}
2718
2719
static void d68040_move16_al_pi(m68k_info *info)
2720
426
{
2721
426
  int data[] = { read_imm_32(info), info->ir & 7 };
2722
426
  int modes[] = { M68K_AM_ABSOLUTE_DATA_LONG, M68K_AM_REGI_ADDR_POST_INC };
2723
2724
426
  LIMIT_CPU_TYPES(info, M68040_PLUS);
2725
2726
224
  build_move16(info, data, modes);
2727
224
}
2728
2729
static void d68040_move16_ai_al(m68k_info *info)
2730
317
{
2731
317
  int data[] = { info->ir & 7, read_imm_32(info) };
2732
317
  int modes[] = { M68K_AM_REG_DIRECT_ADDR, M68K_AM_ABSOLUTE_DATA_LONG };
2733
2734
317
  LIMIT_CPU_TYPES(info, M68040_PLUS);
2735
2736
240
  build_move16(info, data, modes);
2737
240
}
2738
2739
static void d68040_move16_al_ai(m68k_info *info)
2740
138
{
2741
138
  int data[] = { read_imm_32(info), info->ir & 7 };
2742
138
  int modes[] = { M68K_AM_ABSOLUTE_DATA_LONG, M68K_AM_REG_DIRECT_ADDR };
2743
2744
138
  LIMIT_CPU_TYPES(info, M68040_PLUS);
2745
2746
92
  build_move16(info, data, modes);
2747
92
}
2748
2749
static void d68000_muls(m68k_info *info)
2750
1.49k
{
2751
1.49k
  build_er_1(info, M68K_INS_MULS, 2);
2752
1.49k
}
2753
2754
static void d68000_mulu(m68k_info *info)
2755
1.51k
{
2756
1.51k
  build_er_1(info, M68K_INS_MULU, 2);
2757
1.51k
}
2758
2759
static void d68020_mull(m68k_info *info)
2760
350
{
2761
350
  uint32_t extension, insn_signed;
2762
350
  cs_m68k* ext;
2763
350
  cs_m68k_op* op0;
2764
350
  cs_m68k_op* op1;
2765
350
  uint32_t reg_0, reg_1;
2766
2767
350
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2768
2769
282
  extension = read_imm_16(info);
2770
282
  insn_signed = 0;
2771
2772
282
  if (BIT_B((extension)))
2773
106
    insn_signed = 1;
2774
2775
282
  ext = build_init_op(info, insn_signed ? M68K_INS_MULS : M68K_INS_MULU, 2, 4);
2776
2777
282
  op0 = &ext->operands[0];
2778
282
  op1 = &ext->operands[1];
2779
2780
282
  get_ea_mode_op(info, op0, info->ir, 4);
2781
2782
282
  reg_0 = extension & 7;
2783
282
  reg_1 = (extension >> 12) & 7;
2784
2785
282
  op1->address_mode = M68K_AM_NONE;
2786
282
  op1->type = M68K_OP_REG_PAIR;
2787
282
  op1->reg_pair.reg_0 = reg_0 + M68K_REG_D0;
2788
282
  op1->reg_pair.reg_1 = reg_1 + M68K_REG_D0;
2789
2790
282
  if (!BIT_A(extension)) {
2791
142
    op1->type = M68K_OP_REG;
2792
142
    op1->reg = M68K_REG_D0 + reg_1;
2793
142
  }
2794
282
}
2795
2796
static void d68000_nbcd(m68k_info *info)
2797
584
{
2798
584
  build_ea(info, M68K_INS_NBCD, 1);
2799
584
}
2800
2801
static void d68000_neg_8(m68k_info *info)
2802
311
{
2803
311
  build_ea(info, M68K_INS_NEG, 1);
2804
311
}
2805
2806
static void d68000_neg_16(m68k_info *info)
2807
744
{
2808
744
  build_ea(info, M68K_INS_NEG, 2);
2809
744
}
2810
2811
static void d68000_neg_32(m68k_info *info)
2812
225
{
2813
225
  build_ea(info, M68K_INS_NEG, 4);
2814
225
}
2815
2816
static void d68000_negx_8(m68k_info *info)
2817
417
{
2818
417
  build_ea(info, M68K_INS_NEGX, 1);
2819
417
}
2820
2821
static void d68000_negx_16(m68k_info *info)
2822
529
{
2823
529
  build_ea(info, M68K_INS_NEGX, 2);
2824
529
}
2825
2826
static void d68000_negx_32(m68k_info *info)
2827
294
{
2828
294
  build_ea(info, M68K_INS_NEGX, 4);
2829
294
}
2830
2831
static void d68000_nop(m68k_info *info)
2832
70
{
2833
70
  MCInst_setOpcode(info->inst, M68K_INS_NOP);
2834
70
}
2835
2836
static void d68000_not_8(m68k_info *info)
2837
374
{
2838
374
  build_ea(info, M68K_INS_NOT, 1);
2839
374
}
2840
2841
static void d68000_not_16(m68k_info *info)
2842
640
{
2843
640
  build_ea(info, M68K_INS_NOT, 2);
2844
640
}
2845
2846
static void d68000_not_32(m68k_info *info)
2847
225
{
2848
225
  build_ea(info, M68K_INS_NOT, 4);
2849
225
}
2850
2851
static void d68000_or_er_8(m68k_info *info)
2852
1.01k
{
2853
1.01k
  build_er_1(info, M68K_INS_OR, 1);
2854
1.01k
}
2855
2856
static void d68000_or_er_16(m68k_info *info)
2857
568
{
2858
568
  build_er_1(info, M68K_INS_OR, 2);
2859
568
}
2860
2861
static void d68000_or_er_32(m68k_info *info)
2862
1.99k
{
2863
1.99k
  build_er_1(info, M68K_INS_OR, 4);
2864
1.99k
}
2865
2866
static void d68000_or_re_8(m68k_info *info)
2867
523
{
2868
523
  build_re_1(info, M68K_INS_OR, 1);
2869
523
}
2870
2871
static void d68000_or_re_16(m68k_info *info)
2872
907
{
2873
907
  build_re_1(info, M68K_INS_OR, 2);
2874
907
}
2875
2876
static void d68000_or_re_32(m68k_info *info)
2877
1.10k
{
2878
1.10k
  build_re_1(info, M68K_INS_OR, 4);
2879
1.10k
}
2880
2881
static void d68000_ori_8(m68k_info *info)
2882
11.8k
{
2883
11.8k
  build_imm_ea(info, M68K_INS_ORI, 1, read_imm_8(info));
2884
11.8k
}
2885
2886
static void d68000_ori_16(m68k_info *info)
2887
1.15k
{
2888
1.15k
  build_imm_ea(info, M68K_INS_ORI, 2, read_imm_16(info));
2889
1.15k
}
2890
2891
static void d68000_ori_32(m68k_info *info)
2892
1.07k
{
2893
1.07k
  build_imm_ea(info, M68K_INS_ORI, 4, read_imm_32(info));
2894
1.07k
}
2895
2896
static void d68000_ori_to_ccr(m68k_info *info)
2897
515
{
2898
515
  build_imm_special_reg(info, M68K_INS_ORI, read_imm_8(info), 1, M68K_REG_CCR);
2899
515
}
2900
2901
static void d68000_ori_to_sr(m68k_info *info)
2902
308
{
2903
308
  build_imm_special_reg(info, M68K_INS_ORI, read_imm_16(info), 2, M68K_REG_SR);
2904
308
}
2905
2906
static void d68020_pack_rr(m68k_info *info)
2907
742
{
2908
742
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2909
398
  build_rr(info, M68K_INS_PACK, 0, read_imm_16(info));
2910
398
}
2911
2912
static void d68020_pack_mm(m68k_info *info)
2913
1.06k
{
2914
1.06k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
2915
711
  build_mm(info, M68K_INS_PACK, 0, read_imm_16(info));
2916
711
}
2917
2918
static void d68000_pea(m68k_info *info)
2919
373
{
2920
373
  build_ea(info, M68K_INS_PEA, 4);
2921
373
}
2922
2923
static void d68000_reset(m68k_info *info)
2924
863
{
2925
863
  MCInst_setOpcode(info->inst, M68K_INS_RESET);
2926
863
}
2927
2928
static void d68000_ror_s_8(m68k_info *info)
2929
281
{
2930
281
  build_3bit_d(info, M68K_INS_ROR, 1);
2931
281
}
2932
2933
static void d68000_ror_s_16(m68k_info *info)
2934
470
{
2935
470
  build_3bit_d(info, M68K_INS_ROR, 2);
2936
470
}
2937
2938
static void d68000_ror_s_32(m68k_info *info)
2939
229
{
2940
229
  build_3bit_d(info, M68K_INS_ROR, 4);
2941
229
}
2942
2943
static void d68000_ror_r_8(m68k_info *info)
2944
287
{
2945
287
  build_r(info, M68K_INS_ROR, 1);
2946
287
}
2947
2948
static void d68000_ror_r_16(m68k_info *info)
2949
261
{
2950
261
  build_r(info, M68K_INS_ROR, 2);
2951
261
}
2952
2953
static void d68000_ror_r_32(m68k_info *info)
2954
265
{
2955
265
  build_r(info, M68K_INS_ROR, 4);
2956
265
}
2957
2958
static void d68000_ror_ea(m68k_info *info)
2959
247
{
2960
247
  build_ea(info, M68K_INS_ROR, 2);
2961
247
}
2962
2963
static void d68000_rol_s_8(m68k_info *info)
2964
441
{
2965
441
  build_3bit_d(info, M68K_INS_ROL, 1);
2966
441
}
2967
2968
static void d68000_rol_s_16(m68k_info *info)
2969
98
{
2970
98
  build_3bit_d(info, M68K_INS_ROL, 2);
2971
98
}
2972
2973
static void d68000_rol_s_32(m68k_info *info)
2974
216
{
2975
216
  build_3bit_d(info, M68K_INS_ROL, 4);
2976
216
}
2977
2978
static void d68000_rol_r_8(m68k_info *info)
2979
471
{
2980
471
  build_r(info, M68K_INS_ROL, 1);
2981
471
}
2982
2983
static void d68000_rol_r_16(m68k_info *info)
2984
138
{
2985
138
  build_r(info, M68K_INS_ROL, 2);
2986
138
}
2987
2988
static void d68000_rol_r_32(m68k_info *info)
2989
93
{
2990
93
  build_r(info, M68K_INS_ROL, 4);
2991
93
}
2992
2993
static void d68000_rol_ea(m68k_info *info)
2994
467
{
2995
467
  build_ea(info, M68K_INS_ROL, 2);
2996
467
}
2997
2998
static void d68000_roxr_s_8(m68k_info *info)
2999
656
{
3000
656
  build_3bit_d(info, M68K_INS_ROXR, 1);
3001
656
}
3002
3003
static void d68000_roxr_s_16(m68k_info *info)
3004
222
{
3005
222
  build_3bit_d(info, M68K_INS_ROXR, 2);
3006
222
}
3007
3008
static void d68000_roxr_s_32(m68k_info *info)
3009
90
{
3010
90
  build_3bit_d(info, M68K_INS_ROXR, 4);
3011
90
}
3012
3013
static void d68000_roxr_r_8(m68k_info *info)
3014
172
{
3015
172
  build_3bit_d(info, M68K_INS_ROXR, 4);
3016
172
}
3017
3018
static void d68000_roxr_r_16(m68k_info *info)
3019
91
{
3020
91
  build_r(info, M68K_INS_ROXR, 2);
3021
91
}
3022
3023
static void d68000_roxr_r_32(m68k_info *info)
3024
268
{
3025
268
  build_r(info, M68K_INS_ROXR, 4);
3026
268
}
3027
3028
static void d68000_roxr_ea(m68k_info *info)
3029
602
{
3030
602
  build_ea(info, M68K_INS_ROXR, 2);
3031
602
}
3032
3033
static void d68000_roxl_s_8(m68k_info *info)
3034
143
{
3035
143
  build_3bit_d(info, M68K_INS_ROXL, 1);
3036
143
}
3037
3038
static void d68000_roxl_s_16(m68k_info *info)
3039
119
{
3040
119
  build_3bit_d(info, M68K_INS_ROXL, 2);
3041
119
}
3042
3043
static void d68000_roxl_s_32(m68k_info *info)
3044
186
{
3045
186
  build_3bit_d(info, M68K_INS_ROXL, 4);
3046
186
}
3047
3048
static void d68000_roxl_r_8(m68k_info *info)
3049
372
{
3050
372
  build_r(info, M68K_INS_ROXL, 1);
3051
372
}
3052
3053
static void d68000_roxl_r_16(m68k_info *info)
3054
54
{
3055
54
  build_r(info, M68K_INS_ROXL, 2);
3056
54
}
3057
3058
static void d68000_roxl_r_32(m68k_info *info)
3059
93
{
3060
93
  build_r(info, M68K_INS_ROXL, 4);
3061
93
}
3062
3063
static void d68000_roxl_ea(m68k_info *info)
3064
447
{
3065
447
  build_ea(info, M68K_INS_ROXL, 2);
3066
447
}
3067
3068
static void d68010_rtd(m68k_info *info)
3069
797
{
3070
797
  set_insn_group(info, M68K_GRP_RET);
3071
797
  LIMIT_CPU_TYPES(info, M68010_PLUS);
3072
721
  build_absolute_jump_with_immediate(info, M68K_INS_RTD, 0, read_imm_16(info));
3073
721
}
3074
3075
static void d68000_rte(m68k_info *info)
3076
95
{
3077
95
  set_insn_group(info, M68K_GRP_IRET);
3078
95
  MCInst_setOpcode(info->inst, M68K_INS_RTE);
3079
95
}
3080
3081
static void d68020_rtm(m68k_info *info)
3082
231
{
3083
231
  cs_m68k* ext;
3084
231
  cs_m68k_op* op;
3085
3086
231
  set_insn_group(info, M68K_GRP_RET);
3087
3088
231
  LIMIT_CPU_TYPES(info, M68020_ONLY);
3089
3090
0
  build_absolute_jump_with_immediate(info, M68K_INS_RTM, 0, 0);
3091
3092
0
  ext = &info->extension;
3093
0
  op = &ext->operands[0];
3094
3095
0
  op->address_mode = M68K_AM_NONE;
3096
0
  op->type = M68K_OP_REG;
3097
3098
0
  if (BIT_3(info->ir)) {
3099
0
    op->reg = M68K_REG_A0 + (info->ir & 7);
3100
0
  } else {
3101
0
    op->reg = M68K_REG_D0 + (info->ir & 7);
3102
0
  }
3103
0
}
3104
3105
static void d68000_rtr(m68k_info *info)
3106
20
{
3107
20
  set_insn_group(info, M68K_GRP_RET);
3108
20
  MCInst_setOpcode(info->inst, M68K_INS_RTR);
3109
20
}
3110
3111
static void d68000_rts(m68k_info *info)
3112
95
{
3113
95
  set_insn_group(info, M68K_GRP_RET);
3114
95
  MCInst_setOpcode(info->inst, M68K_INS_RTS);
3115
95
}
3116
3117
static void d68000_sbcd_rr(m68k_info *info)
3118
417
{
3119
417
  build_rr(info, M68K_INS_SBCD, 1, 0);
3120
417
}
3121
3122
static void d68000_sbcd_mm(m68k_info *info)
3123
481
{
3124
481
  build_mm(info, M68K_INS_SBCD, 0, read_imm_16(info));
3125
481
}
3126
3127
static void d68000_scc(m68k_info *info)
3128
1.30k
{
3129
1.30k
  cs_m68k* ext = build_init_op(info, s_scc_lut[(info->ir >> 8) & 0xf], 1, 1);
3130
1.30k
  get_ea_mode_op(info, &ext->operands[0], info->ir, 1);
3131
1.30k
}
3132
3133
static void d68000_stop(m68k_info *info)
3134
62
{
3135
62
  build_absolute_jump_with_immediate(info, M68K_INS_STOP, 0, read_imm_16(info));
3136
62
}
3137
3138
static void d68000_sub_er_8(m68k_info *info)
3139
969
{
3140
969
  build_er_1(info, M68K_INS_SUB, 1);
3141
969
}
3142
3143
static void d68000_sub_er_16(m68k_info *info)
3144
1.23k
{
3145
1.23k
  build_er_1(info, M68K_INS_SUB, 2);
3146
1.23k
}
3147
3148
static void d68000_sub_er_32(m68k_info *info)
3149
2.22k
{
3150
2.22k
  build_er_1(info, M68K_INS_SUB, 4);
3151
2.22k
}
3152
3153
static void d68000_sub_re_8(m68k_info *info)
3154
630
{
3155
630
  build_re_1(info, M68K_INS_SUB, 1);
3156
630
}
3157
3158
static void d68000_sub_re_16(m68k_info *info)
3159
506
{
3160
506
  build_re_1(info, M68K_INS_SUB, 2);
3161
506
}
3162
3163
static void d68000_sub_re_32(m68k_info *info)
3164
3.01k
{
3165
3.01k
  build_re_1(info, M68K_INS_SUB, 4);
3166
3.01k
}
3167
3168
static void d68000_suba_16(m68k_info *info)
3169
548
{
3170
548
  build_ea_a(info, M68K_INS_SUBA, 2);
3171
548
}
3172
3173
static void d68000_suba_32(m68k_info *info)
3174
541
{
3175
541
  build_ea_a(info, M68K_INS_SUBA, 4);
3176
541
}
3177
3178
static void d68000_subi_8(m68k_info *info)
3179
1.49k
{
3180
1.49k
  build_imm_ea(info, M68K_INS_SUBI, 1, read_imm_8(info));
3181
1.49k
}
3182
3183
static void d68000_subi_16(m68k_info *info)
3184
399
{
3185
399
  build_imm_ea(info, M68K_INS_SUBI, 2, read_imm_16(info));
3186
399
}
3187
3188
static void d68000_subi_32(m68k_info *info)
3189
260
{
3190
260
  build_imm_ea(info, M68K_INS_SUBI, 4, read_imm_32(info));
3191
260
}
3192
3193
static void d68000_subq_8(m68k_info *info)
3194
1.35k
{
3195
1.35k
  build_3bit_ea(info, M68K_INS_SUBQ, 1);
3196
1.35k
}
3197
3198
static void d68000_subq_16(m68k_info *info)
3199
2.55k
{
3200
2.55k
  build_3bit_ea(info, M68K_INS_SUBQ, 2);
3201
2.55k
}
3202
3203
static void d68000_subq_32(m68k_info *info)
3204
623
{
3205
623
  build_3bit_ea(info, M68K_INS_SUBQ, 4);
3206
623
}
3207
3208
static void d68000_subx_rr_8(m68k_info *info)
3209
382
{
3210
382
  build_rr(info, M68K_INS_SUBX, 1, 0);
3211
382
}
3212
3213
static void d68000_subx_rr_16(m68k_info *info)
3214
321
{
3215
321
  build_rr(info, M68K_INS_SUBX, 2, 0);
3216
321
}
3217
3218
static void d68000_subx_rr_32(m68k_info *info)
3219
538
{
3220
538
  build_rr(info, M68K_INS_SUBX, 4, 0);
3221
538
}
3222
3223
static void d68000_subx_mm_8(m68k_info *info)
3224
383
{
3225
383
  build_mm(info, M68K_INS_SUBX, 1, 0);
3226
383
}
3227
3228
static void d68000_subx_mm_16(m68k_info *info)
3229
249
{
3230
249
  build_mm(info, M68K_INS_SUBX, 2, 0);
3231
249
}
3232
3233
static void d68000_subx_mm_32(m68k_info *info)
3234
186
{
3235
186
  build_mm(info, M68K_INS_SUBX, 4, 0);
3236
186
}
3237
3238
static void d68000_swap(m68k_info *info)
3239
43
{
3240
43
  build_d(info, M68K_INS_SWAP, 0);
3241
43
}
3242
3243
static void d68000_tas(m68k_info *info)
3244
329
{
3245
329
  build_ea(info, M68K_INS_TAS, 1);
3246
329
}
3247
3248
static void d68000_trap(m68k_info *info)
3249
562
{
3250
562
  build_absolute_jump_with_immediate(info, M68K_INS_TRAP, 0, info->ir&0xf);
3251
562
}
3252
3253
static void d68020_trapcc_0(m68k_info *info)
3254
512
{
3255
512
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3256
295
  build_trap(info, 0, 0);
3257
3258
295
  info->extension.op_count = 0;
3259
295
}
3260
3261
static void d68020_trapcc_16(m68k_info *info)
3262
542
{
3263
542
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3264
328
  build_trap(info, 2, read_imm_16(info));
3265
328
}
3266
3267
static void d68020_trapcc_32(m68k_info *info)
3268
150
{
3269
150
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3270
76
  build_trap(info, 4, read_imm_32(info));
3271
76
}
3272
3273
static void d68000_trapv(m68k_info *info)
3274
195
{
3275
195
  MCInst_setOpcode(info->inst, M68K_INS_TRAPV);
3276
195
}
3277
3278
static void d68000_tst_8(m68k_info *info)
3279
315
{
3280
315
  build_ea(info, M68K_INS_TST, 1);
3281
315
}
3282
3283
static void d68020_tst_pcdi_8(m68k_info *info)
3284
209
{
3285
209
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3286
73
  build_ea(info, M68K_INS_TST, 1);
3287
73
}
3288
3289
static void d68020_tst_pcix_8(m68k_info *info)
3290
146
{
3291
146
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3292
95
  build_ea(info, M68K_INS_TST, 1);
3293
95
}
3294
3295
static void d68020_tst_i_8(m68k_info *info)
3296
171
{
3297
171
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3298
101
  build_ea(info, M68K_INS_TST, 1);
3299
101
}
3300
3301
static void d68000_tst_16(m68k_info *info)
3302
327
{
3303
327
  build_ea(info, M68K_INS_TST, 2);
3304
327
}
3305
3306
static void d68020_tst_a_16(m68k_info *info)
3307
1.09k
{
3308
1.09k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3309
474
  build_ea(info, M68K_INS_TST, 2);
3310
474
}
3311
3312
static void d68020_tst_pcdi_16(m68k_info *info)
3313
677
{
3314
677
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3315
77
  build_ea(info, M68K_INS_TST, 2);
3316
77
}
3317
3318
static void d68020_tst_pcix_16(m68k_info *info)
3319
192
{
3320
192
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3321
123
  build_ea(info, M68K_INS_TST, 2);
3322
123
}
3323
3324
static void d68020_tst_i_16(m68k_info *info)
3325
248
{
3326
248
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3327
180
  build_ea(info, M68K_INS_TST, 2);
3328
180
}
3329
3330
static void d68000_tst_32(m68k_info *info)
3331
507
{
3332
507
  build_ea(info, M68K_INS_TST, 4);
3333
507
}
3334
3335
static void d68020_tst_a_32(m68k_info *info)
3336
591
{
3337
591
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3338
357
  build_ea(info, M68K_INS_TST, 4);
3339
357
}
3340
3341
static void d68020_tst_pcdi_32(m68k_info *info)
3342
278
{
3343
278
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3344
69
  build_ea(info, M68K_INS_TST, 4);
3345
69
}
3346
3347
static void d68020_tst_pcix_32(m68k_info *info)
3348
177
{
3349
177
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3350
81
  build_ea(info, M68K_INS_TST, 4);
3351
81
}
3352
3353
static void d68020_tst_i_32(m68k_info *info)
3354
407
{
3355
407
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3356
198
  build_ea(info, M68K_INS_TST, 4);
3357
198
}
3358
3359
static void d68000_unlk(m68k_info *info)
3360
92
{
3361
92
  cs_m68k_op* op;
3362
92
  cs_m68k* ext = build_init_op(info, M68K_INS_UNLK, 1, 0);
3363
3364
92
  op = &ext->operands[0];
3365
3366
92
  op->address_mode = M68K_AM_REG_DIRECT_ADDR;
3367
92
  op->reg = M68K_REG_A0 + (info->ir & 7);
3368
92
}
3369
3370
static void d68020_unpk_rr(m68k_info *info)
3371
1.74k
{
3372
1.74k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3373
1.32k
  build_rr(info, M68K_INS_UNPK, 0, read_imm_16(info));
3374
1.32k
}
3375
3376
static void d68020_unpk_mm(m68k_info *info)
3377
1.32k
{
3378
1.32k
  LIMIT_CPU_TYPES(info, M68020_PLUS);
3379
901
  build_mm(info, M68K_INS_UNPK, 0, read_imm_16(info));
3380
901
}
3381
3382
/* This table is auto-generated. Look in contrib/m68k_instruction_tbl_gen for more info */
3383
#include "M68KInstructionTable.inc"
3384
3385
static int instruction_is_valid(m68k_info *info, const unsigned int word_check)
3386
250k
{
3387
250k
  const unsigned int instruction = info->ir;
3388
250k
  const instruction_struct *i = &g_instruction_table[instruction];
3389
3390
250k
  if ( (i->word2_mask && ((word_check & i->word2_mask) != i->word2_match)) ||
3391
250k
    (i->instruction == d68000_invalid) ) {
3392
1.93k
    d68000_invalid(info);
3393
1.93k
    return 0;
3394
1.93k
  }
3395
3396
248k
  return 1;
3397
250k
}
3398
3399
static int exists_reg_list(uint16_t *regs, uint8_t count, m68k_reg reg)
3400
323k
{
3401
323k
  uint8_t i;
3402
3403
451k
  for (i = 0; i < count; ++i) {
3404
133k
    if (regs[i] == (uint16_t)reg)
3405
5.39k
      return 1;
3406
133k
  }
3407
3408
318k
  return 0;
3409
323k
}
3410
3411
static void add_reg_to_rw_list(m68k_info *info, m68k_reg reg, int write)
3412
349k
{
3413
349k
  if (reg == M68K_REG_INVALID)
3414
26.0k
    return;
3415
3416
323k
  if (write)
3417
190k
  {
3418
190k
    if (exists_reg_list(info->regs_write, info->regs_write_count, reg))
3419
2.87k
      return;
3420
3421
188k
    info->regs_write[info->regs_write_count] = (uint16_t)reg;
3422
188k
    info->regs_write_count++;
3423
188k
  }
3424
132k
  else
3425
132k
  {
3426
132k
    if (exists_reg_list(info->regs_read, info->regs_read_count, reg))
3427
2.52k
      return;
3428
3429
130k
    info->regs_read[info->regs_read_count] = (uint16_t)reg;
3430
130k
    info->regs_read_count++;
3431
130k
  }
3432
323k
}
3433
3434
static void update_am_reg_list(m68k_info *info, cs_m68k_op *op, int write)
3435
112k
{
3436
112k
  switch (op->address_mode) {
3437
777
    case M68K_AM_REG_DIRECT_ADDR:
3438
777
    case M68K_AM_REG_DIRECT_DATA:
3439
777
      add_reg_to_rw_list(info, op->reg, write);
3440
777
      break;
3441
3442
22.3k
    case M68K_AM_REGI_ADDR_POST_INC:
3443
52.9k
    case M68K_AM_REGI_ADDR_PRE_DEC:
3444
52.9k
      add_reg_to_rw_list(info, op->reg, 1);
3445
52.9k
      break;
3446
3447
17.5k
    case M68K_AM_REGI_ADDR:
3448
34.3k
    case M68K_AM_REGI_ADDR_DISP:
3449
34.3k
      add_reg_to_rw_list(info, op->reg, 0);
3450
34.3k
      break;
3451
3452
9.07k
    case M68K_AM_AREGI_INDEX_8_BIT_DISP:
3453
12.1k
    case M68K_AM_AREGI_INDEX_BASE_DISP:
3454
14.8k
    case M68K_AM_MEMI_POST_INDEX:
3455
17.0k
    case M68K_AM_MEMI_PRE_INDEX:
3456
17.8k
    case M68K_AM_PCI_INDEX_8_BIT_DISP:
3457
17.9k
    case M68K_AM_PCI_INDEX_BASE_DISP:
3458
18.6k
    case M68K_AM_PC_MEMI_PRE_INDEX:
3459
18.8k
    case M68K_AM_PC_MEMI_POST_INDEX:
3460
18.8k
      add_reg_to_rw_list(info, op->mem.index_reg, 0);
3461
18.8k
      add_reg_to_rw_list(info, op->mem.base_reg, 0);
3462
18.8k
      break;
3463
3464
    // no register(s) in the other addressing modes
3465
5.24k
    default:
3466
5.24k
      break;
3467
112k
  }
3468
112k
}
3469
3470
static void update_bits_range(m68k_info *info, m68k_reg reg_start, uint8_t bits, int write)
3471
15.4k
{
3472
15.4k
  int i;
3473
3474
139k
  for (i = 0; i < 8; ++i) {
3475
123k
    if (bits & (1 << i)) {
3476
26.5k
      add_reg_to_rw_list(info, reg_start + i, write);
3477
26.5k
    }
3478
123k
  }
3479
15.4k
}
3480
3481
static void update_reg_list_regbits(m68k_info *info, cs_m68k_op *op, int write)
3482
5.15k
{
3483
5.15k
  uint32_t bits = op->register_bits;
3484
5.15k
  update_bits_range(info, M68K_REG_D0, bits & 0xff, write);
3485
5.15k
  update_bits_range(info, M68K_REG_A0, (bits >> 8) & 0xff, write);
3486
5.15k
  update_bits_range(info, M68K_REG_FP0, (bits >> 16) & 0xff, write);
3487
5.15k
}
3488
3489
static void update_op_reg_list(m68k_info *info, cs_m68k_op *op, int write)
3490
420k
{
3491
420k
  switch ((int)op->type) {
3492
191k
    case M68K_OP_REG:
3493
191k
      add_reg_to_rw_list(info, op->reg, write);
3494
191k
      break;
3495
3496
112k
    case M68K_OP_MEM:
3497
112k
      update_am_reg_list(info, op, write);
3498
112k
      break;
3499
3500
5.15k
    case M68K_OP_REG_BITS:
3501
5.15k
      update_reg_list_regbits(info, op, write);
3502
5.15k
      break;
3503
3504
2.86k
    case M68K_OP_REG_PAIR:
3505
2.86k
      add_reg_to_rw_list(info, op->reg_pair.reg_0, write);
3506
2.86k
      add_reg_to_rw_list(info, op->reg_pair.reg_1, write);
3507
2.86k
      break;
3508
420k
  }
3509
420k
}
3510
3511
static void build_regs_read_write_counts(m68k_info *info)
3512
247k
{
3513
247k
  int i;
3514
3515
247k
  if (!info->extension.op_count)
3516
1.93k
    return;
3517
3518
245k
  if (info->extension.op_count == 1) {
3519
74.8k
    update_op_reg_list(info, &info->extension.operands[0], 1);
3520
170k
  } else {
3521
    // first operand is always read
3522
170k
    update_op_reg_list(info, &info->extension.operands[0], 0);
3523
3524
    // remaining write
3525
345k
    for (i = 1; i < info->extension.op_count; ++i)
3526
174k
      update_op_reg_list(info, &info->extension.operands[i], 1);
3527
170k
  }
3528
245k
}
3529
3530
static void m68k_setup_internals(m68k_info* info, MCInst* inst, unsigned int pc, unsigned int cpu_type)
3531
248k
{
3532
248k
  info->inst = inst;
3533
248k
  info->pc = pc;
3534
248k
  info->ir = 0;
3535
248k
  info->type = cpu_type;
3536
248k
  info->address_mask = 0xffffffff;
3537
3538
248k
  switch(info->type) {
3539
75.4k
    case M68K_CPU_TYPE_68000:
3540
75.4k
      info->type = TYPE_68000;
3541
75.4k
      info->address_mask = 0x00ffffff;
3542
75.4k
      break;
3543
0
    case M68K_CPU_TYPE_68010:
3544
0
      info->type = TYPE_68010;
3545
0
      info->address_mask = 0x00ffffff;
3546
0
      break;
3547
0
    case M68K_CPU_TYPE_68EC020:
3548
0
      info->type = TYPE_68020;
3549
0
      info->address_mask = 0x00ffffff;
3550
0
      break;
3551
0
    case M68K_CPU_TYPE_68020:
3552
0
      info->type = TYPE_68020;
3553
0
      info->address_mask = 0xffffffff;
3554
0
      break;
3555
0
    case M68K_CPU_TYPE_68030:
3556
0
      info->type = TYPE_68030;
3557
0
      info->address_mask = 0xffffffff;
3558
0
      break;
3559
172k
    case M68K_CPU_TYPE_68040:
3560
172k
      info->type = TYPE_68040;
3561
172k
      info->address_mask = 0xffffffff;
3562
172k
      break;
3563
0
    default:
3564
0
      info->address_mask = 0;
3565
0
      return;
3566
248k
  }
3567
248k
}
3568
3569
/* ======================================================================== */
3570
/* ================================= API ================================== */
3571
/* ======================================================================== */
3572
3573
/* Disasemble one instruction at pc and store in str_buff */
3574
static unsigned int m68k_disassemble(m68k_info *info, uint64_t pc)
3575
248k
{
3576
248k
  MCInst *inst = info->inst;
3577
248k
  cs_m68k* ext = &info->extension;
3578
248k
  int i;
3579
248k
  unsigned int size;
3580
3581
248k
  inst->Opcode = M68K_INS_INVALID;
3582
3583
248k
  memset(ext, 0, sizeof(cs_m68k));
3584
248k
  ext->op_size.type = M68K_SIZE_TYPE_CPU;
3585
3586
1.24M
  for (i = 0; i < M68K_OPERAND_COUNT; ++i)
3587
992k
    ext->operands[i].type = M68K_OP_REG;
3588
3589
248k
  info->ir = peek_imm_16(info);
3590
248k
  if (instruction_is_valid(info, peek_imm_32(info) & 0xffff)) {
3591
247k
    info->ir = read_imm_16(info);
3592
247k
    g_instruction_table[info->ir].instruction(info);
3593
247k
  }
3594
3595
248k
  size = info->pc - (unsigned int)pc;
3596
248k
  info->pc = (unsigned int)pc;
3597
3598
248k
  return size;
3599
248k
}
3600
3601
bool M68K_getInstruction(csh ud, const uint8_t* code, size_t code_len, MCInst* instr, uint16_t* size, uint64_t address, void* inst_info)
3602
249k
{
3603
#ifdef M68K_DEBUG
3604
  SStream ss;
3605
#endif
3606
249k
  int s;
3607
249k
  int cpu_type = M68K_CPU_TYPE_68000;
3608
249k
  cs_struct* handle = instr->csh;
3609
249k
  m68k_info *info = (m68k_info*)handle->printer_info;
3610
3611
  // code len has to be at least 2 bytes to be valid m68k
3612
3613
249k
  if (code_len < 2) {
3614
941
    *size = 0;
3615
941
    return false;
3616
941
  }
3617
3618
248k
  if (instr->flat_insn->detail) {
3619
248k
    memset(instr->flat_insn->detail, 0, offsetof(cs_detail, m68k)+sizeof(cs_m68k));
3620
248k
  }
3621
3622
248k
  info->groups_count = 0;
3623
248k
  info->regs_read_count = 0;
3624
248k
  info->regs_write_count = 0;
3625
248k
  info->code = code;
3626
248k
  info->code_len = code_len;
3627
248k
  info->baseAddress = address;
3628
3629
248k
  if (handle->mode & CS_MODE_M68K_010)
3630
0
    cpu_type = M68K_CPU_TYPE_68010;
3631
248k
  if (handle->mode & CS_MODE_M68K_020)
3632
0
    cpu_type = M68K_CPU_TYPE_68020;
3633
248k
  if (handle->mode & CS_MODE_M68K_030)
3634
0
    cpu_type = M68K_CPU_TYPE_68030;
3635
248k
  if (handle->mode & CS_MODE_M68K_040)
3636
172k
    cpu_type = M68K_CPU_TYPE_68040;
3637
248k
  if (handle->mode & CS_MODE_M68K_060)
3638
0
    cpu_type = M68K_CPU_TYPE_68040; // 060 = 040 for now
3639
3640
248k
  m68k_setup_internals(info, instr, (unsigned int)address, cpu_type);
3641
248k
  s = m68k_disassemble(info, address);
3642
3643
248k
  if (s == 0) {
3644
936
    *size = 2;
3645
936
    return false;
3646
936
  }
3647
3648
247k
  build_regs_read_write_counts(info);
3649
3650
#ifdef M68K_DEBUG
3651
  SStream_Init(&ss);
3652
  M68K_printInst(instr, &ss, info);
3653
#endif
3654
3655
  // Make sure we always stay within range
3656
247k
  if (s > (int)code_len)
3657
1.01k
    *size = (uint16_t)code_len;
3658
246k
  else
3659
246k
    *size = (uint16_t)s;
3660
3661
247k
  return true;
3662
248k
}
3663