Coverage Report

Created: 2026-05-30 06:22

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonev5/cs.c
Line
Count
Source
1
/* Capstone Disassembly Engine */
2
/* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2019 */
3
#if defined (WIN32) || defined (WIN64) || defined (_WIN32) || defined (_WIN64)
4
#pragma warning(disable:4996)     // disable MSVC's warning on strcpy()
5
#pragma warning(disable:28719)    // disable MSVC's warning on strcpy()
6
#endif
7
#if defined(CAPSTONE_HAS_OSXKERNEL)
8
#include <Availability.h>
9
#include <libkern/libkern.h>
10
#else
11
#include <stddef.h>
12
#include <stdio.h>
13
#include <stdlib.h>
14
#endif
15
16
#include <string.h>
17
#include <capstone/capstone.h>
18
19
#include "utils.h"
20
#include "MCRegisterInfo.h"
21
22
#if defined(_KERNEL_MODE)
23
#include "windows\winkernel_mm.h"
24
#endif
25
26
// Issue #681: Windows kernel does not support formatting float point
27
#if defined(_KERNEL_MODE) && !defined(CAPSTONE_DIET)
28
#if defined(CAPSTONE_HAS_ARM) || defined(CAPSTONE_HAS_ARM64) || defined(CAPSTONE_HAS_M68K)
29
#define CAPSTONE_STR_INTERNAL(x) #x
30
#define CAPSTONE_STR(x) CAPSTONE_STR_INTERNAL(x)
31
#define CAPSTONE_MSVC_WRANING_PREFIX __FILE__ "("CAPSTONE_STR(__LINE__)") : warning message : "
32
33
#pragma message(CAPSTONE_MSVC_WRANING_PREFIX "Windows driver does not support full features for selected architecture(s). Define CAPSTONE_DIET to compile Capstone with only supported features. See issue #681 for details.")
34
35
#undef CAPSTONE_MSVC_WRANING_PREFIX
36
#undef CAPSTONE_STR
37
#undef CAPSTONE_STR_INTERNAL
38
#endif
39
#endif  // defined(_KERNEL_MODE) && !defined(CAPSTONE_DIET)
40
41
#if !defined(CAPSTONE_HAS_OSXKERNEL) && !defined(CAPSTONE_DIET) && !defined(_KERNEL_MODE)
42
41.4k
#define INSN_CACHE_SIZE 32
43
#else
44
// reduce stack variable size for kernel/firmware
45
#define INSN_CACHE_SIZE 8
46
#endif
47
48
// default SKIPDATA mnemonic
49
#ifndef CAPSTONE_DIET
50
41.4k
#define SKIPDATA_MNEM ".byte"
51
#else // No printing is available in diet mode
52
#define SKIPDATA_MNEM NULL
53
#endif
54
55
#include "arch/AArch64/AArch64Module.h"
56
#include "arch/ARM/ARMModule.h"
57
#include "arch/EVM/EVMModule.h"
58
#include "arch/WASM/WASMModule.h"
59
#include "arch/M680X/M680XModule.h"
60
#include "arch/M68K/M68KModule.h"
61
#include "arch/Mips/MipsModule.h"
62
#include "arch/PowerPC/PPCModule.h"
63
#include "arch/Sparc/SparcModule.h"
64
#include "arch/SystemZ/SystemZModule.h"
65
#include "arch/TMS320C64x/TMS320C64xModule.h"
66
#include "arch/X86/X86Module.h"
67
#include "arch/XCore/XCoreModule.h"
68
#include "arch/RISCV/RISCVModule.h"
69
#include "arch/MOS65XX/MOS65XXModule.h"
70
#include "arch/BPF/BPFModule.h"
71
#include "arch/SH/SHModule.h"
72
#include "arch/TriCore/TriCoreModule.h"
73
74
static const struct {
75
  // constructor initialization
76
  cs_err (*arch_init)(cs_struct *);
77
  // support cs_option()
78
  cs_err (*arch_option)(cs_struct *, cs_opt_type, size_t value);
79
  // bitmask for finding disallowed modes for an arch:
80
  // to be called in cs_open()/cs_option()
81
  cs_mode arch_disallowed_mode_mask;
82
} arch_configs[MAX_ARCH] = {
83
#ifdef CAPSTONE_HAS_ARM
84
  {
85
    ARM_global_init,
86
    ARM_option,
87
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_ARM | CS_MODE_V8 | CS_MODE_MCLASS
88
        | CS_MODE_THUMB | CS_MODE_BIG_ENDIAN)
89
  },
90
#else
91
  { NULL, NULL, 0 },
92
#endif
93
#ifdef CAPSTONE_HAS_ARM64
94
  {
95
    AArch64_global_init,
96
    AArch64_option,
97
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_ARM | CS_MODE_BIG_ENDIAN),
98
  },
99
#else
100
  { NULL, NULL, 0 },
101
#endif
102
#ifdef CAPSTONE_HAS_MIPS
103
  {
104
    Mips_global_init,
105
    Mips_option,
106
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_32 | CS_MODE_64 | CS_MODE_MICRO
107
        | CS_MODE_MIPS32R6 | CS_MODE_BIG_ENDIAN | CS_MODE_MIPS2 | CS_MODE_MIPS3),
108
  },
109
#else
110
  { NULL, NULL, 0 },
111
#endif
112
#ifdef CAPSTONE_HAS_X86
113
  {
114
    X86_global_init,
115
    X86_option,
116
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_32 | CS_MODE_64 | CS_MODE_16),
117
  },
118
#else
119
  { NULL, NULL, 0 },
120
#endif
121
#ifdef CAPSTONE_HAS_POWERPC
122
  {
123
    PPC_global_init,
124
    PPC_option,
125
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_32 | CS_MODE_64 | CS_MODE_BIG_ENDIAN
126
        | CS_MODE_QPX | CS_MODE_PS),
127
  },
128
#else
129
  { NULL, NULL, 0 },
130
#endif
131
#ifdef CAPSTONE_HAS_SPARC
132
  {
133
    Sparc_global_init,
134
    Sparc_option,
135
    ~(CS_MODE_BIG_ENDIAN | CS_MODE_V9),
136
  },
137
#else
138
  { NULL, NULL, 0 },
139
#endif
140
#ifdef CAPSTONE_HAS_SYSZ
141
  {
142
    SystemZ_global_init,
143
    SystemZ_option,
144
    ~(CS_MODE_BIG_ENDIAN),
145
  },
146
#else
147
  { NULL, NULL, 0 },
148
#endif
149
#ifdef CAPSTONE_HAS_XCORE
150
  {
151
    XCore_global_init,
152
    XCore_option,
153
    ~(CS_MODE_BIG_ENDIAN),
154
  },
155
#else
156
  { NULL, NULL, 0 },
157
#endif
158
#ifdef CAPSTONE_HAS_M68K
159
  {
160
    M68K_global_init,
161
    M68K_option,
162
    ~(CS_MODE_BIG_ENDIAN | CS_MODE_M68K_000 | CS_MODE_M68K_010 | CS_MODE_M68K_020
163
        | CS_MODE_M68K_030 | CS_MODE_M68K_040 | CS_MODE_M68K_060),
164
  },
165
#else
166
  { NULL, NULL, 0 },
167
#endif
168
#ifdef CAPSTONE_HAS_TMS320C64X
169
  {
170
    TMS320C64x_global_init,
171
    TMS320C64x_option,
172
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_BIG_ENDIAN),
173
  },
174
#else
175
  { NULL, NULL, 0 },
176
#endif
177
#ifdef CAPSTONE_HAS_M680X
178
  {
179
    M680X_global_init,
180
    M680X_option,
181
    ~(CS_MODE_M680X_6301 | CS_MODE_M680X_6309 | CS_MODE_M680X_6800
182
        | CS_MODE_M680X_6801 | CS_MODE_M680X_6805 | CS_MODE_M680X_6808
183
        | CS_MODE_M680X_6809 | CS_MODE_M680X_6811 | CS_MODE_M680X_CPU12
184
        | CS_MODE_M680X_HCS08),
185
  },
186
#else
187
  { NULL, NULL, 0 },
188
#endif
189
#ifdef CAPSTONE_HAS_EVM
190
  {
191
    EVM_global_init,
192
    EVM_option,
193
    0,
194
  },
195
#else
196
  { NULL, NULL, 0 },
197
#endif
198
#ifdef CAPSTONE_HAS_MOS65XX
199
  {
200
    MOS65XX_global_init,
201
    MOS65XX_option,
202
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_MOS65XX_6502 | CS_MODE_MOS65XX_65C02
203
        | CS_MODE_MOS65XX_W65C02 | CS_MODE_MOS65XX_65816_LONG_MX),
204
  },
205
#else
206
  { NULL, NULL, 0 },
207
#endif
208
#ifdef CAPSTONE_HAS_WASM
209
  {
210
    WASM_global_init,
211
    WASM_option,
212
    0,
213
  },
