| |
| |
| |
|
|
| package pe |
|
|
| import ( |
| "encoding/binary" |
| "errors" |
| "fmt" |
| "internal/saferio" |
| "io" |
| "unsafe" |
| ) |
|
|
| const COFFSymbolSize = 18 |
|
|
| |
| type COFFSymbol struct { |
| Name [8]uint8 |
| Value uint32 |
| SectionNumber int16 |
| Type uint16 |
| StorageClass uint8 |
| NumberOfAuxSymbols uint8 |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| func readCOFFSymbols(fh *FileHeader, r io.ReadSeeker) ([]COFFSymbol, error) { |
| if fh.PointerToSymbolTable == 0 { |
| return nil, nil |
| } |
| if fh.NumberOfSymbols <= 0 { |
| return nil, nil |
| } |
| _, err := r.Seek(int64(fh.PointerToSymbolTable), io.SeekStart) |
| if err != nil { |
| return nil, fmt.Errorf("fail to seek to symbol table: %v", err) |
| } |
| c := saferio.SliceCap[COFFSymbol](uint64(fh.NumberOfSymbols)) |
| if c < 0 { |
| return nil, errors.New("too many symbols; file may be corrupt") |
| } |
| syms := make([]COFFSymbol, 0, c) |
| naux := 0 |
| for k := uint32(0); k < fh.NumberOfSymbols; k++ { |
| var sym COFFSymbol |
| if naux == 0 { |
| |
| err = binary.Read(r, binary.LittleEndian, &sym) |
| if err != nil { |
| return nil, fmt.Errorf("fail to read symbol table: %v", err) |
| } |
| |
| naux = int(sym.NumberOfAuxSymbols) |
| } else { |
| |
| |
| |
| |
| naux-- |
| aux := (*COFFSymbolAuxFormat5)(unsafe.Pointer(&sym)) |
| err = binary.Read(r, binary.LittleEndian, aux) |
| if err != nil { |
| return nil, fmt.Errorf("fail to read symbol table: %v", err) |
| } |
| } |
| syms = append(syms, sym) |
| } |
| if naux != 0 { |
| return nil, fmt.Errorf("fail to read symbol table: %d aux symbols unread", naux) |
| } |
| return syms, nil |
| } |
|
|
| |
| func isSymNameOffset(name [8]byte) (bool, uint32) { |
| if name[0] == 0 && name[1] == 0 && name[2] == 0 && name[3] == 0 { |
| offset := binary.LittleEndian.Uint32(name[4:]) |
| if offset == 0 { |
| |
| return false, 0 |
| } |
| return true, offset |
| } |
| return false, 0 |
| } |
|
|
| |
| |
| |
| func (sym *COFFSymbol) FullName(st StringTable) (string, error) { |
| if ok, offset := isSymNameOffset(sym.Name); ok { |
| return st.String(offset) |
| } |
| return cstring(sym.Name[:]), nil |
| } |
|
|
| func removeAuxSymbols(allsyms []COFFSymbol, st StringTable) ([]*Symbol, error) { |
| if len(allsyms) == 0 { |
| return nil, nil |
| } |
| syms := make([]*Symbol, 0) |
| aux := uint8(0) |
| for _, sym := range allsyms { |
| if aux > 0 { |
| aux-- |
| continue |
| } |
| name, err := sym.FullName(st) |
| if err != nil { |
| return nil, err |
| } |
| aux = sym.NumberOfAuxSymbols |
| s := &Symbol{ |
| Name: name, |
| Value: sym.Value, |
| SectionNumber: sym.SectionNumber, |
| Type: sym.Type, |
| StorageClass: sym.StorageClass, |
| } |
| syms = append(syms, s) |
| } |
| return syms, nil |
| } |
|
|
| |
| |
| type Symbol struct { |
| Name string |
| Value uint32 |
| SectionNumber int16 |
| Type uint16 |
| StorageClass uint8 |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| type COFFSymbolAuxFormat5 struct { |
| Size uint32 |
| NumRelocs uint16 |
| NumLineNumbers uint16 |
| Checksum uint32 |
| SecNum uint16 |
| Selection uint8 |
| _ [3]uint8 |
| } |
|
|
| |
| |
| const ( |
| IMAGE_COMDAT_SELECT_NODUPLICATES = 1 |
| IMAGE_COMDAT_SELECT_ANY = 2 |
| IMAGE_COMDAT_SELECT_SAME_SIZE = 3 |
| IMAGE_COMDAT_SELECT_EXACT_MATCH = 4 |
| IMAGE_COMDAT_SELECT_ASSOCIATIVE = 5 |
| IMAGE_COMDAT_SELECT_LARGEST = 6 |
| ) |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| func (f *File) COFFSymbolReadSectionDefAux(idx int) (*COFFSymbolAuxFormat5, error) { |
| var rv *COFFSymbolAuxFormat5 |
| if idx < 0 || idx >= len(f.COFFSymbols) { |
| return rv, fmt.Errorf("invalid symbol index") |
| } |
| pesym := &f.COFFSymbols[idx] |
| const IMAGE_SYM_CLASS_STATIC = 3 |
| if pesym.StorageClass != uint8(IMAGE_SYM_CLASS_STATIC) { |
| return rv, fmt.Errorf("incorrect symbol storage class") |
| } |
| if pesym.NumberOfAuxSymbols == 0 || idx+1 >= len(f.COFFSymbols) { |
| return rv, fmt.Errorf("aux symbol unavailable") |
| } |
| |
| pesymn := &f.COFFSymbols[idx+1] |
| rv = (*COFFSymbolAuxFormat5)(unsafe.Pointer(pesymn)) |
| return rv, nil |
| } |
|
|