Linux Kernel
3.7.1
|
#include <linux/types.h>
Go to the source code of this file.
Data Structures | |
struct | iwl_rb_status |
struct | iwl_tfd_tb |
struct | iwl_tfd |
struct | iwlagn_scd_bc_tbl |
Variables | |
struct iwl_rb_status | __packed |
#define FH_KW_MEM_ADDR_REG (FH_MEM_LOWER_BOUND + 0x97C) |
Keep-Warm (KW) buffer base address.
Driver must allocate a 4KByte buffer that is for keeping the host DRAM powered on (via dummy accesses to DRAM) to maintain low-latency DRAM access when doing Txing or Rxing. The dummy accesses prevent host from going into a power-savings mode that would cause higher DRAM latency, and possible data over/under-runs, before all Tx/Rx is complete.
Driver loads FH_KW_MEM_ADDR_REG with the physical address (bits 35:4) of the buffer, which must be 4K aligned. Once this is set up, the device automatically invokes keep-warm accesses when normal accesses might not be sufficient to maintain fast DRAM response.
Bit fields: 31-0: Keep-warm buffer physical base address [35:4], must be 4K aligned
#define FH_MEM_CBBC_0_15_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0x9D0) |
TFD Circular Buffers Base (CBBC) addresses
Device has 16 base pointer registers, one for each of 16 host-DRAM-resident circular buffers (CBs/queues) containing Transmit Frame Descriptors (TFDs) (see struct iwl_tfd_frame). These 16 pointer registers are offset by 0x04 bytes from one another. Each TFD circular buffer in DRAM must be 256-byte aligned (address bits 0-7 must be 0). Later devices have 20 (5000 series) or 30 (higher) queues, but the registers for them are in different places.
Bit fields in each pointer register: 27-0: TFD CB physical base address [35:8], must be 256-byte aligned
#define FH_MEM_CBBC_0_15_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xA10) |
#define FH_MEM_CBBC_16_19_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xBF0) |
#define FH_MEM_CBBC_16_19_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xC00) |
#define FH_MEM_CBBC_20_31_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xB20) |
#define FH_MEM_CBBC_20_31_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xB80) |
#define FH_MEM_LOWER_BOUND (0x1000) |
#define FH_MEM_RCSR_CHNL0 (FH_MEM_RCSR_LOWER_BOUND) |
#define FH_MEM_RCSR_CHNL0_CONFIG_REG (FH_MEM_RCSR_CHNL0) |
#define FH_MEM_RCSR_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xC00) |
Rx Config/Status Registers (RCSR) Rx Config Reg for channel 0 (only channel used)
Driver must initialize FH_MEM_RCSR_CHNL0_CONFIG_REG as follows for normal operation (see bit fields).
Clearing FH_MEM_RCSR_CHNL0_CONFIG_REG to 0 turns off Rx DMA. Driver should poll FH_MEM_RSSR_RX_STATUS_REG for FH_RSSR_CHNL0_RX_STATUS_CHNL_IDLE (bit 24) before continuing.
Bit fields: 31-30: Rx DMA channel enable: '00' off/pause, '01' pause at end of frame, '10' operate normally 29-24: reserved 23-20: # RBDs in circular buffer = 2^value; use "8" for 256 RBDs (normal), min "5" for 32 RBDs, max "12" for 4096 RBDs. 19-18: reserved 17-16: size of each receive buffer; '00' 4K (normal), '01' 8K, '10' 12K, '11' 16K. 15-14: reserved 13-12: IRQ destination; '00' none, '01' host driver (normal operation) 11- 4: timeout for closing Rx buffer and interrupting host (units 32 usec) typical value 0x10 (about 1/2 msec) 3- 0: reserved
#define FH_MEM_RCSR_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xCC0) |
#define FH_MEM_RSCSR_CHNL0 (FH_MEM_RSCSR_LOWER_BOUND) |
#define FH_MEM_RSCSR_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xBC0) |
Rx SRAM Control and Status Registers (RSCSR)
These registers provide handshake between driver and device for the Rx queue (this queue handles all command responses, notifications, Rx data, etc. sent from uCode to host driver). Unlike Tx, there is only one Rx queue, and only one Rx DMA/FIFO channel. Also unlike Tx, which can concatenate up to 20 DRAM buffers to form a Tx frame, each Receive Buffer Descriptor (RBD) points to only one Rx Buffer (RB); there is a 1:1 mapping between RBDs and RBs.
Driver must allocate host DRAM memory for the following, and set the physical address of each into device registers:
1) Receive Buffer Descriptor (RBD) circular buffer (CB), typically with 256 entries (although any power of 2, up to 4096, is selectable by driver). Each entry (1 dword) points to a receive buffer (RB) of consistent size (typically 4K, although 8K or 16K are also selectable by driver). Driver sets up RB size and number of RBDs in the CB via Rx config register FH_MEM_RCSR_CHNL0_CONFIG_REG.
Bit fields within one RBD: 27-0: Receive Buffer physical address bits [35:8], 256-byte aligned
Driver sets physical address [35:8] of base of RBD circular buffer into FH_RSCSR_CHNL0_RBDCB_BASE_REG [27:0].
2) Rx status buffer, 8 bytes, in which uCode indicates which Rx Buffers (RBs) have been filled, via a "write pointer", actually the index of the RB's corresponding RBD within the circular buffer. Driver sets physical address [35:4] into FH_RSCSR_CHNL0_STTS_WPTR_REG [31:0].
Bit fields in lower dword of Rx status buffer (upper dword not used by driver: 31-12: Not used by driver 11- 0: Index of last filled Rx buffer descriptor (device writes, driver reads this value)
As the driver prepares Receive Buffers (RBs) for device to fill, driver must enter pointers to these RBs into contiguous RBD circular buffer entries, and update the device's "write" index register, FH_RSCSR_CHNL0_RBDCB_WPTR_REG.
This "write" index corresponds to the next RBD that the driver will make available, i.e. one RBD past the tail of the ready-to-fill RBDs within the circular buffer. This value should initially be 0 (before preparing any RBs), should be 8 after preparing the first 8 RBs (for example), and must wrap back to 0 at the end of the circular buffer (but don't wrap before "read" index has advanced past 1! See below). NOTE: DEVICE EXPECTS THE WRITE INDEX TO BE INCREMENTED IN MULTIPLES OF 8.
As the device fills RBs (referenced from contiguous RBDs within the circular buffer), it updates the Rx status buffer in host DRAM, 2) described above, to tell the driver the index of the latest filled RBD. The driver must read this "read" index from DRAM after receiving an Rx interrupt from device
The driver must also internally keep track of a third index, which is the next RBD to process. When receiving an Rx interrupt, driver should process all filled but unprocessed RBs up to, but not including, the RB corresponding to the "read" index. For example, if "read" index becomes "1", driver may process the RB pointed to by RBD 0. Depending on volume of traffic, there may be many RBs to process.
If read index == write index, device thinks there is no room to put new data. Due to this, the maximum number of filled RBs is 255, instead of 256. To be safe, make sure that there is a gap of at least 2 RBDs between "write" and "read" indexes; that is, make sure that there are no more than 254 buffers waiting to be filled.
#define FH_MEM_RSCSR_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xC00) |
#define FH_MEM_RSSR_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xC40) |
Rx Shared Status Registers (RSSR)
After stopping Rx DMA channel (writing 0 to FH_MEM_RCSR_CHNL0_CONFIG_REG), driver must poll FH_MEM_RSSR_RX_STATUS_REG until Rx channel is idle.
Bit fields: 24: 1 = Channel 0 is idle
FH_MEM_RSSR_SHARED_CTRL_REG and FH_MEM_RSSR_RX_ENABLE_ERR_IRQ2DRV contain default values that should not be altered by the driver.
#define FH_MEM_RSSR_RX_ENABLE_ERR_IRQ2DRV (FH_MEM_RSSR_LOWER_BOUND + 0x008) |
#define FH_MEM_RSSR_RX_STATUS_REG (FH_MEM_RSSR_LOWER_BOUND + 0x004) |
#define FH_MEM_RSSR_SHARED_CTRL_REG (FH_MEM_RSSR_LOWER_BOUND) |
#define FH_MEM_RSSR_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xD00) |
#define FH_RCSR_CHNL0_RX_CONFIG_DMA_CHNL_EN_MSK (0xC0000000) /* bits 30-31*/ |
#define FH_RCSR_CHNL0_RX_CONFIG_IRQ_DEST_INT_HOST_VAL (0x00001000) |
#define FH_RCSR_CHNL0_RX_CONFIG_IRQ_DEST_MSK (0x00001000) /* bits 12 */ |
#define FH_RCSR_CHNL0_RX_CONFIG_IRQ_DEST_NO_INT_VAL (0x00000000) |
#define FH_RCSR_CHNL0_RX_CONFIG_RB_SIZE_MSK (0x00030000) /* bits 16-17 */ |
#define FH_RCSR_CHNL0_RX_CONFIG_RB_TIMEOUT_MSK (0x00000FF0) /* bits 4-11 */ |
#define FH_RCSR_CHNL0_RX_CONFIG_RBDBC_SIZE_MSK (0x00F00000) /* bits 20-23 */ |
#define FH_RCSR_CHNL0_RX_CONFIG_SINGLE_FRAME_MSK (0x00008000) /* bit 15 */ |
#define FH_RCSR_RX_CONFIG_CHNL_EN_PAUSE_EOF_VAL (0x40000000) |
#define FH_RSCSR_CHNL0_RBDCB_BASE_REG (FH_MEM_RSCSR_CHNL0 + 0x004) |
#define FH_RSCSR_CHNL0_RBDCB_WPTR_REG (FH_MEM_RSCSR_CHNL0 + 0x008) |
#define FH_RSCSR_CHNL0_STTS_WPTR_REG (FH_MEM_RSCSR_CHNL0) |
#define FH_RSCSR_CHNL0_WPTR (FH_RSCSR_CHNL0_RBDCB_WPTR_REG) |
#define FH_SRVC_CHNL_SRAM_ADDR_REG | ( | _chnl | ) | (FH_SRVC_LOWER_BOUND + ((_chnl) - 9) * 0x4) |
#define FH_SRVC_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0x9C8) |
#define FH_SRVC_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0x9D0) |
#define FH_TCSR_CHNL_TX_BUF_STS_REG | ( | _chnl | ) | (FH_TCSR_LOWER_BOUND + 0x20 * (_chnl) + 0x8) |
#define FH_TCSR_CHNL_TX_BUF_STS_REG_VAL_TFDB_EMPTY (0x00000000) |
#define FH_TCSR_CHNL_TX_BUF_STS_REG_VAL_TFDB_VALID (0x00000003) |
#define FH_TCSR_CHNL_TX_BUF_STS_REG_VAL_TFDB_WAIT (0x00002000) |
#define FH_TCSR_CHNL_TX_CONFIG_REG | ( | _chnl | ) | (FH_TCSR_LOWER_BOUND + 0x20 * (_chnl)) |
#define FH_TCSR_CHNL_TX_CREDIT_REG | ( | _chnl | ) | (FH_TCSR_LOWER_BOUND + 0x20 * (_chnl) + 0x4) |
#define FH_TCSR_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xD00) |
Transmit DMA Channel Control/Status Registers (TCSR)
Device has one configuration register for each of 8 Tx DMA/FIFO channels supported in hardware (don't confuse these with the 16 Tx queues in DRAM, which feed the DMA/FIFO channels); config regs are separated by 0x20 bytes.
To use a Tx DMA channel, driver must initialize its FH_TCSR_CHNL_TX_CONFIG_REG(chnl) with:
FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_ENABLE | FH_TCSR_TX_CONFIG_REG_VAL_DMA_CREDIT_ENABLE_VAL
All other bits should be 0.
Bit fields: 31-30: Tx DMA channel enable: '00' off/pause, '01' pause at end of frame, '10' operate normally 29- 4: Reserved, set to "0" 3: Enable internal DMA requests (1, normal operation), disable (0) 2- 0: Reserved, set to "0"
#define FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_HOST_ENDTFD (0x00100000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_HOST_IFTFD (0x00200000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_HOST_NOINT (0x00000000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_RTC_ENDTFD (0x00400000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_RTC_IFTFD (0x00800000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_RTC_NOINT (0x00000000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_ENABLE (0x80000000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_PAUSE (0x00000000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_DMA_CHNL_PAUSE_EOF (0x40000000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_DMA_CREDIT_DISABLE (0x00000000) |
#define FH_TCSR_TX_CONFIG_REG_VAL_DMA_CREDIT_ENABLE (0x00000008) |
#define FH_TCSR_TX_CONFIG_REG_VAL_MSG_MODE_DRV (0x00000001) |
#define FH_TCSR_TX_CONFIG_REG_VAL_MSG_MODE_TXF (0x00000000) |
#define FH_TCSR_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xE60) |
#define FH_TFDIB_CTRL0_REG | ( | _chnl | ) | (FH_TFDIB_LOWER_BOUND + 0x8 * (_chnl)) |
#define FH_TFDIB_CTRL1_REG | ( | _chnl | ) | (FH_TFDIB_LOWER_BOUND + 0x8 * (_chnl) + 0x4) |
#define FH_TFDIB_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0x900) |
#define FH_TFDIB_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0x958) |
#define FH_TSSR_LOWER_BOUND (FH_MEM_LOWER_BOUND + 0xEA0) |
Tx Shared Status Registers (TSSR)
After stopping Tx DMA channel (writing 0 to FH_TCSR_CHNL_TX_CONFIG_REG(chnl)), driver must poll FH_TSSR_TX_STATUS_REG until selected Tx channel is idle (channel's buffers empty | no pending requests).
Bit fields: 31-24: 1 = Channel buffers empty (channel 7:0) 23-16: 1 = No pending requests (channel 7:0)
#define FH_TSSR_TX_ERROR_REG (FH_TSSR_LOWER_BOUND + 0x018) |
Bit fields for TSSR(Tx Shared Status & Control) error status register: 31: Indicates an address error when accessed to internal memory uCode/driver must write "1" in order to clear this flag 30: Indicates that Host did not send the expected number of dwords to FH uCode/driver must write "1" in order to clear this flag 16-9:Each status bit is for one channel. Indicates that an (Error) ActDMA command was received from the scheduler while the TRB was already full with previous command uCode/driver must write "1" in order to clear this flag 7-0: Each status bit indicates a channel's TxCredit error. When an error bit is set, it indicates that the FH has received a full indication from the RTC TxFIFO and the current value of the TxCredit counter was not equal to zero. This mean that the credit mechanism was not synchronized to the TxFIFO status uCode/driver must write "1" in order to clear this flag
#define FH_TSSR_TX_STATUS_REG (FH_TSSR_LOWER_BOUND + 0x010) |
#define FH_TSSR_TX_STATUS_REG_MSK_CHNL_IDLE | ( | _chnl | ) | ((1 << (_chnl)) << 16) |
#define FH_TSSR_UPPER_BOUND (FH_MEM_LOWER_BOUND + 0xEC0) |
#define FH_TX_CHICKEN_BITS_REG (FH_MEM_LOWER_BOUND + 0xE98) |
#define FH_TX_TRB_REG | ( | _chan | ) | (FH_MEM_LOWER_BOUND + 0x958 + (_chan) * 4) |
#define IWL_TX_DMA_MASK DMA_BIT_MASK(36) |
#define TFD_QUEUE_BC_SIZE (TFD_QUEUE_SIZE_MAX + TFD_QUEUE_SIZE_BC_DUP) |