From 20cb93578240ef4dc88d1212c3baa81111709c40 Mon Sep 17 00:00:00 2001 From: Md Tahsin Rahman Date: Wed, 7 Aug 2019 17:09:50 +0600 Subject: [PATCH 1/3] Add raw_tracepoint support --- elf/elf.go | 22 ++++++++++++++-- elf/module.go | 70 ++++++++++++++++++++++++++++++++++++++++----------- 2 files changed, 75 insertions(+), 17 deletions(-) diff --git a/elf/elf.go b/elf/elf.go index f9e0c7ee..9a888112 100644 --- a/elf/elf.go +++ b/elf/elf.go @@ -557,6 +557,7 @@ func (b *Module) Load(parameters map[string]SectionParams) error { isCgroupSock := strings.HasPrefix(secName, "cgroup/sock") isSocketFilter := strings.HasPrefix(secName, "socket") isTracepoint := strings.HasPrefix(secName, "tracepoint/") + isRawTracepoint := strings.HasPrefix(secName, "raw_tracepoint/") isSchedCls := strings.HasPrefix(secName, "sched_cls/") isSchedAct := strings.HasPrefix(secName, "sched_act/") @@ -579,6 +580,8 @@ func (b *Module) Load(parameters map[string]SectionParams) error { progType = uint32(C.BPF_PROG_TYPE_SOCKET_FILTER) case isTracepoint: progType = uint32(C.BPF_PROG_TYPE_TRACEPOINT) + case isRawTracepoint: + progType = uint32(C.BPF_PROG_TYPE_RAW_TRACEPOINT) case isSchedCls: progType = uint32(C.BPF_PROG_TYPE_SCHED_CLS) case isSchedAct: @@ -598,7 +601,7 @@ func (b *Module) Load(parameters map[string]SectionParams) error { } } - if isKprobe || isKretprobe || isUprobe || isUretprobe || isCgroupSkb || isCgroupSock || isSocketFilter || isTracepoint || isSchedCls || isSchedAct { + if isKprobe || isKretprobe || isUprobe || isUretprobe || isCgroupSkb || isCgroupSock || isSocketFilter || isTracepoint || isRawTracepoint || isSchedCls || isSchedAct { rdata, err := rsection.Data() if err != nil { return err @@ -662,6 +665,12 @@ func (b *Module) Load(parameters map[string]SectionParams) error { insns: insns, fd: int(progFd), } + case isRawTracepoint: + b.rawTracepointPrograms[secName] = &RawTracepointProgram{ + Name: secName, + insns: insns, + fd: int(progFd), + } case isSchedCls: fallthrough case isSchedAct: @@ -690,6 +699,7 @@ func (b *Module) Load(parameters map[string]SectionParams) error { isCgroupSock := strings.HasPrefix(secName, "cgroup/sock") isSocketFilter := strings.HasPrefix(secName, "socket") isTracepoint := strings.HasPrefix(secName, "tracepoint/") + isRawTracepoint := strings.HasPrefix(secName, "raw_tracepoint/") isSchedCls := strings.HasPrefix(secName, "sched_cls/") isSchedAct := strings.HasPrefix(secName, "sched_act/") @@ -711,6 +721,8 @@ func (b *Module) Load(parameters map[string]SectionParams) error { progType = uint32(C.BPF_PROG_TYPE_SOCKET_FILTER) case isTracepoint: progType = uint32(C.BPF_PROG_TYPE_TRACEPOINT) + case isRawTracepoint: + progType = uint32(C.BPF_PROG_TYPE_RAW_TRACEPOINT) case isSchedCls: progType = uint32(C.BPF_PROG_TYPE_SCHED_CLS) case isSchedAct: @@ -730,7 +742,7 @@ func (b *Module) Load(parameters map[string]SectionParams) error { } } - if isKprobe || isKretprobe || isUprobe || isUretprobe || isCgroupSkb || isCgroupSock || isSocketFilter || isTracepoint || isSchedCls || isSchedAct { + if isKprobe || isKretprobe || isUprobe || isUretprobe || isCgroupSkb || isCgroupSock || isSocketFilter || isTracepoint || isRawTracepoint || isSchedCls || isSchedAct { data, err := section.Data() if err != nil { return err @@ -789,6 +801,12 @@ func (b *Module) Load(parameters map[string]SectionParams) error { insns: insns, fd: int(progFd), } + case isRawTracepoint: + b.rawTracepointPrograms[secName] = &RawTracepointProgram{ + Name: secName, + insns: insns, + fd: int(progFd), + } case isSchedCls: fallthrough case isSchedAct: diff --git a/elf/module.go b/elf/module.go index 12cb1694..ecf7a2d9 100644 --- a/elf/module.go +++ b/elf/module.go @@ -88,6 +88,17 @@ int bpf_detach_socket(int sock, int fd) { return setsockopt(sock, SOL_SOCKET, SO_DETACH_BPF, &fd, sizeof(fd)); } + +int bpf_attach_raw_tracepoint(int prog_fd, char *tp_name) +{ + union bpf_attr attr; + + bzero(&attr, sizeof(attr)); + attr.raw_tracepoint.name = ptr_to_u64(tp_name); + attr.raw_tracepoint.prog_fd = prog_fd; + + return syscall(__NR_bpf, BPF_RAW_TRACEPOINT_OPEN, &attr, sizeof(attr)); +} */ import "C" @@ -96,14 +107,15 @@ type Module struct { fileReader io.ReaderAt file *elf.File - log []byte - maps map[string]*Map - probes map[string]*Kprobe - uprobes map[string]*Uprobe - cgroupPrograms map[string]*CgroupProgram - socketFilters map[string]*SocketFilter - tracepointPrograms map[string]*TracepointProgram - schedPrograms map[string]*SchedProgram + log []byte + maps map[string]*Map + probes map[string]*Kprobe + uprobes map[string]*Uprobe + cgroupPrograms map[string]*CgroupProgram + socketFilters map[string]*SocketFilter + tracepointPrograms map[string]*TracepointProgram + rawTracepointPrograms map[string]*RawTracepointProgram + schedPrograms map[string]*SchedProgram compatProbe bool // try to be automatically convert function names depending on kernel versions (SyS_ and __x64_sys_) } @@ -154,6 +166,14 @@ type TracepointProgram struct { efd int } +// RawTracepointProgram represents a tracepoint program +type RawTracepointProgram struct { + Name string + insns *C.struct_bpf_insn + fd int + efd int +} + // SchedProgram represents a traffic classifier program type SchedProgram struct { Name string @@ -163,13 +183,14 @@ type SchedProgram struct { func newModule() *Module { return &Module{ - probes: make(map[string]*Kprobe), - uprobes: make(map[string]*Uprobe), - cgroupPrograms: make(map[string]*CgroupProgram), - socketFilters: make(map[string]*SocketFilter), - tracepointPrograms: make(map[string]*TracepointProgram), - schedPrograms: make(map[string]*SchedProgram), - log: make([]byte, 524288), + probes: make(map[string]*Kprobe), + uprobes: make(map[string]*Uprobe), + cgroupPrograms: make(map[string]*CgroupProgram), + socketFilters: make(map[string]*SocketFilter), + tracepointPrograms: make(map[string]*TracepointProgram), + rawTracepointPrograms: make(map[string]*RawTracepointProgram), + schedPrograms: make(map[string]*SchedProgram), + log: make([]byte, 524288), } } @@ -352,6 +373,25 @@ func (b *Module) EnableTracepoint(secName string) error { return err } +func (b *Module) AttachRawTracepoint(secName string) error { + prog, ok := b.rawTracepointPrograms[secName] + if !ok { + return fmt.Errorf("no such raw tracepoint program %q", secName) + } + progFd := prog.fd + + name := strings.Split(secName, "raw_tracepoint/") + + tpNameCS := C.CString(name[1]) + res, err := C.bpf_attach_raw_tracepoint(C.int(progFd), tpNameCS) + C.free(unsafe.Pointer(tpNameCS)) + + if res < 0 { + return fmt.Errorf("failed to attach raw tracepoint: %v", err) + } + return nil +} + // IterKprobes returns a channel that emits the kprobes that included in the // module. func (b *Module) IterKprobes() <-chan *Kprobe { From ba7f1266be4146590ce4ab8fbc0885fe5136b23e Mon Sep 17 00:00:00 2001 From: Tahsin Date: Thu, 8 Aug 2019 00:15:37 +0600 Subject: [PATCH 2/3] Remove unused efd, add close method --- elf/module.go | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/elf/module.go b/elf/module.go index ecf7a2d9..4d19d603 100644 --- a/elf/module.go +++ b/elf/module.go @@ -171,7 +171,6 @@ type RawTracepointProgram struct { Name string insns *C.struct_bpf_insn fd int - efd int } // SchedProgram represents a traffic classifier program @@ -705,6 +704,15 @@ func (b *Module) closeTracepointPrograms() error { return nil } +func (b *Module) closeRawTracepointPrograms() error { + for _, program := range b.rawTracepointPrograms { + if err := syscall.Close(program.fd); err != nil { + return fmt.Errorf("error closing raw tracepoint program fd: %v", err) + } + } + return nil +} + func (b *Module) closeCgroupPrograms() error { for _, program := range b.cgroupPrograms { if err := syscall.Close(program.fd); err != nil { @@ -805,6 +813,9 @@ func (b *Module) CloseExt(options map[string]CloseOptions) error { if err := b.closeTracepointPrograms(); err != nil { return err } + if err := b.closeRawTracepointPrograms(); err != nil { + return err + } if err := b.closeSocketFilters(); err != nil { return err } From 0faedcbdf3f5c60231aaab31d6b4f712f7919012 Mon Sep 17 00:00:00 2001 From: Md Tahsin Rahman Date: Thu, 8 Aug 2019 15:42:11 +0600 Subject: [PATCH 3/3] Fix warning implicit declaration of ptr_to_u64 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit ../../iovisor/gobpf/elf/module.go: In function ‘bpf_attach_raw_tracepoint’: ../../iovisor/gobpf/elf/module.go:97:29: warning: implicit declaration of function ‘ptr_to_u64’; did you mean ‘strtoull’? [-Wimplicit-function-declaration] attr.raw_tracepoint.name = ptr_to_u64(tp_name); ^~~~~~~~~~ strtoull --- elf/module.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/elf/module.go b/elf/module.go index 4d19d603..8c8aa171 100644 --- a/elf/module.go +++ b/elf/module.go @@ -94,7 +94,7 @@ int bpf_attach_raw_tracepoint(int prog_fd, char *tp_name) union bpf_attr attr; bzero(&attr, sizeof(attr)); - attr.raw_tracepoint.name = ptr_to_u64(tp_name); + attr.raw_tracepoint.name = (__u64) tp_name; attr.raw_tracepoint.prog_fd = prog_fd; return syscall(__NR_bpf, BPF_RAW_TRACEPOINT_OPEN, &attr, sizeof(attr));