• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Driver interaction with extended Linux CFG8021
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * Alternatively, this software may be distributed under the terms of BSD
9  * license.
10  *
11  */
12 
13 #include "driver_nl80211.h"
14 #include "wpa_supplicant_i.h"
15 #include "config.h"
16 #ifdef ANDROID
17 #include "android_drv.h"
18 #endif
19 
20 #define WPA_PS_ENABLED		0
21 #define WPA_PS_DISABLED		1
22 
23 #define MAX_WPSP2PIE_CMD_SIZE		512
24 
25 typedef struct android_wifi_priv_cmd {
26 	char *buf;
27 	int used_len;
28 	int total_len;
29 } android_wifi_priv_cmd;
30 
31 int send_and_recv_msgs(struct wpa_driver_nl80211_data *drv, struct nl_msg *msg,
32 		       int (*valid_handler)(struct nl_msg *, void *),
33 		       void *valid_data);
34 
35 static int drv_errors = 0;
36 
wpa_driver_send_hang_msg(struct wpa_driver_nl80211_data * drv)37 static void wpa_driver_send_hang_msg(struct wpa_driver_nl80211_data *drv)
38 {
39 	drv_errors++;
40 	if (drv_errors > DRV_NUMBER_SEQUENTIAL_ERRORS) {
41 		drv_errors = 0;
42 		wpa_msg(drv->ctx, MSG_INFO, WPA_EVENT_DRIVER_STATE "HANGED");
43 	}
44 }
45 
wpa_driver_set_power_save(void * priv,int state)46 static int wpa_driver_set_power_save(void *priv, int state)
47 {
48 	struct i802_bss *bss = priv;
49 	struct wpa_driver_nl80211_data *drv = bss->drv;
50 	struct nl_msg *msg;
51 	int ret = -1;
52 	enum nl80211_ps_state ps_state;
53 
54 	msg = nlmsg_alloc();
55 	if (!msg)
56 		return -1;
57 
58 	genlmsg_put(msg, 0, 0, drv->global->nl80211_id, 0, 0,
59 		    NL80211_CMD_SET_POWER_SAVE, 0);
60 
61 	if (state == WPA_PS_ENABLED)
62 		ps_state = NL80211_PS_ENABLED;
63 	else
64 		ps_state = NL80211_PS_DISABLED;
65 
66 	NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, drv->ifindex);
67 	NLA_PUT_U32(msg, NL80211_ATTR_PS_STATE, ps_state);
68 
69 	ret = send_and_recv_msgs(drv, msg, NULL, NULL);
70 	msg = NULL;
71 	if (ret < 0)
72 		wpa_printf(MSG_ERROR, "nl80211: Set power mode fail: %d", ret);
73 nla_put_failure:
74 	nlmsg_free(msg);
75 	return ret;
76 }
77 
get_power_mode_handler(struct nl_msg * msg,void * arg)78 static int get_power_mode_handler(struct nl_msg *msg, void *arg)
79 {
80 	struct nlattr *tb[NL80211_ATTR_MAX + 1];
81 	struct genlmsghdr *gnlh = nlmsg_data(nlmsg_hdr(msg));
82 	int *state = (int *)arg;
83 
84 	nla_parse(tb, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
85 		  genlmsg_attrlen(gnlh, 0), NULL);
86 
87 	if (!tb[NL80211_ATTR_PS_STATE])
88 		return NL_SKIP;
89 
90 	if (state) {
91 		*state = (int)nla_get_u32(tb[NL80211_ATTR_PS_STATE]);
92 		wpa_printf(MSG_DEBUG, "nl80211: Get power mode = %d", *state);
93 		*state = (*state == NL80211_PS_ENABLED) ?
94 				WPA_PS_ENABLED : WPA_PS_DISABLED;
95 	}
96 
97 	return NL_SKIP;
98 }
99 
wpa_driver_get_power_save(void * priv,int * state)100 static int wpa_driver_get_power_save(void *priv, int *state)
101 {
102 	struct i802_bss *bss = priv;
103 	struct wpa_driver_nl80211_data *drv = bss->drv;
104 	struct nl_msg *msg;
105 	int ret = -1;
106 	enum nl80211_ps_state ps_state;
107 
108 	msg = nlmsg_alloc();
109 	if (!msg)
110 		return -1;
111 
112 	genlmsg_put(msg, 0, 0, drv->global->nl80211_id, 0, 0,
113 		    NL80211_CMD_GET_POWER_SAVE, 0);
114 
115 	NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, drv->ifindex);
116 
117 	ret = send_and_recv_msgs(drv, msg, get_power_mode_handler, state);
118 	msg = NULL;
119 	if (ret < 0)
120 		wpa_printf(MSG_ERROR, "nl80211: Get power mode fail: %d", ret);
121 nla_put_failure:
122 	nlmsg_free(msg);
123 	return ret;
124 }
125 
wpa_driver_nl80211_driver_cmd(void * priv,char * cmd,char * buf,size_t buf_len)126 int wpa_driver_nl80211_driver_cmd(void *priv, char *cmd, char *buf,
127 				  size_t buf_len )
128 {
129 	struct i802_bss *bss = priv;
130 	struct wpa_driver_nl80211_data *drv = bss->drv;
131 	struct ifreq ifr;
132 	android_wifi_priv_cmd priv_cmd;
133 	int ret = 0;
134 
135 	if (os_strcasecmp(cmd, "STOP") == 0) {
136 		linux_set_iface_flags(drv->global->ioctl_sock, bss->ifname, 0);
137 		wpa_msg(drv->ctx, MSG_INFO, WPA_EVENT_DRIVER_STATE "STOPPED");
138 	} else if (os_strcasecmp(cmd, "START") == 0) {
139 		linux_set_iface_flags(drv->global->ioctl_sock, bss->ifname, 1);
140 		wpa_msg(drv->ctx, MSG_INFO, WPA_EVENT_DRIVER_STATE "STARTED");
141 	} else if (os_strcasecmp(cmd, "MACADDR") == 0) {
142 		u8 macaddr[ETH_ALEN] = {};
143 
144 		ret = linux_get_ifhwaddr(drv->global->ioctl_sock, bss->ifname, macaddr);
145 		if (!ret)
146 			ret = os_snprintf(buf, buf_len,
147 					  "Macaddr = " MACSTR "\n", MAC2STR(macaddr));
148 	} else if (os_strcasecmp(cmd, "RELOAD") == 0) {
149 		wpa_msg(drv->ctx, MSG_INFO, WPA_EVENT_DRIVER_STATE "HANGED");
150 	} else if (os_strncasecmp(cmd, "POWERMODE ", 10) == 0) {
151 		int state;
152 
153 		state = atoi(cmd + 10);
154 		ret = wpa_driver_set_power_save(priv, state);
155 		if (ret < 0)
156 			wpa_driver_send_hang_msg(drv);
157 		else
158 			drv_errors = 0;
159 	} else if (os_strncasecmp(cmd, "GETPOWER", 8) == 0) {
160 		int state = -1;
161 
162 		ret = wpa_driver_get_power_save(priv, &state);
163 		if (!ret && (state != -1)) {
164 			ret = os_snprintf(buf, buf_len, "POWERMODE = %d\n", state);
165 			drv_errors = 0;
166 		} else {
167 			wpa_driver_send_hang_msg(drv);
168 		}
169 	} else { /* Use private command */
170 		memset(&ifr, 0, sizeof(ifr));
171 		memset(&priv_cmd, 0, sizeof(priv_cmd));
172 		os_memcpy(buf, cmd, strlen(cmd) + 1);
173 		os_strncpy(ifr.ifr_name, bss->ifname, IFNAMSIZ);
174 
175 		priv_cmd.buf = buf;
176 		priv_cmd.used_len = buf_len;
177 		priv_cmd.total_len = buf_len;
178 		ifr.ifr_data = &priv_cmd;
179 
180 		if ((ret = ioctl(drv->global->ioctl_sock, SIOCDEVPRIVATE + 1, &ifr)) < 0) {
181 			wpa_printf(MSG_ERROR, "%s: failed to issue private commands\n", __func__);
182 		} else {
183 			drv_errors = 0;
184 			ret = 0;
185 			if ((os_strcasecmp(cmd, "LINKSPEED") == 0) ||
186 			    (os_strcasecmp(cmd, "RSSI") == 0) ||
187 			    (os_strcasecmp(cmd, "GETBAND") == 0) )
188 				ret = strlen(buf);
189 			else if (os_strcasecmp(cmd, "COUNTRY") == 0)
190 				wpa_supplicant_event(drv->ctx,
191 					EVENT_CHANNEL_LIST_CHANGED, NULL);
192 			else if (os_strncasecmp(cmd, "SETBAND", 7) == 0)
193 				wpa_printf(MSG_DEBUG, "%s: %s ", __func__, cmd);
194 			else if (os_strcasecmp(cmd, "P2P_DEV_ADDR") == 0)
195 				wpa_printf(MSG_DEBUG, "%s: P2P: Device address ("MACSTR")",
196 					__func__, MAC2STR(buf));
197 			else if (os_strcasecmp(cmd, "P2P_SET_PS") == 0)
198 				wpa_printf(MSG_DEBUG, "%s: P2P: %s ", __func__, buf);
199 			else if (os_strcasecmp(cmd, "P2P_SET_NOA") == 0)
200 				wpa_printf(MSG_DEBUG, "%s: P2P: %s ", __func__, buf);
201 			else
202 				wpa_printf(MSG_DEBUG, "%s %s len = %d, %d", __func__, buf, ret, strlen(buf));
203 		}
204 	}
205 	return ret;
206 }
207 
wpa_driver_set_p2p_noa(void * priv,u8 count,int start,int duration)208 int wpa_driver_set_p2p_noa(void *priv, u8 count, int start, int duration)
209 {
210 	char buf[MAX_DRV_CMD_SIZE];
211 
212 	memset(buf, 0, sizeof(buf));
213 	wpa_printf(MSG_DEBUG, "%s: Entry", __func__);
214 	snprintf(buf, sizeof(buf), "P2P_SET_NOA %d %d %d", count, start, duration);
215 	return wpa_driver_nl80211_driver_cmd(priv, buf, buf, strlen(buf)+1);
216 }
217 
wpa_driver_get_p2p_noa(void * priv,u8 * buf,size_t len)218 int wpa_driver_get_p2p_noa(void *priv, u8 *buf, size_t len)
219 {
220 	/* Return 0 till we handle p2p_presence request completely in the driver */
221 	return 0;
222 }
223 
wpa_driver_set_p2p_ps(void * priv,int legacy_ps,int opp_ps,int ctwindow)224 int wpa_driver_set_p2p_ps(void *priv, int legacy_ps, int opp_ps, int ctwindow)
225 {
226 	char buf[MAX_DRV_CMD_SIZE];
227 
228 	memset(buf, 0, sizeof(buf));
229 	wpa_printf(MSG_DEBUG, "%s: Entry", __func__);
230 	snprintf(buf, sizeof(buf), "P2P_SET_PS %d %d %d", legacy_ps, opp_ps, ctwindow);
231 	return wpa_driver_nl80211_driver_cmd(priv, buf, buf, strlen(buf) + 1);
232 }
233 
wpa_driver_set_ap_wps_p2p_ie(void * priv,const struct wpabuf * beacon,const struct wpabuf * proberesp,const struct wpabuf * assocresp)234 int wpa_driver_set_ap_wps_p2p_ie(void *priv, const struct wpabuf *beacon,
235 				 const struct wpabuf *proberesp,
236 				 const struct wpabuf *assocresp)
237 {
238 	char buf[MAX_WPSP2PIE_CMD_SIZE];
239 	struct wpabuf *ap_wps_p2p_ie = NULL;
240 	char *_cmd = "SET_AP_WPS_P2P_IE";
241 	char *pbuf;
242 	int ret = 0;
243 	int i;
244 	struct cmd_desc {
245 		int cmd;
246 		const struct wpabuf *src;
247 	} cmd_arr[] = {
248 		{0x1, beacon},
249 		{0x2, proberesp},
250 		{0x4, assocresp},
251 		{-1, NULL}
252 	};
253 
254 	wpa_printf(MSG_DEBUG, "%s: Entry", __func__);
255 	for (i = 0; cmd_arr[i].cmd != -1; i++) {
256 		os_memset(buf, 0, sizeof(buf));
257 		pbuf = buf;
258 		pbuf += sprintf(pbuf, "%s %d", _cmd, cmd_arr[i].cmd);
259 		*pbuf++ = '\0';
260 		ap_wps_p2p_ie = cmd_arr[i].src ?
261 			wpabuf_dup(cmd_arr[i].src) : NULL;
262 		if (ap_wps_p2p_ie) {
263 			os_memcpy(pbuf, wpabuf_head(ap_wps_p2p_ie), wpabuf_len(ap_wps_p2p_ie));
264 			ret = wpa_driver_nl80211_driver_cmd(priv, buf, buf,
265 				strlen(_cmd) + 3 + wpabuf_len(ap_wps_p2p_ie));
266 			wpabuf_free(ap_wps_p2p_ie);
267 			if (ret < 0)
268 				break;
269 		}
270 	}
271 
272 	return ret;
273 }
274