Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(389)

Side by Side Diff: src/IceELFSection.h

Issue 678533005: Subzero: Add basic ELFObjectWriter (text section, symtab, strtab, headers) (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: cleanup Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 //===- subzero/src/IceELFSection.h - Model of ELF sections ------*- C++ -*-===//
2 //
3 // The Subzero Code Generator
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // Representation of ELF sections.
11 //
12 //===----------------------------------------------------------------------===//
13
14 #ifndef SUBZERO_SRC_ICEELFSECTION_H
15 #define SUBZERO_SRC_ICEELFSECTION_H
16
17 #include "IceDefs.h"
18 #include "IceELFStreamer.h"
19
20 using namespace llvm::ELF;
21
22 namespace Ice {
23
24 class ELFStreamer;
25 class ELFStringTableSection;
26
27 // Base representation of an ELF section.
28 class ELFSection {
29 ELFSection(const ELFSection &) = delete;
30 ELFSection &operator=(const ELFSection &) = delete;
31
32 public:
33 // Constructs an ELF section, filling in fields that will be known
34 // once the *type* of section is decided. Other fields may be updated
35 // incrementally or only after the program is completely defined.
36 ELFSection(const IceString &Name, Elf64_Word sh_type,
37 Elf64_Xword sh_flags, Elf64_Xword sh_addralign,
38 Elf64_Xword sh_entsize)
39 : Name(Name), Header(), Number(NoSectionNumber) {
40 Header.sh_type = sh_type;
41 Header.sh_flags = sh_flags;
42 Header.sh_addralign = sh_addralign;
43 Header.sh_entsize = sh_entsize;
44 }
45
46 // Sentinel value for sections without number.
47 static const SizeT NoSectionNumber;
48
49 // Set fields of the header which are only known later.
50 void setNumber(SizeT N) {
51 // Should only set the number once: from NoSectionNumber -> N.
52 assert(Number == NoSectionNumber);
53 Number = N;
54 }
55 SizeT getNumber() const {
56 assert(Number != NoSectionNumber);
57 return Number;
58 }
59
60 IceString getName() const { return Name; }
61 void setNameStrIndex(Elf64_Word sh_name) { Header.sh_name = sh_name; }
62 Elf64_Word getNameStrIndex() const { return Header.sh_name; }
63
64 void setSize(Elf64_Xword sh_size) { Header.sh_size = sh_size; }
65 SizeT getCurrentSize() const { return Header.sh_size; }
66
67 void setLinkNum(Elf64_Word sh_link) { Header.sh_link = sh_link; }
68 Elf64_Word getLinkNum() const { return Header.sh_link; }
69
70 void setInfoNum(Elf64_Word sh_info) { Header.sh_info = sh_info; }
71 Elf64_Word getInfoNum() const { return Header.sh_info; }
72
73 void setFileOffset(Elf64_Off sh_offset) { Header.sh_offset = sh_offset; }
74 Elf64_Off getFileOffset() const { return Header.sh_offset; }
75
76 Elf64_Xword getSectionAlign() const { return Header.sh_addralign; }
77
78 // Write the section header out with the given streamer.
79 template <bool IsELF64> void writeHeader(ELFStreamer &Str);
80
81 protected:
82 ~ELFSection() {}
83
84 // Name of the section in convenient string form (instead of a index
85 // into the Section Header String Table).
86 IceString Name;
87
88 // The fields of the header. Also provides storage for some bookkeeping.
89 Elf64_Shdr Header;
90
91 // The number of the section after laying out sections.
92 SizeT Number;
93 };
94
95
96 // Models text/code sections. Code is written out incrementally and the
97 // size of the section is then updated incrementally.
98 class ELFTextSection : public ELFSection {
99 ELFTextSection(const ELFTextSection &) = delete;
100 ELFTextSection &operator=(const ELFTextSection &) = delete;
101
102 public:
103 using ELFSection::ELFSection;
104
105 void appendData(ELFStreamer &Str, const llvm::StringRef MoreData);
106 };
107
108
109 // Models data/rodata sections. Data is written out incrementally and the
110 // size of the section is then updated incrementally.
111 // Some rodata sections may have fixed entsize and duplicates may be mergeable.
112 class ELFDataSection : public ELFSection {
113 ELFDataSection(const ELFDataSection &) = delete;
114 ELFDataSection &operator=(const ELFDataSection &) = delete;
115
116 public:
117 using ELFSection::ELFSection;
118
119 void appendData(ELFStreamer &Str, const llvm::StringRef MoreData);
120 };
121
122 // Model of ELF symbol table entries. Besides keeping track of the fields
123 // required for an elf symbol table entry it also tracks the number that
124 // represents the symbol's final index in the symbol table.
125 struct ELFSym {
126 Elf64_Sym Sym;
127 SizeT Number;
128
129 static const SizeT UnknownNumber;
130
131 void setNumber(SizeT N) {
132 assert(Number == UnknownNumber);
133 Number = N;
134 }
135
136 SizeT getNumber() const {
137 assert(Number != UnknownNumber);
138 return Number;
139 }
140 };
141
142
143 // Models a symbol table. Symbols may be added up until updateIndices is
144 // called. At that point the indices of each symbol will be finalized.
145 class ELFSymbolTableSection : public ELFSection {
146 public:
147 using ELFSection::ELFSection;
148
149 // Create initial entry for a symbol when it is defined.
150 // Each entry should only be defined once.
151 // We might want to allow Name to be a dummy name initially, then
152 // get updated to the real thing, since Data initializers are read
153 // before the bitcode's symbol table is read.
154 void createDefinedSym(const IceString &Name, uint8_t Type, uint8_t Binding,
155 ELFSection *Section, RelocOffsetT Offset, SizeT Size);
156
157 // Note that a symbol table entry needs to be created for the given
158 // symbol because it is undefined.
159 void noteUndefinedSym(const IceString &Name, ELFSection *NullSection);
160
161 size_t getSectionDataSize() const {
162 return (LocalSymbols.size() + GlobalSymbols.size()) * Header.sh_entsize;
163 }
164
165 size_t getNumLocals() const {
166 return LocalSymbols.size();
167 }
168
169 void updateIndices(const ELFStringTableSection *StrTab);
170
171 void writeData(ELFStreamer &Str, bool IsELF64);
172
173 private:
174 // Map from symbol name + section to its symbol information.
175 typedef std::pair<IceString, ELFSection *> SymtabKey;
176 typedef std::map<SymtabKey, ELFSym> SymMap;
177
178 template <bool IsELF64> void writeSymbolMap(ELFStreamer &Str,
179 const SymMap &Map);
180
181 // Keep Local and Global symbols separate, since the sh_info needs to
182 // know the index of the last LOCAL.
183 SymMap LocalSymbols;
184 SymMap GlobalSymbols;
185 };
186
187
188 // Base model of a relocation section.
189 class ELFRelocationSectionBase : public ELFSection {
190 ELFRelocationSectionBase(const ELFRelocationSectionBase &) = delete;
191 ELFRelocationSectionBase &operator=(const ELFRelocationSectionBase &) = delete ;
192
193 public:
194 ELFRelocationSectionBase(const IceString &Name, Elf64_Word sh_type,
195 Elf64_Xword sh_flags, Elf64_Xword sh_addralign,
196 Elf64_Xword sh_entsize, ELFSection *RelatedSection)
197 : ELFSection(Name, sh_type, sh_flags, sh_addralign, sh_entsize),
198 RelatedSection(RelatedSection) { }
199
200 ELFSection *getRelatedSection() const { return RelatedSection; }
201
202 private:
203 ELFSection *RelatedSection;
204 };
205
206 // ELFRelocationSection which depends on the actual relocation type.
207 // Specializations are needed depending on the ELFCLASS and whether
208 // or not addends are explicit or implicitly embedded in the related
209 // section (ELFCLASS64 pack their r_info field differently from ELFCLASS32).
210 template <typename RelType>
211 class ELFRelocationSection : ELFRelocationSectionBase {
212 ELFRelocationSection(const ELFRelocationSectionBase &) = delete;
213 ELFRelocationSection &operator=(const ELFRelocationSectionBase &) = delete;
214
215 public:
216 using ELFRelocationSectionBase::ELFRelocationSectionBase;
217
218 void addRelocations() {
219 // TODO: fill me in
220 }
221
222 private:
223 typedef std::pair<RelType, ELFSym *> ELFRelSym;
224 typedef std::vector<ELFRelSym> RelocationList;
225 RelocationList Relocations;
226 };
227
228 // Models a string table. The user will build the string table by
229 // adding strings incrementally. At some point, all strings should be
230 // known and layOutStrings() should be called. After that, no other
231 // strings may be added. However, the final offsets of the strings
232 // can be discovered and used to fill out section headers and symbol
233 // table entries.
234 class ELFStringTableSection : public ELFSection {
235 ELFStringTableSection(const ELFStringTableSection &) = delete;
236 ELFStringTableSection &operator=(const ELFStringTableSection &) = delete;
237
238 public:
239 using ELFSection::ELFSection;
240
241 // Add a string to the table, in preparation for final layout.
242 void add(const IceString &Str);
243
244 // Finalizes the layout of the string table and fills in the section Data.
245 void doLayout();
246
247 // Grabs the final index of a string after layout.
248 size_t getIndex(const IceString &Str) const;
249
250 llvm::StringRef getSectionData() const {
251 assert(isLaidOut());
252 return llvm::StringRef(
253 reinterpret_cast<const char *>(StringData.data()), StringData.size());
254 }
255
256 size_t getSectionDataSize() const { return getSectionData().size(); }
257
258 private:
259 bool isLaidOut() const { return !StringData.empty(); }
260
261 // Strings can share a string table entry if they share the same suffix.
262 // E.g., "pop\\0" and "lollipop\0" can both use the characters in
263 // "lolipop\\0", but "pops\\0" cannot, and "unpop\\0" cannot either.
264 // Though, "pop\\0", "lollipop\\0", and "unpop\\0" share "pop\\0" as
265 // the suffix, "pop\\0" can only share the characters with one of them.
266 struct SuffixComparator {
267 bool operator()(const IceString &StrA, const IceString &StrB) const;
268 };
269
270 typedef std::map<IceString, size_t, SuffixComparator> StringToIndexType;
271
272 static const size_t UnknownIndex;
273
274 // Track strings to their index. Index will be UnknownIndex if not
275 // yet laid out.
276 StringToIndexType StringToIndexMap;
277
278 typedef std::vector<uint8_t> RawDataType;
279 RawDataType StringData;
280 };
281
282 template <bool IsELF64> void ELFSection::writeHeader(ELFStreamer &Str) {
283 Str.writeELFWord<IsELF64>(Header.sh_name);
284 Str.writeELFWord<IsELF64>(Header.sh_type);
285 Str.writeELFXword<IsELF64>(Header.sh_flags);
286 Str.writeAddrOrOffset<IsELF64>(Header.sh_addr);
287 Str.writeAddrOrOffset<IsELF64>(Header.sh_offset);
288 Str.writeELFXword<IsELF64>(Header.sh_size);
289 Str.writeELFWord<IsELF64>(Header.sh_link);
290 Str.writeELFWord<IsELF64>(Header.sh_info);
291 Str.writeELFXword<IsELF64>(Header.sh_addralign);
292 Str.writeELFXword<IsELF64>(Header.sh_entsize);
293 }
294
295 template <bool IsELF64> void ELFSymbolTableSection::writeSymbolMap(
296 ELFStreamer &Str, const SymMap &Map) {
297 // The order of the fields is different, so branch on IsELF64.
298 if (IsELF64) {
299 for (auto &KeyValue : Map) {
300 const Elf64_Sym &SymInfo = KeyValue.second.Sym;
301 Str.writeELFWord<IsELF64>(SymInfo.st_name);
302 Str.write8(SymInfo.st_info);
303 Str.write8(SymInfo.st_other);
304 Str.writeLE16(SymInfo.st_shndx);
305 Str.writeAddrOrOffset<IsELF64>(SymInfo.st_value);
306 Str.writeELFXword<IsELF64>(SymInfo.st_size);
307 }
308 } else {
309 for (auto &KeyValue : Map) {
310 const Elf64_Sym &SymInfo = KeyValue.second.Sym;
311 Str.writeELFWord<IsELF64>(SymInfo.st_name);
312 Str.writeAddrOrOffset<IsELF64>(SymInfo.st_value);
313 Str.writeELFWord<IsELF64>(SymInfo.st_size);
314 Str.write8(SymInfo.st_info);
315 Str.write8(SymInfo.st_other);
316 Str.writeLE16(SymInfo.st_shndx);
317 }
318 }
319 }
320
321
322 } // end of namespace Ice
323
324 #endif // SUBZERO_SRC_ICEELFSECTION_H
OLDNEW
« no previous file with comments | « src/IceELFObjectWriter.cpp ('k') | src/IceELFSection.cpp » ('j') | src/IceMemoryRegion.h » ('J')

Powered by Google App Engine
This is Rietveld 408576698