Coverage Report

Created: 2025-08-26 06:30

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