首页 > 代码库 > Winpcap笔记3之打开适配器并捕获数据包

Winpcap笔记3之打开适配器并捕获数据包

上一讲中知道了如何获取适配的信息,这一将我们讲写一个程序蒋每一个通过适配器的数据包打印出来。

打开设备的函数是pcap_open().函数原型是

pcap_t* pcap_open(const char* source,int snaplen,int flags,int read_timeout,struct pcap_rmtauth *auth,char * errbuf);‘

pcap_rmatauth

{

  int type.

     char *username;;//Zero-terminated string containing the username that has to be used on the remote machine for authentication

     char *password;

}

snaplen:snaplen 制定要捕获数据包中的哪些部分。 在一些操作系统中 (比如 xBSD 和 Win32), 驱动可以被配置成只捕获数据包的初始化部分: 这样可以减少应用程序间复制数据的量,从而提高捕获效率。本例中,我们将值定为65535,它比我们能遇到的最大的MTU还要大。因此,我们确信我们总能收到完整的数据包。

flags: 最最重要的flag是用来指示适配器是否要被设置成混杂模式。 一般情况下,适配器只接收发给它自己的数据包, 而那些在其他机器之间通讯的数据包,将会被丢弃。 相反,如果适配器是混杂模式,那么不管这个数据包是不是发给我的,我都会去捕获。也就是说,我会去捕获所有的数据包。 这意味着在一个共享媒介(比如总线型以太网),WinPcap能捕获其他主机的所有的数据包。 大多数用于数据捕获的应用程序都会将适配器设置成混杂模式,所以,我们也会在下面的范例中,使用混杂模式。

                    PCAP_OPENFLAG_PROMISCUOUS:1,它定义了适配器(网卡)是否进入混杂模式(promiscuous mode)。   

      PCAP_OPENFLAG_DATATX_UDP:2,它定义了数据传输(假如是远程抓包)是否用UDP协议来处理。

          PCAP_OPENFLAG_NOCAPTURE_RPCAP:4,它定义了远程探测器是否捕获它自己产生的数据包。

to_ms 指定读取数据的超时时间,以毫秒计(1s=1000ms)。在适配器上进行读取操作(比如用 pcap_dispatch() 或 pcap_next_ex()) 都会在 to_ms 毫秒时间内响应,即使在网络上没有可用的数据包。 在统计模式下to_ms 还可以用来定义统计的时间间隔。 将 to_ms 设置为0意味着没有超时,那么如果没有数据包到达的话,读操作将永远不会返回。 如果设置成-1,则情况恰好相反,无论有没有数据包到达,读操作都会立即返回。

 

read_timeout:以毫秒为单位。read timeout被用来设置在遇到一个数据包的时候读操作不必立即返回,而是等待一段时间,让更多的数据包到来后从OS内核一次读多个数据包。并非所有的平台都支持read timeout;在不支持read timeout的平台上它将被忽略。

auth:一个指向’struct pcap_rmtauth’的指针,保存当一个用户登录到某个远程机器上时的必要信息。假如不是远程抓包,该指针被设置为NULL。

errbuf:一个指向用户申请的缓冲区的指针,存放当该函数出错时的错误信息。

返回值是一个’pcap_t’指针,它可以作为下一步调用(例如pcap_compile()等)的参数,并且指定了一个已经打开的Winpcap会话。在遇到问题的情况下,它返回NULL并且’errbuf’变量保存了错误信息。

 

 

函数1:

 

int pcap_loop(  pcap_t*           p,

 

int                   cnt,

 

pcap_hander    callback,

 

u_char*           user

 

)

 

    收集一群数据包。pcap_loop()与pcap_dispatch()类似,但是它会一直保持读数据包的操作直到cnt包被处理或者发生了错误。当有活动的读超时(read timeout)时它并不返回。然而,对pcap_open_live()指定一个非0的读超时(read timeout),当发生超时的时候调用pcap_dispatch()来接收并处理到来的所有数据包更好。Cnt指明了返回之前要处理数据包的最大数目。如果cnt为负值,pcap_loop()将一直循环(直到发生错误才停止)。如果出错时返回-1;如果cnt用完时返回0;如果在任何包被处理前调用pcap_breakloop()来中止循环将返回-2。所以,如果程序中使用了pcap_breakloop(),必须准确的来判断返回值是-1还是-2,而不能简单的判断<0。

 

 

 

函数2:

 

hypedef void (* pcap_handler)(u_char* user,

 

const struct pcap_pkthdr* pkt_header,

 

const u_char* pkt_data)

 

    接收数据包的回调函数原型。当用户程序使用pcap_dispatch()或者pcap_loop(),数据包以这种回调的方法传给应用程序。用户参数是用户自己定义的包含捕获会话状态的参数,它必须跟pcap_dispatch()和pcap_loop()的参数相一致。pkt_hader是与抓包驱动有关的头。pkt_data指向包里的数据,包括协议头。

 

 

 

结构体1:

 

struct pcap_pkthdr {

 

      struct timeval ts;

 

      bpf_u_int32 caplen;

 

      bpf_u_int32 len;

 

}

 

ts:时间戳

struct timeval {
        long    tv_sec;         /* seconds */
        long    tv_usec;        /* and microseconds */
};

 

cpalen:当前分组的长度

 

len:数据包的长度

 

