github.com/mattn/go@v0.0.0-20171011075504-07f7db3ea99f/src/cmd/link/internal/ld/ldpe.go (about)

     1  // Copyright 2010 The Go Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package ld
     6  
     7  import (
     8  	"cmd/internal/bio"
     9  	"cmd/internal/objabi"
    10  	"cmd/internal/sys"
    11  	"cmd/link/internal/sym"
    12  	"debug/pe"
    13  	"errors"
    14  	"fmt"
    15  	"io"
    16  	"log"
    17  	"sort"
    18  	"strings"
    19  )
    20  
    21  const (
    22  	// TODO: the Microsoft doco says IMAGE_SYM_DTYPE_ARRAY is 3 (same with IMAGE_SYM_DTYPE_POINTER and IMAGE_SYM_DTYPE_FUNCTION)
    23  	IMAGE_SYM_UNDEFINED              = 0
    24  	IMAGE_SYM_ABSOLUTE               = -1
    25  	IMAGE_SYM_DEBUG                  = -2
    26  	IMAGE_SYM_TYPE_NULL              = 0
    27  	IMAGE_SYM_TYPE_VOID              = 1
    28  	IMAGE_SYM_TYPE_CHAR              = 2
    29  	IMAGE_SYM_TYPE_SHORT             = 3
    30  	IMAGE_SYM_TYPE_INT               = 4
    31  	IMAGE_SYM_TYPE_LONG              = 5
    32  	IMAGE_SYM_TYPE_FLOAT             = 6
    33  	IMAGE_SYM_TYPE_DOUBLE            = 7
    34  	IMAGE_SYM_TYPE_STRUCT            = 8
    35  	IMAGE_SYM_TYPE_UNION             = 9
    36  	IMAGE_SYM_TYPE_ENUM              = 10
    37  	IMAGE_SYM_TYPE_MOE               = 11
    38  	IMAGE_SYM_TYPE_BYTE              = 12
    39  	IMAGE_SYM_TYPE_WORD              = 13
    40  	IMAGE_SYM_TYPE_UINT              = 14
    41  	IMAGE_SYM_TYPE_DWORD             = 15
    42  	IMAGE_SYM_TYPE_PCODE             = 32768
    43  	IMAGE_SYM_DTYPE_NULL             = 0
    44  	IMAGE_SYM_DTYPE_POINTER          = 0x10
    45  	IMAGE_SYM_DTYPE_FUNCTION         = 0x20
    46  	IMAGE_SYM_DTYPE_ARRAY            = 0x30
    47  	IMAGE_SYM_CLASS_END_OF_FUNCTION  = -1
    48  	IMAGE_SYM_CLASS_NULL             = 0
    49  	IMAGE_SYM_CLASS_AUTOMATIC        = 1
    50  	IMAGE_SYM_CLASS_EXTERNAL         = 2
    51  	IMAGE_SYM_CLASS_STATIC           = 3
    52  	IMAGE_SYM_CLASS_REGISTER         = 4
    53  	IMAGE_SYM_CLASS_EXTERNAL_DEF     = 5
    54  	IMAGE_SYM_CLASS_LABEL            = 6
    55  	IMAGE_SYM_CLASS_UNDEFINED_LABEL  = 7
    56  	IMAGE_SYM_CLASS_MEMBER_OF_STRUCT = 8
    57  	IMAGE_SYM_CLASS_ARGUMENT         = 9
    58  	IMAGE_SYM_CLASS_STRUCT_TAG       = 10
    59  	IMAGE_SYM_CLASS_MEMBER_OF_UNION  = 11
    60  	IMAGE_SYM_CLASS_UNION_TAG        = 12
    61  	IMAGE_SYM_CLASS_TYPE_DEFINITION  = 13
    62  	IMAGE_SYM_CLASS_UNDEFINED_STATIC = 14
    63  	IMAGE_SYM_CLASS_ENUM_TAG         = 15
    64  	IMAGE_SYM_CLASS_MEMBER_OF_ENUM   = 16
    65  	IMAGE_SYM_CLASS_REGISTER_PARAM   = 17
    66  	IMAGE_SYM_CLASS_BIT_FIELD        = 18
    67  	IMAGE_SYM_CLASS_FAR_EXTERNAL     = 68 /* Not in PECOFF v8 spec */
    68  	IMAGE_SYM_CLASS_BLOCK            = 100
    69  	IMAGE_SYM_CLASS_FUNCTION         = 101
    70  	IMAGE_SYM_CLASS_END_OF_STRUCT    = 102
    71  	IMAGE_SYM_CLASS_FILE             = 103
    72  	IMAGE_SYM_CLASS_SECTION          = 104
    73  	IMAGE_SYM_CLASS_WEAK_EXTERNAL    = 105
    74  	IMAGE_SYM_CLASS_CLR_TOKEN        = 107
    75  	IMAGE_REL_I386_ABSOLUTE          = 0x0000
    76  	IMAGE_REL_I386_DIR16             = 0x0001
    77  	IMAGE_REL_I386_REL16             = 0x0002
    78  	IMAGE_REL_I386_DIR32             = 0x0006
    79  	IMAGE_REL_I386_DIR32NB           = 0x0007
    80  	IMAGE_REL_I386_SEG12             = 0x0009
    81  	IMAGE_REL_I386_SECTION           = 0x000A
    82  	IMAGE_REL_I386_SECREL            = 0x000B
    83  	IMAGE_REL_I386_TOKEN             = 0x000C
    84  	IMAGE_REL_I386_SECREL7           = 0x000D
    85  	IMAGE_REL_I386_REL32             = 0x0014
    86  	IMAGE_REL_AMD64_ABSOLUTE         = 0x0000
    87  	IMAGE_REL_AMD64_ADDR64           = 0x0001
    88  	IMAGE_REL_AMD64_ADDR32           = 0x0002
    89  	IMAGE_REL_AMD64_ADDR32NB         = 0x0003
    90  	IMAGE_REL_AMD64_REL32            = 0x0004
    91  	IMAGE_REL_AMD64_REL32_1          = 0x0005
    92  	IMAGE_REL_AMD64_REL32_2          = 0x0006
    93  	IMAGE_REL_AMD64_REL32_3          = 0x0007
    94  	IMAGE_REL_AMD64_REL32_4          = 0x0008
    95  	IMAGE_REL_AMD64_REL32_5          = 0x0009
    96  	IMAGE_REL_AMD64_SECTION          = 0x000A
    97  	IMAGE_REL_AMD64_SECREL           = 0x000B
    98  	IMAGE_REL_AMD64_SECREL7          = 0x000C
    99  	IMAGE_REL_AMD64_TOKEN            = 0x000D
   100  	IMAGE_REL_AMD64_SREL32           = 0x000E
   101  	IMAGE_REL_AMD64_PAIR             = 0x000F
   102  	IMAGE_REL_AMD64_SSPAN32          = 0x0010
   103  )
   104  
   105  // TODO(brainman): maybe just add ReadAt method to bio.Reader instead of creating peBiobuf
   106  
   107  // peBiobuf makes bio.Reader look like io.ReaderAt.
   108  type peBiobuf bio.Reader
   109  
   110  func (f *peBiobuf) ReadAt(p []byte, off int64) (int, error) {
   111  	ret := ((*bio.Reader)(f)).Seek(off, 0)
   112  	if ret < 0 {
   113  		return 0, errors.New("fail to seek")
   114  	}
   115  	n, err := f.Read(p)
   116  	if err != nil {
   117  		return 0, err
   118  	}
   119  	return n, nil
   120  }
   121  
   122  func ldpe(ctxt *Link, input *bio.Reader, pkg string, length int64, pn string) {
   123  	err := ldpeError(ctxt, input, pkg, length, pn)
   124  	if err != nil {
   125  		Errorf(nil, "%s: malformed pe file: %v", pn, err)
   126  	}
   127  }
   128  
   129  func ldpeError(ctxt *Link, input *bio.Reader, pkg string, length int64, pn string) error {
   130  	if ctxt.Debugvlog != 0 {
   131  		ctxt.Logf("%5.2f ldpe %s\n", Cputime(), pn)
   132  	}
   133  
   134  	localSymVersion := ctxt.Syms.IncVersion()
   135  
   136  	sectsyms := make(map[*pe.Section]*sym.Symbol)
   137  	sectdata := make(map[*pe.Section][]byte)
   138  
   139  	// Some input files are archives containing multiple of
   140  	// object files, and pe.NewFile seeks to the start of
   141  	// input file and get confused. Create section reader
   142  	// to stop pe.NewFile looking before current position.
   143  	sr := io.NewSectionReader((*peBiobuf)(input), input.Offset(), 1<<63-1)
   144  
   145  	// TODO: replace pe.NewFile with pe.Load (grep for "add Load function" in debug/pe for details)
   146  	f, err := pe.NewFile(sr)
   147  	if err != nil {
   148  		return err
   149  	}
   150  	defer f.Close()
   151  
   152  	// TODO return error if found .cormeta
   153  
   154  	// create symbols for mapped sections
   155  	for _, sect := range f.Sections {
   156  		if sect.Characteristics&IMAGE_SCN_MEM_DISCARDABLE != 0 {
   157  			continue
   158  		}
   159  
   160  		if sect.Characteristics&(IMAGE_SCN_CNT_CODE|IMAGE_SCN_CNT_INITIALIZED_DATA|IMAGE_SCN_CNT_UNINITIALIZED_DATA) == 0 {
   161  			// This has been seen for .idata sections, which we
   162  			// want to ignore. See issues 5106 and 5273.
   163  			continue
   164  		}
   165  
   166  		name := fmt.Sprintf("%s(%s)", pkg, sect.Name)
   167  		s := ctxt.Syms.Lookup(name, localSymVersion)
   168  
   169  		switch sect.Characteristics & (IMAGE_SCN_CNT_UNINITIALIZED_DATA | IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE | IMAGE_SCN_CNT_CODE | IMAGE_SCN_MEM_EXECUTE) {
   170  		case IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ: //.rdata
   171  			s.Type = sym.SRODATA
   172  
   173  		case IMAGE_SCN_CNT_UNINITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE: //.bss
   174  			s.Type = sym.SNOPTRBSS
   175  
   176  		case IMAGE_SCN_CNT_INITIALIZED_DATA | IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE: //.data
   177  			s.Type = sym.SNOPTRDATA
   178  
   179  		case IMAGE_SCN_CNT_CODE | IMAGE_SCN_MEM_EXECUTE | IMAGE_SCN_MEM_READ: //.text
   180  			s.Type = sym.STEXT
   181  
   182  		default:
   183  			return fmt.Errorf("unexpected flags %#06x for PE section %s", sect.Characteristics, sect.Name)
   184  		}
   185  
   186  		if s.Type != sym.SNOPTRBSS {
   187  			data, err := sect.Data()
   188  			if err != nil {
   189  				return err
   190  			}
   191  			sectdata[sect] = data
   192  			s.P = data
   193  		}
   194  		s.Size = int64(sect.Size)
   195  		sectsyms[sect] = s
   196  		if sect.Name == ".rsrc" {
   197  			setpersrc(ctxt, s)
   198  		}
   199  	}
   200  
   201  	// load relocations
   202  	for _, rsect := range f.Sections {
   203  		if _, found := sectsyms[rsect]; !found {
   204  			continue
   205  		}
   206  		if rsect.NumberOfRelocations == 0 {
   207  			continue
   208  		}
   209  		if rsect.Characteristics&IMAGE_SCN_MEM_DISCARDABLE != 0 {
   210  			continue
   211  		}
   212  		if rsect.Characteristics&(IMAGE_SCN_CNT_CODE|IMAGE_SCN_CNT_INITIALIZED_DATA|IMAGE_SCN_CNT_UNINITIALIZED_DATA) == 0 {
   213  			// This has been seen for .idata sections, which we
   214  			// want to ignore. See issues 5106 and 5273.
   215  			continue
   216  		}
   217  
   218  		rs := make([]sym.Reloc, rsect.NumberOfRelocations)
   219  		for j, r := range rsect.Relocs {
   220  			rp := &rs[j]
   221  			if int(r.SymbolTableIndex) >= len(f.COFFSymbols) {
   222  				return fmt.Errorf("relocation number %d symbol index idx=%d cannot be large then number of symbols %d", j, r.SymbolTableIndex, len(f.COFFSymbols))
   223  			}
   224  			pesym := &f.COFFSymbols[r.SymbolTableIndex]
   225  			gosym, err := readpesym(ctxt, f, pesym, sectsyms, localSymVersion)
   226  			if err != nil {
   227  				return err
   228  			}
   229  			if gosym == nil {
   230  				name, err := pesym.FullName(f.StringTable)
   231  				if err != nil {
   232  					name = string(pesym.Name[:])
   233  				}
   234  				return fmt.Errorf("reloc of invalid sym %s idx=%d type=%d", name, r.SymbolTableIndex, pesym.Type)
   235  			}
   236  
   237  			rp.Sym = gosym
   238  			rp.Siz = 4
   239  			rp.Off = int32(r.VirtualAddress)
   240  			switch r.Type {
   241  			default:
   242  				Errorf(sectsyms[rsect], "%s: unknown relocation type %d;", pn, r.Type)
   243  				fallthrough
   244  
   245  			case IMAGE_REL_I386_REL32, IMAGE_REL_AMD64_REL32,
   246  				IMAGE_REL_AMD64_ADDR32, // R_X86_64_PC32
   247  				IMAGE_REL_AMD64_ADDR32NB:
   248  				rp.Type = objabi.R_PCREL
   249  
   250  				rp.Add = int64(int32(Le32(sectdata[rsect][rp.Off:])))
   251  
   252  			case IMAGE_REL_I386_DIR32NB, IMAGE_REL_I386_DIR32:
   253  				rp.Type = objabi.R_ADDR
   254  
   255  				// load addend from image
   256  				rp.Add = int64(int32(Le32(sectdata[rsect][rp.Off:])))
   257  
   258  			case IMAGE_REL_AMD64_ADDR64: // R_X86_64_64
   259  				rp.Siz = 8
   260  
   261  				rp.Type = objabi.R_ADDR
   262  
   263  				// load addend from image
   264  				rp.Add = int64(Le64(sectdata[rsect][rp.Off:]))
   265  			}
   266  
   267  			// ld -r could generate multiple section symbols for the
   268  			// same section but with different values, we have to take
   269  			// that into account
   270  			if issect(pesym) {
   271  				rp.Add += int64(pesym.Value)
   272  			}
   273  		}
   274  
   275  		sort.Sort(sym.RelocByOff(rs[:rsect.NumberOfRelocations]))
   276  
   277  		s := sectsyms[rsect]
   278  		s.R = rs
   279  		s.R = s.R[:rsect.NumberOfRelocations]
   280  	}
   281  
   282  	// enter sub-symbols into symbol table.
   283  	for i, numaux := 0, 0; i < len(f.COFFSymbols); i += numaux + 1 {
   284  		pesym := &f.COFFSymbols[i]
   285  
   286  		numaux = int(pesym.NumberOfAuxSymbols)
   287  
   288  		name, err := pesym.FullName(f.StringTable)
   289  		if err != nil {
   290  			return err
   291  		}
   292  		if name == "" {
   293  			continue
   294  		}
   295  		if issect(pesym) {
   296  			continue
   297  		}
   298  		if int(pesym.SectionNumber) > len(f.Sections) {
   299  			continue
   300  		}
   301  		if pesym.SectionNumber == IMAGE_SYM_DEBUG {
   302  			continue
   303  		}
   304  		var sect *pe.Section
   305  		if pesym.SectionNumber > 0 {
   306  			sect = f.Sections[pesym.SectionNumber-1]
   307  			if _, found := sectsyms[sect]; !found {
   308  				continue
   309  			}
   310  		}
   311  
   312  		s, err := readpesym(ctxt, f, pesym, sectsyms, localSymVersion)
   313  		if err != nil {
   314  			return err
   315  		}
   316  
   317  		if pesym.SectionNumber == 0 { // extern
   318  			if s.Type == sym.SDYNIMPORT {
   319  				s.Plt = -2 // flag for dynimport in PE object files.
   320  			}
   321  			if s.Type == sym.SXREF && pesym.Value > 0 { // global data
   322  				s.Type = sym.SNOPTRDATA
   323  				s.Size = int64(pesym.Value)
   324  			}
   325  
   326  			continue
   327  		} else if pesym.SectionNumber > 0 && int(pesym.SectionNumber) <= len(f.Sections) {
   328  			sect = f.Sections[pesym.SectionNumber-1]
   329  			if _, found := sectsyms[sect]; !found {
   330  				Errorf(s, "%s: missing sect.sym", pn)
   331  			}
   332  		} else {
   333  			Errorf(s, "%s: sectnum < 0!", pn)
   334  		}
   335  
   336  		if sect == nil {
   337  			return nil
   338  		}
   339  
   340  		if s.Outer != nil {
   341  			if s.Attr.DuplicateOK() {
   342  				continue
   343  			}
   344  			Exitf("%s: duplicate symbol reference: %s in both %s and %s", pn, s.Name, s.Outer.Name, sectsyms[sect].Name)
   345  		}
   346  
   347  		sectsym := sectsyms[sect]
   348  		s.Sub = sectsym.Sub
   349  		sectsym.Sub = s
   350  		s.Type = sectsym.Type | sym.SSUB
   351  		s.Value = int64(pesym.Value)
   352  		s.Size = 4
   353  		s.Outer = sectsym
   354  		if sectsym.Type == sym.STEXT {
   355  			if s.Attr.External() && !s.Attr.DuplicateOK() {
   356  				Errorf(s, "%s: duplicate symbol definition", pn)
   357  			}
   358  			s.Attr |= sym.AttrExternal
   359  		}
   360  	}
   361  
   362  	// Sort outer lists by address, adding to textp.
   363  	// This keeps textp in increasing address order.
   364  	for _, sect := range f.Sections {
   365  		s := sectsyms[sect]
   366  		if s == nil {
   367  			continue
   368  		}
   369  		if s.Sub != nil {
   370  			s.Sub = sym.SortSub(s.Sub)
   371  		}
   372  		if s.Type == sym.STEXT {
   373  			if s.Attr.OnList() {
   374  				log.Fatalf("symbol %s listed multiple times", s.Name)
   375  			}
   376  			s.Attr |= sym.AttrOnList
   377  			ctxt.Textp = append(ctxt.Textp, s)
   378  			for s = s.Sub; s != nil; s = s.Sub {
   379  				if s.Attr.OnList() {
   380  					log.Fatalf("symbol %s listed multiple times", s.Name)
   381  				}
   382  				s.Attr |= sym.AttrOnList
   383  				ctxt.Textp = append(ctxt.Textp, s)
   384  			}
   385  		}
   386  	}
   387  
   388  	return nil
   389  }
   390  
   391  func issect(s *pe.COFFSymbol) bool {
   392  	return s.StorageClass == IMAGE_SYM_CLASS_STATIC && s.Type == 0 && s.Name[0] == '.'
   393  }
   394  
   395  func readpesym(ctxt *Link, f *pe.File, pesym *pe.COFFSymbol, sectsyms map[*pe.Section]*sym.Symbol, localSymVersion int) (*sym.Symbol, error) {
   396  	symname, err := pesym.FullName(f.StringTable)
   397  	if err != nil {
   398  		return nil, err
   399  	}
   400  	var name string
   401  	if issect(pesym) {
   402  		name = sectsyms[f.Sections[pesym.SectionNumber-1]].Name
   403  	} else {
   404  		name = symname
   405  		if strings.HasPrefix(name, "__imp_") {
   406  			name = name[6:] // __imp_Name => Name
   407  		}
   408  		if ctxt.Arch.Family == sys.I386 && name[0] == '_' {
   409  			name = name[1:] // _Name => Name
   410  		}
   411  	}
   412  
   413  	// remove last @XXX
   414  	if i := strings.LastIndex(name, "@"); i >= 0 {
   415  		name = name[:i]
   416  	}
   417  
   418  	var s *sym.Symbol
   419  	switch pesym.Type {
   420  	default:
   421  		return nil, fmt.Errorf("%s: invalid symbol type %d", symname, pesym.Type)
   422  
   423  	case IMAGE_SYM_DTYPE_FUNCTION, IMAGE_SYM_DTYPE_NULL:
   424  		switch pesym.StorageClass {
   425  		case IMAGE_SYM_CLASS_EXTERNAL: //global
   426  			s = ctxt.Syms.Lookup(name, 0)
   427  
   428  		case IMAGE_SYM_CLASS_NULL, IMAGE_SYM_CLASS_STATIC, IMAGE_SYM_CLASS_LABEL:
   429  			s = ctxt.Syms.Lookup(name, localSymVersion)
   430  			s.Attr |= sym.AttrDuplicateOK
   431  
   432  		default:
   433  			return nil, fmt.Errorf("%s: invalid symbol binding %d", symname, pesym.StorageClass)
   434  		}
   435  	}
   436  
   437  	if s != nil && s.Type == 0 && (pesym.StorageClass != IMAGE_SYM_CLASS_STATIC || pesym.Value != 0) {
   438  		s.Type = sym.SXREF
   439  	}
   440  	if strings.HasPrefix(symname, "__imp_") {
   441  		s.Got = -2 // flag for __imp_
   442  	}
   443  
   444  	return s, nil
   445  }