Coverage Report

Created: 2025-07-11 06:32

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