214
#else
215
  { NULL, NULL, 0 },
216
#endif
217
#ifdef CAPSTONE_HAS_BPF
218
  {
219
    BPF_global_init,
220
    BPF_option,
221
    ~(CS_MODE_LITTLE_ENDIAN | CS_MODE_BPF_CLASSIC | CS_MODE_BPF_EXTENDED
222
        | CS_MODE_BIG_ENDIAN),
223
  },
224
#else
225
  { NULL, NULL, 0 },
226
#endif
227
#ifdef CAPSTONE_HAS_RISCV
228
  {
229
    RISCV_global_init,
230
    RISCV_option,
231
    ~(CS_MODE_RISCV32 | CS_MODE_RISCV64 | CS_MODE_RISCVC),
232
  },
233
#else
234
  { NULL, NULL, 0 },
235
#endif
236
#ifdef CAPSTONE_HAS_SH
237
  {
238
    SH_global_init,
239
    SH_option,
240
    ~(CS_MODE_SH2 | CS_MODE_SH2A | CS_MODE_SH3 |
241
      CS_MODE_SH4 | CS_MODE_SH4A |
242
      CS_MODE_SHFPU | CS_MODE_SHDSP|CS_MODE_BIG_ENDIAN),
243
  },
244
#else
245
  { NULL, NULL, 0 },
246
#endif
247
#ifdef CAPSTONE_HAS_TRICORE
248
  {
249
    TRICORE_global_init,
250
    TRICORE_option,
251
    ~(CS_MODE_TRICORE_110 | CS_MODE_TRICORE_120 | CS_MODE_TRICORE_130
252
    | CS_MODE_TRICORE_131 | CS_MODE_TRICORE_160 | CS_MODE_TRICORE_161
253
    | CS_MODE_TRICORE_162 | CS_MODE_LITTLE_ENDIAN),
254
  },
255
#else
256
  { NULL, NULL, 0 },
257
#endif
258
};
259
260
// bitmask of enabled architectures
261
static const uint32_t all_arch = 0
262
#ifdef CAPSTONE_HAS_ARM
263
  | (1 << CS_ARCH_ARM)
264
#endif
265
#ifdef CAPSTONE_HAS_ARM64
266
  | (1 << CS_ARCH_ARM64)
267
#endif
268
#ifdef CAPSTONE_HAS_MIPS
269
  | (1 << CS_ARCH_MIPS)
270
#endif
271
#ifdef CAPSTONE_HAS_X86
272
  | (1 << CS_ARCH_X86)
273
#endif
274
#ifdef CAPSTONE_HAS_POWERPC
275
  | (1 << CS_ARCH_PPC)
276
#endif
277
#ifdef CAPSTONE_HAS_SPARC
278
  | (1 << CS_ARCH_SPARC)
279
#endif
280
#ifdef CAPSTONE_HAS_SYSZ
281
  | (1 << CS_ARCH_SYSZ)
282
#endif
283
#ifdef CAPSTONE_HAS_XCORE
284
  | (1 << CS_ARCH_XCORE)
285
#endif
286
#ifdef CAPSTONE_HAS_M68K
287
  | (1 << CS_ARCH_M68K)
288
#endif
289
#ifdef CAPSTONE_HAS_TMS320C64X
290
  | (1 << CS_ARCH_TMS320C64X)
291
#endif
292
#ifdef CAPSTONE_HAS_M680X
293
  | (1 << CS_ARCH_M680X)
294
#endif
295
#ifdef CAPSTONE_HAS_EVM
296
  | (1 << CS_ARCH_EVM)
297
#endif
298
#ifdef CAPSTONE_HAS_MOS65XX
299
  | (1 << CS_ARCH_MOS65XX)
300
#endif
301
#ifdef CAPSTONE_HAS_WASM
302
  | (1 << CS_ARCH_WASM)
303
#endif
304
#ifdef CAPSTONE_HAS_BPF
305
  | (1 << CS_ARCH_BPF)
306
#endif
307
#ifdef CAPSTONE_HAS_RISCV
308
  | (1 << CS_ARCH_RISCV)
309
#endif
310
#ifdef CAPSTONE_HAS_SH
311
  | (1 << CS_ARCH_SH)
312
#endif
313
#ifdef CAPSTONE_HAS_TRICORE
314
  | (1 << CS_ARCH_TRICORE)
