4.3.1 main函数分析

Android平台中,main函数定义于main.c中,代码如下所示。

[—>main.c::main]

  1. int main(int argc, char *argv[])
  2. {
  3. int c, i;
  4. struct wpa_interface *ifaces, *iface;
  5. int iface_count, exitcode = -1;
  6. struct wpa_params params;
  7. struct wpa_global *global;
  8. /*
  9. Android平台中,下面这个函数的实现在os_unix.c中。Android对其做了一些修改,主要是权
  10. 限方面的设置防止某些情况下被破解者利用权限漏洞以获取root权限。
  11. */
  12. if (os_program_init())
  13. return -1;
  14.  
  15. os_memset(&params, 0, sizeof(params));
  16. params.wpa_debug_level = MSG_INFO;
  17.  
  18. iface = ifaces = os_zalloc(sizeof(struct wpa_interface));
  19. ......
  20. iface_count = 1;
  21. wpa_supplicant_fd_workaround(); // 输入输出重定向到/dev/null设备
  22.  
  23. for (;;) { // 参数解析,由图4-3所知,Note 2中WPAS启动只使用了4个参数
  24. c = getopt(argc, argv, "b:Bc:C:D:de:f:g:hi:KLNo:O:p:P:qstuvW");
  25. if (c < 0)
  26. break;
  27. switch (c) {
  28. ......
  29. case 'c':
  30. // 指定配置文件名。注意,该参数赋值给了wpa_interface中的变量
  31. iface->confname = optarg;
  32. break;
  33. ......
  34. case 'D':
  35. // 指定driver名称。注意,该参数赋值给了wpa_interface中的变量
  36. iface->driver = optarg;
  37. break;
  38. ......
  39. case 'e':
  40. // 指定初始随机数文件,用于后续随机数的生成
  41. params.entropy_file = optarg; break;
  42. ......
  43. case 'i':
  44. iface->ifname = optarg; // 指定网络设备接口名,本例是"wlan0"
  45. break;
  46. ......
  47. }
  48. }
  49.  
  50. exitcode = 0;
  51. // 关键函数①:根据传入的参数,创建并初始化一个wpa_global对象
  52. global = wpa_supplicant_init(&params);
  53. ......
  54.  
  55. for (i = 0; exitcode == 0 && i < iface_count; i++) {
  56. ......
  57. // 关键函数②:WPAS支持操作多个无线网络设备,此处需将它们一一添加到WPAS中
  58. // WPAS内部将初始化这些设备
  59. if (wpa_supplicant_add_iface(global, &ifaces[i]) == NULL)
  60. exitcode = -1;
  61. }
  62. // Android平台中,wpa_supplicant通过select或epoll方式实现多路I/O复用。相关解释见下文
  63. if (exitcode == 0)
  64. exitcode = wpa_supplicant_run(global);
  65.  
  66. wpa_supplicant_deinit(global);
  67. ......// 退出
  68. return exitcode;
  69. }

main函数中出现了几个重要的数据结构和两个关键函数。

注意 虽然WPAS代码遵循C语法,但笔者也将称结构体实例称为对象。

先来认识这几个重要数据结构,如图4-7所示。

4.3.1 main函数分析 - 图1

图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代表当前编译到系统中的driver wrapper个数(详情见下文)。另外,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平台中更多和随机数有关的知识。