| Peter Collingbourne | 66ac1d6 | 2016-04-22 20:21:26 +0000 | [diff] [blame] | 1 | //===- SymbolListFile.cpp -------------------------------------------------===// |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 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 | // This file contains the parser/evaluator of the linker script. |
| 11 | // It does not construct an AST but consume linker script directives directly. |
| 12 | // Results are written to Driver or Config object. |
| 13 | // |
| 14 | //===----------------------------------------------------------------------===// |
| 15 | |
| Peter Collingbourne | 66ac1d6 | 2016-04-22 20:21:26 +0000 | [diff] [blame] | 16 | #include "SymbolListFile.h" |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 17 | #include "Config.h" |
| 18 | #include "ScriptParser.h" |
| 19 | #include "llvm/Support/MemoryBuffer.h" |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 20 | |
| 21 | using namespace llvm; |
| Rui Ueyama | 2506866 | 2016-07-16 03:08:26 +0000 | [diff] [blame] | 22 | using namespace llvm::ELF; |
| 23 | |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 24 | using namespace lld; |
| 25 | using namespace lld::elf; |
| 26 | |
| 27 | // Parse the --dynamic-list argument. A dynamic list is in the form |
| 28 | // |
| 29 | // { symbol1; symbol2; [...]; symbolN }; |
| 30 | // |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 31 | // Multiple groups can be defined in the same file, and they are merged |
| 32 | // into a single group. |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 33 | |
| 34 | class DynamicListParser final : public ScriptParserBase { |
| 35 | public: |
| 36 | DynamicListParser(StringRef S) : ScriptParserBase(S) {} |
| Rui Ueyama | 4a46539 | 2016-04-22 22:59:24 +0000 | [diff] [blame] | 37 | void run(); |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 38 | }; |
| 39 | |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 40 | void DynamicListParser::run() { |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 41 | while (!atEOF()) { |
| 42 | expect("{"); |
| 43 | while (!Error) { |
| 44 | Config->DynamicList.push_back(next()); |
| 45 | expect(";"); |
| 46 | if (skip("}")) |
| 47 | break; |
| 48 | } |
| 49 | expect(";"); |
| 50 | } |
| Adhemerval Zanella | 9df0720 | 2016-04-13 18:51:11 +0000 | [diff] [blame] | 51 | } |
| 52 | |
| 53 | void elf::parseDynamicList(MemoryBufferRef MB) { |
| 54 | DynamicListParser(MB.getBuffer()).run(); |
| 55 | } |
| Peter Collingbourne | 66ac1d6 | 2016-04-22 20:21:26 +0000 | [diff] [blame] | 56 | |
| 57 | // Parse the --version-script argument. We currently only accept the following |
| 58 | // version script syntax: |
| 59 | // |
| 60 | // { [ global: symbol1; symbol2; [...]; symbolN; ] local: *; }; |
| 61 | // |
| 62 | // No wildcards are supported, other than for the local entry. Symbol versioning |
| 63 | // is also not supported. |
| 64 | |
| 65 | class VersionScriptParser final : public ScriptParserBase { |
| 66 | public: |
| 67 | VersionScriptParser(StringRef S) : ScriptParserBase(S) {} |
| 68 | |
| Rui Ueyama | 4a46539 | 2016-04-22 22:59:24 +0000 | [diff] [blame] | 69 | void run(); |
| George Rimar | 95eeb77 | 2016-06-16 18:47:04 +0000 | [diff] [blame] | 70 | |
| 71 | private: |
| George Rimar | d356630 | 2016-06-20 11:55:12 +0000 | [diff] [blame] | 72 | void parseVersion(StringRef Version); |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 73 | void parseGlobal(StringRef Version); |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 74 | void parseLocal(); |
| Peter Collingbourne | 66ac1d6 | 2016-04-22 20:21:26 +0000 | [diff] [blame] | 75 | }; |
| 76 | |
| George Rimar | 7899d48 | 2016-07-12 07:44:40 +0000 | [diff] [blame] | 77 | size_t elf::defineSymbolVersion(StringRef Version) { |
| 78 | // Identifiers start at 2 because 0 and 1 are reserved |
| 79 | // for VER_NDX_LOCAL and VER_NDX_GLOBAL constants. |
| 80 | size_t VersionId = Config->SymbolVersions.size() + 2; |
| 81 | Config->SymbolVersions.push_back(elf::Version(Version, VersionId)); |
| 82 | return VersionId; |
| 83 | } |
| 84 | |
| George Rimar | d356630 | 2016-06-20 11:55:12 +0000 | [diff] [blame] | 85 | void VersionScriptParser::parseVersion(StringRef Version) { |
| George Rimar | 7899d48 | 2016-07-12 07:44:40 +0000 | [diff] [blame] | 86 | defineSymbolVersion(Version); |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 87 | |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 88 | if (skip("global:") || peek() != "local:") |
| 89 | parseGlobal(Version); |
| 90 | if (skip("local:")) |
| 91 | parseLocal(); |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 92 | expect("}"); |
| George Rimar | 33b9de4 | 2016-07-01 11:45:10 +0000 | [diff] [blame] | 93 | |
| 94 | // Each version may have a parent version. For example, "Ver2" defined as |
| 95 | // "Ver2 { global: foo; local: *; } Ver1;" has "Ver1" as a parent. This |
| 96 | // version hierarchy is, probably against your instinct, purely for human; the |
| 97 | // runtime doesn't care about them at all. In LLD, we simply skip the token. |
| George Rimar | 0ab48b7 | 2016-06-22 14:40:45 +0000 | [diff] [blame] | 98 | if (!Version.empty() && peek() != ";") |
| George Rimar | 33b9de4 | 2016-07-01 11:45:10 +0000 | [diff] [blame] | 99 | next(); |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 100 | expect(";"); |
| 101 | } |
| 102 | |
| 103 | void VersionScriptParser::parseLocal() { |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 104 | Config->DefaultSymbolVersion = VER_NDX_LOCAL; |
| Peter Collingbourne | 66ac1d6 | 2016-04-22 20:21:26 +0000 | [diff] [blame] | 105 | expect("*"); |
| 106 | expect(";"); |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 107 | } |
| 108 | |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 109 | void VersionScriptParser::parseGlobal(StringRef Version) { |
| George Rimar | dd64bb3 | 2016-07-13 08:19:04 +0000 | [diff] [blame] | 110 | std::vector<StringRef> *Globals; |
| Rafael Espindola | 857d7c5 | 2016-06-28 21:48:33 +0000 | [diff] [blame] | 111 | if (Version.empty()) |
| George Rimar | d356630 | 2016-06-20 11:55:12 +0000 | [diff] [blame] | 112 | Globals = &Config->VersionScriptGlobals; |
| Rafael Espindola | 857d7c5 | 2016-06-28 21:48:33 +0000 | [diff] [blame] | 113 | else |
| George Rimar | d356630 | 2016-06-20 11:55:12 +0000 | [diff] [blame] | 114 | Globals = &Config->SymbolVersions.back().Globals; |
| George Rimar | d356630 | 2016-06-20 11:55:12 +0000 | [diff] [blame] | 115 | |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 116 | for (;;) { |
| George Rimar | da5fa1e | 2016-06-23 07:23:07 +0000 | [diff] [blame] | 117 | StringRef Cur = peek(); |
| George Rimar | dd64bb3 | 2016-07-13 08:19:04 +0000 | [diff] [blame] | 118 | if (Cur == "extern") |
| 119 | setError("extern keyword is not supported"); |
| George Rimar | da5fa1e | 2016-06-23 07:23:07 +0000 | [diff] [blame] | 120 | if (Cur == "}" || Cur == "local:" || Error) |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 121 | return; |
| George Rimar | da5fa1e | 2016-06-23 07:23:07 +0000 | [diff] [blame] | 122 | next(); |
| George Rimar | dd64bb3 | 2016-07-13 08:19:04 +0000 | [diff] [blame] | 123 | Globals->push_back(Cur); |
| Rafael Espindola | f70fb04 | 2016-06-17 13:38:09 +0000 | [diff] [blame] | 124 | expect(";"); |
| 125 | } |
| George Rimar | 95eeb77 | 2016-06-16 18:47:04 +0000 | [diff] [blame] | 126 | } |
| 127 | |
| 128 | void VersionScriptParser::run() { |
| 129 | StringRef Msg = "anonymous version definition is used in " |
| 130 | "combination with other version definitions"; |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 131 | if (skip("{")) { |
| George Rimar | d356630 | 2016-06-20 11:55:12 +0000 | [diff] [blame] | 132 | parseVersion(""); |
| George Rimar | 95eeb77 | 2016-06-16 18:47:04 +0000 | [diff] [blame] | 133 | if (!atEOF()) |
| 134 | setError(Msg); |
| 135 | return; |
| 136 | } |
| 137 | |
| 138 | while (!atEOF() && !Error) { |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 139 | StringRef Version = next(); |
| 140 | if (Version == "{") { |
| George Rimar | 95eeb77 | 2016-06-16 18:47:04 +0000 | [diff] [blame] | 141 | setError(Msg); |
| 142 | return; |
| 143 | } |
| Rui Ueyama | 52654eb | 2016-07-16 03:45:59 +0000 | [diff] [blame^] | 144 | expect("{"); |
| 145 | parseVersion(Version); |
| George Rimar | 95eeb77 | 2016-06-16 18:47:04 +0000 | [diff] [blame] | 146 | } |
| Peter Collingbourne | 66ac1d6 | 2016-04-22 20:21:26 +0000 | [diff] [blame] | 147 | } |
| 148 | |
| 149 | void elf::parseVersionScript(MemoryBufferRef MB) { |
| 150 | VersionScriptParser(MB.getBuffer()).run(); |
| 151 | } |