315
#endif
316
;
317
318
319
#if defined(CAPSTONE_USE_SYS_DYN_MEM)
320
#if !defined(CAPSTONE_HAS_OSXKERNEL) && !defined(_KERNEL_MODE)
321
// default
322
cs_malloc_t cs_mem_malloc = malloc;
323
cs_calloc_t cs_mem_calloc = calloc;
324
cs_realloc_t cs_mem_realloc = realloc;
325
cs_free_t cs_mem_free = free;
326
#if defined(_WIN32_WCE)
327
cs_vsnprintf_t cs_vsnprintf = _vsnprintf;
328
#else
329
cs_vsnprintf_t cs_vsnprintf = vsnprintf;
330
#endif  // defined(_WIN32_WCE)
331
332
#elif defined(_KERNEL_MODE)
333
// Windows driver
334
cs_malloc_t cs_mem_malloc = cs_winkernel_malloc;
335
cs_calloc_t cs_mem_calloc = cs_winkernel_calloc;
336
cs_realloc_t cs_mem_realloc = cs_winkernel_realloc;
337
cs_free_t cs_mem_free = cs_winkernel_free;
338
cs_vsnprintf_t cs_vsnprintf = cs_winkernel_vsnprintf;
339
#else
340
// OSX kernel
341
extern void* kern_os_malloc(size_t size);
342
extern void kern_os_free(void* addr);
343
extern void* kern_os_realloc(void* addr, size_t nsize);
344
345
static void* cs_kern_os_calloc(size_t num, size_t size)
346
{
347
  return kern_os_malloc(num * size); // malloc bzeroes the buffer
348
}
349
350
cs_malloc_t cs_mem_malloc = kern_os_malloc;
351
cs_calloc_t cs_mem_calloc = cs_kern_os_calloc;
352
cs_realloc_t cs_mem_realloc = kern_os_realloc;
353
cs_free_t cs_mem_free = kern_os_free;
354
cs_vsnprintf_t cs_vsnprintf = vsnprintf;
355
#endif  // !defined(CAPSTONE_HAS_OSXKERNEL) && !defined(_KERNEL_MODE)
356
#else
357
// User-defined
358
cs_malloc_t cs_mem_malloc = NULL;
359
cs_calloc_t cs_mem_calloc = NULL;
360
cs_realloc_t cs_mem_realloc = NULL;
361
cs_free_t cs_mem_free = NULL;
362
cs_vsnprintf_t cs_vsnprintf = NULL;
363
364
#endif  // defined(CAPSTONE_USE_SYS_DYN_MEM)
365
366
CAPSTONE_EXPORT
367
unsigned int CAPSTONE_API cs_version(int *major, int *minor)
368
0
{
369
0
  if (major != NULL && minor != NULL) {
370
0
    *major = CS_API_MAJOR;
371
0
    *minor = CS_API_MINOR;
372
0
  }
373
374
0
  return (CS_API_MAJOR << 8) + CS_API_MINOR;
375
0
}
376
377
CAPSTONE_EXPORT
378
bool CAPSTONE_API cs_support(int query)
379
0
{
380
0
  if (query == CS_ARCH_ALL)
381
0
    return all_arch == ((1 << CS_ARCH_ARM)   | (1 << CS_ARCH_ARM64)      |
382
0
            (1 << CS_ARCH_MIPS)  | (1 << CS_ARCH_X86)        |
383
0
            (1 << CS_ARCH_PPC)   | (1 << CS_ARCH_SPARC)      |
384
0
            (1 << CS_ARCH_SYSZ)  | (1 << CS_ARCH_XCORE)      |
385
0
            (1 << CS_ARCH_M68K)  | (1 << CS_ARCH_TMS320C64X) |
386
0
            (1 << CS_ARCH_M680X) | (1 << CS_ARCH_EVM)        |
387
0
            (1 << CS_ARCH_RISCV) | (1 << CS_ARCH_MOS65XX)    |
388
0
            (1 << CS_ARCH_WASM)  | (1 << CS_ARCH_BPF)        |
389
0
            (1 << CS_ARCH_SH)    | (1 << CS_ARCH_TRICORE));
390
391
0
  if ((unsigned int)query < CS_ARCH_MAX)
392
0
    return all_arch & (1 << query);
393
394
0
  if (query == CS_SUPPORT_DIET) {
395
#ifdef CAPSTONE_DIET
396
    return true;
397
#else
398
0
    return false;
399
0
#endif
400
0
  }
401
402
0
  if (query == CS_SUPPORT_X86_REDUCE) {
403
#if defined(CAPSTONE_HAS_X86) && defined(CAPSTONE_X86_REDUCE)
404
    return true;
405
#else
406
0
    return false;
407
0
#endif
408
0
  }
409
410
  // unsupported query
411
0
  return false;
412
0
}
413
414
CAPSTONE_EXPORT
415
cs_err CAPSTONE_API cs_errno(csh handle)
416
0
{
417
0
  struct cs_struct *ud;
418
0
  if (!handle)
419
0
    return CS_ERR_CSH;
420
421
0
  ud = (struct cs_struct *)(uintptr_t)handle;
422
423
0
  return ud->errnum;
424
0
}
425
426
CAPSTONE_EXPORT
427
const char * CAPSTONE_API cs_strerror(cs_err code)
428
0
{
429
0
  switch(code) {
430
0
    default:
431
0
      return "Unknown error code";
432
0
    case CS_ERR_OK:
433
0
      return "OK (CS_ERR_OK)";
434
0
    case CS_ERR_MEM:
435
0
      return "Out of memory (CS_ERR_MEM)";
436
0
    case CS_ERR_ARCH:
437
0
      return "Invalid/unsupported architecture(CS_ERR_ARCH)";
438
0
    case CS_ERR_HANDLE:
439
0
      return "Invalid handle (CS_ERR_HANDLE)";
440
0
    case CS_ERR_CSH:
441
0
      return "Invalid csh (CS_ERR_CSH)";
442
0
    case CS_ERR_MODE:
443
0
      return "Invalid mode (CS_ERR_MODE)";
444
0
    case CS_ERR_OPTION:
445
0
      return "Invalid option (CS_ERR_OPTION)";
446
0
    case CS_ERR_DETAIL:
447
0
      return "Details are unavailable (CS_ERR_DETAIL)";
448
0
    case CS_ERR_MEMSETUP:
449
0
      return "Dynamic memory management uninitialized (CS_ERR_MEMSETUP)";
450
0
    case CS_ERR_VERSION:
451
0
      return "Different API version between core & binding (CS_ERR_VERSION)";
452
0
    case CS_ERR_DIET:
453
0
      return "Information irrelevant in diet engine (CS_ERR_DIET)";
454
0
    case CS_ERR_SKIPDATA:
455
0
      return "Information irrelevant for 'data' instruction in SKIPDATA mode (CS_ERR_SKIPDATA)";
456
0
    case CS_ERR_X86_ATT:
457
0
      return "AT&T syntax is unavailable (CS_ERR_X86_ATT)";
458
0
    case CS_ERR_X86_INTEL:
459
0
      return "INTEL syntax is unavailable (CS_ERR_X86_INTEL)";
460
0
    case CS_ERR_X86_MASM:
461
0
      return "MASM syntax is unavailable (CS_ERR_X86_MASM)";
462
0
  }
463
0
}
464
465
CAPSTONE_EXPORT
466
cs_err CAPSTONE_API cs_open(cs_arch arch, cs_mode mode, csh *handle)
467
41.4k
{
468
41.4k
  cs_err err;
469
41.4k
  struct cs_struct *ud;
470
41.4k
  if (!cs_mem_malloc || !cs_mem_calloc || !cs_mem_realloc || !cs_mem_free || !cs_vsnprintf)
471
    // Error: before cs_open(), dynamic memory management must be initialized
472
    // with cs_option(CS_OPT_MEM)
473
0
    return CS_ERR_MEMSETUP;
474
475
41.4k
  if (arch < CS_ARCH_MAX && arch_configs[arch].arch_init) {
476
    // verify if requested mode is valid
477
41.4k
    if (mode & arch_configs[arch].arch_disallowed_mode_mask) {
478
2
      *handle = 0;
479
2
      return CS_ERR_MODE;
480
2
    }
481
482
41.4k
    ud = cs_mem_calloc(1, sizeof(*ud));
483
41.4k
    if (!ud) {
484
      // memory insufficient
485
0
      return CS_ERR_MEM;
486
0
    }
487
488
41.4k
    ud->errnum = CS_ERR_OK;
489
41.4k
    ud->arch = arch;
490
41.4k
    ud->mode = mode;
491
    // by default, do not break instruction into details
492
41.4k
    ud->detail = CS_OPT_OFF;
493
494
    // default skipdata setup
495
41.4k
    ud->skipdata_setup.mnemonic = SKIPDATA_MNEM;
496
497
41.4k
    err = arch_configs[ud->arch].arch_init(ud);
498
41.4k
    if (err) {
499
0
      cs_mem_free(ud);
500
0
      *handle = 0;
501
0
      return err;
502
0
    }
503
504
41.4k
    *handle = (uintptr_t)ud;
505
506
41.4k
    return CS_ERR_OK;
507
41.4k
  } else {
508
0
    *handle = 0;
509
0
    return CS_ERR_ARCH;
510
0
  }
511
41.4k
}
512
513
CAPSTONE_EXPORT
514
cs_err CAPSTONE_API cs_close(csh *handle)
515
41.4k
{
516
41.4k
  struct cs_struct *ud;
517
41.4k
  struct insn_mnem *next, *tmp;
518
519
41.4k
  if (*handle == 0)
520
    // invalid handle
521
0
    return CS_ERR_CSH;
522
523
41.4k
  ud = (struct cs_struct *)(*handle);
524
525
41.4k
  if (ud->printer_info)
526
39.3k
    cs_mem_free(ud->printer_info);
527
528
  // free the linked list of customized mnemonic
529
41.4k
  tmp = ud->mnem_list;
530
41.4k
  while(tmp) {
531
0
    next = tmp->next;
532
0
    cs_mem_free(tmp);
533
0
    tmp = next;
534
0
  }
535
536
41.4k
  cs_mem_free(ud->insn_cache);
537
41.4k
  cs_mem_free(ud->x86_insn_lut);
538
41.4k
  cs_mem_free(ud->x86_insn_reg_lut);
539
540
41.4k
  memset(ud, 0, sizeof(*ud));
541
41.4k
  cs_mem_free(ud);
542
543
  // invalidate this handle by ZERO out its value.
544
  // this is to make sure it is unusable after cs_close()
545
41.4k
  *handle = 0;
546
547
41.4k
  return CS_ERR_OK;
548
41.4k
}
549
550
// replace str1 in target with str2; target starts with str1
551
// output is put into result (which is array of char with size CS_MNEMONIC_SIZE)
552
// return 0 on success, -1 on failure
553
static int str_replace(char *result, char *target, const char *str1, char *str2)
554
0
{
555
  // only perform replacement if the output fits into result
556
0
  if (strlen(target) - strlen(str1) + strlen(str2) < CS_MNEMONIC_SIZE - 1)  {
557
    // copy str2 to begining of result
558
0
    strcpy(result, str2);
559
    // skip str1 - already replaced by str2
560
0
    strcat(result, target + strlen(str1));
561
562
0
    return 0;
563
0
  } else
564
0
    return -1;
565
0
}
566
567
// fill insn with mnemonic & operands info
568
static void fill_insn(struct cs_struct *handle, cs_insn *insn, char *buffer, MCInst *mci,
569
    PostPrinter_t postprinter, const uint8_t *code)
