-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathelfreader.go
More file actions
95 lines (86 loc) · 2.16 KB
/
elfreader.go
File metadata and controls
95 lines (86 loc) · 2.16 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
// Copyright The Parca Authors
// SPDX-License-Identifier: Apache-2.0
package usdt // import "github.com/parca-dev/usdt"
import (
"debug/elf"
"io"
"os"
)
// StdlibELFReader implements ELFReader using Go's debug/elf package.
type StdlibELFReader struct {
f *elf.File
closer io.Closer // non-nil when opened from a path
}
// NewStdlibELFReader creates an ELFReader from an io.ReaderAt.
func NewStdlibELFReader(r io.ReaderAt) (*StdlibELFReader, error) {
f, err := elf.NewFile(r)
if err != nil {
return nil, err
}
return &StdlibELFReader{f: f}, nil
}
// OpenELF opens an ELF file by path and returns a StdlibELFReader.
// The caller must call Close() when done.
func OpenELF(path string) (*StdlibELFReader, error) {
file, err := os.Open(path)
if err != nil {
return nil, err
}
f, err := elf.NewFile(file)
if err != nil {
file.Close()
return nil, err
}
return &StdlibELFReader{f: f, closer: file}, nil
}
// Sections returns all ELF section headers with their data.
func (r *StdlibELFReader) Sections() ([]ELFSection, error) {
sections := make([]ELFSection, 0, len(r.f.Sections))
for _, s := range r.f.Sections {
sec := ELFSection{
Name: s.Name,
Addr: s.Addr,
}
// Only read data for the sections we actually need
if s.Name == ".note.stapsdt" || s.Name == ".stapsdt.base" {
data, err := io.ReadAll(s.Open())
if err != nil {
return nil, err
}
sec.Data = data
}
sections = append(sections, sec)
}
return sections, nil
}
// LoadSegments returns all PT_LOAD program headers.
func (r *StdlibELFReader) LoadSegments() []ELFProg {
var segs []ELFProg
for _, p := range r.f.Progs {
if p.Type == elf.PT_LOAD {
segs = append(segs, ELFProg{
Vaddr: p.Vaddr,
Memsz: p.Memsz,
Off: p.Off,
})
}
}
return segs
}
// Close closes the underlying file.
func (r *StdlibELFReader) Close() error {
if r.closer != nil {
return r.closer.Close()
}
return nil
}
// ParseProbesFromFile is a convenience function that opens an ELF file,
// parses its USDT probes, and closes it.
func ParseProbesFromFile(path string) ([]Probe, error) {
r, err := OpenELF(path)
if err != nil {
return nil, err
}
defer r.Close()
return ParseProbes(r)
}