网络编程工程实训_实验三 使用Libpcap库离线抓包并解析

实验任务

使用前期搭建好的开发环境(Centos7.0+gcc+libpcap),使用C/C++语言开发一个应用程序,打开离线数据包文件(前期使用tcpdump抓取的数据包文件,已经发送给大家),从其中抓取一个数据包,并对数据包内容进行解析。

实验内容:

1.使用libpcap库的pcap_t *pcap_open_offline(const char *fname, char *errbuf)函数,打开离线文件(离线文件以绝对路径的做为第一个参数,保存出错信息的字符串做为第二个参数,返回一个设备指针),并根据打开成功与否,打印“Hello+你的姓名!”(成功)或“Sorry+你的姓名!”(失败)
2.使用上一步中获取的设备,使用pcap_next()函数(或pcap_next_ex()函数),从离线文件中抓取一个数据包,并根据抓包成功与否,打印“Hello+你的姓名!”(成功)或“Sorry+你的姓名!”(失败)
3.针对你抓取到的数据包进行如下解析:
数据链路层头部:查找并阅读RFC894,针对数据链路层头部进行解析,并输出各个字段名称+对应字段取值,并解释含义。如:TYPE=0x8864表示以太网负载为PPPoE数据,YPE=0x8864表示以太网负载为IP数据。
PPPoE头部解析:查找并阅读RFC 2516&

  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用socket来抓包,需要在程序中创建一个原始套接字。下面是一个简单的示例程序,可以抓取指定IP地址和端口号的网络数据包: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <arpa/inet.h> #include <sys/socket.h> #include <netinet/in.h> #include <netinet/ip.h> #include <netinet/tcp.h> #define BUFSIZE 65536 int main(int argc, char *argv[]) { int sock_raw, sock_tcp, recv_len; char buffer[BUFSIZE]; struct sockaddr_in saddr; struct iphdr *iph; struct tcphdr *tcph; if (argc != 3) { printf("Usage: %s <source IP> <destination IP>\n", argv[0]); exit(1); } // 创建原始套接字 sock_raw = socket(AF_INET, SOCK_RAW, IPPROTO_TCP); if (sock_raw < 0) { perror("socket() error"); exit(1); } // 绑定原始套接字到指定源IP地址 saddr.sin_family = AF_INET; saddr.sin_addr.s_addr = inet_addr(argv[1]); if (bind(sock_raw, (struct sockaddr *)&saddr, sizeof(saddr)) < 0) { perror("bind() error"); close(sock_raw); exit(1); } // 过滤指定目的IP地址和端口号的数据包 struct sockaddr_in daddr; daddr.sin_family = AF_INET; daddr.sin_addr.s_addr = inet_addr(argv[2]); struct sock_filter filter[] = { {0x28, 0, 0, 0x0000000c}, {0x15, 0, 5, 0x00000800}, {0x20, 0, 0, 0x0000001e}, {0x15, 0, 3, daddr.sin_addr.s_addr}, {0x20, 0, 0, 0x00000022}, {0x15, 0, 1, 0x0000006f}, {0x6, 0, 0, 0x0000ffff}, {0x6, 0, 0, 0x00000000}, }; struct sock_fprog fprog; fprog.len = sizeof(filter)/sizeof(filter[0]); fprog.filter = filter; if (setsockopt(sock_raw, SOL_SOCKET, SO_ATTACH_FILTER, &fprog, sizeof(fprog)) < 0) { perror("setsockopt() error"); close(sock_raw); exit(1); } // 创建TCP套接字 sock_tcp = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (sock_tcp < 0) { perror("socket() error"); close(sock_raw); exit(1); } // 接收网络数据包并分析 while (1) { recv_len = recv(sock_raw, buffer, BUFSIZE, 0); if (recv_len < 0) { perror("recv() error"); close(sock_raw); close(sock_tcp); exit(1); } iph = (struct iphdr *)buffer; if (iph->protocol == IPPROTO_TCP) { tcph = (struct tcphdr *)(buffer + iph->ihl*4); if (ntohs(tcph->dest) == 80) { printf("Source IP: %s\n", inet_ntoa(*(struct in_addr *)&iph->saddr)); printf("Destination IP: %s\n", inet_ntoa(*(struct in_addr *)&iph->daddr)); printf("Source port: %d\n", ntohs(tcph->source)); printf("Destination port: %d\n", ntohs(tcph->dest)); } } } close(sock_raw); close(sock_tcp); return 0; } ``` 该示例程序使用了一个过滤器来过滤指定目的IP地址和端口号的数据包。过滤器是一个由一组过滤规则组成的程序,用于过滤网络数据包。在本示例中,过滤器使用了Linux内核提供的BPF虚拟机进行解释和执行。可以通过修改过滤器规则来实现更复杂的过滤功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值