/*
* 截获数据包的试验。先打印出所有网络适配器的列表,然后选择
* 想在哪个适配器上截获数据包。然后通过pcap_loop()函数将截获
* 的数据包传给回调函数packet_handler()处理。

* 通过该程序初步了解了使用winpcap截获数据包的步骤以及一些在
* 截获数据包时非常重要的函数和结构体。
*/

 

 1 //打开适配器捕获数据包
 2 #include "pcap.h"
 3 
 4 /* packet handler 函数原型 */
 5 void packet_handler(u_char *param, const struct pcap_pkthdr *header, const u_char *pkt_data);
 6 
 7 int main()
 8 {
 9     pcap_if_t *alldevs;
10     pcap_if_t *d;
11     int inum;
12     int i = 0;
13     pcap_t *adhandle;
14     char errbuf[PCAP_ERRBUF_SIZE];
15 
16     /* 获取本机设备列表 */
17     if (pcap_findalldevs_ex(PCAP_SRC_IF_STRING, NULL, &alldevs, errbuf) == -1)
18     {
19         fprintf(stderr, "Error in pcap_findalldevs: %s\n", errbuf);
20         exit(1);
21     }
22 
23     /* 打印列表 */
24     for (d = alldevs; d; d = d->next)
25     {
26         printf("%d. %s", ++i, d->name);
27         if (d->description)
28             printf(" (%s)\n", d->description);
29         else
30             printf(" (No description available)\n");
31     }
32 
33     if (i == 0)
34     {
35         printf("\nNo interfaces found! Make sure WinPcap is installed.\n");
36         return -1;
37     }
38 
39     printf("Enter the interface number (1-%d):", i);
40     scanf("%d", &inum);
41 
42     if (inum < 1 || inum > i)
43     {
44         printf("\nInterface number out of range.\n");
45         /* 释放设备列表 */
46         pcap_freealldevs(alldevs);
47         return -1;
48     }
49 
50     /* 跳转到选中的适配器 */
51     for (d = alldevs, i = 0; i < inum - 1; d = d->next, i++);
52 
53     /* 打开设备 */
54     if ((adhandle = pcap_open(d->name,          // 设备名
55         65536,            // 65535保证能捕获到不同数据链路层上的每个数据包的全部内容
56         PCAP_OPENFLAG_PROMISCUOUS,    // 混杂模式
57         1000,             // 读取超时时间
58         NULL,             // 远程机器验证
59         errbuf            // 错误缓冲池
60         )) == NULL)
61     {
62         fprintf(stderr, "\nUnable to open the adapter. %s is not supported by WinPcap\n", d->name);
63         /* 释放设备列表 */
64         pcap_freealldevs(alldevs);
65         return -1;
66     }
67 
68     printf("\nlistening on %s...\n", d->description);
69 
70     /* 释放设备列表 */
71     pcap_freealldevs(alldevs);
72 
73     /* 开始捕获 */
74     pcap_loop(adhandle, 0, packet_handler, NULL);
75 
76     return 0;
77 }
78 
79 
80 /* 每次捕获到数据包时,libpcap都会自动调用这个回调函数 */
81 void packet_handler(u_char *param, const struct pcap_pkthdr *header, const u_char *pkt_data)
82 {
83     struct tm *ltime;
84     char timestr[16];
85     time_t local_tv_sec;
86 
87     /* 将时间戳转换成可识别的格式 */
88     local_tv_sec = header->ts.tv_sec;
89     ltime = localtime(&local_tv_sec);
90     strftime(timestr, sizeof timestr, "%H:%M:%S", ltime);
91 
92     printf("%s,%.6d len:%d\n", timestr, header->ts.tv_usec, header->len);
93 
94 }

 

技术分享

 

当适配器被打开,捕获工作就可以用 pcap_dispatch() 或 pcap_loop()进行。 这两个函数非常的相似,区别就是 pcap_ dispatch() 当超时时间到了(timeout expires)就返回 (尽管不能保证) ,而 pcap_loop() 不会因此而返回,只有当 cnt 数据包被捕获,所以,pcap_loop()会在一小段时间内,阻塞网络的利用。pcap_loop()对于我们这个简单的范例来说,可以满足需求,不过, pcap_dispatch() 函数一般用于比较复杂的程序中。

这两个函数都有一个 回调 参数, packet_handler指向一个可以接收数据包的函数。 这个函数会在收到每个新的数据包并收到一个通用状态时被libpcap所调用 ( 与函数 pcap_loop() 和 pcap_dispatch() 中的 user 参数相似),数据包的首部一般有一些诸如时间戳,数据包长度的信息,还有包含了协议首部的实际数据。 注意:冗余校验码CRC不再支持,因为帧到达适配器,并经过校验确认以后,适配器就会将CRC删除,与此同时,大部分适配器会直接丢弃CRC错误的数据包,所以,WinPcap没法捕获到它们。

 

上面的程序将每一个数据包的时间戳和长度从 pcap_pkthdr 的首部解析出来,并打印在屏幕上。

 

请注意,使用 pcap_loop() 函数可能会遇到障碍,主要因为它直接由数据包捕获驱动所调用。因此,用户程序是不能直接控制它的。另一个实现方法(也是提高可读性的方法),是使用 pcap_next_ex() 函数。有关这个函数的使用,我们将在下一讲为您展示。 (不用回调方法捕获数据包).

 

Winpcap笔记3之打开适配器并捕获数据包