blob: cb741991bc49654f6da05de24a7eef37cfc8f8bd [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>
44Verbose("v", cl::desc("Print information about actions taken"));
45
46static cl::opt<bool>
47DumpAsm("d", cl::desc("Print assembly as linked"), cl::Hidden);
48
49// LoadFile - Read the specified bitcode file in and return it. This routine
50// searches the link path for the specified file to try to find it...
51//
Dan Gohman4f7a1562009-09-12 21:55:12 +000052static inline std::auto_ptr<Module> LoadFile(const char *argv0,
53 const std::string &FN,
Owen Anderson92adb182009-07-01 23:13:44 +000054 LLVMContext& Context) {
Dan Gohmanf17a25c2007-07-18 16:29:46 +000055 sys::Path Filename;
56 if (!Filename.set(FN)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000057 errs() << "Invalid file name: '" << FN << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000058 return std::auto_ptr<Module>();
59 }
60
Dan Gohman4f7a1562009-09-12 21:55:12 +000061 SMDiagnostic Err;
Dan Gohmanf17a25c2007-07-18 16:29:46 +000062 if (Filename.exists()) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000063 if (Verbose) errs() << "Loading '" << Filename.c_str() << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000064 Module* Result = 0;
65
Chris Lattnerb1aa85b2009-08-23 22:45:37 +000066 const std::string &FNStr = Filename.str();
Dan Gohman4f7a1562009-09-12 21:55:12 +000067 Result = ParseIRFile(FNStr, Err, Context);
Dan Gohmanf17a25c2007-07-18 16:29:46 +000068 if (Result) return std::auto_ptr<Module>(Result); // Load successful!
69
Dan Gohman4f7a1562009-09-12 21:55:12 +000070 if (Verbose)
71 Err.Print(argv0, errs());
Dan Gohmanf17a25c2007-07-18 16:29:46 +000072 } else {
Dan Gohmanb714fab2009-07-16 15:30:09 +000073 errs() << "Bitcode file: '" << Filename.c_str() << "' does not exist.\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000074 }
75
76 return std::auto_ptr<Module>();
77}
78
79int main(int argc, char **argv) {
Chris Lattnere6012df2009-03-06 05:34:10 +000080 // Print a stack trace if we signal out.
Dan Gohmanf17a25c2007-07-18 16:29:46 +000081 sys::PrintStackTraceOnErrorSignal();
Chris Lattnere6012df2009-03-06 05:34:10 +000082 PrettyStackTraceProgram X(argc, argv);
83
Owen Andersone84b8b32009-07-15 22:16:10 +000084 LLVMContext &Context = getGlobalContext();
Chris Lattnere6012df2009-03-06 05:34:10 +000085 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
86 cl::ParseCommandLineOptions(argc, argv, "llvm linker\n");
Dan Gohmanf17a25c2007-07-18 16:29:46 +000087
88 unsigned BaseArg = 0;
89 std::string ErrorMessage;
90
Dan Gohman4f7a1562009-09-12 21:55:12 +000091 std::auto_ptr<Module> Composite(LoadFile(argv[0],
92 InputFilenames[BaseArg], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +000093 if (Composite.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +000094 errs() << argv[0] << ": error loading file '"
95 << InputFilenames[BaseArg] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +000096 return 1;
97 }
98
99 for (unsigned i = BaseArg+1; i < InputFilenames.size(); ++i) {
Dan Gohman4f7a1562009-09-12 21:55:12 +0000100 std::auto_ptr<Module> M(LoadFile(argv[0],
101 InputFilenames[i], Context));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000102 if (M.get() == 0) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000103 errs() << argv[0] << ": error loading file '" <<InputFilenames[i]<< "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000104 return 1;
105 }
106
Dan Gohmanb714fab2009-07-16 15:30:09 +0000107 if (Verbose) errs() << "Linking in '" << InputFilenames[i] << "'\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000108
109 if (Linker::LinkModules(Composite.get(), M.get(), &ErrorMessage)) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000110 errs() << argv[0] << ": link error in '" << InputFilenames[i]
111 << "': " << ErrorMessage << "\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000112 return 1;
113 }
114 }
115
116 // TODO: Iterate over the -l list and link in any modules containing
117 // global symbols that have not been resolved so far.
118
Dan Gohmanb714fab2009-07-16 15:30:09 +0000119 if (DumpAsm) errs() << "Here's the assembly:\n" << *Composite.get();
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000120
Chris Lattner4e95ea92009-08-23 02:56:05 +0000121 std::string ErrorInfo;
122 std::auto_ptr<raw_ostream>
123 Out(new raw_fd_ostream(OutputFilename.c_str(), ErrorInfo,
Dan Gohman176426d2009-08-25 15:34:52 +0000124 raw_fd_ostream::F_Binary));
Chris Lattner4e95ea92009-08-23 02:56:05 +0000125 if (!ErrorInfo.empty()) {
126 errs() << ErrorInfo << '\n';
Chris Lattner4e95ea92009-08-23 02:56:05 +0000127 return 1;
128 }
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000129
130 // Make sure that the Out file gets unlinked from the disk if we get a
131 // SIGINT
Chris Lattner4e95ea92009-08-23 02:56:05 +0000132 if (OutputFilename != "-")
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000133 sys::RemoveFileOnSignal(sys::Path(OutputFilename));
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000134
135 if (verifyModule(*Composite.get())) {
Dan Gohmanb714fab2009-07-16 15:30:09 +0000136 errs() << argv[0] << ": linked module is broken!\n";
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000137 return 1;
138 }
139
Dan Gohmanb714fab2009-07-16 15:30:09 +0000140 if (Verbose) errs() << "Writing bitcode...\n";
Dan Gohman176426d2009-08-25 15:34:52 +0000141 if (Force || !CheckBitcodeOutputToConsole(*Out, true))
142 WriteBitcodeToFile(Composite.get(), *Out);
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000143
Dan Gohmanf17a25c2007-07-18 16:29:46 +0000144 return 0;
145}