blob: 0df7328ffc348ca9dec058e6ee1c52886d4b9fbb [file] [log] [blame]
Chris Lattner8f71f042003-10-20 17:58:43 +00001//===-- llvm-dis.cpp - The low-level LLVM disassembler --------------------===//
Misha Brukman650ba8e2005-04-22 00:00:37 +00002//
John Criswell09344dc2003-10-20 17:47:21 +00003// The LLVM Compiler Infrastructure
4//
Chris Lattner345353d2007-12-29 20:44:31 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Misha Brukman650ba8e2005-04-22 00:00:37 +00007//
John Criswell09344dc2003-10-20 17:47:21 +00008//===----------------------------------------------------------------------===//
Chris Lattner2f7c9632001-06-06 20:29:01 +00009//
10// This utility may be invoked in the following manner:
Gabor Greife16561c2007-07-05 17:07:56 +000011// llvm-dis [options] - Read LLVM bitcode from stdin, write asm to stdout
12// llvm-dis [options] x.bc - Read LLVM bitcode from the x.bc file, write asm
Misha Brukmanf12549d2003-08-28 21:34:13 +000013// to the x.ll file.
Chris Lattnerf284ac52001-06-13 19:55:41 +000014// Options:
15// --help - Output information about command line switches
Chris Lattner2f7c9632001-06-06 20:29:01 +000016//
Chris Lattner6d56c6b2001-10-13 07:06:57 +000017//===----------------------------------------------------------------------===//
Chris Lattner2f7c9632001-06-06 20:29:01 +000018
Chandler Carruth9fb823b2013-01-02 11:36:10 +000019#include "llvm/IR/LLVMContext.h"
Chandler Carruth4d88a1c2012-12-04 10:44:52 +000020#include "llvm/Bitcode/ReaderWriter.h"
Chandler Carruth9aca9182014-01-07 12:34:26 +000021#include "llvm/IR/AssemblyAnnotationWriter.h"
Chandler Carruth9a4c9e52014-03-06 00:46:21 +000022#include "llvm/IR/DebugInfo.h"
Chandler Carruth9fb823b2013-01-02 11:36:10 +000023#include "llvm/IR/IntrinsicInst.h"
24#include "llvm/IR/Module.h"
25#include "llvm/IR/Type.h"
Reid Spencer7c16caa2004-09-01 22:55:40 +000026#include "llvm/Support/CommandLine.h"
Derek Schuff8b2dcad2012-02-06 22:30:29 +000027#include "llvm/Support/DataStream.h"
Benjamin Kramerd59664f2014-04-29 23:26:49 +000028#include "llvm/Support/FileSystem.h"
Chris Lattner7f2f0932010-09-02 23:21:44 +000029#include "llvm/Support/FormattedStream.h"
Chris Lattner76d46322006-12-06 01:18:01 +000030#include "llvm/Support/ManagedStatic.h"
Chris Lattner6694f602007-04-29 07:54:31 +000031#include "llvm/Support/MemoryBuffer.h"
Chris Lattnere3fc2d12009-03-06 05:34:10 +000032#include "llvm/Support/PrettyStackTrace.h"
Michael J. Spencer447762d2010-11-29 18:16:10 +000033#include "llvm/Support/Signals.h"
Chandler Carruth4d88a1c2012-12-04 10:44:52 +000034#include "llvm/Support/ToolOutputFile.h"
Michael J. Spencer7b6fef82010-12-09 17:36:48 +000035#include "llvm/Support/system_error.h"
Brian Gaeke960707c2003-11-11 22:41:34 +000036using namespace llvm;
37
Chris Lattner64a67272002-07-25 16:31:09 +000038static cl::opt<std::string>
Gabor Greife16561c2007-07-05 17:07:56 +000039InputFilename(cl::Positional, cl::desc("<input bitcode>"), cl::init("-"));
Chris Lattnerf5cad152002-07-22 02:10:13 +000040
Chris Lattner64a67272002-07-25 16:31:09 +000041static cl::opt<std::string>
Misha Brukman650ba8e2005-04-22 00:00:37 +000042OutputFilename("o", cl::desc("Override output filename"),
Chris Lattnerf5cad152002-07-22 02:10:13 +000043 cl::value_desc("filename"));
44
45static cl::opt<bool>
Dan Gohman61a87962009-08-25 15:34:52 +000046Force("f", cl::desc("Enable binary output on terminals"));
Chris Lattnerf5cad152002-07-22 02:10:13 +000047
Chris Lattner216835d2007-02-07 04:39:35 +000048static cl::opt<bool>
49DontPrint("disable-output", cl::desc("Don't output the .ll file"), cl::Hidden);
50
Chris Lattner7f2f0932010-09-02 23:21:44 +000051static cl::opt<bool>
52ShowAnnotations("show-annotations",
53 cl::desc("Add informational comments to the .ll file"));
54
55namespace {
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +000056
Devang Patel982efb52011-03-11 18:07:33 +000057static void printDebugLoc(const DebugLoc &DL, formatted_raw_ostream &OS) {
58 OS << DL.getLine() << ":" << DL.getCol();
59 if (MDNode *N = DL.getInlinedAt(getGlobalContext())) {
60 DebugLoc IDL = DebugLoc::getFromDILocation(N);
61 if (!IDL.isUnknown()) {
62 OS << "@";
63 printDebugLoc(IDL,OS);
64 }
65 }
66}
Chris Lattner7f2f0932010-09-02 23:21:44 +000067class CommentWriter : public AssemblyAnnotationWriter {
68public:
69 void emitFunctionAnnot(const Function *F,
Craig Toppere56917c2014-03-08 08:27:28 +000070 formatted_raw_ostream &OS) override {
Chris Lattner7f2f0932010-09-02 23:21:44 +000071 OS << "; [#uses=" << F->getNumUses() << ']'; // Output # uses
72 OS << '\n';
73 }
Craig Toppere56917c2014-03-08 08:27:28 +000074 void printInfoComment(const Value &V, formatted_raw_ostream &OS) override {
Devang Patel982efb52011-03-11 18:07:33 +000075 bool Padded = false;
76 if (!V.getType()->isVoidTy()) {
77 OS.PadToColumn(50);
78 Padded = true;
Chris Lattner0f214eb2011-06-18 21:18:23 +000079 OS << "; [#uses=" << V.getNumUses() << " type=" << *V.getType() << "]"; // Output # uses and type
Devang Patel982efb52011-03-11 18:07:33 +000080 }
81 if (const Instruction *I = dyn_cast<Instruction>(&V)) {
82 const DebugLoc &DL = I->getDebugLoc();
83 if (!DL.isUnknown()) {
84 if (!Padded) {
85 OS.PadToColumn(50);
86 Padded = true;
87 OS << ";";
88 }
89 OS << " [debug line = ";
90 printDebugLoc(DL,OS);
91 OS << "]";
92 }
93 if (const DbgDeclareInst *DDI = dyn_cast<DbgDeclareInst>(I)) {
94 DIVariable Var(DDI->getVariable());
95 if (!Padded) {
96 OS.PadToColumn(50);
Devang Patel982efb52011-03-11 18:07:33 +000097 OS << ";";
98 }
99 OS << " [debug variable = " << Var.getName() << "]";
100 }
101 else if (const DbgValueInst *DVI = dyn_cast<DbgValueInst>(I)) {
102 DIVariable Var(DVI->getVariable());
103 if (!Padded) {
104 OS.PadToColumn(50);
Devang Patel982efb52011-03-11 18:07:33 +0000105 OS << ";";
106 }
107 OS << " [debug variable = " << Var.getName() << "]";
108 }
109 }
Chris Lattner7f2f0932010-09-02 23:21:44 +0000110 }
111};
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +0000112
Chris Lattner7f2f0932010-09-02 23:21:44 +0000113} // end anon namespace
114
Chris Lattner2f7c9632001-06-06 20:29:01 +0000115int main(int argc, char **argv) {
Chris Lattnere3fc2d12009-03-06 05:34:10 +0000116 // Print a stack trace if we signal out.
117 sys::PrintStackTraceOnErrorSignal();
118 PrettyStackTraceProgram X(argc, argv);
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +0000119
Owen Anderson19251ec2009-07-15 22:16:10 +0000120 LLVMContext &Context = getGlobalContext();
Chris Lattnere3fc2d12009-03-06 05:34:10 +0000121 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +0000122
123
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000124 cl::ParseCommandLineOptions(argc, argv, "llvm .bc -> .ll disassembler\n");
Chris Lattner12439ff2004-02-19 20:32:12 +0000125
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000126 std::string ErrorMessage;
Ahmed Charles56440fd2014-03-06 05:51:42 +0000127 std::unique_ptr<Module> M;
Michael J. Spencer7b6fef82010-12-09 17:36:48 +0000128
Derek Schuff8b2dcad2012-02-06 22:30:29 +0000129 // Use the bitcode streaming interface
130 DataStreamer *streamer = getDataFileStreamer(InputFilename, &ErrorMessage);
131 if (streamer) {
132 std::string DisplayFilename;
133 if (InputFilename == "-")
134 DisplayFilename = "<stdin>";
Michael J. Spencera646f392010-12-16 22:37:52 +0000135 else
Derek Schuff8b2dcad2012-02-06 22:30:29 +0000136 DisplayFilename = InputFilename;
137 M.reset(getStreamedBitcodeModule(DisplayFilename, streamer, Context,
138 &ErrorMessage));
Craig Toppere6cb63e2014-04-25 04:24:47 +0000139 if(M.get()) {
Rafael Espindolae9fab9b2014-01-14 23:51:27 +0000140 if (error_code EC = M->materializeAllPermanently()) {
141 ErrorMessage = EC.message();
142 M.reset();
143 }
Derek Schuff8b2dcad2012-02-06 22:30:29 +0000144 }
Michael J. Spencera646f392010-12-16 22:37:52 +0000145 }
Chris Lattner76d46322006-12-06 01:18:01 +0000146
Craig Toppere6cb63e2014-04-25 04:24:47 +0000147 if (!M.get()) {
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000148 errs() << argv[0] << ": ";
149 if (ErrorMessage.size())
150 errs() << ErrorMessage << "\n";
151 else
152 errs() << "bitcode didn't read correctly.\n";
153 return 1;
154 }
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +0000155
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000156 // Just use stdout. We won't actually print anything on it.
157 if (DontPrint)
158 OutputFilename = "-";
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +0000159
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000160 if (OutputFilename.empty()) { // Unspecified output, infer it.
161 if (InputFilename == "-") {
162 OutputFilename = "-";
163 } else {
164 const std::string &IFN = InputFilename;
165 int Len = IFN.length();
166 // If the source ends in .bc, strip it off.
167 if (IFN[Len-3] == '.' && IFN[Len-2] == 'b' && IFN[Len-1] == 'c')
168 OutputFilename = std::string(IFN.begin(), IFN.end()-3)+".ll";
169 else
170 OutputFilename = IFN+".ll";
171 }
172 }
Dan Gohmane5929232009-09-11 20:46:33 +0000173
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000174 std::string ErrorInfo;
Ahmed Charles56440fd2014-03-06 05:51:42 +0000175 std::unique_ptr<tool_output_file> Out(
176 new tool_output_file(OutputFilename.c_str(), ErrorInfo, sys::fs::F_None));
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000177 if (!ErrorInfo.empty()) {
178 errs() << ErrorInfo << '\n';
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000179 return 1;
180 }
181
Ahmed Charles56440fd2014-03-06 05:51:42 +0000182 std::unique_ptr<AssemblyAnnotationWriter> Annotator;
Chris Lattner7f2f0932010-09-02 23:21:44 +0000183 if (ShowAnnotations)
184 Annotator.reset(new CommentWriter());
Michael J. Spencer7fda8fe2010-12-16 16:23:30 +0000185
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000186 // All that llvm-dis does is write the assembly to a file.
Dan Gohman972c9c52009-09-15 15:33:42 +0000187 if (!DontPrint)
Chris Lattner7f2f0932010-09-02 23:21:44 +0000188 M->print(Out->os(), Annotator.get());
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000189
Dan Gohman268b0f42010-08-20 01:07:01 +0000190 // Declare success.
191 Out->keep();
192
Chris Lattner9e6f1f12009-08-23 02:51:22 +0000193 return 0;
Chris Lattner2f7c9632001-06-06 20:29:01 +0000194}