Linux之bpfjit使用分析和mini-tcpdump实现




#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <net/if.h>
#include <sys/ioctl.h>
#include <arpa/inet.h>
#include <bpf/bpf.h>
#include <bpf/libbpf.h>
 
#define MAX_IFACES 16
#define BUF_SIZE 65535
 
int main(int argc, char **argv) {
    char buf[BUF_SIZE];
    struct bpf_prog_load_attr prog_load_attr = {
        .file = "mini_tcpdump.o",
        .prog_type = BPF_PROG_TYPE_SOCKET_FILTER,
    };
    struct bpf_object *obj;
    struct bpf_prog_info info = {};
    bpf_u_int32 netmask;
    char iface[IFNAMSIZ];
    int fd, err, i, ifindex;
    unsigned int iface_count = 0;
    struct ifreq ifr;
    struct sockaddr_ll sll;
 
    if (argc != 2) {
        fprintf(stderr, "Usage: %s <interface>\n", argv[0]);
        return 1;
    }
 
    strncpy(iface, argv[1], IFNAMSIZ - 1);
 
    if ((err = bpf_prog_load_xattr(&prog_load_attr, &obj, &fd))) {
        fprintf(stderr, "bpf_prog_load_xattr: %s\n", strerror(err));
        return 1;
    }
 
    if (bpf_obj_get_info(bpf_program__fd(prog_load_attr.prog), &info, sizeof(info))) {
        fprintf(stderr, "bpf_obj_get_info failed\n");
        return 1;
    }
 
    ifindex = if_nametoindex(iface);
    if (!ifindex) {
        perror("if_nametoindex");
        return 1;
    }
 
    memset(&ifr, 0, sizeof(ifr));
    strncpy(ifr.ifr_name, iface, IFNAMSIZ - 1);
 
    if (ioctl(fd, BIOCSETIF, &ifr) < 0) {
        perror("ioctl(BIOCSETIF)");
        return 1;
    }
 
    netmask = (bpf_u_int32)-1;
    if (ioctl(fd, BIOCSETWF, &netmask) < 0) {
        perror("ioctl(BIOCSETWF)");
        return 1;
    }
 
    memset(&sll, 0, sizeof(sll));
    sll.sll_family = AF_PACKET;
    sll.sll_ifindex = ifindex;
    sll.sll_protocol = htons(ETH_P_IP);
 
    if (bind(fd, (struct sockaddr *)&sll, sizeof(sll)) < 0) {
        perror("bind");
        return 1;
    }
 
    while (1) {
        int len = read(fd, buf, sizeof(buf));
        if (len < 0) {
            if (errno == EAGAIN)
                continue;
            perror("read");
            break;
        }
 
        // 处理读取到的数据包
        // ...
    }
 
    close(fd);
    return 0;
}

这段代码示例展示了如何加载一个BPF程序,并将其绑定到一个网络接口上,然后读取并处理数据包。它使用了BPF系统调用和socket编程接口,并且提供了一个简单的错误处理框架。这是一个基于Linux BPF JIT的实际应用示例,它可以作为开发

评论已关闭

推荐阅读

DDPG 模型解析,附Pytorch完整代码
2024年11月24日
DQN 模型解析,附Pytorch完整代码
2024年11月24日
AIGC实战——Transformer模型
2024年12月01日
Socket TCP 和 UDP 编程基础(Python)
2024年11月30日
python , tcp , udp
如何使用 ChatGPT 进行学术润色?你需要这些指令
2024年12月01日
AI
最新 Python 调用 OpenAi 详细教程实现问答、图像合成、图像理解、语音合成、语音识别(详细教程)
2024年11月24日
ChatGPT 和 DALL·E 2 配合生成故事绘本
2024年12月01日
omegaconf,一个超强的 Python 库!
2024年11月24日
【视觉AIGC识别】误差特征、人脸伪造检测、其他类型假图检测
2024年12月01日
[超级详细]如何在深度学习训练模型过程中使用 GPU 加速
2024年11月29日
Python 物理引擎pymunk最完整教程
2024年11月27日
MediaPipe 人体姿态与手指关键点检测教程
2024年11月27日
深入了解 Taipy:Python 打造 Web 应用的全面教程
2024年11月26日
基于Transformer的时间序列预测模型
2024年11月25日
Python在金融大数据分析中的AI应用(股价分析、量化交易)实战
2024年11月25日
AIGC Gradio系列学习教程之Components
2024年12月01日
Python3 `asyncio` — 异步 I/O,事件循环和并发工具
2024年11月30日
llama-factory SFT系列教程:大模型在自定义数据集 LoRA 训练与部署
2024年12月01日
Python 多线程和多进程用法
2024年11月24日
Python socket详解,全网最全教程
2024年11月27日