|  | // Copyright 2016 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package pe | 
|  |  | 
|  | import ( | 
|  | "encoding/binary" | 
|  | "fmt" | 
|  | "io" | 
|  | "strconv" | 
|  | ) | 
|  |  | 
|  | // SectionHeader32 represents real PE COFF section header. | 
|  | type SectionHeader32 struct { | 
|  | Name                 [8]uint8 | 
|  | VirtualSize          uint32 | 
|  | VirtualAddress       uint32 | 
|  | SizeOfRawData        uint32 | 
|  | PointerToRawData     uint32 | 
|  | PointerToRelocations uint32 | 
|  | PointerToLineNumbers uint32 | 
|  | NumberOfRelocations  uint16 | 
|  | NumberOfLineNumbers  uint16 | 
|  | Characteristics      uint32 | 
|  | } | 
|  |  | 
|  | // fullName finds real name of section sh. Normally name is stored | 
|  | // in sh.Name, but if it is longer then 8 characters, it is stored | 
|  | // in COFF string table st instead. | 
|  | func (sh *SectionHeader32) fullName(st StringTable) (string, error) { | 
|  | if sh.Name[0] != '/' { | 
|  | return cstring(sh.Name[:]), nil | 
|  | } | 
|  | i, err := strconv.Atoi(cstring(sh.Name[1:])) | 
|  | if err != nil { | 
|  | return "", err | 
|  | } | 
|  | return st.String(uint32(i)) | 
|  | } | 
|  |  | 
|  | // TODO(brainman): copy all IMAGE_REL_* consts from ldpe.go here | 
|  |  | 
|  | // Reloc represents a PE COFF relocation. | 
|  | // Each section contains its own relocation list. | 
|  | type Reloc struct { | 
|  | VirtualAddress   uint32 | 
|  | SymbolTableIndex uint32 | 
|  | Type             uint16 | 
|  | } | 
|  |  | 
|  | func readRelocs(sh *SectionHeader, r io.ReadSeeker) ([]Reloc, error) { | 
|  | if sh.NumberOfRelocations <= 0 { | 
|  | return nil, nil | 
|  | } | 
|  | _, err := r.Seek(int64(sh.PointerToRelocations), seekStart) | 
|  | if err != nil { | 
|  | return nil, fmt.Errorf("fail to seek to %q section relocations: %v", sh.Name, err) | 
|  | } | 
|  | relocs := make([]Reloc, sh.NumberOfRelocations) | 
|  | err = binary.Read(r, binary.LittleEndian, relocs) | 
|  | if err != nil { | 
|  | return nil, fmt.Errorf("fail to read section relocations: %v", err) | 
|  | } | 
|  | return relocs, nil | 
|  | } | 
|  |  | 
|  | // SectionHeader is similar to SectionHeader32 with Name | 
|  | // field replaced by Go string. | 
|  | type SectionHeader struct { | 
|  | Name                 string | 
|  | VirtualSize          uint32 | 
|  | VirtualAddress       uint32 | 
|  | Size                 uint32 | 
|  | Offset               uint32 | 
|  | PointerToRelocations uint32 | 
|  | PointerToLineNumbers uint32 | 
|  | NumberOfRelocations  uint16 | 
|  | NumberOfLineNumbers  uint16 | 
|  | Characteristics      uint32 | 
|  | } | 
|  |  | 
|  | // Section provides access to PE COFF section. | 
|  | type Section struct { | 
|  | SectionHeader | 
|  | Relocs []Reloc | 
|  |  | 
|  | // Embed ReaderAt for ReadAt method. | 
|  | // Do not embed SectionReader directly | 
|  | // to avoid having Read and Seek. | 
|  | // If a client wants Read and Seek it must use | 
|  | // Open() to avoid fighting over the seek offset | 
|  | // with other clients. | 
|  | io.ReaderAt | 
|  | sr *io.SectionReader | 
|  | } | 
|  |  | 
|  | // Data reads and returns the contents of the PE section s. | 
|  | func (s *Section) Data() ([]byte, error) { | 
|  | dat := make([]byte, s.sr.Size()) | 
|  | n, err := s.sr.ReadAt(dat, 0) | 
|  | if n == len(dat) { | 
|  | err = nil | 
|  | } | 
|  | return dat[0:n], err | 
|  | } | 
|  |  | 
|  | // Open returns a new ReadSeeker reading the PE section s. | 
|  | func (s *Section) Open() io.ReadSeeker { | 
|  | return io.NewSectionReader(s.sr, 0, 1<<63-1) | 
|  | } |