github.com/varialus/godfly@v0.0.0-20130904042352-1934f9f095ab/src/pkg/debug/pe/file.go (about) 1 // Copyright 2009 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 pe implements access to PE (Microsoft Windows Portable Executable) files. 6 package pe 7 8 import ( 9 "debug/dwarf" 10 "encoding/binary" 11 "errors" 12 "fmt" 13 "io" 14 "os" 15 "strconv" 16 ) 17 18 // A File represents an open PE file. 19 type File struct { 20 FileHeader 21 Sections []*Section 22 Symbols []*Symbol 23 24 closer io.Closer 25 } 26 27 type SectionHeader struct { 28 Name string 29 VirtualSize uint32 30 VirtualAddress uint32 31 Size uint32 32 Offset uint32 33 PointerToRelocations uint32 34 PointerToLineNumbers uint32 35 NumberOfRelocations uint16 36 NumberOfLineNumbers uint16 37 Characteristics uint32 38 } 39 40 type Section struct { 41 SectionHeader 42 43 // Embed ReaderAt for ReadAt method. 44 // Do not embed SectionReader directly 45 // to avoid having Read and Seek. 46 // If a client wants Read and Seek it must use 47 // Open() to avoid fighting over the seek offset 48 // with other clients. 49 io.ReaderAt 50 sr *io.SectionReader 51 } 52 53 type Symbol struct { 54 Name string 55 Value uint32 56 SectionNumber int16 57 Type uint16 58 StorageClass uint8 59 } 60 61 type ImportDirectory struct { 62 OriginalFirstThunk uint32 63 TimeDateStamp uint32 64 ForwarderChain uint32 65 Name uint32 66 FirstThunk uint32 67 68 dll string 69 } 70 71 // Data reads and returns the contents of the PE section. 72 func (s *Section) Data() ([]byte, error) { 73 dat := make([]byte, s.sr.Size()) 74 n, err := s.sr.ReadAt(dat, 0) 75 return dat[0:n], err 76 } 77 78 // Open returns a new ReadSeeker reading the PE section. 79 func (s *Section) Open() io.ReadSeeker { return io.NewSectionReader(s.sr, 0, 1<<63-1) } 80 81 type FormatError struct { 82 off int64 83 msg string 84 val interface{} 85 } 86 87 func (e *FormatError) Error() string { 88 msg := e.msg 89 if e.val != nil { 90 msg += fmt.Sprintf(" '%v'", e.val) 91 } 92 msg += fmt.Sprintf(" in record at byte %#x", e.off) 93 return msg 94 } 95 96 // Open opens the named file using os.Open and prepares it for use as a PE binary. 97 func Open(name string) (*File, error) { 98 f, err := os.Open(name) 99 if err != nil { 100 return nil, err 101 } 102 ff, err := NewFile(f) 103 if err != nil { 104 f.Close() 105 return nil, err 106 } 107 ff.closer = f 108 return ff, nil 109 } 110 111 // Close closes the File. 112 // If the File was created using NewFile directly instead of Open, 113 // Close has no effect. 114 func (f *File) Close() error { 115 var err error 116 if f.closer != nil { 117 err = f.closer.Close() 118 f.closer = nil 119 } 120 return err 121 } 122 123 // NewFile creates a new File for accessing a PE binary in an underlying reader. 124 func NewFile(r io.ReaderAt) (*File, error) { 125 f := new(File) 126 sr := io.NewSectionReader(r, 0, 1<<63-1) 127 128 var dosheader [96]byte 129 if _, err := r.ReadAt(dosheader[0:], 0); err != nil { 130 return nil, err 131 } 132 var base int64 133 if dosheader[0] == 'M' && dosheader[1] == 'Z' { 134 signoff := int64(binary.LittleEndian.Uint32(dosheader[0x3c:])) 135 var sign [4]byte 136 r.ReadAt(sign[:], signoff) 137 if !(sign[0] == 'P' && sign[1] == 'E' && sign[2] == 0 && sign[3] == 0) { 138 return nil, errors.New("Invalid PE File Format.") 139 } 140 base = signoff + 4 141 } else { 142 base = int64(0) 143 } 144 sr.Seek(base, os.SEEK_SET) 145 if err := binary.Read(sr, binary.LittleEndian, &f.FileHeader); err != nil { 146 return nil, err 147 } 148 if f.FileHeader.Machine != IMAGE_FILE_MACHINE_UNKNOWN && f.FileHeader.Machine != IMAGE_FILE_MACHINE_AMD64 && f.FileHeader.Machine != IMAGE_FILE_MACHINE_I386 { 149 return nil, errors.New("Invalid PE File Format.") 150 } 151 152 var ss []byte 153 if f.FileHeader.NumberOfSymbols > 0 { 154 // Get COFF string table, which is located at the end of the COFF symbol table. 155 sr.Seek(int64(f.FileHeader.PointerToSymbolTable+COFFSymbolSize*f.FileHeader.NumberOfSymbols), os.SEEK_SET) 156 var l uint32 157 if err := binary.Read(sr, binary.LittleEndian, &l); err != nil { 158 return nil, err 159 } 160 ss = make([]byte, l) 161 if _, err := r.ReadAt(ss, int64(f.FileHeader.PointerToSymbolTable+COFFSymbolSize*f.FileHeader.NumberOfSymbols)); err != nil { 162 return nil, err 163 } 164 165 // Process COFF symbol table. 166 sr.Seek(int64(f.FileHeader.PointerToSymbolTable), os.SEEK_SET) 167 aux := uint8(0) 168 for i := 0; i < int(f.FileHeader.NumberOfSymbols); i++ { 169 cs := new(COFFSymbol) 170 if err := binary.Read(sr, binary.LittleEndian, cs); err != nil { 171 return nil, err 172 } 173 if aux > 0 { 174 aux-- 175 continue 176 } 177 var name string 178 if cs.Name[0] == 0 && cs.Name[1] == 0 && cs.Name[2] == 0 && cs.Name[3] == 0 { 179 si := int(binary.LittleEndian.Uint32(cs.Name[4:])) 180 name, _ = getString(ss, si) 181 } else { 182 name = cstring(cs.Name[:]) 183 } 184 aux = cs.NumberOfAuxSymbols 185 s := &Symbol{ 186 Name: name, 187 Value: cs.Value, 188 SectionNumber: cs.SectionNumber, 189 Type: cs.Type, 190 StorageClass: cs.StorageClass, 191 } 192 f.Symbols = append(f.Symbols, s) 193 } 194 } 195 196 // Process sections. 197 sr.Seek(base, os.SEEK_SET) 198 binary.Read(sr, binary.LittleEndian, &f.FileHeader) 199 sr.Seek(int64(f.FileHeader.SizeOfOptionalHeader), os.SEEK_CUR) //Skip OptionalHeader 200 f.Sections = make([]*Section, f.FileHeader.NumberOfSections) 201 for i := 0; i < int(f.FileHeader.NumberOfSections); i++ { 202 sh := new(SectionHeader32) 203 if err := binary.Read(sr, binary.LittleEndian, sh); err != nil { 204 return nil, err 205 } 206 var name string 207 if sh.Name[0] == '\x2F' { 208 si, _ := strconv.Atoi(cstring(sh.Name[1:])) 209 name, _ = getString(ss, si) 210 } else { 211 name = cstring(sh.Name[0:]) 212 } 213 s := new(Section) 214 s.SectionHeader = SectionHeader{ 215 Name: name, 216 VirtualSize: uint32(sh.VirtualSize), 217 VirtualAddress: uint32(sh.VirtualAddress), 218 Size: uint32(sh.SizeOfRawData), 219 Offset: uint32(sh.PointerToRawData), 220 PointerToRelocations: uint32(sh.PointerToRelocations), 221 PointerToLineNumbers: uint32(sh.PointerToLineNumbers), 222 NumberOfRelocations: uint16(sh.NumberOfRelocations), 223 NumberOfLineNumbers: uint16(sh.NumberOfLineNumbers), 224 Characteristics: uint32(sh.Characteristics), 225 } 226 s.sr = io.NewSectionReader(r, int64(s.SectionHeader.Offset), int64(s.SectionHeader.Size)) 227 s.ReaderAt = s.sr 228 f.Sections[i] = s 229 } 230 return f, nil 231 } 232 233 func cstring(b []byte) string { 234 var i int 235 for i = 0; i < len(b) && b[i] != 0; i++ { 236 } 237 return string(b[0:i]) 238 } 239 240 // getString extracts a string from symbol string table. 241 func getString(section []byte, start int) (string, bool) { 242 if start < 0 || start >= len(section) { 243 return "", false 244 } 245 246 for end := start; end < len(section); end++ { 247 if section[end] == 0 { 248 return string(section[start:end]), true 249 } 250 } 251 return "", false 252 } 253 254 // Section returns the first section with the given name, or nil if no such 255 // section exists. 256 func (f *File) Section(name string) *Section { 257 for _, s := range f.Sections { 258 if s.Name == name { 259 return s 260 } 261 } 262 return nil 263 } 264 265 func (f *File) DWARF() (*dwarf.Data, error) { 266 // There are many other DWARF sections, but these 267 // are the required ones, and the debug/dwarf package 268 // does not use the others, so don't bother loading them. 269 var names = [...]string{"abbrev", "info", "str"} 270 var dat [len(names)][]byte 271 for i, name := range names { 272 name = ".debug_" + name 273 s := f.Section(name) 274 if s == nil { 275 continue 276 } 277 b, err := s.Data() 278 if err != nil && uint32(len(b)) < s.Size { 279 return nil, err 280 } 281 dat[i] = b 282 } 283 284 abbrev, info, str := dat[0], dat[1], dat[2] 285 return dwarf.New(abbrev, nil, nil, info, nil, nil, nil, str) 286 } 287 288 // ImportedSymbols returns the names of all symbols 289 // referred to by the binary f that are expected to be 290 // satisfied by other libraries at dynamic load time. 291 // It does not return weak symbols. 292 func (f *File) ImportedSymbols() ([]string, error) { 293 pe64 := f.Machine == IMAGE_FILE_MACHINE_AMD64 294 ds := f.Section(".idata") 295 if ds == nil { 296 // not dynamic, so no libraries 297 return nil, nil 298 } 299 d, err := ds.Data() 300 if err != nil { 301 return nil, err 302 } 303 var ida []ImportDirectory 304 for len(d) > 0 { 305 var dt ImportDirectory 306 dt.OriginalFirstThunk = binary.LittleEndian.Uint32(d[0:4]) 307 dt.Name = binary.LittleEndian.Uint32(d[12:16]) 308 dt.FirstThunk = binary.LittleEndian.Uint32(d[16:20]) 309 d = d[20:] 310 if dt.OriginalFirstThunk == 0 { 311 break 312 } 313 ida = append(ida, dt) 314 } 315 names, _ := ds.Data() 316 var all []string 317 for _, dt := range ida { 318 dt.dll, _ = getString(names, int(dt.Name-ds.VirtualAddress)) 319 d, _ = ds.Data() 320 // seek to OriginalFirstThunk 321 d = d[dt.OriginalFirstThunk-ds.VirtualAddress:] 322 for len(d) > 0 { 323 if pe64 { // 64bit 324 va := binary.LittleEndian.Uint64(d[0:8]) 325 d = d[8:] 326 if va == 0 { 327 break 328 } 329 if va&0x8000000000000000 > 0 { // is Ordinal 330 // TODO add dynimport ordinal support. 331 } else { 332 fn, _ := getString(names, int(uint32(va)-ds.VirtualAddress+2)) 333 all = append(all, fn+":"+dt.dll) 334 } 335 } else { // 32bit 336 va := binary.LittleEndian.Uint32(d[0:4]) 337 d = d[4:] 338 if va == 0 { 339 break 340 } 341 if va&0x80000000 > 0 { // is Ordinal 342 // TODO add dynimport ordinal support. 343 //ord := va&0x0000FFFF 344 } else { 345 fn, _ := getString(names, int(va-ds.VirtualAddress+2)) 346 all = append(all, fn+":"+dt.dll) 347 } 348 } 349 } 350 } 351 352 return all, nil 353 } 354 355 // ImportedLibraries returns the names of all libraries 356 // referred to by the binary f that are expected to be 357 // linked with the binary at dynamic link time. 358 func (f *File) ImportedLibraries() ([]string, error) { 359 // TODO 360 // cgo -dynimport don't use this for windows PE, so just return. 361 return nil, nil 362 }