1 // SPDX-License-Identifier: BSD-3-Clause-Clear
2 /*
3 * Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
4 */
5
6 #include <linux/module.h>
7 #include <linux/slab.h>
8 #include <linux/remoteproc.h>
9 #include <linux/firmware.h>
10 #include <linux/of.h>
11 #include "core.h"
12 #include "dp_tx.h"
13 #include "dp_rx.h"
14 #include "debug.h"
15 #include "hif.h"
16 #include "wow.h"
17
18 unsigned int ath11k_debug_mask;
19 EXPORT_SYMBOL(ath11k_debug_mask);
20 module_param_named(debug_mask, ath11k_debug_mask, uint, 0644);
21 MODULE_PARM_DESC(debug_mask, "Debugging mask");
22
23 static unsigned int ath11k_crypto_mode;
24 module_param_named(crypto_mode, ath11k_crypto_mode, uint, 0644);
25 MODULE_PARM_DESC(crypto_mode, "crypto mode: 0-hardware, 1-software");
26
27 /* frame mode values are mapped as per enum ath11k_hw_txrx_mode */
28 unsigned int ath11k_frame_mode = ATH11K_HW_TXRX_NATIVE_WIFI;
29 module_param_named(frame_mode, ath11k_frame_mode, uint, 0644);
30 MODULE_PARM_DESC(frame_mode,
31 "Datapath frame mode (0: raw, 1: native wifi (default), 2: ethernet)");
32
33 static const struct ath11k_hw_params ath11k_hw_params[] = {
34 {
35 .hw_rev = ATH11K_HW_IPQ8074,
36 .name = "ipq8074 hw2.0",
37 .fw = {
38 .dir = "IPQ8074/hw2.0",
39 .board_size = 256 * 1024,
40 .cal_size = 256 * 1024,
41 },
42 .max_radios = 3,
43 .bdf_addr = 0x4B0C0000,
44 .hw_ops = &ipq8074_ops,
45 .ring_mask = &ath11k_hw_ring_mask_ipq8074,
46 .internal_sleep_clock = false,
47 .regs = &ipq8074_regs,
48 .qmi_service_ins_id = ATH11K_QMI_WLFW_SERVICE_INS_ID_V01_IPQ8074,
49 .host_ce_config = ath11k_host_ce_config_ipq8074,
50 .ce_count = 12,
51 .target_ce_config = ath11k_target_ce_config_wlan_ipq8074,
52 .target_ce_count = 11,
53 .svc_to_ce_map = ath11k_target_service_to_ce_map_wlan_ipq8074,
54 .svc_to_ce_map_len = 21,
55 .single_pdev_only = false,
56 .rxdma1_enable = true,
57 .num_rxmda_per_pdev = 1,
58 .rx_mac_buf_ring = false,
59 .vdev_start_delay = false,
60 .htt_peer_map_v2 = true,
61 .tcl_0_only = false,
62 .spectral_fft_sz = 2,
63
64 .interface_modes = BIT(NL80211_IFTYPE_STATION) |
65 BIT(NL80211_IFTYPE_AP) |
66 BIT(NL80211_IFTYPE_MESH_POINT),
67 .supports_monitor = true,
68 .supports_shadow_regs = false,
69 .idle_ps = false,
70 .cold_boot_calib = true,
71 .supports_suspend = false,
72 .hal_desc_sz = sizeof(struct hal_rx_desc_ipq8074),
73 .fix_l1ss = true,
74 .wakeup_mhi = false,
75 },
76 {
77 .hw_rev = ATH11K_HW_IPQ6018_HW10,
78 .name = "ipq6018 hw1.0",
79 .fw = {
80 .dir = "IPQ6018/hw1.0",
81 .board_size = 256 * 1024,
82 .cal_size = 256 * 1024,
83 },
84 .max_radios = 2,
85 .bdf_addr = 0x4ABC0000,
86 .hw_ops = &ipq6018_ops,
87 .ring_mask = &ath11k_hw_ring_mask_ipq8074,
88 .internal_sleep_clock = false,
89 .regs = &ipq8074_regs,
90 .qmi_service_ins_id = ATH11K_QMI_WLFW_SERVICE_INS_ID_V01_IPQ8074,
91 .host_ce_config = ath11k_host_ce_config_ipq8074,
92 .ce_count = 12,
93 .target_ce_config = ath11k_target_ce_config_wlan_ipq8074,
94 .target_ce_count = 11,
95 .svc_to_ce_map = ath11k_target_service_to_ce_map_wlan_ipq6018,
96 .svc_to_ce_map_len = 19,
97 .single_pdev_only = false,
98 .rxdma1_enable = true,
99 .num_rxmda_per_pdev = 1,
100 .rx_mac_buf_ring = false,
101 .vdev_start_delay = false,
102 .htt_peer_map_v2 = true,
103 .tcl_0_only = false,
104 .spectral_fft_sz = 4,
105
106 .interface_modes = BIT(NL80211_IFTYPE_STATION) |
107 BIT(NL80211_IFTYPE_AP) |
108 BIT(NL80211_IFTYPE_MESH_POINT),
109 .supports_monitor = true,
110 .supports_shadow_regs = false,
111 .idle_ps = false,
112 .cold_boot_calib = true,
113 .supports_suspend = false,
114 .hal_desc_sz = sizeof(struct hal_rx_desc_ipq8074),
115 .fix_l1ss = true,
116 .wakeup_mhi = false,
117 },
118 {
119 .name = "qca6390 hw2.0",
120 .hw_rev = ATH11K_HW_QCA6390_HW20,
121 .fw = {
122 .dir = "QCA6390/hw2.0",
123 .board_size = 256 * 1024,
124 .cal_size = 256 * 1024,
125 },
126 .max_radios = 3,
127 .bdf_addr = 0x4B0C0000,
128 .hw_ops = &qca6390_ops,
129 .ring_mask = &ath11k_hw_ring_mask_qca6390,
130 .internal_sleep_clock = true,
131 .regs = &qca6390_regs,
132 .qmi_service_ins_id = ATH11K_QMI_WLFW_SERVICE_INS_ID_V01_QCA6390,
133 .host_ce_config = ath11k_host_ce_config_qca6390,
134 .ce_count = 9,
135 .target_ce_config = ath11k_target_ce_config_wlan_qca6390,
136 .target_ce_count = 9,
137 .svc_to_ce_map = ath11k_target_service_to_ce_map_wlan_qca6390,
138 .svc_to_ce_map_len = 14,
139 .single_pdev_only = true,
140 .rxdma1_enable = false,
141 .num_rxmda_per_pdev = 2,
142 .rx_mac_buf_ring = true,
143 .vdev_start_delay = true,
144 .htt_peer_map_v2 = false,
145 .tcl_0_only = true,
146 .spectral_fft_sz = 0,
147
148 .interface_modes = BIT(NL80211_IFTYPE_STATION) |
149 BIT(NL80211_IFTYPE_AP),
150 .supports_monitor = false,
151 .supports_shadow_regs = true,
152 .idle_ps = true,
153 .cold_boot_calib = false,
154 .supports_suspend = true,
155 .hal_desc_sz = sizeof(struct hal_rx_desc_ipq8074),
156 .fix_l1ss = true,
157 .wakeup_mhi = true,
158 },
159 {
160 .name = "qcn9074 hw1.0",
161 .hw_rev = ATH11K_HW_QCN9074_HW10,
162 .fw = {
163 .dir = "QCN9074/hw1.0",
164 .board_size = 256 * 1024,
165 .cal_size = 256 * 1024,
166 },
167 .max_radios = 1,
168 .single_pdev_only = false,
169 .qmi_service_ins_id = ATH11K_QMI_WLFW_SERVICE_INS_ID_V01_QCN9074,
170 .hw_ops = &qcn9074_ops,
171 .ring_mask = &ath11k_hw_ring_mask_qcn9074,
172 .internal_sleep_clock = false,
173 .regs = &qcn9074_regs,
174 .host_ce_config = ath11k_host_ce_config_qcn9074,
175 .ce_count = 6,
176 .target_ce_config = ath11k_target_ce_config_wlan_qcn9074,
177 .target_ce_count = 9,
178 .svc_to_ce_map = ath11k_target_service_to_ce_map_wlan_qcn9074,
179 .svc_to_ce_map_len = 18,
180 .rxdma1_enable = true,
181 .num_rxmda_per_pdev = 1,
182 .rx_mac_buf_ring = false,
183 .vdev_start_delay = false,
184 .htt_peer_map_v2 = true,
185 .tcl_0_only = false,
186 .interface_modes = BIT(NL80211_IFTYPE_STATION) |
187 BIT(NL80211_IFTYPE_AP) |
188 BIT(NL80211_IFTYPE_MESH_POINT),
189 .supports_monitor = true,
190 .supports_shadow_regs = false,
191 .idle_ps = false,
192 .cold_boot_calib = false,
193 .supports_suspend = false,
194 .hal_desc_sz = sizeof(struct hal_rx_desc_qcn9074),
195 .fix_l1ss = true,
196 .wakeup_mhi = false,
197 },
198 {
199 .name = "wcn6855 hw2.0",
200 .hw_rev = ATH11K_HW_WCN6855_HW20,
201 .fw = {
202 .dir = "WCN6855/hw2.0",
203 .board_size = 256 * 1024,
204 .cal_size = 256 * 1024,
205 },
206 .max_radios = 3,
207 .bdf_addr = 0x4B0C0000,
208 .hw_ops = &wcn6855_ops,
209 .ring_mask = &ath11k_hw_ring_mask_qca6390,
210 .internal_sleep_clock = true,
211 .regs = &wcn6855_regs,
212 .qmi_service_ins_id = ATH11K_QMI_WLFW_SERVICE_INS_ID_V01_QCA6390,
213 .host_ce_config = ath11k_host_ce_config_qca6390,
214 .ce_count = 9,
215 .target_ce_config = ath11k_target_ce_config_wlan_qca6390,
216 .target_ce_count = 9,
217 .svc_to_ce_map = ath11k_target_service_to_ce_map_wlan_qca6390,
218 .svc_to_ce_map_len = 14,
219 .single_pdev_only = true,
220 .rxdma1_enable = false,
221 .num_rxmda_per_pdev = 2,
222 .rx_mac_buf_ring = true,
223 .vdev_start_delay = true,
224 .htt_peer_map_v2 = false,
225 .tcl_0_only = true,
226 .spectral_fft_sz = 0,
227
228 .interface_modes = BIT(NL80211_IFTYPE_STATION) |
229 BIT(NL80211_IFTYPE_AP),
230 .supports_monitor = false,
231 .supports_shadow_regs = true,
232 .idle_ps = true,
233 .cold_boot_calib = false,
234 .supports_suspend = true,
235 .hal_desc_sz = sizeof(struct hal_rx_desc_wcn6855),
236 .fix_l1ss = false,
237 .wakeup_mhi = true,
238 },
239 };
240
ath11k_core_suspend(struct ath11k_base * ab)241 int ath11k_core_suspend(struct ath11k_base *ab)
242 {
243 int ret;
244
245 if (!ab->hw_params.supports_suspend)
246 return -EOPNOTSUPP;
247
248 /* TODO: there can frames in queues so for now add delay as a hack.
249 * Need to implement to handle and remove this delay.
250 */
251 msleep(500);
252
253 ret = ath11k_dp_rx_pktlog_stop(ab, true);
254 if (ret) {
255 ath11k_warn(ab, "failed to stop dp rx (and timer) pktlog during suspend: %d\n",
256 ret);
257 return ret;
258 }
259
260 ret = ath11k_wow_enable(ab);
261 if (ret) {
262 ath11k_warn(ab, "failed to enable wow during suspend: %d\n", ret);
263 return ret;
264 }
265
266 ret = ath11k_dp_rx_pktlog_stop(ab, false);
267 if (ret) {
268 ath11k_warn(ab, "failed to stop dp rx pktlog during suspend: %d\n",
269 ret);
270 return ret;
271 }
272
273 ath11k_ce_stop_shadow_timers(ab);
274 ath11k_dp_stop_shadow_timers(ab);
275
276 ath11k_hif_irq_disable(ab);
277 ath11k_hif_ce_irq_disable(ab);
278
279 ret = ath11k_hif_suspend(ab);
280 if (ret) {
281 ath11k_warn(ab, "failed to suspend hif: %d\n", ret);
282 return ret;
283 }
284
285 return 0;
286 }
287 EXPORT_SYMBOL(ath11k_core_suspend);
288
ath11k_core_resume(struct ath11k_base * ab)289 int ath11k_core_resume(struct ath11k_base *ab)
290 {
291 int ret;
292
293 if (!ab->hw_params.supports_suspend)
294 return -EOPNOTSUPP;
295
296 ret = ath11k_hif_resume(ab);
297 if (ret) {
298 ath11k_warn(ab, "failed to resume hif during resume: %d\n", ret);
299 return ret;
300 }
301
302 ath11k_hif_ce_irq_enable(ab);
303 ath11k_hif_irq_enable(ab);
304
305 ret = ath11k_dp_rx_pktlog_start(ab);
306 if (ret) {
307 ath11k_warn(ab, "failed to start rx pktlog during resume: %d\n",
308 ret);
309 return ret;
310 }
311
312 ret = ath11k_wow_wakeup(ab);
313 if (ret) {
314 ath11k_warn(ab, "failed to wakeup wow during resume: %d\n", ret);
315 return ret;
316 }
317
318 return 0;
319 }
320 EXPORT_SYMBOL(ath11k_core_resume);
321
ath11k_core_check_dt(struct ath11k_base * ab)322 int ath11k_core_check_dt(struct ath11k_base *ab)
323 {
324 size_t max_len = sizeof(ab->qmi.target.bdf_ext);
325 const char *variant = NULL;
326 struct device_node *node;
327
328 node = ab->dev->of_node;
329 if (!node)
330 return -ENOENT;
331
332 of_property_read_string(node, "qcom,ath11k-calibration-variant",
333 &variant);
334 if (!variant)
335 return -ENODATA;
336
337 if (strscpy(ab->qmi.target.bdf_ext, variant, max_len) < 0)
338 ath11k_dbg(ab, ATH11K_DBG_BOOT,
339 "bdf variant string is longer than the buffer can accommodate (variant: %s)\n",
340 variant);
341
342 return 0;
343 }
344
ath11k_core_create_board_name(struct ath11k_base * ab,char * name,size_t name_len)345 static int ath11k_core_create_board_name(struct ath11k_base *ab, char *name,
346 size_t name_len)
347 {
348 /* strlen(',variant=') + strlen(ab->qmi.target.bdf_ext) */
349 char variant[9 + ATH11K_QMI_BDF_EXT_STR_LENGTH] = { 0 };
350
351 if (ab->qmi.target.bdf_ext[0] != '\0')
352 scnprintf(variant, sizeof(variant), ",variant=%s",
353 ab->qmi.target.bdf_ext);
354
355 switch (ab->id.bdf_search) {
356 case ATH11K_BDF_SEARCH_BUS_AND_BOARD:
357 scnprintf(name, name_len,
358 "bus=%s,vendor=%04x,device=%04x,subsystem-vendor=%04x,subsystem-device=%04x,qmi-chip-id=%d,qmi-board-id=%d%s",
359 ath11k_bus_str(ab->hif.bus),
360 ab->id.vendor, ab->id.device,
361 ab->id.subsystem_vendor,
362 ab->id.subsystem_device,
363 ab->qmi.target.chip_id,
364 ab->qmi.target.board_id,
365 variant);
366 break;
367 default:
368 scnprintf(name, name_len,
369 "bus=%s,qmi-chip-id=%d,qmi-board-id=%d%s",
370 ath11k_bus_str(ab->hif.bus),
371 ab->qmi.target.chip_id,
372 ab->qmi.target.board_id, variant);
373 break;
374 }
375
376 ath11k_dbg(ab, ATH11K_DBG_BOOT, "boot using board name '%s'\n", name);
377
378 return 0;
379 }
380
ath11k_core_firmware_request(struct ath11k_base * ab,const char * file)381 const struct firmware *ath11k_core_firmware_request(struct ath11k_base *ab,
382 const char *file)
383 {
384 const struct firmware *fw;
385 char path[100];
386 int ret;
387
388 if (file == NULL)
389 return ERR_PTR(-ENOENT);
390
391 ath11k_core_create_firmware_path(ab, file, path, sizeof(path));
392
393 ret = firmware_request_nowarn(&fw, path, ab->dev);
394 if (ret)
395 return ERR_PTR(ret);
396
397 ath11k_dbg(ab, ATH11K_DBG_BOOT, "boot firmware request %s size %zu\n",
398 path, fw->size);
399
400 return fw;
401 }
402
ath11k_core_free_bdf(struct ath11k_base * ab,struct ath11k_board_data * bd)403 void ath11k_core_free_bdf(struct ath11k_base *ab, struct ath11k_board_data *bd)
404 {
405 if (!IS_ERR(bd->fw))
406 release_firmware(bd->fw);
407
408 memset(bd, 0, sizeof(*bd));
409 }
410
ath11k_core_parse_bd_ie_board(struct ath11k_base * ab,struct ath11k_board_data * bd,const void * buf,size_t buf_len,const char * boardname,int bd_ie_type)411 static int ath11k_core_parse_bd_ie_board(struct ath11k_base *ab,
412 struct ath11k_board_data *bd,
413 const void *buf, size_t buf_len,
414 const char *boardname,
415 int bd_ie_type)
416 {
417 const struct ath11k_fw_ie *hdr;
418 bool name_match_found;
419 int ret, board_ie_id;
420 size_t board_ie_len;
421 const void *board_ie_data;
422
423 name_match_found = false;
424
425 /* go through ATH11K_BD_IE_BOARD_ elements */
426 while (buf_len > sizeof(struct ath11k_fw_ie)) {
427 hdr = buf;
428 board_ie_id = le32_to_cpu(hdr->id);
429 board_ie_len = le32_to_cpu(hdr->len);
430 board_ie_data = hdr->data;
431
432 buf_len -= sizeof(*hdr);
433 buf += sizeof(*hdr);
434
435 if (buf_len < ALIGN(board_ie_len, 4)) {
436 ath11k_err(ab, "invalid ATH11K_BD_IE_BOARD length: %zu < %zu\n",
437 buf_len, ALIGN(board_ie_len, 4));
438 ret = -EINVAL;
439 goto out;
440 }
441
442 switch (board_ie_id) {
443 case ATH11K_BD_IE_BOARD_NAME:
444 ath11k_dbg_dump(ab, ATH11K_DBG_BOOT, "board name", "",
445 board_ie_data, board_ie_len);
446
447 if (board_ie_len != strlen(boardname))
448 break;
449
450 ret = memcmp(board_ie_data, boardname, strlen(boardname));
451 if (ret)
452 break;
453
454 name_match_found = true;
455 ath11k_dbg(ab, ATH11K_DBG_BOOT,
456 "boot found match for name '%s'",
457 boardname);
458 break;
459 case ATH11K_BD_IE_BOARD_DATA:
460 if (!name_match_found)
461 /* no match found */
462 break;
463
464 ath11k_dbg(ab, ATH11K_DBG_BOOT,
465 "boot found board data for '%s'", boardname);
466
467 bd->data = board_ie_data;
468 bd->len = board_ie_len;
469
470 ret = 0;
471 goto out;
472 default:
473 ath11k_warn(ab, "unknown ATH11K_BD_IE_BOARD found: %d\n",
474 board_ie_id);
475 break;
476 }
477
478 /* jump over the padding */
479 board_ie_len = ALIGN(board_ie_len, 4);
480
481 buf_len -= board_ie_len;
482 buf += board_ie_len;
483 }
484
485 /* no match found */
486 ret = -ENOENT;
487
488 out:
489 return ret;
490 }
491
ath11k_core_fetch_board_data_api_n(struct ath11k_base * ab,struct ath11k_board_data * bd,const char * boardname)492 static int ath11k_core_fetch_board_data_api_n(struct ath11k_base *ab,
493 struct ath11k_board_data *bd,
494 const char *boardname)
495 {
496 size_t len, magic_len;
497 const u8 *data;
498 char *filename, filepath[100];
499 size_t ie_len;
500 struct ath11k_fw_ie *hdr;
501 int ret, ie_id;
502
503 filename = ATH11K_BOARD_API2_FILE;
504
505 if (!bd->fw)
506 bd->fw = ath11k_core_firmware_request(ab, filename);
507
508 if (IS_ERR(bd->fw))
509 return PTR_ERR(bd->fw);
510
511 data = bd->fw->data;
512 len = bd->fw->size;
513
514 ath11k_core_create_firmware_path(ab, filename,
515 filepath, sizeof(filepath));
516
517 /* magic has extra null byte padded */
518 magic_len = strlen(ATH11K_BOARD_MAGIC) + 1;
519 if (len < magic_len) {
520 ath11k_err(ab, "failed to find magic value in %s, file too short: %zu\n",
521 filepath, len);
522 ret = -EINVAL;
523 goto err;
524 }
525
526 if (memcmp(data, ATH11K_BOARD_MAGIC, magic_len)) {
527 ath11k_err(ab, "found invalid board magic\n");
528 ret = -EINVAL;
529 goto err;
530 }
531
532 /* magic is padded to 4 bytes */
533 magic_len = ALIGN(magic_len, 4);
534 if (len < magic_len) {
535 ath11k_err(ab, "failed: %s too small to contain board data, len: %zu\n",
536 filepath, len);
537 ret = -EINVAL;
538 goto err;
539 }
540
541 data += magic_len;
542 len -= magic_len;
543
544 while (len > sizeof(struct ath11k_fw_ie)) {
545 hdr = (struct ath11k_fw_ie *)data;
546 ie_id = le32_to_cpu(hdr->id);
547 ie_len = le32_to_cpu(hdr->len);
548
549 len -= sizeof(*hdr);
550 data = hdr->data;
551
552 if (len < ALIGN(ie_len, 4)) {
553 ath11k_err(ab, "invalid length for board ie_id %d ie_len %zu len %zu\n",
554 ie_id, ie_len, len);
555 ret = -EINVAL;
556 goto err;
557 }
558
559 switch (ie_id) {
560 case ATH11K_BD_IE_BOARD:
561 ret = ath11k_core_parse_bd_ie_board(ab, bd, data,
562 ie_len,
563 boardname,
564 ATH11K_BD_IE_BOARD);
565 if (ret == -ENOENT)
566 /* no match found, continue */
567 break;
568 else if (ret)
569 /* there was an error, bail out */
570 goto err;
571 /* either found or error, so stop searching */
572 goto out;
573 }
574
575 /* jump over the padding */
576 ie_len = ALIGN(ie_len, 4);
577
578 len -= ie_len;
579 data += ie_len;
580 }
581
582 out:
583 if (!bd->data || !bd->len) {
584 ath11k_err(ab,
585 "failed to fetch board data for %s from %s\n",
586 boardname, filepath);
587 ret = -ENODATA;
588 goto err;
589 }
590
591 return 0;
592
593 err:
594 ath11k_core_free_bdf(ab, bd);
595 return ret;
596 }
597
ath11k_core_fetch_board_data_api_1(struct ath11k_base * ab,struct ath11k_board_data * bd)598 static int ath11k_core_fetch_board_data_api_1(struct ath11k_base *ab,
599 struct ath11k_board_data *bd)
600 {
601 bd->fw = ath11k_core_firmware_request(ab, ATH11K_DEFAULT_BOARD_FILE);
602 if (IS_ERR(bd->fw))
603 return PTR_ERR(bd->fw);
604
605 bd->data = bd->fw->data;
606 bd->len = bd->fw->size;
607
608 return 0;
609 }
610
611 #define BOARD_NAME_SIZE 200
ath11k_core_fetch_bdf(struct ath11k_base * ab,struct ath11k_board_data * bd)612 int ath11k_core_fetch_bdf(struct ath11k_base *ab, struct ath11k_board_data *bd)
613 {
614 char boardname[BOARD_NAME_SIZE];
615 int ret;
616
617 ret = ath11k_core_create_board_name(ab, boardname, BOARD_NAME_SIZE);
618 if (ret) {
619 ath11k_err(ab, "failed to create board name: %d", ret);
620 return ret;
621 }
622
623 ab->bd_api = 2;
624 ret = ath11k_core_fetch_board_data_api_n(ab, bd, boardname);
625 if (!ret)
626 goto success;
627
628 ab->bd_api = 1;
629 ret = ath11k_core_fetch_board_data_api_1(ab, bd);
630 if (ret) {
631 ath11k_err(ab, "failed to fetch board-2.bin or board.bin from %s\n",
632 ab->hw_params.fw.dir);
633 return ret;
634 }
635
636 success:
637 ath11k_dbg(ab, ATH11K_DBG_BOOT, "using board api %d\n", ab->bd_api);
638 return 0;
639 }
640
ath11k_core_stop(struct ath11k_base * ab)641 static void ath11k_core_stop(struct ath11k_base *ab)
642 {
643 if (!test_bit(ATH11K_FLAG_CRASH_FLUSH, &ab->dev_flags))
644 ath11k_qmi_firmware_stop(ab);
645
646 ath11k_hif_stop(ab);
647 ath11k_wmi_detach(ab);
648 ath11k_dp_pdev_reo_cleanup(ab);
649
650 /* De-Init of components as needed */
651 }
652
ath11k_core_soc_create(struct ath11k_base * ab)653 static int ath11k_core_soc_create(struct ath11k_base *ab)
654 {
655 int ret;
656
657 ret = ath11k_qmi_init_service(ab);
658 if (ret) {
659 ath11k_err(ab, "failed to initialize qmi :%d\n", ret);
660 return ret;
661 }
662
663 ret = ath11k_debugfs_soc_create(ab);
664 if (ret) {
665 ath11k_err(ab, "failed to create ath11k debugfs\n");
666 goto err_qmi_deinit;
667 }
668
669 ret = ath11k_hif_power_up(ab);
670 if (ret) {
671 ath11k_err(ab, "failed to power up :%d\n", ret);
672 goto err_debugfs_reg;
673 }
674
675 return 0;
676
677 err_debugfs_reg:
678 ath11k_debugfs_soc_destroy(ab);
679 err_qmi_deinit:
680 ath11k_qmi_deinit_service(ab);
681 return ret;
682 }
683
ath11k_core_soc_destroy(struct ath11k_base * ab)684 static void ath11k_core_soc_destroy(struct ath11k_base *ab)
685 {
686 ath11k_debugfs_soc_destroy(ab);
687 ath11k_dp_free(ab);
688 ath11k_reg_free(ab);
689 ath11k_qmi_deinit_service(ab);
690 }
691
ath11k_core_pdev_create(struct ath11k_base * ab)692 static int ath11k_core_pdev_create(struct ath11k_base *ab)
693 {
694 int ret;
695
696 ret = ath11k_debugfs_pdev_create(ab);
697 if (ret) {
698 ath11k_err(ab, "failed to create core pdev debugfs: %d\n", ret);
699 return ret;
700 }
701
702 ret = ath11k_dp_pdev_alloc(ab);
703 if (ret) {
704 ath11k_err(ab, "failed to attach DP pdev: %d\n", ret);
705 goto err_pdev_debug;
706 }
707
708 ret = ath11k_mac_register(ab);
709 if (ret) {
710 ath11k_err(ab, "failed register the radio with mac80211: %d\n", ret);
711 goto err_dp_pdev_free;
712 }
713
714 ret = ath11k_thermal_register(ab);
715 if (ret) {
716 ath11k_err(ab, "could not register thermal device: %d\n",
717 ret);
718 goto err_mac_unregister;
719 }
720
721 ret = ath11k_spectral_init(ab);
722 if (ret) {
723 ath11k_err(ab, "failed to init spectral %d\n", ret);
724 goto err_thermal_unregister;
725 }
726
727 return 0;
728
729 err_thermal_unregister:
730 ath11k_thermal_unregister(ab);
731 err_mac_unregister:
732 ath11k_mac_unregister(ab);
733 err_dp_pdev_free:
734 ath11k_dp_pdev_free(ab);
735 err_pdev_debug:
736 ath11k_debugfs_pdev_destroy(ab);
737
738 return ret;
739 }
740
ath11k_core_pdev_destroy(struct ath11k_base * ab)741 static void ath11k_core_pdev_destroy(struct ath11k_base *ab)
742 {
743 ath11k_spectral_deinit(ab);
744 ath11k_thermal_unregister(ab);
745 ath11k_mac_unregister(ab);
746 ath11k_hif_irq_disable(ab);
747 ath11k_dp_pdev_free(ab);
748 ath11k_debugfs_pdev_destroy(ab);
749 }
750
ath11k_core_start(struct ath11k_base * ab,enum ath11k_firmware_mode mode)751 static int ath11k_core_start(struct ath11k_base *ab,
752 enum ath11k_firmware_mode mode)
753 {
754 int ret;
755
756 ret = ath11k_qmi_firmware_start(ab, mode);
757 if (ret) {
758 ath11k_err(ab, "failed to attach wmi: %d\n", ret);
759 return ret;
760 }
761
762 ret = ath11k_wmi_attach(ab);
763 if (ret) {
764 ath11k_err(ab, "failed to attach wmi: %d\n", ret);
765 goto err_firmware_stop;
766 }
767
768 ret = ath11k_htc_init(ab);
769 if (ret) {
770 ath11k_err(ab, "failed to init htc: %d\n", ret);
771 goto err_wmi_detach;
772 }
773
774 ret = ath11k_hif_start(ab);
775 if (ret) {
776 ath11k_err(ab, "failed to start HIF: %d\n", ret);
777 goto err_wmi_detach;
778 }
779
780 ret = ath11k_htc_wait_target(&ab->htc);
781 if (ret) {
782 ath11k_err(ab, "failed to connect to HTC: %d\n", ret);
783 goto err_hif_stop;
784 }
785
786 ret = ath11k_dp_htt_connect(&ab->dp);
787 if (ret) {
788 ath11k_err(ab, "failed to connect to HTT: %d\n", ret);
789 goto err_hif_stop;
790 }
791
792 ret = ath11k_wmi_connect(ab);
793 if (ret) {
794 ath11k_err(ab, "failed to connect wmi: %d\n", ret);
795 goto err_hif_stop;
796 }
797
798 ret = ath11k_htc_start(&ab->htc);
799 if (ret) {
800 ath11k_err(ab, "failed to start HTC: %d\n", ret);
801 goto err_hif_stop;
802 }
803
804 ret = ath11k_wmi_wait_for_service_ready(ab);
805 if (ret) {
806 ath11k_err(ab, "failed to receive wmi service ready event: %d\n",
807 ret);
808 goto err_hif_stop;
809 }
810
811 ret = ath11k_mac_allocate(ab);
812 if (ret) {
813 ath11k_err(ab, "failed to create new hw device with mac80211 :%d\n",
814 ret);
815 goto err_hif_stop;
816 }
817
818 ath11k_dp_pdev_pre_alloc(ab);
819
820 ret = ath11k_dp_pdev_reo_setup(ab);
821 if (ret) {
822 ath11k_err(ab, "failed to initialize reo destination rings: %d\n", ret);
823 goto err_mac_destroy;
824 }
825
826 ret = ath11k_wmi_cmd_init(ab);
827 if (ret) {
828 ath11k_err(ab, "failed to send wmi init cmd: %d\n", ret);
829 goto err_reo_cleanup;
830 }
831
832 ret = ath11k_wmi_wait_for_unified_ready(ab);
833 if (ret) {
834 ath11k_err(ab, "failed to receive wmi unified ready event: %d\n",
835 ret);
836 goto err_reo_cleanup;
837 }
838
839 /* put hardware to DBS mode */
840 if (ab->hw_params.single_pdev_only) {
841 ret = ath11k_wmi_set_hw_mode(ab, WMI_HOST_HW_MODE_DBS);
842 if (ret) {
843 ath11k_err(ab, "failed to send dbs mode: %d\n", ret);
844 goto err_hif_stop;
845 }
846 }
847
848 ret = ath11k_dp_tx_htt_h2t_ver_req_msg(ab);
849 if (ret) {
850 ath11k_err(ab, "failed to send htt version request message: %d\n",
851 ret);
852 goto err_reo_cleanup;
853 }
854
855 return 0;
856
857 err_reo_cleanup:
858 ath11k_dp_pdev_reo_cleanup(ab);
859 err_mac_destroy:
860 ath11k_mac_destroy(ab);
861 err_hif_stop:
862 ath11k_hif_stop(ab);
863 err_wmi_detach:
864 ath11k_wmi_detach(ab);
865 err_firmware_stop:
866 ath11k_qmi_firmware_stop(ab);
867
868 return ret;
869 }
870
ath11k_core_qmi_firmware_ready(struct ath11k_base * ab)871 int ath11k_core_qmi_firmware_ready(struct ath11k_base *ab)
872 {
873 int ret;
874
875 ret = ath11k_ce_init_pipes(ab);
876 if (ret) {
877 ath11k_err(ab, "failed to initialize CE: %d\n", ret);
878 return ret;
879 }
880
881 ret = ath11k_dp_alloc(ab);
882 if (ret) {
883 ath11k_err(ab, "failed to init DP: %d\n", ret);
884 return ret;
885 }
886
887 switch (ath11k_crypto_mode) {
888 case ATH11K_CRYPT_MODE_SW:
889 set_bit(ATH11K_FLAG_HW_CRYPTO_DISABLED, &ab->dev_flags);
890 set_bit(ATH11K_FLAG_RAW_MODE, &ab->dev_flags);
891 break;
892 case ATH11K_CRYPT_MODE_HW:
893 clear_bit(ATH11K_FLAG_HW_CRYPTO_DISABLED, &ab->dev_flags);
894 clear_bit(ATH11K_FLAG_RAW_MODE, &ab->dev_flags);
895 break;
896 default:
897 ath11k_info(ab, "invalid crypto_mode: %d\n", ath11k_crypto_mode);
898 return -EINVAL;
899 }
900
901 if (ath11k_frame_mode == ATH11K_HW_TXRX_RAW)
902 set_bit(ATH11K_FLAG_RAW_MODE, &ab->dev_flags);
903
904 mutex_lock(&ab->core_lock);
905 ret = ath11k_core_start(ab, ATH11K_FIRMWARE_MODE_NORMAL);
906 if (ret) {
907 ath11k_err(ab, "failed to start core: %d\n", ret);
908 goto err_dp_free;
909 }
910
911 ret = ath11k_core_pdev_create(ab);
912 if (ret) {
913 ath11k_err(ab, "failed to create pdev core: %d\n", ret);
914 goto err_core_stop;
915 }
916 ath11k_hif_irq_enable(ab);
917 mutex_unlock(&ab->core_lock);
918
919 return 0;
920
921 err_core_stop:
922 ath11k_core_stop(ab);
923 ath11k_mac_destroy(ab);
924 err_dp_free:
925 ath11k_dp_free(ab);
926 mutex_unlock(&ab->core_lock);
927 return ret;
928 }
929
ath11k_core_reconfigure_on_crash(struct ath11k_base * ab)930 static int ath11k_core_reconfigure_on_crash(struct ath11k_base *ab)
931 {
932 int ret;
933
934 mutex_lock(&ab->core_lock);
935 ath11k_thermal_unregister(ab);
936 ath11k_hif_irq_disable(ab);
937 ath11k_dp_pdev_free(ab);
938 ath11k_spectral_deinit(ab);
939 ath11k_hif_stop(ab);
940 ath11k_wmi_detach(ab);
941 ath11k_dp_pdev_reo_cleanup(ab);
942 mutex_unlock(&ab->core_lock);
943
944 ath11k_dp_free(ab);
945 ath11k_hal_srng_deinit(ab);
946
947 ab->free_vdev_map = (1LL << (ab->num_radios * TARGET_NUM_VDEVS)) - 1;
948
949 ret = ath11k_hal_srng_init(ab);
950 if (ret)
951 return ret;
952
953 clear_bit(ATH11K_FLAG_CRASH_FLUSH, &ab->dev_flags);
954
955 ret = ath11k_core_qmi_firmware_ready(ab);
956 if (ret)
957 goto err_hal_srng_deinit;
958
959 clear_bit(ATH11K_FLAG_RECOVERY, &ab->dev_flags);
960
961 return 0;
962
963 err_hal_srng_deinit:
964 ath11k_hal_srng_deinit(ab);
965 return ret;
966 }
967
ath11k_core_halt(struct ath11k * ar)968 void ath11k_core_halt(struct ath11k *ar)
969 {
970 struct ath11k_base *ab = ar->ab;
971
972 lockdep_assert_held(&ar->conf_mutex);
973
974 ar->num_created_vdevs = 0;
975 ar->allocated_vdev_map = 0;
976
977 ath11k_mac_scan_finish(ar);
978 ath11k_mac_peer_cleanup_all(ar);
979 cancel_delayed_work_sync(&ar->scan.timeout);
980 cancel_work_sync(&ar->regd_update_work);
981
982 rcu_assign_pointer(ab->pdevs_active[ar->pdev_idx], NULL);
983 synchronize_rcu();
984 INIT_LIST_HEAD(&ar->arvifs);
985 idr_init(&ar->txmgmt_idr);
986 }
987
ath11k_core_restart(struct work_struct * work)988 static void ath11k_core_restart(struct work_struct *work)
989 {
990 struct ath11k_base *ab = container_of(work, struct ath11k_base, restart_work);
991 struct ath11k *ar;
992 struct ath11k_pdev *pdev;
993 int i, ret = 0;
994
995 spin_lock_bh(&ab->base_lock);
996 ab->stats.fw_crash_counter++;
997 spin_unlock_bh(&ab->base_lock);
998
999 for (i = 0; i < ab->num_radios; i++) {
1000 pdev = &ab->pdevs[i];
1001 ar = pdev->ar;
1002 if (!ar || ar->state == ATH11K_STATE_OFF)
1003 continue;
1004
1005 ieee80211_stop_queues(ar->hw);
1006 ath11k_mac_drain_tx(ar);
1007 complete(&ar->scan.started);
1008 complete(&ar->scan.completed);
1009 complete(&ar->peer_assoc_done);
1010 complete(&ar->peer_delete_done);
1011 complete(&ar->install_key_done);
1012 complete(&ar->vdev_setup_done);
1013 complete(&ar->vdev_delete_done);
1014 complete(&ar->bss_survey_done);
1015 complete(&ar->thermal.wmi_sync);
1016
1017 wake_up(&ar->dp.tx_empty_waitq);
1018 idr_for_each(&ar->txmgmt_idr,
1019 ath11k_mac_tx_mgmt_pending_free, ar);
1020 idr_destroy(&ar->txmgmt_idr);
1021 }
1022
1023 wake_up(&ab->wmi_ab.tx_credits_wq);
1024 wake_up(&ab->peer_mapping_wq);
1025
1026 ret = ath11k_core_reconfigure_on_crash(ab);
1027 if (ret) {
1028 ath11k_err(ab, "failed to reconfigure driver on crash recovery\n");
1029 return;
1030 }
1031
1032 for (i = 0; i < ab->num_radios; i++) {
1033 pdev = &ab->pdevs[i];
1034 ar = pdev->ar;
1035 if (!ar || ar->state == ATH11K_STATE_OFF)
1036 continue;
1037
1038 mutex_lock(&ar->conf_mutex);
1039
1040 switch (ar->state) {
1041 case ATH11K_STATE_ON:
1042 ar->state = ATH11K_STATE_RESTARTING;
1043 ath11k_core_halt(ar);
1044 ieee80211_restart_hw(ar->hw);
1045 break;
1046 case ATH11K_STATE_OFF:
1047 ath11k_warn(ab,
1048 "cannot restart radio %d that hasn't been started\n",
1049 i);
1050 break;
1051 case ATH11K_STATE_RESTARTING:
1052 break;
1053 case ATH11K_STATE_RESTARTED:
1054 ar->state = ATH11K_STATE_WEDGED;
1055 fallthrough;
1056 case ATH11K_STATE_WEDGED:
1057 ath11k_warn(ab,
1058 "device is wedged, will not restart radio %d\n", i);
1059 break;
1060 }
1061 mutex_unlock(&ar->conf_mutex);
1062 }
1063 complete(&ab->driver_recovery);
1064 }
1065
ath11k_init_hw_params(struct ath11k_base * ab)1066 static int ath11k_init_hw_params(struct ath11k_base *ab)
1067 {
1068 const struct ath11k_hw_params *hw_params = NULL;
1069 int i;
1070
1071 for (i = 0; i < ARRAY_SIZE(ath11k_hw_params); i++) {
1072 hw_params = &ath11k_hw_params[i];
1073
1074 if (hw_params->hw_rev == ab->hw_rev)
1075 break;
1076 }
1077
1078 if (i == ARRAY_SIZE(ath11k_hw_params)) {
1079 ath11k_err(ab, "Unsupported hardware version: 0x%x\n", ab->hw_rev);
1080 return -EINVAL;
1081 }
1082
1083 ab->hw_params = *hw_params;
1084
1085 ath11k_info(ab, "%s\n", ab->hw_params.name);
1086
1087 return 0;
1088 }
1089
ath11k_core_pre_init(struct ath11k_base * ab)1090 int ath11k_core_pre_init(struct ath11k_base *ab)
1091 {
1092 int ret;
1093
1094 ret = ath11k_init_hw_params(ab);
1095 if (ret) {
1096 ath11k_err(ab, "failed to get hw params: %d\n", ret);
1097 return ret;
1098 }
1099
1100 return 0;
1101 }
1102 EXPORT_SYMBOL(ath11k_core_pre_init);
1103
ath11k_core_init(struct ath11k_base * ab)1104 int ath11k_core_init(struct ath11k_base *ab)
1105 {
1106 int ret;
1107
1108 ret = ath11k_core_soc_create(ab);
1109 if (ret) {
1110 ath11k_err(ab, "failed to create soc core: %d\n", ret);
1111 return ret;
1112 }
1113
1114 return 0;
1115 }
1116 EXPORT_SYMBOL(ath11k_core_init);
1117
ath11k_core_deinit(struct ath11k_base * ab)1118 void ath11k_core_deinit(struct ath11k_base *ab)
1119 {
1120 mutex_lock(&ab->core_lock);
1121
1122 ath11k_core_pdev_destroy(ab);
1123 ath11k_core_stop(ab);
1124
1125 mutex_unlock(&ab->core_lock);
1126
1127 ath11k_hif_power_down(ab);
1128 ath11k_mac_destroy(ab);
1129 ath11k_core_soc_destroy(ab);
1130 }
1131 EXPORT_SYMBOL(ath11k_core_deinit);
1132
ath11k_core_free(struct ath11k_base * ab)1133 void ath11k_core_free(struct ath11k_base *ab)
1134 {
1135 kfree(ab);
1136 }
1137 EXPORT_SYMBOL(ath11k_core_free);
1138
ath11k_core_alloc(struct device * dev,size_t priv_size,enum ath11k_bus bus,const struct ath11k_bus_params * bus_params)1139 struct ath11k_base *ath11k_core_alloc(struct device *dev, size_t priv_size,
1140 enum ath11k_bus bus,
1141 const struct ath11k_bus_params *bus_params)
1142 {
1143 struct ath11k_base *ab;
1144
1145 ab = kzalloc(sizeof(*ab) + priv_size, GFP_KERNEL);
1146 if (!ab)
1147 return NULL;
1148
1149 init_completion(&ab->driver_recovery);
1150
1151 ab->workqueue = create_singlethread_workqueue("ath11k_wq");
1152 if (!ab->workqueue)
1153 goto err_sc_free;
1154
1155 mutex_init(&ab->core_lock);
1156 spin_lock_init(&ab->base_lock);
1157
1158 INIT_LIST_HEAD(&ab->peers);
1159 init_waitqueue_head(&ab->peer_mapping_wq);
1160 init_waitqueue_head(&ab->wmi_ab.tx_credits_wq);
1161 init_waitqueue_head(&ab->qmi.cold_boot_waitq);
1162 INIT_WORK(&ab->restart_work, ath11k_core_restart);
1163 timer_setup(&ab->rx_replenish_retry, ath11k_ce_rx_replenish_retry, 0);
1164 init_completion(&ab->htc_suspend);
1165 init_completion(&ab->wow.wakeup_completed);
1166
1167 ab->dev = dev;
1168 ab->bus_params = *bus_params;
1169 ab->hif.bus = bus;
1170
1171 return ab;
1172
1173 err_sc_free:
1174 kfree(ab);
1175 return NULL;
1176 }
1177 EXPORT_SYMBOL(ath11k_core_alloc);
1178
1179 MODULE_DESCRIPTION("Core module for Qualcomm Atheros 802.11ax wireless LAN cards.");
1180 MODULE_LICENSE("Dual BSD/GPL");
1181