570
3.00M
{
571
3.00M
#ifndef CAPSTONE_DIET
572
3.00M
  char *sp, *mnem;
573
3.00M
#endif
574
3.00M
  uint16_t copy_size = MIN(sizeof(insn->bytes), insn->size);
575
576
  // fill the instruction bytes.
577
  // we might skip some redundant bytes in front in the case of X86
578
3.00M
  memcpy(insn->bytes, code + insn->size - copy_size, copy_size);
579
3.00M
  insn->op_str[0] = '\0';
580
3.00M
  insn->size = copy_size;
581
582
  // alias instruction might have ID saved in OpcodePub
583
3.00M
  if (MCInst_getOpcodePub(mci))
584
212k
    insn->id = MCInst_getOpcodePub(mci);
585
586
  // post printer handles some corner cases (hacky)
587
3.00M
  if (postprinter)
588
2.21M
    postprinter((csh)handle, insn, buffer, mci);
589
590
3.00M
#ifndef CAPSTONE_DIET
591
3.00M
  mnem = insn->mnemonic;
592
  // memset(mnem, 0, CS_MNEMONIC_SIZE);
593
17.2M
  for (sp = buffer; *sp; sp++) {
594
17.1M
    if (*sp == ' '|| *sp == '\t')
595
2.83M
      break;
596
14.2M
    if (*sp == '|')  // lock|rep prefix for x86
597
87.9k
      *sp = ' ';
598
    // copy to @mnemonic
599
14.2M
    *mnem = *sp;
600
14.2M
    mnem++;
601
14.2M
  }
602
603
3.00M
  *mnem = '\0';
604
605
  // we might have customized mnemonic
606
3.00M
  if (handle->mnem_list) {
607
0
    struct insn_mnem *tmp = handle->mnem_list;
608
0
    while(tmp) {
609
0
      if (tmp->insn.id == insn->id) {
610
0
        char str[CS_MNEMONIC_SIZE];
611
612
0
        if (!str_replace(str, insn->mnemonic, cs_insn_name((csh)handle, insn->id), tmp->insn.mnemonic)) {
613
          // copy result to mnemonic
614
0
          (void)strncpy(insn->mnemonic, str, sizeof(insn->mnemonic) - 1);
615
0
          insn->mnemonic[sizeof(insn->mnemonic) - 1] = '\0';
616
0
        }
617
618
0
        break;
619
0
      }
620
0
      tmp = tmp->next;
621
0
    }
622
0
  }
623
624
  // copy @op_str
625
3.00M
  if (*sp) {
626
    // find the next non-space char
627
2.83M
    sp++;
628
2.83M
    for (; ((*sp == ' ') || (*sp == '\t')); sp++);
629
2.83M
    strncpy(insn->op_str, sp, sizeof(insn->op_str) - 1);
630
2.83M
    insn->op_str[sizeof(insn->op_str) - 1] = '\0';
631
2.83M
  } else
632
168k
    insn->op_str[0] = '\0';
633
634
3.00M
#endif
635
3.00M
}
636
637
// how many bytes will we skip when encountering data (CS_OPT_SKIPDATA)?
638
// this very much depends on instruction alignment requirement of each arch.
639
static uint8_t skipdata_size(cs_struct *handle)
640
0
{
641
0
  switch(handle->arch) {
642
0
    default:
643
      // should never reach
644
0
      return (uint8_t)-1;
645
0
    case CS_ARCH_ARM:
646
      // skip 2 bytes on Thumb mode.
647
0
      if (handle->mode & CS_MODE_THUMB)
648
0
        return 2;
649
      // otherwise, skip 4 bytes
650
0
      return 4;
651
0
    case CS_ARCH_ARM64:
652
0
    case CS_ARCH_MIPS:
653
0
    case CS_ARCH_PPC:
654
0
    case CS_ARCH_SPARC:
655
      // skip 4 bytes
656
0
      return 4;
657
0
    case CS_ARCH_SYSZ:
658
      // SystemZ instruction's length can be 2, 4 or 6 bytes,
659
      // so we just skip 2 bytes
660
0
      return 2;
661
0
    case CS_ARCH_X86:
662
      // X86 has no restriction on instruction alignment
663
0
      return 1;
664
0
    case CS_ARCH_XCORE:
665
      // XCore instruction's length can be 2 or 4 bytes,
666
      // so we just skip 2 bytes
667
0
      return 2;
668
0
    case CS_ARCH_M68K:
669
      // M68K has 2 bytes instruction alignment but contain multibyte instruction so we skip 2 bytes
670
0
      return 2;
671
0
    case CS_ARCH_TMS320C64X:
672
      // TMS320C64x alignment is 4.
673
0
      return 4;
674
0
    case CS_ARCH_M680X:
675
      // M680X alignment is 1.
676
0
      return 1;
677
0
    case CS_ARCH_EVM:
678
      // EVM alignment is 1.
679
0
      return 1;
680
0
    case CS_ARCH_WASM:
681
      //WASM alignment is 1
682
0
      return 1;
683
0
    case CS_ARCH_MOS65XX:
684
      // MOS65XX alignment is 1.
685
0
      return 1;
686
0
    case CS_ARCH_BPF:
687
      // both classic and extended BPF have alignment 8.
688
0
      return 8;
689
0
    case CS_ARCH_RISCV:
690
      // special compress mode
691
0
      if (handle->mode & CS_MODE_RISCVC)
692
0
        return 2;
693
0
      return 4;
694
0
    case CS_ARCH_SH:
695
0
      return 2;
696
0
    case CS_ARCH_TRICORE:
697
      // TriCore instruction's length can be 2 or 4 bytes,
698
      // so we just skip 2 bytes
699
0
      return 2;
700
0
  }
701
0
}
702
703
CAPSTONE_EXPORT
704
cs_err CAPSTONE_API cs_option(csh ud, cs_opt_type type, size_t value)
705
53.3k
{
706
53.3k
  struct cs_struct *handle;
707
53.3k
  cs_opt_mnem *opt;
708
709
  // cs_option() can be called with NULL handle just for CS_OPT_MEM
710
  // This is supposed to be executed before all other APIs (even cs_open())
711
53.3k
  if (type == CS_OPT_MEM) {
712
0
    cs_opt_mem *mem = (cs_opt_mem *)value;
713
714
0
    cs_mem_malloc = mem->malloc;
715
0
    cs_mem_calloc = mem->calloc;
716
0
    cs_mem_realloc = mem->realloc;
717
0
    cs_mem_free = mem->free;
718
0
    cs_vsnprintf = mem->vsnprintf;
719
720
0
    return CS_ERR_OK;
721
0
  }
722
723
53.3k
  handle = (struct cs_struct *)(uintptr_t)ud;
724
53.3k
  if (!handle)
725
0
    return CS_ERR_CSH;
726
727
53.3k
  switch(type) {
728
11.8k
    default:
729
11.8k
      break;
730
731
11.8k
    case CS_OPT_UNSIGNED:
732
0
      handle->imm_unsigned = (cs_opt_value)value;
733
0
      return CS_ERR_OK;
734
735
41.4k
    case CS_OPT_DETAIL:
736
41.4k
      handle->detail = (cs_opt_value)value;
737
41.4k
      return CS_ERR_OK;
738
739
0
    case CS_OPT_SKIPDATA:
740
0
      handle->skipdata = (value == CS_OPT_ON);
741
0
      if (handle->skipdata) {
742
0
        if (handle->skipdata_size == 0) {
743
          // set the default skipdata size
744
0
          handle->skipdata_size = skipdata_size(handle);
745
0
        }
746
0
      }
747
0
      return CS_ERR_OK;
748
749
0
    case CS_OPT_SKIPDATA_SETUP:
750
0
      if (value) {
751
0
        handle->skipdata_setup = *((cs_opt_skipdata *)value);
752
0
        if (handle->skipdata_setup.mnemonic == NULL) {
753
0
          handle->skipdata_setup.mnemonic = SKIPDATA_MNEM;
754
0
        }
755
0
      }
756
0
      return CS_ERR_OK;
757
758
0
    case CS_OPT_MNEMONIC:
759
0
      opt = (cs_opt_mnem *)value;
760
0
      if (opt->id) {
761
0
        if (opt->mnemonic) {
762
0
          struct insn_mnem *tmp;
763
764
          // add new instruction, or replace existing instruction
765
          // 1. find if we already had this insn in the linked list
766
0
          tmp = handle->mnem_list;
767
0
          while(tmp) {
768
0
            if (tmp->insn.id == opt->id) {
769
              // found this instruction, so replace its mnemonic
770
0
              (void)strncpy(tmp->insn.mnemonic, opt->mnemonic, sizeof(tmp->insn.mnemonic) - 1);
771
0
              tmp->insn.mnemonic[sizeof(tmp->insn.mnemonic) - 1] = '\0';
772
0
              break;
773
0
            }
774
0
            tmp = tmp->next;
775
0
          }
776
777
          // 2. add this instruction if we have not had it yet
778
0
          if (!tmp) {
779
0
            tmp = cs_mem_malloc(sizeof(*tmp));
780
0
            tmp->insn.id = opt->id;
781
0
            (void)strncpy(tmp->insn.mnemonic, opt->mnemonic, sizeof(tmp->insn.mnemonic) - 1);
782
0
            tmp->insn.mnemonic[sizeof(tmp->insn.mnemonic) - 1] = '\0';
783
            // this new instruction is heading the list
784
0
            tmp->next = handle->mnem_list;
785
0
            handle->mnem_list = tmp;
786
0
          }
787
0
          return CS_ERR_OK;
788
0
        } else {
789
0
          struct insn_mnem *prev, *tmp;
790
791
          // we want to delete an existing instruction
792
          // iterate the list to find the instruction to remove it
793
0
          tmp = handle->mnem_list;
794
0
          prev = tmp;
795
0
          while(tmp) {
796
0
            if (tmp->insn.id == opt->id) {
797
              // delete this instruction
798
0
              if (tmp == prev) {
799
                // head of the list
800
0
                handle->mnem_list = tmp->next;
801
0
              } else {
802
0
                prev->next = tmp->next;
803
0
              }
804
0
              cs_mem_free(tmp);
805
0
              break;
806
0
            }
807
0
            prev = tmp;
808
0
            tmp = tmp->next;
809
0
          }
810
0
        }
811
0
      }
812
0
      return CS_ERR_OK;
813
814
0
    case CS_OPT_MODE:
815
      // verify if requested mode is valid
816
0
      if (value & arch_configs[handle->arch].arch_disallowed_mode_mask) {
817
0
        return CS_ERR_OPTION;
818
0
      }
819
0
      break;
820
53.3k
  }
821
822
11.8k
  return arch_configs[handle->arch].arch_option(handle, type, value);
823
53.3k
}
824
825
// generate @op_str for data instruction of SKIPDATA
826
#ifndef CAPSTONE_DIET
827
static void skipdata_opstr(char *opstr, const uint8_t *buffer, size_t size)
828
0
{
829
0
  char *p = opstr;
830
0
  int len;
831
0
  size_t i;
832
0
  size_t available = sizeof(((cs_insn*)NULL)->op_str);
833
834
0
  if (!size) {
835
0
    opstr[0] = '\0';
836
0
    return;
837
0
  }
838
839
0
  len = cs_snprintf(p, available, "0x%02x", buffer[0]);
840
0
  p+= len;
841
0
  available -= len;
842
843
0
  for(i = 1; i < size; i++) {
844
0
    len = cs_snprintf(p, available, ", 0x%02x", buffer[i]);
845
0
    if (len < 0) {
846
0
      break;
847
0
    }
848
0
    if ((size_t)len > available - 1) {
849
0
      break;
850
0
    }
851
0
    p+= len;
852
0
    available -= len;
853
0
  }
854
0
}
855
#endif
856
857
// dynamicly allocate memory to contain disasm insn
858
// NOTE: caller must free() the allocated memory itself to avoid memory leaking
859
CAPSTONE_EXPORT
860
size_t CAPSTONE_API cs_disasm(csh ud, const uint8_t *buffer, size_t size, uint64_t offset, size_t count, cs_insn **insn)
861
41.4k
{
862
41.4k
  struct cs_struct *handle;
863
41.4k
  MCInst mci;
864
41.4k
  uint16_t insn_size;
865
41.4k
  size_t c = 0, i;
866
41.4k
  unsigned int f = 0; // index of the next instruction in the cache
867
41.4k
  cs_insn *insn_cache;  // cache contains disassembled instructions
868
41.4k
  void *total = NULL;
869
41.4k
  size_t total_size = 0;  // total size of output buffer containing all insns
870
41.4k
  bool r;
871
41.4k
  void *tmp;
872
41.4k
  size_t skipdata_bytes;
873
41.4k
  uint64_t offset_org; // save all the original info of the buffer
874
41.4k
  size_t size_org;
875
41.4k
  const uint8_t *buffer_org;
876
41.4k
  unsigned int cache_size = INSN_CACHE_SIZE;
877
41.4k
  size_t next_offset;
878
879
41.4k
  handle = (struct cs_struct *)(uintptr_t)ud;
880
41.4k
  if (!handle) {
881
    // FIXME: how to handle this case:
882
    // handle->errnum = CS_ERR_HANDLE;
883
0
    return 0;
884
0
  }
885
886
41.4k
  handle->errnum = CS_ERR_OK;
887
888
  // reset IT block of ARM structure
889
41.4k
  if (handle->arch == CS_ARCH_ARM)
890
7.73k
    handle->ITBlock.size = 0;
891
892
41.4k
#ifdef CAPSTONE_USE_SYS_DYN_MEM
893
41.4k
  if (count > 0 && count <= INSN_CACHE_SIZE)
894
0
    cache_size = (unsigned int) count;
895
41.4k
#endif
896
897
  // save the original offset for SKIPDATA
898
41.4k
  buffer_org = buffer;
899
41.4k
  offset_org = offset;
900
41.4k
  size_org = size;
901
902
41.4k
  total_size = sizeof(cs_insn) * cache_size;
903
41.4k
  total = cs_mem_calloc(sizeof(cs_insn), cache_size);
904
41.4k
  if (total == NULL) {
905
    // insufficient memory
906
0
    handle->errnum = CS_ERR_MEM;
907
0
    return 0;
908
0
  }
909
910
41.4k
  insn_cache = total;
911
912
3.04M
  while (size > 0) {
913
3.03M
    MCInst_Init(&mci);
914
3.03M
    mci.csh = handle;
915
916
    // relative branches need to know the address & size of current insn
917
3.03M
    mci.address = offset;
918
919
3.03M
    if (handle->detail) {
920
      // allocate memory for @detail pointer
921
3.03M
      insn_cache->detail = cs_mem_malloc(sizeof(cs_detail));
922
3.03M
    } else {
923
0
      insn_cache->detail = NULL;
924
0
    }
925
926
    // save all the information for non-detailed mode
927
3.03M
    mci.flat_insn = insn_cache;
928
3.03M
    mci.flat_insn->address = offset;
929
#ifdef CAPSTONE_DIET
930
    // zero out mnemonic & op_str
931
    mci.flat_insn->mnemonic[0] = '\0';
932
    mci.flat_insn->op_str[0] = '\0';
933
#endif
934
935
3.03M
    r = handle->disasm(ud, buffer, size, &mci, &insn_size, offset, handle->getinsn_info);
936
3.03M
    if (r) {
937
3.00M
      SStream ss;
938
3.00M
      SStream_Init(&ss);
939
940
3.00M
      mci.flat_insn->size = insn_size;
941
942
      // map internal instruction opcode to public insn ID
943
944
3.00M
      handle->insn_id(handle, insn_cache, mci.Opcode);
945
946
3.00M
      handle->printer(&mci, &ss, handle->printer_info);
947
3.00M
      fill_insn(handle, insn_cache, ss.buffer, &mci, handle->post_printer, buffer);
948
949
      // adjust for pseudo opcode (X86)
950
3.00M
      if (handle->arch == CS_ARCH_X86 && insn_cache->id != X86_INS_VCMP)
951
769k
        insn_cache->id += mci.popcode_adjust;
952
953
3.00M
      next_offset = insn_size;
954
3.00M
    } else {
955
      // encounter a broken instruction
956
957
      // free memory of @detail pointer
958
24.3k
      if (handle->detail) {
959
24.3k
        cs_mem_free(insn_cache->detail);
960
24.3k
      }
961
962
      // if there is no request to skip data, or remaining data is too small,
963
      // then bail out
964
24.3k
      if (!handle->skipdata || handle->skipdata_size > size)
965
24.3k
        break;
966
967
0
      if (handle->skipdata_setup.callback) {
968
0
        skipdata_bytes = handle->skipdata_setup.callback(buffer_org, size_org,
969
0
            (size_t)(offset - offset_org), handle->skipdata_setup.user_data);
970
0
        if (skipdata_bytes > size)
971
          // remaining data is not enough
972
0
          break;
973
974
0
        if (!skipdata_bytes)
975
          // user requested not to skip data, so bail out
976
0
          break;
977
0
      } else
978
0
        skipdata_bytes = handle->skipdata_size;
979
980
      // we have to skip some amount of data, depending on arch & mode
981
      // invalid ID for this "data" instruction
982
0
      insn_cache->id = 0;
983
0
      insn_cache->address = offset;
984
0
      insn_cache->size = (uint16_t)MIN(
985
0
        skipdata_bytes, sizeof(insn_cache->bytes));
986
0
      memcpy(insn_cache->bytes, buffer,
987
0
             MIN(skipdata_bytes, sizeof(insn_cache->bytes)));
988
#ifdef CAPSTONE_DIET
989
      insn_cache->mnemonic[0] = '\0';
990
      insn_cache->op_str[0] = '\0';
991
#else
992
0
      strncpy(insn_cache->mnemonic, handle->skipdata_setup.mnemonic,
993
0
          sizeof(insn_cache->mnemonic) - 1);
994
0
      skipdata_opstr(insn_cache->op_str, buffer, skipdata_bytes);
995
0
#endif
996
0
      insn_cache->detail = NULL;
997
998
0
      next_offset = skipdata_bytes;
999
0
    }
1000
1001
    // one more instruction entering the cache
1002
3.00M
    f++;
1003
1004
    // one more instruction disassembled
1005
3.00M
    c++;
1006
3.00M
    if (count > 0 && c == count)
1007
      // already got requested number of instructions
1008
0
      break;
1009
1010
3.00M
    if (f == cache_size) {
1011
      // full cache, so expand the cache to contain incoming insns
1012
30.8k
      cache_size = cache_size * 8 / 5; // * 1.6 ~ golden ratio
1013
30.8k
      total_size += (sizeof(cs_insn) * cache_size);
1014
30.8k
      tmp = cs_mem_realloc(total, total_size);
1015
30.8k
      if (tmp == NULL) { // insufficient memory
1016
0
        if (handle->detail) {
1017
0
          insn_cache = (cs_insn *)total;
1018
0
          for (i = 0; i < c; i++, insn_cache++)
1019
0
            cs_mem_free(insn_cache->detail);
1020
0
        }
1021
1022
0
        cs_mem_free(total);
1023
0
        *insn = NULL;
1024
0
        handle->errnum = CS_ERR_MEM;
1025
0
        return 0;
1026
0
      }
1027
1028
30.8k
      total = tmp;
1029
      // continue to fill in the cache after the last instruction
1030
30.8k
      insn_cache = (cs_insn *)((char *)total + sizeof(cs_insn) * c);
1031
1032
      // reset f back to 0, so we fill in the cache from begining
1033
30.8k
      f = 0;
1034
30.8k
    } else
1035
2.97M
      insn_cache++;
1036
1037
3.00M
    buffer += next_offset;
1038
3.00M
    size -= next_offset;
1039
3.00M
    offset += next_offset;
1040
3.00M
  }
1041
1042
41.4k
  if (!c) {
1043
    // we did not disassemble any instruction
1044
724
    cs_mem_free(total);
1045
724
    total = NULL;
1046
40.7k
  } else if (f != cache_size) {
1047
    // total did not fully use the last cache, so downsize it
1048
40.7k
    tmp = cs_mem_realloc(total, total_size - (cache_size - f) * sizeof(*insn_cache));
1049
40.7k
    if (tmp == NULL) { // insufficient memory
1050
      // free all detail pointers
1051
0
      if (handle->detail) {
1052
0
        insn_cache = (cs_insn *)total;
1053
0
        for (i = 0; i < c; i++, insn_cache++)
1054
0
          cs_mem_free(insn_cache->detail);
1055
0
      }
1056
1057
0
      cs_mem_free(total);
1058
0
      *insn = NULL;
1059
1060
0
      handle->errnum = CS_ERR_MEM;
1061
0
      return 0;
1062
0
    }
1063
1064
40.7k
    total = tmp;
1065
40.7k
  }
1066
1067
41.4k
  *insn = total;
1068
1069
41.4k
  return c;
1070
41.4k
}
1071
1072
CAPSTONE_EXPORT
1073
void CAPSTONE_API cs_free(cs_insn *insn, size_t count)
1074
40.7k
{
1075
40.7k
  size_t i;
1076
1077
  // free all detail pointers
1078
3.04M
  for (i = 0; i < count; i++)
1079
3.00M
    cs_mem_free(insn[i].detail);
1080
1081
  // then free pointer to cs_insn array
1082
40.7k
  cs_mem_free(insn);
1083
40.7k
}
1084
1085
CAPSTONE_EXPORT
1086
cs_insn * CAPSTONE_API cs_malloc(csh ud)
1087
0
{
1088
0
  cs_insn *insn;
1089
0
  struct cs_struct *handle = (struct cs_struct *)(uintptr_t)ud;
1090
1091
0
  insn = cs_mem_malloc(sizeof(cs_insn));
1092
0
  if (!insn) {
1093
    // insufficient memory
1094
0
    handle->errnum = CS_ERR_MEM;
1095
0
    return NULL;
1096
0
  } else {
1097
0
    if (handle->detail) {
1098
      // allocate memory for @detail pointer
1099
0
      insn->detail = cs_mem_malloc(sizeof(cs_detail));
1100
0
      if (insn->detail == NULL) { // insufficient memory
1101
0
        cs_mem_free(insn);
1102
0
        handle->errnum = CS_ERR_MEM;
1103
0
        return NULL;
1104
0
      }
1105
0
    } else
1106
0
      insn->detail = NULL;
1107
0
  }
1108
1109
0
  return insn;
1110
0
}
1111
1112
// iterator for instruction "single-stepping"
1113
CAPSTONE_EXPORT
1114
bool CAPSTONE_API cs_disasm_iter(csh ud, const uint8_t **code, size_t *size,
1115
    uint64_t *address, cs_insn *insn)
