Skip to content

Commit 0e10fd4

Browse files
committed
Merge branch 'Open-coded task_vma iter'
Dave Marchevsky says: ==================== At Meta we have a profiling daemon which periodically collects information on many hosts. This collection usually involves grabbing stacks (user and kernel) using perf_event BPF progs and later symbolicating them. For user stacks we try to use BPF_F_USER_BUILD_ID and rely on remote symbolication, but BPF_F_USER_BUILD_ID doesn't always succeed. In those cases we must fall back to digging around in /proc/PID/maps to map virtual address to (binary, offset). The /proc/PID/maps digging does not occur synchronously with stack collection, so the process might already be gone, in which case it won't have /proc/PID/maps and we will fail to symbolicate. This 'exited process problem' doesn't occur very often as most of the prod services we care to profile are long-lived daemons, but there are enough usecases to warrant a workaround: a BPF program which can be optionally loaded at data collection time and essentially walks /proc/PID/maps. Currently this is done by walking the vma list: struct vm_area_struct* mmap = BPF_CORE_READ(mm, mmap); mmap_next = BPF_CORE_READ(rmap, vm_next); /* in a loop */ Since commit 763ecb0 ("mm: remove the vma linked list") there's no longer a vma linked list to walk. Walking the vma maple tree is not as simple as hopping struct vm_area_struct->vm_next. Luckily, commit f39af05 ("mm: add VMA iterator"), another commit in that series, added struct vma_iterator and for_each_vma macro for easy vma iteration. If similar functionality was exposed to BPF programs, it would be perfect for our usecase. This series adds such functionality, specifically a BPF equivalent of for_each_vma using the open-coded iterator style. Notes: * This approach was chosen after discussion on a previous series [0] which attempted to solve the same problem by adding a BPF_F_VMA_NEXT flag to bpf_find_vma. * Unlike the task_vma bpf_iter, the open-coded iterator kfuncs here do not drop the vma read lock between iterations. See Alexei's response in [0]. * The [vsyscall] page isn't really part of task->mm's vmas, but /proc/PID/maps returns information about it anyways. The vma iter added here does not do the same. See comment on selftest in patch 3. * bpf_iter_task_vma allocates a _data struct which contains - among other things - struct vma_iterator, using BPF allocator and keeps a pointer to the bpf_iter_task_vma_data. This is done in order to prevent changes to struct ma_state - which is wrapped by struct vma_iterator - from necessitating changes to uapi struct bpf_iter_task_vma. Changelog: v6 -> v7: https://lore.kernel.org/bpf/20231010185944.3888849-1-davemarchevsky@fb.com/ Patch numbers correspond to their position in v6 Patch 2 ("selftests/bpf: Rename bpf_iter_task_vma.c to bpf_iter_task_vmas.c") * Add Andrii ack Patch 3 ("bpf: Introduce task_vma open-coded iterator kfuncs") * Add Andrii ack * Add missing __diag_ignore_all for -Wmissing-prototypes (Song) Patch 4 ("selftests/bpf: Add tests for open-coded task_vma iter") * Remove two unnecessary header includes (Andrii) * Remove extraneous !vmas_seen check (Andrii) New Patch ("bpf: Add BPF_KFUNC_{START,END}_defs macros") * After talking to Andrii, this is an attempt to clean up __diag_ignore_all spam everywhere kfuncs are defined. If nontrivial changes are needed, let's apply the other 4 and I'll respin as a standalone patch. v5 -> v6: https://lore.kernel.org/bpf/20231010175637.3405682-1-davemarchevsky@fb.com/ Patch 4 ("selftests/bpf: Add tests for open-coded task_vma iter") * Remove extraneous blank line. I did this manually to the .patch file for v5, which caused BPF CI to complain about failing to apply the series v4 -> v5: https://lore.kernel.org/bpf/20231002195341.2940874-1-davemarchevsky@fb.com/ Patch numbers correspond to their position in v4 New Patch ("selftests/bpf: Rename bpf_iter_task_vma.c to bpf_iter_task_vmas.c") * Patch 2's renaming of this selftest, and associated changes in the userspace runner, are split out into this separate commit (Andrii) Patch 2 ("bpf: Introduce task_vma open-coded iterator kfuncs") * Remove bpf_iter_task_vma kfuncs from libbpf's bpf_helpers.h, they'll be added to selftests' bpf_experimental.h in selftests patch below (Andrii) * Split bpf_iter_task_vma.c renaming into separate commit (Andrii) Patch 3 ("selftests/bpf: Add tests for open-coded task_vma iter") * Add bpf_iter_task_vma kfuncs to bpf_experimental.h (Andrii) * Remove '?' from prog SEC, open_and_load the skel in one operation (Andrii) * Ensure that fclose() always happens in test runner (Andrii) * Use global var w/ 1000 (vm_start, vm_end) structs instead of two MAP_TYPE_ARRAY's w/ 1k u64s each (Andrii) v3 -> v4: https://lore.kernel.org/bpf/20230822050558.2937659-1-davemarchevsky@fb.com/ Patch 1 ("bpf: Don't explicitly emit BTF for struct btf_iter_num") * Add Andrii ack Patch 2 ("bpf: Introduce task_vma open-coded iterator kfuncs") * Mark bpf_iter_task_vma_new args KF_RCU and remove now-unnecessary !task check (Yonghong) * Although KF_RCU is a function-level flag, in reality it only applies to the task_struct *task parameter, as the other two params are a scalar int and a specially-handled KF_ARG_PTR_TO_ITER * Remove struct bpf_iter_task_vma definition from uapi headers, define in kernel/bpf/task_iter.c instead (Andrii) Patch 3 ("selftests/bpf: Add tests for open-coded task_vma iter") * Use a local var when looping over vmas to track map idx. Update vmas_seen global after done iterating. Don't start iterating or update vmas_seen if vmas_seen global is nonzero. (Andrii) * Move getpgid() call to correct spot - above skel detach. (Andrii) v2 -> v3: https://lore.kernel.org/bpf/20230821173415.1970776-1-davemarchevsky@fb.com/ Patch 1 ("bpf: Don't explicitly emit BTF for struct btf_iter_num") * Add Yonghong ack Patch 2 ("bpf: Introduce task_vma open-coded iterator kfuncs") * UAPI bpf header and tools/ version should match * Add bpf_iter_task_vma_kern_data which bpf_iter_task_vma_kern points to, bpf_mem_alloc/free it instead of just vma_iterator. (Alexei) * Inner data ptr == NULL implies initialization failed v1 -> v2: https://lore.kernel.org/bpf/20230810183513.684836-1-davemarchevsky@fb.com/ * Patch 1 * Now removes the unnecessary BTF_TYPE_EMIT instead of changing the type (Yonghong) * Patch 2 * Don't do unnecessary BTF_TYPE_EMIT (Yonghong) * Bump task refcount to prevent ->mm reuse (Yonghong) * Keep a pointer to vma_iterator in bpf_iter_task_vma, alloc/free via BPF mem allocator (Yonghong, Stanislav) * Patch 3 [0]: https://lore.kernel.org/bpf/20230801145414.418145-1-davemarchevsky@fb.com/ ==================== Signed-off-by: Andrii Nakryiko <andrii@kernel.org>
2 parents ba8ea72 + e0e1a7a commit 0e10fd4

