2023-01-17 23:59:04 +03:00
|
|
|
package internal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"debug/elf"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
)
|
|
|
|
|
|
|
|
type SafeELFFile struct {
|
|
|
|
*elf.File
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewSafeELFFile reads an ELF safely.
|
|
|
|
//
|
|
|
|
// Any panic during parsing is turned into an error. This is necessary since
|
|
|
|
// there are a bunch of unfixed bugs in debug/elf.
|
|
|
|
//
|
|
|
|
// https://github.com/golang/go/issues?q=is%3Aissue+is%3Aopen+debug%2Felf+in%3Atitle
|
|
|
|
func NewSafeELFFile(r io.ReaderAt) (safe *SafeELFFile, err error) {
|
|
|
|
defer func() {
|
|
|
|
r := recover()
|
|
|
|
if r == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
safe = nil
|
|
|
|
err = fmt.Errorf("reading ELF file panicked: %s", r)
|
|
|
|
}()
|
|
|
|
|
|
|
|
file, err := elf.NewFile(r)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &SafeELFFile{file}, nil
|
|
|
|
}
|
|
|
|
|
2023-04-03 12:16:17 +03:00
|
|
|
// OpenSafeELFFile reads an ELF from a file.
|
|
|
|
//
|
|
|
|
// It works like NewSafeELFFile, with the exception that safe.Close will
|
|
|
|
// close the underlying file.
|
|
|
|
func OpenSafeELFFile(path string) (safe *SafeELFFile, err error) {
|
|
|
|
defer func() {
|
|
|
|
r := recover()
|
|
|
|
if r == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
safe = nil
|
|
|
|
err = fmt.Errorf("reading ELF file panicked: %s", r)
|
|
|
|
}()
|
|
|
|
|
|
|
|
file, err := elf.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &SafeELFFile{file}, nil
|
|
|
|
}
|
|
|
|
|
2023-01-17 23:59:04 +03:00
|
|
|
// Symbols is the safe version of elf.File.Symbols.
|
|
|
|
func (se *SafeELFFile) Symbols() (syms []elf.Symbol, err error) {
|
|
|
|
defer func() {
|
|
|
|
r := recover()
|
|
|
|
if r == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
syms = nil
|
|
|
|
err = fmt.Errorf("reading ELF symbols panicked: %s", r)
|
|
|
|
}()
|
|
|
|
|
|
|
|
syms, err = se.File.Symbols()
|
|
|
|
return
|
|
|
|
}
|
2023-04-03 12:16:17 +03:00
|
|
|
|
|
|
|
// DynamicSymbols is the safe version of elf.File.DynamicSymbols.
|
|
|
|
func (se *SafeELFFile) DynamicSymbols() (syms []elf.Symbol, err error) {
|
|
|
|
defer func() {
|
|
|
|
r := recover()
|
|
|
|
if r == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
syms = nil
|
|
|
|
err = fmt.Errorf("reading ELF dynamic symbols panicked: %s", r)
|
|
|
|
}()
|
|
|
|
|
|
|
|
syms, err = se.File.DynamicSymbols()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// SectionsByType returns all sections in the file with the specified section type.
|
|
|
|
func (se *SafeELFFile) SectionsByType(typ elf.SectionType) []*elf.Section {
|
|
|
|
sections := make([]*elf.Section, 0, 1)
|
|
|
|
for _, section := range se.Sections {
|
|
|
|
if section.Type == typ {
|
|
|
|
sections = append(sections, section)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sections
|
|
|
|
}
|