Coverage Report

Created: 2026-03-13 06:50

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonenext/arch/MOS65XX/MOS65XXDisassembler.c
Line
Count
Source
1
/* Capstone Disassembly Engine */
2
/* MOS65XX Backend by Sebastian Macke <sebastian@macke.de> 2018 */
3
4
#include "capstone/mos65xx.h"
5
#include "MOS65XXDisassembler.h"
6
#include "MOS65XXDisassemblerInternals.h"
7
8
typedef struct OpInfo {
9
  mos65xx_insn ins;
10
  mos65xx_address_mode am;
11
  int operand_bytes;
12
} OpInfo;
13
14
static const struct OpInfo OpInfoTable[] = {
15
16
#include "m6502.inc"
17
#include "m65c02.inc"
18
#include "mw65c02.inc"
19
#include "m65816.inc"
20
21
};
22
23
#ifndef CAPSTONE_DIET
24
static const char *const RegNames[] = { "invalid", "A",  "X", "Y", "P",
25
          "SP",    "DP", "B", "K" };
26
27
static const char *const GroupNames[] = {
28
  NULL, "jump", "call", "ret", "int", "iret", "branch_relative"
29
};
30
31
typedef struct InstructionInfo {
32
  const char *name;
33
  mos65xx_group_type group_type;
34
  mos65xx_reg write, read;
35
  bool modifies_status;
36
} InstructionInfo;
37
38
static const struct InstructionInfo InstructionInfoTable[] = {
39
40
#include "instruction_info.inc"
41
42
};
43
#endif
44
45
#ifndef CAPSTONE_DIET
46
static void fillDetails(MCInst *MI, struct OpInfo opinfo, int cpu_type)
47
14.8k
{
48
14.8k
  int i;
49
14.8k
  cs_detail *detail = MI->flat_insn->detail;
50
51
14.8k
  InstructionInfo insinfo = InstructionInfoTable[opinfo.ins];
52
53
14.8k
  detail->mos65xx.am = opinfo.am;
54
14.8k
  detail->mos65xx.modifies_flags = insinfo.modifies_status;
55
14.8k
  detail->groups_count = 0;
56
14.8k
  detail->regs_read_count = 0;
57
14.8k
  detail->regs_write_count = 0;
58
14.8k
  detail->mos65xx.op_count = 0;
59
60
14.8k
  if (insinfo.group_type != MOS65XX_GRP_INVALID) {
61
3.12k
    detail->groups[detail->groups_count] = insinfo.group_type;
62
3.12k
    detail->groups_count++;
63
3.12k
  }
64
65
14.8k
  if (opinfo.am == MOS65XX_AM_REL || opinfo.am == MOS65XX_AM_ZP_REL) {
66
503
    detail->groups[detail->groups_count] =
67
503
      MOS65XX_GRP_BRANCH_RELATIVE;
68
503
    detail->groups_count++;
69
503
  }
70
71
14.8k
  if (insinfo.read != MOS65XX_REG_INVALID) {
72
5.30k
    detail->regs_read[detail->regs_read_count++] = insinfo.read;
73
5.30k
  } else
74
9.59k
    switch (opinfo.am) {
75
898
    case MOS65XX_AM_ACC:
76
898
      detail->regs_read[detail->regs_read_count++] =
77
898
        MOS65XX_REG_ACC;
78
898
      break;
79
389
    case MOS65XX_AM_ZP_Y:
80
927
    case MOS65XX_AM_ZP_IND_Y:
81
1.65k
    case MOS65XX_AM_ABS_Y:
82
1.65k
    case MOS65XX_AM_ZP_IND_LONG_Y:
83
1.65k
      detail->regs_read[detail->regs_read_count++] =
84
1.65k
        MOS65XX_REG_Y;
85
1.65k
      break;
86
87
847
    case MOS65XX_AM_ZP_X:
88
1.57k
    case MOS65XX_AM_ZP_X_IND:
89
2.10k
    case MOS65XX_AM_ABS_X:
90
2.10k
    case MOS65XX_AM_ABS_X_IND:
91
2.10k
    case MOS65XX_AM_ABS_LONG_X:
92
2.10k
      detail->regs_read[detail->regs_read_count++] =
93
2.10k
        MOS65XX_REG_X;
94
2.10k
      break;
95
96
0
    case MOS65XX_AM_SR:
97
0
      detail->regs_read[detail->regs_read_count++] =
98
0
        MOS65XX_REG_SP;
99
0
      break;
100
0
    case MOS65XX_AM_SR_IND_Y:
101
0
      detail->regs_read[detail->regs_read_count++] =
102
0
        MOS65XX_REG_SP;
103
0
      detail->regs_read[detail->regs_read_count++] =
104
0
        MOS65XX_REG_Y;
105
0
      break;
106
107
4.93k
    default:
108
4.93k
      break;
109
9.59k
    }
110
111
14.8k
  if (insinfo.write != MOS65XX_REG_INVALID) {
112
5.01k
    detail->regs_write[detail->regs_write_count++] = insinfo.write;
113
9.88k
  } else if (opinfo.am == MOS65XX_AM_ACC) {
114
898
    detail->regs_write[detail->regs_write_count++] =
115
898
      MOS65XX_REG_ACC;
116
898
  }
117
118
14.8k
  switch (opinfo.ins) {
119
542
  case MOS65XX_INS_ADC:
120
1.07k
  case MOS65XX_INS_SBC:
121
1.89k
  case MOS65XX_INS_ROL:
122
2.43k
  case MOS65XX_INS_ROR:
123
    /* these read carry flag (and decimal for ADC/SBC) */
124
2.43k
    detail->regs_read[detail->regs_read_count++] = MOS65XX_REG_P;
125
2.43k
    break;
126
  /* stack operations */
127
0
  case MOS65XX_INS_JSL:
128
347
  case MOS65XX_INS_JSR:
129
347
  case MOS65XX_INS_PEA:
130
347
  case MOS65XX_INS_PEI:
131
347
  case MOS65XX_INS_PER:
132
2.11k
  case MOS65XX_INS_PHA:
133
2.11k
  case MOS65XX_INS_PHB:
134
2.11k
  case MOS65XX_INS_PHD:
135
2.11k
  case MOS65XX_INS_PHK:
136
2.77k
  case MOS65XX_INS_PHP:
137
2.77k
  case MOS65XX_INS_PHX:
138
2.77k
  case MOS65XX_INS_PHY:
139
3.04k
  case MOS65XX_INS_PLA:
140
3.04k
  case MOS65XX_INS_PLB:
141
3.04k
  case MOS65XX_INS_PLD:
142
3.55k
  case MOS65XX_INS_PLP:
143
3.55k
  case MOS65XX_INS_PLX:
144
3.55k
  case MOS65XX_INS_PLY:
145
3.82k
  case MOS65XX_INS_RTI:
146
3.82k
  case MOS65XX_INS_RTL:
147
4.29k
  case MOS65XX_INS_RTS:
148
4.29k
    detail->regs_read[detail->regs_read_count++] = MOS65XX_REG_SP;
149
4.29k
    detail->regs_write[detail->regs_write_count++] = MOS65XX_REG_SP;
150
4.29k
    break;
151
8.16k
  default:
152
8.16k
    break;
153
14.8k
  }
154
155
14.8k
  if (cpu_type == MOS65XX_CPU_TYPE_65816) {
156
0
    switch (opinfo.am) {
157
0
    case MOS65XX_AM_ZP:
158
0
    case MOS65XX_AM_ZP_X:
159
0
    case MOS65XX_AM_ZP_Y:
160
0
    case MOS65XX_AM_ZP_IND:
161
0
    case MOS65XX_AM_ZP_X_IND:
162
0
    case MOS65XX_AM_ZP_IND_Y:
163
0
    case MOS65XX_AM_ZP_IND_LONG:
164
0
    case MOS65XX_AM_ZP_IND_LONG_Y:
165
0
      detail->regs_read[detail->regs_read_count++] =
166
0
        MOS65XX_REG_DP;
167
0
      break;
168
0
    case MOS65XX_AM_BLOCK:
169
0
      detail->regs_read[detail->regs_read_count++] =
170
0
        MOS65XX_REG_ACC;
171
0
      detail->regs_read[detail->regs_read_count++] =
172
0
        MOS65XX_REG_X;
173
0
      detail->regs_read[detail->regs_read_count++] =
174
0
        MOS65XX_REG_Y;
175
0
      detail->regs_write[detail->regs_write_count++] =
176
0
        MOS65XX_REG_ACC;
177
0
      detail->regs_write[detail->regs_write_count++] =
178
0
        MOS65XX_REG_X;
179
0
      detail->regs_write[detail->regs_write_count++] =
180
0
        MOS65XX_REG_Y;
181
0
      detail->regs_write[detail->regs_write_count++] =
182
0
        MOS65XX_REG_B;
183
0
      break;
184
0
    default:
185
0
      break;
186
0
    }
187
188
0
    switch (opinfo.am) {
189
0
    case MOS65XX_AM_ZP_IND:
190
0
    case MOS65XX_AM_ZP_X_IND:
191
0
    case MOS65XX_AM_ZP_IND_Y:
192
0
    case MOS65XX_AM_ABS:
193
0
    case MOS65XX_AM_ABS_X:
194
0
    case MOS65XX_AM_ABS_Y:
195
0
    case MOS65XX_AM_ABS_X_IND:
196
      /* these depend on the databank to generate a 24-bit address */
197
      /* exceptions: PEA, PEI, and JMP (abs) */
198
0
      if (opinfo.ins == MOS65XX_INS_PEI ||
199
0
          opinfo.ins == MOS65XX_INS_PEA)
200
0
        break;
201
0
      detail->regs_read[detail->regs_read_count++] =
202
0
        MOS65XX_REG_B;
203
0
      break;
204
0
    default:
205
0
      break;
206
0
    }
207
0
  }
208
209
14.8k
  if (insinfo.modifies_status) {
210
8.59k
    detail->regs_write[detail->regs_write_count++] = MOS65XX_REG_P;
211
8.59k
  }
212
213
14.8k
  switch (opinfo.am) {
214
5.12k
  case MOS65XX_AM_IMP:
215
5.12k
    break;
216
439
  case MOS65XX_AM_IMM:
217
439
    detail->mos65xx.operands[detail->mos65xx.op_count].type =
218
439
      MOS65XX_OP_IMM;
219
439
    detail->mos65xx.operands[detail->mos65xx.op_count].imm =
220
439
      MI->Operands[0].ImmVal;
221
439
    detail->mos65xx.op_count++;
222
439
    break;
223
898
  case MOS65XX_AM_ACC:
224
898
    detail->mos65xx.operands[detail->mos65xx.op_count].type =
225
898
      MOS65XX_OP_REG;
226
898
    detail->mos65xx.operands[detail->mos65xx.op_count].reg =
227
898
      MOS65XX_REG_ACC;
228
898
    detail->mos65xx.op_count++;
229
898
    break;
230
503
  case MOS65XX_AM_REL: {
231
503
    int value = MI->Operands[0].ImmVal;
232
503
    if (MI->op1_size == 1)
233
503
      value = 2 + (signed char)value;
234
0
    else
235
0
      value = 3 + (signed short)value;
236
503
    detail->mos65xx.operands[detail->mos65xx.op_count].type =
237
503
      MOS65XX_OP_MEM;
238
503
    detail->mos65xx.operands[detail->mos65xx.op_count].mem =
239
503
      (MI->address + value) & 0xffff;
240
503
    detail->mos65xx.op_count++;
241
503
    break;
242
0
  }
243
0
  case MOS65XX_AM_ZP_REL: {
244
0
    int value = 3 + (signed char)MI->Operands[1].ImmVal;
245
    /* BBR0, zp, rel  and BBS0, zp, rel */
246
0
    detail->mos65xx.operands[detail->mos65xx.op_count].type =
247
0
      MOS65XX_OP_MEM;
248
0
    detail->mos65xx.operands[detail->mos65xx.op_count].mem =
249
0
      MI->Operands[0].ImmVal;
250
0
    detail->mos65xx.operands[detail->mos65xx.op_count + 1].type =
251
0
      MOS65XX_OP_MEM;
252
0
    detail->mos65xx.operands[detail->mos65xx.op_count + 1].mem =
253
0
      (MI->address + value) & 0xffff;
254
0
    detail->mos65xx.op_count += 2;
255
0
    break;
256
0
  }
257
7.93k
  default:
258
15.8k
    for (i = 0; i < MI->size; ++i) {
259
7.93k
      detail->mos65xx.operands[detail->mos65xx.op_count].type =
260
7.93k
        MOS65XX_OP_MEM;
261
7.93k
      detail->mos65xx.operands[detail->mos65xx.op_count].mem =
262
7.93k
        MI->Operands[i].ImmVal;
263
7.93k
      detail->mos65xx.op_count++;
264
7.93k
    }
265
7.93k
    break;
266
14.8k
  }
267
14.8k
}
268
#endif
269
270
void MOS65XX_printInst(MCInst *MI, struct SStream *O, void *PrinterInfo)
271
14.8k
{
272
14.8k
#ifndef CAPSTONE_DIET
273
14.8k
  unsigned int value;
274
14.8k
  unsigned opcode = MCInst_getOpcode(MI);
275
14.8k
  mos65xx_info *info = (mos65xx_info *)PrinterInfo;
276
277
14.8k
  OpInfo opinfo = OpInfoTable[opcode];
278
279
14.8k
  const char *prefix = info->hex_prefix ? info->hex_prefix : "0x";
280
281
14.8k
  SStream_concat0(O, InstructionInfoTable[opinfo.ins].name);
282
14.8k
  switch (opinfo.ins) {
283
  /* special case - bit included as part of the instruction name */
284
0
  case MOS65XX_INS_BBR:
285
0
  case MOS65XX_INS_BBS:
286
0
  case MOS65XX_INS_RMB:
287
0
  case MOS65XX_INS_SMB:
288
0
    SStream_concat(O, "%" PRId8, (opcode >> 4) & 0x07);
289
0
    break;
290
14.8k
  default:
291
14.8k
    break;
292
14.8k
  }
293
294
14.8k
  value = MI->Operands[0].ImmVal;
295
296
14.8k
  switch (opinfo.am) {
297
0
  default:
298
0
    break;
299
300
5.12k
  case MOS65XX_AM_IMP:
301
5.12k
    break;
302
303
898
  case MOS65XX_AM_ACC:
304
898
    SStream_concat0(O, " a");
305
898
    break;
306
307
439
  case MOS65XX_AM_IMM:
308
439
    if (MI->imm_size == 1)
309
439
      SStream_concat(O, " #%s%02" PRIx32, prefix, value);
310
0
    else
311
0
      SStream_concat(O, " #%s%04" PRIx32, prefix, value);
312
439
    break;
313
314
736
  case MOS65XX_AM_ZP:
315
736
    SStream_concat(O, " %s%02" PRIx32, prefix, value);
316
736
    break;
317
318
740
  case MOS65XX_AM_ABS:
319
740
    SStream_concat(O, " %s%04" PRIx32, prefix, value);
320
740
    break;
321
322
0
  case MOS65XX_AM_ABS_LONG_X:
323
0
    SStream_concat(O, " %s%06" PRIx32 ", x", prefix, value);
324
0
    break;
325
326
773
  case MOS65XX_AM_INT:
327
773
    SStream_concat(O, " %s%02" PRIx32, prefix, value);
328
773
    break;
329
330
624
  case MOS65XX_AM_ABS_X:
331
624
    SStream_concat(O, " %s%04" PRIx32 ", x", prefix, value);
332
624
    break;
333
334
1.07k
  case MOS65XX_AM_ABS_Y:
335
1.07k
    SStream_concat(O, " %s%04" PRIx32 ", y", prefix, value);
336
1.07k
    break;
337
338
0
  case MOS65XX_AM_ABS_LONG:
339
0
    SStream_concat(O, " %s%06" PRIx32, prefix, value);
340
0
    break;
341
342
1.10k
  case MOS65XX_AM_ZP_X:
343
1.10k
    SStream_concat(O, " %s%02" PRIx32 ", x", prefix, value);
344
1.10k
    break;
345
346
424
  case MOS65XX_AM_ZP_Y:
347
424
    SStream_concat(O, " %s%02" PRIx32 ", y", prefix, value);
348
424
    break;
349
350
503
  case MOS65XX_AM_REL: {
351
503
    if (MI->op1_size == 1)
352
503
      value = 2 + (int8_t)value;
353
0
    else
354
0
      value = 3 + (int16_t)value;
355
356
503
    uint32_t addr = MI->address;
357
503
    SStream_concat(O, " %s%04" PRIx16, prefix,
358
503
             (addr + value) & 0xffff);
359
360
503
    break;
361
0
  }
362
758
  case MOS65XX_AM_ABS_IND:
363
758
    SStream_concat(O, " (%s%04" PRIx32 ")", prefix, value);
364
758
    break;
365
366
0
  case MOS65XX_AM_ABS_X_IND:
367
0
    SStream_concat(O, " (%s%04" PRIx32 ", x)", prefix, value);
368
0
    break;
369
370
0
  case MOS65XX_AM_ABS_IND_LONG:
371
0
    SStream_concat(O, " [%s%04" PRIx32 "]", prefix, value);
372
0
    break;
373
374
0
  case MOS65XX_AM_ZP_IND:
375
0
    SStream_concat(O, " (%s%02" PRIx32 ")", prefix, value);
376
0
    break;
377
378
865
  case MOS65XX_AM_ZP_X_IND:
379
865
    SStream_concat(O, " (%s%02" PRIx32 ", x)", prefix, value);
380
865
    break;
381
382
832
  case MOS65XX_AM_ZP_IND_Y:
383
832
    SStream_concat(O, " (%s%02" PRIx32 "), y", prefix, value);
384
832
    break;
385
386
0
  case MOS65XX_AM_ZP_IND_LONG:
387
0
    SStream_concat(O, " [%s%02" PRIx32 "]", prefix, value);
388
0
    break;
389
390
0
  case MOS65XX_AM_ZP_IND_LONG_Y:
391
0
    SStream_concat(O, " [%s%02" PRIx32 "], y", prefix, value);
392
0
    break;
393
394
0
  case MOS65XX_AM_SR:
395
0
    SStream_concat(O, " %s%02" PRIx32 ", s", prefix, value);
396
0
    break;
397
398
0
  case MOS65XX_AM_SR_IND_Y:
399
0
    SStream_concat(O, " (%s%02" PRIx32 ", s), y", prefix, value);
400
0
    break;
401
402
0
  case MOS65XX_AM_BLOCK:
403
0
    SStream_concat(O, " %s%02" PRIx32 ", %s%02" PRIx32, prefix,
404
0
             (uint32_t)MI->Operands[0].ImmVal, prefix,
405
0
             (uint32_t)MI->Operands[1].ImmVal);
406
0
    break;
407
408
0
  case MOS65XX_AM_ZP_REL: {
409
0
    value = 3 + (int8_t)MI->Operands[1].ImmVal;
410
0
    uint32_t addr = MI->address;
411
0
    uint32_t target = (addr + value) & 0xffff;
412
    /* BBR0, zp, rel  and BBS0, zp, rel */
413
0
    SStream_concat(O, " %s%02" PRIx32 ", %s%04" PRIx32, prefix,
414
0
             (uint32_t)MI->Operands[0].ImmVal, prefix,
415
0
             target);
416
0
    break;
417
0
  }
418
14.8k
  }
419
14.8k
#endif
420
14.8k
}
421
422
bool MOS65XX_getInstruction(csh ud, const uint8_t *code, size_t code_len,
423
          MCInst *MI, uint16_t *size, uint64_t address,
424
          void *inst_info)
