blob: 18fe18adea2179b6800381e55567aa3a21c86f8a [file] [log] [blame]
Michael J. Spencer84487f12015-07-24 21:03:07 +00001//===- Symbols.h ----------------------------------------------------------===//
2//
3// The LLVM Linker
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#ifndef LLD_ELF_SYMBOLS_H
11#define LLD_ELF_SYMBOLS_H
12
13#include "lld/Core/LLVM.h"
14#include "llvm/Object/ELF.h"
15
16namespace lld {
17namespace elf2 {
18
19using llvm::object::ELFFile;
20
21class Chunk;
22class InputFile;
23class SymbolBody;
Michael J. Spencercdae0a42015-07-28 22:58:25 +000024template <class ELFT> class ObjectFile;
Michael J. Spencer84487f12015-07-24 21:03:07 +000025
26// A real symbol object, SymbolBody, is usually accessed indirectly
27// through a Symbol. There's always one Symbol for each symbol name.
28// The resolver updates SymbolBody pointers as it resolves symbols.
29struct Symbol {
30 explicit Symbol(SymbolBody *P) : Body(P) {}
31 SymbolBody *Body;
32};
33
34// The base class for real symbol classes.
35class SymbolBody {
36public:
37 enum Kind {
Rafael Espindolaae1b23b2015-08-11 17:10:02 +000038 DefinedFirst = 0,
39 DefinedRegularKind = 0,
Rafael Espindolab13df652015-08-11 17:33:02 +000040 DefinedWeakKind = 1,
41 DefinedLast = 1,
Rafael Espindola76e24ea2015-08-11 17:57:05 +000042 UndefinedWeakKind = 2,
43 UndefinedKind = 3
Michael J. Spencer84487f12015-07-24 21:03:07 +000044 };
45
Michael J. Spencercdae0a42015-07-28 22:58:25 +000046 Kind kind() const { return static_cast<Kind>(SymbolKind); }
Michael J. Spencer84487f12015-07-24 21:03:07 +000047
Michael J. Spencer84487f12015-07-24 21:03:07 +000048 // Returns the symbol name.
Michael J. Spencercdae0a42015-07-28 22:58:25 +000049 StringRef getName() const { return Name; }
Michael J. Spencer84487f12015-07-24 21:03:07 +000050
51 // A SymbolBody has a backreference to a Symbol. Originally they are
52 // doubly-linked. A backreference will never change. But the pointer
53 // in the Symbol may be mutated by the resolver. If you have a
54 // pointer P to a SymbolBody and are not sure whether the resolver
55 // has chosen the object among other objects having the same name,
56 // you can access P->Backref->Body to get the resolver's result.
57 void setBackref(Symbol *P) { Backref = P; }
58 SymbolBody *getReplacement() { return Backref ? Backref->Body : this; }
59
60 // Decides which symbol should "win" in the symbol table, this or
61 // the Other. Returns 1 if this wins, -1 if the Other wins, or 0 if
62 // they are duplicate (conflicting) symbols.
Rui Ueyamaa7ccb292015-07-27 20:39:01 +000063 int compare(SymbolBody *Other);
Michael J. Spencer84487f12015-07-24 21:03:07 +000064
65protected:
Michael J. Spencercdae0a42015-07-28 22:58:25 +000066 SymbolBody(Kind K, StringRef N = "")
Rafael Espindolae3335d82015-08-05 13:26:54 +000067 : SymbolKind(K), Name(N) {}
Michael J. Spencer84487f12015-07-24 21:03:07 +000068
Michael J. Spencercdae0a42015-07-28 22:58:25 +000069protected:
70 const unsigned SymbolKind : 8;
Michael J. Spencercdae0a42015-07-28 22:58:25 +000071 StringRef Name;
Michael J. Spencer84487f12015-07-24 21:03:07 +000072 Symbol *Backref = nullptr;
73};
74
75// The base class for any defined symbols, including absolute symbols,
76// etc.
77class Defined : public SymbolBody {
78public:
Rui Ueyama4f89fda2015-08-11 23:37:25 +000079 explicit Defined(Kind K, StringRef N = "") : SymbolBody(K, N) {}
Michael J. Spencer84487f12015-07-24 21:03:07 +000080
81 static bool classof(const SymbolBody *S) {
82 Kind K = S->kind();
83 return DefinedFirst <= K && K <= DefinedLast;
84 }
Michael J. Spencer84487f12015-07-24 21:03:07 +000085};
86
87// Regular defined symbols read from object file symbol tables.
Rafael Espindola791e9f92015-08-11 17:51:57 +000088class DefinedRegular : public Defined {
Michael J. Spencer84487f12015-07-24 21:03:07 +000089public:
Rui Ueyama4f89fda2015-08-11 23:37:25 +000090 explicit DefinedRegular(StringRef N) : Defined(DefinedRegularKind, N) {}
Michael J. Spencer84487f12015-07-24 21:03:07 +000091
92 static bool classof(const SymbolBody *S) {
93 return S->kind() == DefinedRegularKind;
94 }
Michael J. Spencer84487f12015-07-24 21:03:07 +000095};
96
Rafael Espindola791e9f92015-08-11 17:51:57 +000097class DefinedWeak : public Defined {
Rafael Espindolab13df652015-08-11 17:33:02 +000098public:
Rui Ueyama4f89fda2015-08-11 23:37:25 +000099 explicit DefinedWeak(StringRef N) : Defined(DefinedWeakKind, N) {}
Rafael Espindolab13df652015-08-11 17:33:02 +0000100
101 static bool classof(const SymbolBody *S) {
102 return S->kind() == DefinedWeakKind;
103 }
104};
105
Michael J. Spencer84487f12015-07-24 21:03:07 +0000106// Undefined symbols.
107class Undefined : public SymbolBody {
108public:
Michael J. Spencercdae0a42015-07-28 22:58:25 +0000109 explicit Undefined(StringRef N) : SymbolBody(UndefinedKind, N) {}
Michael J. Spencer84487f12015-07-24 21:03:07 +0000110
111 static bool classof(const SymbolBody *S) {
112 return S->kind() == UndefinedKind;
113 }
Michael J. Spencer84487f12015-07-24 21:03:07 +0000114};
115
Rafael Espindola76e24ea2015-08-11 17:57:05 +0000116class UndefinedWeak : public SymbolBody {
117public:
118 explicit UndefinedWeak(StringRef N) : SymbolBody(UndefinedWeakKind, N) {}
119
120 static bool classof(const SymbolBody *S) {
121 return S->kind() == UndefinedWeakKind;
122 }
123};
124
Michael J. Spencer84487f12015-07-24 21:03:07 +0000125} // namespace elf2
126} // namespace lld
127
128#endif