mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 280b058d48
			
		
	
	
		280b058d48
		
	
	
	
	
		
			
			Also monitor the tracepoint xdp_exception. This tracepoint is usually invoked by the drivers. Programs themselves can activate this by returning XDP_ABORTED, which will drop the packet but also trigger the tracepoint. This is useful for distinguishing intentional (XDP_DROP) vs. ebpf-program error cases that cased a drop (XDP_ABORTED). Drivers also use this tracepoint for reporting on XDP actions that are unknown to the specific driver. This can help the user to detect if a driver e.g. doesn't implement XDP_REDIRECT yet. Signed-off-by: Jesper Dangaard Brouer <brouer@redhat.com> Signed-off-by: David S. Miller <davem@davemloft.net>
		
			
				
	
	
		
			121 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* XDP monitor tool, based on tracepoints
 | |
|  *
 | |
|  *  Copyright(c) 2017 Jesper Dangaard Brouer, Red Hat Inc.
 | |
|  */
 | |
| #include <uapi/linux/bpf.h>
 | |
| #include "bpf_helpers.h"
 | |
| 
 | |
| struct bpf_map_def SEC("maps") redirect_err_cnt = {
 | |
| 	.type = BPF_MAP_TYPE_PERCPU_ARRAY,
 | |
| 	.key_size = sizeof(u32),
 | |
| 	.value_size = sizeof(u64),
 | |
| 	.max_entries = 2,
 | |
| 	/* TODO: have entries for all possible errno's */
 | |
| };
 | |
| 
 | |
| #define XDP_UNKNOWN	XDP_REDIRECT + 1
 | |
| struct bpf_map_def SEC("maps") exception_cnt = {
 | |
| 	.type		= BPF_MAP_TYPE_PERCPU_ARRAY,
 | |
| 	.key_size	= sizeof(u32),
 | |
| 	.value_size	= sizeof(u64),
 | |
| 	.max_entries	= XDP_UNKNOWN + 1,
 | |
| };
 | |
| 
 | |
| /* Tracepoint format: /sys/kernel/debug/tracing/events/xdp/xdp_redirect/format
 | |
|  * Code in:                kernel/include/trace/events/xdp.h
 | |
|  */
 | |
| struct xdp_redirect_ctx {
 | |
| 	u64 __pad;		// First 8 bytes are not accessible by bpf code
 | |
| 	int prog_id;		//	offset:8;  size:4; signed:1;
 | |
| 	u32 act;		//	offset:12  size:4; signed:0;
 | |
| 	int ifindex;		//	offset:16  size:4; signed:1;
 | |
| 	int err;		//	offset:20  size:4; signed:1;
 | |
| 	int to_ifindex;		//	offset:24  size:4; signed:1;
 | |
| 	u32 map_id;		//	offset:28  size:4; signed:0;
 | |
| 	int map_index;		//	offset:32  size:4; signed:1;
 | |
| };				//	offset:36
 | |
| 
 | |
| enum {
 | |
| 	XDP_REDIRECT_SUCCESS = 0,
 | |
| 	XDP_REDIRECT_ERROR = 1
 | |
| };
 | |
| 
 | |
| static __always_inline
 | |
| int xdp_redirect_collect_stat(struct xdp_redirect_ctx *ctx)
 | |
| {
 | |
| 	u32 key = XDP_REDIRECT_ERROR;
 | |
| 	int err = ctx->err;
 | |
| 	u64 *cnt;
 | |
| 
 | |
| 	if (!err)
 | |
| 		key = XDP_REDIRECT_SUCCESS;
 | |
| 
 | |
| 	cnt  = bpf_map_lookup_elem(&redirect_err_cnt, &key);
 | |
| 	if (!cnt)
 | |
| 		return 1;
 | |
| 	*cnt += 1;
 | |
| 
 | |
| 	return 0; /* Indicate event was filtered (no further processing)*/
 | |
| 	/*
 | |
| 	 * Returning 1 here would allow e.g. a perf-record tracepoint
 | |
| 	 * to see and record these events, but it doesn't work well
 | |
| 	 * in-practice as stopping perf-record also unload this
 | |
| 	 * bpf_prog.  Plus, there is additional overhead of doing so.
 | |
| 	 */
 | |
| }
 | |
| 
 | |
| SEC("tracepoint/xdp/xdp_redirect_err")
 | |
| int trace_xdp_redirect_err(struct xdp_redirect_ctx *ctx)
 | |
| {
 | |
| 	return xdp_redirect_collect_stat(ctx);
 | |
| }
 | |
| 
 | |
| 
 | |
| SEC("tracepoint/xdp/xdp_redirect_map_err")
 | |
| int trace_xdp_redirect_map_err(struct xdp_redirect_ctx *ctx)
 | |
| {
 | |
| 	return xdp_redirect_collect_stat(ctx);
 | |
| }
 | |
| 
 | |
| /* Likely unloaded when prog starts */
 | |
| SEC("tracepoint/xdp/xdp_redirect")
 | |
| int trace_xdp_redirect(struct xdp_redirect_ctx *ctx)
 | |
| {
 | |
| 	return xdp_redirect_collect_stat(ctx);
 | |
| }
 | |
| 
 | |
| /* Likely unloaded when prog starts */
 | |
| SEC("tracepoint/xdp/xdp_redirect_map")
 | |
| int trace_xdp_redirect_map(struct xdp_redirect_ctx *ctx)
 | |
| {
 | |
| 	return xdp_redirect_collect_stat(ctx);
 | |
| }
 | |
| 
 | |
| /* Tracepoint format: /sys/kernel/debug/tracing/events/xdp/xdp_exception/format
 | |
|  * Code in:                kernel/include/trace/events/xdp.h
 | |
|  */
 | |
| struct xdp_exception_ctx {
 | |
| 	u64 __pad;	// First 8 bytes are not accessible by bpf code
 | |
| 	int prog_id;	//	offset:8;  size:4; signed:1;
 | |
| 	u32 act;	//	offset:12; size:4; signed:0;
 | |
| 	int ifindex;	//	offset:16; size:4; signed:1;
 | |
| };
 | |
| 
 | |
| SEC("tracepoint/xdp/xdp_exception")
 | |
| int trace_xdp_exception(struct xdp_exception_ctx *ctx)
 | |
| {
 | |
| 	u64 *cnt;;
 | |
| 	u32 key;
 | |
| 
 | |
| 	key = ctx->act;
 | |
| 	if (key > XDP_REDIRECT)
 | |
| 		key = XDP_UNKNOWN;
 | |
| 
 | |
| 	cnt = bpf_map_lookup_elem(&exception_cnt, &key);
 | |
| 	if (!cnt)
 | |
| 		return 1;
 | |
| 	*cnt += 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 |