425
15.0k
{
426
15.0k
  int i;
427
15.0k
  unsigned char opcode;
428
15.0k
  unsigned char len;
429
15.0k
  unsigned cpu_offset = 0;
430
15.0k
  int cpu_type = MOS65XX_CPU_TYPE_6502;
431
15.0k
  cs_struct *handle = MI->csh;
432
15.0k
  mos65xx_info *info = (mos65xx_info *)handle->printer_info;
433
15.0k
  OpInfo opinfo;
434
435
15.0k
  if (code_len == 0) {
436
0
    *size = 1;
437
0
    return false;
438
0
  }
439
440
15.0k
  cpu_type = info->cpu_type;
441
15.0k
  cpu_offset = cpu_type * 256;
442
443
15.0k
  opcode = code[0];
444
15.0k
  opinfo = OpInfoTable[cpu_offset + opcode];
445
15.0k
  if (opinfo.ins == MOS65XX_INS_INVALID) {
446
100
    *size = 1;
447
100
    return false;
448
100
  }
449
450
14.9k
  len = opinfo.operand_bytes + 1;
451
452
14.9k
  if (cpu_type == MOS65XX_CPU_TYPE_65816 && opinfo.am == MOS65XX_AM_IMM) {
453
0
    switch (opinfo.ins) {
454
0
    case MOS65XX_INS_CPX:
455
0
    case MOS65XX_INS_CPY:
456
0
    case MOS65XX_INS_LDX:
457
0
    case MOS65XX_INS_LDY:
458
0
      if (info->long_x)
459
0
        ++len;
460
0
      break;
461
0
    case MOS65XX_INS_ADC:
462
0
    case MOS65XX_INS_AND:
463
0
    case MOS65XX_INS_BIT:
464
0
    case MOS65XX_INS_CMP:
465
0
    case MOS65XX_INS_EOR:
466
0
    case MOS65XX_INS_LDA:
467
0
    case MOS65XX_INS_ORA:
468
0
    case MOS65XX_INS_SBC:
469
0
      if (info->long_m)
470
0
        ++len;
471
0
      break;
472
0
    default:
473
0
      break;
474
0
    }
475
0
  }
476
477
14.9k
  if (code_len < len) {
478
59
    *size = 1;
479
59
    return false;
480
59
  }
481
482
14.8k
  MI->address = address;
483
484
14.8k
  MCInst_setOpcode(MI, cpu_offset + opcode);
485
14.8k
  MCInst_setOpcodePub(MI, opinfo.ins);
486
487
14.8k
  *size = len;
488
489
  /* needed to differentiate relative vs relative long */
490
14.8k
  MI->op1_size = len - 1;
491
14.8k
  if (opinfo.ins == MOS65XX_INS_NOP) {
492
593
    for (i = 1; i < len; ++i)
493
0
      MCOperand_CreateImm0(MI, code[i]);
494
593
  }
495
496
14.8k
  switch (opinfo.am) {
497
0
  case MOS65XX_AM_ZP_REL:
498
0
    MCOperand_CreateImm0(MI, code[1]);
499
0
    MCOperand_CreateImm0(MI, code[2]);
500
0
    break;
501
0
  case MOS65XX_AM_BLOCK:
502
0
    MCOperand_CreateImm0(MI, code[2]);
503
0
    MCOperand_CreateImm0(MI, code[1]);
504
0
    break;
505
5.12k
  case MOS65XX_AM_IMP:
506
6.02k
  case MOS65XX_AM_ACC:
507
6.02k
    break;
508
509
439
  case MOS65XX_AM_IMM:
510
439
    MI->has_imm = 1;
511
439
    MI->imm_size = len - 1;
512
    /* 65816 immediate is either 1 or 2 bytes */
513
    /* drop through */
514
8.87k
  default:
515
8.87k
    if (len == 2)
516
5.67k
      MCOperand_CreateImm0(MI, code[1]);
517
3.19k
    else if (len == 3)
518
3.19k
      MCOperand_CreateImm0(MI, (code[2] << 8) | code[1]);
519
0
    else if (len == 4)
520
0
      MCOperand_CreateImm0(
521
0
        MI, (code[3] << 16) | (code[2] << 8) | code[1]);
522
8.87k
    break;
523
14.8k
  }
524
525
14.8k
#ifndef CAPSTONE_DIET
526
14.8k
  if (MI->flat_insn->detail) {
527
14.8k
    fillDetails(MI, opinfo, cpu_type);
528
14.8k
  }
529
14.8k
#endif
530
531
14.8k
  return true;
532
14.8k
}
533
534
const char *MOS65XX_insn_name(csh handle, unsigned int id)
535
14.8k
{
536
#ifdef CAPSTONE_DIET
537
  return NULL;
538
#else
539
14.8k
  if (id >= ARR_SIZE(InstructionInfoTable)) {
540
0
    return NULL;
541
0
  }
542
14.8k
  return InstructionInfoTable[id].name;
543
14.8k
#endif
544
14.8k
}
545
546
const char *MOS65XX_reg_name(csh handle, unsigned int reg)
547
35.5k
{
548
#ifdef CAPSTONE_DIET
549
  return NULL;
550
#else
551
35.5k
  if (reg >= ARR_SIZE(RegNames)) {
552
0
    return NULL;
553
0
  }
554
35.5k
  return RegNames[(int)reg];
555
35.5k
#endif
556
35.5k
}
557
558
void MOS65XX_get_insn_id(cs_struct *h, cs_insn *insn, unsigned int id)
559
14.8k
{
560
  /* id is cpu_offset + opcode */
561
14.8k
  if (id < ARR_SIZE(OpInfoTable)) {
562
14.8k
    insn->id = OpInfoTable[id].ins;
563
14.8k
  }
564
14.8k
}
565
566
const char *MOS65XX_group_name(csh handle, unsigned int id)
567
3.63k
{
568
#ifdef CAPSTONE_DIET
569
  return NULL;
570
#else
571
3.63k
  if (id >= ARR_SIZE(GroupNames)) {
572
0
    return NULL;
573
0
  }
574
3.63k
  return GroupNames[(int)id];
575
3.63k
#endif
576
3.63k
}