1 // Copyright 2013 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 // Parsing of ELF executables (Linux, FreeBSD, and so on). 6 7 package objfile 8 9 import ( 10 "debug/dwarf" 11 "debug/elf" 12 "encoding/binary" 13 "fmt" 14 "io" 15 ) 16 17 type elfFile struct { 18 elf *elf.File 19 } 20 21 func openElf(r io.ReaderAt) (rawFile, error) { 22 f, err := elf.NewFile(r) 23 if err != nil { 24 return nil, err 25 } 26 return &elfFile{f}, nil 27 } 28 29 func (f *elfFile) symbols() ([]Sym, error) { 30 elfSyms, err := f.elf.Symbols() 31 if err != nil { 32 return nil, err 33 } 34 35 var syms []Sym 36 for _, s := range elfSyms { 37 sym := Sym{Addr: s.Value, Name: s.Name, Size: int64(s.Size), Code: '?'} 38 switch s.Section { 39 case elf.SHN_UNDEF: 40 sym.Code = 'U' 41 case elf.SHN_COMMON: 42 sym.Code = 'B' 43 default: 44 i := int(s.Section) 45 if i < 0 || i >= len(f.elf.Sections) { 46 break 47 } 48 sect := f.elf.Sections[i] 49 switch sect.Flags & (elf.SHF_WRITE | elf.SHF_ALLOC | elf.SHF_EXECINSTR) { 50 case elf.SHF_ALLOC | elf.SHF_EXECINSTR: 51 sym.Code = 'T' 52 case elf.SHF_ALLOC: 53 sym.Code = 'R' 54 case elf.SHF_ALLOC | elf.SHF_WRITE: 55 sym.Code = 'D' 56 } 57 } 58 if elf.ST_BIND(s.Info) == elf.STB_LOCAL { 59 sym.Code += 'a' - 'A' 60 } 61 syms = append(syms, sym) 62 } 63 64 return syms, nil 65 } 66 67 func (f *elfFile) pcln() (textStart uint64, symtab, pclntab []byte, err error) { 68 if sect := f.elf.Section(".text"); sect != nil { 69 textStart = sect.Addr 70 } 71 if sect := f.elf.Section(".gosymtab"); sect != nil { 72 if symtab, err = sect.Data(); err != nil { 73 return 0, nil, nil, err 74 } 75 } 76 if sect := f.elf.Section(".gopclntab"); sect != nil { 77 if pclntab, err = sect.Data(); err != nil { 78 return 0, nil, nil, err 79 } 80 } 81 return textStart, symtab, pclntab, nil 82 } 83 84 func (f *elfFile) text() (textStart uint64, text []byte, err error) { 85 sect := f.elf.Section(".text") 86 if sect == nil { 87 return 0, nil, fmt.Errorf("text section not found") 88 } 89 textStart = sect.Addr 90 text, err = sect.Data() 91 return 92 } 93 94 func (f *elfFile) goarch() string { 95 switch f.elf.Machine { 96 case elf.EM_386: 97 return "386" 98 case elf.EM_X86_64: 99 return "amd64" 100 case elf.EM_ARM: 101 return "arm" 102 case elf.EM_AARCH64: 103 return "arm64" 104 case elf.EM_PPC64: 105 if f.elf.ByteOrder == binary.LittleEndian { 106 return "ppc64le" 107 } 108 return "ppc64" 109 case elf.EM_S390: 110 return "s390x" 111 } 112 return "" 113 } 114 115 func (f *elfFile) loadAddress() (uint64, error) { 116 for _, p := range f.elf.Progs { 117 if p.Type == elf.PT_LOAD { 118 return p.Vaddr, nil 119 } 120 } 121 return 0, fmt.Errorf("unknown load address") 122 } 123 124 func (f *elfFile) dwarf() (*dwarf.Data, error) { 125 return f.elf.DWARF() 126 } 127