Coverage Report

Created: 2025-07-01 07:03

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