gpt4 book ai didi

c++ - libpcap 和无线信号捕获

转载 作者:塔克拉玛干 更新时间:2023-11-03 02:16:59 26 4
gpt4 key购买 nike

我正在尝试编写一个 c++ 应用程序 (linux) 来捕获无线数据包以及相关的信号强度(以 dBm 为单位)。捕获部分很简单,但问题是我找不到任何关于如何获取每个数据包信号强度的文档。

它是标题的一部分吗?

这是我目前所拥有的:

printf("Device: %s\n", dev);
printf("Number of packets: %d\n", num_packets);
printf("Filter expression: %s\n", filter_exp);

/* open capture device */
pcap_t *handler = pcap_create("wlan0", errbuf);
if (handler == NULL)
{
exit(-1);
}
if(pcap_set_rfmon(handler,1)==0 )
{
printf("monitor mode enabled\n");
}
pcap_set_snaplen(handler, 2048); // Set the snapshot length to 2048
pcap_set_promisc(handler, 0); // Turn promiscuous mode off
pcap_set_timeout(handler, 512); // Set the timeout to 512 milliseconds
int status = pcap_activate(handler);


/* now we can set our callback function */
pcap_loop(handle, num_packets, got_packet, NULL);

这里是 got_packet 代码:

/* define ethernet header */
ethernet = (struct sniff_ethernet*)(packet);

/* define/compute ip header offset */
ip = (struct sniff_ip*)(packet + SIZE_ETHERNET);
size_ip = IP_HL(ip)*4;
if (size_ip < 20) {
printf(" * Invalid IP header length: %u bytes\n", size_ip);
return;
}

/* print source and destination IP addresses */
printf(" From: %s\n", inet_ntoa(ip->ip_src));
printf(" To: %s\n", inet_ntoa(ip->ip_dst));

/* determine protocol */
switch(ip->ip_p) {
case IPPROTO_TCP:
printf(" Protocol: TCP\n");
break;
case IPPROTO_UDP:
printf(" Protocol: UDP\n");
return;
case IPPROTO_ICMP:
printf(" Protocol: ICMP\n");
return;
case IPPROTO_IP:
printf(" Protocol: IP\n");
return;
default:
printf(" Protocol: unknown\n");
return;
}

/* define/compute tcp header offset */
tcp = (struct sniff_tcp*)(packet + SIZE_ETHERNET + size_ip);
size_tcp = TH_OFF(tcp)*4;
if (size_tcp < 20) {
printf(" * Invalid TCP header length: %u bytes\n", size_tcp);
return;
}

printf(" Src port: %d\n", ntohs(tcp->th_sport));
printf(" Dst port: %d\n", ntohs(tcp->th_dport));

/* define/compute tcp payload (segment) offset */
payload = (char *)(packet + SIZE_ETHERNET + size_ip + size_tcp);

/* compute tcp payload (segment) size */
size_payload = ntohs(ip->ip_len) - (size_ip + size_tcp);

/*
* Print payload data; it might be binary, so don't just
* treat it as a string.
*/
if (size_payload > 0) {
printf(" Payload (%d bytes):\n", size_payload);
//print_payload(payload, size_payload);
}

如有任何帮助,我们将不胜感激。

更新:/******************/这是一个更新:因此,根据我的研究,正如 Guy Scott 提到的那样,我正在寻找错误的信息。我需要查看无线数据包,而不是加载以太网数据包。所以这是更新后的代码:

 pcap_set_snaplen(handle, 2048);  // Set the snapshot length to 2048
pcap_set_promisc(handle, 1); // Turn promiscuous mode off
pcap_set_timeout(handle, 512); // Set the timeout to 512 milliseconds
int status = pcap_activate(handle);

if(pcap_set_datalink(handle, DLT_IEEE802_11_RADIO) == -1) {
printf("Couldn't set datalink type %s: %s\n", device, pcap_geterr(handle));
}

所以现在的问题是解析数据包,这似乎是一个非常困难的问题。我对源地址、目标地址和相关信号等感兴趣。我不知道如何匹配和加载数据包中的数据并将其匹配到 radiotap 结构。

struct ieee80211_radiotap_header {
u_int8_t it_version; /* set to 0 */
u_int8_t it_pad;
u_int16_t it_len; /* entire length */
u_int32_t it_present; /* fields present */
} __attribute__((__packed__));

/* Presence bits */
#define RADIOTAP_TSFT 0
#define RADIOTAP_FLAGS 1
#define RADIOTAP_RATE 2
#define RADIOTAP_CHANNEL 3
#define RADIOTAP_FHSS 4
#define RADIOTAP_ANTENNA_SIGNAL 5
#define RADIOTAP_ANTENNA_NOISE 6
#define RADIOTAP_LOCK_QUALITY 7
#define RADIOTAP_TX_ATTENUATION 8
#define RADIOTAP_DB_TX_ATTENUATION 9
#define RADIOTAP_DBM_TX_POWER 10
#define RADIOTAP_ANTENNA 11
#define RADIOTAP_DB_ANTENNA_SIGNAL 12

void process_packet (u_char * args, const struct pcap_pkthdr *header, const u_char * packet)
{
struct ieee80211_radiotap_header *packet_header = (struct ieee80211_radiotap_header *) header;
// This is where I am stuck

一旦我捕获了数据包,有人能告诉我如何从中提取上述值吗?

谢谢

最佳答案

每个调用 pcap_open_live()pcap_create()/pcap_activate()pcap_open_offline() 的程序> 如果调用成功,应该调用 pcap_datalink() 来找出捕获的链路层报头类型是什么。

此规则没有异常(exception)。

然后看the link-layer header types page for tcpdump.org查看 pcap_datalink() 返回的值的含义。将它们与此处列出的 DLT_ 值进行比较。您可能获得的是 DLT_IEEE802_11,它没有信号强度信息,以及 DLT_PRISM_HEADERDLT_IEEE802_11_RADIODLT_IEEE802_11_RADIO_AVS ,其中确实有信号强度信息。请参阅后三种可能性的链接,了解有关如何在数据包数据中表示信号强度信息(和其他 radio 元数据)的信息。

(而且,是的,这是 radio 元数据的三个选项,因此另一个答案中给出的链接指向不完整的来源;大多数时候您可能会得到radiotap header ,而不是 AVS 或 Prism header 。Radiotap 更通用,因为它被设计为可扩展的,但解析起来更复杂。)

关于c++ - libpcap 和无线信号捕获,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28772086/

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