blob: fae4d107b1a22698de07649db36d703653498314 [file] [log] [blame]
Dan Gohmanf17a25c2007-07-18 16:29:46 +00001//===- llvm-link.cpp - Low-level LLVM linker ------------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner5f5a5732007-12-29 20:44:31 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Dan Gohmanf17a25c2007-07-18 16:29:46 +00007//
8//===----------------------------------------------------------------------===//
9//
10// This utility may be invoked in the following manner:
11// llvm-link a.bc b.bc c.bc -o x.bc
12//
13//===----------------------------------------------------------------------===//
14
15#include "llvm/Linker.h"
Owen Anderson25209b42009-07-01 16:58:40 +000016#include "llvm/LLVMContext.h"
Dan Gohmanf17a25c2007-07-18 16:29:46 +000017#include "llvm/Module.h"
18#include "llvm/Analysis/Verifier.h"
19#include "llvm/Bitcode/ReaderWriter.h"
20#include "llvm/Support/CommandLine.h"
21#include "llvm/Support/ManagedStatic.h"
22#include "llvm/Support/MemoryBuffer.h"
Chris Lattnere6012df2009-03-06 05:34:10 +000023#include "llvm/Support/PrettyStackTrace.h"
Chris Lattnerb1aa85b2009-08-23 22:45:37 +000024#include "llvm/Support/raw_ostream.h"
Dan Gohman176426d2009-08-25 15:34:52 +000025#include "llvm/Support/SystemUtils.h"
Dan Gohman4f7a1562009-09-12 21:55:12 +000026#include "llvm/Support/IRReader.h"
Dan Gohmanf17a25c2007-07-18 16:29:46 +000027#include "llvm/System/Signals.h"
28#include "llvm/System/Path.h"
Dan Gohmanf17a25c2007-07-18 16:29:46 +000029#include <memory>
30using namespace llvm;
31
32static cl::list<std::string>
33InputFilenames(cl::Positional, cl::OneOrMore,
34 cl::desc("<input bitcode files>"));
35
36static cl::opt<std::string>
37OutputFilename("o", cl::desc("Override output filename"), cl::init("-"),
38 cl::value_desc("filename"));
39
Dan Gohman176426d2009-08-25 15:34:52 +000040static cl::opt<bool>
41Force("f", cl::desc("Enable binary output on terminals"));
Dan Gohmanf17a25c2007-07-18 16:29:46 +000042
43static cl::opt<bool>
Dan Gohmand5939a42009-09-15 15:35:07 +000044OutputAssembly("S",
45 cl::desc("Write output as LLVM assembly"), cl::Hidden);
46
47static cl::opt<bool>
Dan Gohmanf17a25c2007-07-18 16:29:46 +000048Verbose("v", cl::desc("Print information about actions taken"));
49
50static cl::opt<bool>
51DumpAsm("d", cl::desc("Print assembly as linked"), cl::Hidden);
52
53// LoadFile - Read the specified bitcode file in and return it. This routine
54// searches the link path for the specified file to try to find it...
55//
Dan Gohman4f7a1562009-09-12 21:55:12 +000056static inline std::auto_ptr<Module> LoadFile(const char *argv0,
57 const std::string &FN,
Owen Anderson92adb182009-07-01 23:13:44 +000058 LLVMContext& Context) {
Dan Gohmanf17a25c2007-07-18 16:29:46 +000059 sys::Path Filename;
60 if (!Filename.set(FN)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000061 errs() << "Invalid file name: '" << FN << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000062 return std::auto_ptr<Module>();
63 }
64
Dan Gohman4f7a1562009-09-12 21:55:12 +000065 SMDiagnostic Err;
Dan Gohmanf17a25c2007-07-18 16:29:46 +000066 if (Filename.exists()) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000067 if (Verbose) errs() << "Loading '" << Filename.c_str() << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000068 Module* Result = 0;
69
Chris Lattnerb1aa85b2009-08-23 22:45:37 +000070 const std::string &FNStr = Filename.str();
Dan Gohman4f7a1562009-09-12 21:55:12 +000071 Result = ParseIRFile(FNStr, Err, Context);
Dan Gohmanf17a25c2007-07-18 16:29:46 +000072 if (Result) return std::auto_ptr<Module>(Result); // Load successful!
73
Dan Gohman4f7a1562009-09-12 21:55:12 +000074 if (Verbose)
75 Err.Print(argv0, errs());
Dan Gohmanf17a25c2007-07-18 16:29:46 +000076 } else {
Dan Gohmanb714fab2009-07-16 15:30:09 +000077 errs() << "Bitcode file: '" << Filename.c_str() << "' does not exist.\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000078 }
79
80 return std::auto_ptr<Module>();
81}
82
83int main(int argc, char **argv) {
Chris Lattnere6012df2009-03-06 05:34:10 +000084 // Print a stack trace if we signal out.
Dan Gohmanf17a25c2007-07-18 16:29:46 +000085 sys::PrintStackTraceOnErrorSignal();
Chris Lattnere6012df2009-03-06 05:34:10 +000086 PrettyStackTraceProgram X(argc, argv);
87
Owen Andersone84b8b32009-07-15 22:16:10 +000088 LLVMContext &Context = getGlobalContext();
Chris Lattnere6012df2009-03-06 05:34:10 +000089 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
90 cl::ParseCommandLineOptions(argc, argv, "llvm linker\n");
Dan Gohmanf17a25c2007-07-18 16:29:46 +000091
92 unsigned BaseArg = 0;
93 std::string ErrorMessage;
94
Dan Gohman4f7a1562009-09-12 21:55:12 +000095 std::auto_ptr<Module> Composite(LoadFile(argv[0],
96 InputFilenames[BaseArg], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +000097 if (Composite.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000098 errs() << argv[0] << ": error loading file '"
99 << InputFilenames[BaseArg] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000100 return 1;
101 }
102
103 for (unsigned i = BaseArg+1; i < InputFilenames.size(); ++i) {
Dan Gohman4f7a1562009-09-12 21:55:12 +0000104 std::auto_ptr<Module> M(LoadFile(argv[0],
105 InputFilenames[i], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000106 if (M.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000107 errs() << argv[0] << ": error loading file '" <<InputFilenames[i]<< "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000108 return 1;
109 }
110
Dan Gohmanb714fab2009-07-16 15:30:09 +0000111 if (Verbose) errs() << "Linking in '" << InputFilenames[i] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000112
113 if (Linker::LinkModules(Composite.get(), M.get(), &ErrorMessage)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000114 errs() << argv[0] << ": link error in '" << InputFilenames[i]
115 << "': " << ErrorMessage << "\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000116 return 1;
117 }
118 }
119
120 // TODO: Iterate over the -l list and link in any modules containing
121 // global symbols that have not been resolved so far.
122
Dan Gohmand5939a42009-09-15 15:35:07 +0000123 if (DumpAsm) errs() << "Here's the assembly:\n" << *Composite;
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000124
Chris Lattner4e95ea92009-08-23 02:56:05 +0000125 std::string ErrorInfo;
126 std::auto_ptr<raw_ostream>
127 Out(new raw_fd_ostream(OutputFilename.c_str(), ErrorInfo,
Dan Gohman176426d2009-08-25 15:34:52 +0000128 raw_fd_ostream::F_Binary));
Chris Lattner4e95ea92009-08-23 02:56:05 +0000129 if (!ErrorInfo.empty()) {
130 errs() << ErrorInfo << '\n';
Chris Lattner4e95ea92009-08-23 02:56:05 +0000131 return 1;
132 }
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000133
134 // Make sure that the Out file gets unlinked from the disk if we get a
135 // SIGINT
Chris Lattner4e95ea92009-08-23 02:56:05 +0000136 if (OutputFilename != "-")
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000137 sys::RemoveFileOnSignal(sys::Path(OutputFilename));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000138
Dan Gohmand5939a42009-09-15 15:35:07 +0000139 if (verifyModule(*Composite)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000140 errs() << argv[0] << ": linked module is broken!\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000141 return 1;
142 }
143
Dan Gohmanb714fab2009-07-16 15:30:09 +0000144 if (Verbose) errs() << "Writing bitcode...\n";
Dan Gohmand5939a42009-09-15 15:35:07 +0000145 if (OutputAssembly) {
146 *Out << *Composite;
147 } else if (Force || !CheckBitcodeOutputToConsole(*Out, true))
Dan Gohman176426d2009-08-25 15:34:52 +0000148 WriteBitcodeToFile(Composite.get(), *Out);
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000149
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000150 return 0;
151}