Coverage Report

Created: 2025-07-18 06:43

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