switch tools to bitcode instead of bytecode
[oota-llvm.git] / tools / llvm-dis / llvm-dis.cpp
1 //===-- llvm-dis.cpp - The low-level LLVM disassembler --------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file was developed by the LLVM research group and is distributed under
6 // the University of Illinois Open Source License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This utility may be invoked in the following manner:
11 //  llvm-dis [options]      - Read LLVM bytecode from stdin, write asm to stdout
12 //  llvm-dis [options] x.bc - Read LLVM bytecode from the x.bc file, write asm
13 //                            to the x.ll file.
14 //  Options:
15 //      --help   - Output information about command line switches
16 //
17 //===----------------------------------------------------------------------===//
18
19 #include "llvm/Module.h"
20 #include "llvm/PassManager.h"
21 #include "llvm/Bitcode/ReaderWriter.h"
22 #include "llvm/Assembly/PrintModulePass.h"
23 #include "llvm/Support/CommandLine.h"
24 #include "llvm/Support/ManagedStatic.h"
25 #include "llvm/Support/MemoryBuffer.h"
26 #include "llvm/Support/Streams.h"
27 #include "llvm/System/Signals.h"
28 #include <iostream>
29 #include <fstream>
30 #include <memory>
31 using namespace llvm;
32
33 static cl::opt<std::string>
34 InputFilename(cl::Positional, cl::desc("<input bytecode>"), cl::init("-"));
35
36 static cl::opt<std::string>
37 OutputFilename("o", cl::desc("Override output filename"),
38                cl::value_desc("filename"));
39
40 static cl::opt<bool>
41 Force("f", cl::desc("Overwrite output files"));
42
43 static cl::opt<bool>
44 DontPrint("disable-output", cl::desc("Don't output the .ll file"), cl::Hidden);
45
46 int main(int argc, char **argv) {
47   llvm_shutdown_obj X;  // Call llvm_shutdown() on exit.
48   try {
49     cl::ParseCommandLineOptions(argc, argv, " llvm .bc -> .ll disassembler\n");
50     sys::PrintStackTraceOnErrorSignal();
51
52     std::ostream *Out = &std::cout;  // Default to printing to stdout.
53     std::string ErrorMessage;
54
55     std::auto_ptr<Module> M;
56    
57     MemoryBuffer *Buffer
58       = MemoryBuffer::getFileOrSTDIN(&InputFilename[0], InputFilename.size());
59
60     if (Buffer == 0)
61       ErrorMessage = "Error reading file '" + InputFilename + "'";
62     else
63       M.reset(ParseBitcodeFile(Buffer, &ErrorMessage));
64     delete Buffer;
65
66     if (M.get() == 0) {
67       cerr << argv[0] << ": ";
68       if (ErrorMessage.size())
69         cerr << ErrorMessage << "\n";
70       else
71         cerr << "bytecode didn't read correctly.\n";
72       return 1;
73     }
74     
75     if (DontPrint) {
76       // Just use stdout.  We won't actually print anything on it.
77     } else if (OutputFilename != "") {   // Specified an output filename?
78       if (OutputFilename != "-") { // Not stdout?
79         if (!Force && std::ifstream(OutputFilename.c_str())) {
80           // If force is not specified, make sure not to overwrite a file!
81           cerr << argv[0] << ": error opening '" << OutputFilename
82                << "': file exists! Sending to standard output.\n";
83         } else {
84           Out = new std::ofstream(OutputFilename.c_str());
85         }
86       }
87     } else {
88       if (InputFilename == "-") {
89         OutputFilename = "-";
90       } else {
91         std::string IFN = InputFilename;
92         int Len = IFN.length();
93         if (IFN[Len-3] == '.' && IFN[Len-2] == 'b' && IFN[Len-1] == 'c') {
94           // Source ends in .bc
95           OutputFilename = std::string(IFN.begin(), IFN.end()-3)+".ll";
96         } else {
97           OutputFilename = IFN+".ll";
98         }
99
100         if (!Force && std::ifstream(OutputFilename.c_str())) {
101           // If force is not specified, make sure not to overwrite a file!
102           cerr << argv[0] << ": error opening '" << OutputFilename
103                << "': file exists! Sending to standard output.\n";
104         } else {
105           Out = new std::ofstream(OutputFilename.c_str());
106
107           // Make sure that the Out file gets unlinked from the disk if we get a
108           // SIGINT
109           sys::RemoveFileOnSignal(sys::Path(OutputFilename));
110         }
111       }
112     }
113
114     if (!Out->good()) {
115       cerr << argv[0] << ": error opening " << OutputFilename
116            << ": sending to stdout instead!\n";
117       Out = &std::cout;
118     }
119
120     // All that llvm-dis does is write the assembly to a file.
121     if (!DontPrint) {
122       PassManager Passes;
123       OStream L(*Out);
124       Passes.add(new PrintModulePass(&L));
125       Passes.run(*M.get());
126     }
127
128     if (Out != &std::cout) {
129       ((std::ofstream*)Out)->close();
130       delete Out;
131     }
132     return 0;
133   } catch (const std::string& msg) {
134     cerr << argv[0] << ": " << msg << "\n";
135   } catch (...) {
136     cerr << argv[0] << ": Unexpected unknown exception occurred.\n";
137   }
138
139   return 1;
140 }
141