1116
0
{
1117
0
  if (*size == 0)
1118
0
    return false;
1119
1120
0
  struct cs_struct *handle;
1121
0
  uint16_t insn_size;
1122
0
  MCInst mci;
1123
0
  bool r;
1124
1125
0
  handle = (struct cs_struct *)(uintptr_t)ud;
1126
0
  if (!handle) {
1127
0
    return false;
1128
0
  }
1129
1130
0
  handle->errnum = CS_ERR_OK;
1131
1132
0
  MCInst_Init(&mci);
1133
0
  mci.csh = handle;
1134
1135
  // relative branches need to know the address & size of current insn
1136
0
  mci.address = *address;
1137
1138
  // save all the information for non-detailed mode
1139
0
  mci.flat_insn = insn;
1140
0
  mci.flat_insn->address = *address;
1141
#ifdef CAPSTONE_DIET
1142
  // zero out mnemonic & op_str
1143
  mci.flat_insn->mnemonic[0] = '\0';
1144
  mci.flat_insn->op_str[0] = '\0';
1145
#endif
1146
1147
0
  r = handle->disasm(ud, *code, *size, &mci, &insn_size, *address, handle->getinsn_info);
1148
0
  if (r) {
1149
0
    SStream ss;
1150
0
    SStream_Init(&ss);
1151
1152
0
    mci.flat_insn->size = insn_size;
1153
1154
    // map internal instruction opcode to public insn ID
1155
0
    handle->insn_id(handle, insn, mci.Opcode);
1156
1157
0
    handle->printer(&mci, &ss, handle->printer_info);
1158
1159
0
    fill_insn(handle, insn, ss.buffer, &mci, handle->post_printer, *code);
1160
1161
    // adjust for pseudo opcode (X86)
1162
0
    if (handle->arch == CS_ARCH_X86)
1163
0
      insn->id += mci.popcode_adjust;
1164
1165
0
    *code += insn_size;
1166
0
    *size -= insn_size;
1167
0
    *address += insn_size;
1168
0
  } else {   // encounter a broken instruction
1169
0
    size_t skipdata_bytes;
1170
1171
    // if there is no request to skip data, or remaining data is too small,
1172
    // then bail out
1173
0
    if (!handle->skipdata || handle->skipdata_size > *size)
1174
0
      return false;
1175
1176
0
    if (handle->skipdata_setup.callback) {
1177
0
      skipdata_bytes = handle->skipdata_setup.callback(*code, *size,
1178
0
          0, handle->skipdata_setup.user_data);
1179
0
      if (skipdata_bytes > *size)
1180
        // remaining data is not enough
1181
0
        return false;
1182
1183
0
      if (!skipdata_bytes)
1184
        // user requested not to skip data, so bail out
1185
0
        return false;
1186
0
    } else
1187
0
      skipdata_bytes = handle->skipdata_size;
1188
1189
    // we have to skip some amount of data, depending on arch & mode
1190
0
    insn->id = 0; // invalid ID for this "data" instruction
1191
0
    insn->address = *address;
1192
0
    insn->size = (uint16_t)MIN(skipdata_bytes, sizeof(insn->bytes));
1193
0
    memcpy(insn->bytes, *code,
1194
0
           MIN(skipdata_bytes, sizeof(insn->bytes)));
1195
#ifdef CAPSTONE_DIET
1196
    insn->mnemonic[0] = '\0';
1197
    insn->op_str[0] = '\0';
1198
#else
1199
0
    strncpy(insn->mnemonic, handle->skipdata_setup.mnemonic,
1200
0
        sizeof(insn->mnemonic) - 1);
1201
0
    skipdata_opstr(insn->op_str, *code, skipdata_bytes);
1202
0
#endif
1203
1204
0
    *code += skipdata_bytes;
1205
0
    *size -= skipdata_bytes;
1206
0
    *address += skipdata_bytes;
1207
0
  }
1208
1209
0
  return true;
1210
0
}
1211
1212
// return friendly name of register in a string
1213
CAPSTONE_EXPORT
1214
const char * CAPSTONE_API cs_reg_name(csh ud, unsigned int reg)
1215
2.29M
{
1216
2.29M
  struct cs_struct *handle = (struct cs_struct *)(uintptr_t)ud;
1217
1218
2.29M
  if (!handle || handle->reg_name == NULL) {
1219
0
    return NULL;
1220
0
  }
1221
1222
2.29M
  return handle->reg_name(ud, reg);
1223
2.29M
}
1224
1225
CAPSTONE_EXPORT
1226
const char * CAPSTONE_API cs_insn_name(csh ud, unsigned int insn)
1227
3.00M
{
1228
3.00M
  struct cs_struct *handle = (struct cs_struct *)(uintptr_t)ud;
1229
1230
3.00M
  if (!handle || handle->insn_name == NULL) {
1231
0
    return NULL;
1232
0
  }
1233
1234
3.00M
  return handle->insn_name(ud, insn);
1235
3.00M
}
1236
1237
CAPSTONE_EXPORT
1238
const char * CAPSTONE_API cs_group_name(csh ud, unsigned int group)
1239
2.47M
{
1240
2.47M
  struct cs_struct *handle = (struct cs_struct *)(uintptr_t)ud;
1241
1242
2.47M
  if (!handle || handle->group_name == NULL) {
1243
0
    return NULL;
1244
0
  }
1245
1246
2.47M
  return handle->group_name(ud, group);
1247
2.47M
}
1248
1249
CAPSTONE_EXPORT
1250
bool CAPSTONE_API cs_insn_group(csh ud, const cs_insn *insn, unsigned int group_id)
1251
0
{
1252
0
  struct cs_struct *handle;
1253
0
  if (!ud)
1254
0
    return false;
1255
1256
0
  handle = (struct cs_struct *)(uintptr_t)ud;
1257
1258
0
  if (!handle->detail) {
1259
0
    handle->errnum = CS_ERR_DETAIL;
1260
0
    return false;
1261
0
  }
1262
1263
0
  if (!insn->id) {
1264
0
    handle->errnum = CS_ERR_SKIPDATA;
1265
0
    return false;
1266
0
  }
1267
1268
0
  if (!insn->detail) {
1269
0
    handle->errnum = CS_ERR_DETAIL;
1270
0
    return false;
1271
0
  }
1272
1273
0
  return arr_exist8(insn->detail->groups, insn->detail->groups_count, group_id);
1274
0
}
1275
1276
CAPSTONE_EXPORT
1277
bool CAPSTONE_API cs_reg_read(csh ud, const cs_insn *insn, unsigned int reg_id)
1278
0
{
1279
0
  struct cs_struct *handle;
1280
0
  if (!ud)
1281
0
    return false;
1282
1283
0
  handle = (struct cs_struct *)(uintptr_t)ud;
1284
1285
0
  if (!handle->detail) {
1286
0
    handle->errnum = CS_ERR_DETAIL;
1287
0
    return false;
1288
0
  }
1289
1290
0
  if (!insn->id) {
1291
0
    handle->errnum = CS_ERR_SKIPDATA;
1292
0
    return false;
1293
0
  }
1294
1295
0
  if (!insn->detail) {
1296
0
    handle->errnum = CS_ERR_DETAIL;
1297
0
    return false;
1298
0
  }
1299
1300
0
  return arr_exist(insn->detail->regs_read, insn->detail->regs_read_count, reg_id);
1301
0
}
1302
1303
CAPSTONE_EXPORT
1304
bool CAPSTONE_API cs_reg_write(csh ud, const cs_insn *insn, unsigned int reg_id)
1305
1.21M
{
1306
1.21M
  struct cs_struct *handle;
1307
1.21M
  if (!ud)
1308
0
    return false;
1309
1310
1.21M
  handle = (struct cs_struct *)(uintptr_t)ud;
1311
1312
1.21M
  if (!handle->detail) {
1313
0
    handle->errnum = CS_ERR_DETAIL;
1314
0
    return false;
1315
0
  }
1316
1317
1.21M
  if (!insn->id) {
1318
0
    handle->errnum = CS_ERR_SKIPDATA;
1319
0
    return false;
1320
0
  }
1321
1322
1.21M
  if (!insn->detail) {
1323
0
    handle->errnum = CS_ERR_DETAIL;
1324
0
    return false;
1325
0
  }
1326
1327
1.21M
  return arr_exist(insn->detail->regs_write, insn->detail->regs_write_count, reg_id);
1328
1.21M
}
1329
1330
CAPSTONE_EXPORT
1331
int CAPSTONE_API cs_op_count(csh ud, const cs_insn *insn, unsigned int op_type)
1332
0
{
1333
0
  struct cs_struct *handle;
1334
0
  unsigned int count = 0, i;
1335
0
  if (!ud)
1336
0
    return -1;
1337
1338
0
  handle = (struct cs_struct *)(uintptr_t)ud;
1339
1340
0
  if (!handle->detail) {
1341
0
    handle->errnum = CS_ERR_DETAIL;
1342
0
    return -1;
1343
0
  }
1344
1345
0
  if (!insn->id) {
1346
0
    handle->errnum = CS_ERR_SKIPDATA;
1347
0
    return -1;
1348
0
  }
1349
1350
0
  if (!insn->detail) {
1351
0
    handle->errnum = CS_ERR_DETAIL;
1352
0
    return -1;
1353
0
  }
1354
1355
0
  handle->errnum = CS_ERR_OK;
1356
1357
0
  switch (handle->arch) {
1358
0
    default:
1359
0
      handle->errnum = CS_ERR_HANDLE;
1360
0
      return -1;
1361
0
    case CS_ARCH_ARM:
1362
0
      for (i = 0; i < insn->detail->arm.op_count; i++)
1363
0
        if (insn->detail->arm.operands[i].type == (arm_op_type)op_type)
1364
0
          count++;
1365
0
      break;
1366
0
    case CS_ARCH_ARM64:
1367
0
      for (i = 0; i < insn->detail->arm64.op_count; i++)
1368
0
        if (insn->detail->arm64.operands[i].type == (arm64_op_type)op_type)
1369
0
          count++;
1370
0
      break;
1371
0
    case CS_ARCH_X86:
1372
0
      for (i = 0; i < insn->detail->x86.op_count; i++)
1373
0
        if (insn->detail->x86.operands[i].type == (x86_op_type)op_type)
1374
0
          count++;
1375
0
      break;
1376
0
    case CS_ARCH_MIPS:
1377
0
      for (i = 0; i < insn->detail->mips.op_count; i++)
1378
0
        if (insn->detail->mips.operands[i].type == (mips_op_type)op_type)
1379
0
          count++;
1380
0
      break;
1381
0
    case CS_ARCH_PPC:
1382
0
      for (i = 0; i < insn->detail->ppc.op_count; i++)
1383
0
        if (insn->detail->ppc.operands[i].type == (ppc_op_type)op_type)
1384
0
          count++;
1385
0
      break;
1386
0
    case CS_ARCH_SPARC:
1387
0
      for (i = 0; i < insn->detail->sparc.op_count; i++)
1388
0
        if (insn->detail->sparc.operands[i].type == (sparc_op_type)op_type)
1389
0
          count++;
1390
0
      break;
1391
0
    case CS_ARCH_SYSZ:
1392
0
      for (i = 0; i < insn->detail->sysz.op_count; i++)
1393
0
        if (insn->detail->sysz.operands[i].type == (sysz_op_type)op_type)
1394
0
          count++;
1395
0
      break;
1396
0
    case CS_ARCH_XCORE:
1397
0
      for (i = 0; i < insn->detail->xcore.op_count; i++)
1398
0
        if (insn->detail->xcore.operands[i].type == (xcore_op_type)op_type)
1399
0
          count++;
1400
0
      break;
1401
0
    case CS_ARCH_M68K:
1402
0
      for (i = 0; i < insn->detail->m68k.op_count; i++)
1403
0
        if (insn->detail->m68k.operands[i].type == (m68k_op_type)op_type)
1404
0
          count++;
1405
0
      break;
1406
0
    case CS_ARCH_TMS320C64X:
1407
0
      for (i = 0; i < insn->detail->tms320c64x.op_count; i++)
1408
0
        if (insn->detail->tms320c64x.operands[i].type == (tms320c64x_op_type)op_type)
1409
0
          count++;
1410
0
      break;
1411
0
    case CS_ARCH_M680X:
1412
0
      for (i = 0; i < insn->detail->m680x.op_count; i++)
1413
0
        if (insn->detail->m680x.operands[i].type == (m680x_op_type)op_type)
1414
0
          count++;
1415
0
      break;
1416
0
    case CS_ARCH_EVM:
1417
0
      break;
1418
0
    case CS_ARCH_MOS65XX:
1419
0
      for (i = 0; i < insn->detail->mos65xx.op_count; i++)
1420
0
        if (insn->detail->mos65xx.operands[i].type == (mos65xx_op_type)op_type)
1421
0
          count++;
1422
0
      break;
1423
0
    case CS_ARCH_WASM:
1424
0
      for (i = 0; i < insn->detail->wasm.op_count; i++)
1425
0
        if (insn->detail->wasm.operands[i].type == (wasm_op_type)op_type)
1426
0
          count++;
1427
0
      break;
1428
0
    case CS_ARCH_BPF:
1429
0
      for (i = 0; i < insn->detail->bpf.op_count; i++)
1430
0
        if (insn->detail->bpf.operands[i].type == (bpf_op_type)op_type)
1431
0
          count++;
1432
0
      break;
1433
0
    case CS_ARCH_RISCV:
1434
0
      for (i = 0; i < insn->detail->riscv.op_count; i++)
1435
0
        if (insn->detail->riscv.operands[i].type == (riscv_op_type)op_type)
1436
0
          count++;
1437
0
      break;
1438
0
    case CS_ARCH_TRICORE:
1439
0
      for (i = 0; i < insn->detail->tricore.op_count; i++)
1440
0
        if (insn->detail->tricore.operands[i].type == (tricore_op_type)op_type)
1441
0
          count++;
1442
0
      break;
1443
0
  }
1444
1445
0
  return count;
1446
0
}
1447
1448
CAPSTONE_EXPORT
1449
int CAPSTONE_API cs_op_index(csh ud, const cs_insn *insn, unsigned int op_type,
1450
    unsigned int post)
