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

Side by Side Diff: src/IceELFSection.cpp

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: stuff 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.cpp - Representation of ELF sections -----===//
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 // This file defines how ELF sections are represented.
11 //
12 //===----------------------------------------------------------------------===//
13
14 #include "IceDefs.h"
15 #include "IceELFSection.h"
16 #include "IceELFStreamer.h"
17
18 using namespace llvm::ELF;
19
20 namespace Ice {
21
22 // The dummy NULL section will be assigned number 0, and it is referenced
23 // by the dummy 0-th symbol in the symbol table, so we can't use that.
24 const SizeT ELFSection::NoSectionNumber = std::numeric_limits<SizeT>::max();
25
26 // Text sections.
27
28 void ELFTextSection::appendData(ELFStreamer &Str,
29 const llvm::StringRef MoreData) {
30 Str.writeBytes(MoreData);
31 Header.sh_size += MoreData.size();
32 }
33
34 // Data sections.
35
36 void ELFDataSection::appendData(ELFStreamer &Str,
37 const llvm::StringRef MoreData) {
38 Str.writeBytes(MoreData);
39 Header.sh_size += MoreData.size();
40 }
41
42 // Relocation sections.
43
44 // Symbol tables.
45
46 // The dummy 0-th symbol will be assigned number 0, so don't use that.
47 const SizeT ELFSym::UnknownNumber = std::numeric_limits<SizeT>::max();
48
49 void ELFSymbolTableSection::createDefinedSym(const IceString &Name,
50 uint8_t Type, uint8_t Binding,
51 ELFSection *Section,
52 RelocOffsetT Offset, SizeT Size) {
53 ELFSym NewSymbol = ELFSym();
54 NewSymbol.Sym.setBindingAndType(Binding, Type);
55 NewSymbol.Sym.st_value = Offset;
56 NewSymbol.Sym.st_size = Size;
57 NewSymbol.Number = ELFSym::UnknownNumber;
58 SymtabKey Key = {Name, Section};
59 bool Unique;
60 if (Type == STB_LOCAL)
61 Unique = LocalSymbols.insert(std::make_pair(Key, NewSymbol)).second;
62 else
63 Unique = GlobalSymbols.insert(std::make_pair(Key, NewSymbol)).second;
64 assert(Unique);
65 }
66
67 void ELFSymbolTableSection::noteUndefinedSym(const IceString &Name,
68 ELFSection *NullSection) {
69 ELFSym NewSymbol = ELFSym();
70 NewSymbol.Sym.setBindingAndType(STB_GLOBAL, STT_NOTYPE);
71 NewSymbol.Number = ELFSym::UnknownNumber;
72 SymtabKey Key = {Name, NullSection};
73 GlobalSymbols.insert(std::make_pair(Key, NewSymbol));
74 }
75
76 void ELFSymbolTableSection::updateIndices(const ELFStringTableSection *StrTab) {
77 SizeT SymNumber = 0;
78 for (auto &KeyValue : LocalSymbols) {
79 const IceString &Name = KeyValue.first.first;
80 ELFSection *Section = KeyValue.first.second;
81 Elf64_Sym &SymInfo = KeyValue.second.Sym;
82 if (!Name.empty())
83 SymInfo.st_name = StrTab->getIndex(Name);
84 SymInfo.st_shndx = Section->getNumber();
85 KeyValue.second.setNumber(SymNumber++);
86 }
87 for (auto &KeyValue : GlobalSymbols) {
88 const IceString &Name = KeyValue.first.first;
89 ELFSection *Section = KeyValue.first.second;
90 Elf64_Sym &SymInfo = KeyValue.second.Sym;
91 if (!Name.empty())
92 SymInfo.st_name = StrTab->getIndex(Name);
93 SymInfo.st_shndx = Section->getNumber();
94 KeyValue.second.setNumber(SymNumber++);
95 }
96 }
97
98 void ELFSymbolTableSection::writeData(ELFStreamer &Str, bool IsELF64) {
99 if (IsELF64) {
100 writeSymbolMap<true>(Str, LocalSymbols);
101 writeSymbolMap<true>(Str, GlobalSymbols);
102 } else {
103 writeSymbolMap<false>(Str, LocalSymbols);
104 writeSymbolMap<false>(Str, GlobalSymbols);
105 }
106 }
107
108 // String tables.
109
110 // The first byte of the string table should be \00, so it is an
111 // invalid index. Indices start out as unknown until layout is complete.
112 const size_t ELFStringTableSection::UnknownIndex = 0;
113
114 void ELFStringTableSection::add(const IceString &Str) {
115 assert(!isLaidOut());
116 assert(!Str.empty());
117 StringToIndexMap.insert(std::make_pair(Str, UnknownIndex));
118 }
119
120 size_t ELFStringTableSection::getIndex(const IceString &Str) const {
121 assert(isLaidOut());
122 StringToIndexType::const_iterator It = StringToIndexMap.find(Str);
123 if (It == StringToIndexMap.end()) {
124 llvm_unreachable("String index not found");
125 return UnknownIndex;
126 }
127 return It->second;
128 }
129
130 bool ELFStringTableSection::SuffixComparator::
131 operator()(const IceString &StrA, const IceString &StrB) const {
132 size_t LenA = StrA.size();
133 size_t LenB = StrB.size();
134 size_t CommonLen = std::min(LenA, LenB);
135 // If there is a difference in the common suffix, use that diff to sort.
136 for (size_t i = 0; i < CommonLen; ++i) {
137 char a = StrA[LenA - i - 1];
138 char b = StrB[LenB - i - 1];
139 if (a != b)
140 return a > b;
141 }
142 // If the common suffixes are completely equal, let the longer one come
143 // first, so that it can be laid out first and its characters shared.
144 return LenA > LenB;
145 }
146
147 void ELFStringTableSection::doLayout() {
148 assert(!isLaidOut());
149 llvm::StringRef Prev;
150
151 // String table starts with 0 byte.
152 StringData.push_back(0);
153
154 for (auto &StringIndex : StringToIndexMap) {
155 assert(StringIndex.second == UnknownIndex);
156 llvm::StringRef Cur = llvm::StringRef(StringIndex.first);
157 if (Prev.endswith(Cur)) {
158 // Prev is already in the StringData, and Cur is shorter than Prev
159 // based on the sort.
160 StringIndex.second = StringData.size() - Cur.size() - 1;
161 continue;
162 }
163 StringIndex.second = StringData.size();
164 std::copy(Cur.begin(), Cur.end(), back_inserter(StringData));
165 StringData.push_back(0);
166 Prev = Cur;
167 }
168 }
169
170 } // end of namespace Ice
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698