gpt4 book ai didi

c - 为什么samples/bpf 中的ebpf 程序不起作用?

转载 作者:行者123 更新时间:2023-11-30 14:37:29 28 4
gpt4 key购买 nike

目标:在4.18.0内核源码树的samples/bpf目录下编写一个新的ebpf示例,编译并执行.

问题:编译后,当我运行sudo ./mine时,它就会终止。

mine_kern.c

#include <uapi/linux/bpf.h>
#include <uapi/linux/if_ether.h>
#include <uapi/linux/ip.h>
#include <linux/in.h>
#include <linux/if_packet.h>
#include "bpf_helpers.h"

int icmp_filter(struct __sk_buff *skb){

int proto = load_byte(skb, ETH_HLEN + offsetof(struct iphdr, protocol));
if(proto == IPPROTO_ICMP && skb->pkt_type == PACKET_OUTGOING){
return -1;
} else {
return 0;
}
}

char _license[] SEC("license") = "GPL";

mine_user.c

#include <stdio.h>
#include <assert.h>
#include <linux/bpf.h>
#include <bpf/bpf.h>
#include "bpf_load.h"
#include "sock_example.h"
#include <unistd.h>
#include <arpa/inet.h>

int main(int ac, char **argv)
{
char filename[256];
FILE *f;
int i, sock;

snprintf(filename, sizeof(filename), "%s_kern.o", argv[0]);

if (load_bpf_file(filename)) {
printf("%s", bpf_log_buf);
return 1;
}

sock = open_raw_sock("lo");

assert(setsockopt(sock, SOL_SOCKET, SO_ATTACH_BPF, prog_fd,
sizeof(prog_fd[0])) == 0);

f = popen("ping -c5 localhost", "r");
(void) f;

char buf[65535];

for(i=0; i<20; i++){
int res = recvfrom(sock, buf, sizeof(buf), 0, NULL, 0);
printf("res=%d\n", res);
}

return 0;
}

我还修改了 samples/bpf 内的 Makefile,添加了 mine_user.cmine_kern.c如有需要。问题:这段代码有什么问题?

最佳答案

由于 load_bpf_file() 加载函数的方式,您需要将 BPF 程序函数放在单独的 ELF 部分中。例如,我可以使用以下命令加载程序:

SEC("socket")
int icmp_filter(struct __sk_buff *skb){
...
}

之后,我在运行程序时看到一系列 res=-1 。这是因为您的套接字被 sock_example.h 中的 open_raw_sock() 设置为非阻塞:

sock = socket(PF_PACKET, SOCK_RAW|SOCK_NONBLOCK|SOCK_CLOEXEC, htons(ETH_P_ALL));

因此,当没有要接收的数据包时,recvfrom() 只需返回 -1(并将 errno 设置为 - EGAIN——顺便说一下,你应该考虑打印strerror(errno))而不是等待数据包。所以您可能也想改变这一点。

关于c - 为什么samples/bpf 中的ebpf 程序不起作用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57291946/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com