上层应用与wpa_supplicant,wpa_supplicant与kernel 相关socket创建交互分析
2013-09-16 21:43 ...平..淡... 阅读(3987) 评论(0) 编辑 收藏 举报单独拿出来,分析以下上层应用与wpa_supplicant wpa_supplicant与kernel 的socket交互。
关联上层应用与wpa_supplicant的socket的创建、连接流程分析
[-->./wpa_supplicant/ctrl_iface_unix.c]
1.wpa_supplicant初始化时,在wpa_supplicant_ctrl_iface_init方法中,会通过priv->sock = socket(PF_UNIX, SOCK_DGRAM, 0); 创建了一个socket,然后bind到指定的地址中的路径(/data/misc/wifi/sockets/p2p0)。
这之后,便会把socket的fd保存到eloop_run中发rfds中了。由select监听变化。
2.然后在打开wifi之后,应用层会去连接wpa_supplicant,在wifi.c:: wifi_connect_to_supplicant--->wifi_connect_on_socket_path方法中,会创建ctrl_conn与monitor_conn来和wpa_supplicant交互,它们各自通过wpa_ctrl_open方法创建了一个socket(它们的目的端地址路径都是/data/misc/wifi/sockets/p2p0,可在wifi_connect_to_supplicant方法中打log查看)。然后,通过bind、connect方法与wpa_supplicant初始化时创建的priv->sock建立了连接。
ps:之前已经讨论过monitor_conn的作用是监听wpa_supplicant传递过来的消息。为什么呢,其实是因为通过wpa_ctrl_attach(monitor_conn[index])语句,将monitor_conn中的socket信息(包括源端地址)保存到wpa_supplicant的某个链表结构中,这样wpa_supplicant就能通过socket发送消息给上层应用了。
贴出部分log:
//wpa_supplicant初始化时,创建的socket关联的路径 D/wpa_supplicant(2575): cb--1-addr.sun_path is wpa_/data/misc/wifi/sockets D/wpa_supplicant(2575): cb--2-addr.sun_path is /data/misc/wifi/sockets/p2p0 D/wpa_supplicant(2575): cb--1-addr.sun_path is wpa_wlan0 //创建ctrl_conn和monitor_conn时,socket关联的源端、目的端地址 I/wpa_ctrl(595): cb---fd is 254, ctrl->local.sun_path is /data/misc/wifi/sockets/wpa_ctrl_595-3, ctrl->dest.sun_path is /data/misc/wifi/sockets/p2p0 I/wpa_ctrl(595): cb---fd is 255, ctrl->local.sun_path is /data/misc/wifi/sockets/wpa_ctrl_595-4, ctrl->dest.sun_path is /data/misc/wifi/sockets/p2p0
关联kernel与wpa_supplicant的socket的创建流程分析 ---可参考” wpa_supplicant与kernel交互"
因为在”保存驱动接口”,select_driver方法中调用了global_init方法(会根据用户态的结构体wpa_driver_nl80211_ops中查找对应方法,即nl80211_global_init)。
在nl80211_global_init方法中,有两条关键语句:
//(1) 初始化netlink,并注册事件接收函数 global->netlink = netlink_init(cfg); //(2) 此global->ioctl_sock用作为ioctl命令的fd global->ioctl_sock = socket(PF_INET, SOCK_DGRAM, 0);
(1)netlink_init方法中创建了一个socket,并添加到eloop_run方法中的rfds中。用于从kernel态发送事件给用户态
netlink->sock = socket(PF_NETLINK, SOCK_RAW, NETLINK_ROUTE); ...... eloop_register_read_sock(netlink->sock, netlink_receive, netlink,NULL);
(2)该socket用于从用户态发送请求给kernel态