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

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

Powered by Google App Engine
This is Rietveld 408576698