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