问题描述
我是 eBPF 的新手,想学习如何做一些基本的事情。我的问题是如何为我的 eBPF 代码编写 C 代码,以便以十六进制打印(bpf_trace_printk
)获得的数据包的 UPD 有效负载。我试过没有运气。这是我当前的代码:
int udppingpong(struct __sk_buff *skb)
{
void *data = (void *)(long)skb->data;
void *data_end = (void *)(long)skb->data_end;
struct ethhdr *eth = data;
struct iphdr *ip;
struct udphdr *udpdata;
if ((void *)eth + sizeof(*eth) > data_end) {
return TC_ACT_UNSPEC;
}
ip = data + sizeof(*eth);
if ((void *)ip + sizeof(*ip) > data_end) {
return TC_ACT_UNSPEC;
}
udpdata = (void *)ip + sizeof(*ip);
if ((void *)udpdata + sizeof(*udpdata) > data_end) {
return TC_ACT_UNSPEC;
}
if (eth->h_proto != htons(ETH_P_IP)) {
return TC_ACT_UNSPEC;
}
if (ip->protocol != IPPROTO_UDP) {
return TC_ACT_UNSPEC;
}
unsigned int payload_size;
unsigned char *payload;
payload_size = ntohs(udpdata->len) - sizeof(*udpdata);
payload = (unsigned char *)udpdata + sizeof(*udpdata);
if ((void *)payload + payload_size > data_end) {
return TC_ACT_UNSPEC;
}
__be16 port = udpdata->dest;
__be16 portToFilter = htons(7878);
if (port != portToFilter) {
return TC_ACT_OK;
}
__u32 src_ip = ip->saddr;
bpf_trace_printk("proto= %d,src= %lu\n",ip->protocol,src_ip); // --> This shows in decimal and network format (reversed),how to show actual IP like 1.2.3.4?
bpf_trace_printk("payload= %02x\n",payload); // --> HOW? I need it in hex to compare what is received
return TC_ACT_OK;
}
特别注意带有痕迹的最后几行。你能帮我打印十六进制格式的 UDP 负载以及源 IP 吗?
谢谢。
解决方法
我强烈建议在用户空间进行这种后期处理; bpf_trace_printk
无论如何都不适用于生产环境(请参阅它在系统日志中留下的大量警告)。使用 bpf_trace_printk
打印 UDP 负载也很困难且效率低下。
要在用户空间进行后处理,您可以依赖 bpf_skb_output
或其在 bcc 中的更高级别对应物 perf_submit_skb()
。这将允许您将数据包传递到用户空间,然后用户空间可以显示其 UDP 负载。
您可以在 bcc 存储库中找到 tutorial 和 an example。
您可以在 BPF 方面做什么:
- 以 IP 地址格式打印
src_ip
相当容易。您可以关注this StackOverflow answer。 - 您无法打印完整的、可变长度的 UDP 负载,但您可以通过将前 N 个字节传递给
bpf_trace_printk
来打印它们,如下所示。
__u32 *data = payload;
bpf_trace_printk("payload= %x %x %x\n",*data,*(data+1),*(data+2));