1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright (c) 2008-2011 Atheros Communications Inc.
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Permission to use, copy, modify, and/or distribute this software for any
5*4882a593Smuzhiyun * purpose with or without fee is hereby granted, provided that the above
6*4882a593Smuzhiyun * copyright notice and this permission notice appear in all copies.
7*4882a593Smuzhiyun *
8*4882a593Smuzhiyun * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9*4882a593Smuzhiyun * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10*4882a593Smuzhiyun * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11*4882a593Smuzhiyun * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12*4882a593Smuzhiyun * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13*4882a593Smuzhiyun * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14*4882a593Smuzhiyun * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15*4882a593Smuzhiyun */
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
18*4882a593Smuzhiyun
19*4882a593Smuzhiyun #include <linux/dma-mapping.h>
20*4882a593Smuzhiyun #include <linux/slab.h>
21*4882a593Smuzhiyun #include <linux/ath9k_platform.h>
22*4882a593Smuzhiyun #include <linux/module.h>
23*4882a593Smuzhiyun #include <linux/of.h>
24*4882a593Smuzhiyun #include <linux/of_net.h>
25*4882a593Smuzhiyun #include <linux/relay.h>
26*4882a593Smuzhiyun #include <linux/dmi.h>
27*4882a593Smuzhiyun #include <net/ieee80211_radiotap.h>
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun #include "ath9k.h"
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun struct ath9k_eeprom_ctx {
32*4882a593Smuzhiyun struct completion complete;
33*4882a593Smuzhiyun struct ath_hw *ah;
34*4882a593Smuzhiyun };
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun static char *dev_info = "ath9k";
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun MODULE_AUTHOR("Atheros Communications");
39*4882a593Smuzhiyun MODULE_DESCRIPTION("Support for Atheros 802.11n wireless LAN cards.");
40*4882a593Smuzhiyun MODULE_SUPPORTED_DEVICE("Atheros 802.11n WLAN cards");
41*4882a593Smuzhiyun MODULE_LICENSE("Dual BSD/GPL");
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun static unsigned int ath9k_debug = ATH_DBG_DEFAULT;
44*4882a593Smuzhiyun module_param_named(debug, ath9k_debug, uint, 0);
45*4882a593Smuzhiyun MODULE_PARM_DESC(debug, "Debugging mask");
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun int ath9k_modparam_nohwcrypt;
48*4882a593Smuzhiyun module_param_named(nohwcrypt, ath9k_modparam_nohwcrypt, int, 0444);
49*4882a593Smuzhiyun MODULE_PARM_DESC(nohwcrypt, "Disable hardware encryption");
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun int ath9k_led_blink;
52*4882a593Smuzhiyun module_param_named(blink, ath9k_led_blink, int, 0444);
53*4882a593Smuzhiyun MODULE_PARM_DESC(blink, "Enable LED blink on activity");
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun static int ath9k_led_active_high = -1;
56*4882a593Smuzhiyun module_param_named(led_active_high, ath9k_led_active_high, int, 0444);
57*4882a593Smuzhiyun MODULE_PARM_DESC(led_active_high, "Invert LED polarity");
58*4882a593Smuzhiyun
59*4882a593Smuzhiyun static int ath9k_btcoex_enable;
60*4882a593Smuzhiyun module_param_named(btcoex_enable, ath9k_btcoex_enable, int, 0444);
61*4882a593Smuzhiyun MODULE_PARM_DESC(btcoex_enable, "Enable wifi-BT coexistence");
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun static int ath9k_bt_ant_diversity;
64*4882a593Smuzhiyun module_param_named(bt_ant_diversity, ath9k_bt_ant_diversity, int, 0444);
65*4882a593Smuzhiyun MODULE_PARM_DESC(bt_ant_diversity, "Enable WLAN/BT RX antenna diversity");
66*4882a593Smuzhiyun
67*4882a593Smuzhiyun static int ath9k_ps_enable;
68*4882a593Smuzhiyun module_param_named(ps_enable, ath9k_ps_enable, int, 0444);
69*4882a593Smuzhiyun MODULE_PARM_DESC(ps_enable, "Enable WLAN PowerSave");
70*4882a593Smuzhiyun
71*4882a593Smuzhiyun #ifdef CONFIG_ATH9K_CHANNEL_CONTEXT
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun int ath9k_use_chanctx;
74*4882a593Smuzhiyun module_param_named(use_chanctx, ath9k_use_chanctx, int, 0444);
75*4882a593Smuzhiyun MODULE_PARM_DESC(use_chanctx, "Enable channel context for concurrency");
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun #endif /* CONFIG_ATH9K_CHANNEL_CONTEXT */
78*4882a593Smuzhiyun
79*4882a593Smuzhiyun int ath9k_use_msi;
80*4882a593Smuzhiyun module_param_named(use_msi, ath9k_use_msi, int, 0444);
81*4882a593Smuzhiyun MODULE_PARM_DESC(use_msi, "Use MSI instead of INTx if possible");
82*4882a593Smuzhiyun
83*4882a593Smuzhiyun bool is_ath9k_unloaded;
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun #ifdef CONFIG_MAC80211_LEDS
86*4882a593Smuzhiyun static const struct ieee80211_tpt_blink ath9k_tpt_blink[] = {
87*4882a593Smuzhiyun { .throughput = 0 * 1024, .blink_time = 334 },
88*4882a593Smuzhiyun { .throughput = 1 * 1024, .blink_time = 260 },
89*4882a593Smuzhiyun { .throughput = 5 * 1024, .blink_time = 220 },
90*4882a593Smuzhiyun { .throughput = 10 * 1024, .blink_time = 190 },
91*4882a593Smuzhiyun { .throughput = 20 * 1024, .blink_time = 170 },
92*4882a593Smuzhiyun { .throughput = 50 * 1024, .blink_time = 150 },
93*4882a593Smuzhiyun { .throughput = 70 * 1024, .blink_time = 130 },
94*4882a593Smuzhiyun { .throughput = 100 * 1024, .blink_time = 110 },
95*4882a593Smuzhiyun { .throughput = 200 * 1024, .blink_time = 80 },
96*4882a593Smuzhiyun { .throughput = 300 * 1024, .blink_time = 50 },
97*4882a593Smuzhiyun };
98*4882a593Smuzhiyun #endif
99*4882a593Smuzhiyun
set_use_msi(const struct dmi_system_id * dmi)100*4882a593Smuzhiyun static int __init set_use_msi(const struct dmi_system_id *dmi)
101*4882a593Smuzhiyun {
102*4882a593Smuzhiyun ath9k_use_msi = 1;
103*4882a593Smuzhiyun return 1;
104*4882a593Smuzhiyun }
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun static const struct dmi_system_id ath9k_quirks[] __initconst = {
107*4882a593Smuzhiyun {
108*4882a593Smuzhiyun .callback = set_use_msi,
109*4882a593Smuzhiyun .ident = "Dell Inspiron 24-3460",
110*4882a593Smuzhiyun .matches = {
111*4882a593Smuzhiyun DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
112*4882a593Smuzhiyun DMI_MATCH(DMI_PRODUCT_NAME, "Inspiron 24-3460"),
113*4882a593Smuzhiyun },
114*4882a593Smuzhiyun },
115*4882a593Smuzhiyun {
116*4882a593Smuzhiyun .callback = set_use_msi,
117*4882a593Smuzhiyun .ident = "Dell Vostro 3262",
118*4882a593Smuzhiyun .matches = {
119*4882a593Smuzhiyun DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
120*4882a593Smuzhiyun DMI_MATCH(DMI_PRODUCT_NAME, "Vostro 3262"),
121*4882a593Smuzhiyun },
122*4882a593Smuzhiyun },
123*4882a593Smuzhiyun {
124*4882a593Smuzhiyun .callback = set_use_msi,
125*4882a593Smuzhiyun .ident = "Dell Inspiron 3472",
126*4882a593Smuzhiyun .matches = {
127*4882a593Smuzhiyun DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
128*4882a593Smuzhiyun DMI_MATCH(DMI_PRODUCT_NAME, "Inspiron 3472"),
129*4882a593Smuzhiyun },
130*4882a593Smuzhiyun },
131*4882a593Smuzhiyun {
132*4882a593Smuzhiyun .callback = set_use_msi,
133*4882a593Smuzhiyun .ident = "Dell Vostro 15-3572",
134*4882a593Smuzhiyun .matches = {
135*4882a593Smuzhiyun DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
136*4882a593Smuzhiyun DMI_MATCH(DMI_PRODUCT_NAME, "Vostro 15-3572"),
137*4882a593Smuzhiyun },
138*4882a593Smuzhiyun },
139*4882a593Smuzhiyun {
140*4882a593Smuzhiyun .callback = set_use_msi,
141*4882a593Smuzhiyun .ident = "Dell Inspiron 14-3473",
142*4882a593Smuzhiyun .matches = {
143*4882a593Smuzhiyun DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
144*4882a593Smuzhiyun DMI_MATCH(DMI_PRODUCT_NAME, "Inspiron 14-3473"),
145*4882a593Smuzhiyun },
146*4882a593Smuzhiyun },
147*4882a593Smuzhiyun {}
148*4882a593Smuzhiyun };
149*4882a593Smuzhiyun
150*4882a593Smuzhiyun static void ath9k_deinit_softc(struct ath_softc *sc);
151*4882a593Smuzhiyun
ath9k_op_ps_wakeup(struct ath_common * common)152*4882a593Smuzhiyun static void ath9k_op_ps_wakeup(struct ath_common *common)
153*4882a593Smuzhiyun {
154*4882a593Smuzhiyun ath9k_ps_wakeup((struct ath_softc *) common->priv);
155*4882a593Smuzhiyun }
156*4882a593Smuzhiyun
ath9k_op_ps_restore(struct ath_common * common)157*4882a593Smuzhiyun static void ath9k_op_ps_restore(struct ath_common *common)
158*4882a593Smuzhiyun {
159*4882a593Smuzhiyun ath9k_ps_restore((struct ath_softc *) common->priv);
160*4882a593Smuzhiyun }
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun static const struct ath_ps_ops ath9k_ps_ops = {
163*4882a593Smuzhiyun .wakeup = ath9k_op_ps_wakeup,
164*4882a593Smuzhiyun .restore = ath9k_op_ps_restore,
165*4882a593Smuzhiyun };
166*4882a593Smuzhiyun
167*4882a593Smuzhiyun /*
168*4882a593Smuzhiyun * Read and write, they both share the same lock. We do this to serialize
169*4882a593Smuzhiyun * reads and writes on Atheros 802.11n PCI devices only. This is required
170*4882a593Smuzhiyun * as the FIFO on these devices can only accept sanely 2 requests.
171*4882a593Smuzhiyun */
172*4882a593Smuzhiyun
ath9k_iowrite32(void * hw_priv,u32 val,u32 reg_offset)173*4882a593Smuzhiyun static void ath9k_iowrite32(void *hw_priv, u32 val, u32 reg_offset)
174*4882a593Smuzhiyun {
175*4882a593Smuzhiyun struct ath_hw *ah = hw_priv;
176*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
177*4882a593Smuzhiyun struct ath_softc *sc = (struct ath_softc *) common->priv;
178*4882a593Smuzhiyun
179*4882a593Smuzhiyun if (NR_CPUS > 1 && ah->config.serialize_regmode == SER_REG_MODE_ON) {
180*4882a593Smuzhiyun unsigned long flags;
181*4882a593Smuzhiyun spin_lock_irqsave(&sc->sc_serial_rw, flags);
182*4882a593Smuzhiyun iowrite32(val, sc->mem + reg_offset);
183*4882a593Smuzhiyun spin_unlock_irqrestore(&sc->sc_serial_rw, flags);
184*4882a593Smuzhiyun } else
185*4882a593Smuzhiyun iowrite32(val, sc->mem + reg_offset);
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun
ath9k_ioread32(void * hw_priv,u32 reg_offset)188*4882a593Smuzhiyun static unsigned int ath9k_ioread32(void *hw_priv, u32 reg_offset)
189*4882a593Smuzhiyun {
190*4882a593Smuzhiyun struct ath_hw *ah = hw_priv;
191*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
192*4882a593Smuzhiyun struct ath_softc *sc = (struct ath_softc *) common->priv;
193*4882a593Smuzhiyun u32 val;
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun if (NR_CPUS > 1 && ah->config.serialize_regmode == SER_REG_MODE_ON) {
196*4882a593Smuzhiyun unsigned long flags;
197*4882a593Smuzhiyun spin_lock_irqsave(&sc->sc_serial_rw, flags);
198*4882a593Smuzhiyun val = ioread32(sc->mem + reg_offset);
199*4882a593Smuzhiyun spin_unlock_irqrestore(&sc->sc_serial_rw, flags);
200*4882a593Smuzhiyun } else
201*4882a593Smuzhiyun val = ioread32(sc->mem + reg_offset);
202*4882a593Smuzhiyun return val;
203*4882a593Smuzhiyun }
204*4882a593Smuzhiyun
ath9k_multi_ioread32(void * hw_priv,u32 * addr,u32 * val,u16 count)205*4882a593Smuzhiyun static void ath9k_multi_ioread32(void *hw_priv, u32 *addr,
206*4882a593Smuzhiyun u32 *val, u16 count)
207*4882a593Smuzhiyun {
208*4882a593Smuzhiyun int i;
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun for (i = 0; i < count; i++)
211*4882a593Smuzhiyun val[i] = ath9k_ioread32(hw_priv, addr[i]);
212*4882a593Smuzhiyun }
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun
__ath9k_reg_rmw(struct ath_softc * sc,u32 reg_offset,u32 set,u32 clr)215*4882a593Smuzhiyun static unsigned int __ath9k_reg_rmw(struct ath_softc *sc, u32 reg_offset,
216*4882a593Smuzhiyun u32 set, u32 clr)
217*4882a593Smuzhiyun {
218*4882a593Smuzhiyun u32 val;
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun val = ioread32(sc->mem + reg_offset);
221*4882a593Smuzhiyun val &= ~clr;
222*4882a593Smuzhiyun val |= set;
223*4882a593Smuzhiyun iowrite32(val, sc->mem + reg_offset);
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun return val;
226*4882a593Smuzhiyun }
227*4882a593Smuzhiyun
ath9k_reg_rmw(void * hw_priv,u32 reg_offset,u32 set,u32 clr)228*4882a593Smuzhiyun static unsigned int ath9k_reg_rmw(void *hw_priv, u32 reg_offset, u32 set, u32 clr)
229*4882a593Smuzhiyun {
230*4882a593Smuzhiyun struct ath_hw *ah = hw_priv;
231*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
232*4882a593Smuzhiyun struct ath_softc *sc = (struct ath_softc *) common->priv;
233*4882a593Smuzhiyun unsigned long flags;
234*4882a593Smuzhiyun u32 val;
235*4882a593Smuzhiyun
236*4882a593Smuzhiyun if (NR_CPUS > 1 && ah->config.serialize_regmode == SER_REG_MODE_ON) {
237*4882a593Smuzhiyun spin_lock_irqsave(&sc->sc_serial_rw, flags);
238*4882a593Smuzhiyun val = __ath9k_reg_rmw(sc, reg_offset, set, clr);
239*4882a593Smuzhiyun spin_unlock_irqrestore(&sc->sc_serial_rw, flags);
240*4882a593Smuzhiyun } else
241*4882a593Smuzhiyun val = __ath9k_reg_rmw(sc, reg_offset, set, clr);
242*4882a593Smuzhiyun
243*4882a593Smuzhiyun return val;
244*4882a593Smuzhiyun }
245*4882a593Smuzhiyun
246*4882a593Smuzhiyun /**************************/
247*4882a593Smuzhiyun /* Initialization */
248*4882a593Smuzhiyun /**************************/
249*4882a593Smuzhiyun
ath9k_reg_notifier(struct wiphy * wiphy,struct regulatory_request * request)250*4882a593Smuzhiyun static void ath9k_reg_notifier(struct wiphy *wiphy,
251*4882a593Smuzhiyun struct regulatory_request *request)
252*4882a593Smuzhiyun {
253*4882a593Smuzhiyun struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
254*4882a593Smuzhiyun struct ath_softc *sc = hw->priv;
255*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
256*4882a593Smuzhiyun struct ath_regulatory *reg = ath9k_hw_regulatory(ah);
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun ath_reg_notifier_apply(wiphy, request, reg);
259*4882a593Smuzhiyun
260*4882a593Smuzhiyun /* synchronize DFS detector if regulatory domain changed */
261*4882a593Smuzhiyun if (sc->dfs_detector != NULL)
262*4882a593Smuzhiyun sc->dfs_detector->set_dfs_domain(sc->dfs_detector,
263*4882a593Smuzhiyun request->dfs_region);
264*4882a593Smuzhiyun
265*4882a593Smuzhiyun /* Set tx power */
266*4882a593Smuzhiyun if (!ah->curchan)
267*4882a593Smuzhiyun return;
268*4882a593Smuzhiyun
269*4882a593Smuzhiyun sc->cur_chan->txpower = 2 * ah->curchan->chan->max_power;
270*4882a593Smuzhiyun ath9k_ps_wakeup(sc);
271*4882a593Smuzhiyun ath9k_hw_set_txpowerlimit(ah, sc->cur_chan->txpower, false);
272*4882a593Smuzhiyun ath9k_cmn_update_txpow(ah, sc->cur_chan->cur_txpower,
273*4882a593Smuzhiyun sc->cur_chan->txpower,
274*4882a593Smuzhiyun &sc->cur_chan->cur_txpower);
275*4882a593Smuzhiyun ath9k_ps_restore(sc);
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun /*
279*4882a593Smuzhiyun * This function will allocate both the DMA descriptor structure, and the
280*4882a593Smuzhiyun * buffers it contains. These are used to contain the descriptors used
281*4882a593Smuzhiyun * by the system.
282*4882a593Smuzhiyun */
ath_descdma_setup(struct ath_softc * sc,struct ath_descdma * dd,struct list_head * head,const char * name,int nbuf,int ndesc,bool is_tx)283*4882a593Smuzhiyun int ath_descdma_setup(struct ath_softc *sc, struct ath_descdma *dd,
284*4882a593Smuzhiyun struct list_head *head, const char *name,
285*4882a593Smuzhiyun int nbuf, int ndesc, bool is_tx)
286*4882a593Smuzhiyun {
287*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(sc->sc_ah);
288*4882a593Smuzhiyun u8 *ds;
289*4882a593Smuzhiyun int i, bsize, desc_len;
290*4882a593Smuzhiyun
291*4882a593Smuzhiyun ath_dbg(common, CONFIG, "%s DMA: %u buffers %u desc/buf\n",
292*4882a593Smuzhiyun name, nbuf, ndesc);
293*4882a593Smuzhiyun
294*4882a593Smuzhiyun INIT_LIST_HEAD(head);
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun if (is_tx)
297*4882a593Smuzhiyun desc_len = sc->sc_ah->caps.tx_desc_len;
298*4882a593Smuzhiyun else
299*4882a593Smuzhiyun desc_len = sizeof(struct ath_desc);
300*4882a593Smuzhiyun
301*4882a593Smuzhiyun /* ath_desc must be a multiple of DWORDs */
302*4882a593Smuzhiyun if ((desc_len % 4) != 0) {
303*4882a593Smuzhiyun ath_err(common, "ath_desc not DWORD aligned\n");
304*4882a593Smuzhiyun BUG_ON((desc_len % 4) != 0);
305*4882a593Smuzhiyun return -ENOMEM;
306*4882a593Smuzhiyun }
307*4882a593Smuzhiyun
308*4882a593Smuzhiyun dd->dd_desc_len = desc_len * nbuf * ndesc;
309*4882a593Smuzhiyun
310*4882a593Smuzhiyun /*
311*4882a593Smuzhiyun * Need additional DMA memory because we can't use
312*4882a593Smuzhiyun * descriptors that cross the 4K page boundary. Assume
313*4882a593Smuzhiyun * one skipped descriptor per 4K page.
314*4882a593Smuzhiyun */
315*4882a593Smuzhiyun if (!(sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_4KB_SPLITTRANS)) {
316*4882a593Smuzhiyun u32 ndesc_skipped =
317*4882a593Smuzhiyun ATH_DESC_4KB_BOUND_NUM_SKIPPED(dd->dd_desc_len);
318*4882a593Smuzhiyun u32 dma_len;
319*4882a593Smuzhiyun
320*4882a593Smuzhiyun while (ndesc_skipped) {
321*4882a593Smuzhiyun dma_len = ndesc_skipped * desc_len;
322*4882a593Smuzhiyun dd->dd_desc_len += dma_len;
323*4882a593Smuzhiyun
324*4882a593Smuzhiyun ndesc_skipped = ATH_DESC_4KB_BOUND_NUM_SKIPPED(dma_len);
325*4882a593Smuzhiyun }
326*4882a593Smuzhiyun }
327*4882a593Smuzhiyun
328*4882a593Smuzhiyun /* allocate descriptors */
329*4882a593Smuzhiyun dd->dd_desc = dmam_alloc_coherent(sc->dev, dd->dd_desc_len,
330*4882a593Smuzhiyun &dd->dd_desc_paddr, GFP_KERNEL);
331*4882a593Smuzhiyun if (!dd->dd_desc)
332*4882a593Smuzhiyun return -ENOMEM;
333*4882a593Smuzhiyun
334*4882a593Smuzhiyun ds = dd->dd_desc;
335*4882a593Smuzhiyun ath_dbg(common, CONFIG, "%s DMA map: %p (%u) -> %llx (%u)\n",
336*4882a593Smuzhiyun name, ds, (u32) dd->dd_desc_len,
337*4882a593Smuzhiyun ito64(dd->dd_desc_paddr), /*XXX*/(u32) dd->dd_desc_len);
338*4882a593Smuzhiyun
339*4882a593Smuzhiyun /* allocate buffers */
340*4882a593Smuzhiyun if (is_tx) {
341*4882a593Smuzhiyun struct ath_buf *bf;
342*4882a593Smuzhiyun
343*4882a593Smuzhiyun bsize = sizeof(struct ath_buf) * nbuf;
344*4882a593Smuzhiyun bf = devm_kzalloc(sc->dev, bsize, GFP_KERNEL);
345*4882a593Smuzhiyun if (!bf)
346*4882a593Smuzhiyun return -ENOMEM;
347*4882a593Smuzhiyun
348*4882a593Smuzhiyun for (i = 0; i < nbuf; i++, bf++, ds += (desc_len * ndesc)) {
349*4882a593Smuzhiyun bf->bf_desc = ds;
350*4882a593Smuzhiyun bf->bf_daddr = DS2PHYS(dd, ds);
351*4882a593Smuzhiyun
352*4882a593Smuzhiyun if (!(sc->sc_ah->caps.hw_caps &
353*4882a593Smuzhiyun ATH9K_HW_CAP_4KB_SPLITTRANS)) {
354*4882a593Smuzhiyun /*
355*4882a593Smuzhiyun * Skip descriptor addresses which can cause 4KB
356*4882a593Smuzhiyun * boundary crossing (addr + length) with a 32 dword
357*4882a593Smuzhiyun * descriptor fetch.
358*4882a593Smuzhiyun */
359*4882a593Smuzhiyun while (ATH_DESC_4KB_BOUND_CHECK(bf->bf_daddr)) {
360*4882a593Smuzhiyun BUG_ON((caddr_t) bf->bf_desc >=
361*4882a593Smuzhiyun ((caddr_t) dd->dd_desc +
362*4882a593Smuzhiyun dd->dd_desc_len));
363*4882a593Smuzhiyun
364*4882a593Smuzhiyun ds += (desc_len * ndesc);
365*4882a593Smuzhiyun bf->bf_desc = ds;
366*4882a593Smuzhiyun bf->bf_daddr = DS2PHYS(dd, ds);
367*4882a593Smuzhiyun }
368*4882a593Smuzhiyun }
369*4882a593Smuzhiyun list_add_tail(&bf->list, head);
370*4882a593Smuzhiyun }
371*4882a593Smuzhiyun } else {
372*4882a593Smuzhiyun struct ath_rxbuf *bf;
373*4882a593Smuzhiyun
374*4882a593Smuzhiyun bsize = sizeof(struct ath_rxbuf) * nbuf;
375*4882a593Smuzhiyun bf = devm_kzalloc(sc->dev, bsize, GFP_KERNEL);
376*4882a593Smuzhiyun if (!bf)
377*4882a593Smuzhiyun return -ENOMEM;
378*4882a593Smuzhiyun
379*4882a593Smuzhiyun for (i = 0; i < nbuf; i++, bf++, ds += (desc_len * ndesc)) {
380*4882a593Smuzhiyun bf->bf_desc = ds;
381*4882a593Smuzhiyun bf->bf_daddr = DS2PHYS(dd, ds);
382*4882a593Smuzhiyun
383*4882a593Smuzhiyun if (!(sc->sc_ah->caps.hw_caps &
384*4882a593Smuzhiyun ATH9K_HW_CAP_4KB_SPLITTRANS)) {
385*4882a593Smuzhiyun /*
386*4882a593Smuzhiyun * Skip descriptor addresses which can cause 4KB
387*4882a593Smuzhiyun * boundary crossing (addr + length) with a 32 dword
388*4882a593Smuzhiyun * descriptor fetch.
389*4882a593Smuzhiyun */
390*4882a593Smuzhiyun while (ATH_DESC_4KB_BOUND_CHECK(bf->bf_daddr)) {
391*4882a593Smuzhiyun BUG_ON((caddr_t) bf->bf_desc >=
392*4882a593Smuzhiyun ((caddr_t) dd->dd_desc +
393*4882a593Smuzhiyun dd->dd_desc_len));
394*4882a593Smuzhiyun
395*4882a593Smuzhiyun ds += (desc_len * ndesc);
396*4882a593Smuzhiyun bf->bf_desc = ds;
397*4882a593Smuzhiyun bf->bf_daddr = DS2PHYS(dd, ds);
398*4882a593Smuzhiyun }
399*4882a593Smuzhiyun }
400*4882a593Smuzhiyun list_add_tail(&bf->list, head);
401*4882a593Smuzhiyun }
402*4882a593Smuzhiyun }
403*4882a593Smuzhiyun return 0;
404*4882a593Smuzhiyun }
405*4882a593Smuzhiyun
ath9k_init_queues(struct ath_softc * sc)406*4882a593Smuzhiyun static int ath9k_init_queues(struct ath_softc *sc)
407*4882a593Smuzhiyun {
408*4882a593Smuzhiyun int i = 0;
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun sc->beacon.beaconq = ath9k_hw_beaconq_setup(sc->sc_ah);
411*4882a593Smuzhiyun sc->beacon.cabq = ath_txq_setup(sc, ATH9K_TX_QUEUE_CAB, 0);
412*4882a593Smuzhiyun ath_cabq_update(sc);
413*4882a593Smuzhiyun
414*4882a593Smuzhiyun sc->tx.uapsdq = ath_txq_setup(sc, ATH9K_TX_QUEUE_UAPSD, 0);
415*4882a593Smuzhiyun
416*4882a593Smuzhiyun for (i = 0; i < IEEE80211_NUM_ACS; i++) {
417*4882a593Smuzhiyun sc->tx.txq_map[i] = ath_txq_setup(sc, ATH9K_TX_QUEUE_DATA, i);
418*4882a593Smuzhiyun sc->tx.txq_map[i]->mac80211_qnum = i;
419*4882a593Smuzhiyun }
420*4882a593Smuzhiyun return 0;
421*4882a593Smuzhiyun }
422*4882a593Smuzhiyun
ath9k_init_misc(struct ath_softc * sc)423*4882a593Smuzhiyun static void ath9k_init_misc(struct ath_softc *sc)
424*4882a593Smuzhiyun {
425*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(sc->sc_ah);
426*4882a593Smuzhiyun int i = 0;
427*4882a593Smuzhiyun
428*4882a593Smuzhiyun timer_setup(&common->ani.timer, ath_ani_calibrate, 0);
429*4882a593Smuzhiyun
430*4882a593Smuzhiyun common->last_rssi = ATH_RSSI_DUMMY_MARKER;
431*4882a593Smuzhiyun eth_broadcast_addr(common->bssidmask);
432*4882a593Smuzhiyun sc->beacon.slottime = 9;
433*4882a593Smuzhiyun
434*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(sc->beacon.bslot); i++)
435*4882a593Smuzhiyun sc->beacon.bslot[i] = NULL;
436*4882a593Smuzhiyun
437*4882a593Smuzhiyun if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_ANT_DIV_COMB)
438*4882a593Smuzhiyun sc->ant_comb.count = ATH_ANT_DIV_COMB_INIT_COUNT;
439*4882a593Smuzhiyun
440*4882a593Smuzhiyun sc->spec_priv.ah = sc->sc_ah;
441*4882a593Smuzhiyun sc->spec_priv.spec_config.enabled = 0;
442*4882a593Smuzhiyun sc->spec_priv.spec_config.short_repeat = true;
443*4882a593Smuzhiyun sc->spec_priv.spec_config.count = 8;
444*4882a593Smuzhiyun sc->spec_priv.spec_config.endless = false;
445*4882a593Smuzhiyun sc->spec_priv.spec_config.period = 0xFF;
446*4882a593Smuzhiyun sc->spec_priv.spec_config.fft_period = 0xF;
447*4882a593Smuzhiyun }
448*4882a593Smuzhiyun
ath9k_init_pcoem_platform(struct ath_softc * sc)449*4882a593Smuzhiyun static void ath9k_init_pcoem_platform(struct ath_softc *sc)
450*4882a593Smuzhiyun {
451*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
452*4882a593Smuzhiyun struct ath9k_hw_capabilities *pCap = &ah->caps;
453*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun if (!IS_ENABLED(CONFIG_ATH9K_PCOEM))
456*4882a593Smuzhiyun return;
457*4882a593Smuzhiyun
458*4882a593Smuzhiyun if (common->bus_ops->ath_bus_type != ATH_PCI)
459*4882a593Smuzhiyun return;
460*4882a593Smuzhiyun
461*4882a593Smuzhiyun if (sc->driver_data & (ATH9K_PCI_CUS198 |
462*4882a593Smuzhiyun ATH9K_PCI_CUS230)) {
463*4882a593Smuzhiyun ah->config.xlna_gpio = 9;
464*4882a593Smuzhiyun ah->config.xatten_margin_cfg = true;
465*4882a593Smuzhiyun ah->config.alt_mingainidx = true;
466*4882a593Smuzhiyun ah->config.ant_ctrl_comm2g_switch_enable = 0x000BBB88;
467*4882a593Smuzhiyun sc->ant_comb.low_rssi_thresh = 20;
468*4882a593Smuzhiyun sc->ant_comb.fast_div_bias = 3;
469*4882a593Smuzhiyun
470*4882a593Smuzhiyun ath_info(common, "Set parameters for %s\n",
471*4882a593Smuzhiyun (sc->driver_data & ATH9K_PCI_CUS198) ?
472*4882a593Smuzhiyun "CUS198" : "CUS230");
473*4882a593Smuzhiyun }
474*4882a593Smuzhiyun
475*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_CUS217)
476*4882a593Smuzhiyun ath_info(common, "CUS217 card detected\n");
477*4882a593Smuzhiyun
478*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_CUS252)
479*4882a593Smuzhiyun ath_info(common, "CUS252 card detected\n");
480*4882a593Smuzhiyun
481*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_AR9565_1ANT)
482*4882a593Smuzhiyun ath_info(common, "WB335 1-ANT card detected\n");
483*4882a593Smuzhiyun
484*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_AR9565_2ANT)
485*4882a593Smuzhiyun ath_info(common, "WB335 2-ANT card detected\n");
486*4882a593Smuzhiyun
487*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_KILLER)
488*4882a593Smuzhiyun ath_info(common, "Killer Wireless card detected\n");
489*4882a593Smuzhiyun
490*4882a593Smuzhiyun /*
491*4882a593Smuzhiyun * Some WB335 cards do not support antenna diversity. Since
492*4882a593Smuzhiyun * we use a hardcoded value for AR9565 instead of using the
493*4882a593Smuzhiyun * EEPROM/OTP data, remove the combining feature from
494*4882a593Smuzhiyun * the HW capabilities bitmap.
495*4882a593Smuzhiyun */
496*4882a593Smuzhiyun if (sc->driver_data & (ATH9K_PCI_AR9565_1ANT | ATH9K_PCI_AR9565_2ANT)) {
497*4882a593Smuzhiyun if (!(sc->driver_data & ATH9K_PCI_BT_ANT_DIV))
498*4882a593Smuzhiyun pCap->hw_caps &= ~ATH9K_HW_CAP_ANT_DIV_COMB;
499*4882a593Smuzhiyun }
500*4882a593Smuzhiyun
501*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_BT_ANT_DIV) {
502*4882a593Smuzhiyun pCap->hw_caps |= ATH9K_HW_CAP_BT_ANT_DIV;
503*4882a593Smuzhiyun ath_info(common, "Set BT/WLAN RX diversity capability\n");
504*4882a593Smuzhiyun }
505*4882a593Smuzhiyun
506*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_D3_L1_WAR) {
507*4882a593Smuzhiyun ah->config.pcie_waen = 0x0040473b;
508*4882a593Smuzhiyun ath_info(common, "Enable WAR for ASPM D3/L1\n");
509*4882a593Smuzhiyun }
510*4882a593Smuzhiyun
511*4882a593Smuzhiyun /*
512*4882a593Smuzhiyun * The default value of pll_pwrsave is 1.
513*4882a593Smuzhiyun * For certain AR9485 cards, it is set to 0.
514*4882a593Smuzhiyun * For AR9462, AR9565 it's set to 7.
515*4882a593Smuzhiyun */
516*4882a593Smuzhiyun ah->config.pll_pwrsave = 1;
517*4882a593Smuzhiyun
518*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_NO_PLL_PWRSAVE) {
519*4882a593Smuzhiyun ah->config.pll_pwrsave = 0;
520*4882a593Smuzhiyun ath_info(common, "Disable PLL PowerSave\n");
521*4882a593Smuzhiyun }
522*4882a593Smuzhiyun
523*4882a593Smuzhiyun if (sc->driver_data & ATH9K_PCI_LED_ACT_HI)
524*4882a593Smuzhiyun ah->config.led_active_high = true;
525*4882a593Smuzhiyun }
526*4882a593Smuzhiyun
ath9k_eeprom_request_cb(const struct firmware * eeprom_blob,void * ctx)527*4882a593Smuzhiyun static void ath9k_eeprom_request_cb(const struct firmware *eeprom_blob,
528*4882a593Smuzhiyun void *ctx)
529*4882a593Smuzhiyun {
530*4882a593Smuzhiyun struct ath9k_eeprom_ctx *ec = ctx;
531*4882a593Smuzhiyun
532*4882a593Smuzhiyun if (eeprom_blob)
533*4882a593Smuzhiyun ec->ah->eeprom_blob = eeprom_blob;
534*4882a593Smuzhiyun
535*4882a593Smuzhiyun complete(&ec->complete);
536*4882a593Smuzhiyun }
537*4882a593Smuzhiyun
ath9k_eeprom_request(struct ath_softc * sc,const char * name)538*4882a593Smuzhiyun static int ath9k_eeprom_request(struct ath_softc *sc, const char *name)
539*4882a593Smuzhiyun {
540*4882a593Smuzhiyun struct ath9k_eeprom_ctx ec;
541*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
542*4882a593Smuzhiyun int err;
543*4882a593Smuzhiyun
544*4882a593Smuzhiyun /* try to load the EEPROM content asynchronously */
545*4882a593Smuzhiyun init_completion(&ec.complete);
546*4882a593Smuzhiyun ec.ah = sc->sc_ah;
547*4882a593Smuzhiyun
548*4882a593Smuzhiyun err = request_firmware_nowait(THIS_MODULE, 1, name, sc->dev, GFP_KERNEL,
549*4882a593Smuzhiyun &ec, ath9k_eeprom_request_cb);
550*4882a593Smuzhiyun if (err < 0) {
551*4882a593Smuzhiyun ath_err(ath9k_hw_common(ah),
552*4882a593Smuzhiyun "EEPROM request failed\n");
553*4882a593Smuzhiyun return err;
554*4882a593Smuzhiyun }
555*4882a593Smuzhiyun
556*4882a593Smuzhiyun wait_for_completion(&ec.complete);
557*4882a593Smuzhiyun
558*4882a593Smuzhiyun if (!ah->eeprom_blob) {
559*4882a593Smuzhiyun ath_err(ath9k_hw_common(ah),
560*4882a593Smuzhiyun "Unable to load EEPROM file %s\n", name);
561*4882a593Smuzhiyun return -EINVAL;
562*4882a593Smuzhiyun }
563*4882a593Smuzhiyun
564*4882a593Smuzhiyun return 0;
565*4882a593Smuzhiyun }
566*4882a593Smuzhiyun
ath9k_eeprom_release(struct ath_softc * sc)567*4882a593Smuzhiyun static void ath9k_eeprom_release(struct ath_softc *sc)
568*4882a593Smuzhiyun {
569*4882a593Smuzhiyun release_firmware(sc->sc_ah->eeprom_blob);
570*4882a593Smuzhiyun }
571*4882a593Smuzhiyun
ath9k_init_platform(struct ath_softc * sc)572*4882a593Smuzhiyun static int ath9k_init_platform(struct ath_softc *sc)
573*4882a593Smuzhiyun {
574*4882a593Smuzhiyun struct ath9k_platform_data *pdata = sc->dev->platform_data;
575*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
576*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
577*4882a593Smuzhiyun int ret;
578*4882a593Smuzhiyun
579*4882a593Smuzhiyun if (!pdata)
580*4882a593Smuzhiyun return 0;
581*4882a593Smuzhiyun
582*4882a593Smuzhiyun if (!pdata->use_eeprom) {
583*4882a593Smuzhiyun ah->ah_flags &= ~AH_USE_EEPROM;
584*4882a593Smuzhiyun ah->gpio_mask = pdata->gpio_mask;
585*4882a593Smuzhiyun ah->gpio_val = pdata->gpio_val;
586*4882a593Smuzhiyun ah->led_pin = pdata->led_pin;
587*4882a593Smuzhiyun ah->is_clk_25mhz = pdata->is_clk_25mhz;
588*4882a593Smuzhiyun ah->get_mac_revision = pdata->get_mac_revision;
589*4882a593Smuzhiyun ah->external_reset = pdata->external_reset;
590*4882a593Smuzhiyun ah->disable_2ghz = pdata->disable_2ghz;
591*4882a593Smuzhiyun ah->disable_5ghz = pdata->disable_5ghz;
592*4882a593Smuzhiyun
593*4882a593Smuzhiyun if (!pdata->endian_check)
594*4882a593Smuzhiyun ah->ah_flags |= AH_NO_EEP_SWAP;
595*4882a593Smuzhiyun }
596*4882a593Smuzhiyun
597*4882a593Smuzhiyun if (pdata->eeprom_name) {
598*4882a593Smuzhiyun ret = ath9k_eeprom_request(sc, pdata->eeprom_name);
599*4882a593Smuzhiyun if (ret)
600*4882a593Smuzhiyun return ret;
601*4882a593Smuzhiyun }
602*4882a593Smuzhiyun
603*4882a593Smuzhiyun if (pdata->led_active_high)
604*4882a593Smuzhiyun ah->config.led_active_high = true;
605*4882a593Smuzhiyun
606*4882a593Smuzhiyun if (pdata->tx_gain_buffalo)
607*4882a593Smuzhiyun ah->config.tx_gain_buffalo = true;
608*4882a593Smuzhiyun
609*4882a593Smuzhiyun if (pdata->macaddr)
610*4882a593Smuzhiyun ether_addr_copy(common->macaddr, pdata->macaddr);
611*4882a593Smuzhiyun
612*4882a593Smuzhiyun return 0;
613*4882a593Smuzhiyun }
614*4882a593Smuzhiyun
ath9k_of_init(struct ath_softc * sc)615*4882a593Smuzhiyun static int ath9k_of_init(struct ath_softc *sc)
616*4882a593Smuzhiyun {
617*4882a593Smuzhiyun struct device_node *np = sc->dev->of_node;
618*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
619*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
620*4882a593Smuzhiyun enum ath_bus_type bus_type = common->bus_ops->ath_bus_type;
621*4882a593Smuzhiyun const char *mac;
622*4882a593Smuzhiyun char eeprom_name[100];
623*4882a593Smuzhiyun int ret;
624*4882a593Smuzhiyun
625*4882a593Smuzhiyun if (!of_device_is_available(np))
626*4882a593Smuzhiyun return 0;
627*4882a593Smuzhiyun
628*4882a593Smuzhiyun ath_dbg(common, CONFIG, "parsing configuration from OF node\n");
629*4882a593Smuzhiyun
630*4882a593Smuzhiyun if (of_property_read_bool(np, "qca,no-eeprom")) {
631*4882a593Smuzhiyun /* ath9k-eeprom-<bus>-<id>.bin */
632*4882a593Smuzhiyun scnprintf(eeprom_name, sizeof(eeprom_name),
633*4882a593Smuzhiyun "ath9k-eeprom-%s-%s.bin",
634*4882a593Smuzhiyun ath_bus_type_to_string(bus_type), dev_name(ah->dev));
635*4882a593Smuzhiyun
636*4882a593Smuzhiyun ret = ath9k_eeprom_request(sc, eeprom_name);
637*4882a593Smuzhiyun if (ret)
638*4882a593Smuzhiyun return ret;
639*4882a593Smuzhiyun
640*4882a593Smuzhiyun ah->ah_flags &= ~AH_USE_EEPROM;
641*4882a593Smuzhiyun ah->ah_flags |= AH_NO_EEP_SWAP;
642*4882a593Smuzhiyun }
643*4882a593Smuzhiyun
644*4882a593Smuzhiyun mac = of_get_mac_address(np);
645*4882a593Smuzhiyun if (!IS_ERR(mac))
646*4882a593Smuzhiyun ether_addr_copy(common->macaddr, mac);
647*4882a593Smuzhiyun
648*4882a593Smuzhiyun return 0;
649*4882a593Smuzhiyun }
650*4882a593Smuzhiyun
ath9k_init_softc(u16 devid,struct ath_softc * sc,const struct ath_bus_ops * bus_ops)651*4882a593Smuzhiyun static int ath9k_init_softc(u16 devid, struct ath_softc *sc,
652*4882a593Smuzhiyun const struct ath_bus_ops *bus_ops)
653*4882a593Smuzhiyun {
654*4882a593Smuzhiyun struct ath_hw *ah = NULL;
655*4882a593Smuzhiyun struct ath9k_hw_capabilities *pCap;
656*4882a593Smuzhiyun struct ath_common *common;
657*4882a593Smuzhiyun int ret = 0, i;
658*4882a593Smuzhiyun int csz = 0;
659*4882a593Smuzhiyun
660*4882a593Smuzhiyun ah = devm_kzalloc(sc->dev, sizeof(struct ath_hw), GFP_KERNEL);
661*4882a593Smuzhiyun if (!ah)
662*4882a593Smuzhiyun return -ENOMEM;
663*4882a593Smuzhiyun
664*4882a593Smuzhiyun ah->dev = sc->dev;
665*4882a593Smuzhiyun ah->hw = sc->hw;
666*4882a593Smuzhiyun ah->hw_version.devid = devid;
667*4882a593Smuzhiyun ah->ah_flags |= AH_USE_EEPROM;
668*4882a593Smuzhiyun ah->led_pin = -1;
669*4882a593Smuzhiyun ah->reg_ops.read = ath9k_ioread32;
670*4882a593Smuzhiyun ah->reg_ops.multi_read = ath9k_multi_ioread32;
671*4882a593Smuzhiyun ah->reg_ops.write = ath9k_iowrite32;
672*4882a593Smuzhiyun ah->reg_ops.rmw = ath9k_reg_rmw;
673*4882a593Smuzhiyun pCap = &ah->caps;
674*4882a593Smuzhiyun
675*4882a593Smuzhiyun common = ath9k_hw_common(ah);
676*4882a593Smuzhiyun
677*4882a593Smuzhiyun /* Will be cleared in ath9k_start() */
678*4882a593Smuzhiyun set_bit(ATH_OP_INVALID, &common->op_flags);
679*4882a593Smuzhiyun
680*4882a593Smuzhiyun sc->sc_ah = ah;
681*4882a593Smuzhiyun sc->dfs_detector = dfs_pattern_detector_init(common, NL80211_DFS_UNSET);
682*4882a593Smuzhiyun sc->tx99_power = MAX_RATE_POWER + 1;
683*4882a593Smuzhiyun init_waitqueue_head(&sc->tx_wait);
684*4882a593Smuzhiyun sc->cur_chan = &sc->chanctx[0];
685*4882a593Smuzhiyun if (!ath9k_is_chanctx_enabled())
686*4882a593Smuzhiyun sc->cur_chan->hw_queue_base = 0;
687*4882a593Smuzhiyun
688*4882a593Smuzhiyun common->ops = &ah->reg_ops;
689*4882a593Smuzhiyun common->bus_ops = bus_ops;
690*4882a593Smuzhiyun common->ps_ops = &ath9k_ps_ops;
691*4882a593Smuzhiyun common->ah = ah;
692*4882a593Smuzhiyun common->hw = sc->hw;
693*4882a593Smuzhiyun common->priv = sc;
694*4882a593Smuzhiyun common->debug_mask = ath9k_debug;
695*4882a593Smuzhiyun common->btcoex_enabled = ath9k_btcoex_enable == 1;
696*4882a593Smuzhiyun common->disable_ani = false;
697*4882a593Smuzhiyun
698*4882a593Smuzhiyun /*
699*4882a593Smuzhiyun * Platform quirks.
700*4882a593Smuzhiyun */
701*4882a593Smuzhiyun ath9k_init_pcoem_platform(sc);
702*4882a593Smuzhiyun
703*4882a593Smuzhiyun ret = ath9k_init_platform(sc);
704*4882a593Smuzhiyun if (ret)
705*4882a593Smuzhiyun return ret;
706*4882a593Smuzhiyun
707*4882a593Smuzhiyun ret = ath9k_of_init(sc);
708*4882a593Smuzhiyun if (ret)
709*4882a593Smuzhiyun return ret;
710*4882a593Smuzhiyun
711*4882a593Smuzhiyun if (ath9k_led_active_high != -1)
712*4882a593Smuzhiyun ah->config.led_active_high = ath9k_led_active_high == 1;
713*4882a593Smuzhiyun
714*4882a593Smuzhiyun /*
715*4882a593Smuzhiyun * Enable WLAN/BT RX Antenna diversity only when:
716*4882a593Smuzhiyun *
717*4882a593Smuzhiyun * - BTCOEX is disabled.
718*4882a593Smuzhiyun * - the user manually requests the feature.
719*4882a593Smuzhiyun * - the HW cap is set using the platform data.
720*4882a593Smuzhiyun */
721*4882a593Smuzhiyun if (!common->btcoex_enabled && ath9k_bt_ant_diversity &&
722*4882a593Smuzhiyun (pCap->hw_caps & ATH9K_HW_CAP_BT_ANT_DIV))
723*4882a593Smuzhiyun common->bt_ant_diversity = 1;
724*4882a593Smuzhiyun
725*4882a593Smuzhiyun spin_lock_init(&common->cc_lock);
726*4882a593Smuzhiyun spin_lock_init(&sc->intr_lock);
727*4882a593Smuzhiyun spin_lock_init(&sc->sc_serial_rw);
728*4882a593Smuzhiyun spin_lock_init(&sc->sc_pm_lock);
729*4882a593Smuzhiyun spin_lock_init(&sc->chan_lock);
730*4882a593Smuzhiyun mutex_init(&sc->mutex);
731*4882a593Smuzhiyun tasklet_setup(&sc->intr_tq, ath9k_tasklet);
732*4882a593Smuzhiyun tasklet_setup(&sc->bcon_tasklet, ath9k_beacon_tasklet);
733*4882a593Smuzhiyun
734*4882a593Smuzhiyun timer_setup(&sc->sleep_timer, ath_ps_full_sleep, 0);
735*4882a593Smuzhiyun INIT_WORK(&sc->hw_reset_work, ath_reset_work);
736*4882a593Smuzhiyun INIT_WORK(&sc->paprd_work, ath_paprd_calibrate);
737*4882a593Smuzhiyun INIT_DELAYED_WORK(&sc->hw_pll_work, ath_hw_pll_work);
738*4882a593Smuzhiyun INIT_DELAYED_WORK(&sc->hw_check_work, ath_hw_check_work);
739*4882a593Smuzhiyun
740*4882a593Smuzhiyun ath9k_init_channel_context(sc);
741*4882a593Smuzhiyun
742*4882a593Smuzhiyun /*
743*4882a593Smuzhiyun * Cache line size is used to size and align various
744*4882a593Smuzhiyun * structures used to communicate with the hardware.
745*4882a593Smuzhiyun */
746*4882a593Smuzhiyun ath_read_cachesize(common, &csz);
747*4882a593Smuzhiyun common->cachelsz = csz << 2; /* convert to bytes */
748*4882a593Smuzhiyun
749*4882a593Smuzhiyun /* Initializes the hardware for all supported chipsets */
750*4882a593Smuzhiyun ret = ath9k_hw_init(ah);
751*4882a593Smuzhiyun if (ret)
752*4882a593Smuzhiyun goto err_hw;
753*4882a593Smuzhiyun
754*4882a593Smuzhiyun ret = ath9k_init_queues(sc);
755*4882a593Smuzhiyun if (ret)
756*4882a593Smuzhiyun goto err_queues;
757*4882a593Smuzhiyun
758*4882a593Smuzhiyun ret = ath9k_init_btcoex(sc);
759*4882a593Smuzhiyun if (ret)
760*4882a593Smuzhiyun goto err_btcoex;
761*4882a593Smuzhiyun
762*4882a593Smuzhiyun ret = ath9k_cmn_init_channels_rates(common);
763*4882a593Smuzhiyun if (ret)
764*4882a593Smuzhiyun goto err_btcoex;
765*4882a593Smuzhiyun
766*4882a593Smuzhiyun ret = ath9k_init_p2p(sc);
767*4882a593Smuzhiyun if (ret)
768*4882a593Smuzhiyun goto err_btcoex;
769*4882a593Smuzhiyun
770*4882a593Smuzhiyun ath9k_cmn_init_crypto(sc->sc_ah);
771*4882a593Smuzhiyun ath9k_init_misc(sc);
772*4882a593Smuzhiyun ath_chanctx_init(sc);
773*4882a593Smuzhiyun ath9k_offchannel_init(sc);
774*4882a593Smuzhiyun
775*4882a593Smuzhiyun if (common->bus_ops->aspm_init)
776*4882a593Smuzhiyun common->bus_ops->aspm_init(common);
777*4882a593Smuzhiyun
778*4882a593Smuzhiyun return 0;
779*4882a593Smuzhiyun
780*4882a593Smuzhiyun err_btcoex:
781*4882a593Smuzhiyun for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++)
782*4882a593Smuzhiyun if (ATH_TXQ_SETUP(sc, i))
783*4882a593Smuzhiyun ath_tx_cleanupq(sc, &sc->tx.txq[i]);
784*4882a593Smuzhiyun err_queues:
785*4882a593Smuzhiyun ath9k_hw_deinit(ah);
786*4882a593Smuzhiyun err_hw:
787*4882a593Smuzhiyun ath9k_eeprom_release(sc);
788*4882a593Smuzhiyun dev_kfree_skb_any(sc->tx99_skb);
789*4882a593Smuzhiyun return ret;
790*4882a593Smuzhiyun }
791*4882a593Smuzhiyun
ath9k_init_band_txpower(struct ath_softc * sc,int band)792*4882a593Smuzhiyun static void ath9k_init_band_txpower(struct ath_softc *sc, int band)
793*4882a593Smuzhiyun {
794*4882a593Smuzhiyun struct ieee80211_supported_band *sband;
795*4882a593Smuzhiyun struct ieee80211_channel *chan;
796*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
797*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
798*4882a593Smuzhiyun struct cfg80211_chan_def chandef;
799*4882a593Smuzhiyun int i;
800*4882a593Smuzhiyun
801*4882a593Smuzhiyun sband = &common->sbands[band];
802*4882a593Smuzhiyun for (i = 0; i < sband->n_channels; i++) {
803*4882a593Smuzhiyun chan = &sband->channels[i];
804*4882a593Smuzhiyun ah->curchan = &ah->channels[chan->hw_value];
805*4882a593Smuzhiyun cfg80211_chandef_create(&chandef, chan, NL80211_CHAN_HT20);
806*4882a593Smuzhiyun ath9k_cmn_get_channel(sc->hw, ah, &chandef);
807*4882a593Smuzhiyun ath9k_hw_set_txpowerlimit(ah, MAX_COMBINED_POWER, true);
808*4882a593Smuzhiyun }
809*4882a593Smuzhiyun }
810*4882a593Smuzhiyun
ath9k_init_txpower_limits(struct ath_softc * sc)811*4882a593Smuzhiyun static void ath9k_init_txpower_limits(struct ath_softc *sc)
812*4882a593Smuzhiyun {
813*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
814*4882a593Smuzhiyun struct ath9k_channel *curchan = ah->curchan;
815*4882a593Smuzhiyun
816*4882a593Smuzhiyun if (ah->caps.hw_caps & ATH9K_HW_CAP_2GHZ)
817*4882a593Smuzhiyun ath9k_init_band_txpower(sc, NL80211_BAND_2GHZ);
818*4882a593Smuzhiyun if (ah->caps.hw_caps & ATH9K_HW_CAP_5GHZ)
819*4882a593Smuzhiyun ath9k_init_band_txpower(sc, NL80211_BAND_5GHZ);
820*4882a593Smuzhiyun
821*4882a593Smuzhiyun ah->curchan = curchan;
822*4882a593Smuzhiyun }
823*4882a593Smuzhiyun
824*4882a593Smuzhiyun static const struct ieee80211_iface_limit if_limits[] = {
825*4882a593Smuzhiyun { .max = 2048, .types = BIT(NL80211_IFTYPE_STATION) },
826*4882a593Smuzhiyun { .max = 8, .types =
827*4882a593Smuzhiyun #ifdef CONFIG_MAC80211_MESH
828*4882a593Smuzhiyun BIT(NL80211_IFTYPE_MESH_POINT) |
829*4882a593Smuzhiyun #endif
830*4882a593Smuzhiyun BIT(NL80211_IFTYPE_AP) },
831*4882a593Smuzhiyun { .max = 1, .types = BIT(NL80211_IFTYPE_P2P_CLIENT) |
832*4882a593Smuzhiyun BIT(NL80211_IFTYPE_P2P_GO) },
833*4882a593Smuzhiyun };
834*4882a593Smuzhiyun
835*4882a593Smuzhiyun #ifdef CONFIG_WIRELESS_WDS
836*4882a593Smuzhiyun static const struct ieee80211_iface_limit wds_limits[] = {
837*4882a593Smuzhiyun { .max = 2048, .types = BIT(NL80211_IFTYPE_WDS) },
838*4882a593Smuzhiyun };
839*4882a593Smuzhiyun #endif
840*4882a593Smuzhiyun
841*4882a593Smuzhiyun #ifdef CONFIG_ATH9K_CHANNEL_CONTEXT
842*4882a593Smuzhiyun
843*4882a593Smuzhiyun static const struct ieee80211_iface_limit if_limits_multi[] = {
844*4882a593Smuzhiyun { .max = 2, .types = BIT(NL80211_IFTYPE_STATION) |
845*4882a593Smuzhiyun BIT(NL80211_IFTYPE_AP) |
846*4882a593Smuzhiyun BIT(NL80211_IFTYPE_P2P_CLIENT) |
847*4882a593Smuzhiyun BIT(NL80211_IFTYPE_P2P_GO) },
848*4882a593Smuzhiyun { .max = 1, .types = BIT(NL80211_IFTYPE_ADHOC) },
849*4882a593Smuzhiyun { .max = 1, .types = BIT(NL80211_IFTYPE_P2P_DEVICE) },
850*4882a593Smuzhiyun };
851*4882a593Smuzhiyun
852*4882a593Smuzhiyun static const struct ieee80211_iface_combination if_comb_multi[] = {
853*4882a593Smuzhiyun {
854*4882a593Smuzhiyun .limits = if_limits_multi,
855*4882a593Smuzhiyun .n_limits = ARRAY_SIZE(if_limits_multi),
856*4882a593Smuzhiyun .max_interfaces = 3,
857*4882a593Smuzhiyun .num_different_channels = 2,
858*4882a593Smuzhiyun .beacon_int_infra_match = true,
859*4882a593Smuzhiyun },
860*4882a593Smuzhiyun };
861*4882a593Smuzhiyun
862*4882a593Smuzhiyun #endif /* CONFIG_ATH9K_CHANNEL_CONTEXT */
863*4882a593Smuzhiyun
864*4882a593Smuzhiyun static const struct ieee80211_iface_combination if_comb[] = {
865*4882a593Smuzhiyun {
866*4882a593Smuzhiyun .limits = if_limits,
867*4882a593Smuzhiyun .n_limits = ARRAY_SIZE(if_limits),
868*4882a593Smuzhiyun .max_interfaces = 2048,
869*4882a593Smuzhiyun .num_different_channels = 1,
870*4882a593Smuzhiyun .beacon_int_infra_match = true,
871*4882a593Smuzhiyun #ifdef CONFIG_ATH9K_DFS_CERTIFIED
872*4882a593Smuzhiyun .radar_detect_widths = BIT(NL80211_CHAN_WIDTH_20_NOHT) |
873*4882a593Smuzhiyun BIT(NL80211_CHAN_WIDTH_20) |
874*4882a593Smuzhiyun BIT(NL80211_CHAN_WIDTH_40),
875*4882a593Smuzhiyun #endif
876*4882a593Smuzhiyun },
877*4882a593Smuzhiyun #ifdef CONFIG_WIRELESS_WDS
878*4882a593Smuzhiyun {
879*4882a593Smuzhiyun .limits = wds_limits,
880*4882a593Smuzhiyun .n_limits = ARRAY_SIZE(wds_limits),
881*4882a593Smuzhiyun .max_interfaces = 2048,
882*4882a593Smuzhiyun .num_different_channels = 1,
883*4882a593Smuzhiyun .beacon_int_infra_match = true,
884*4882a593Smuzhiyun },
885*4882a593Smuzhiyun #endif
886*4882a593Smuzhiyun };
887*4882a593Smuzhiyun
888*4882a593Smuzhiyun #ifdef CONFIG_ATH9K_CHANNEL_CONTEXT
ath9k_set_mcc_capab(struct ath_softc * sc,struct ieee80211_hw * hw)889*4882a593Smuzhiyun static void ath9k_set_mcc_capab(struct ath_softc *sc, struct ieee80211_hw *hw)
890*4882a593Smuzhiyun {
891*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
892*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
893*4882a593Smuzhiyun
894*4882a593Smuzhiyun if (!ath9k_is_chanctx_enabled())
895*4882a593Smuzhiyun return;
896*4882a593Smuzhiyun
897*4882a593Smuzhiyun ieee80211_hw_set(hw, QUEUE_CONTROL);
898*4882a593Smuzhiyun hw->queues = ATH9K_NUM_TX_QUEUES;
899*4882a593Smuzhiyun hw->offchannel_tx_hw_queue = hw->queues - 1;
900*4882a593Smuzhiyun hw->wiphy->interface_modes &= ~ BIT(NL80211_IFTYPE_WDS);
901*4882a593Smuzhiyun hw->wiphy->iface_combinations = if_comb_multi;
902*4882a593Smuzhiyun hw->wiphy->n_iface_combinations = ARRAY_SIZE(if_comb_multi);
903*4882a593Smuzhiyun hw->wiphy->max_scan_ssids = 255;
904*4882a593Smuzhiyun hw->wiphy->max_scan_ie_len = IEEE80211_MAX_DATA_LEN;
905*4882a593Smuzhiyun hw->wiphy->max_remain_on_channel_duration = 10000;
906*4882a593Smuzhiyun hw->chanctx_data_size = sizeof(void *);
907*4882a593Smuzhiyun hw->extra_beacon_tailroom =
908*4882a593Smuzhiyun sizeof(struct ieee80211_p2p_noa_attr) + 9;
909*4882a593Smuzhiyun
910*4882a593Smuzhiyun ath_dbg(common, CHAN_CTX, "Use channel contexts\n");
911*4882a593Smuzhiyun }
912*4882a593Smuzhiyun #endif /* CONFIG_ATH9K_CHANNEL_CONTEXT */
913*4882a593Smuzhiyun
ath9k_set_hw_capab(struct ath_softc * sc,struct ieee80211_hw * hw)914*4882a593Smuzhiyun static void ath9k_set_hw_capab(struct ath_softc *sc, struct ieee80211_hw *hw)
915*4882a593Smuzhiyun {
916*4882a593Smuzhiyun struct ath_hw *ah = sc->sc_ah;
917*4882a593Smuzhiyun struct ath_common *common = ath9k_hw_common(ah);
918*4882a593Smuzhiyun
919*4882a593Smuzhiyun ieee80211_hw_set(hw, SUPPORTS_HT_CCK_RATES);
920*4882a593Smuzhiyun ieee80211_hw_set(hw, SUPPORTS_RC_TABLE);
921*4882a593Smuzhiyun ieee80211_hw_set(hw, REPORTS_TX_ACK_STATUS);
922*4882a593Smuzhiyun ieee80211_hw_set(hw, SPECTRUM_MGMT);
923*4882a593Smuzhiyun ieee80211_hw_set(hw, PS_NULLFUNC_STACK);
924*4882a593Smuzhiyun ieee80211_hw_set(hw, SIGNAL_DBM);
925*4882a593Smuzhiyun ieee80211_hw_set(hw, RX_INCLUDES_FCS);
926*4882a593Smuzhiyun ieee80211_hw_set(hw, HOST_BROADCAST_PS_BUFFERING);
927*4882a593Smuzhiyun ieee80211_hw_set(hw, SUPPORT_FAST_XMIT);
928*4882a593Smuzhiyun ieee80211_hw_set(hw, SUPPORTS_CLONED_SKBS);
929*4882a593Smuzhiyun
930*4882a593Smuzhiyun if (ath9k_ps_enable)
931*4882a593Smuzhiyun ieee80211_hw_set(hw, SUPPORTS_PS);
932*4882a593Smuzhiyun
933*4882a593Smuzhiyun if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_HT) {
934*4882a593Smuzhiyun ieee80211_hw_set(hw, AMPDU_AGGREGATION);
935*4882a593Smuzhiyun
936*4882a593Smuzhiyun if (AR_SREV_9280_20_OR_LATER(ah))
937*4882a593Smuzhiyun hw->radiotap_mcs_details |=
938*4882a593Smuzhiyun IEEE80211_RADIOTAP_MCS_HAVE_STBC;
939*4882a593Smuzhiyun }
940*4882a593Smuzhiyun
941*4882a593Smuzhiyun if (AR_SREV_9160_10_OR_LATER(sc->sc_ah) || ath9k_modparam_nohwcrypt)
942*4882a593Smuzhiyun ieee80211_hw_set(hw, MFP_CAPABLE);
943*4882a593Smuzhiyun
944*4882a593Smuzhiyun hw->wiphy->features |= NL80211_FEATURE_ACTIVE_MONITOR |
945*4882a593Smuzhiyun NL80211_FEATURE_AP_MODE_CHAN_WIDTH_CHANGE |
946*4882a593Smuzhiyun NL80211_FEATURE_P2P_GO_CTWIN;
947*4882a593Smuzhiyun
948*4882a593Smuzhiyun if (!IS_ENABLED(CONFIG_ATH9K_TX99)) {
949*4882a593Smuzhiyun hw->wiphy->interface_modes =
950*4882a593Smuzhiyun BIT(NL80211_IFTYPE_P2P_GO) |
951*4882a593Smuzhiyun BIT(NL80211_IFTYPE_P2P_CLIENT) |
952*4882a593Smuzhiyun BIT(NL80211_IFTYPE_AP) |
953*4882a593Smuzhiyun BIT(NL80211_IFTYPE_STATION) |
954*4882a593Smuzhiyun BIT(NL80211_IFTYPE_ADHOC) |
955*4882a593Smuzhiyun BIT(NL80211_IFTYPE_MESH_POINT) |
956*4882a593Smuzhiyun #ifdef CONFIG_WIRELESS_WDS
957*4882a593Smuzhiyun BIT(NL80211_IFTYPE_WDS) |
958*4882a593Smuzhiyun #endif
959*4882a593Smuzhiyun BIT(NL80211_IFTYPE_OCB);
960*4882a593Smuzhiyun
961*4882a593Smuzhiyun if (ath9k_is_chanctx_enabled())
962*4882a593Smuzhiyun hw->wiphy->interface_modes |=
963*4882a593Smuzhiyun BIT(NL80211_IFTYPE_P2P_DEVICE);
964*4882a593Smuzhiyun
965*4882a593Smuzhiyun hw->wiphy->iface_combinations = if_comb;
966*4882a593Smuzhiyun hw->wiphy->n_iface_combinations = ARRAY_SIZE(if_comb);
967*4882a593Smuzhiyun }
968*4882a593Smuzhiyun
969*4882a593Smuzhiyun hw->wiphy->flags &= ~WIPHY_FLAG_PS_ON_BY_DEFAULT;
970*4882a593Smuzhiyun
971*4882a593Smuzhiyun hw->wiphy->flags |= WIPHY_FLAG_IBSS_RSN;
972*4882a593Smuzhiyun hw->wiphy->flags |= WIPHY_FLAG_SUPPORTS_TDLS;
973*4882a593Smuzhiyun hw->wiphy->flags |= WIPHY_FLAG_HAS_REMAIN_ON_CHANNEL;
974*4882a593Smuzhiyun hw->wiphy->flags |= WIPHY_FLAG_SUPPORTS_5_10_MHZ;
975*4882a593Smuzhiyun hw->wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH;
976*4882a593Smuzhiyun hw->wiphy->flags |= WIPHY_FLAG_AP_UAPSD;
977*4882a593Smuzhiyun
978*4882a593Smuzhiyun hw->queues = 4;
979*4882a593Smuzhiyun hw->max_rates = 4;
980*4882a593Smuzhiyun hw->max_listen_interval = 10;
981*4882a593Smuzhiyun hw->max_rate_tries = 10;
982*4882a593Smuzhiyun hw->sta_data_size = sizeof(struct ath_node);
983*4882a593Smuzhiyun hw->vif_data_size = sizeof(struct ath_vif);
984*4882a593Smuzhiyun hw->txq_data_size = sizeof(struct ath_atx_tid);
985*4882a593Smuzhiyun hw->extra_tx_headroom = 4;
986*4882a593Smuzhiyun
987*4882a593Smuzhiyun hw->wiphy->available_antennas_rx = BIT(ah->caps.max_rxchains) - 1;
988*4882a593Smuzhiyun hw->wiphy->available_antennas_tx = BIT(ah->caps.max_txchains) - 1;
989*4882a593Smuzhiyun
990*4882a593Smuzhiyun /* single chain devices with rx diversity */
991*4882a593Smuzhiyun if (ah->caps.hw_caps & ATH9K_HW_CAP_ANT_DIV_COMB)
992*4882a593Smuzhiyun hw->wiphy->available_antennas_rx = BIT(0) | BIT(1);
993*4882a593Smuzhiyun
994*4882a593Smuzhiyun sc->ant_rx = hw->wiphy->available_antennas_rx;
995*4882a593Smuzhiyun sc->ant_tx = hw->wiphy->available_antennas_tx;
996*4882a593Smuzhiyun
997*4882a593Smuzhiyun if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_2GHZ)
998*4882a593Smuzhiyun hw->wiphy->bands[NL80211_BAND_2GHZ] =
999*4882a593Smuzhiyun &common->sbands[NL80211_BAND_2GHZ];
1000*4882a593Smuzhiyun if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_5GHZ)
1001*4882a593Smuzhiyun hw->wiphy->bands[NL80211_BAND_5GHZ] =
1002*4882a593Smuzhiyun &common->sbands[NL80211_BAND_5GHZ];
1003*4882a593Smuzhiyun
1004*4882a593Smuzhiyun #ifdef CONFIG_ATH9K_CHANNEL_CONTEXT
1005*4882a593Smuzhiyun ath9k_set_mcc_capab(sc, hw);
1006*4882a593Smuzhiyun #endif
1007*4882a593Smuzhiyun ath9k_init_wow(hw);
1008*4882a593Smuzhiyun ath9k_cmn_reload_chainmask(ah);
1009*4882a593Smuzhiyun
1010*4882a593Smuzhiyun SET_IEEE80211_PERM_ADDR(hw, common->macaddr);
1011*4882a593Smuzhiyun
1012*4882a593Smuzhiyun wiphy_ext_feature_set(hw->wiphy, NL80211_EXT_FEATURE_CQM_RSSI_LIST);
1013*4882a593Smuzhiyun wiphy_ext_feature_set(hw->wiphy, NL80211_EXT_FEATURE_AIRTIME_FAIRNESS);
1014*4882a593Smuzhiyun wiphy_ext_feature_set(hw->wiphy,
1015*4882a593Smuzhiyun NL80211_EXT_FEATURE_MULTICAST_REGISTRATIONS);
1016*4882a593Smuzhiyun wiphy_ext_feature_set(hw->wiphy, NL80211_EXT_FEATURE_CAN_REPLACE_PTK0);
1017*4882a593Smuzhiyun }
1018*4882a593Smuzhiyun
ath9k_init_device(u16 devid,struct ath_softc * sc,const struct ath_bus_ops * bus_ops)1019*4882a593Smuzhiyun int ath9k_init_device(u16 devid, struct ath_softc *sc,
1020*4882a593Smuzhiyun const struct ath_bus_ops *bus_ops)
1021*4882a593Smuzhiyun {
1022*4882a593Smuzhiyun struct ieee80211_hw *hw = sc->hw;
1023*4882a593Smuzhiyun struct ath_common *common;
1024*4882a593Smuzhiyun struct ath_hw *ah;
1025*4882a593Smuzhiyun int error = 0;
1026*4882a593Smuzhiyun struct ath_regulatory *reg;
1027*4882a593Smuzhiyun
1028*4882a593Smuzhiyun /* Bring up device */
1029*4882a593Smuzhiyun error = ath9k_init_softc(devid, sc, bus_ops);
1030*4882a593Smuzhiyun if (error)
1031*4882a593Smuzhiyun return error;
1032*4882a593Smuzhiyun
1033*4882a593Smuzhiyun ah = sc->sc_ah;
1034*4882a593Smuzhiyun common = ath9k_hw_common(ah);
1035*4882a593Smuzhiyun ath9k_set_hw_capab(sc, hw);
1036*4882a593Smuzhiyun
1037*4882a593Smuzhiyun /* Initialize regulatory */
1038*4882a593Smuzhiyun error = ath_regd_init(&common->regulatory, sc->hw->wiphy,
1039*4882a593Smuzhiyun ath9k_reg_notifier);
1040*4882a593Smuzhiyun if (error)
1041*4882a593Smuzhiyun goto deinit;
1042*4882a593Smuzhiyun
1043*4882a593Smuzhiyun reg = &common->regulatory;
1044*4882a593Smuzhiyun
1045*4882a593Smuzhiyun /* Setup TX DMA */
1046*4882a593Smuzhiyun error = ath_tx_init(sc, ATH_TXBUF);
1047*4882a593Smuzhiyun if (error != 0)
1048*4882a593Smuzhiyun goto deinit;
1049*4882a593Smuzhiyun
1050*4882a593Smuzhiyun /* Setup RX DMA */
1051*4882a593Smuzhiyun error = ath_rx_init(sc, ATH_RXBUF);
1052*4882a593Smuzhiyun if (error != 0)
1053*4882a593Smuzhiyun goto deinit;
1054*4882a593Smuzhiyun
1055*4882a593Smuzhiyun ath9k_init_txpower_limits(sc);
1056*4882a593Smuzhiyun
1057*4882a593Smuzhiyun #ifdef CONFIG_MAC80211_LEDS
1058*4882a593Smuzhiyun /* must be initialized before ieee80211_register_hw */
1059*4882a593Smuzhiyun sc->led_cdev.default_trigger = ieee80211_create_tpt_led_trigger(sc->hw,
1060*4882a593Smuzhiyun IEEE80211_TPT_LEDTRIG_FL_RADIO, ath9k_tpt_blink,
1061*4882a593Smuzhiyun ARRAY_SIZE(ath9k_tpt_blink));
1062*4882a593Smuzhiyun #endif
1063*4882a593Smuzhiyun
1064*4882a593Smuzhiyun /* Register with mac80211 */
1065*4882a593Smuzhiyun error = ieee80211_register_hw(hw);
1066*4882a593Smuzhiyun if (error)
1067*4882a593Smuzhiyun goto rx_cleanup;
1068*4882a593Smuzhiyun
1069*4882a593Smuzhiyun error = ath9k_init_debug(ah);
1070*4882a593Smuzhiyun if (error) {
1071*4882a593Smuzhiyun ath_err(common, "Unable to create debugfs files\n");
1072*4882a593Smuzhiyun goto unregister;
1073*4882a593Smuzhiyun }
1074*4882a593Smuzhiyun
1075*4882a593Smuzhiyun /* Handle world regulatory */
1076*4882a593Smuzhiyun if (!ath_is_world_regd(reg)) {
1077*4882a593Smuzhiyun error = regulatory_hint(hw->wiphy, reg->alpha2);
1078*4882a593Smuzhiyun if (error)
1079*4882a593Smuzhiyun goto debug_cleanup;
1080*4882a593Smuzhiyun }
1081*4882a593Smuzhiyun
1082*4882a593Smuzhiyun ath_init_leds(sc);
1083*4882a593Smuzhiyun ath_start_rfkill_poll(sc);
1084*4882a593Smuzhiyun
1085*4882a593Smuzhiyun return 0;
1086*4882a593Smuzhiyun
1087*4882a593Smuzhiyun debug_cleanup:
1088*4882a593Smuzhiyun ath9k_deinit_debug(sc);
1089*4882a593Smuzhiyun unregister:
1090*4882a593Smuzhiyun ieee80211_unregister_hw(hw);
1091*4882a593Smuzhiyun rx_cleanup:
1092*4882a593Smuzhiyun ath_rx_cleanup(sc);
1093*4882a593Smuzhiyun deinit:
1094*4882a593Smuzhiyun ath9k_deinit_softc(sc);
1095*4882a593Smuzhiyun return error;
1096*4882a593Smuzhiyun }
1097*4882a593Smuzhiyun
1098*4882a593Smuzhiyun /*****************************/
1099*4882a593Smuzhiyun /* De-Initialization */
1100*4882a593Smuzhiyun /*****************************/
1101*4882a593Smuzhiyun
ath9k_deinit_softc(struct ath_softc * sc)1102*4882a593Smuzhiyun static void ath9k_deinit_softc(struct ath_softc *sc)
1103*4882a593Smuzhiyun {
1104*4882a593Smuzhiyun int i = 0;
1105*4882a593Smuzhiyun
1106*4882a593Smuzhiyun ath9k_deinit_p2p(sc);
1107*4882a593Smuzhiyun ath9k_deinit_btcoex(sc);
1108*4882a593Smuzhiyun
1109*4882a593Smuzhiyun for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++)
1110*4882a593Smuzhiyun if (ATH_TXQ_SETUP(sc, i))
1111*4882a593Smuzhiyun ath_tx_cleanupq(sc, &sc->tx.txq[i]);
1112*4882a593Smuzhiyun
1113*4882a593Smuzhiyun del_timer_sync(&sc->sleep_timer);
1114*4882a593Smuzhiyun ath9k_hw_deinit(sc->sc_ah);
1115*4882a593Smuzhiyun if (sc->dfs_detector != NULL)
1116*4882a593Smuzhiyun sc->dfs_detector->exit(sc->dfs_detector);
1117*4882a593Smuzhiyun
1118*4882a593Smuzhiyun ath9k_eeprom_release(sc);
1119*4882a593Smuzhiyun }
1120*4882a593Smuzhiyun
ath9k_deinit_device(struct ath_softc * sc)1121*4882a593Smuzhiyun void ath9k_deinit_device(struct ath_softc *sc)
1122*4882a593Smuzhiyun {
1123*4882a593Smuzhiyun struct ieee80211_hw *hw = sc->hw;
1124*4882a593Smuzhiyun
1125*4882a593Smuzhiyun ath9k_ps_wakeup(sc);
1126*4882a593Smuzhiyun
1127*4882a593Smuzhiyun wiphy_rfkill_stop_polling(sc->hw->wiphy);
1128*4882a593Smuzhiyun ath_deinit_leds(sc);
1129*4882a593Smuzhiyun
1130*4882a593Smuzhiyun ath9k_ps_restore(sc);
1131*4882a593Smuzhiyun
1132*4882a593Smuzhiyun ath9k_deinit_debug(sc);
1133*4882a593Smuzhiyun ath9k_deinit_wow(hw);
1134*4882a593Smuzhiyun ieee80211_unregister_hw(hw);
1135*4882a593Smuzhiyun ath_rx_cleanup(sc);
1136*4882a593Smuzhiyun ath9k_deinit_softc(sc);
1137*4882a593Smuzhiyun }
1138*4882a593Smuzhiyun
1139*4882a593Smuzhiyun /************************/
1140*4882a593Smuzhiyun /* Module Hooks */
1141*4882a593Smuzhiyun /************************/
1142*4882a593Smuzhiyun
ath9k_init(void)1143*4882a593Smuzhiyun static int __init ath9k_init(void)
1144*4882a593Smuzhiyun {
1145*4882a593Smuzhiyun int error;
1146*4882a593Smuzhiyun
1147*4882a593Smuzhiyun error = ath_pci_init();
1148*4882a593Smuzhiyun if (error < 0) {
1149*4882a593Smuzhiyun pr_err("No PCI devices found, driver not installed\n");
1150*4882a593Smuzhiyun error = -ENODEV;
1151*4882a593Smuzhiyun goto err_out;
1152*4882a593Smuzhiyun }
1153*4882a593Smuzhiyun
1154*4882a593Smuzhiyun error = ath_ahb_init();
1155*4882a593Smuzhiyun if (error < 0) {
1156*4882a593Smuzhiyun error = -ENODEV;
1157*4882a593Smuzhiyun goto err_pci_exit;
1158*4882a593Smuzhiyun }
1159*4882a593Smuzhiyun
1160*4882a593Smuzhiyun dmi_check_system(ath9k_quirks);
1161*4882a593Smuzhiyun
1162*4882a593Smuzhiyun return 0;
1163*4882a593Smuzhiyun
1164*4882a593Smuzhiyun err_pci_exit:
1165*4882a593Smuzhiyun ath_pci_exit();
1166*4882a593Smuzhiyun err_out:
1167*4882a593Smuzhiyun return error;
1168*4882a593Smuzhiyun }
1169*4882a593Smuzhiyun module_init(ath9k_init);
1170*4882a593Smuzhiyun
ath9k_exit(void)1171*4882a593Smuzhiyun static void __exit ath9k_exit(void)
1172*4882a593Smuzhiyun {
1173*4882a593Smuzhiyun is_ath9k_unloaded = true;
1174*4882a593Smuzhiyun ath_ahb_exit();
1175*4882a593Smuzhiyun ath_pci_exit();
1176*4882a593Smuzhiyun pr_info("%s: Driver unloaded\n", dev_info);
1177*4882a593Smuzhiyun }
1178*4882a593Smuzhiyun module_exit(ath9k_exit);
1179