xref: /OK3568_Linux_fs/kernel/drivers/net/wireless/ath/ath9k/init.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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