OLD | NEW |
(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 |
OLD | NEW |