ELF: Also record the type of undefined symbols.

Tested with a weak undefined. Testing with a plain undefined will have to wait
for support for -shared.

llvm-svn: 245069
This commit is contained in:
Rafael Espindola 2015-08-14 16:46:28 +00:00
parent a57d015154
commit 1bd885aba4
6 changed files with 84 additions and 22 deletions

View File

@ -86,11 +86,11 @@ SymbolBody *elf2::ObjectFile<ELFT>::createSymbolBody(StringRef StringTable,
error("unexpected binding"); error("unexpected binding");
case STB_GLOBAL: case STB_GLOBAL:
if (Sym->isUndefined()) if (Sym->isUndefined())
return new (Alloc) Undefined(Name); return new (Alloc) Undefined<ELFT>(Name, *Sym);
return new (Alloc) DefinedRegular<ELFT>(Name, *Sym); return new (Alloc) DefinedRegular<ELFT>(Name, *Sym);
case STB_WEAK: case STB_WEAK:
if (Sym->isUndefined()) if (Sym->isUndefined())
return new (Alloc) UndefinedWeak(Name); return new (Alloc) UndefinedWeak<ELFT>(Name, *Sym);
return new (Alloc) DefinedWeak<ELFT>(Name, *Sym); return new (Alloc) DefinedWeak<ELFT>(Name, *Sym);
} }
} }

View File

@ -16,7 +16,9 @@ using namespace llvm;
using namespace lld; using namespace lld;
using namespace lld::elf2; using namespace lld::elf2;
SymbolTable::SymbolTable() { resolve(new (Alloc) Undefined("_start")); } SymbolTable::SymbolTable() {
resolve(new (Alloc) SyntheticUndefined("_start"));
}
void SymbolTable::addFile(std::unique_ptr<InputFile> File) { void SymbolTable::addFile(std::unique_ptr<InputFile> File) {
File->parse(); File->parse();
@ -39,9 +41,9 @@ void SymbolTable::addObject(ObjectFileBase *File) {
void SymbolTable::reportRemainingUndefines() { void SymbolTable::reportRemainingUndefines() {
for (auto &I : Symtab) { for (auto &I : Symtab) {
Symbol *Sym = I.second; SymbolBody *Body = I.second->Body;
if (auto *Undef = dyn_cast<Undefined>(Sym->Body)) if (Body->isStrongUndefined())
error(Twine("undefined symbol: ") + Undef->getName()); error(Twine("undefined symbol: ") + Body->getName());
} }
} }

View File

@ -38,6 +38,7 @@ int SymbolBody::compare(SymbolBody *Other) {
case DefinedWeakKind: case DefinedWeakKind:
case UndefinedKind: case UndefinedKind:
case UndefinedWeakKind: case UndefinedWeakKind:
case UndefinedSyntheticKind:
return 1; return 1;
} }
llvm_unreachable("unknown symbol kind"); llvm_unreachable("unknown symbol kind");

View File

@ -40,11 +40,16 @@ public:
DefinedWeakKind = 1, DefinedWeakKind = 1,
DefinedLast = 1, DefinedLast = 1,
UndefinedWeakKind = 2, UndefinedWeakKind = 2,
UndefinedKind = 3 UndefinedKind = 3,
UndefinedSyntheticKind = 4
}; };
Kind kind() const { return static_cast<Kind>(SymbolKind); } Kind kind() const { return static_cast<Kind>(SymbolKind); }
bool isStrongUndefined() {
return SymbolKind == UndefinedKind || SymbolKind == UndefinedSyntheticKind;
}
// Returns the symbol name. // Returns the symbol name.
StringRef getName() const { return Name; } StringRef getName() const { return Name; }
@ -70,19 +75,43 @@ protected:
Symbol *Backref = nullptr; Symbol *Backref = nullptr;
}; };
// The base class for any defined symbols, including absolute symbols, // This is for symbols created from elf files and not from the command line.
// etc. // Since they come from object files, they have a Elf_Sym.
template <class ELFT> class Defined : public SymbolBody { //
public: // FIXME: Another alternative is to give every symbol an Elf_Sym. To do that
// we have to delay creating the symbol table until the output format is
// known and some of its methods will be templated. We should experiment with
// that once we have a bit more code.
template <class ELFT> class ELFSymbolBody : public SymbolBody {
protected:
typedef typename llvm::object::ELFFile<ELFT>::Elf_Sym Elf_Sym; typedef typename llvm::object::ELFFile<ELFT>::Elf_Sym Elf_Sym;
const Elf_Sym &Sym; ELFSymbolBody(Kind K, StringRef Name, const Elf_Sym &Sym)
: SymbolBody(K, Name), Sym(Sym) {}
explicit Defined(Kind K, StringRef N, const Elf_Sym &Sym) public:
: SymbolBody(K, N), Sym(Sym) {} const Elf_Sym &Sym;
static bool classof(const SymbolBody *S) { static bool classof(const SymbolBody *S) {
Kind K = S->kind(); Kind K = S->kind();
return DefinedFirst <= K && K <= DefinedLast; return K >= DefinedFirst && K <= UndefinedKind;
}
};
// The base class for any defined symbols, including absolute symbols,
// etc.
template <class ELFT> class Defined : public ELFSymbolBody<ELFT> {
typedef ELFSymbolBody<ELFT> Base;
typedef typename Base::Kind Kind;
public:
typedef typename Base::Elf_Sym Elf_Sym;
explicit Defined(Kind K, StringRef N, const Elf_Sym &Sym)
: ELFSymbolBody<ELFT>(K, N, Sym) {}
static bool classof(const SymbolBody *S) {
Kind K = S->kind();
return Base::DefinedFirst <= K && K <= Base::DefinedLast;
} }
}; };
@ -114,21 +143,38 @@ public:
}; };
// Undefined symbols. // Undefined symbols.
class Undefined : public SymbolBody { class SyntheticUndefined : public SymbolBody {
public: public:
explicit Undefined(StringRef N) : SymbolBody(UndefinedKind, N) {} explicit SyntheticUndefined(StringRef N) : SymbolBody(UndefinedKind, N) {}
static bool classof(const SymbolBody *S) { static bool classof(const SymbolBody *S) {
return S->kind() == UndefinedKind; return S->kind() == UndefinedKind;
} }
}; };
class UndefinedWeak : public SymbolBody { template <class ELFT> class Undefined : public ELFSymbolBody<ELFT> {
typedef ELFSymbolBody<ELFT> Base;
typedef typename Base::Elf_Sym Elf_Sym;
public: public:
explicit UndefinedWeak(StringRef N) : SymbolBody(UndefinedWeakKind, N) {} explicit Undefined(StringRef N, const Elf_Sym &Sym)
: ELFSymbolBody<ELFT>(Base::UndefinedKind, N, Sym) {}
static bool classof(const SymbolBody *S) { static bool classof(const SymbolBody *S) {
return S->kind() == UndefinedWeakKind; return S->kind() == Base::UndefinedKind;
}
};
template <class ELFT> class UndefinedWeak : public ELFSymbolBody<ELFT> {
typedef ELFSymbolBody<ELFT> Base;
typedef typename Base::Elf_Sym Elf_Sym;
public:
explicit UndefinedWeak(StringRef N, const Elf_Sym &Sym)
: ELFSymbolBody<ELFT>(Base::UndefinedWeakKind, N, Sym) {}
static bool classof(const SymbolBody *S) {
return S->kind() == Base::UndefinedWeakKind;
} }
}; };

