Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
1
fork

Configure Feed

Select the types of activity you want to include in your feed.

at master 95 lines 2.1 kB view raw
1// SPDX-License-Identifier: GPL-2.0 2 3#include <vmlinux.h> 4#include <bpf/bpf_core_read.h> 5#include <bpf/bpf_helpers.h> 6#include <bpf/bpf_tracing.h> 7#include "bpf_misc.h" 8 9char _license[] SEC("license") = "GPL"; 10 11/* typically virtio scsi has max SGs of 6 */ 12#define VIRTIO_MAX_SGS 6 13 14/* Verifier will fail with SG_MAX = 128. The failure can be 15 * workarounded with a smaller SG_MAX, e.g. 10. 16 */ 17#define WORKAROUND 18#ifdef WORKAROUND 19#define SG_MAX 10 20#else 21/* typically virtio blk has max SEG of 128 */ 22#define SG_MAX 128 23#endif 24 25#define SG_CHAIN 0x01UL 26#define SG_END 0x02UL 27 28#define sg_is_chain(sg) ((sg)->page_link & SG_CHAIN) 29#define sg_is_last(sg) ((sg)->page_link & SG_END) 30#define sg_chain_ptr(sg) \ 31 ((struct scatterlist *) ((sg)->page_link & ~(SG_CHAIN | SG_END))) 32 33static inline struct scatterlist *__sg_next(struct scatterlist *sgp) 34{ 35 struct scatterlist sg; 36 37 bpf_probe_read_kernel(&sg, sizeof(sg), sgp); 38 if (sg_is_last(&sg)) 39 return NULL; 40 41 sgp++; 42 43 bpf_probe_read_kernel(&sg, sizeof(sg), sgp); 44 if (sg_is_chain(&sg)) 45 sgp = sg_chain_ptr(&sg); 46 47 return sgp; 48} 49 50static inline struct scatterlist *get_sgp(struct scatterlist **sgs, int i) 51{ 52 struct scatterlist *sgp; 53 54 bpf_probe_read_kernel(&sgp, sizeof(sgp), sgs + i); 55 return sgp; 56} 57 58int run_once = 0; 59int result = 0; 60 61SEC("kprobe/virtqueue_add_sgs") 62int BPF_KPROBE(trace_virtqueue_add_sgs, void *unused, struct scatterlist **sgs, 63 unsigned int out_sgs, unsigned int in_sgs) 64{ 65 struct scatterlist *sgp = NULL; 66 __u64 length1 = 0, length2 = 0; 67 unsigned int i, n, len; 68 69 if (run_once != 0) 70 return 0; 71 72 for (i = 0; (i < VIRTIO_MAX_SGS) && (i < out_sgs); i++) { 73 __sink(out_sgs); 74 for (n = 0, sgp = get_sgp(sgs, i); sgp && (n < SG_MAX); 75 sgp = __sg_next(sgp)) { 76 len = BPF_CORE_READ(sgp, length); 77 length1 += len; 78 n++; 79 } 80 } 81 82 for (i = 0; (i < VIRTIO_MAX_SGS) && (i < in_sgs); i++) { 83 __sink(in_sgs); 84 for (n = 0, sgp = get_sgp(sgs, i); sgp && (n < SG_MAX); 85 sgp = __sg_next(sgp)) { 86 len = BPF_CORE_READ(sgp, length); 87 length2 += len; 88 n++; 89 } 90 } 91 92 run_once = 1; 93 result = length2 - length1; 94 return 0; 95}