1
2
3
4
5
6
7
8 package ld
9
10 import (
11 "cmd/internal/objabi"
12 "cmd/internal/sys"
13 "cmd/link/internal/loader"
14 "cmd/link/internal/sym"
15 "debug/pe"
16 "encoding/binary"
17 "fmt"
18 "internal/buildcfg"
19 "math"
20 "slices"
21 "sort"
22 "strconv"
23 "strings"
24 )
25
26 type IMAGE_IMPORT_DESCRIPTOR struct {
27 OriginalFirstThunk uint32
28 TimeDateStamp uint32
29 ForwarderChain uint32
30 Name uint32
31 FirstThunk uint32
32 }
33
34 type IMAGE_EXPORT_DIRECTORY struct {
35 Characteristics uint32
36 TimeDateStamp uint32
37 MajorVersion uint16
38 MinorVersion uint16
39 Name uint32
40 Base uint32
41 NumberOfFunctions uint32
42 NumberOfNames uint32
43 AddressOfFunctions uint32
44 AddressOfNames uint32
45 AddressOfNameOrdinals uint32
46 }
47
48 var (
49
50
51 PEBASE int64
52
53
54
55 PESECTALIGN int64 = 0x1000
56
57
58
59
60 PEFILEALIGN int64 = 2 << 8
61 )
62
63 const (
64 IMAGE_SCN_CNT_CODE = 0x00000020
65 IMAGE_SCN_CNT_INITIALIZED_DATA = 0x00000040
66 IMAGE_SCN_CNT_UNINITIALIZED_DATA = 0x00000080
67 IMAGE_SCN_LNK_OTHER = 0x00000100
68 IMAGE_SCN_LNK_INFO = 0x00000200
69 IMAGE_SCN_LNK_REMOVE = 0x00000800
70 IMAGE_SCN_LNK_COMDAT = 0x00001000
71 IMAGE_SCN_GPREL = 0x00008000
72 IMAGE_SCN_MEM_PURGEABLE = 0x00020000
73 IMAGE_SCN_MEM_16BIT = 0x00020000
74 IMAGE_SCN_MEM_LOCKED = 0x00040000
75 IMAGE_SCN_MEM_PRELOAD = 0x00080000
76 IMAGE_SCN_ALIGN_1BYTES = 0x00100000
77 IMAGE_SCN_ALIGN_2BYTES = 0x00200000
78 IMAGE_SCN_ALIGN_4BYTES = 0x00300000
79 IMAGE_SCN_ALIGN_8BYTES = 0x00400000
80 IMAGE_SCN_ALIGN_16BYTES = 0x00500000
81 IMAGE_SCN_ALIGN_32BYTES = 0x00600000
82 IMAGE_SCN_ALIGN_64BYTES = 0x00700000
83 IMAGE_SCN_ALIGN_128BYTES = 0x00800000
84 IMAGE_SCN_ALIGN_256BYTES = 0x00900000
85 IMAGE_SCN_ALIGN_512BYTES = 0x00A00000
86 IMAGE_SCN_ALIGN_1024BYTES = 0x00B00000
87 IMAGE_SCN_ALIGN_2048BYTES = 0x00C00000
88 IMAGE_SCN_ALIGN_4096BYTES = 0x00D00000
89 IMAGE_SCN_ALIGN_8192BYTES = 0x00E00000
90 IMAGE_SCN_LNK_NRELOC_OVFL = 0x01000000
91 IMAGE_SCN_MEM_DISCARDABLE = 0x02000000
92 IMAGE_SCN_MEM_NOT_CACHED = 0x04000000
93 IMAGE_SCN_MEM_NOT_PAGED = 0x08000000
94 IMAGE_SCN_MEM_SHARED = 0x10000000
95 IMAGE_SCN_MEM_EXECUTE = 0x20000000
96 IMAGE_SCN_MEM_READ = 0x40000000
97 IMAGE_SCN_MEM_WRITE = 0x80000000
98 )
99
100
101
102 const (
103 IMAGE_SYM_TYPE_NULL = 0
104 IMAGE_SYM_TYPE_STRUCT = 8
105 IMAGE_SYM_DTYPE_FUNCTION = 2
106 IMAGE_SYM_DTYPE_ARRAY = 3
107 IMAGE_SYM_CLASS_EXTERNAL = 2
108 IMAGE_SYM_CLASS_STATIC = 3
109
110 IMAGE_REL_I386_DIR32 = 0x0006
111 IMAGE_REL_I386_DIR32NB = 0x0007
112 IMAGE_REL_I386_SECREL = 0x000B
113 IMAGE_REL_I386_REL32 = 0x0014
114
115 IMAGE_REL_AMD64_ADDR64 = 0x0001
116 IMAGE_REL_AMD64_ADDR32 = 0x0002
117 IMAGE_REL_AMD64_ADDR32NB = 0x0003
118 IMAGE_REL_AMD64_REL32 = 0x0004
119 IMAGE_REL_AMD64_SECREL = 0x000B
120
121 IMAGE_REL_ARM_ABSOLUTE = 0x0000
122 IMAGE_REL_ARM_ADDR32 = 0x0001
123 IMAGE_REL_ARM_ADDR32NB = 0x0002
124 IMAGE_REL_ARM_BRANCH24 = 0x0003
125 IMAGE_REL_ARM_BRANCH11 = 0x0004
126 IMAGE_REL_ARM_SECREL = 0x000F
127
128 IMAGE_REL_ARM64_ABSOLUTE = 0x0000
129 IMAGE_REL_ARM64_ADDR32 = 0x0001
130 IMAGE_REL_ARM64_ADDR32NB = 0x0002
131 IMAGE_REL_ARM64_BRANCH26 = 0x0003
132 IMAGE_REL_ARM64_PAGEBASE_REL21 = 0x0004
133 IMAGE_REL_ARM64_REL21 = 0x0005
134 IMAGE_REL_ARM64_PAGEOFFSET_12A = 0x0006
135 IMAGE_REL_ARM64_PAGEOFFSET_12L = 0x0007
136 IMAGE_REL_ARM64_SECREL = 0x0008
137 IMAGE_REL_ARM64_SECREL_LOW12A = 0x0009
138 IMAGE_REL_ARM64_SECREL_HIGH12A = 0x000A
139 IMAGE_REL_ARM64_SECREL_LOW12L = 0x000B
140 IMAGE_REL_ARM64_TOKEN = 0x000C
141 IMAGE_REL_ARM64_SECTION = 0x000D
142 IMAGE_REL_ARM64_ADDR64 = 0x000E
143 IMAGE_REL_ARM64_BRANCH19 = 0x000F
144 IMAGE_REL_ARM64_BRANCH14 = 0x0010
145 IMAGE_REL_ARM64_REL32 = 0x0011
146
147 IMAGE_REL_BASED_HIGHLOW = 3
148 IMAGE_REL_BASED_DIR64 = 10
149 )
150
151 const (
152 PeMinimumTargetMajorVersion = 6
153 PeMinimumTargetMinorVersion = 1
154 )
155
156
157
158
159 var dosstub = []uint8{
160 0x4d,
161 0x5a,
162 0x90,
163 0x00,
164 0x03,
165 0x00,
166 0x00,
167 0x00,
168 0x04,
169 0x00,
170 0x00,
171 0x00,
172 0xff,
173 0xff,
174 0x00,
175 0x00,
176 0x8b,
177 0x00,
178 0x00,
179 0x00,
180 0x00,
181 0x00,
182 0x00,
183 0x00,
184 0x40,
185 0x00,
186 0x00,
187 0x00,
188 0x00,
189 0x00,
190 0x00,
191 0x00,
192 0x00,
193 0x00,
194 0x00,
195 0x00,
196 0x00,
197 0x00,
198 0x00,
199 0x00,
200 0x00,
201 0x00,
202 0x00,
203 0x00,
204 0x00,
205 0x00,
206 0x00,
207 0x00,
208 0x00,
209 0x00,
210 0x00,
211 0x00,
212 0x00,
213 0x00,
214 0x00,
215 0x00,
216 0x00,
217 0x00,
218 0x00,
219 0x00,
220 0x80,
221 0x00,
222 0x00,
223 0x00,
224 0x0e,
225 0x1f,
226 0xba,
227 0x0e,
228 0x00,
229 0xb4,
230 0x09,
231 0xcd,
232 0x21,
233 0xb8,
234 0x01,
235 0x4c,
236 0xcd,
237 0x21,
238 0x54,
239 0x68,
240 0x69,
241 0x73,
242 0x20,
243 0x70,
244 0x72,
245 0x6f,
246 0x67,
247 0x72,
248 0x61,
249 0x6d,
250 0x20,
251 0x63,
252 0x61,
253 0x6e,
254 0x6e,
255 0x6f,
256 0x74,
257 0x20,
258 0x62,
259 0x65,
260 0x20,
261 0x72,
262 0x75,
263 0x6e,
264 0x20,
265 0x69,
266 0x6e,
267 0x20,
268 0x44,
269 0x4f,
270 0x53,
271 0x20,
272 0x6d,
273 0x6f,
274 0x64,
275 0x65,
276 0x2e,
277 0x0d,
278 0x0d,
279 0x0a,
280 0x24,
281 0x00,
282 0x00,
283 0x00,
284 0x00,
285 0x00,
286 0x00,
287 0x00,
288 }
289
290 type Imp struct {
291 s loader.Sym
292 off uint64
293 next *Imp
294 argsize int
295 }
296
297 type Dll struct {
298 name string
299 nameoff uint64
300 thunkoff uint64
301 ms *Imp
302 next *Dll
303 }
304
305 var (
306 rsrcsyms []loader.Sym
307 PESECTHEADR int32
308 PEFILEHEADR int32
309 pe64 bool
310 dr *Dll
311
312 dexport []loader.Sym
313 )
314
315
316 type peStringTable struct {
317 strings []string
318 stringsLen int
319 }
320
321
322 func (t *peStringTable) size() int {
323
324 return t.stringsLen + 4
325 }
326
327
328 func (t *peStringTable) add(str string) int {
329 off := t.size()
330 t.strings = append(t.strings, str)
331 t.stringsLen += len(str) + 1
332 return off
333 }
334
335
336 func (t *peStringTable) write(out *OutBuf) {
337 out.Write32(uint32(t.size()))
338 for _, s := range t.strings {
339 out.WriteString(s)
340 out.Write8(0)
341 }
342 }
343
344
345 type peSection struct {
346 name string
347 shortName string
348 index int
349 virtualSize uint32
350 virtualAddress uint32
351 sizeOfRawData uint32
352 pointerToRawData uint32
353 pointerToRelocations uint32
354 numberOfRelocations uint16
355 characteristics uint32
356 }
357
358
359 func (sect *peSection) checkOffset(off int64) {
360 if off != int64(sect.pointerToRawData) {
361 Errorf("%s.PointerToRawData = %#x, want %#x", sect.name, uint64(int64(sect.pointerToRawData)), uint64(off))
362 errorexit()
363 }
364 }
365
366
367
368 func (sect *peSection) checkSegment(seg *sym.Segment) {
369 if seg.Vaddr-uint64(PEBASE) != uint64(sect.virtualAddress) {
370 Errorf("%s.VirtualAddress = %#x, want %#x", sect.name, uint64(int64(sect.virtualAddress)), uint64(int64(seg.Vaddr-uint64(PEBASE))))
371 errorexit()
372 }
373 if seg.Fileoff != uint64(sect.pointerToRawData) {
374 Errorf("%s.PointerToRawData = %#x, want %#x", sect.name, uint64(int64(sect.pointerToRawData)), uint64(int64(seg.Fileoff)))
375 errorexit()
376 }
377 }
378
379
380
381
382 func (sect *peSection) pad(out *OutBuf, n uint32) {
383 out.WriteStringN("", int(sect.sizeOfRawData-n))
384 }
385
386
387 func (sect *peSection) write(out *OutBuf, linkmode LinkMode) error {
388 h := pe.SectionHeader32{
389 VirtualSize: sect.virtualSize,
390 SizeOfRawData: sect.sizeOfRawData,
391 PointerToRawData: sect.pointerToRawData,
392 PointerToRelocations: sect.pointerToRelocations,
393 NumberOfRelocations: sect.numberOfRelocations,
394 Characteristics: sect.characteristics,
395 }
396 if linkmode != LinkExternal {
397 h.VirtualAddress = sect.virtualAddress
398 }
399 copy(h.Name[:], sect.shortName)
400 return binary.Write(out, binary.LittleEndian, h)
401 }
402
403
404
405
406
407 func (sect *peSection) emitRelocations(out *OutBuf, relocfn func() int) {
408 sect.pointerToRelocations = uint32(out.Offset())
409
410 out.Write32(0)
411 out.Write32(0)
412 out.Write16(0)
413
414 n := relocfn() + 1
415
416 cpos := out.Offset()
417 out.SeekSet(int64(sect.pointerToRelocations))
418 out.Write32(uint32(n))
419 out.SeekSet(cpos)
420 if n > 0x10000 {
421 n = 0x10000
422 sect.characteristics |= IMAGE_SCN_LNK_NRELOC_OVFL
423 } else {
424 sect.pointerToRelocations += 10
425 }
426 sect.numberOfRelocations = uint16(n - 1)
427 }
428
429
430 type peFile struct {
431 sections []*peSection
432 stringTable peStringTable
433 textSect *peSection
434 rdataSect *peSection
435 dataSect *peSection
436 bssSect *peSection
437 ctorsSect *peSection
438 pdataSect *peSection
439 xdataSect *peSection
440 nextSectOffset uint32
441 nextFileOffset uint32
442 symtabOffset int64
443 symbolCount int
444 dataDirectory [16]pe.DataDirectory
445 }
446
447
448 func (f *peFile) addSection(name string, sectsize int, filesize int) *peSection {
449 sect := &peSection{
450 name: name,
451 shortName: name,
452 index: len(f.sections) + 1,
453 virtualAddress: f.nextSectOffset,
454 pointerToRawData: f.nextFileOffset,
455 }
456 f.nextSectOffset = uint32(Rnd(int64(f.nextSectOffset)+int64(sectsize), PESECTALIGN))
457 if filesize > 0 {
458 sect.virtualSize = uint32(sectsize)
459 sect.sizeOfRawData = uint32(Rnd(int64(filesize), PEFILEALIGN))
460 f.nextFileOffset += sect.sizeOfRawData
461 } else {
462 sect.sizeOfRawData = uint32(sectsize)
463 }
464 f.sections = append(f.sections, sect)
465 return sect
466 }
467
468
469
470
471 func (f *peFile) addDWARFSection(name string, size int) *peSection {
472 if size == 0 {
473 Exitf("DWARF section %q is empty", name)
474 }
475
476
477
478
479
480
481 off := f.stringTable.add(name)
482 h := f.addSection(name, size, size)
483 h.shortName = fmt.Sprintf("/%d", off)
484 h.characteristics = IMAGE_SCN_ALIGN_1BYTES | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_DISCARDABLE | IMAGE_SCN_CNT_INITIALIZED_DATA
485 return h
486 }
487
488
489 func (f *peFile) addDWARF() {
490 if *FlagS {
491 return
492 }
493 if *FlagW {
494 return
495 }
496 for _, sect := range Segdwarf.Sections {
497 h := f.addDWARFSection(sect.Name, int(sect.Length))
498 fileoff := sect.Vaddr - Segdwarf.Vaddr + Segdwarf.Fileoff
499 if uint64(h.pointerToRawData) != fileoff {
500 Exitf("%s.PointerToRawData = %#x, want %#x", sect.Name, h.pointerToRawData, fileoff)
501 }
502 }
503 }
504
505
506 func (f *peFile) addSEH(ctxt *Link) {
507
508
509 if Segpdata.Length == 0 {
510 return
511 }
512 d := pefile.addSection(".pdata", int(Segpdata.Length), int(Segpdata.Length))
513 d.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ
514 if ctxt.LinkMode == LinkExternal {
515
516 d.characteristics |= IMAGE_SCN_ALIGN_4BYTES
517 }
518 pefile.pdataSect = d
519 d.checkSegment(&Segpdata)
520 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_EXCEPTION].VirtualAddress = d.virtualAddress
521 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_EXCEPTION].Size = d.virtualSize
522
523 if Segxdata.Length > 0 {
524 d = pefile.addSection(".xdata", int(Segxdata.Length), int(Segxdata.Length))
525 d.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ
526 if ctxt.LinkMode == LinkExternal {
527
528 d.characteristics |= IMAGE_SCN_ALIGN_4BYTES
529 }
530 pefile.xdataSect = d
531 d.checkSegment(&Segxdata)
532 }
533 }
534
535
536 func (f *peFile) addInitArray(ctxt *Link) *peSection {
537
538
539
540
541
542 var size int
543 var alignment uint32
544 if pe64 {
545 size = 8
546 alignment = IMAGE_SCN_ALIGN_8BYTES
547 } else {
548 size = 4
549 alignment = IMAGE_SCN_ALIGN_4BYTES
550 }
551 sect := f.addSection(".ctors", size, size)
552 sect.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE | alignment
553 sect.sizeOfRawData = uint32(size)
554 ctxt.Out.SeekSet(int64(sect.pointerToRawData))
555 sect.checkOffset(ctxt.Out.Offset())
556
557 init_entry := ctxt.loader.Lookup(*flagEntrySymbol, 0)
558 addr := uint64(ctxt.loader.SymValue(init_entry)) - ctxt.loader.SymSect(init_entry).Vaddr
559 if pe64 {
560 ctxt.Out.Write64(addr)
561 } else {
562 ctxt.Out.Write32(uint32(addr))
563 }
564 return sect
565 }
566
567
568 func (f *peFile) emitRelocations(ctxt *Link) {
569 for ctxt.Out.Offset()&7 != 0 {
570 ctxt.Out.Write8(0)
571 }
572
573 ldr := ctxt.loader
574
575
576
577 relocsect := func(sect *sym.Section, syms []loader.Sym, base uint64) int {
578
579 if sect.Vaddr >= sect.Seg.Vaddr+sect.Seg.Filelen {
580 return 0
581 }
582 sect.Reloff = uint64(ctxt.Out.Offset())
583 for i, s := range syms {
584 if !ldr.AttrReachable(s) {
585 continue
586 }
587 if uint64(ldr.SymValue(s)) >= sect.Vaddr {
588 syms = syms[i:]
589 break
590 }
591 }
592 eaddr := int64(sect.Vaddr + sect.Length)
593 for _, s := range syms {
594 if !ldr.AttrReachable(s) {
595 continue
596 }
597 if ldr.SymValue(s) >= eaddr {
598 break
599 }
600
601
602 relocs := ldr.Relocs(s)
603 for ri := 0; ri < relocs.Count(); ri++ {
604 r := relocs.At(ri)
605 rr, ok := extreloc(ctxt, ldr, s, r)
606 if !ok {
607 continue
608 }
609 if rr.Xsym == 0 {
610 ctxt.Errorf(s, "missing xsym in relocation")
611 continue
612 }
613 if ldr.SymDynid(rr.Xsym) < 0 {
614 ctxt.Errorf(s, "reloc %d to non-coff symbol %s (outer=%s) %d", r.Type(), ldr.SymName(r.Sym()), ldr.SymName(rr.Xsym), ldr.SymType(r.Sym()))
615 }
616 if !thearch.PEreloc1(ctxt.Arch, ctxt.Out, ldr, s, rr, int64(uint64(ldr.SymValue(s)+int64(r.Off()))-base)) {
617 ctxt.Errorf(s, "unsupported obj reloc %v/%d to %s", r.Type(), r.Siz(), ldr.SymName(r.Sym()))
618 }
619 }
620 }
621 sect.Rellen = uint64(ctxt.Out.Offset()) - sect.Reloff
622 const relocLen = 4 + 4 + 2
623 return int(sect.Rellen / relocLen)
624 }
625
626 type relsect struct {
627 peSect *peSection
628 seg *sym.Segment
629 syms []loader.Sym
630 }
631 sects := []relsect{
632 {f.textSect, &Segtext, ctxt.Textp},
633 {f.rdataSect, &Segrodata, ctxt.datap},
634 {f.dataSect, &Segdata, ctxt.datap},
635 }
636 if len(sehp.pdata) != 0 {
637 sects = append(sects, relsect{f.pdataSect, &Segpdata, sehp.pdata})
638 }
639 if len(sehp.xdata) != 0 {
640 sects = append(sects, relsect{f.xdataSect, &Segxdata, sehp.xdata})
641 }
642 for _, s := range sects {
643 s.peSect.emitRelocations(ctxt.Out, func() int {
644 var n int
645 for _, sect := range s.seg.Sections {
646 n += relocsect(sect, s.syms, s.seg.Vaddr)
647 }
648 return n
649 })
650 }
651
652 dwarfLoop:
653 for i := 0; i < len(Segdwarf.Sections); i++ {
654 sect := Segdwarf.Sections[i]
655 si := dwarfp[i]
656 if si.secSym() != loader.Sym(sect.Sym) ||
657 ldr.SymSect(si.secSym()) != sect {
658 panic("inconsistency between dwarfp and Segdwarf")
659 }
660 for _, pesect := range f.sections {
661 if sect.Name == pesect.name {
662 pesect.emitRelocations(ctxt.Out, func() int {
663 return relocsect(sect, si.syms, sect.Vaddr)
664 })
665 continue dwarfLoop
666 }
667 }
668 Errorf("emitRelocations: could not find %q section", sect.Name)
669 }
670
671 if f.ctorsSect == nil {
672 return
673 }
674
675 f.ctorsSect.emitRelocations(ctxt.Out, func() int {
676 dottext := ldr.Lookup(".text", 0)
677 ctxt.Out.Write32(0)
678 ctxt.Out.Write32(uint32(ldr.SymDynid(dottext)))
679 switch buildcfg.GOARCH {
680 default:
681 ctxt.Errorf(dottext, "unknown architecture for PE: %q\n", buildcfg.GOARCH)
682 case "386":
683 ctxt.Out.Write16(IMAGE_REL_I386_DIR32)
684 case "amd64":
685 ctxt.Out.Write16(IMAGE_REL_AMD64_ADDR64)
686 case "arm":
687 ctxt.Out.Write16(IMAGE_REL_ARM_ADDR32)
688 case "arm64":
689 ctxt.Out.Write16(IMAGE_REL_ARM64_ADDR64)
690 }
691 return 1
692 })
693 }
694
695
696
697 func (f *peFile) writeSymbol(out *OutBuf, ldr *loader.Loader, s loader.Sym, name string, value int64, sectidx int, typ uint16, class uint8) {
698 if len(name) > 8 {
699 out.Write32(0)
700 out.Write32(uint32(f.stringTable.add(name)))
701 } else {
702 out.WriteStringN(name, 8)
703 }
704 out.Write32(uint32(value))
705 out.Write16(uint16(sectidx))
706 out.Write16(typ)
707 out.Write8(class)
708 out.Write8(0)
709
710 ldr.SetSymDynid(s, int32(f.symbolCount))
711
712 f.symbolCount++
713 }
714
715
716
717 func (f *peFile) mapToPESection(ldr *loader.Loader, s loader.Sym, linkmode LinkMode) (pesectidx int, offset int64, err error) {
718 sect := ldr.SymSect(s)
719 if sect == nil {
720 return 0, 0, fmt.Errorf("could not map %s symbol with no section", ldr.SymName(s))
721 }
722 if sect.Seg == &Segtext {
723 return f.textSect.index, int64(uint64(ldr.SymValue(s)) - Segtext.Vaddr), nil
724 }
725 if sect.Seg == &Segrodata {
726 return f.rdataSect.index, int64(uint64(ldr.SymValue(s)) - Segrodata.Vaddr), nil
727 }
728 if sect.Seg != &Segdata {
729 return 0, 0, fmt.Errorf("could not map %s symbol with non .text or .rdata or .data section", ldr.SymName(s))
730 }
731 v := uint64(ldr.SymValue(s)) - Segdata.Vaddr
732 if linkmode != LinkExternal {
733 return f.dataSect.index, int64(v), nil
734 }
735 if ldr.SymType(s).IsDATA() {
736 return f.dataSect.index, int64(v), nil
737 }
738
739
740 if v < Segdata.Filelen {
741 return f.dataSect.index, int64(v), nil
742 }
743 return f.bssSect.index, int64(v - Segdata.Filelen), nil
744 }
745
746 var isLabel = make(map[loader.Sym]bool)
747
748 func AddPELabelSym(ldr *loader.Loader, s loader.Sym) {
749 isLabel[s] = true
750 }
751
752
753 func (f *peFile) writeSymbols(ctxt *Link) {
754 ldr := ctxt.loader
755 addsym := func(s loader.Sym) {
756 t := ldr.SymType(s)
757 if ldr.SymSect(s) == nil && t != sym.SDYNIMPORT && t != sym.SHOSTOBJ && t != sym.SUNDEFEXT {
758 return
759 }
760
761 name := ldr.SymName(s)
762
763
764 if ctxt.Is386() && ctxt.IsExternal() &&
765 (t == sym.SHOSTOBJ || t == sym.SUNDEFEXT || ldr.AttrCgoExport(s) ||
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785 name == "go:buildid" || name == "type:*") {
786 name = "_" + name
787 }
788
789 name = mangleABIName(ctxt, ldr, s, name)
790
791 var peSymType uint16 = IMAGE_SYM_TYPE_NULL
792 switch {
793 case t.IsText(), t == sym.SDYNIMPORT, t == sym.SHOSTOBJ, t == sym.SUNDEFEXT:
794
795
796
797
798
799 peSymType = IMAGE_SYM_DTYPE_FUNCTION<<4 + IMAGE_SYM_TYPE_NULL
800 }
801 sect, value, err := f.mapToPESection(ldr, s, ctxt.LinkMode)
802 if err != nil {
803 switch t {
804 case sym.SDYNIMPORT, sym.SHOSTOBJ, sym.SUNDEFEXT:
805 default:
806 ctxt.Errorf(s, "addpesym: %v", err)
807 }
808 }
809 class := IMAGE_SYM_CLASS_EXTERNAL
810 if ldr.IsFileLocal(s) || ldr.AttrVisibilityHidden(s) || ldr.AttrLocal(s) {
811 class = IMAGE_SYM_CLASS_STATIC
812 }
813 f.writeSymbol(ctxt.Out, ldr, s, name, value, sect, peSymType, uint8(class))
814 }
815
816 if ctxt.LinkMode == LinkExternal {
817
818
819 for _, pesect := range f.sections {
820 s := ldr.LookupOrCreateSym(pesect.name, 0)
821 f.writeSymbol(ctxt.Out, ldr, s, pesect.name, 0, pesect.index, IMAGE_SYM_TYPE_NULL, IMAGE_SYM_CLASS_STATIC)
822 }
823 }
824
825
826 s := ldr.Lookup("runtime.text", 0)
827 if ldr.SymType(s).IsText() {
828 addsym(s)
829 }
830 s = ldr.Lookup("runtime.etext", 0)
831 if ldr.SymType(s).IsText() {
832 addsym(s)
833 }
834
835
836 for _, s := range ctxt.Textp {
837 addsym(s)
838 }
839
840 shouldBeInSymbolTable := func(s loader.Sym) bool {
841 if ldr.AttrNotInSymbolTable(s) {
842 return false
843 }
844 name := ldr.SymName(s)
845 if name == "" || name[0] == '.' {
846 return false
847 }
848 return true
849 }
850
851
852 for s := loader.Sym(1); s < loader.Sym(ldr.NSym()); s++ {
853 if !ldr.AttrReachable(s) {
854 continue
855 }
856 t := ldr.SymType(s)
857 if t >= sym.SELFRXSECT && t < sym.SXREF {
858 if t == sym.STLSBSS {
859 continue
860 }
861 if !shouldBeInSymbolTable(s) {
862 continue
863 }
864 addsym(s)
865 }
866
867 switch t {
868 case sym.SDYNIMPORT, sym.SHOSTOBJ, sym.SUNDEFEXT:
869 addsym(s)
870 default:
871 if len(isLabel) > 0 && isLabel[s] {
872 addsym(s)
873 }
874 }
875 }
876 }
877
878
879 func (f *peFile) writeSymbolTableAndStringTable(ctxt *Link) {
880 f.symtabOffset = ctxt.Out.Offset()
881
882
883 if !*FlagS || ctxt.LinkMode == LinkExternal {
884 f.writeSymbols(ctxt)
885 }
886
887
888 size := f.stringTable.size() + 18*f.symbolCount
889 var h *peSection
890 if ctxt.LinkMode != LinkExternal {
891
892
893 h = f.addSection(".symtab", size, size)
894 h.characteristics = IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_DISCARDABLE
895 h.checkOffset(f.symtabOffset)
896 }
897
898
899 f.stringTable.write(ctxt.Out)
900 if ctxt.LinkMode != LinkExternal {
901 h.pad(ctxt.Out, uint32(size))
902 }
903 }
904
905
906 func (f *peFile) writeFileHeader(ctxt *Link) {
907 var fh pe.FileHeader
908
909 switch ctxt.Arch.Family {
910 default:
911 Exitf("unknown PE architecture: %v", ctxt.Arch.Family)
912 case sys.AMD64:
913 fh.Machine = pe.IMAGE_FILE_MACHINE_AMD64
914 case sys.I386:
915 fh.Machine = pe.IMAGE_FILE_MACHINE_I386
916 case sys.ARM:
917 fh.Machine = pe.IMAGE_FILE_MACHINE_ARMNT
918 case sys.ARM64:
919 fh.Machine = pe.IMAGE_FILE_MACHINE_ARM64
920 }
921
922 fh.NumberOfSections = uint16(len(f.sections))
923
924
925
926 fh.TimeDateStamp = 0
927
928 if ctxt.LinkMode != LinkExternal {
929 fh.Characteristics = pe.IMAGE_FILE_EXECUTABLE_IMAGE
930 switch ctxt.Arch.Family {
931 case sys.AMD64, sys.I386:
932 if ctxt.BuildMode != BuildModePIE {
933 fh.Characteristics |= pe.IMAGE_FILE_RELOCS_STRIPPED
934 }
935 }
936 }
937 if pe64 {
938 var oh64 pe.OptionalHeader64
939 fh.SizeOfOptionalHeader = uint16(binary.Size(&oh64))
940 fh.Characteristics |= pe.IMAGE_FILE_LARGE_ADDRESS_AWARE
941 } else {
942 var oh pe.OptionalHeader32
943 fh.SizeOfOptionalHeader = uint16(binary.Size(&oh))
944 fh.Characteristics |= pe.IMAGE_FILE_32BIT_MACHINE
945 }
946
947 fh.PointerToSymbolTable = uint32(f.symtabOffset)
948 fh.NumberOfSymbols = uint32(f.symbolCount)
949
950 binary.Write(ctxt.Out, binary.LittleEndian, &fh)
951 }
952
953
954 func (f *peFile) writeOptionalHeader(ctxt *Link) {
955 var oh pe.OptionalHeader32
956 var oh64 pe.OptionalHeader64
957
958 if pe64 {
959 oh64.Magic = 0x20b
960 } else {
961 oh.Magic = 0x10b
962 oh.BaseOfData = f.dataSect.virtualAddress
963 }
964
965
966 oh64.MajorLinkerVersion = 3
967 oh.MajorLinkerVersion = 3
968 oh64.MinorLinkerVersion = 0
969 oh.MinorLinkerVersion = 0
970 oh64.SizeOfCode = f.textSect.sizeOfRawData
971 oh.SizeOfCode = f.textSect.sizeOfRawData
972 oh64.SizeOfInitializedData = f.dataSect.sizeOfRawData
973 oh.SizeOfInitializedData = f.dataSect.sizeOfRawData
974 oh64.SizeOfUninitializedData = 0
975 oh.SizeOfUninitializedData = 0
976 if ctxt.LinkMode != LinkExternal {
977 oh64.AddressOfEntryPoint = uint32(Entryvalue(ctxt) - PEBASE)
978 oh.AddressOfEntryPoint = uint32(Entryvalue(ctxt) - PEBASE)
979 }
980 oh64.BaseOfCode = f.textSect.virtualAddress
981 oh.BaseOfCode = f.textSect.virtualAddress
982 oh64.ImageBase = uint64(PEBASE)
983 oh.ImageBase = uint32(PEBASE)
984 oh64.SectionAlignment = uint32(PESECTALIGN)
985 oh.SectionAlignment = uint32(PESECTALIGN)
986 oh64.FileAlignment = uint32(PEFILEALIGN)
987 oh.FileAlignment = uint32(PEFILEALIGN)
988 oh64.MajorOperatingSystemVersion = PeMinimumTargetMajorVersion
989 oh.MajorOperatingSystemVersion = PeMinimumTargetMajorVersion
990 oh64.MinorOperatingSystemVersion = PeMinimumTargetMinorVersion
991 oh.MinorOperatingSystemVersion = PeMinimumTargetMinorVersion
992 oh64.MajorImageVersion = 1
993 oh.MajorImageVersion = 1
994 oh64.MinorImageVersion = 0
995 oh.MinorImageVersion = 0
996 oh64.MajorSubsystemVersion = PeMinimumTargetMajorVersion
997 oh.MajorSubsystemVersion = PeMinimumTargetMajorVersion
998 oh64.MinorSubsystemVersion = PeMinimumTargetMinorVersion
999 oh.MinorSubsystemVersion = PeMinimumTargetMinorVersion
1000 oh64.SizeOfImage = f.nextSectOffset
1001 oh.SizeOfImage = f.nextSectOffset
1002 oh64.SizeOfHeaders = uint32(PEFILEHEADR)
1003 oh.SizeOfHeaders = uint32(PEFILEHEADR)
1004 if windowsgui {
1005 oh64.Subsystem = pe.IMAGE_SUBSYSTEM_WINDOWS_GUI
1006 oh.Subsystem = pe.IMAGE_SUBSYSTEM_WINDOWS_GUI
1007 } else {
1008 oh64.Subsystem = pe.IMAGE_SUBSYSTEM_WINDOWS_CUI
1009 oh.Subsystem = pe.IMAGE_SUBSYSTEM_WINDOWS_CUI
1010 }
1011
1012
1013 oh64.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_TERMINAL_SERVER_AWARE
1014 oh.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_TERMINAL_SERVER_AWARE
1015
1016
1017 oh64.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_NX_COMPAT
1018 oh.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_NX_COMPAT
1019
1020
1021 if needPEBaseReloc(ctxt) {
1022 oh64.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_DYNAMIC_BASE
1023 oh.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_DYNAMIC_BASE
1024 }
1025
1026
1027 if ctxt.BuildMode == BuildModePIE {
1028 oh64.DllCharacteristics |= pe.IMAGE_DLLCHARACTERISTICS_HIGH_ENTROPY_VA
1029 }
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055 oh64.SizeOfStackReserve = 0x00200000
1056 if !iscgo {
1057 oh64.SizeOfStackCommit = 0x00001000
1058 } else {
1059
1060
1061
1062 oh64.SizeOfStackCommit = 0x00200000 - 0x2000
1063 }
1064
1065 oh.SizeOfStackReserve = 0x00100000
1066 if !iscgo {
1067 oh.SizeOfStackCommit = 0x00001000
1068 } else {
1069 oh.SizeOfStackCommit = 0x00100000 - 0x2000
1070 }
1071
1072 oh64.SizeOfHeapReserve = 0x00100000
1073 oh.SizeOfHeapReserve = 0x00100000
1074 oh64.SizeOfHeapCommit = 0x00001000
1075 oh.SizeOfHeapCommit = 0x00001000
1076 oh64.NumberOfRvaAndSizes = 16
1077 oh.NumberOfRvaAndSizes = 16
1078
1079 if pe64 {
1080 oh64.DataDirectory = f.dataDirectory
1081 } else {
1082 oh.DataDirectory = f.dataDirectory
1083 }
1084
1085 if pe64 {
1086 binary.Write(ctxt.Out, binary.LittleEndian, &oh64)
1087 } else {
1088 binary.Write(ctxt.Out, binary.LittleEndian, &oh)
1089 }
1090 }
1091
1092 var pefile peFile
1093
1094 func Peinit(ctxt *Link) {
1095 var l int
1096
1097 if ctxt.Arch.PtrSize == 8 {
1098
1099 pe64 = true
1100 PEBASE = 1 << 32
1101 if ctxt.Arch.Family == sys.AMD64 {
1102
1103
1104
1105 PEBASE = 1 << 22
1106 }
1107 var oh64 pe.OptionalHeader64
1108 l = binary.Size(&oh64)
1109 } else {
1110
1111 PEBASE = 1 << 22
1112 var oh pe.OptionalHeader32
1113 l = binary.Size(&oh)
1114 }
1115
1116 if ctxt.LinkMode == LinkExternal {
1117
1118
1119
1120
1121 PESECTALIGN = 32
1122 PEFILEALIGN = 0
1123
1124 PEBASE = 0
1125 }
1126
1127 var sh [16]pe.SectionHeader32
1128 var fh pe.FileHeader
1129 PEFILEHEADR = int32(Rnd(int64(len(dosstub)+binary.Size(&fh)+l+binary.Size(&sh)), PEFILEALIGN))
1130 if ctxt.LinkMode != LinkExternal {
1131 PESECTHEADR = int32(Rnd(int64(PEFILEHEADR), PESECTALIGN))
1132 } else {
1133 PESECTHEADR = 0
1134 }
1135 pefile.nextSectOffset = uint32(PESECTHEADR)
1136 pefile.nextFileOffset = uint32(PEFILEHEADR)
1137
1138 if ctxt.LinkMode == LinkInternal {
1139
1140 for _, name := range [2]string{"__image_base__", "_image_base__"} {
1141 sb := ctxt.loader.CreateSymForUpdate(name, 0)
1142 sb.SetType(sym.SDATA)
1143 sb.SetValue(PEBASE)
1144 ctxt.loader.SetAttrSpecial(sb.Sym(), true)
1145 ctxt.loader.SetAttrLocal(sb.Sym(), true)
1146 }
1147 }
1148
1149 HEADR = PEFILEHEADR
1150 if *FlagRound == -1 {
1151 *FlagRound = PESECTALIGN
1152 }
1153 if *FlagTextAddr == -1 {
1154 *FlagTextAddr = Rnd(PEBASE, *FlagRound) + int64(PESECTHEADR)
1155 }
1156 }
1157
1158 func pewrite(ctxt *Link) {
1159 ctxt.Out.SeekSet(0)
1160 if ctxt.LinkMode != LinkExternal {
1161 ctxt.Out.Write(dosstub)
1162 ctxt.Out.WriteStringN("PE", 4)
1163 }
1164
1165 pefile.writeFileHeader(ctxt)
1166
1167 pefile.writeOptionalHeader(ctxt)
1168
1169 for _, sect := range pefile.sections {
1170 sect.write(ctxt.Out, ctxt.LinkMode)
1171 }
1172 }
1173
1174 func strput(out *OutBuf, s string) {
1175 out.WriteString(s)
1176 out.Write8(0)
1177
1178 if (len(s)+1)%2 != 0 {
1179 out.Write8(0)
1180 }
1181 }
1182
1183 func initdynimport(ctxt *Link) *Dll {
1184 ldr := ctxt.loader
1185 var d *Dll
1186
1187 dr = nil
1188 var m *Imp
1189 for s := loader.Sym(1); s < loader.Sym(ldr.NSym()); s++ {
1190 if !ldr.AttrReachable(s) || ldr.SymType(s) != sym.SDYNIMPORT {
1191 continue
1192 }
1193 dynlib := ldr.SymDynimplib(s)
1194 for d = dr; d != nil; d = d.next {
1195 if d.name == dynlib {
1196 m = new(Imp)
1197 break
1198 }
1199 }
1200
1201 if d == nil {
1202 d = new(Dll)
1203 d.name = dynlib
1204 d.next = dr
1205 dr = d
1206 m = new(Imp)
1207 }
1208
1209
1210
1211
1212
1213 m.argsize = -1
1214 extName := ldr.SymExtname(s)
1215 if i := strings.IndexByte(extName, '%'); i >= 0 {
1216 var err error
1217 m.argsize, err = strconv.Atoi(extName[i+1:])
1218 if err != nil {
1219 ctxt.Errorf(s, "failed to parse stdcall decoration: %v", err)
1220 }
1221 m.argsize *= ctxt.Arch.PtrSize
1222 ldr.SetSymExtname(s, extName[:i])
1223 }
1224
1225 m.s = s
1226 m.next = d.ms
1227 d.ms = m
1228 }
1229
1230 if ctxt.IsExternal() {
1231
1232 for d := dr; d != nil; d = d.next {
1233 for m = d.ms; m != nil; m = m.next {
1234 sb := ldr.MakeSymbolUpdater(m.s)
1235 sb.SetType(sym.SDATA)
1236 sb.Grow(int64(ctxt.Arch.PtrSize))
1237 dynName := sb.Extname()
1238
1239 if ctxt.Is386() && m.argsize >= 0 {
1240 dynName += fmt.Sprintf("@%d", m.argsize)
1241 }
1242 dynSym := ldr.CreateSymForUpdate(dynName, 0)
1243 dynSym.SetType(sym.SHOSTOBJ)
1244 r, _ := sb.AddRel(objabi.R_ADDR)
1245 r.SetSym(dynSym.Sym())
1246 r.SetSiz(uint8(ctxt.Arch.PtrSize))
1247 }
1248 }
1249 } else {
1250 dynamic := ldr.CreateSymForUpdate(".windynamic", 0)
1251 dynamic.SetType(sym.SWINDOWS)
1252 for d := dr; d != nil; d = d.next {
1253 for m = d.ms; m != nil; m = m.next {
1254 sb := ldr.MakeSymbolUpdater(m.s)
1255 sb.SetType(sym.SWINDOWS)
1256 sb.SetValue(dynamic.Size())
1257 dynamic.SetSize(dynamic.Size() + int64(ctxt.Arch.PtrSize))
1258 dynamic.AddInteriorSym(m.s)
1259 }
1260
1261 dynamic.SetSize(dynamic.Size() + int64(ctxt.Arch.PtrSize))
1262 }
1263 }
1264
1265 return dr
1266 }
1267
1268
1269
1270 func peimporteddlls() []string {
1271 var dlls []string
1272
1273 for d := dr; d != nil; d = d.next {
1274 dlls = append(dlls, "-l"+strings.TrimSuffix(d.name, ".dll"))
1275 }
1276
1277 return dlls
1278 }
1279
1280 func addimports(ctxt *Link, datsect *peSection) {
1281 ldr := ctxt.loader
1282 startoff := ctxt.Out.Offset()
1283 dynamic := ldr.LookupOrCreateSym(".windynamic", 0)
1284
1285
1286 n := uint64(0)
1287
1288 for d := dr; d != nil; d = d.next {
1289 n++
1290 }
1291 ctxt.Out.SeekSet(startoff + int64(binary.Size(&IMAGE_IMPORT_DESCRIPTOR{}))*int64(n+1))
1292
1293
1294 for d := dr; d != nil; d = d.next {
1295 d.nameoff = uint64(ctxt.Out.Offset()) - uint64(startoff)
1296 strput(ctxt.Out, d.name)
1297 }
1298
1299
1300 for d := dr; d != nil; d = d.next {
1301 for m := d.ms; m != nil; m = m.next {
1302 m.off = uint64(pefile.nextSectOffset) + uint64(ctxt.Out.Offset()) - uint64(startoff)
1303 ctxt.Out.Write16(0)
1304 strput(ctxt.Out, ldr.SymExtname(m.s))
1305 }
1306 }
1307
1308
1309 oftbase := uint64(ctxt.Out.Offset()) - uint64(startoff)
1310
1311 n = uint64(ctxt.Out.Offset())
1312 for d := dr; d != nil; d = d.next {
1313 d.thunkoff = uint64(ctxt.Out.Offset()) - n
1314 for m := d.ms; m != nil; m = m.next {
1315 if pe64 {
1316 ctxt.Out.Write64(m.off)
1317 } else {
1318 ctxt.Out.Write32(uint32(m.off))
1319 }
1320 }
1321
1322 if pe64 {
1323 ctxt.Out.Write64(0)
1324 } else {
1325 ctxt.Out.Write32(0)
1326 }
1327 }
1328
1329
1330 n = uint64(ctxt.Out.Offset()) - uint64(startoff)
1331
1332 isect := pefile.addSection(".idata", int(n), int(n))
1333 isect.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE
1334 isect.checkOffset(startoff)
1335 isect.pad(ctxt.Out, uint32(n))
1336 endoff := ctxt.Out.Offset()
1337
1338
1339 ftbase := uint64(ldr.SymValue(dynamic)) - uint64(datsect.virtualAddress) - uint64(PEBASE)
1340
1341 ctxt.Out.SeekSet(int64(uint64(datsect.pointerToRawData) + ftbase))
1342 for d := dr; d != nil; d = d.next {
1343 for m := d.ms; m != nil; m = m.next {
1344 if pe64 {
1345 ctxt.Out.Write64(m.off)
1346 } else {
1347 ctxt.Out.Write32(uint32(m.off))
1348 }
1349 }
1350
1351 if pe64 {
1352 ctxt.Out.Write64(0)
1353 } else {
1354 ctxt.Out.Write32(0)
1355 }
1356 }
1357
1358
1359 out := ctxt.Out
1360 out.SeekSet(startoff)
1361
1362 for d := dr; d != nil; d = d.next {
1363 out.Write32(uint32(uint64(isect.virtualAddress) + oftbase + d.thunkoff))
1364 out.Write32(0)
1365 out.Write32(0)
1366 out.Write32(uint32(uint64(isect.virtualAddress) + d.nameoff))
1367 out.Write32(uint32(uint64(datsect.virtualAddress) + ftbase + d.thunkoff))
1368 }
1369
1370 out.Write32(0)
1371 out.Write32(0)
1372 out.Write32(0)
1373 out.Write32(0)
1374 out.Write32(0)
1375
1376
1377 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress = isect.virtualAddress
1378 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_IMPORT].Size = isect.virtualSize
1379 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_IAT].VirtualAddress = uint32(ldr.SymValue(dynamic) - PEBASE)
1380 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_IAT].Size = uint32(ldr.SymSize(dynamic))
1381
1382 out.SeekSet(endoff)
1383 }
1384
1385 func initdynexport(ctxt *Link) {
1386 ldr := ctxt.loader
1387 for s := loader.Sym(1); s < loader.Sym(ldr.NSym()); s++ {
1388 if !ldr.AttrReachable(s) || !ldr.AttrCgoExportDynamic(s) {
1389 continue
1390 }
1391 if len(dexport) >= math.MaxUint16 {
1392 ctxt.Errorf(s, "pe dynexport table is full")
1393 errorexit()
1394 }
1395
1396 dexport = append(dexport, s)
1397 }
1398
1399 sort.Slice(dexport, func(i, j int) bool { return ldr.SymExtname(dexport[i]) < ldr.SymExtname(dexport[j]) })
1400 }
1401
1402 func addexports(ctxt *Link) {
1403 ldr := ctxt.loader
1404 var e IMAGE_EXPORT_DIRECTORY
1405
1406 nexport := len(dexport)
1407 size := binary.Size(&e) + 10*nexport + len(*flagOutfile) + 1
1408 for _, s := range dexport {
1409 size += len(ldr.SymExtname(s)) + 1
1410 }
1411
1412 if nexport == 0 {
1413 return
1414 }
1415
1416 sect := pefile.addSection(".edata", size, size)
1417 sect.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ
1418 sect.checkOffset(ctxt.Out.Offset())
1419 va := int(sect.virtualAddress)
1420 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_EXPORT].VirtualAddress = uint32(va)
1421 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_EXPORT].Size = sect.virtualSize
1422
1423 vaName := va + binary.Size(&e) + nexport*4
1424 vaAddr := va + binary.Size(&e)
1425 vaNa := va + binary.Size(&e) + nexport*8
1426
1427 e.Characteristics = 0
1428 e.MajorVersion = 0
1429 e.MinorVersion = 0
1430 e.NumberOfFunctions = uint32(nexport)
1431 e.NumberOfNames = uint32(nexport)
1432 e.Name = uint32(va+binary.Size(&e)) + uint32(nexport)*10
1433 e.Base = 1
1434 e.AddressOfFunctions = uint32(vaAddr)
1435 e.AddressOfNames = uint32(vaName)
1436 e.AddressOfNameOrdinals = uint32(vaNa)
1437
1438 out := ctxt.Out
1439
1440
1441 binary.Write(out, binary.LittleEndian, &e)
1442
1443
1444 for _, s := range dexport {
1445 out.Write32(uint32(ldr.SymValue(s) - PEBASE))
1446 }
1447
1448
1449 v := int(e.Name + uint32(len(*flagOutfile)) + 1)
1450
1451 for _, s := range dexport {
1452 out.Write32(uint32(v))
1453 v += len(ldr.SymExtname(s)) + 1
1454 }
1455
1456
1457 for i := 0; i < nexport; i++ {
1458 out.Write16(uint16(i))
1459 }
1460
1461
1462 out.WriteStringN(*flagOutfile, len(*flagOutfile)+1)
1463
1464 for _, s := range dexport {
1465 name := ldr.SymExtname(s)
1466 out.WriteStringN(name, len(name)+1)
1467 }
1468 sect.pad(out, uint32(size))
1469 }
1470
1471
1472 type peBaseRelocEntry struct {
1473 typeOff uint16
1474 }
1475
1476
1477
1478
1479
1480
1481 type peBaseRelocBlock struct {
1482 entries []peBaseRelocEntry
1483 }
1484
1485
1486
1487
1488 type pePages []uint32
1489
1490
1491
1492
1493
1494 type peBaseRelocTable struct {
1495 blocks map[uint32]peBaseRelocBlock
1496
1497
1498
1499 pages pePages
1500 }
1501
1502 func (rt *peBaseRelocTable) init(ctxt *Link) {
1503 rt.blocks = make(map[uint32]peBaseRelocBlock)
1504 }
1505
1506 func (rt *peBaseRelocTable) addentry(ldr *loader.Loader, s loader.Sym, r *loader.Reloc) {
1507
1508
1509 const pageSize = 0x1000
1510 const pageMask = pageSize - 1
1511
1512 addr := ldr.SymValue(s) + int64(r.Off()) - int64(PEBASE)
1513 page := uint32(addr &^ pageMask)
1514 off := uint32(addr & pageMask)
1515
1516 b, ok := rt.blocks[page]
1517 if !ok {
1518 rt.pages = append(rt.pages, page)
1519 }
1520
1521 e := peBaseRelocEntry{
1522 typeOff: uint16(off & 0xFFF),
1523 }
1524
1525
1526 switch r.Siz() {
1527 default:
1528 Exitf("unsupported relocation size %d\n", r.Siz)
1529 case 4:
1530 e.typeOff |= uint16(IMAGE_REL_BASED_HIGHLOW << 12)
1531 case 8:
1532 e.typeOff |= uint16(IMAGE_REL_BASED_DIR64 << 12)
1533 }
1534
1535 b.entries = append(b.entries, e)
1536 rt.blocks[page] = b
1537 }
1538
1539 func (rt *peBaseRelocTable) write(ctxt *Link) {
1540 out := ctxt.Out
1541
1542
1543 slices.Sort(rt.pages)
1544
1545
1546 if out.Offset()&3 != 0 {
1547 Errorf("internal error, start of .reloc not 32-bit aligned")
1548 }
1549
1550 for _, p := range rt.pages {
1551 b := rt.blocks[p]
1552
1553
1554
1555
1556 if len(b.entries)&1 != 0 {
1557 b.entries = append(b.entries, peBaseRelocEntry{})
1558 }
1559
1560 const sizeOfPEbaseRelocBlock = 8
1561 blockSize := uint32(sizeOfPEbaseRelocBlock + len(b.entries)*2)
1562 out.Write32(p)
1563 out.Write32(blockSize)
1564
1565 for _, e := range b.entries {
1566 out.Write16(e.typeOff)
1567 }
1568 }
1569 }
1570
1571 func addPEBaseRelocSym(ldr *loader.Loader, s loader.Sym, rt *peBaseRelocTable) {
1572 relocs := ldr.Relocs(s)
1573 for ri := 0; ri < relocs.Count(); ri++ {
1574 r := relocs.At(ri)
1575 if r.Type() >= objabi.ElfRelocOffset {
1576 continue
1577 }
1578 if r.Siz() == 0 {
1579 continue
1580 }
1581 rs := r.Sym()
1582 if rs == 0 {
1583 continue
1584 }
1585 if !ldr.AttrReachable(s) {
1586 continue
1587 }
1588
1589 switch r.Type() {
1590 default:
1591 case objabi.R_ADDR:
1592 rt.addentry(ldr, s, &r)
1593 }
1594 }
1595 }
1596
1597 func needPEBaseReloc(ctxt *Link) bool {
1598
1599
1600 if (ctxt.Arch.Family == sys.I386 || ctxt.Arch.Family == sys.AMD64) && ctxt.BuildMode != BuildModePIE {
1601 return false
1602 }
1603 return true
1604 }
1605
1606 func addPEBaseReloc(ctxt *Link) {
1607 if !needPEBaseReloc(ctxt) {
1608 return
1609 }
1610
1611 var rt peBaseRelocTable
1612 rt.init(ctxt)
1613
1614
1615 ldr := ctxt.loader
1616 for _, s := range ctxt.Textp {
1617 addPEBaseRelocSym(ldr, s, &rt)
1618 }
1619 for _, s := range ctxt.datap {
1620 addPEBaseRelocSym(ldr, s, &rt)
1621 }
1622
1623
1624 startoff := ctxt.Out.Offset()
1625 rt.write(ctxt)
1626 size := ctxt.Out.Offset() - startoff
1627
1628
1629 rsect := pefile.addSection(".reloc", int(size), int(size))
1630 rsect.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_DISCARDABLE
1631 rsect.checkOffset(startoff)
1632 rsect.pad(ctxt.Out, uint32(size))
1633
1634 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_BASERELOC].VirtualAddress = rsect.virtualAddress
1635 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_BASERELOC].Size = rsect.virtualSize
1636 }
1637
1638 func (ctxt *Link) dope() {
1639 initdynimport(ctxt)
1640 initdynexport(ctxt)
1641 writeSEH(ctxt)
1642 }
1643
1644 func setpersrc(ctxt *Link, syms []loader.Sym) {
1645 if len(rsrcsyms) != 0 {
1646 Errorf("too many .rsrc sections")
1647 }
1648 rsrcsyms = syms
1649 }
1650
1651 func addpersrc(ctxt *Link) {
1652 if len(rsrcsyms) == 0 {
1653 return
1654 }
1655
1656 var size int64
1657 for _, rsrcsym := range rsrcsyms {
1658 size += ctxt.loader.SymSize(rsrcsym)
1659 }
1660 h := pefile.addSection(".rsrc", int(size), int(size))
1661 h.characteristics = IMAGE_SCN_MEM_READ | IMAGE_SCN_CNT_INITIALIZED_DATA
1662 h.checkOffset(ctxt.Out.Offset())
1663
1664 for _, rsrcsym := range rsrcsyms {
1665
1666
1667
1668 splitResources := strings.Contains(ctxt.loader.SymName(rsrcsym), ".rsrc$")
1669 relocs := ctxt.loader.Relocs(rsrcsym)
1670 data := ctxt.loader.Data(rsrcsym)
1671 for ri := 0; ri < relocs.Count(); ri++ {
1672 r := relocs.At(ri)
1673 p := data[r.Off():]
1674 val := uint32(int64(h.virtualAddress) + r.Add())
1675 if splitResources {
1676
1677
1678
1679
1680
1681
1682
1683
1684 val += uint32(len(data))
1685 }
1686 binary.LittleEndian.PutUint32(p, val)
1687 }
1688 ctxt.Out.Write(data)
1689 }
1690 h.pad(ctxt.Out, uint32(size))
1691
1692
1693 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_RESOURCE].VirtualAddress = h.virtualAddress
1694 pefile.dataDirectory[pe.IMAGE_DIRECTORY_ENTRY_RESOURCE].Size = h.virtualSize
1695 }
1696
1697 func asmbPe(ctxt *Link) {
1698 t := pefile.addSection(".text", int(Segtext.Length), int(Segtext.Length))
1699 t.characteristics = IMAGE_SCN_CNT_CODE | IMAGE_SCN_MEM_EXECUTE | IMAGE_SCN_MEM_READ
1700 if ctxt.LinkMode == LinkExternal {
1701
1702
1703 t.characteristics |= IMAGE_SCN_ALIGN_32BYTES
1704 }
1705 t.checkSegment(&Segtext)
1706 pefile.textSect = t
1707
1708 ro := pefile.addSection(".rdata", int(Segrodata.Length), int(Segrodata.Length))
1709 ro.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ
1710 if ctxt.LinkMode == LinkExternal {
1711
1712
1713 ro.characteristics |= IMAGE_SCN_ALIGN_32BYTES
1714 }
1715 ro.checkSegment(&Segrodata)
1716 pefile.rdataSect = ro
1717
1718 var d *peSection
1719 if ctxt.LinkMode != LinkExternal {
1720 d = pefile.addSection(".data", int(Segdata.Length), int(Segdata.Filelen))
1721 d.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE
1722 d.checkSegment(&Segdata)
1723 pefile.dataSect = d
1724 } else {
1725 d = pefile.addSection(".data", int(Segdata.Filelen), int(Segdata.Filelen))
1726 d.characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE | IMAGE_SCN_ALIGN_32BYTES
1727 d.checkSegment(&Segdata)
1728 pefile.dataSect = d
1729
1730 b := pefile.addSection(".bss", int(Segdata.Length-Segdata.Filelen), 0)
1731 b.characteristics = IMAGE_SCN_CNT_UNINITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE | IMAGE_SCN_ALIGN_32BYTES
1732 b.pointerToRawData = 0
1733 pefile.bssSect = b
1734 }
1735
1736 pefile.addSEH(ctxt)
1737 pefile.addDWARF()
1738
1739 if ctxt.LinkMode == LinkExternal {
1740 pefile.ctorsSect = pefile.addInitArray(ctxt)
1741 }
1742
1743 ctxt.Out.SeekSet(int64(pefile.nextFileOffset))
1744 if ctxt.LinkMode != LinkExternal {
1745 addimports(ctxt, d)
1746 addexports(ctxt)
1747 addPEBaseReloc(ctxt)
1748 }
1749 pefile.writeSymbolTableAndStringTable(ctxt)
1750 addpersrc(ctxt)
1751 if ctxt.LinkMode == LinkExternal {
1752 pefile.emitRelocations(ctxt)
1753 }
1754
1755 pewrite(ctxt)
1756 }
1757
View as plain text