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