File tree

8 files changed

+216
-15
lines changed

8 files changed

+216
-15
lines changed

kernel/bpf/bpf_iter.c

Lines changed: 0 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -793,8 +793,6 @@ __bpf_kfunc int bpf_iter_num_new(struct bpf_iter_num *it, int start, int end)
793793
BUILD_BUG_ON(sizeof(struct bpf_iter_num_kern) != sizeof(struct bpf_iter_num));
794794
BUILD_BUG_ON(__alignof__(struct bpf_iter_num_kern) != __alignof__(struct bpf_iter_num));
795795

796-
BTF_TYPE_EMIT(struct btf_iter_num);
797-
798796
/* start == end is legit, it's an empty range and we'll just get NULL
799797
* on first (and any subsequent) bpf_iter_num_next() call
800798
*/

kernel/bpf/helpers.c

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2552,6 +2552,9 @@ BTF_ID_FLAGS(func, bpf_dynptr_slice_rdwr, KF_RET_NULL)
25522552
BTF_ID_FLAGS(func, bpf_iter_num_new, KF_ITER_NEW)
25532553
BTF_ID_FLAGS(func, bpf_iter_num_next, KF_ITER_NEXT | KF_RET_NULL)
25542554
BTF_ID_FLAGS(func, bpf_iter_num_destroy, KF_ITER_DESTROY)
2555+
BTF_ID_FLAGS(func, bpf_iter_task_vma_new, KF_ITER_NEW | KF_RCU)
2556+
BTF_ID_FLAGS(func, bpf_iter_task_vma_next, KF_ITER_NEXT | KF_RET_NULL)
2557+
BTF_ID_FLAGS(func, bpf_iter_task_vma_destroy, KF_ITER_DESTROY)
25552558
BTF_ID_FLAGS(func, bpf_dynptr_adjust)
25562559
BTF_ID_FLAGS(func, bpf_dynptr_is_null)
25572560
BTF_ID_FLAGS(func, bpf_dynptr_is_rdonly)

