1 // SPDX-License-Identifier: GPL-2.0
2 /*
3 * Copyright (c) 2012 - 2018 Microchip Technology Inc., and its subsidiaries.
4 * All rights reserved.
5 */
6
7 #include <linux/ieee80211.h>
8
9 #include "coreconfigurator.h"
10
11 #define TAG_PARAM_OFFSET (MAC_HDR_LEN + TIME_STAMP_LEN + \
12 BEACON_INTERVAL_LEN + CAP_INFO_LEN)
13
14 enum sub_frame_type {
15 ASSOC_REQ = 0x00,
16 ASSOC_RSP = 0x10,
17 REASSOC_REQ = 0x20,
18 REASSOC_RSP = 0x30,
19 PROBE_REQ = 0x40,
20 PROBE_RSP = 0x50,
21 BEACON = 0x80,
22 ATIM = 0x90,
23 DISASOC = 0xA0,
24 AUTH = 0xB0,
25 DEAUTH = 0xC0,
26 ACTION = 0xD0,
27 PS_POLL = 0xA4,
28 RTS = 0xB4,
29 CTS = 0xC4,
30 ACK = 0xD4,
31 CFEND = 0xE4,
32 CFEND_ACK = 0xF4,
33 DATA = 0x08,
34 DATA_ACK = 0x18,
35 DATA_POLL = 0x28,
36 DATA_POLL_ACK = 0x38,
37 NULL_FRAME = 0x48,
38 CFACK = 0x58,
39 CFPOLL = 0x68,
40 CFPOLL_ACK = 0x78,
41 QOS_DATA = 0x88,
42 QOS_DATA_ACK = 0x98,
43 QOS_DATA_POLL = 0xA8,
44 QOS_DATA_POLL_ACK = 0xB8,
45 QOS_NULL_FRAME = 0xC8,
46 QOS_CFPOLL = 0xE8,
47 QOS_CFPOLL_ACK = 0xF8,
48 BLOCKACK_REQ = 0x84,
49 BLOCKACK = 0x94,
50 FRAME_SUBTYPE_FORCE_32BIT = 0xFFFFFFFF
51 };
52
get_beacon_period(u8 * data)53 static inline u16 get_beacon_period(u8 *data)
54 {
55 u16 bcn_per;
56
57 bcn_per = data[0];
58 bcn_per |= (data[1] << 8);
59
60 return bcn_per;
61 }
62
get_beacon_timestamp_lo(u8 * data)63 static inline u32 get_beacon_timestamp_lo(u8 *data)
64 {
65 u32 time_stamp = 0;
66 u32 index = MAC_HDR_LEN;
67
68 time_stamp |= data[index++];
69 time_stamp |= (data[index++] << 8);
70 time_stamp |= (data[index++] << 16);
71 time_stamp |= (data[index] << 24);
72
73 return time_stamp;
74 }
75
get_beacon_timestamp_hi(u8 * data)76 static inline u32 get_beacon_timestamp_hi(u8 *data)
77 {
78 u32 time_stamp = 0;
79 u32 index = (MAC_HDR_LEN + 4);
80
81 time_stamp |= data[index++];
82 time_stamp |= (data[index++] << 8);
83 time_stamp |= (data[index++] << 16);
84 time_stamp |= (data[index] << 24);
85
86 return time_stamp;
87 }
88
get_sub_type(u8 * header)89 static inline enum sub_frame_type get_sub_type(u8 *header)
90 {
91 return ((enum sub_frame_type)(header[0] & 0xFC));
92 }
93
get_to_ds(u8 * header)94 static inline u8 get_to_ds(u8 *header)
95 {
96 return (header[1] & 0x01);
97 }
98
get_from_ds(u8 * header)99 static inline u8 get_from_ds(u8 *header)
100 {
101 return ((header[1] & 0x02) >> 1);
102 }
103
get_address1(u8 * msa,u8 * addr)104 static inline void get_address1(u8 *msa, u8 *addr)
105 {
106 memcpy(addr, msa + 4, 6);
107 }
108
get_address2(u8 * msa,u8 * addr)109 static inline void get_address2(u8 *msa, u8 *addr)
110 {
111 memcpy(addr, msa + 10, 6);
112 }
113
get_address3(u8 * msa,u8 * addr)114 static inline void get_address3(u8 *msa, u8 *addr)
115 {
116 memcpy(addr, msa + 16, 6);
117 }
118
get_BSSID(u8 * data,u8 * bssid)119 static inline void get_BSSID(u8 *data, u8 *bssid)
120 {
121 if (get_from_ds(data) == 1)
122 get_address2(data, bssid);
123 else if (get_to_ds(data) == 1)
124 get_address1(data, bssid);
125 else
126 get_address3(data, bssid);
127 }
128
get_ssid(u8 * data,u8 * ssid,u8 * p_ssid_len)129 static inline void get_ssid(u8 *data, u8 *ssid, u8 *p_ssid_len)
130 {
131 u8 i, j, len;
132
133 len = data[TAG_PARAM_OFFSET + 1];
134 j = TAG_PARAM_OFFSET + 2;
135
136 if (len >= MAX_SSID_LEN)
137 len = 0;
138
139 for (i = 0; i < len; i++, j++)
140 ssid[i] = data[j];
141
142 ssid[len] = '\0';
143
144 *p_ssid_len = len;
145 }
146
get_cap_info(u8 * data)147 static inline u16 get_cap_info(u8 *data)
148 {
149 u16 cap_info = 0;
150 u16 index = MAC_HDR_LEN;
151 enum sub_frame_type st;
152
153 st = get_sub_type(data);
154
155 if (st == BEACON || st == PROBE_RSP)
156 index += TIME_STAMP_LEN + BEACON_INTERVAL_LEN;
157
158 cap_info = data[index];
159 cap_info |= (data[index + 1] << 8);
160
161 return cap_info;
162 }
163
get_asoc_status(u8 * data)164 static inline u16 get_asoc_status(u8 *data)
165 {
166 u16 asoc_status;
167
168 asoc_status = data[3];
169 return (asoc_status << 8) | data[2];
170 }
171
get_tim_elm(u8 * msa,u16 rx_len,u16 tag_param_offset)172 static u8 *get_tim_elm(u8 *msa, u16 rx_len, u16 tag_param_offset)
173 {
174 u16 index;
175
176 index = tag_param_offset;
177
178 while (index < (rx_len - FCS_LEN)) {
179 if (msa[index] == WLAN_EID_TIM)
180 return &msa[index];
181 index += (IE_HDR_LEN + msa[index + 1]);
182 }
183
184 return NULL;
185 }
186
get_current_channel_802_11n(u8 * msa,u16 rx_len)187 static u8 get_current_channel_802_11n(u8 *msa, u16 rx_len)
188 {
189 u16 index;
190
191 index = TAG_PARAM_OFFSET;
192 while (index < (rx_len - FCS_LEN)) {
193 if (msa[index] == WLAN_EID_DS_PARAMS)
194 return msa[index + 2];
195 index += msa[index + 1] + IE_HDR_LEN;
196 }
197
198 return 0;
199 }
200
wilc_parse_network_info(u8 * msg_buffer,struct network_info ** ret_network_info)201 s32 wilc_parse_network_info(u8 *msg_buffer,
202 struct network_info **ret_network_info)
203 {
204 struct network_info *network_info;
205 u8 *wid_val, *msa, *tim_elm, *ies;
206 u32 tsf_lo, tsf_hi;
207 u16 wid_len, rx_len, ies_len;
208 u8 msg_type, index;
209
210 msg_type = msg_buffer[0];
211
212 if ('N' != msg_type)
213 return -EFAULT;
214
215 wid_len = MAKE_WORD16(msg_buffer[6], msg_buffer[7]);
216 wid_val = &msg_buffer[8];
217
218 network_info = kzalloc(sizeof(*network_info), GFP_KERNEL);
219 if (!network_info)
220 return -ENOMEM;
221
222 network_info->rssi = wid_val[0];
223
224 msa = &wid_val[1];
225
226 rx_len = wid_len - 1;
227 network_info->cap_info = get_cap_info(msa);
228 network_info->tsf_lo = get_beacon_timestamp_lo(msa);
229
230 tsf_lo = get_beacon_timestamp_lo(msa);
231 tsf_hi = get_beacon_timestamp_hi(msa);
232
233 network_info->tsf_hi = tsf_lo | ((u64)tsf_hi << 32);
234
235 get_ssid(msa, network_info->ssid, &network_info->ssid_len);
236 get_BSSID(msa, network_info->bssid);
237
238 network_info->ch = get_current_channel_802_11n(msa, rx_len
239 + FCS_LEN);
240
241 index = MAC_HDR_LEN + TIME_STAMP_LEN;
242
243 network_info->beacon_period = get_beacon_period(msa + index);
244
245 index += BEACON_INTERVAL_LEN + CAP_INFO_LEN;
246
247 tim_elm = get_tim_elm(msa, rx_len + FCS_LEN, index);
248 if (tim_elm)
249 network_info->dtim_period = tim_elm[3];
250 ies = &msa[TAG_PARAM_OFFSET];
251 ies_len = rx_len - TAG_PARAM_OFFSET;
252
253 if (ies_len > 0) {
254 network_info->ies = kmemdup(ies, ies_len, GFP_KERNEL);
255 if (!network_info->ies) {
256 kfree(network_info);
257 return -ENOMEM;
258 }
259 }
260 network_info->ies_len = ies_len;
261
262 *ret_network_info = network_info;
263
264 return 0;
265 }
266
wilc_parse_assoc_resp_info(u8 * buffer,u32 buffer_len,struct connect_info * ret_conn_info)267 s32 wilc_parse_assoc_resp_info(u8 *buffer, u32 buffer_len,
268 struct connect_info *ret_conn_info)
269 {
270 u8 *ies;
271 u16 ies_len;
272
273 ret_conn_info->status = get_asoc_status(buffer);
274 if (ret_conn_info->status == WLAN_STATUS_SUCCESS) {
275 ies = &buffer[CAP_INFO_LEN + STATUS_CODE_LEN + AID_LEN];
276 ies_len = buffer_len - (CAP_INFO_LEN + STATUS_CODE_LEN +
277 AID_LEN);
278
279 ret_conn_info->resp_ies = kmemdup(ies, ies_len, GFP_KERNEL);
280 if (!ret_conn_info->resp_ies)
281 return -ENOMEM;
282
283 ret_conn_info->resp_ies_len = ies_len;
284 }
285
286 return 0;
287 }
288