| // SPDX-License-Identifier: GPL-2.0 |
| // Copyright (c) 2019 Facebook |
| |
| #include <linux/ptrace.h> |
| #include <linux/bpf.h> |
| #include <bpf/bpf_helpers.h> |
| |
| const struct { |
| unsigned a[4]; |
| /* |
| * if the struct's size is multiple of 16, compiler will put it into |
| * .rodata.cst16 section, which is not recognized by libbpf; work |
| * around this by ensuring we don't have 16-aligned struct |
| */ |
| char _y; |
| } rdonly_values = { .a = {2, 3, 4, 5} }; |
| |
| struct { |
| unsigned did_run; |
| unsigned iters; |
| unsigned sum; |
| } res = {}; |
| |
| SEC("raw_tracepoint/sys_enter:skip_loop") |
| int skip_loop(struct pt_regs *ctx) |
| { |
| /* prevent compiler to optimize everything out */ |
| unsigned * volatile p = (void *)&rdonly_values.a; |
| unsigned iters = 0, sum = 0; |
| |
| /* we should never enter this loop */ |
| while (*p & 1) { |
| iters++; |
| sum += *p; |
| p++; |
| } |
| res.did_run = 1; |
| res.iters = iters; |
| res.sum = sum; |
| return 0; |
| } |
| |
| SEC("raw_tracepoint/sys_enter:part_loop") |
| int part_loop(struct pt_regs *ctx) |
| { |
| /* prevent compiler to optimize everything out */ |
| unsigned * volatile p = (void *)&rdonly_values.a; |
| unsigned iters = 0, sum = 0; |
| |
| /* validate verifier can derive loop termination */ |
| while (*p < 5) { |
| iters++; |
| sum += *p; |
| p++; |
| } |
| res.did_run = 1; |
| res.iters = iters; |
| res.sum = sum; |
| return 0; |
| } |
| |
| SEC("raw_tracepoint/sys_enter:full_loop") |
| int full_loop(struct pt_regs *ctx) |
| { |
| /* prevent compiler to optimize everything out */ |
| unsigned * volatile p = (void *)&rdonly_values.a; |
| int i = sizeof(rdonly_values.a) / sizeof(rdonly_values.a[0]); |
| unsigned iters = 0, sum = 0; |
| |
| /* validate verifier can allow full loop as well */ |
| while (i > 0 ) { |
| iters++; |
| sum += *p; |
| p++; |
| i--; |
| } |
| res.did_run = 1; |
| res.iters = iters; |
| res.sum = sum; |
| return 0; |
| } |
| |
| char _license[] SEC("license") = "GPL"; |