xref: /freebsd/sys/dev/iwi/if_iwivar.h (revision 71625ec9ad2a9bc8c09784fbd23b759830e0ee5f)
1ceaec73dSDamien Bergamini 
2ceaec73dSDamien Bergamini /*-
3*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
4718cf2ccSPedro F. Giffuni  *
5c10b1400SMax Laier  * Copyright (c) 2004, 2005
6ceaec73dSDamien Bergamini  *      Damien Bergamini <damien.bergamini@free.fr>. All rights reserved.
7ceaec73dSDamien Bergamini  *
8ceaec73dSDamien Bergamini  * Redistribution and use in source and binary forms, with or without
9ceaec73dSDamien Bergamini  * modification, are permitted provided that the following conditions
10ceaec73dSDamien Bergamini  * are met:
11ceaec73dSDamien Bergamini  * 1. Redistributions of source code must retain the above copyright
12ceaec73dSDamien Bergamini  *    notice unmodified, this list of conditions, and the following
13ceaec73dSDamien Bergamini  *    disclaimer.
14ceaec73dSDamien Bergamini  * 2. Redistributions in binary form must reproduce the above copyright
15ceaec73dSDamien Bergamini  *    notice, this list of conditions and the following disclaimer in the
16ceaec73dSDamien Bergamini  *    documentation and/or other materials provided with the distribution.
17ceaec73dSDamien Bergamini  *
18ceaec73dSDamien Bergamini  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19ceaec73dSDamien Bergamini  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20ceaec73dSDamien Bergamini  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21ceaec73dSDamien Bergamini  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22ceaec73dSDamien Bergamini  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23ceaec73dSDamien Bergamini  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24ceaec73dSDamien Bergamini  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25ceaec73dSDamien Bergamini  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26ceaec73dSDamien Bergamini  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27ceaec73dSDamien Bergamini  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28ceaec73dSDamien Bergamini  * SUCH DAMAGE.
29ceaec73dSDamien Bergamini  */
30ceaec73dSDamien Bergamini 
31ceaec73dSDamien Bergamini struct iwi_rx_radiotap_header {
32ceaec73dSDamien Bergamini 	struct ieee80211_radiotap_header wr_ihdr;
33ceaec73dSDamien Bergamini 	uint8_t		wr_flags;
34ceaec73dSDamien Bergamini 	uint8_t		wr_rate;
35ceaec73dSDamien Bergamini 	uint16_t	wr_chan_freq;
36ceaec73dSDamien Bergamini 	uint16_t	wr_chan_flags;
375463c4a4SSam Leffler 	int8_t		wr_antsignal;
385463c4a4SSam Leffler 	int8_t		wr_antnoise;
39ceaec73dSDamien Bergamini 	uint8_t		wr_antenna;
40786ac703SAndriy Voskoboinyk } __packed __aligned(8);
41ceaec73dSDamien Bergamini 
42ceaec73dSDamien Bergamini #define IWI_RX_RADIOTAP_PRESENT						\
43ceaec73dSDamien Bergamini 	((1 << IEEE80211_RADIOTAP_FLAGS) |				\
44ceaec73dSDamien Bergamini 	 (1 << IEEE80211_RADIOTAP_RATE) |				\
45ceaec73dSDamien Bergamini 	 (1 << IEEE80211_RADIOTAP_CHANNEL) |				\
4607f8bbeaSSam Leffler 	 (1 << IEEE80211_RADIOTAP_DBM_ANTSIGNAL) |			\
4707f8bbeaSSam Leffler 	 (1 << IEEE80211_RADIOTAP_DBM_ANTNOISE) |			\
48ceaec73dSDamien Bergamini 	 (1 << IEEE80211_RADIOTAP_ANTENNA))
49ceaec73dSDamien Bergamini 
50ceaec73dSDamien Bergamini struct iwi_tx_radiotap_header {
51ceaec73dSDamien Bergamini 	struct ieee80211_radiotap_header wt_ihdr;
52ceaec73dSDamien Bergamini 	uint8_t		wt_flags;
53786ac703SAndriy Voskoboinyk 	uint8_t		wt_pad;
54ceaec73dSDamien Bergamini 	uint16_t	wt_chan_freq;
55ceaec73dSDamien Bergamini 	uint16_t	wt_chan_flags;
56786ac703SAndriy Voskoboinyk } __packed;
57ceaec73dSDamien Bergamini 
58ceaec73dSDamien Bergamini #define IWI_TX_RADIOTAP_PRESENT						\
59ceaec73dSDamien Bergamini 	((1 << IEEE80211_RADIOTAP_FLAGS) |				\
60ceaec73dSDamien Bergamini 	 (1 << IEEE80211_RADIOTAP_CHANNEL))
61ceaec73dSDamien Bergamini 
62ceaec73dSDamien Bergamini struct iwi_cmd_ring {
63ceaec73dSDamien Bergamini 	bus_dma_tag_t		desc_dmat;
64ceaec73dSDamien Bergamini 	bus_dmamap_t		desc_map;
65ceaec73dSDamien Bergamini 	bus_addr_t		physaddr;
66ceaec73dSDamien Bergamini 	struct iwi_cmd_desc	*desc;
67ceaec73dSDamien Bergamini 	int			count;
68ceaec73dSDamien Bergamini 	int			queued;
69ceaec73dSDamien Bergamini 	int			cur;
70ceaec73dSDamien Bergamini 	int			next;
71ceaec73dSDamien Bergamini };
72ceaec73dSDamien Bergamini 
73ceaec73dSDamien Bergamini struct iwi_tx_data {
74ceaec73dSDamien Bergamini 	bus_dmamap_t		map;
75ceaec73dSDamien Bergamini 	struct mbuf		*m;
76ceaec73dSDamien Bergamini 	struct ieee80211_node	*ni;
77ceaec73dSDamien Bergamini };
78ceaec73dSDamien Bergamini 
79ceaec73dSDamien Bergamini struct iwi_tx_ring {
80ceaec73dSDamien Bergamini 	bus_dma_tag_t		desc_dmat;
81ceaec73dSDamien Bergamini 	bus_dma_tag_t		data_dmat;
82ceaec73dSDamien Bergamini 	bus_dmamap_t		desc_map;
83ceaec73dSDamien Bergamini 	bus_addr_t		physaddr;
84ec01dc2bSDamien Bergamini 	bus_addr_t		csr_ridx;
85ec01dc2bSDamien Bergamini 	bus_addr_t		csr_widx;
86ceaec73dSDamien Bergamini 	struct iwi_tx_desc	*desc;
87ceaec73dSDamien Bergamini 	struct iwi_tx_data	*data;
88ceaec73dSDamien Bergamini 	int			count;
89ceaec73dSDamien Bergamini 	int			queued;
90ceaec73dSDamien Bergamini 	int			cur;
91ceaec73dSDamien Bergamini 	int			next;
92ceaec73dSDamien Bergamini };
93ceaec73dSDamien Bergamini 
94ceaec73dSDamien Bergamini struct iwi_rx_data {
95ceaec73dSDamien Bergamini 	bus_dmamap_t	map;
96ceaec73dSDamien Bergamini 	bus_addr_t	physaddr;
97ceaec73dSDamien Bergamini 	uint32_t	reg;
98ceaec73dSDamien Bergamini 	struct mbuf	*m;
99ceaec73dSDamien Bergamini };
100ceaec73dSDamien Bergamini 
101ceaec73dSDamien Bergamini struct iwi_rx_ring {
102ceaec73dSDamien Bergamini 	bus_dma_tag_t		data_dmat;
103ceaec73dSDamien Bergamini 	struct iwi_rx_data	*data;
104ceaec73dSDamien Bergamini 	int			count;
105ceaec73dSDamien Bergamini 	int			cur;
106ceaec73dSDamien Bergamini };
107ceaec73dSDamien Bergamini 
10828a38073SDamien Bergamini struct iwi_node {
10928a38073SDamien Bergamini 	struct ieee80211_node	in_node;
11028a38073SDamien Bergamini 	int			in_station;
11128a38073SDamien Bergamini #define IWI_MAX_IBSSNODE	32
11228a38073SDamien Bergamini };
11328a38073SDamien Bergamini 
114c10b1400SMax Laier struct iwi_fw {
11533d54970SLuigi Rizzo 	const struct firmware	*fp;		/* image handle */
116c10b1400SMax Laier 	const char		*data;		/* firmware image data */
117c10b1400SMax Laier 	size_t			size;		/* firmware image size */
118c10b1400SMax Laier 	const char		*name;		/* associated image name */
119c10b1400SMax Laier };
120c10b1400SMax Laier 
121b032f27cSSam Leffler struct iwi_vap {
122b032f27cSSam Leffler 	struct ieee80211vap	iwi_vap;
123b032f27cSSam Leffler 
124b032f27cSSam Leffler 	int			(*iwi_newstate)(struct ieee80211vap *,
125b032f27cSSam Leffler 				    enum ieee80211_state, int);
126b032f27cSSam Leffler };
127b032f27cSSam Leffler #define	IWI_VAP(vap)	((struct iwi_vap *)(vap))
128b032f27cSSam Leffler 
129ceaec73dSDamien Bergamini struct iwi_softc {
1307a79cebfSGleb Smirnoff 	struct mtx		sc_mtx;
1317a79cebfSGleb Smirnoff 	struct ieee80211com	sc_ic;
1327a79cebfSGleb Smirnoff 	struct mbufq		sc_snd;
133ceaec73dSDamien Bergamini 	device_t		sc_dev;
134ceaec73dSDamien Bergamini 
1357a79cebfSGleb Smirnoff 	void			(*sc_node_free)(struct ieee80211_node *);
1367a79cebfSGleb Smirnoff 
137c10b1400SMax Laier 	uint8_t			sc_mcast[IEEE80211_ADDR_LEN];
13828a38073SDamien Bergamini 	struct unrhdr		*sc_unr;
139ceaec73dSDamien Bergamini 
140ceaec73dSDamien Bergamini 	uint32_t		flags;
141408146edSDamien Bergamini #define IWI_FLAG_FW_INITED	(1 << 0)
142c10b1400SMax Laier #define	IWI_FLAG_BUSY		(1 << 3)	/* busy sending a command */
143c10b1400SMax Laier #define	IWI_FLAG_ASSOCIATED	(1 << 4)	/* currently associated  */
14468e8e04eSSam Leffler #define IWI_FLAG_CHANNEL_SCAN	(1 << 5)
14568e8e04eSSam Leffler 	uint32_t		fw_state;
14668e8e04eSSam Leffler #define IWI_FW_IDLE		0
14768e8e04eSSam Leffler #define IWI_FW_LOADING		1
14868e8e04eSSam Leffler #define IWI_FW_ASSOCIATING	2
14968e8e04eSSam Leffler #define IWI_FW_DISASSOCIATING	3
15068e8e04eSSam Leffler #define IWI_FW_SCANNING		4
151ceaec73dSDamien Bergamini 	struct iwi_cmd_ring	cmdq;
152ec01dc2bSDamien Bergamini 	struct iwi_tx_ring	txq[WME_NUM_AC];
153ceaec73dSDamien Bergamini 	struct iwi_rx_ring	rxq;
154ceaec73dSDamien Bergamini 
155ceaec73dSDamien Bergamini 	struct resource		*irq;
156ceaec73dSDamien Bergamini 	struct resource		*mem;
157ceaec73dSDamien Bergamini 	bus_space_tag_t		sc_st;
158ceaec73dSDamien Bergamini 	bus_space_handle_t	sc_sh;
159ceaec73dSDamien Bergamini 	void 			*sc_ih;
160ceaec73dSDamien Bergamini 
161484f6530SLuigi Rizzo 	/*
162484f6530SLuigi Rizzo 	 * The card needs external firmware images to work, which is made of a
163484f6530SLuigi Rizzo 	 * bootloader, microcode and firmware proper. In version 3.00 and
164484f6530SLuigi Rizzo 	 * above, all pieces are contained in a single image, preceded by a
165484f6530SLuigi Rizzo 	 * struct iwi_firmware_hdr indicating the size of the 3 pieces.
166484f6530SLuigi Rizzo 	 * Old firmware < 3.0 has separate boot and ucode, so we need to
167484f6530SLuigi Rizzo 	 * load all of them explicitly.
168484f6530SLuigi Rizzo 	 * To avoid issues related to fragmentation, we keep the block of
169484f6530SLuigi Rizzo 	 * dma-ble memory around until detach time, and reallocate it when
170484f6530SLuigi Rizzo 	 * it becomes too small. fw_dma_size is the size currently allocated.
171484f6530SLuigi Rizzo 	 */
172c10b1400SMax Laier 	int			fw_dma_size;
173484f6530SLuigi Rizzo 	uint32_t		fw_flags;	/* allocation status */
174484f6530SLuigi Rizzo #define	IWI_FW_HAVE_DMAT	0x01
175484f6530SLuigi Rizzo #define	IWI_FW_HAVE_MAP		0x02
176484f6530SLuigi Rizzo #define	IWI_FW_HAVE_PHY		0x04
177c10b1400SMax Laier 	bus_dma_tag_t		fw_dmat;
178c10b1400SMax Laier 	bus_dmamap_t		fw_map;
179c10b1400SMax Laier 	bus_addr_t		fw_physaddr;
180c10b1400SMax Laier 	void			*fw_virtaddr;
181c10b1400SMax Laier 	enum ieee80211_opmode	fw_mode;	/* mode of current firmware */
182c10b1400SMax Laier 	struct iwi_fw		fw_boot;	/* boot firmware */
183c10b1400SMax Laier 	struct iwi_fw		fw_uc;		/* microcode */
184c10b1400SMax Laier 	struct iwi_fw		fw_fw;		/* operating mode support */
185c10b1400SMax Laier 
186c10b1400SMax Laier 	int			curchan;	/* current h/w channel # */
187acea4241SDamien Bergamini 	int			antenna;
188ceaec73dSDamien Bergamini 	int			bluetooth;
189c10b1400SMax Laier 	struct iwi_associate	assoc;
190c10b1400SMax Laier 	struct iwi_wme_params	wme[3];
19168e8e04eSSam Leffler 	u_int			sc_scangen;
192c10b1400SMax Laier 
193c10b1400SMax Laier 	struct task		sc_radiontask;	/* radio on processing */
194c10b1400SMax Laier 	struct task		sc_radiofftask;	/* radio off processing */
195c10b1400SMax Laier 	struct task		sc_restarttask;	/* restart adapter processing */
1965efea30fSAndrew Thompson 	struct task		sc_disassoctask;
19740668dc5SBernhard Schmidt 	struct task		sc_monitortask;
198c10b1400SMax Laier 
1997a79cebfSGleb Smirnoff 	unsigned int		sc_running : 1,	/* initialized */
2007a79cebfSGleb Smirnoff 				sc_softled : 1,	/* enable LED gpio status */
201c10b1400SMax Laier 				sc_ledstate: 1,	/* LED on/off state */
202c10b1400SMax Laier 				sc_blinking: 1;	/* LED blink operation active */
203c10b1400SMax Laier 	u_int			sc_nictype;	/* NIC type from EEPROM */
204c10b1400SMax Laier 	u_int			sc_ledpin;	/* mask for activity LED */
205c10b1400SMax Laier 	u_int			sc_ledidle;	/* idle polling interval */
206c10b1400SMax Laier 	int			sc_ledevent;	/* time of last LED event */
207c10b1400SMax Laier 	u_int8_t		sc_rxrate;	/* current rx rate for LED */
208c10b1400SMax Laier 	u_int8_t		sc_rxrix;
209c10b1400SMax Laier 	u_int8_t		sc_txrate;	/* current tx rate for LED */
210c10b1400SMax Laier 	u_int8_t		sc_txrix;
211c10b1400SMax Laier 	u_int16_t		sc_ledoff;	/* off time for current blink */
212c10b1400SMax Laier 	struct callout		sc_ledtimer;	/* led off timer */
21368e8e04eSSam Leffler 	struct callout		sc_wdtimer;	/* watchdog timer */
214b032f27cSSam Leffler 	struct callout		sc_rftimer;	/* rfkill timer */
215ceaec73dSDamien Bergamini 
216ceaec73dSDamien Bergamini 	int			sc_tx_timer;
21768e8e04eSSam Leffler 	int			sc_state_timer;	/* firmware state timer */
21868e8e04eSSam Leffler 	int			sc_busy_timer;	/* firmware cmd timer */
219ceaec73dSDamien Bergamini 
220b032f27cSSam Leffler 	struct iwi_rx_radiotap_header sc_rxtap;
221b032f27cSSam Leffler 	struct iwi_tx_radiotap_header sc_txtap;
2224982d693SAdrian Chadd 
2234982d693SAdrian Chadd 	struct iwi_notif_link_quality sc_linkqual;
2244982d693SAdrian Chadd 	int			sc_linkqual_valid;
225ceaec73dSDamien Bergamini };
226ceaec73dSDamien Bergamini 
22768e8e04eSSam Leffler #define	IWI_STATE_BEGIN(_sc, _state)	do {			\
22868e8e04eSSam Leffler 	KASSERT(_sc->fw_state == IWI_FW_IDLE,			\
229853569c6SSam Leffler 	    ("iwi firmware not idle, state %s", iwi_fw_states[_sc->fw_state]));\
23068e8e04eSSam Leffler 	_sc->fw_state = _state;					\
23168e8e04eSSam Leffler 	_sc->sc_state_timer = 5;				\
232b032f27cSSam Leffler 	DPRINTF(("enter %s state\n", iwi_fw_states[_state]));	\
23368e8e04eSSam Leffler } while (0)
23468e8e04eSSam Leffler 
23568e8e04eSSam Leffler #define	IWI_STATE_END(_sc, _state)	do {			\
23668e8e04eSSam Leffler 	if (_sc->fw_state == _state)				\
237b032f27cSSam Leffler 		DPRINTF(("exit %s state\n", iwi_fw_states[_state])); \
23868e8e04eSSam Leffler 	 else							\
239b032f27cSSam Leffler 		DPRINTF(("expected %s state, got %s\n",	\
240b032f27cSSam Leffler 		    iwi_fw_states[_state], iwi_fw_states[_sc->fw_state])); \
24168e8e04eSSam Leffler 	_sc->fw_state = IWI_FW_IDLE;				\
24268e8e04eSSam Leffler 	wakeup(_sc);						\
24368e8e04eSSam Leffler 	_sc->sc_state_timer = 0;				\
24468e8e04eSSam Leffler } while (0)
245c10b1400SMax Laier /*
246c10b1400SMax Laier  * NB.: This models the only instance of async locking in iwi_init_locked
247c10b1400SMax Laier  *	and must be kept in sync.
248c10b1400SMax Laier  */
24968e8e04eSSam Leffler #define	IWI_LOCK_INIT(sc) \
25068e8e04eSSam Leffler 	mtx_init(&(sc)->sc_mtx, device_get_nameunit((sc)->sc_dev), \
25168e8e04eSSam Leffler 	    MTX_NETWORK_LOCK, MTX_DEF)
25268e8e04eSSam Leffler #define	IWI_LOCK_DESTROY(sc)	mtx_destroy(&(sc)->sc_mtx)
253c10b1400SMax Laier #define	IWI_LOCK_DECL	int	__waslocked = 0
25468e8e04eSSam Leffler #define IWI_LOCK_ASSERT(sc)	mtx_assert(&(sc)->sc_mtx, MA_OWNED)
255c10b1400SMax Laier #define IWI_LOCK(sc)	do {				\
256c10b1400SMax Laier 	if (!(__waslocked = mtx_owned(&(sc)->sc_mtx)))	\
257c10b1400SMax Laier 		mtx_lock(&(sc)->sc_mtx);		\
258c10b1400SMax Laier } while (0)
259c10b1400SMax Laier #define IWI_UNLOCK(sc)	do {			\
260c10b1400SMax Laier 	if (!__waslocked)			\
261c10b1400SMax Laier 		mtx_unlock(&(sc)->sc_mtx);	\
262c10b1400SMax Laier } while (0)
263