1451
0
{
1452
0
  struct cs_struct *handle;
1453
0
  unsigned int count = 0, i;
1454
0
  if (!ud)
1455
0
    return -1;
1456
1457
0
  handle = (struct cs_struct *)(uintptr_t)ud;
1458
1459
0
  if (!handle->detail) {
1460
0
    handle->errnum = CS_ERR_DETAIL;
1461
0
    return -1;
1462
0
  }
1463
1464
0
  if (!insn->id) {
1465
0
    handle->errnum = CS_ERR_SKIPDATA;
1466
0
    return -1;
1467
0
  }
1468
1469
0
  if (!insn->detail) {
1470
0
    handle->errnum = CS_ERR_DETAIL;
1471
0
    return -1;
1472
0
  }
1473
1474
0
  handle->errnum = CS_ERR_OK;
1475
1476
0
  switch (handle->arch) {
1477
0
    default:
1478
0
      handle->errnum = CS_ERR_HANDLE;
1479
0
      return -1;
1480
0
    case CS_ARCH_ARM:
1481
0
      for (i = 0; i < insn->detail->arm.op_count; i++) {
1482
0
        if (insn->detail->arm.operands[i].type == (arm_op_type)op_type)
1483
0
          count++;
1484
0
        if (count == post)
1485
0
          return i;
1486
0
      }
1487
0
      break;
1488
0
    case CS_ARCH_ARM64:
1489
0
      for (i = 0; i < insn->detail->arm64.op_count; i++) {
1490
0
        if (insn->detail->arm64.operands[i].type == (arm64_op_type)op_type)
1491
0
          count++;
1492
0
        if (count == post)
1493
0
          return i;
1494
0
      }
1495
0
      break;
1496
0
    case CS_ARCH_X86:
1497
0
      for (i = 0; i < insn->detail->x86.op_count; i++) {
1498
0
        if (insn->detail->x86.operands[i].type == (x86_op_type)op_type)
1499
0
          count++;
1500
0
        if (count == post)
1501
0
          return i;
1502
0
      }
1503
0
      break;
1504
0
    case CS_ARCH_MIPS:
1505
0
      for (i = 0; i < insn->detail->mips.op_count; i++) {
1506
0
        if (insn->detail->mips.operands[i].type == (mips_op_type)op_type)
1507
0
          count++;
1508
0
        if (count == post)
1509
0
          return i;
1510
0
      }
1511
0
      break;
1512
0
    case CS_ARCH_PPC:
1513
0
      for (i = 0; i < insn->detail->ppc.op_count; i++) {
1514
0
        if (insn->detail->ppc.operands[i].type == (ppc_op_type)op_type)
1515
0
          count++;
1516
0
        if (count == post)
1517
0
          return i;
1518
0
      }
1519
0
      break;
1520
0
    case CS_ARCH_SPARC:
1521
0
      for (i = 0; i < insn->detail->sparc.op_count; i++) {
1522
0
        if (insn->detail->sparc.operands[i].type == (sparc_op_type)op_type)
1523
0
          count++;
1524
0
        if (count == post)
1525
0
          return i;
1526
0
      }
1527
0
      break;
1528
0
    case CS_ARCH_SYSZ:
1529
0
      for (i = 0; i < insn->detail->sysz.op_count; i++) {
1530
0
        if (insn->detail->sysz.operands[i].type == (sysz_op_type)op_type)
1531
0
          count++;
1532
0
        if (count == post)
1533
0
          return i;
1534
0
      }
1535
0
      break;
1536
0
    case CS_ARCH_XCORE:
1537
0
      for (i = 0; i < insn->detail->xcore.op_count; i++) {
1538
0
        if (insn->detail->xcore.operands[i].type == (xcore_op_type)op_type)
1539
0
          count++;
1540
0
        if (count == post)
1541
0
          return i;
1542
0
      }
1543
0
      break;
1544
0
    case CS_ARCH_TRICORE:
1545
0
      for (i = 0; i < insn->detail->tricore.op_count; i++) {
1546
0
        if (insn->detail->tricore.operands[i].type == (tricore_op_type)op_type)
1547
0
          count++;
1548
0
        if (count == post)
1549
0
          return i;
1550
0
      }
1551
0
      break;
1552
0
    case CS_ARCH_M68K:
1553
0
      for (i = 0; i < insn->detail->m68k.op_count; i++) {
1554
0
        if (insn->detail->m68k.operands[i].type == (m68k_op_type)op_type)
1555
0
          count++;
1556
0
        if (count == post)
1557
0
          return i;
1558
0
      }
1559
0
      break;
1560
0
    case CS_ARCH_TMS320C64X:
1561
0
      for (i = 0; i < insn->detail->tms320c64x.op_count; i++) {
1562
0
        if (insn->detail->tms320c64x.operands[i].type == (tms320c64x_op_type)op_type)
1563
0
          count++;
1564
0
        if (count == post)
1565
0
          return i;
1566
0
      }
1567
0
      break;
1568
0
    case CS_ARCH_M680X:
1569
0
      for (i = 0; i < insn->detail->m680x.op_count; i++) {
1570
0
        if (insn->detail->m680x.operands[i].type == (m680x_op_type)op_type)
1571
0
          count++;
1572
0
        if (count == post)
1573
0
          return i;
1574
0
      }
1575
0
      break;
1576
0
    case CS_ARCH_EVM:
1577
#if 0
1578
      for (i = 0; i < insn->detail->evm.op_count; i++) {
1579
        if (insn->detail->evm.operands[i].type == (evm_op_type)op_type)
1580
          count++;
1581
        if (count == post)
1582
          return i;
1583
      }
1584
#endif
1585
0
      break;
1586
0
    case CS_ARCH_MOS65XX:
1587
0
      for (i = 0; i < insn->detail->mos65xx.op_count; i++) {
1588
0
        if (insn->detail->mos65xx.operands[i].type == (mos65xx_op_type)op_type)
1589
0
          count++;
1590
0
        if (count == post)
1591
0
          return i;
1592
0
      }
1593
0
      break;
1594
0
    case CS_ARCH_WASM:
1595
0
      for (i = 0; i < insn->detail->wasm.op_count; i++) {
1596
0
        if (insn->detail->wasm.operands[i].type == (wasm_op_type)op_type)
1597
0
          count++;
1598
0
        if (count == post)
1599
0
          return i;
1600
0
      }
1601
0
      break;
1602
0
    case CS_ARCH_BPF:
1603
0
      for (i = 0; i < insn->detail->bpf.op_count; i++) {
1604
0
        if (insn->detail->bpf.operands[i].type == (bpf_op_type)op_type)
1605
0
          count++;
1606
0
        if (count == post)
1607
0
          return i;
1608
0
      }
1609
0
      break;
1610
0
    case CS_ARCH_RISCV:
1611
0
      for (i = 0; i < insn->detail->riscv.op_count; i++) {
1612
0
        if (insn->detail->riscv.operands[i].type == (riscv_op_type)op_type)
1613
0
          count++;
1614
0
        if (count == post)
1615
0
          return i;
1616
0
      }
1617
0
      break;
1618
0
    case CS_ARCH_SH:
1619
0
      for (i = 0; i < insn->detail->sh.op_count; i++) {
1620
0
        if (insn->detail->sh.operands[i].type == (sh_op_type)op_type)
1621
0
          count++;
1622
0
        if (count == post)
1623
0
          return i;
1624
0
      }
1625
0
      break;
1626
0
  }
1627
1628
0
  return -1;
1629
0
}
1630
1631
CAPSTONE_EXPORT
1632
cs_err CAPSTONE_API cs_regs_access(csh ud, const cs_insn *insn,
1633
    cs_regs regs_read, uint8_t *regs_read_count,
1634
    cs_regs regs_write, uint8_t *regs_write_count)
