forked from aquasecurity/libbpfgo
-
Notifications
You must be signed in to change notification settings - Fork 0
/
buf-perf.go
112 lines (94 loc) · 2.06 KB
/
buf-perf.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
package libbpfgo
/*
#cgo LDFLAGS: -lelf -lz
#include "libbpfgo.h"
*/
import "C"
import (
"fmt"
"sync"
"syscall"
)
//
// PerfBuffer
//
type PerfBuffer struct {
pb *C.struct_perf_buffer
bpfMap *BPFMap
slot uint
eventsChan chan []byte
lostChan chan uint64
stop chan struct{}
closed bool
wg sync.WaitGroup
}
// Poll will wait until timeout in milliseconds to gather
// data from the perf buffer.
func (pb *PerfBuffer) Poll(timeout int) {
pb.stop = make(chan struct{})
pb.wg.Add(1)
go pb.poll(timeout)
}
// Deprecated: use PerfBuffer.Poll() instead.
func (pb *PerfBuffer) Start() {
pb.Poll(300)
}
func (pb *PerfBuffer) Stop() {
if pb.stop == nil {
return
}
// Signal the poll goroutine to exit
close(pb.stop)
// The event and lost channels should be drained here since the consumer
// may have stopped at this point. Failure to drain it will
// result in a deadlock: the channel will fill up and the poll
// goroutine will block in the callback.
go func() {
// revive:disable:empty-block
for range pb.eventsChan {
}
if pb.lostChan != nil {
for range pb.lostChan {
}
}
// revive:enable:empty-block
}()
// Wait for the poll goroutine to exit
pb.wg.Wait()
// Close the channel -- this is useful for the consumer but
// also to terminate the drain goroutine above.
close(pb.eventsChan)
if pb.lostChan != nil {
close(pb.lostChan)
}
// Reset pb.stop to allow multiple safe calls to Stop()
pb.stop = nil
}
func (pb *PerfBuffer) Close() {
if pb.closed {
return
}
pb.Stop()
C.perf_buffer__free(pb.pb)
eventChannels.remove(pb.slot)
pb.closed = true
}
// todo: consider writing the perf polling in go as c to go calls (callback) are expensive
func (pb *PerfBuffer) poll(timeout int) error {
defer pb.wg.Done()
for {
select {
case <-pb.stop:
return nil
default:
retC := C.perf_buffer__poll(pb.pb, C.int(timeout))
if retC < 0 {
errno := syscall.Errno(-retC)
if errno == syscall.EINTR {
continue
}
return fmt.Errorf("error polling perf buffer: %w", errno)
}
}
}
}