xref: /aosp_15_r20/external/llvm/tools/llvm-dis/llvm-dis.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===-- llvm-dis.cpp - The low-level LLVM disassembler --------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker //
10*9880d681SAndroid Build Coastguard Worker // This utility may be invoked in the following manner:
11*9880d681SAndroid Build Coastguard Worker //  llvm-dis [options]      - Read LLVM bitcode from stdin, write asm to stdout
12*9880d681SAndroid Build Coastguard Worker //  llvm-dis [options] x.bc - Read LLVM bitcode from the x.bc file, write asm
13*9880d681SAndroid Build Coastguard Worker //                            to the x.ll file.
14*9880d681SAndroid Build Coastguard Worker //  Options:
15*9880d681SAndroid Build Coastguard Worker //      --help   - Output information about command line switches
16*9880d681SAndroid Build Coastguard Worker //
17*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
18*9880d681SAndroid Build Coastguard Worker 
19*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/LLVMContext.h"
20*9880d681SAndroid Build Coastguard Worker #include "llvm/Bitcode/ReaderWriter.h"
21*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/AssemblyAnnotationWriter.h"
22*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/DebugInfo.h"
23*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/DiagnosticInfo.h"
24*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/DiagnosticPrinter.h"
25*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/IntrinsicInst.h"
26*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/Module.h"
27*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/Type.h"
28*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/CommandLine.h"
29*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/DataStream.h"
30*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Error.h"
31*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/FileSystem.h"
32*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/FormattedStream.h"
33*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ManagedStatic.h"
34*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/MemoryBuffer.h"
35*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/PrettyStackTrace.h"
36*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Signals.h"
37*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ToolOutputFile.h"
38*9880d681SAndroid Build Coastguard Worker #include <system_error>
39*9880d681SAndroid Build Coastguard Worker using namespace llvm;
40*9880d681SAndroid Build Coastguard Worker 
41*9880d681SAndroid Build Coastguard Worker static cl::opt<std::string>
42*9880d681SAndroid Build Coastguard Worker InputFilename(cl::Positional, cl::desc("<input bitcode>"), cl::init("-"));
43*9880d681SAndroid Build Coastguard Worker 
44*9880d681SAndroid Build Coastguard Worker static cl::opt<std::string>
45*9880d681SAndroid Build Coastguard Worker OutputFilename("o", cl::desc("Override output filename"),
46*9880d681SAndroid Build Coastguard Worker                cl::value_desc("filename"));
47*9880d681SAndroid Build Coastguard Worker 
48*9880d681SAndroid Build Coastguard Worker static cl::opt<bool>
49*9880d681SAndroid Build Coastguard Worker Force("f", cl::desc("Enable binary output on terminals"));
50*9880d681SAndroid Build Coastguard Worker 
51*9880d681SAndroid Build Coastguard Worker static cl::opt<bool>
52*9880d681SAndroid Build Coastguard Worker DontPrint("disable-output", cl::desc("Don't output the .ll file"), cl::Hidden);
53*9880d681SAndroid Build Coastguard Worker 
54*9880d681SAndroid Build Coastguard Worker static cl::opt<bool>
55*9880d681SAndroid Build Coastguard Worker ShowAnnotations("show-annotations",
56*9880d681SAndroid Build Coastguard Worker                 cl::desc("Add informational comments to the .ll file"));
57*9880d681SAndroid Build Coastguard Worker 
58*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> PreserveAssemblyUseListOrder(
59*9880d681SAndroid Build Coastguard Worker     "preserve-ll-uselistorder",
60*9880d681SAndroid Build Coastguard Worker     cl::desc("Preserve use-list order when writing LLVM assembly."),
61*9880d681SAndroid Build Coastguard Worker     cl::init(false), cl::Hidden);
62*9880d681SAndroid Build Coastguard Worker 
63*9880d681SAndroid Build Coastguard Worker static cl::opt<bool>
64*9880d681SAndroid Build Coastguard Worker     MaterializeMetadata("materialize-metadata",
65*9880d681SAndroid Build Coastguard Worker                         cl::desc("Load module without materializing metadata, "
66*9880d681SAndroid Build Coastguard Worker                                  "then materialize only the metadata"));
67*9880d681SAndroid Build Coastguard Worker 
68*9880d681SAndroid Build Coastguard Worker namespace {
69*9880d681SAndroid Build Coastguard Worker 
printDebugLoc(const DebugLoc & DL,formatted_raw_ostream & OS)70*9880d681SAndroid Build Coastguard Worker static void printDebugLoc(const DebugLoc &DL, formatted_raw_ostream &OS) {
71*9880d681SAndroid Build Coastguard Worker   OS << DL.getLine() << ":" << DL.getCol();
72*9880d681SAndroid Build Coastguard Worker   if (DILocation *IDL = DL.getInlinedAt()) {
73*9880d681SAndroid Build Coastguard Worker     OS << "@";
74*9880d681SAndroid Build Coastguard Worker     printDebugLoc(IDL, OS);
75*9880d681SAndroid Build Coastguard Worker   }
76*9880d681SAndroid Build Coastguard Worker }
77*9880d681SAndroid Build Coastguard Worker class CommentWriter : public AssemblyAnnotationWriter {
78*9880d681SAndroid Build Coastguard Worker public:
emitFunctionAnnot(const Function * F,formatted_raw_ostream & OS)79*9880d681SAndroid Build Coastguard Worker   void emitFunctionAnnot(const Function *F,
80*9880d681SAndroid Build Coastguard Worker                          formatted_raw_ostream &OS) override {
81*9880d681SAndroid Build Coastguard Worker     OS << "; [#uses=" << F->getNumUses() << ']';  // Output # uses
82*9880d681SAndroid Build Coastguard Worker     OS << '\n';
83*9880d681SAndroid Build Coastguard Worker   }
printInfoComment(const Value & V,formatted_raw_ostream & OS)84*9880d681SAndroid Build Coastguard Worker   void printInfoComment(const Value &V, formatted_raw_ostream &OS) override {
85*9880d681SAndroid Build Coastguard Worker     bool Padded = false;
86*9880d681SAndroid Build Coastguard Worker     if (!V.getType()->isVoidTy()) {
87*9880d681SAndroid Build Coastguard Worker       OS.PadToColumn(50);
88*9880d681SAndroid Build Coastguard Worker       Padded = true;
89*9880d681SAndroid Build Coastguard Worker       // Output # uses and type
90*9880d681SAndroid Build Coastguard Worker       OS << "; [#uses=" << V.getNumUses() << " type=" << *V.getType() << "]";
91*9880d681SAndroid Build Coastguard Worker     }
92*9880d681SAndroid Build Coastguard Worker     if (const Instruction *I = dyn_cast<Instruction>(&V)) {
93*9880d681SAndroid Build Coastguard Worker       if (const DebugLoc &DL = I->getDebugLoc()) {
94*9880d681SAndroid Build Coastguard Worker         if (!Padded) {
95*9880d681SAndroid Build Coastguard Worker           OS.PadToColumn(50);
96*9880d681SAndroid Build Coastguard Worker           Padded = true;
97*9880d681SAndroid Build Coastguard Worker           OS << ";";
98*9880d681SAndroid Build Coastguard Worker         }
99*9880d681SAndroid Build Coastguard Worker         OS << " [debug line = ";
100*9880d681SAndroid Build Coastguard Worker         printDebugLoc(DL,OS);
101*9880d681SAndroid Build Coastguard Worker         OS << "]";
102*9880d681SAndroid Build Coastguard Worker       }
103*9880d681SAndroid Build Coastguard Worker       if (const DbgDeclareInst *DDI = dyn_cast<DbgDeclareInst>(I)) {
104*9880d681SAndroid Build Coastguard Worker         if (!Padded) {
105*9880d681SAndroid Build Coastguard Worker           OS.PadToColumn(50);
106*9880d681SAndroid Build Coastguard Worker           OS << ";";
107*9880d681SAndroid Build Coastguard Worker         }
108*9880d681SAndroid Build Coastguard Worker         OS << " [debug variable = " << DDI->getVariable()->getName() << "]";
109*9880d681SAndroid Build Coastguard Worker       }
110*9880d681SAndroid Build Coastguard Worker       else if (const DbgValueInst *DVI = dyn_cast<DbgValueInst>(I)) {
111*9880d681SAndroid Build Coastguard Worker         if (!Padded) {
112*9880d681SAndroid Build Coastguard Worker           OS.PadToColumn(50);
113*9880d681SAndroid Build Coastguard Worker           OS << ";";
114*9880d681SAndroid Build Coastguard Worker         }
115*9880d681SAndroid Build Coastguard Worker         OS << " [debug variable = " << DVI->getVariable()->getName() << "]";
116*9880d681SAndroid Build Coastguard Worker       }
117*9880d681SAndroid Build Coastguard Worker     }
118*9880d681SAndroid Build Coastguard Worker   }
119*9880d681SAndroid Build Coastguard Worker };
120*9880d681SAndroid Build Coastguard Worker 
121*9880d681SAndroid Build Coastguard Worker } // end anon namespace
122*9880d681SAndroid Build Coastguard Worker 
diagnosticHandler(const DiagnosticInfo & DI,void * Context)123*9880d681SAndroid Build Coastguard Worker static void diagnosticHandler(const DiagnosticInfo &DI, void *Context) {
124*9880d681SAndroid Build Coastguard Worker   raw_ostream &OS = errs();
125*9880d681SAndroid Build Coastguard Worker   OS << (char *)Context << ": ";
126*9880d681SAndroid Build Coastguard Worker   switch (DI.getSeverity()) {
127*9880d681SAndroid Build Coastguard Worker   case DS_Error: OS << "error: "; break;
128*9880d681SAndroid Build Coastguard Worker   case DS_Warning: OS << "warning: "; break;
129*9880d681SAndroid Build Coastguard Worker   case DS_Remark: OS << "remark: "; break;
130*9880d681SAndroid Build Coastguard Worker   case DS_Note: OS << "note: "; break;
131*9880d681SAndroid Build Coastguard Worker   }
132*9880d681SAndroid Build Coastguard Worker 
133*9880d681SAndroid Build Coastguard Worker   DiagnosticPrinterRawOStream DP(OS);
134*9880d681SAndroid Build Coastguard Worker   DI.print(DP);
135*9880d681SAndroid Build Coastguard Worker   OS << '\n';
136*9880d681SAndroid Build Coastguard Worker 
137*9880d681SAndroid Build Coastguard Worker   if (DI.getSeverity() == DS_Error)
138*9880d681SAndroid Build Coastguard Worker     exit(1);
139*9880d681SAndroid Build Coastguard Worker }
140*9880d681SAndroid Build Coastguard Worker 
openInputFile(LLVMContext & Context)141*9880d681SAndroid Build Coastguard Worker static Expected<std::unique_ptr<Module>> openInputFile(LLVMContext &Context) {
142*9880d681SAndroid Build Coastguard Worker   if (MaterializeMetadata) {
143*9880d681SAndroid Build Coastguard Worker     ErrorOr<std::unique_ptr<MemoryBuffer>> MBOrErr =
144*9880d681SAndroid Build Coastguard Worker         MemoryBuffer::getFileOrSTDIN(InputFilename);
145*9880d681SAndroid Build Coastguard Worker     if (!MBOrErr)
146*9880d681SAndroid Build Coastguard Worker       return errorCodeToError(MBOrErr.getError());
147*9880d681SAndroid Build Coastguard Worker     ErrorOr<std::unique_ptr<Module>> MOrErr =
148*9880d681SAndroid Build Coastguard Worker         getLazyBitcodeModule(std::move(*MBOrErr), Context,
149*9880d681SAndroid Build Coastguard Worker                              /*ShouldLazyLoadMetadata=*/true);
150*9880d681SAndroid Build Coastguard Worker     if (!MOrErr)
151*9880d681SAndroid Build Coastguard Worker       return errorCodeToError(MOrErr.getError());
152*9880d681SAndroid Build Coastguard Worker     (*MOrErr)->materializeMetadata();
153*9880d681SAndroid Build Coastguard Worker     return std::move(*MOrErr);
154*9880d681SAndroid Build Coastguard Worker   } else {
155*9880d681SAndroid Build Coastguard Worker     std::string ErrorMessage;
156*9880d681SAndroid Build Coastguard Worker     std::unique_ptr<DataStreamer> Streamer =
157*9880d681SAndroid Build Coastguard Worker         getDataFileStreamer(InputFilename, &ErrorMessage);
158*9880d681SAndroid Build Coastguard Worker     if (!Streamer)
159*9880d681SAndroid Build Coastguard Worker       return make_error<StringError>(ErrorMessage, inconvertibleErrorCode());
160*9880d681SAndroid Build Coastguard Worker     std::string DisplayFilename;
161*9880d681SAndroid Build Coastguard Worker     if (InputFilename == "-")
162*9880d681SAndroid Build Coastguard Worker       DisplayFilename = "<stdin>";
163*9880d681SAndroid Build Coastguard Worker     else
164*9880d681SAndroid Build Coastguard Worker       DisplayFilename = InputFilename;
165*9880d681SAndroid Build Coastguard Worker     ErrorOr<std::unique_ptr<Module>> MOrErr =
166*9880d681SAndroid Build Coastguard Worker         getStreamedBitcodeModule(DisplayFilename, std::move(Streamer), Context);
167*9880d681SAndroid Build Coastguard Worker     (*MOrErr)->materializeAll();
168*9880d681SAndroid Build Coastguard Worker     return std::move(*MOrErr);
169*9880d681SAndroid Build Coastguard Worker   }
170*9880d681SAndroid Build Coastguard Worker }
171*9880d681SAndroid Build Coastguard Worker 
main(int argc,char ** argv)172*9880d681SAndroid Build Coastguard Worker int main(int argc, char **argv) {
173*9880d681SAndroid Build Coastguard Worker   // Print a stack trace if we signal out.
174*9880d681SAndroid Build Coastguard Worker   sys::PrintStackTraceOnErrorSignal(argv[0]);
175*9880d681SAndroid Build Coastguard Worker   PrettyStackTraceProgram X(argc, argv);
176*9880d681SAndroid Build Coastguard Worker 
177*9880d681SAndroid Build Coastguard Worker   LLVMContext Context;
178*9880d681SAndroid Build Coastguard Worker   llvm_shutdown_obj Y;  // Call llvm_shutdown() on exit.
179*9880d681SAndroid Build Coastguard Worker 
180*9880d681SAndroid Build Coastguard Worker   Context.setDiagnosticHandler(diagnosticHandler, argv[0]);
181*9880d681SAndroid Build Coastguard Worker 
182*9880d681SAndroid Build Coastguard Worker   cl::ParseCommandLineOptions(argc, argv, "llvm .bc -> .ll disassembler\n");
183*9880d681SAndroid Build Coastguard Worker 
184*9880d681SAndroid Build Coastguard Worker   Expected<std::unique_ptr<Module>> MOrErr = openInputFile(Context);
185*9880d681SAndroid Build Coastguard Worker   if (!MOrErr) {
186*9880d681SAndroid Build Coastguard Worker     handleAllErrors(MOrErr.takeError(), [&](ErrorInfoBase &EIB) {
187*9880d681SAndroid Build Coastguard Worker       errs() << argv[0] << ": ";
188*9880d681SAndroid Build Coastguard Worker       EIB.log(errs());
189*9880d681SAndroid Build Coastguard Worker       errs() << '\n';
190*9880d681SAndroid Build Coastguard Worker     });
191*9880d681SAndroid Build Coastguard Worker     return 1;
192*9880d681SAndroid Build Coastguard Worker   }
193*9880d681SAndroid Build Coastguard Worker   std::unique_ptr<Module> M = std::move(*MOrErr);
194*9880d681SAndroid Build Coastguard Worker 
195*9880d681SAndroid Build Coastguard Worker   // Just use stdout.  We won't actually print anything on it.
196*9880d681SAndroid Build Coastguard Worker   if (DontPrint)
197*9880d681SAndroid Build Coastguard Worker     OutputFilename = "-";
198*9880d681SAndroid Build Coastguard Worker 
199*9880d681SAndroid Build Coastguard Worker   if (OutputFilename.empty()) { // Unspecified output, infer it.
200*9880d681SAndroid Build Coastguard Worker     if (InputFilename == "-") {
201*9880d681SAndroid Build Coastguard Worker       OutputFilename = "-";
202*9880d681SAndroid Build Coastguard Worker     } else {
203*9880d681SAndroid Build Coastguard Worker       StringRef IFN = InputFilename;
204*9880d681SAndroid Build Coastguard Worker       OutputFilename = (IFN.endswith(".bc") ? IFN.drop_back(3) : IFN).str();
205*9880d681SAndroid Build Coastguard Worker       OutputFilename += ".ll";
206*9880d681SAndroid Build Coastguard Worker     }
207*9880d681SAndroid Build Coastguard Worker   }
208*9880d681SAndroid Build Coastguard Worker 
209*9880d681SAndroid Build Coastguard Worker   std::error_code EC;
210*9880d681SAndroid Build Coastguard Worker   std::unique_ptr<tool_output_file> Out(
211*9880d681SAndroid Build Coastguard Worker       new tool_output_file(OutputFilename, EC, sys::fs::F_None));
212*9880d681SAndroid Build Coastguard Worker   if (EC) {
213*9880d681SAndroid Build Coastguard Worker     errs() << EC.message() << '\n';
214*9880d681SAndroid Build Coastguard Worker     return 1;
215*9880d681SAndroid Build Coastguard Worker   }
216*9880d681SAndroid Build Coastguard Worker 
217*9880d681SAndroid Build Coastguard Worker   std::unique_ptr<AssemblyAnnotationWriter> Annotator;
218*9880d681SAndroid Build Coastguard Worker   if (ShowAnnotations)
219*9880d681SAndroid Build Coastguard Worker     Annotator.reset(new CommentWriter());
220*9880d681SAndroid Build Coastguard Worker 
221*9880d681SAndroid Build Coastguard Worker   // All that llvm-dis does is write the assembly to a file.
222*9880d681SAndroid Build Coastguard Worker   if (!DontPrint)
223*9880d681SAndroid Build Coastguard Worker     M->print(Out->os(), Annotator.get(), PreserveAssemblyUseListOrder);
224*9880d681SAndroid Build Coastguard Worker 
225*9880d681SAndroid Build Coastguard Worker   // Declare success.
226*9880d681SAndroid Build Coastguard Worker   Out->keep();
227*9880d681SAndroid Build Coastguard Worker 
228*9880d681SAndroid Build Coastguard Worker   return 0;
229*9880d681SAndroid Build Coastguard Worker }
230