blob: c60e56ae9b6e78765631b5bc5609cef3deb087c3 [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"
Chris Lattnere6012df2009-03-06 05:34:10 +000022#include "llvm/Support/PrettyStackTrace.h"
Chris Lattnerb1aa85b2009-08-23 22:45:37 +000023#include "llvm/Support/raw_ostream.h"
Dan Gohman176426d2009-08-25 15:34:52 +000024#include "llvm/Support/SystemUtils.h"
Dan Gohman4f7a1562009-09-12 21:55:12 +000025#include "llvm/Support/IRReader.h"
Dan Gohmanf17a25c2007-07-18 16:29:46 +000026#include "llvm/System/Signals.h"
27#include "llvm/System/Path.h"
Dan Gohmanf17a25c2007-07-18 16:29:46 +000028#include <memory>
29using namespace llvm;
30
31static cl::list<std::string>
32InputFilenames(cl::Positional, cl::OneOrMore,
33 cl::desc("<input bitcode files>"));
34
35static cl::opt<std::string>
36OutputFilename("o", cl::desc("Override output filename"), cl::init("-"),
37 cl::value_desc("filename"));
38
Dan Gohman176426d2009-08-25 15:34:52 +000039static cl::opt<bool>
40Force("f", cl::desc("Enable binary output on terminals"));
Dan Gohmanf17a25c2007-07-18 16:29:46 +000041
42static cl::opt<bool>
Dan Gohmand5939a42009-09-15 15:35:07 +000043OutputAssembly("S",
44 cl::desc("Write output as LLVM assembly"), cl::Hidden);
45
46static cl::opt<bool>
Dan Gohmanf17a25c2007-07-18 16:29:46 +000047Verbose("v", cl::desc("Print information about actions taken"));
48
49static cl::opt<bool>
50DumpAsm("d", cl::desc("Print assembly as linked"), cl::Hidden);
51
52// LoadFile - Read the specified bitcode file in and return it. This routine
53// searches the link path for the specified file to try to find it...
54//
Dan Gohman4f7a1562009-09-12 21:55:12 +000055static inline std::auto_ptr<Module> LoadFile(const char *argv0,
56 const std::string &FN,
Owen Anderson92adb182009-07-01 23:13:44 +000057 LLVMContext& Context) {
Dan Gohmanf17a25c2007-07-18 16:29:46 +000058 sys::Path Filename;
59 if (!Filename.set(FN)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000060 errs() << "Invalid file name: '" << FN << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000061 return std::auto_ptr<Module>();
62 }
63
Dan Gohman4f7a1562009-09-12 21:55:12 +000064 SMDiagnostic Err;
Dan Gohmanf17a25c2007-07-18 16:29:46 +000065 if (Filename.exists()) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000066 if (Verbose) errs() << "Loading '" << Filename.c_str() << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000067 Module* Result = 0;
68
Chris Lattnerb1aa85b2009-08-23 22:45:37 +000069 const std::string &FNStr = Filename.str();
Dan Gohman4f7a1562009-09-12 21:55:12 +000070 Result = ParseIRFile(FNStr, Err, Context);
Dan Gohmanf17a25c2007-07-18 16:29:46 +000071 if (Result) return std::auto_ptr<Module>(Result); // Load successful!
72
Dan Gohman4f7a1562009-09-12 21:55:12 +000073 if (Verbose)
74 Err.Print(argv0, errs());
Dan Gohmanf17a25c2007-07-18 16:29:46 +000075 } else {
Dan Gohmanb714fab2009-07-16 15:30:09 +000076 errs() << "Bitcode file: '" << Filename.c_str() << "' does not exist.\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000077 }
78
79 return std::auto_ptr<Module>();
80}
81
82int main(int argc, char **argv) {
Chris Lattnere6012df2009-03-06 05:34:10 +000083 // Print a stack trace if we signal out.
Dan Gohmanf17a25c2007-07-18 16:29:46 +000084 sys::PrintStackTraceOnErrorSignal();
Chris Lattnere6012df2009-03-06 05:34:10 +000085 PrettyStackTraceProgram X(argc, argv);
86
Owen Andersone84b8b32009-07-15 22:16:10 +000087 LLVMContext &Context = getGlobalContext();
Chris Lattnere6012df2009-03-06 05:34:10 +000088 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
89 cl::ParseCommandLineOptions(argc, argv, "llvm linker\n");
Dan Gohmanf17a25c2007-07-18 16:29:46 +000090
91 unsigned BaseArg = 0;
92 std::string ErrorMessage;
93
Dan Gohman4f7a1562009-09-12 21:55:12 +000094 std::auto_ptr<Module> Composite(LoadFile(argv[0],
95 InputFilenames[BaseArg], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +000096 if (Composite.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000097 errs() << argv[0] << ": error loading file '"
98 << InputFilenames[BaseArg] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000099 return 1;
100 }
101
102 for (unsigned i = BaseArg+1; i < InputFilenames.size(); ++i) {
Dan Gohman4f7a1562009-09-12 21:55:12 +0000103 std::auto_ptr<Module> M(LoadFile(argv[0],
104 InputFilenames[i], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000105 if (M.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000106 errs() << argv[0] << ": error loading file '" <<InputFilenames[i]<< "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000107 return 1;
108 }
109
Dan Gohmanb714fab2009-07-16 15:30:09 +0000110 if (Verbose) errs() << "Linking in '" << InputFilenames[i] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000111
112 if (Linker::LinkModules(Composite.get(), M.get(), &ErrorMessage)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000113 errs() << argv[0] << ": link error in '" << InputFilenames[i]
114 << "': " << ErrorMessage << "\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000115 return 1;
116 }
117 }
118
119 // TODO: Iterate over the -l list and link in any modules containing
120 // global symbols that have not been resolved so far.
121
Dan Gohmand5939a42009-09-15 15:35:07 +0000122 if (DumpAsm) errs() << "Here's the assembly:\n" << *Composite;
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000123
Chris Lattner4e95ea92009-08-23 02:56:05 +0000124 std::string ErrorInfo;
125 std::auto_ptr<raw_ostream>
126 Out(new raw_fd_ostream(OutputFilename.c_str(), ErrorInfo,
Dan Gohman176426d2009-08-25 15:34:52 +0000127 raw_fd_ostream::F_Binary));
Chris Lattner4e95ea92009-08-23 02:56:05 +0000128 if (!ErrorInfo.empty()) {
129 errs() << ErrorInfo << '\n';
Chris Lattner4e95ea92009-08-23 02:56:05 +0000130 return 1;
131 }
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000132
133 // Make sure that the Out file gets unlinked from the disk if we get a
134 // SIGINT
Chris Lattner4e95ea92009-08-23 02:56:05 +0000135 if (OutputFilename != "-")
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000136 sys::RemoveFileOnSignal(sys::Path(OutputFilename));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000137
Dan Gohmand5939a42009-09-15 15:35:07 +0000138 if (verifyModule(*Composite)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000139 errs() << argv[0] << ": linked module is broken!\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000140 return 1;
141 }
142
Dan Gohmanb714fab2009-07-16 15:30:09 +0000143 if (Verbose) errs() << "Writing bitcode...\n";
Dan Gohmand5939a42009-09-15 15:35:07 +0000144 if (OutputAssembly) {
145 *Out << *Composite;
146 } else if (Force || !CheckBitcodeOutputToConsole(*Out, true))
Dan Gohman176426d2009-08-25 15:34:52 +0000147 WriteBitcodeToFile(Composite.get(), *Out);
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000148
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000149 return 0;
150}