llvm-project/lld/unittests/DriverTests/GnuLdDriverTest.cpp

300 lines
9.0 KiB
C++

//===- lld/unittest/GnuLdDriverTest.cpp -----------------------------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
///
/// \file
/// \brief GNU ld driver tests.
///
//===----------------------------------------------------------------------===//
#include "DriverTest.h"
#include "lld/ReaderWriter/ELFLinkingContext.h"
#include "llvm/Support/MemoryBuffer.h"
using namespace llvm;
using namespace lld;
namespace {
class GnuLdParserTest
: public ParserTest<GnuLdDriver, std::unique_ptr<ELFLinkingContext>> {
protected:
const LinkingContext *linkingContext() override { return _ctx.get(); }
};
class LinkerScriptTest : public testing::Test {
protected:
void SetUp() override {
llvm::Triple triple(llvm::sys::getDefaultTargetTriple());
_ctx = GnuLdDriver::createELFLinkingContext(triple);
}
void parse(StringRef script, bool nostdlib = false) {
std::unique_ptr<MemoryBuffer> mb = MemoryBuffer::getMemBuffer(
script, "foo.so");
std::string s;
raw_string_ostream out(s);
std::error_code ec =
GnuLdDriver::evalLinkerScript(*_ctx, std::move(mb), out, nostdlib);
EXPECT_FALSE(ec);
}
std::unique_ptr<ELFLinkingContext> _ctx;
};
} // anonymous namespace
TEST_F(GnuLdParserTest, Empty) {
EXPECT_FALSE(parse("ld", nullptr));
EXPECT_EQ(linkingContext(), nullptr);
EXPECT_EQ("No input files\n", errorMessage());
}
// -o
TEST_F(GnuLdParserTest, Output) {
EXPECT_TRUE(parse("ld", "a.o", "-o", "foo", nullptr));
EXPECT_EQ("foo", _ctx->outputPath());
}
TEST_F(GnuLdParserTest, OutputDefault) {
EXPECT_TRUE(parse("ld", "abc.o", nullptr));
EXPECT_EQ("a.out", _ctx->outputPath());
}
// --noinhibit-exec
TEST_F(GnuLdParserTest, NoinhibitExec) {
EXPECT_TRUE(parse("ld", "a.o", "--noinhibit-exec", nullptr));
EXPECT_TRUE(_ctx->allowRemainingUndefines());
}
// --entry
TEST_F(GnuLdParserTest, Entry) {
EXPECT_TRUE(parse("ld", "a.o", "--entry", "foo", nullptr));
EXPECT_EQ("foo", _ctx->entrySymbolName());
}
TEST_F(GnuLdParserTest, EntryShort) {
EXPECT_TRUE(parse("ld", "a.o", "-e", "foo", nullptr));
EXPECT_EQ("foo", _ctx->entrySymbolName());
}
TEST_F(GnuLdParserTest, EntryJoined) {
EXPECT_TRUE(parse("ld", "a.o", "--entry=foo", nullptr));
EXPECT_EQ("foo", _ctx->entrySymbolName());
}
// --export-dynamic
TEST_F(GnuLdParserTest, ExportDynamic) {
EXPECT_TRUE(parse("ld", "a.o", "--export-dynamic", nullptr));
EXPECT_TRUE(_ctx->shouldExportDynamic());
}
TEST_F(GnuLdParserTest, NoExportDynamic) {
EXPECT_TRUE(parse("ld", "a.o", "--no-export-dynamic", nullptr));
EXPECT_FALSE(_ctx->shouldExportDynamic());
}
// --init
TEST_F(GnuLdParserTest, Init) {
EXPECT_TRUE(parse("ld", "a.o", "-init", "foo", "-init", "bar", nullptr));
EXPECT_EQ("bar", _ctx->initFunction());
}
TEST_F(GnuLdParserTest, InitJoined) {
EXPECT_TRUE(parse("ld", "a.o", "-init=foo", nullptr));
EXPECT_EQ("foo", _ctx->initFunction());
}
// --soname
TEST_F(GnuLdParserTest, SOName) {
EXPECT_TRUE(parse("ld", "a.o", "--soname=foo", nullptr));
EXPECT_EQ("foo", _ctx->sharedObjectName());
}
TEST_F(GnuLdParserTest, SONameSingleDash) {
EXPECT_TRUE(parse("ld", "a.o", "-soname=foo", nullptr));
EXPECT_EQ("foo", _ctx->sharedObjectName());
}
TEST_F(GnuLdParserTest, SONameH) {
EXPECT_TRUE(parse("ld", "a.o", "-h", "foo", nullptr));
EXPECT_EQ("foo", _ctx->sharedObjectName());
}
// -rpath
TEST_F(GnuLdParserTest, Rpath) {
EXPECT_TRUE(parse("ld", "a.o", "-rpath", "foo:bar", nullptr));
EXPECT_EQ(2, _ctx->getRpathList().size());
EXPECT_EQ("foo", _ctx->getRpathList()[0]);
EXPECT_EQ("bar", _ctx->getRpathList()[1]);
}
TEST_F(GnuLdParserTest, RpathEq) {
EXPECT_TRUE(parse("ld", "a.o", "-rpath=foo", nullptr));
EXPECT_EQ(1, _ctx->getRpathList().size());
EXPECT_EQ("foo", _ctx->getRpathList()[0]);
}
// --defsym
TEST_F(GnuLdParserTest, DefsymDecimal) {
EXPECT_TRUE(parse("ld", "a.o", "--defsym=sym=1000", nullptr));
assert(_ctx.get());
auto map = _ctx->getAbsoluteSymbols();
EXPECT_EQ((size_t)1, map.size());
EXPECT_EQ((uint64_t)1000, map["sym"]);
}
TEST_F(GnuLdParserTest, DefsymHexadecimal) {
EXPECT_TRUE(parse("ld", "a.o", "--defsym=sym=0x1000", nullptr));
auto map = _ctx->getAbsoluteSymbols();
EXPECT_EQ((size_t)1, map.size());
EXPECT_EQ((uint64_t)0x1000, map["sym"]);
}
TEST_F(GnuLdParserTest, DefsymAlias) {
EXPECT_TRUE(parse("ld", "a.o", "--defsym=sym=abc", nullptr));
auto map = _ctx->getAliases();
EXPECT_EQ((size_t)1, map.size());
EXPECT_EQ("abc", map["sym"]);
}
TEST_F(GnuLdParserTest, DefsymOctal) {
EXPECT_TRUE(parse("ld", "a.o", "--defsym=sym=0777", nullptr));
auto map = _ctx->getAbsoluteSymbols();
EXPECT_EQ((size_t)1, map.size());
EXPECT_EQ((uint64_t)0777, map["sym"]);
}
TEST_F(GnuLdParserTest, DefsymMisssingSymbol) {
EXPECT_FALSE(parse("ld", "a.o", "--defsym==0", nullptr));
}
TEST_F(GnuLdParserTest, DefsymMisssingValue) {
EXPECT_FALSE(parse("ld", "a.o", "--defsym=sym=", nullptr));
}
// --as-needed
TEST_F(GnuLdParserTest, AsNeeded) {
EXPECT_TRUE(parse("ld", "a.o", "--as-needed", "b.o", "c.o",
"--no-as-needed", "d.o", nullptr));
std::vector<std::unique_ptr<Node>> &nodes = _ctx->getNodes();
EXPECT_EQ((size_t)4, nodes.size());
EXPECT_FALSE(cast<FileNode>(nodes[0].get())->asNeeded());
EXPECT_TRUE(cast<FileNode>(nodes[1].get())->asNeeded());
EXPECT_TRUE(cast<FileNode>(nodes[2].get())->asNeeded());
EXPECT_FALSE(cast<FileNode>(nodes[3].get())->asNeeded());
}
// Emulation
TEST_F(GnuLdParserTest, Emulation) {
EXPECT_TRUE(parse("mips-linux-gnu-ld", "a.o", "-m", "elf64ltsmip", nullptr));
EXPECT_EQ(Triple::mips64el, _ctx->getTriple().getArch());
EXPECT_TRUE(
parse("mips64el-linux-gnu-ld", "a.o", "-m", "elf32btsmip", nullptr));
EXPECT_EQ(Triple::mips, _ctx->getTriple().getArch());
EXPECT_TRUE(
parse("mipsel-linux-gnu-ld", "a.o", "-m", "elf32btsmipn32", nullptr));
EXPECT_EQ(Triple::mips, _ctx->getTriple().getArch());
EXPECT_TRUE(
parse("mips-linux-gnu-ld", "a.o", "-m", "elf32ltsmipn32", nullptr));
EXPECT_EQ(Triple::mipsel, _ctx->getTriple().getArch());
}
// Linker script
TEST_F(LinkerScriptTest, Input) {
parse("INPUT(/x /y)");
std::vector<std::unique_ptr<Node>> &nodes = _ctx->getNodes();
EXPECT_EQ((size_t)2, nodes.size());
EXPECT_EQ("/x", cast<FileNode>(nodes[0].get())->getFile()->path());
EXPECT_EQ("/y", cast<FileNode>(nodes[1].get())->getFile()->path());
}
TEST_F(LinkerScriptTest, Group) {
parse("GROUP(/x /y)");
std::vector<std::unique_ptr<Node>> &nodes = _ctx->getNodes();
EXPECT_EQ((size_t)3, nodes.size());
EXPECT_EQ("/x", cast<FileNode>(nodes[0].get())->getFile()->path());
EXPECT_EQ("/y", cast<FileNode>(nodes[1].get())->getFile()->path());
EXPECT_EQ(2, cast<GroupEnd>(nodes[2].get())->getSize());
}
TEST_F(LinkerScriptTest, SearchDir) {
parse("SEARCH_DIR(\"/foo/bar\")");
std::vector<StringRef> paths = _ctx->getSearchPaths();
EXPECT_EQ((size_t)1, paths.size());
EXPECT_EQ("/foo/bar", paths[0]);
}
TEST_F(LinkerScriptTest, Entry) {
parse("ENTRY(blah)");
EXPECT_EQ("blah", _ctx->entrySymbolName());
}
TEST_F(LinkerScriptTest, Output) {
parse("OUTPUT(\"/path/to/output\")");
EXPECT_EQ("/path/to/output", _ctx->outputPath());
}
// Test that search paths are ignored when nostdlib is set.
TEST_F(LinkerScriptTest, IgnoreSearchDirNoStdLib) {
parse("SEARCH_DIR(\"/foo/bar\")", true /*nostdlib*/);
std::vector<StringRef> paths = _ctx->getSearchPaths();
EXPECT_EQ((size_t)0, paths.size());
}
TEST_F(LinkerScriptTest, ExprEval) {
parse("SECTIONS { symbol = 0x4000 + 0x40; \n"
". = (symbol >= 0x4040)? (0x5001 * 2 & 0xFFF0) << 1 : 0}");
EXPECT_EQ((size_t)1, _ctx->linkerScriptSema().getLinkerScripts().size());
script::LinkerScript *ls =
_ctx->linkerScriptSema().getLinkerScripts()[0]->get();
EXPECT_EQ((size_t)1, ls->_commands.size());
auto *secs = dyn_cast<const script::Sections>(*ls->_commands.begin());
EXPECT_TRUE(secs != nullptr);
EXPECT_EQ(2, secs->end() - secs->begin());
auto command = secs->begin();
auto *sa1 = dyn_cast<const script::SymbolAssignment>(*command);
EXPECT_TRUE(sa1 != nullptr);
EXPECT_EQ(script::SymbolAssignment::Simple, sa1->assignmentKind());
EXPECT_EQ(script::SymbolAssignment::Default, sa1->assignmentVisibility());
++command;
auto *sa2 = dyn_cast<const script::SymbolAssignment>(*command);
EXPECT_TRUE(sa2 != nullptr);
EXPECT_EQ(script::SymbolAssignment::Simple, sa2->assignmentKind());
EXPECT_EQ(script::SymbolAssignment::Default, sa2->assignmentVisibility());
script::Expression::SymbolTableTy mySymbolTable;
auto ans = sa1->expr()->evalExpr(mySymbolTable);
EXPECT_FALSE(ans.getError());
int64_t result = *ans;
EXPECT_EQ(0x4040, result);
mySymbolTable[sa1->symbol()] = result;
auto ans2 = sa2->expr()->evalExpr(mySymbolTable);
EXPECT_FALSE(ans2.getError());
result = *ans2;
EXPECT_EQ(0x14000, result);
EXPECT_EQ(0, sa2->symbol().compare(StringRef(".")));
}