Coverage Report

Created: 2025-07-04 06:11

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