View File

@ -224,16 +224,16 @@ template <class ELFT> void SymbolTableSection<ELFT>::writeTo(uint8_t *Buf) {
uint8_t Type = 0; uint8_t Type = 0;
switch (Body->kind()) { switch (Body->kind()) {
case SymbolBody::UndefinedKind: case SymbolBody::UndefinedKind:
case SymbolBody::UndefinedSyntheticKind:
llvm_unreachable("Should be defined by now"); llvm_unreachable("Should be defined by now");
case SymbolBody::DefinedRegularKind: case SymbolBody::DefinedRegularKind:
Binding = STB_GLOBAL; Binding = STB_GLOBAL;
Type = cast<DefinedRegular<ELFT>>(Body)->Sym.getType(); Type = cast<DefinedRegular<ELFT>>(Body)->Sym.getType();
break; break;
case SymbolBody::DefinedWeakKind: case SymbolBody::DefinedWeakKind:
Type = cast<DefinedWeak<ELFT>>(Body)->Sym.getType();
// Fallthrough
case SymbolBody::UndefinedWeakKind: case SymbolBody::UndefinedWeakKind:
Binding = STB_WEAK; Binding = STB_WEAK;
Type = cast<ELFSymbolBody<ELFT>>(Body)->Sym.getType();
break; break;
} }
ESym->setBindingAndType(Binding, Type); ESym->setBindingAndType(Binding, Type);

View File

@ -11,6 +11,10 @@ _start:
.weak foo .weak foo
foo: foo:
.type bar, @object
.weak bar
.long bar
// CHECK: Symbols [ // CHECK: Symbols [
// CHECK-NEXT: Symbol { // CHECK-NEXT: Symbol {
// CHECK-NEXT: Name: (0) // CHECK-NEXT: Name: (0)
@ -31,6 +35,15 @@ foo:
// CHECK-NEXT: Section: Undefined (0x0) // CHECK-NEXT: Section: Undefined (0x0)
// CHECK-NEXT: } // CHECK-NEXT: }
// CHECK-NEXT: Symbol { // CHECK-NEXT: Symbol {
// CHECK-NEXT: Name: bar
// CHECK-NEXT: Value: 0x0
// CHECK-NEXT: Size: 0
// CHECK-NEXT: Binding: Weak (0x2)
// CHECK-NEXT: Type: Object (0x1)
// CHECK-NEXT: Other: 0
// CHECK-NEXT: Section: Undefined (0x0)
// CHECK-NEXT: }
// CHECK-NEXT: Symbol {
// CHECK-NEXT: Name: foo // CHECK-NEXT: Name: foo
// CHECK-NEXT: Value: 0x0 // CHECK-NEXT: Value: 0x0
// CHECK-NEXT: Size: 0 // CHECK-NEXT: Size: 0