xref: /openbsd-src/gnu/llvm/llvm/tools/llvm-c-test/disassemble.c (revision 09467b48e8bc8b4905716062da846024139afbf2)
1*09467b48Spatrick /*===-- disassemble.c - tool for testing libLLVM and llvm-c API -----------===*\
2*09467b48Spatrick |*                                                                            *|
3*09467b48Spatrick |* Part of the LLVM Project, under the Apache License v2.0 with LLVM          *|
4*09467b48Spatrick |* Exceptions.                                                                *|
5*09467b48Spatrick |* See https://llvm.org/LICENSE.txt for license information.                  *|
6*09467b48Spatrick |* SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception                    *|
7*09467b48Spatrick |*                                                                            *|
8*09467b48Spatrick |*===----------------------------------------------------------------------===*|
9*09467b48Spatrick |*                                                                            *|
10*09467b48Spatrick |* This file implements the --disassemble command in llvm-c-test.             *|
11*09467b48Spatrick |* --disassemble reads lines from stdin, parses them as a triple and hex      *|
12*09467b48Spatrick |*  machine code, and prints disassembly of the machine code.                 *|
13*09467b48Spatrick |*                                                                            *|
14*09467b48Spatrick \*===----------------------------------------------------------------------===*/
15*09467b48Spatrick 
16*09467b48Spatrick #include "llvm-c-test.h"
17*09467b48Spatrick #include "llvm-c/Disassembler.h"
18*09467b48Spatrick #include "llvm-c/Target.h"
19*09467b48Spatrick #include <stdio.h>
20*09467b48Spatrick #include <stdlib.h>
21*09467b48Spatrick #include <string.h>
22*09467b48Spatrick 
pprint(int pos,unsigned char * buf,int len,const char * disasm)23*09467b48Spatrick static void pprint(int pos, unsigned char *buf, int len, const char *disasm) {
24*09467b48Spatrick   int i;
25*09467b48Spatrick   printf("%04x:  ", pos);
26*09467b48Spatrick   for (i = 0; i < 8; i++) {
27*09467b48Spatrick     if (i < len) {
28*09467b48Spatrick       printf("%02x ", buf[i]);
29*09467b48Spatrick     } else {
30*09467b48Spatrick       printf("   ");
31*09467b48Spatrick     }
32*09467b48Spatrick   }
33*09467b48Spatrick 
34*09467b48Spatrick   printf("   %s\n", disasm);
35*09467b48Spatrick }
36*09467b48Spatrick 
do_disassemble(const char * triple,const char * features,unsigned char * buf,int siz)37*09467b48Spatrick static void do_disassemble(const char *triple, const char *features,
38*09467b48Spatrick                            unsigned char *buf, int siz) {
39*09467b48Spatrick   LLVMDisasmContextRef D = LLVMCreateDisasmCPUFeatures(triple, "", features,
40*09467b48Spatrick                                                        NULL, 0, NULL, NULL);
41*09467b48Spatrick   char outline[1024];
42*09467b48Spatrick   int pos;
43*09467b48Spatrick 
44*09467b48Spatrick   if (!D) {
45*09467b48Spatrick     printf("ERROR: Couldn't create disassembler for triple %s\n", triple);
46*09467b48Spatrick     return;
47*09467b48Spatrick   }
48*09467b48Spatrick 
49*09467b48Spatrick   pos = 0;
50*09467b48Spatrick   while (pos < siz) {
51*09467b48Spatrick     size_t l = LLVMDisasmInstruction(D, buf + pos, siz - pos, 0, outline,
52*09467b48Spatrick                                      sizeof(outline));
53*09467b48Spatrick     if (!l) {
54*09467b48Spatrick       pprint(pos, buf + pos, 1, "\t???");
55*09467b48Spatrick       pos++;
56*09467b48Spatrick     } else {
57*09467b48Spatrick       pprint(pos, buf + pos, l, outline);
58*09467b48Spatrick       pos += l;
59*09467b48Spatrick     }
60*09467b48Spatrick   }
61*09467b48Spatrick 
62*09467b48Spatrick   LLVMDisasmDispose(D);
63*09467b48Spatrick }
64*09467b48Spatrick 
handle_line(char ** tokens,int ntokens)65*09467b48Spatrick static void handle_line(char **tokens, int ntokens) {
66*09467b48Spatrick   unsigned char disbuf[128];
67*09467b48Spatrick   size_t disbuflen = 0;
68*09467b48Spatrick   const char *triple = tokens[0];
69*09467b48Spatrick   const char *features = tokens[1];
70*09467b48Spatrick   int i;
71*09467b48Spatrick 
72*09467b48Spatrick   printf("triple: %s, features: %s\n", triple, features);
73*09467b48Spatrick   if (!strcmp(features, "NULL"))
74*09467b48Spatrick     features = "";
75*09467b48Spatrick 
76*09467b48Spatrick   for (i = 2; i < ntokens; i++) {
77*09467b48Spatrick     disbuf[disbuflen++] = strtol(tokens[i], NULL, 16);
78*09467b48Spatrick     if (disbuflen >= sizeof(disbuf)) {
79*09467b48Spatrick       fprintf(stderr, "Warning: Too long line, truncating\n");
80*09467b48Spatrick       break;
81*09467b48Spatrick     }
82*09467b48Spatrick   }
83*09467b48Spatrick   do_disassemble(triple, features, disbuf, disbuflen);
84*09467b48Spatrick }
85*09467b48Spatrick 
llvm_disassemble(void)86*09467b48Spatrick int llvm_disassemble(void) {
87*09467b48Spatrick   LLVMInitializeAllTargetInfos();
88*09467b48Spatrick   LLVMInitializeAllTargetMCs();
89*09467b48Spatrick   LLVMInitializeAllDisassemblers();
90*09467b48Spatrick 
91*09467b48Spatrick   llvm_tokenize_stdin(handle_line);
92*09467b48Spatrick 
93*09467b48Spatrick   return 0;
94*09467b48Spatrick }
95