4.3.1 main函数分析
最后更新于:2022-04-02 06:03:02
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(¶ms, 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(¶ms);
......
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://docs.gechiui.com/gc-content/uploads/sites/kancloud/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平台中更多和随机数有关的知识。
';