Coverage Report

Created: 2025-07-11 06:32

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