2012-12-08 08:47:36 +08:00
|
|
|
//===- lib/Driver/Driver.cpp - Linker Driver Emulator ---------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Linker
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "lld/Driver/Driver.h"
|
2013-05-29 02:37:39 +08:00
|
|
|
|
2013-04-05 02:59:24 +08:00
|
|
|
#include "lld/Core/LLVM.h"
|
2013-05-29 02:55:39 +08:00
|
|
|
#include "lld/Core/Instrumentation.h"
|
2013-04-05 02:59:24 +08:00
|
|
|
#include "lld/Core/PassManager.h"
|
2013-05-29 02:37:39 +08:00
|
|
|
#include "lld/Core/Parallel.h"
|
|
|
|
#include "lld/Core/Resolver.h"
|
2013-04-05 02:59:24 +08:00
|
|
|
#include "lld/ReaderWriter/Reader.h"
|
|
|
|
#include "lld/ReaderWriter/Writer.h"
|
2012-12-08 08:47:36 +08:00
|
|
|
|
2013-04-05 02:59:24 +08:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
|
|
#include "llvm/Option/Arg.h"
|
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/Path.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
|
2013-10-09 07:01:52 +08:00
|
|
|
#include <mutex>
|
|
|
|
|
2013-04-05 02:59:24 +08:00
|
|
|
namespace lld {
|
|
|
|
|
|
|
|
/// This is where the link is actually performed.
|
2013-10-08 23:43:48 +08:00
|
|
|
bool Driver::link(LinkingContext &context, raw_ostream &diagnostics) {
|
2013-04-05 02:59:24 +08:00
|
|
|
// Honor -mllvm
|
2013-08-07 06:31:59 +08:00
|
|
|
if (!context.llvmOptions().empty()) {
|
|
|
|
unsigned numArgs = context.llvmOptions().size();
|
|
|
|
const char **args = new const char *[numArgs + 2];
|
2013-04-05 02:59:24 +08:00
|
|
|
args[0] = "lld (LLVM option parsing)";
|
|
|
|
for (unsigned i = 0; i != numArgs; ++i)
|
2013-08-07 06:31:59 +08:00
|
|
|
args[i + 1] = context.llvmOptions()[i];
|
2013-04-05 02:59:24 +08:00
|
|
|
args[numArgs + 1] = 0;
|
|
|
|
llvm::cl::ParseCommandLineOptions(numArgs + 1, args);
|
|
|
|
}
|
2013-08-22 06:57:10 +08:00
|
|
|
InputGraph &inputGraph = context.inputGraph();
|
2013-10-07 10:47:09 +08:00
|
|
|
if (!inputGraph.size())
|
2013-09-25 07:26:34 +08:00
|
|
|
return false;
|
2013-04-05 02:59:24 +08:00
|
|
|
|
2013-10-07 10:47:09 +08:00
|
|
|
bool fail = false;
|
|
|
|
|
2013-04-05 02:59:24 +08:00
|
|
|
// Read inputs
|
2013-05-29 02:55:39 +08:00
|
|
|
ScopedTask readTask(getDefaultDomain(), "Read Args");
|
2013-05-29 02:37:39 +08:00
|
|
|
TaskGroup tg;
|
2013-10-07 10:47:09 +08:00
|
|
|
int index = 0;
|
2013-10-09 07:01:52 +08:00
|
|
|
std::mutex diagnosticsMutex;
|
2013-08-31 13:27:38 +08:00
|
|
|
for (auto &ie : inputGraph.inputElements()) {
|
2013-10-07 10:47:09 +08:00
|
|
|
tg.spawn([&, index] {
|
2013-10-09 07:01:52 +08:00
|
|
|
// Writes to the same output stream is not guaranteed to be thread-safe.
|
|
|
|
// We buffer the diagnostics output to a separate string-backed output
|
|
|
|
// stream, acquire the lock, and then print it out.
|
|
|
|
std::string buf;
|
|
|
|
llvm::raw_string_ostream stream(buf);
|
|
|
|
|
|
|
|
if (error_code ec = ie->parse(context, stream)) {
|
|
|
|
FileNode *fileNode = llvm::dyn_cast<FileNode>(ie.get());
|
|
|
|
stream << fileNode->errStr(ec) << "\n";
|
2013-05-29 02:37:39 +08:00
|
|
|
fail = true;
|
|
|
|
}
|
2013-10-09 07:01:52 +08:00
|
|
|
|
|
|
|
stream.flush();
|
|
|
|
if (!buf.empty()) {
|
|
|
|
std::lock_guard<std::mutex> lock(diagnosticsMutex);
|
|
|
|
diagnostics << buf;
|
|
|
|
}
|
2013-05-29 02:37:39 +08:00
|
|
|
});
|
|
|
|
++index;
|
2013-04-05 02:59:24 +08:00
|
|
|
}
|
2013-05-29 02:37:39 +08:00
|
|
|
tg.sync();
|
2013-05-29 02:55:39 +08:00
|
|
|
readTask.end();
|
2013-05-29 02:37:39 +08:00
|
|
|
|
|
|
|
if (fail)
|
2013-09-25 07:26:34 +08:00
|
|
|
return false;
|
2013-05-29 02:37:39 +08:00
|
|
|
|
2013-10-07 10:47:09 +08:00
|
|
|
std::unique_ptr<SimpleFileNode> fileNode(
|
|
|
|
new SimpleFileNode("Internal Files"));
|
2013-08-31 13:27:38 +08:00
|
|
|
|
2013-10-07 10:47:09 +08:00
|
|
|
InputGraph::FileVectorT internalFiles;
|
|
|
|
context.createInternalFiles(internalFiles);
|
2013-08-31 13:27:38 +08:00
|
|
|
|
2013-10-07 10:47:09 +08:00
|
|
|
if (internalFiles.size()) {
|
|
|
|
fileNode->appendInputFiles(std::move(internalFiles));
|
|
|
|
}
|
2013-04-05 02:59:24 +08:00
|
|
|
|
|
|
|
// Give target a chance to add files.
|
2013-10-07 10:47:09 +08:00
|
|
|
InputGraph::FileVectorT implicitFiles;
|
|
|
|
context.createImplicitFiles(implicitFiles);
|
|
|
|
if (implicitFiles.size()) {
|
|
|
|
fileNode->appendInputFiles(std::move(implicitFiles));
|
|
|
|
}
|
|
|
|
|
|
|
|
context.inputGraph().insertOneElementAt(std::move(fileNode),
|
|
|
|
InputGraph::Position::BEGIN);
|
|
|
|
|
|
|
|
context.inputGraph().assignOrdinals();
|
|
|
|
|
|
|
|
context.inputGraph().doPostProcess();
|
2013-04-05 02:59:24 +08:00
|
|
|
|
|
|
|
// Do core linking.
|
2013-05-29 02:55:39 +08:00
|
|
|
ScopedTask resolveTask(getDefaultDomain(), "Resolve");
|
2013-10-07 10:47:09 +08:00
|
|
|
Resolver resolver(context);
|
2013-10-11 14:16:33 +08:00
|
|
|
if (!resolver.resolve()) {
|
2013-08-07 06:31:59 +08:00
|
|
|
if (!context.allowRemainingUndefines())
|
2013-09-25 07:26:34 +08:00
|
|
|
return false;
|
2013-04-05 02:59:24 +08:00
|
|
|
}
|
|
|
|
MutableFile &merged = resolver.resultFile();
|
2013-05-29 02:55:39 +08:00
|
|
|
resolveTask.end();
|
2013-04-05 02:59:24 +08:00
|
|
|
|
|
|
|
// Run passes on linked atoms.
|
2013-05-29 02:55:39 +08:00
|
|
|
ScopedTask passTask(getDefaultDomain(), "Passes");
|
2013-04-05 02:59:24 +08:00
|
|
|
PassManager pm;
|
2013-08-07 06:31:59 +08:00
|
|
|
context.addPasses(pm);
|
2013-04-05 02:59:24 +08:00
|
|
|
pm.runOnFile(merged);
|
2013-05-29 02:55:39 +08:00
|
|
|
passTask.end();
|
2013-04-05 02:59:24 +08:00
|
|
|
|
|
|
|
// Give linked atoms to Writer to generate output file.
|
2013-05-29 02:55:39 +08:00
|
|
|
ScopedTask writeTask(getDefaultDomain(), "Write");
|
2013-08-07 06:31:59 +08:00
|
|
|
if (error_code ec = context.writeFile(merged)) {
|
|
|
|
diagnostics << "Failed to write file '" << context.outputPath()
|
2013-07-16 07:55:07 +08:00
|
|
|
<< "': " << ec.message() << "\n";
|
2013-09-25 07:26:34 +08:00
|
|
|
return false;
|
2013-04-05 02:59:24 +08:00
|
|
|
}
|
|
|
|
|
2013-09-25 07:26:34 +08:00
|
|
|
return true;
|
2013-04-05 02:59:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|