kernel/bpf/task_iter.c

Lines changed: 91 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,9 @@
77
#include <linux/fs.h>
88
#include <linux/fdtable.h>
99
#include <linux/filter.h>
10+
#include <linux/bpf_mem_alloc.h>
1011
#include <linux/btf_ids.h>
12+
#include <linux/mm_types.h>
1113
#include "mmap_unlock_work.h"
1214

1315
static const char * const iter_task_type_names[] = {
@@ -803,6 +805,95 @@ const struct bpf_func_proto bpf_find_vma_proto = {
803805
.arg5_type = ARG_ANYTHING,
804806
};
805807

808+
struct bpf_iter_task_vma_kern_data {
809+
struct task_struct *task;
810+
struct mm_struct *mm;
811+
struct mmap_unlock_irq_work *work;
812+
struct vma_iterator vmi;
813+
};
814+
815+
struct bpf_iter_task_vma {
816+
/* opaque iterator state; having __u64 here allows to preserve correct
817+
* alignment requirements in vmlinux.h, generated from BTF
818+
*/
819+
__u64 __opaque[1];
820+
} __attribute__((aligned(8)));
821+
822+
/* Non-opaque version of bpf_iter_task_vma */
823+
struct bpf_iter_task_vma_kern {
824+
struct bpf_iter_task_vma_kern_data *data;
825+
} __attribute__((aligned(8)));
826+
827+
__diag_push();
828+
__diag_ignore_all("-Wmissing-prototypes",
829+
"Global functions as their definitions will be in vmlinux BTF");
830+
831+
__bpf_kfunc int bpf_iter_task_vma_new(struct bpf_iter_task_vma *it,
832+
struct task_struct *task, u64 addr)
833+
{
834+
struct bpf_iter_task_vma_kern *kit = (void *)it;
835+
bool irq_work_busy = false;
836+
int err;
837+
838+
BUILD_BUG_ON(sizeof(struct bpf_iter_task_vma_kern) != sizeof(struct bpf_iter_task_vma));
839+
BUILD_BUG_ON(__alignof__(struct bpf_iter_task_vma_kern) != __alignof__(struct bpf_iter_task_vma));
840+
841+
/* is_iter_reg_valid_uninit guarantees that kit hasn't been initialized
842+
* before, so non-NULL kit->data doesn't point to previously
843+
* bpf_mem_alloc'd bpf_iter_task_vma_kern_data
844+
*/
845+
kit->data = bpf_mem_alloc(&bpf_global_ma, sizeof(struct bpf_iter_task_vma_kern_data));
846+
if (!kit->data)
847+
return -ENOMEM;
848+
849+
kit->data->task = get_task_struct(task);
850+
kit->data->mm = task->mm;
851+
if (!kit->data->mm) {
852+
err = -ENOENT;
853+
goto err_cleanup_iter;
854+
}
855+
856+
/* kit->data->work == NULL is valid after bpf_mmap_unlock_get_irq_work */
857+
irq_work_busy = bpf_mmap_unlock_get_irq_work(&kit->data->work);
858+
if (irq_work_busy || !mmap_read_trylock(kit->data->mm)) {
859+
err = -EBUSY;
860+
goto err_cleanup_iter;
861+
}
862+
863+
vma_iter_init(&kit->data->vmi, kit->data->mm, addr);
864+
return 0;
865+
866+
err_cleanup_iter:
867+
if (kit->data->task)
868+
put_task_struct(kit->data->task);
869+
bpf_mem_free(&bpf_global_ma, kit->data);
870+
/* NULL kit->data signals failed bpf_iter_task_vma initialization */
871+
kit->data = NULL;
872+
return err;
873+
}
874+
875+
__bpf_kfunc struct vm_area_struct *bpf_iter_task_vma_next(struct bpf_iter_task_vma *it)
876+
{
877+
struct bpf_iter_task_vma_kern *kit = (void *)it;
878+
879+
if (!kit->data) /* bpf_iter_task_vma_new failed */
880+
return NULL;
881+
return vma_next(&kit->data->vmi);
882+
}
883+
884+
__bpf_kfunc void bpf_iter_task_vma_destroy(struct bpf_iter_task_vma *it)
885+
{
886+
struct bpf_iter_task_vma_kern *kit = (void *)it;
887+
888+
if (kit->data) {
889+
bpf_mmap_unlock_mm(kit->data->work, kit->data->mm);
890+
put_task_struct(kit->data->task);
891+
bpf_mem_free(&bpf_global_ma, kit->data);
892+
}
893+
}
894+
895+
__diag_pop();
896+
806897
DEFINE_PER_CPU(struct mmap_unlock_irq_work, mmap_unlock_work);
807898

808899
static void do_mmap_read_unlock(struct irq_work *entry)

tools/testing/selftests/bpf/bpf_experimental.h

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,14 @@ extern void *bpf_percpu_obj_new_impl(__u64 local_type_id, void *meta) __ksym;
159159
*/
160160
extern void bpf_percpu_obj_drop_impl(void *kptr, void *meta) __ksym;
161161

162+
struct bpf_iter_task_vma;
163+
164+
extern int bpf_iter_task_vma_new(struct bpf_iter_task_vma *it,
165+
struct task_struct *task,
166+
unsigned long addr) __ksym;
167+
extern struct vm_area_struct *bpf_iter_task_vma_next(struct bpf_iter_task_vma *it) __ksym;
168+
extern void bpf_iter_task_vma_destroy(struct bpf_iter_task_vma *it) __ksym;
169+
162170
/* Convenience macro to wrap over bpf_obj_drop_impl */
163171
#define bpf_percpu_obj_drop(kptr) bpf_percpu_obj_drop_impl(kptr, NULL)
164172

tools/testing/selftests/bpf/prog_tests/bpf_iter.c

Lines changed: 13 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
#include "bpf_iter_task.skel.h"
1111
#include "bpf_iter_task_stack.skel.h"
1212
#include "bpf_iter_task_file.skel.h"
13-
#include "bpf_iter_task_vma.skel.h"
13+
#include "bpf_iter_task_vmas.skel.h"
1414
#include "bpf_iter_task_btf.skel.h"
1515
#include "bpf_iter_tcp4.skel.h"
1616
#include "bpf_iter_tcp6.skel.h"
@@ -1399,19 +1399,19 @@ static void str_strip_first_line(char *str)
13991399
static void test_task_vma_common(struct bpf_iter_attach_opts *opts)
14001400
{
14011401
int err, iter_fd = -1, proc_maps_fd = -1;
1402-
struct bpf_iter_task_vma *skel;
1402+
struct bpf_iter_task_vmas *skel;
14031403
int len, read_size = 4;
14041404
char maps_path[64];
14051405

1406-
skel = bpf_iter_task_vma__open();
1407-
if (!ASSERT_OK_PTR(skel, "bpf_iter_task_vma__open"))
1406+
skel = bpf_iter_task_vmas__open();
1407+
if (!ASSERT_OK_PTR(skel, "bpf_iter_task_vmas__open"))
14081408
return;
14091409

14101410
skel->bss->pid = getpid();
14111411
skel->bss->one_task = opts ? 1 : 0;
14121412

1413-
err = bpf_iter_task_vma__load(skel);
1414-
if (!ASSERT_OK(err, "bpf_iter_task_vma__load"))
1413+
err = bpf_iter_task_vmas__load(skel);
1414+
if (!ASSERT_OK(err, "bpf_iter_task_vmas__load"))
14151415
goto out;
14161416

14171417
skel->links.proc_maps = bpf_program__attach_iter(
@@ -1462,25 +1462,25 @@ static void test_task_vma_common(struct bpf_iter_attach_opts *opts)
14621462
out:
14631463
close(proc_maps_fd);
14641464
close(iter_fd);
1465-
bpf_iter_task_vma__destroy(skel);
1465+
bpf_iter_task_vmas__destroy(skel);
14661466
}
14671467

14681468
static void test_task_vma_dead_task(void)
14691469
{
1470-
struct bpf_iter_task_vma *skel;
1470+
struct bpf_iter_task_vmas *skel;
14711471
int wstatus, child_pid = -1;
14721472
time_t start_tm, cur_tm;
14731473
int err, iter_fd = -1;
14741474
int wait_sec = 3;
14751475

1476-
skel = bpf_iter_task_vma__open();
1477-
if (!ASSERT_OK_PTR(skel, "bpf_iter_task_vma__open"))
1476+
skel = bpf_iter_task_vmas__open();
1477+
if (!ASSERT_OK_PTR(skel, "bpf_iter_task_vmas__open"))
14781478
return;
14791479

14801480
skel->bss->pid = getpid();
14811481

1482-
err = bpf_iter_task_vma__load(skel);
1483-
if (!ASSERT_OK(err, "bpf_iter_task_vma__load"))
1482+
err = bpf_iter_task_vmas__load(skel);
1483+
if (!ASSERT_OK(err, "bpf_iter_task_vmas__load"))
14841484
goto out;
14851485

14861486
skel->links.proc_maps = bpf_program__attach_iter(
@@ -1533,7 +1533,7 @@ static void test_task_vma_dead_task(void)
15331533
out:
15341534
waitpid(child_pid, &wstatus, 0);
15351535
close(iter_fd);
1536-
bpf_iter_task_vma__destroy(skel);
1536+
bpf_iter_task_vmas__destroy(skel);
15371537
}
15381538

15391539
void test_bpf_sockmap_map_iter_fd(void)

tools/testing/selftests/bpf/prog_tests/iters.c

Lines changed: 58 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
#include "iters_looping.skel.h"
99
#include "iters_num.skel.h"
1010
#include "iters_testmod_seq.skel.h"
11+
#include "iters_task_vma.skel.h"
1112

1213
static void subtest_num_iters(void)
1314
{
@@ -90,6 +91,61 @@ static void subtest_testmod_seq_iters(void)
9091
iters_testmod_seq__destroy(skel);
9192
}
9293

94+
static void subtest_task_vma_iters(void)
95+
{
96+
unsigned long start, end, bpf_iter_start, bpf_iter_end;
97+
struct iters_task_vma *skel;
98+
char rest_of_line[1000];
99+
unsigned int seen;
100+
FILE *f = NULL;
101+
int err;
102+
103+
skel = iters_task_vma__open_and_load();
104+
if (!ASSERT_OK_PTR(skel, "skel_open_and_load"))
105+
return;
106+
107+
skel->bss->target_pid = getpid();
108+
109+
err = iters_task_vma__attach(skel);
110+
if (!ASSERT_OK(err, "skel_attach"))
111+
goto cleanup;
112+
113+
getpgid(skel->bss->target_pid);
114+
iters_task_vma__detach(skel);
115+
116+
if (!ASSERT_GT(skel->bss->vmas_seen, 0, "vmas_seen_gt_zero"))
117+
goto cleanup;
118+
119+
f = fopen("/proc/self/maps", "r");
120+
if (!ASSERT_OK_PTR(f, "proc_maps_fopen"))
121+
goto cleanup;
122+
123+
seen = 0;
124+
while (fscanf(f, "%lx-%lx %[^\n]\n", &start, &end, rest_of_line) == 3) {
125+
/* [vsyscall] vma isn't _really_ part of task->mm vmas.
126+
* /proc/PID/maps returns it when out of vmas - see get_gate_vma
127+
* calls in fs/proc/task_mmu.c
128+
*/
129+
if (strstr(rest_of_line, "[vsyscall]"))
130+
continue;
131+
132+
bpf_iter_start = skel->bss->vm_ranges[seen].vm_start;
133+
bpf_iter_end = skel->bss->vm_ranges[seen].vm_end;
134+
135+
ASSERT_EQ(bpf_iter_start, start, "vma->vm_start match");
136+
ASSERT_EQ(bpf_iter_end, end, "vma->vm_end match");
137+
seen++;
138+
}
139+
140+
if (!ASSERT_EQ(skel->bss->vmas_seen, seen, "vmas_seen_eq"))
141+
goto cleanup;
142+
143+
cleanup:
144+
if (f)
145+
fclose(f);
146+
iters_task_vma__destroy(skel);
147+
}
148+
93149
void test_iters(void)
94150
{
95151
RUN_TESTS(iters_state_safety);
@@ -103,4 +159,6 @@ void test_iters(void)
103159
subtest_num_iters();
104160
if (test__start_subtest("testmod_seq"))
105161
subtest_testmod_seq_iters();
162+
if (test__start_subtest("task_vma"))
163+
subtest_task_vma_iters();
106164
}
Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
// SPDX-License-Identifier: GPL-2.0
2+
/* Copyright (c) 2023 Meta Platforms, Inc. and affiliates. */
3+
4+
#include "vmlinux.h"
5+
#include "bpf_experimental.h"
6+
#include <bpf/bpf_helpers.h>
7+
#include "bpf_misc.h"
8+
9+
pid_t target_pid = 0;
10+
unsigned int vmas_seen = 0;
11+
12+
struct {
13+
__u64 vm_start;
14+
__u64 vm_end;
15+
} vm_ranges[1000];
16+
17+
SEC("raw_tp/sys_enter")
18+
int iter_task_vma_for_each(const void *ctx)
19+
{
20+
struct task_struct *task = bpf_get_current_task_btf();
21+
struct vm_area_struct *vma;
22+
unsigned int seen = 0;
23+
24+
if (task->pid != target_pid)
25+
return 0;
26+
27+
if (vmas_seen)
28+
return 0;
29+
30+
bpf_for_each(task_vma, vma, task, 0) {
31+
if (seen >= 1000)
32+
break;
33+
34+
vm_ranges[seen].vm_start = vma->vm_start;
35+
vm_ranges[seen].vm_end = vma->vm_end;
36+
seen++;
37+
}
38+
39+
vmas_seen = seen;
40+
return 0;
41+
}
42+
43+
char _license[] SEC("license") = "GPL";

0 commit comments

Comments
 (0)