1635
0
{
1636
0
  struct cs_struct *handle;
1637
1638
0
  if (!ud)
1639
0
    return -1;
1640
1641
0
  handle = (struct cs_struct *)(uintptr_t)ud;
1642
1643
#ifdef CAPSTONE_DIET
1644
  // This API does not work in DIET mode
1645
  handle->errnum = CS_ERR_DIET;
1646
  return CS_ERR_DIET;
1647
#else
1648
0
  if (!handle->detail) {
1649
0
    handle->errnum = CS_ERR_DETAIL;
1650
0
    return CS_ERR_DETAIL;
1651
0
  }
1652
1653
0
  if (!insn->id) {
1654
0
    handle->errnum = CS_ERR_SKIPDATA;
1655
0
    return CS_ERR_SKIPDATA;
1656
0
  }
1657
1658
0
  if (!insn->detail) {
1659
0
    handle->errnum = CS_ERR_DETAIL;
1660
0
    return CS_ERR_DETAIL;
1661
0
  }
1662
1663
0
  if (handle->reg_access) {
1664
0
    handle->reg_access(insn, regs_read, regs_read_count, regs_write, regs_write_count);
1665
0
  } else {
1666
    // this arch is unsupported yet
1667
0
    handle->errnum = CS_ERR_ARCH;
1668
0
    return CS_ERR_ARCH;
1669
0
  }
1670
1671
0
  return CS_ERR_OK;
1672
0
#endif
1673
0
}