1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Portions copyright (C) 2023 Broadcom Limited
5  *
6  * Licensed under the Apache License, Version 2.0 (the "License");
7  * you may not use this file except in compliance with the License.
8  * You may obtain a copy of the License at
9  *
10  *     http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing, software
13  * distributed under the License is distributed on an "AS IS" BASIS,
14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  * See the License for the specific language governing permissions and
16  * limitations under the License.
17  */
18 
19 #include <stdint.h>
20 #include <fcntl.h>
21 #include <sys/socket.h>
22 #include <netlink/genl/genl.h>
23 #include <netlink/genl/family.h>
24 #include <netlink/genl/ctrl.h>
25 #include <linux/rtnetlink.h>
26 #include <netpacket/packet.h>
27 #include <linux/filter.h>
28 #include <linux/errqueue.h>
29 
30 #include <linux/pkt_sched.h>
31 #include <netlink/object-api.h>
32 #include <netlink/netlink.h>
33 #include <netlink/socket.h>
34 #include <netlink/handlers.h>
35 
36 #include <hardware_legacy/wifi_hal.h>
37 #include "common.h"
38 #include "cpp_bindings.h"
39 
40 /* test mode flag for halutil only */
41 bool halutil_mode = false;
getIfaceInfo(wifi_interface_handle handle)42 interface_info *getIfaceInfo(wifi_interface_handle handle)
43 {
44     return (interface_info *)handle;
45 }
46 
getWifiHandle(wifi_interface_handle handle)47 wifi_handle getWifiHandle(wifi_interface_handle handle)
48 {
49     return getIfaceInfo(handle)->handle;
50 }
51 
getHalInfo(wifi_handle handle)52 hal_info *getHalInfo(wifi_handle handle)
53 {
54     return (hal_info *)handle;
55 }
56 
getHalInfo(wifi_interface_handle handle)57 hal_info *getHalInfo(wifi_interface_handle handle)
58 {
59     return getHalInfo(getWifiHandle(handle));
60 }
61 
getWifiHandle(hal_info * info)62 wifi_handle getWifiHandle(hal_info *info)
63 {
64     return (wifi_handle)info;
65 }
66 
getIfaceHandle(interface_info * info)67 wifi_interface_handle getIfaceHandle(interface_info *info)
68 {
69     return (wifi_interface_handle)info;
70 }
71 
wifi_register_handler(wifi_handle handle,int cmd,nl_recvmsg_msg_cb_t func,void * arg)72 wifi_error wifi_register_handler(wifi_handle handle, int cmd, nl_recvmsg_msg_cb_t func, void *arg)
73 {
74     hal_info *info = (hal_info *)handle;
75 
76     /* TODO: check for multiple handlers? */
77     pthread_mutex_lock(&info->cb_lock);
78 
79     wifi_error result = WIFI_ERROR_OUT_OF_MEMORY;
80 
81     if (info->num_event_cb < info->alloc_event_cb) {
82         info->event_cb[info->num_event_cb].nl_cmd  = cmd;
83         info->event_cb[info->num_event_cb].vendor_id  = 0;
84         info->event_cb[info->num_event_cb].vendor_subcmd  = 0;
85         info->event_cb[info->num_event_cb].cb_func = func;
86         info->event_cb[info->num_event_cb].cb_arg  = arg;
87         ALOGV("Successfully added event handler %p:%p for command %d at %d",
88                 arg, func, cmd, info->num_event_cb);
89         info->num_event_cb++;
90         result = WIFI_SUCCESS;
91     }
92 
93     pthread_mutex_unlock(&info->cb_lock);
94     return result;
95 }
96 
wifi_register_vendor_handler(wifi_handle handle,uint32_t id,int subcmd,nl_recvmsg_msg_cb_t func,void * arg)97 wifi_error wifi_register_vendor_handler(wifi_handle handle,
98         uint32_t id, int subcmd, nl_recvmsg_msg_cb_t func, void *arg)
99 {
100     hal_info *info = (hal_info *)handle;
101 
102     /* TODO: check for multiple handlers? */
103     pthread_mutex_lock(&info->cb_lock);
104 
105     wifi_error result = WIFI_ERROR_OUT_OF_MEMORY;
106 
107     if (info->num_event_cb < info->alloc_event_cb) {
108         /* To avoid an unwanted duplication of the record, find first.
109          * Update it if the same record is already exist.
110          * KEY => [nl_cmd, vendor_id, vendor_subcmd]
111          */
112         int i = 0;
113         bool is_update = false;
114         for (i = 0; i < info->num_event_cb; i++) {
115             if ((info->event_cb[i].nl_cmd == NL80211_CMD_VENDOR) &&
116                     (info->event_cb[i].vendor_id == id) &&
117                     (info->event_cb[i].vendor_subcmd == subcmd)) {
118                 is_update = true;
119                 break;
120             }
121         }
122 
123         if (is_update) {
124             info->event_cb[i].cb_func = func;
125             info->event_cb[i].cb_arg = arg;
126         } else {
127             info->event_cb[info->num_event_cb].nl_cmd  = NL80211_CMD_VENDOR;
128             info->event_cb[info->num_event_cb].vendor_id  = id;
129             info->event_cb[info->num_event_cb].vendor_subcmd  = subcmd;
130             info->event_cb[info->num_event_cb].cb_func = func;
131             info->event_cb[info->num_event_cb].cb_arg  = arg;
132             info->num_event_cb++;
133         }
134         ALOGI("%s ""event handler %p:%p for vendor 0x%0x and subcmd 0x%0x at %d",
135             is_update ? "Updated" : "Added", arg, func, id, subcmd, info->num_event_cb);
136         result = WIFI_SUCCESS;
137     }
138 
139     pthread_mutex_unlock(&info->cb_lock);
140     return result;
141 }
142 
wifi_unregister_handler(wifi_handle handle,int cmd)143 void wifi_unregister_handler(wifi_handle handle, int cmd)
144 {
145     hal_info *info = (hal_info *)handle;
146 
147     if (cmd == NL80211_CMD_VENDOR) {
148         ALOGE("Must use wifi_unregister_vendor_handler to remove vendor handlers");
149         return;
150     }
151 
152     pthread_mutex_lock(&info->cb_lock);
153 
154     for (int i = 0; i < info->num_event_cb; i++) {
155         if (info->event_cb[i].nl_cmd == cmd) {
156             ALOGV("Successfully removed event handler %p:%p for cmd = 0x%0x from %d",
157                     info->event_cb[i].cb_arg, info->event_cb[i].cb_func, cmd, i);
158 
159             memmove(&info->event_cb[i], &info->event_cb[i+1],
160                 (info->num_event_cb - i - 1) * sizeof(cb_info));
161             info->num_event_cb--;
162             break;
163         }
164     }
165 
166     pthread_mutex_unlock(&info->cb_lock);
167 }
168 
wifi_unregister_vendor_handler_without_lock(wifi_handle handle,uint32_t id,int subcmd)169 void wifi_unregister_vendor_handler_without_lock(wifi_handle handle, uint32_t id, int subcmd)
170 {
171     hal_info *info = (hal_info *)handle;
172 
173     for (int i = 0; i < info->num_event_cb; i++) {
174         if (info->event_cb[i].nl_cmd == NL80211_CMD_VENDOR
175                 && info->event_cb[i].vendor_id == id
176                 && info->event_cb[i].vendor_subcmd == subcmd) {
177             ALOGI("Successfully removed event handler %p:%p for vendor 0x%0x, subcmd 0x%0x from %d",
178                     info->event_cb[i].cb_arg, info->event_cb[i].cb_func, id, subcmd, i);
179             memmove(&info->event_cb[i], &info->event_cb[i+1],
180                 (info->num_event_cb - i - 1) * sizeof(cb_info));
181             info->num_event_cb--;
182             break;
183         }
184     }
185 }
186 
wifi_unregister_vendor_handler(wifi_handle handle,uint32_t id,int subcmd)187 void wifi_unregister_vendor_handler(wifi_handle handle, uint32_t id, int subcmd)
188 {
189     hal_info *info = (hal_info *)handle;
190 
191     pthread_mutex_lock(&info->cb_lock);
192     wifi_unregister_vendor_handler_without_lock(handle, id, subcmd);
193     pthread_mutex_unlock(&info->cb_lock);
194 }
195 
wifi_register_cmd(wifi_handle handle,int id,WifiCommand * cmd)196 wifi_error wifi_register_cmd(wifi_handle handle, int id, WifiCommand *cmd)
197 {
198     hal_info *info = (hal_info *)handle;
199 
200     ALOGV("registering command %d", id);
201 
202     wifi_error result = WIFI_ERROR_OUT_OF_MEMORY;
203 
204     if (info->num_cmd < info->alloc_cmd) {
205         info->cmd[info->num_cmd].id   = id;
206         info->cmd[info->num_cmd].cmd  = cmd;
207         ALOGV("Successfully added command %d: %p at %d", id, cmd, info->num_cmd);
208         info->num_cmd++;
209         result = WIFI_SUCCESS;
210     } else {
211         ALOGE("Failed to add command %d: %p at %d, reached max limit %d",
212                 id, cmd, info->num_cmd, info->alloc_cmd);
213     }
214 
215     return result;
216 }
217 
wifi_unregister_cmd(wifi_handle handle,int id)218 WifiCommand *wifi_unregister_cmd(wifi_handle handle, int id)
219 {
220     hal_info *info = (hal_info *)handle;
221 
222     ALOGV("un-registering command %d", id);
223 
224     WifiCommand *cmd = NULL;
225 
226     for (int i = 0; i < info->num_cmd; i++) {
227         if (info->cmd[i].id == id) {
228             cmd = info->cmd[i].cmd;
229             memmove(&info->cmd[i], &info->cmd[i+1], (info->num_cmd - i - 1) * sizeof(cmd_info));
230             info->num_cmd--;
231             ALOGV("Successfully removed command %d: %p from %d", id, cmd, i);
232             break;
233         }
234     }
235 
236     if (!cmd) {
237         ALOGI("Failed to remove command %d: %p", id, cmd);
238     }
239 
240     return cmd;
241 }
242 
wifi_get_cmd(wifi_handle handle,int id)243 WifiCommand *wifi_get_cmd(wifi_handle handle, int id)
244 {
245     hal_info *info = (hal_info *)handle;
246 
247     WifiCommand *cmd = NULL;
248 
249     for (int i = 0; i < info->num_cmd; i++) {
250         if (info->cmd[i].id == id) {
251             cmd = info->cmd[i].cmd;
252             break;
253         }
254     }
255 
256     return cmd;
257 }
258 
wifi_unregister_cmd(wifi_handle handle,WifiCommand * cmd)259 void wifi_unregister_cmd(wifi_handle handle, WifiCommand *cmd)
260 {
261     hal_info *info = (hal_info *)handle;
262 
263     for (int i = 0; i < info->num_cmd; i++) {
264         if (info->cmd[i].cmd == cmd) {
265             int id = info->cmd[i].id;
266             memmove(&info->cmd[i], &info->cmd[i+1], (info->num_cmd - i - 1) * sizeof(cmd_info));
267             info->num_cmd--;
268             ALOGV("Successfully removed command %d: %p from %d", id, cmd, i);
269             break;
270         }
271     }
272 }
273 
wifi_cancel_cmd(wifi_request_id id,wifi_interface_handle iface)274 wifi_error wifi_cancel_cmd(wifi_request_id id, wifi_interface_handle iface)
275 {
276     wifi_handle handle = getWifiHandle(iface);
277 
278     WifiCommand *cmd = wifi_unregister_cmd(handle, id);
279     ALOGV("Cancel WifiCommand = %p", cmd);
280     if (cmd) {
281         cmd->cancel();
282         cmd->releaseRef();
283         return WIFI_SUCCESS;
284     }
285 
286     return WIFI_ERROR_INVALID_ARGS;
287 }
288 
wifi_get_cancel_cmd(wifi_request_id id,wifi_interface_handle iface)289 wifi_error wifi_get_cancel_cmd(wifi_request_id id, wifi_interface_handle iface)
290 {
291     wifi_handle handle = getWifiHandle(iface);
292     WifiCommand *cmd = wifi_get_cmd(handle, id);
293     ALOGV("Get Cancel WifiCommand = %p", cmd);
294     if (cmd) {
295         cmd->cancel();
296         cmd->releaseRef();
297         return WIFI_SUCCESS;
298     }
299 
300     return WIFI_ERROR_INVALID_ARGS;
301 }
set_hautil_mode(bool util_mode)302 void set_hautil_mode(bool util_mode)
303 {
304     halutil_mode = util_mode;
305 }
get_halutil_mode()306 bool get_halutil_mode()
307 {
308     return halutil_mode;
309 }
310 /* pretty hex print a contiguous buffer */
prhex(const char * msg,u8 * buf,u32 nbytes)311 void prhex(const char *msg, u8 *buf, u32 nbytes)
312 {
313     char line[128];
314     char *p;
315     int len = sizeof(line);
316     int nchar;
317     u32 i;
318 
319     if (msg && (msg[0] != '\0')) {
320         ALOGE("%s: len %d\n", msg, nbytes);
321     }
322 
323     p = line;
324     for (i = 0; i < nbytes; i++) {
325         if (i % 16 == 0) {
326             nchar = snprintf(p, len, "  %04d: ", i);    /* line prefix */
327             p += nchar;
328             len -= nchar;
329         }
330 
331         if (len > 0) {
332             nchar = snprintf(p, len, "%02x ", buf[i]);
333             p += nchar;
334             len -= nchar;
335         }
336 
337         if (i % 16 == 15) {
338             ALOGE("%s\n", line);       /* flush line */
339             p = line;
340             len = sizeof(line);
341         }
342     }
343 
344     /* flush last partial line */
345     if (p != line) {
346         ALOGE("%s\n", line);
347     }
348 }
349