blob: 6dc04e6722871f171bd046b97f3e5ff1055631f2 [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 Gohmanf17a25c2007-07-18 16:29:46 +000025#include "llvm/System/Signals.h"
26#include "llvm/System/Path.h"
Dan Gohmanf17a25c2007-07-18 16:29:46 +000027#include <memory>
28using namespace llvm;
29
30static cl::list<std::string>
31InputFilenames(cl::Positional, cl::OneOrMore,
32 cl::desc("<input bitcode files>"));
33
34static cl::opt<std::string>
35OutputFilename("o", cl::desc("Override output filename"), cl::init("-"),
36 cl::value_desc("filename"));
37
38static cl::opt<bool> Force("f", cl::desc("Overwrite output files"));
39
40static cl::opt<bool>
41Verbose("v", cl::desc("Print information about actions taken"));
42
43static cl::opt<bool>
44DumpAsm("d", cl::desc("Print assembly as linked"), cl::Hidden);
45
46// LoadFile - Read the specified bitcode file in and return it. This routine
47// searches the link path for the specified file to try to find it...
48//
Owen Anderson25209b42009-07-01 16:58:40 +000049static inline std::auto_ptr<Module> LoadFile(const std::string &FN,
Owen Anderson92adb182009-07-01 23:13:44 +000050 LLVMContext& Context) {
Dan Gohmanf17a25c2007-07-18 16:29:46 +000051 sys::Path Filename;
52 if (!Filename.set(FN)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000053 errs() << "Invalid file name: '" << FN << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000054 return std::auto_ptr<Module>();
55 }
56
57 std::string ErrorMessage;
58 if (Filename.exists()) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000059 if (Verbose) errs() << "Loading '" << Filename.c_str() << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000060 Module* Result = 0;
61
Chris Lattnerb1aa85b2009-08-23 22:45:37 +000062 const std::string &FNStr = Filename.str();
Dan Gohmanf17a25c2007-07-18 16:29:46 +000063 if (MemoryBuffer *Buffer = MemoryBuffer::getFileOrSTDIN(FNStr,
64 &ErrorMessage)) {
Owen Anderson25209b42009-07-01 16:58:40 +000065 Result = ParseBitcodeFile(Buffer, Context, &ErrorMessage);
Dan Gohmanf17a25c2007-07-18 16:29:46 +000066 delete Buffer;
67 }
68 if (Result) return std::auto_ptr<Module>(Result); // Load successful!
69
70 if (Verbose) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000071 errs() << "Error opening bitcode file: '" << Filename.c_str() << "'";
72 if (ErrorMessage.size()) errs() << ": " << ErrorMessage;
73 errs() << "\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000074 }
75 } 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
Owen Andersona148fdd2009-07-01 21:22:36 +000094 std::auto_ptr<Module> Composite(LoadFile(InputFilenames[BaseArg], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +000095 if (Composite.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000096 errs() << argv[0] << ": error loading file '"
97 << InputFilenames[BaseArg] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000098 return 1;
99 }
100
101 for (unsigned i = BaseArg+1; i < InputFilenames.size(); ++i) {
Owen Andersona148fdd2009-07-01 21:22:36 +0000102 std::auto_ptr<Module> M(LoadFile(InputFilenames[i], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000103 if (M.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000104 errs() << argv[0] << ": error loading file '" <<InputFilenames[i]<< "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000105 return 1;
106 }
107
Dan Gohmanb714fab2009-07-16 15:30:09 +0000108 if (Verbose) errs() << "Linking in '" << InputFilenames[i] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000109
110 if (Linker::LinkModules(Composite.get(), M.get(), &ErrorMessage)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000111 errs() << argv[0] << ": link error in '" << InputFilenames[i]
112 << "': " << ErrorMessage << "\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000113 return 1;
114 }
115 }
116
117 // TODO: Iterate over the -l list and link in any modules containing
118 // global symbols that have not been resolved so far.
119
Dan Gohmanb714fab2009-07-16 15:30:09 +0000120 if (DumpAsm) errs() << "Here's the assembly:\n" << *Composite.get();
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000121
Chris Lattner4e95ea92009-08-23 02:56:05 +0000122 std::string ErrorInfo;
123 std::auto_ptr<raw_ostream>
124 Out(new raw_fd_ostream(OutputFilename.c_str(), ErrorInfo,
125 raw_fd_ostream::F_Binary |
126 (Force ? raw_fd_ostream::F_Force : 0)));
127 if (!ErrorInfo.empty()) {
128 errs() << ErrorInfo << '\n';
129 if (!Force)
130 errs() << "Use -f command line argument to force output\n";
131 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
139 if (verifyModule(*Composite.get())) {
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 Gohmanf17a25c2007-07-18 16:29:46 +0000145 WriteBitcodeToFile(Composite.get(), *Out);
146
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000147 return 0;
148}