助力软件开发企业降本增效 PHP / java源码系统,只需一次付费,代码终身使用! 广告
Android平台中,main函数定义于main.c中,代码如下所示。 **main.c::main** ~~~ int main(int argc, char *argv[]) { int c, i; struct wpa_interface *ifaces, *iface; int iface_count, exitcode = -1; struct wpa_params params; struct wpa_global *global; /* Android平台中,下面这个函数的实现在os_unix.c中。Android对其做了一些修改,主要是权 限方面的设置防止某些情况下被破解者利用权限漏洞以获取root权限。 */ if (os_program_init()) return -1; os_memset(&params, 0, sizeof(params)); params.wpa_debug_level = MSG_INFO; iface = ifaces = os_zalloc(sizeof(struct wpa_interface)); ...... iface_count = 1; wpa_supplicant_fd_workaround(); // 输入输出重定向到/dev/null设备 for (;;) { // 参数解析,由图4-3所知,Note 2中WPAS启动只使用了4个参数 c = getopt(argc, argv, "b:Bc:C:D:de:f:g:hi:KLNo:O:p:P:qstuvW"); if (c < 0) break; switch (c) { ...... case 'c': // 指定配置文件名。注意,该参数赋值给了wpa_interface中的变量 iface->confname = optarg; break; ...... case 'D': // 指定driver名称。注意,该参数赋值给了wpa_interface中的变量 iface->driver = optarg; break; ...... case 'e': // 指定初始随机数文件,用于后续随机数的生成[^①] params.entropy_file = optarg; break; ...... case 'i': iface->ifname = optarg; // 指定网络设备接口名,本例是"wlan0" break; ...... } } exitcode = 0; // 关键函数①:根据传入的参数,创建并初始化一个wpa_global对象 global = wpa_supplicant_init(&params); ...... for (i = 0; exitcode == 0 && i < iface_count; i++) { ...... // 关键函数②:WPAS支持操作多个无线网络设备,此处需将它们一一添加到WPAS中 // WPAS内部将初始化这些设备 if (wpa_supplicant_add_iface(global, &ifaces[i]) == NULL) exitcode = -1; } // Android平台中,wpa_supplicant通过select或epoll方式实现多路I/O复用。相关解释见下文 if (exitcode == 0) exitcode = wpa_supplicant_run(global); wpa_supplicant_deinit(global); ......// 退出 return exitcode; } ~~~ main函数中出现了几个重要的数据结构和两个关键函数。 >[info] **注意** 虽然WPAS代码遵循C语法,但笔者也将称结构体实例称为对象。 先来认识这几个重要数据结构,如图4-7所示。 :-: ![](https://box.kancloud.cn/c513f785efed8ed7690d0dde4a3bd11c_1256x346.jpg) 图4-7 main函数中重要的数据结构 图4-7中: * wpa_interface用于描述一个无线网络设备。该参数在初始化时用到。 * wpa_global是一个全局性质的上下文信息。它通过ifaces变量指向一个wpa_supplicant对象(以后介绍wpa_supplicant时,读者将发现系统内的所有wpa_supplicant对象将通过单向链表连接在一起。所以,严格意义上来说,ifaces变量指向一个wpa_supplicant对象链表)。drv_priv包含driver wrapper所需要的全局上下文信息。其drv_count代表当前编译到系统中的driverwrapper个数(详情见下文)。另外,wpa_global有一个全局控制接口,如果设置该接口,其他wpa_interface设置的控制接口将被替代。 * wpa_supplicant是WPAS的核心数据结构。一个interface对应有一个wpa_supplicant对象,其内部包含非常多的成员变量(图4-7并未画出,下文详细介绍)。另外,系统中所有wpa_supplicant对象都通过next变量链接在一起。 * ctrl_iface_global_priv是全局控制接口的信息,内部包含一个用于通信的socket句柄。 **提示** 由于篇幅原因,笔者将根据情况略去数据结构中部分成员变量的介绍。 下面分析关键函数wpa_supplicant_init。 [^①]:读者可阅读《深入理解Android:卷Ⅱ》3.3节以了解Android平台中更多和随机数有关的知识。