xref: /freebsd/contrib/llvm-project/lld/COFF/DLL.cpp (revision 90ec6a30353aa7caaf995ea50e2e23aa5a099600)
1 //===- DLL.cpp ------------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file defines various types of chunks for the DLL import or export
10 // descriptor tables. They are inherently Windows-specific.
11 // You need to read Microsoft PE/COFF spec to understand details
12 // about the data structures.
13 //
14 // If you are not particularly interested in linking against Windows
15 // DLL, you can skip this file, and you should still be able to
16 // understand the rest of the linker.
17 //
18 //===----------------------------------------------------------------------===//
19 
20 #include "DLL.h"
21 #include "Chunks.h"
22 #include "llvm/Object/COFF.h"
23 #include "llvm/Support/Endian.h"
24 #include "llvm/Support/Path.h"
25 
26 using namespace llvm;
27 using namespace llvm::object;
28 using namespace llvm::support::endian;
29 using namespace llvm::COFF;
30 
31 namespace lld {
32 namespace coff {
33 namespace {
34 
35 // Import table
36 
37 // A chunk for the import descriptor table.
38 class HintNameChunk : public NonSectionChunk {
39 public:
40   HintNameChunk(StringRef n, uint16_t h) : name(n), hint(h) {}
41 
42   size_t getSize() const override {
43     // Starts with 2 byte Hint field, followed by a null-terminated string,
44     // ends with 0 or 1 byte padding.
45     return alignTo(name.size() + 3, 2);
46   }
47 
48   void writeTo(uint8_t *buf) const override {
49     memset(buf, 0, getSize());
50     write16le(buf, hint);
51     memcpy(buf + 2, name.data(), name.size());
52   }
53 
54 private:
55   StringRef name;
56   uint16_t hint;
57 };
58 
59 // A chunk for the import descriptor table.
60 class LookupChunk : public NonSectionChunk {
61 public:
62   explicit LookupChunk(Chunk *c) : hintName(c) {
63     setAlignment(config->wordsize);
64   }
65   size_t getSize() const override { return config->wordsize; }
66 
67   void writeTo(uint8_t *buf) const override {
68     if (config->is64())
69       write64le(buf, hintName->getRVA());
70     else
71       write32le(buf, hintName->getRVA());
72   }
73 
74   Chunk *hintName;
75 };
76 
77 // A chunk for the import descriptor table.
78 // This chunk represent import-by-ordinal symbols.
79 // See Microsoft PE/COFF spec 7.1. Import Header for details.
80 class OrdinalOnlyChunk : public NonSectionChunk {
81 public:
82   explicit OrdinalOnlyChunk(uint16_t v) : ordinal(v) {
83     setAlignment(config->wordsize);
84   }
85   size_t getSize() const override { return config->wordsize; }
86 
87   void writeTo(uint8_t *buf) const override {
88     // An import-by-ordinal slot has MSB 1 to indicate that
89     // this is import-by-ordinal (and not import-by-name).
90     if (config->is64()) {
91       write64le(buf, (1ULL << 63) | ordinal);
92     } else {
93       write32le(buf, (1ULL << 31) | ordinal);
94     }
95   }
96 
97   uint16_t ordinal;
98 };
99 
100 // A chunk for the import descriptor table.
101 class ImportDirectoryChunk : public NonSectionChunk {
102 public:
103   explicit ImportDirectoryChunk(Chunk *n) : dllName(n) {}
104   size_t getSize() const override { return sizeof(ImportDirectoryTableEntry); }
105 
106   void writeTo(uint8_t *buf) const override {
107     memset(buf, 0, getSize());
108 
109     auto *e = (coff_import_directory_table_entry *)(buf);
110     e->ImportLookupTableRVA = lookupTab->getRVA();
111     e->NameRVA = dllName->getRVA();
112     e->ImportAddressTableRVA = addressTab->getRVA();
113   }
114 
115   Chunk *dllName;
116   Chunk *lookupTab;
117   Chunk *addressTab;
118 };
119 
120 // A chunk representing null terminator in the import table.
121 // Contents of this chunk is always null bytes.
122 class NullChunk : public NonSectionChunk {
123 public:
124   explicit NullChunk(size_t n) : size(n) { hasData = false; }
125   size_t getSize() const override { return size; }
126 
127   void writeTo(uint8_t *buf) const override {
128     memset(buf, 0, size);
129   }
130 
131 private:
132   size_t size;
133 };
134 
135 static std::vector<std::vector<DefinedImportData *>>
136 binImports(const std::vector<DefinedImportData *> &imports) {
137   // Group DLL-imported symbols by DLL name because that's how
138   // symbols are laid out in the import descriptor table.
139   auto less = [](const std::string &a, const std::string &b) {
140     return config->dllOrder[a] < config->dllOrder[b];
141   };
142   std::map<std::string, std::vector<DefinedImportData *>,
143            bool(*)(const std::string &, const std::string &)> m(less);
144   for (DefinedImportData *sym : imports)
145     m[sym->getDLLName().lower()].push_back(sym);
146 
147   std::vector<std::vector<DefinedImportData *>> v;
148   for (auto &kv : m) {
149     // Sort symbols by name for each group.
150     std::vector<DefinedImportData *> &syms = kv.second;
151     std::sort(syms.begin(), syms.end(),
152               [](DefinedImportData *a, DefinedImportData *b) {
153                 return a->getName() < b->getName();
154               });
155     v.push_back(std::move(syms));
156   }
157   return v;
158 }
159 
160 // Export table
161 // See Microsoft PE/COFF spec 4.3 for details.
162 
163 // A chunk for the delay import descriptor table etnry.
164 class DelayDirectoryChunk : public NonSectionChunk {
165 public:
166   explicit DelayDirectoryChunk(Chunk *n) : dllName(n) {}
167 
168   size_t getSize() const override {
169     return sizeof(delay_import_directory_table_entry);
170   }
171 
172   void writeTo(uint8_t *buf) const override {
173     memset(buf, 0, getSize());
174 
175     auto *e = (delay_import_directory_table_entry *)(buf);
176     e->Attributes = 1;
177     e->Name = dllName->getRVA();
178     e->ModuleHandle = moduleHandle->getRVA();
179     e->DelayImportAddressTable = addressTab->getRVA();
180     e->DelayImportNameTable = nameTab->getRVA();
181   }
182 
183   Chunk *dllName;
184   Chunk *moduleHandle;
185   Chunk *addressTab;
186   Chunk *nameTab;
187 };
188 
189 // Initial contents for delay-loaded functions.
190 // This code calls __delayLoadHelper2 function to resolve a symbol
191 // which then overwrites its jump table slot with the result
192 // for subsequent function calls.
193 static const uint8_t thunkX64[] = {
194     0x48, 0x8D, 0x05, 0, 0, 0, 0,       // lea     rax, [__imp_<FUNCNAME>]
195     0xE9, 0, 0, 0, 0,                   // jmp     __tailMerge_<lib>
196 };
197 
198 static const uint8_t tailMergeX64[] = {
199     0x51,                               // push    rcx
200     0x52,                               // push    rdx
201     0x41, 0x50,                         // push    r8
202     0x41, 0x51,                         // push    r9
203     0x48, 0x83, 0xEC, 0x48,             // sub     rsp, 48h
204     0x66, 0x0F, 0x7F, 0x04, 0x24,       // movdqa  xmmword ptr [rsp], xmm0
205     0x66, 0x0F, 0x7F, 0x4C, 0x24, 0x10, // movdqa  xmmword ptr [rsp+10h], xmm1
206     0x66, 0x0F, 0x7F, 0x54, 0x24, 0x20, // movdqa  xmmword ptr [rsp+20h], xmm2
207     0x66, 0x0F, 0x7F, 0x5C, 0x24, 0x30, // movdqa  xmmword ptr [rsp+30h], xmm3
208     0x48, 0x8B, 0xD0,                   // mov     rdx, rax
209     0x48, 0x8D, 0x0D, 0, 0, 0, 0,       // lea     rcx, [___DELAY_IMPORT_...]
210     0xE8, 0, 0, 0, 0,                   // call    __delayLoadHelper2
211     0x66, 0x0F, 0x6F, 0x04, 0x24,       // movdqa  xmm0, xmmword ptr [rsp]
212     0x66, 0x0F, 0x6F, 0x4C, 0x24, 0x10, // movdqa  xmm1, xmmword ptr [rsp+10h]
213     0x66, 0x0F, 0x6F, 0x54, 0x24, 0x20, // movdqa  xmm2, xmmword ptr [rsp+20h]
214     0x66, 0x0F, 0x6F, 0x5C, 0x24, 0x30, // movdqa  xmm3, xmmword ptr [rsp+30h]
215     0x48, 0x83, 0xC4, 0x48,             // add     rsp, 48h
216     0x41, 0x59,                         // pop     r9
217     0x41, 0x58,                         // pop     r8
218     0x5A,                               // pop     rdx
219     0x59,                               // pop     rcx
220     0xFF, 0xE0,                         // jmp     rax
221 };
222 
223 static const uint8_t thunkX86[] = {
224     0xB8, 0, 0, 0, 0,  // mov   eax, offset ___imp__<FUNCNAME>
225     0xE9, 0, 0, 0, 0,  // jmp   __tailMerge_<lib>
226 };
227 
228 static const uint8_t tailMergeX86[] = {
229     0x51,              // push  ecx
230     0x52,              // push  edx
231     0x50,              // push  eax
232     0x68, 0, 0, 0, 0,  // push  offset ___DELAY_IMPORT_DESCRIPTOR_<DLLNAME>_dll
233     0xE8, 0, 0, 0, 0,  // call  ___delayLoadHelper2@8
234     0x5A,              // pop   edx
235     0x59,              // pop   ecx
236     0xFF, 0xE0,        // jmp   eax
237 };
238 
239 static const uint8_t thunkARM[] = {
240     0x40, 0xf2, 0x00, 0x0c, // mov.w   ip, #0 __imp_<FUNCNAME>
241     0xc0, 0xf2, 0x00, 0x0c, // mov.t   ip, #0 __imp_<FUNCNAME>
242     0x00, 0xf0, 0x00, 0xb8, // b.w     __tailMerge_<lib>
243 };
244 
245 static const uint8_t tailMergeARM[] = {
246     0x2d, 0xe9, 0x0f, 0x48, // push.w  {r0, r1, r2, r3, r11, lr}
247     0x0d, 0xf2, 0x10, 0x0b, // addw    r11, sp, #16
248     0x2d, 0xed, 0x10, 0x0b, // vpush   {d0, d1, d2, d3, d4, d5, d6, d7}
249     0x61, 0x46,             // mov     r1, ip
250     0x40, 0xf2, 0x00, 0x00, // mov.w   r0, #0 DELAY_IMPORT_DESCRIPTOR
251     0xc0, 0xf2, 0x00, 0x00, // mov.t   r0, #0 DELAY_IMPORT_DESCRIPTOR
252     0x00, 0xf0, 0x00, 0xd0, // bl      #0 __delayLoadHelper2
253     0x84, 0x46,             // mov     ip, r0
254     0xbd, 0xec, 0x10, 0x0b, // vpop    {d0, d1, d2, d3, d4, d5, d6, d7}
255     0xbd, 0xe8, 0x0f, 0x48, // pop.w   {r0, r1, r2, r3, r11, lr}
256     0x60, 0x47,             // bx      ip
257 };
258 
259 static const uint8_t thunkARM64[] = {
260     0x11, 0x00, 0x00, 0x90, // adrp    x17, #0      __imp_<FUNCNAME>
261     0x31, 0x02, 0x00, 0x91, // add     x17, x17, #0 :lo12:__imp_<FUNCNAME>
262     0x00, 0x00, 0x00, 0x14, // b       __tailMerge_<lib>
263 };
264 
265 static const uint8_t tailMergeARM64[] = {
266     0xfd, 0x7b, 0xb3, 0xa9, // stp     x29, x30, [sp, #-208]!
267     0xfd, 0x03, 0x00, 0x91, // mov     x29, sp
268     0xe0, 0x07, 0x01, 0xa9, // stp     x0, x1, [sp, #16]
269     0xe2, 0x0f, 0x02, 0xa9, // stp     x2, x3, [sp, #32]
270     0xe4, 0x17, 0x03, 0xa9, // stp     x4, x5, [sp, #48]
271     0xe6, 0x1f, 0x04, 0xa9, // stp     x6, x7, [sp, #64]
272     0xe0, 0x87, 0x02, 0xad, // stp     q0, q1, [sp, #80]
273     0xe2, 0x8f, 0x03, 0xad, // stp     q2, q3, [sp, #112]
274     0xe4, 0x97, 0x04, 0xad, // stp     q4, q5, [sp, #144]
275     0xe6, 0x9f, 0x05, 0xad, // stp     q6, q7, [sp, #176]
276     0xe1, 0x03, 0x11, 0xaa, // mov     x1, x17
277     0x00, 0x00, 0x00, 0x90, // adrp    x0, #0     DELAY_IMPORT_DESCRIPTOR
278     0x00, 0x00, 0x00, 0x91, // add     x0, x0, #0 :lo12:DELAY_IMPORT_DESCRIPTOR
279     0x00, 0x00, 0x00, 0x94, // bl      #0 __delayLoadHelper2
280     0xf0, 0x03, 0x00, 0xaa, // mov     x16, x0
281     0xe6, 0x9f, 0x45, 0xad, // ldp     q6, q7, [sp, #176]
282     0xe4, 0x97, 0x44, 0xad, // ldp     q4, q5, [sp, #144]
283     0xe2, 0x8f, 0x43, 0xad, // ldp     q2, q3, [sp, #112]
284     0xe0, 0x87, 0x42, 0xad, // ldp     q0, q1, [sp, #80]
285     0xe6, 0x1f, 0x44, 0xa9, // ldp     x6, x7, [sp, #64]
286     0xe4, 0x17, 0x43, 0xa9, // ldp     x4, x5, [sp, #48]
287     0xe2, 0x0f, 0x42, 0xa9, // ldp     x2, x3, [sp, #32]
288     0xe0, 0x07, 0x41, 0xa9, // ldp     x0, x1, [sp, #16]
289     0xfd, 0x7b, 0xcd, 0xa8, // ldp     x29, x30, [sp], #208
290     0x00, 0x02, 0x1f, 0xd6, // br      x16
291 };
292 
293 // A chunk for the delay import thunk.
294 class ThunkChunkX64 : public NonSectionChunk {
295 public:
296   ThunkChunkX64(Defined *i, Chunk *tm) : imp(i), tailMerge(tm) {}
297 
298   size_t getSize() const override { return sizeof(thunkX64); }
299 
300   void writeTo(uint8_t *buf) const override {
301     memcpy(buf, thunkX64, sizeof(thunkX64));
302     write32le(buf + 3, imp->getRVA() - rva - 7);
303     write32le(buf + 8, tailMerge->getRVA() - rva - 12);
304   }
305 
306   Defined *imp = nullptr;
307   Chunk *tailMerge = nullptr;
308 };
309 
310 class TailMergeChunkX64 : public NonSectionChunk {
311 public:
312   TailMergeChunkX64(Chunk *d, Defined *h) : desc(d), helper(h) {}
313 
314   size_t getSize() const override { return sizeof(tailMergeX64); }
315 
316   void writeTo(uint8_t *buf) const override {
317     memcpy(buf, tailMergeX64, sizeof(tailMergeX64));
318     write32le(buf + 39, desc->getRVA() - rva - 43);
319     write32le(buf + 44, helper->getRVA() - rva - 48);
320   }
321 
322   Chunk *desc = nullptr;
323   Defined *helper = nullptr;
324 };
325 
326 class ThunkChunkX86 : public NonSectionChunk {
327 public:
328   ThunkChunkX86(Defined *i, Chunk *tm) : imp(i), tailMerge(tm) {}
329 
330   size_t getSize() const override { return sizeof(thunkX86); }
331 
332   void writeTo(uint8_t *buf) const override {
333     memcpy(buf, thunkX86, sizeof(thunkX86));
334     write32le(buf + 1, imp->getRVA() + config->imageBase);
335     write32le(buf + 6, tailMerge->getRVA() - rva - 10);
336   }
337 
338   void getBaserels(std::vector<Baserel> *res) override {
339     res->emplace_back(rva + 1);
340   }
341 
342   Defined *imp = nullptr;
343   Chunk *tailMerge = nullptr;
344 };
345 
346 class TailMergeChunkX86 : public NonSectionChunk {
347 public:
348   TailMergeChunkX86(Chunk *d, Defined *h) : desc(d), helper(h) {}
349 
350   size_t getSize() const override { return sizeof(tailMergeX86); }
351 
352   void writeTo(uint8_t *buf) const override {
353     memcpy(buf, tailMergeX86, sizeof(tailMergeX86));
354     write32le(buf + 4, desc->getRVA() + config->imageBase);
355     write32le(buf + 9, helper->getRVA() - rva - 13);
356   }
357 
358   void getBaserels(std::vector<Baserel> *res) override {
359     res->emplace_back(rva + 4);
360   }
361 
362   Chunk *desc = nullptr;
363   Defined *helper = nullptr;
364 };
365 
366 class ThunkChunkARM : public NonSectionChunk {
367 public:
368   ThunkChunkARM(Defined *i, Chunk *tm) : imp(i), tailMerge(tm) {
369     setAlignment(2);
370   }
371 
372   size_t getSize() const override { return sizeof(thunkARM); }
373 
374   void writeTo(uint8_t *buf) const override {
375     memcpy(buf, thunkARM, sizeof(thunkARM));
376     applyMOV32T(buf + 0, imp->getRVA() + config->imageBase);
377     applyBranch24T(buf + 8, tailMerge->getRVA() - rva - 12);
378   }
379 
380   void getBaserels(std::vector<Baserel> *res) override {
381     res->emplace_back(rva + 0, IMAGE_REL_BASED_ARM_MOV32T);
382   }
383 
384   Defined *imp = nullptr;
385   Chunk *tailMerge = nullptr;
386 };
387 
388 class TailMergeChunkARM : public NonSectionChunk {
389 public:
390   TailMergeChunkARM(Chunk *d, Defined *h) : desc(d), helper(h) {
391     setAlignment(2);
392   }
393 
394   size_t getSize() const override { return sizeof(tailMergeARM); }
395 
396   void writeTo(uint8_t *buf) const override {
397     memcpy(buf, tailMergeARM, sizeof(tailMergeARM));
398     applyMOV32T(buf + 14, desc->getRVA() + config->imageBase);
399     applyBranch24T(buf + 22, helper->getRVA() - rva - 26);
400   }
401 
402   void getBaserels(std::vector<Baserel> *res) override {
403     res->emplace_back(rva + 14, IMAGE_REL_BASED_ARM_MOV32T);
404   }
405 
406   Chunk *desc = nullptr;
407   Defined *helper = nullptr;
408 };
409 
410 class ThunkChunkARM64 : public NonSectionChunk {
411 public:
412   ThunkChunkARM64(Defined *i, Chunk *tm) : imp(i), tailMerge(tm) {
413     setAlignment(4);
414   }
415 
416   size_t getSize() const override { return sizeof(thunkARM64); }
417 
418   void writeTo(uint8_t *buf) const override {
419     memcpy(buf, thunkARM64, sizeof(thunkARM64));
420     applyArm64Addr(buf + 0, imp->getRVA(), rva + 0, 12);
421     applyArm64Imm(buf + 4, imp->getRVA() & 0xfff, 0);
422     applyArm64Branch26(buf + 8, tailMerge->getRVA() - rva - 8);
423   }
424 
425   Defined *imp = nullptr;
426   Chunk *tailMerge = nullptr;
427 };
428 
429 class TailMergeChunkARM64 : public NonSectionChunk {
430 public:
431   TailMergeChunkARM64(Chunk *d, Defined *h) : desc(d), helper(h) {
432     setAlignment(4);
433   }
434 
435   size_t getSize() const override { return sizeof(tailMergeARM64); }
436 
437   void writeTo(uint8_t *buf) const override {
438     memcpy(buf, tailMergeARM64, sizeof(tailMergeARM64));
439     applyArm64Addr(buf + 44, desc->getRVA(), rva + 44, 12);
440     applyArm64Imm(buf + 48, desc->getRVA() & 0xfff, 0);
441     applyArm64Branch26(buf + 52, helper->getRVA() - rva - 52);
442   }
443 
444   Chunk *desc = nullptr;
445   Defined *helper = nullptr;
446 };
447 
448 // A chunk for the import descriptor table.
449 class DelayAddressChunk : public NonSectionChunk {
450 public:
451   explicit DelayAddressChunk(Chunk *c) : thunk(c) {
452     setAlignment(config->wordsize);
453   }
454   size_t getSize() const override { return config->wordsize; }
455 
456   void writeTo(uint8_t *buf) const override {
457     if (config->is64()) {
458       write64le(buf, thunk->getRVA() + config->imageBase);
459     } else {
460       uint32_t bit = 0;
461       // Pointer to thumb code must have the LSB set, so adjust it.
462       if (config->machine == ARMNT)
463         bit = 1;
464       write32le(buf, (thunk->getRVA() + config->imageBase) | bit);
465     }
466   }
467 
468   void getBaserels(std::vector<Baserel> *res) override {
469     res->emplace_back(rva);
470   }
471 
472   Chunk *thunk;
473 };
474 
475 // Export table
476 // Read Microsoft PE/COFF spec 5.3 for details.
477 
478 // A chunk for the export descriptor table.
479 class ExportDirectoryChunk : public NonSectionChunk {
480 public:
481   ExportDirectoryChunk(int i, int j, Chunk *d, Chunk *a, Chunk *n, Chunk *o)
482       : maxOrdinal(i), nameTabSize(j), dllName(d), addressTab(a), nameTab(n),
483         ordinalTab(o) {}
484 
485   size_t getSize() const override {
486     return sizeof(export_directory_table_entry);
487   }
488 
489   void writeTo(uint8_t *buf) const override {
490     memset(buf, 0, getSize());
491 
492     auto *e = (export_directory_table_entry *)(buf);
493     e->NameRVA = dllName->getRVA();
494     e->OrdinalBase = 0;
495     e->AddressTableEntries = maxOrdinal + 1;
496     e->NumberOfNamePointers = nameTabSize;
497     e->ExportAddressTableRVA = addressTab->getRVA();
498     e->NamePointerRVA = nameTab->getRVA();
499     e->OrdinalTableRVA = ordinalTab->getRVA();
500   }
501 
502   uint16_t maxOrdinal;
503   uint16_t nameTabSize;
504   Chunk *dllName;
505   Chunk *addressTab;
506   Chunk *nameTab;
507   Chunk *ordinalTab;
508 };
509 
510 class AddressTableChunk : public NonSectionChunk {
511 public:
512   explicit AddressTableChunk(size_t maxOrdinal) : size(maxOrdinal + 1) {}
513   size_t getSize() const override { return size * 4; }
514 
515   void writeTo(uint8_t *buf) const override {
516     memset(buf, 0, getSize());
517 
518     for (const Export &e : config->exports) {
519       uint8_t *p = buf + e.ordinal * 4;
520       uint32_t bit = 0;
521       // Pointer to thumb code must have the LSB set, so adjust it.
522       if (config->machine == ARMNT && !e.data)
523         bit = 1;
524       if (e.forwardChunk) {
525         write32le(p, e.forwardChunk->getRVA() | bit);
526       } else {
527         write32le(p, cast<Defined>(e.sym)->getRVA() | bit);
528       }
529     }
530   }
531 
532 private:
533   size_t size;
534 };
535 
536 class NamePointersChunk : public NonSectionChunk {
537 public:
538   explicit NamePointersChunk(std::vector<Chunk *> &v) : chunks(v) {}
539   size_t getSize() const override { return chunks.size() * 4; }
540 
541   void writeTo(uint8_t *buf) const override {
542     for (Chunk *c : chunks) {
543       write32le(buf, c->getRVA());
544       buf += 4;
545     }
546   }
547 
548 private:
549   std::vector<Chunk *> chunks;
550 };
551 
552 class ExportOrdinalChunk : public NonSectionChunk {
553 public:
554   explicit ExportOrdinalChunk(size_t i) : size(i) {}
555   size_t getSize() const override { return size * 2; }
556 
557   void writeTo(uint8_t *buf) const override {
558     for (Export &e : config->exports) {
559       if (e.noname)
560         continue;
561       write16le(buf, e.ordinal);
562       buf += 2;
563     }
564   }
565 
566 private:
567   size_t size;
568 };
569 
570 } // anonymous namespace
571 
572 void IdataContents::create() {
573   std::vector<std::vector<DefinedImportData *>> v = binImports(imports);
574 
575   // Create .idata contents for each DLL.
576   for (std::vector<DefinedImportData *> &syms : v) {
577     // Create lookup and address tables. If they have external names,
578     // we need to create hintName chunks to store the names.
579     // If they don't (if they are import-by-ordinals), we store only
580     // ordinal values to the table.
581     size_t base = lookups.size();
582     for (DefinedImportData *s : syms) {
583       uint16_t ord = s->getOrdinal();
584       if (s->getExternalName().empty()) {
585         lookups.push_back(make<OrdinalOnlyChunk>(ord));
586         addresses.push_back(make<OrdinalOnlyChunk>(ord));
587         continue;
588       }
589       auto *c = make<HintNameChunk>(s->getExternalName(), ord);
590       lookups.push_back(make<LookupChunk>(c));
591       addresses.push_back(make<LookupChunk>(c));
592       hints.push_back(c);
593     }
594     // Terminate with null values.
595     lookups.push_back(make<NullChunk>(config->wordsize));
596     addresses.push_back(make<NullChunk>(config->wordsize));
597 
598     for (int i = 0, e = syms.size(); i < e; ++i)
599       syms[i]->setLocation(addresses[base + i]);
600 
601     // Create the import table header.
602     dllNames.push_back(make<StringChunk>(syms[0]->getDLLName()));
603     auto *dir = make<ImportDirectoryChunk>(dllNames.back());
604     dir->lookupTab = lookups[base];
605     dir->addressTab = addresses[base];
606     dirs.push_back(dir);
607   }
608   // Add null terminator.
609   dirs.push_back(make<NullChunk>(sizeof(ImportDirectoryTableEntry)));
610 }
611 
612 std::vector<Chunk *> DelayLoadContents::getChunks() {
613   std::vector<Chunk *> v;
614   v.insert(v.end(), dirs.begin(), dirs.end());
615   v.insert(v.end(), names.begin(), names.end());
616   v.insert(v.end(), hintNames.begin(), hintNames.end());
617   v.insert(v.end(), dllNames.begin(), dllNames.end());
618   return v;
619 }
620 
621 std::vector<Chunk *> DelayLoadContents::getDataChunks() {
622   std::vector<Chunk *> v;
623   v.insert(v.end(), moduleHandles.begin(), moduleHandles.end());
624   v.insert(v.end(), addresses.begin(), addresses.end());
625   return v;
626 }
627 
628 uint64_t DelayLoadContents::getDirSize() {
629   return dirs.size() * sizeof(delay_import_directory_table_entry);
630 }
631 
632 void DelayLoadContents::create(Defined *h) {
633   helper = h;
634   std::vector<std::vector<DefinedImportData *>> v = binImports(imports);
635 
636   // Create .didat contents for each DLL.
637   for (std::vector<DefinedImportData *> &syms : v) {
638     // Create the delay import table header.
639     dllNames.push_back(make<StringChunk>(syms[0]->getDLLName()));
640     auto *dir = make<DelayDirectoryChunk>(dllNames.back());
641 
642     size_t base = addresses.size();
643     Chunk *tm = newTailMergeChunk(dir);
644     for (DefinedImportData *s : syms) {
645       Chunk *t = newThunkChunk(s, tm);
646       auto *a = make<DelayAddressChunk>(t);
647       addresses.push_back(a);
648       thunks.push_back(t);
649       StringRef extName = s->getExternalName();
650       if (extName.empty()) {
651         names.push_back(make<OrdinalOnlyChunk>(s->getOrdinal()));
652       } else {
653         auto *c = make<HintNameChunk>(extName, 0);
654         names.push_back(make<LookupChunk>(c));
655         hintNames.push_back(c);
656       }
657     }
658     thunks.push_back(tm);
659     // Terminate with null values.
660     addresses.push_back(make<NullChunk>(8));
661     names.push_back(make<NullChunk>(8));
662 
663     for (int i = 0, e = syms.size(); i < e; ++i)
664       syms[i]->setLocation(addresses[base + i]);
665     auto *mh = make<NullChunk>(8);
666     mh->setAlignment(8);
667     moduleHandles.push_back(mh);
668 
669     // Fill the delay import table header fields.
670     dir->moduleHandle = mh;
671     dir->addressTab = addresses[base];
672     dir->nameTab = names[base];
673     dirs.push_back(dir);
674   }
675   // Add null terminator.
676   dirs.push_back(make<NullChunk>(sizeof(delay_import_directory_table_entry)));
677 }
678 
679 Chunk *DelayLoadContents::newTailMergeChunk(Chunk *dir) {
680   switch (config->machine) {
681   case AMD64:
682     return make<TailMergeChunkX64>(dir, helper);
683   case I386:
684     return make<TailMergeChunkX86>(dir, helper);
685   case ARMNT:
686     return make<TailMergeChunkARM>(dir, helper);
687   case ARM64:
688     return make<TailMergeChunkARM64>(dir, helper);
689   default:
690     llvm_unreachable("unsupported machine type");
691   }
692 }
693 
694 Chunk *DelayLoadContents::newThunkChunk(DefinedImportData *s,
695                                         Chunk *tailMerge) {
696   switch (config->machine) {
697   case AMD64:
698     return make<ThunkChunkX64>(s, tailMerge);
699   case I386:
700     return make<ThunkChunkX86>(s, tailMerge);
701   case ARMNT:
702     return make<ThunkChunkARM>(s, tailMerge);
703   case ARM64:
704     return make<ThunkChunkARM64>(s, tailMerge);
705   default:
706     llvm_unreachable("unsupported machine type");
707   }
708 }
709 
710 EdataContents::EdataContents() {
711   uint16_t maxOrdinal = 0;
712   for (Export &e : config->exports)
713     maxOrdinal = std::max(maxOrdinal, e.ordinal);
714 
715   auto *dllName = make<StringChunk>(sys::path::filename(config->outputFile));
716   auto *addressTab = make<AddressTableChunk>(maxOrdinal);
717   std::vector<Chunk *> names;
718   for (Export &e : config->exports)
719     if (!e.noname)
720       names.push_back(make<StringChunk>(e.exportName));
721 
722   std::vector<Chunk *> forwards;
723   for (Export &e : config->exports) {
724     if (e.forwardTo.empty())
725       continue;
726     e.forwardChunk = make<StringChunk>(e.forwardTo);
727     forwards.push_back(e.forwardChunk);
728   }
729 
730   auto *nameTab = make<NamePointersChunk>(names);
731   auto *ordinalTab = make<ExportOrdinalChunk>(names.size());
732   auto *dir = make<ExportDirectoryChunk>(maxOrdinal, names.size(), dllName,
733                                          addressTab, nameTab, ordinalTab);
734   chunks.push_back(dir);
735   chunks.push_back(dllName);
736   chunks.push_back(addressTab);
737   chunks.push_back(nameTab);
738   chunks.push_back(ordinalTab);
739   chunks.insert(chunks.end(), names.begin(), names.end());
740   chunks.insert(chunks.end(), forwards.begin(), forwards.end());
741 }
742 
743 } // namespace coff
744 } // namespace lld
745