-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain.go
129 lines (97 loc) · 3.59 KB
/
main.go
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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// Copyright 2024 Leon Hwang.
// SPDX-License-Identifier: Apache-2.0
package main
import (
"context"
"errors"
"log"
"os"
"os/signal"
"syscall"
"github.com/cilium/ebpf"
"github.com/cilium/ebpf/ringbuf"
"github.com/knightsc/gapstone"
"golang.org/x/sync/errgroup"
"golang.org/x/sys/unix"
"github.com/Asphaltt/bpflbr/internal/assert"
"github.com/Asphaltt/bpflbr/internal/bpflbr"
)
//go:generate go run github.com/cilium/ebpf/cmd/bpf2go -cc clang lbr ./bpf/lbr.c -- -g -D__TARGET_ARCH_x86 -I./bpf/headers -Wall
//go:generate go run github.com/cilium/ebpf/cmd/bpf2go -cc clang feat ./bpf/feature.c -- -g -D__TARGET_ARCH_x86 -I./bpf/headers -Wall
func main() {
flags, err := bpflbr.ParseFlags()
assert.NoErr(err, "Failed to parse flags: %v")
progs, err := flags.ParseProgs()
assert.NoErr(err, "Failed to parse bpf prog infos: %v")
if flags.DumpProg() {
assert.SliceLen(progs, 1, "Only one --prog is allowed for --dump-jited")
bpflbr.DumpProg(progs)
return
}
featBPFSpec, err := loadFeat()
assert.NoErr(err, "Failed to load feat bpf spec: %v")
err = bpflbr.DetectBPFFeatures(featBPFSpec)
assert.NoErr(err, "Failed to detect bpf features: %v")
lbrPerfEvents, err := bpflbr.OpenLbrPerfEvent()
if err != nil && errors.Is(err, unix.ENOENT) {
log.Fatalln("LBR is not supported on current system")
}
assert.NoErr(err, "Failed to open LBR perf event: %v")
defer lbrPerfEvents.Close()
kallsyms, err := bpflbr.NewKallsyms()
assert.NoErr(err, "Failed to read /proc/kallsyms: %v")
vmlinux, err := bpflbr.FindVmlinux()
assert.NoErr(err, "Failed to find vmlinux: %v")
textAddr, err := bpflbr.ReadTextAddrFromVmlinux(vmlinux)
assert.NoErr(err, "Failed to read .text address from vmlinux: %v")
kaslrOffset := textAddr - kallsyms.Stext()
addr2line, err := bpflbr.NewAddr2Line(vmlinux, kaslrOffset, kallsyms.SysBPF())
assert.NoErr(err, "Failed to create addr2line: %v")
engine, err := gapstone.New(int(gapstone.CS_ARCH_X86), int(gapstone.CS_MODE_64))
assert.NoErr(err, "Failed to create capstone engine: %v")
defer engine.Close()
bpfProgs, err := bpflbr.NewBPFProgs(engine, progs, false)
assert.NoErr(err, "Failed to get bpf progs: %v")
defer bpfProgs.Close()
tracingTargets := bpfProgs.Tracings()
assert.SliceNotEmpty(tracingTargets, "No bpf progs found")
bpfSpec, err := loadLbr()
assert.NoErr(err, "Failed to load bpf spec: %v")
events, err := ebpf.NewMap(bpfSpec.Maps["events"])
assert.NoErr(err, "Failed to create events map: %v")
defer events.Close()
reusedMaps := map[string]*ebpf.Map{
"events": events,
}
tracings, err := bpflbr.NewBPFTracing(bpfSpec, reusedMaps, tracingTargets)
assert.NoVerifierErr(err, "Failed to trace bpf progs: %v")
defer tracings.Close()
err = bpfProgs.AddProgs(tracings.Progs(), engine)
assert.NoErr(err, "Failed to add bpf progs: %v")
kallsyms, err = bpflbr.NewKallsyms()
assert.NoErr(err, "Failed to reread /proc/kallsyms: %v")
reader, err := ringbuf.NewReader(events)
assert.NoErr(err, "Failed to create ringbuf reader: %v")
defer reader.Close()
w := os.Stdout
if flags.OutputFile() != "" {
f, err := os.OpenFile(flags.OutputFile(), os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0o644)
assert.NoErr(err, "Failed to create output file: %v")
defer f.Close()
w = f
}
log.Print("bpflbr is running..")
ctx, stop := signal.NotifyContext(context.Background(), os.Interrupt, syscall.SIGTERM)
defer stop()
errg, ctx := errgroup.WithContext(ctx)
errg.Go(func() error {
<-ctx.Done()
_ = reader.Close()
return nil
})
errg.Go(func() error {
return bpflbr.Run(reader, bpfProgs, addr2line, kallsyms, w)
})
err = errg.Wait()
assert.NoErr(err, "Failed: %v")
}