aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-4965-hw.h38
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-4965.c21
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-5000-hw.h29
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-5000.c52
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-fh.h5
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-helpers.h102
6 files changed, 62 insertions, 185 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl-4965-hw.h b/drivers/net/wireless/iwlwifi/iwl-4965-hw.h
index b66dd093084d..f3f41a6a7c8c 100644
--- a/drivers/net/wireless/iwlwifi/iwl-4965-hw.h
+++ b/drivers/net/wireless/iwlwifi/iwl-4965-hw.h
@@ -111,7 +111,6 @@
111#define PCI_CFG_CMD_REG_INT_DIS_MSK 0x04 111#define PCI_CFG_CMD_REG_INT_DIS_MSK 0x04
112#define PCI_CFG_PMC_PME_FROM_D3COLD_SUPPORT (0x80000000) 112#define PCI_CFG_PMC_PME_FROM_D3COLD_SUPPORT (0x80000000)
113 113
114#define TFD_QUEUE_SIZE_MAX (256)
115 114
116#define IWL_NUM_SCAN_RATES (2) 115#define IWL_NUM_SCAN_RATES (2)
117 116
@@ -815,8 +814,6 @@ enum {
815 * up to 7 DMA channels (FIFOs). Each Tx queue is supported by a circular array 814 * up to 7 DMA channels (FIFOs). Each Tx queue is supported by a circular array
816 * in DRAM containing 256 Transmit Frame Descriptors (TFDs). 815 * in DRAM containing 256 Transmit Frame Descriptors (TFDs).
817 */ 816 */
818#define IWL49_MAX_WIN_SIZE 64
819#define IWL49_QUEUE_SIZE 256
820#define IWL49_NUM_FIFOS 7 817#define IWL49_NUM_FIFOS 7
821#define IWL49_CMD_FIFO_NUM 4 818#define IWL49_CMD_FIFO_NUM 4
822#define IWL49_NUM_QUEUES 16 819#define IWL49_NUM_QUEUES 16
@@ -882,26 +879,7 @@ struct iwl_tfd {
882 879
883 880
884/** 881/**
885 * struct iwl4965_queue_byte_cnt_entry 882 * struct iwl4965_schedq_bc_tbl
886 *
887 * Byte Count Table Entry
888 *
889 * Bit fields:
890 * 15-12: reserved
891 * 11- 0: total to-be-transmitted byte count of frame (does not include command)
892 */
893struct iwl4965_queue_byte_cnt_entry {
894 __le16 val;
895 /* __le16 byte_cnt:12; */
896#define IWL_byte_cnt_POS 0
897#define IWL_byte_cnt_LEN 12
898#define IWL_byte_cnt_SYM val
899 /* __le16 rsvd:4; */
900} __attribute__ ((packed));
901
902
903/**
904 * struct iwl4965_sched_queue_byte_cnt_tbl
905 * 883 *
906 * Byte Count table 884 * Byte Count table
907 * 885 *
@@ -915,15 +893,12 @@ struct iwl4965_queue_byte_cnt_entry {
915 * count table for the chosen Tx queue. If the TFD index is 0-63, the driver 893 * count table for the chosen Tx queue. If the TFD index is 0-63, the driver
916 * must duplicate the byte count entry in corresponding index 256-319. 894 * must duplicate the byte count entry in corresponding index 256-319.
917 * 895 *
918 * "dont_care" padding puts each byte count table on a 1024-byte boundary; 896 * padding puts each byte count table on a 1024-byte boundary;
919 * 4965 assumes tables are separated by 1024 bytes. 897 * 4965 assumes tables are separated by 1024 bytes.
920 */ 898 */
921struct iwl4965_sched_queue_byte_cnt_tbl { 899struct iwl4965_schedq_bc_tbl {
922 struct iwl4965_queue_byte_cnt_entry tfd_offset[IWL49_QUEUE_SIZE + 900 __le16 tfd_offset[TFD_QUEUE_BC_SIZE];
923 IWL49_MAX_WIN_SIZE]; 901 u8 pad[1024 - (TFD_QUEUE_BC_SIZE) * sizeof(__le16)];
924 u8 dont_care[1024 -
925 (IWL49_QUEUE_SIZE + IWL49_MAX_WIN_SIZE) *
926 sizeof(__le16)];
927} __attribute__ ((packed)); 902} __attribute__ ((packed));
928 903
929 904
@@ -951,8 +926,7 @@ struct iwl4965_sched_queue_byte_cnt_tbl {
951 * 31- 0: Not used 926 * 31- 0: Not used
952 */ 927 */
953struct iwl4965_shared { 928struct iwl4965_shared {
954 struct iwl4965_sched_queue_byte_cnt_tbl 929 struct iwl4965_schedq_bc_tbl queues_bc_tbls[IWL49_NUM_QUEUES];
955 queues_byte_cnt_tbls[IWL49_NUM_QUEUES];
956 __le32 rb_closed; 930 __le32 rb_closed;
957 931
958 /* __le32 rb_closed_stts_rb_num:12; */ 932 /* __le32 rb_closed_stts_rb_num:12; */
diff --git a/drivers/net/wireless/iwlwifi/iwl-4965.c b/drivers/net/wireless/iwlwifi/iwl-4965.c
index aad32a3ffd19..1f22140bec1a 100644
--- a/drivers/net/wireless/iwlwifi/iwl-4965.c
+++ b/drivers/net/wireless/iwlwifi/iwl-4965.c
@@ -716,7 +716,7 @@ static int iwl4965_alive_notify(struct iwl_priv *priv)
716 /* Tel 4965 where to find Tx byte count tables */ 716 /* Tel 4965 where to find Tx byte count tables */
717 iwl_write_prph(priv, IWL49_SCD_DRAM_BASE_ADDR, 717 iwl_write_prph(priv, IWL49_SCD_DRAM_BASE_ADDR,
718 (priv->shared_phys + 718 (priv->shared_phys +
719 offsetof(struct iwl4965_shared, queues_byte_cnt_tbls)) >> 10); 719 offsetof(struct iwl4965_shared, queues_bc_tbls)) >> 10);
720 720
721 /* Disable chain mode for all queues */ 721 /* Disable chain mode for all queues */
722 iwl_write_prph(priv, IWL49_SCD_QUEUECHAIN_SEL, 0); 722 iwl_write_prph(priv, IWL49_SCD_QUEUECHAIN_SEL, 0);
@@ -1668,21 +1668,22 @@ static void iwl4965_txq_update_byte_cnt_tbl(struct iwl_priv *priv,
1668 struct iwl_tx_queue *txq, 1668 struct iwl_tx_queue *txq,
1669 u16 byte_cnt) 1669 u16 byte_cnt)
1670{ 1670{
1671 int len;
1672 int txq_id = txq->q.id;
1673 struct iwl4965_shared *shared_data = priv->shared_virt; 1671 struct iwl4965_shared *shared_data = priv->shared_virt;
1672 int txq_id = txq->q.id;
1673 int write_ptr = txq->q.write_ptr;
1674 int len = byte_cnt + IWL_TX_CRC_SIZE + IWL_TX_DELIMITER_SIZE;
1675 __le16 bc_ent;
1674 1676
1675 len = byte_cnt + IWL_TX_CRC_SIZE + IWL_TX_DELIMITER_SIZE; 1677 WARN_ON(len > 0xFFF || write_ptr >= TFD_QUEUE_SIZE_MAX);
1676 1678
1679 bc_ent = cpu_to_le16(len & 0xFFF);
1677 /* Set up byte count within first 256 entries */ 1680 /* Set up byte count within first 256 entries */
1678 IWL_SET_BITS16(shared_data->queues_byte_cnt_tbls[txq_id]. 1681 shared_data->queues_bc_tbls[txq_id].tfd_offset[write_ptr] = bc_ent;
1679 tfd_offset[txq->q.write_ptr], byte_cnt, len);
1680 1682
1681 /* If within first 64 entries, duplicate at end */ 1683 /* If within first 64 entries, duplicate at end */
1682 if (txq->q.write_ptr < IWL49_MAX_WIN_SIZE) 1684 if (write_ptr < TFD_QUEUE_SIZE_BC_DUP)
1683 IWL_SET_BITS16(shared_data->queues_byte_cnt_tbls[txq_id]. 1685 shared_data->queues_bc_tbls[txq_id].
1684 tfd_offset[IWL49_QUEUE_SIZE + txq->q.write_ptr], 1686 tfd_offset[TFD_QUEUE_SIZE_MAX + write_ptr] = bc_ent;
1685 byte_cnt, len);
1686} 1687}
1687 1688
1688/** 1689/**
diff --git a/drivers/net/wireless/iwlwifi/iwl-5000-hw.h b/drivers/net/wireless/iwlwifi/iwl-5000-hw.h
index fa0644321e4f..49ede54bfad9 100644
--- a/drivers/net/wireless/iwlwifi/iwl-5000-hw.h
+++ b/drivers/net/wireless/iwlwifi/iwl-5000-hw.h
@@ -76,30 +76,31 @@
76/* EERPROM */ 76/* EERPROM */
77#define IWL_5000_EEPROM_IMG_SIZE 2048 77#define IWL_5000_EEPROM_IMG_SIZE 2048
78 78
79
80#define IWL50_MAX_WIN_SIZE 64
81#define IWL50_QUEUE_SIZE 256
82#define IWL50_CMD_FIFO_NUM 7 79#define IWL50_CMD_FIFO_NUM 7
83#define IWL50_NUM_QUEUES 20 80#define IWL50_NUM_QUEUES 20
84#define IWL50_NUM_AMPDU_QUEUES 10 81#define IWL50_NUM_AMPDU_QUEUES 10
85#define IWL50_FIRST_AMPDU_QUEUE 10 82#define IWL50_FIRST_AMPDU_QUEUE 10
86 83
87#define IWL_sta_id_POS 12
88#define IWL_sta_id_LEN 4
89#define IWL_sta_id_SYM val
90
91/* Fixed (non-configurable) rx data from phy */ 84/* Fixed (non-configurable) rx data from phy */
92 85
93/* Base physical address of iwl5000_shared is provided to SCD_DRAM_BASE_ADDR 86/**
94 * and &iwl5000_shared.val0 is provided to FH_RSCSR_CHNL0_STTS_WPTR_REG */ 87 * struct iwl5000_schedq_bc_tbl scheduler byte count table
95struct iwl5000_sched_queue_byte_cnt_tbl { 88 * base physical address of iwl5000_shared
96 struct iwl4965_queue_byte_cnt_entry tfd_offset[IWL50_QUEUE_SIZE + 89 * is provided to SCD_DRAM_BASE_ADDR
97 IWL50_MAX_WIN_SIZE]; 90 * @tfd_offset 0-12 - tx command byte count
91 * 12-16 - station index
92 */
93struct iwl5000_schedq_bc_tbl {
94 __le16 tfd_offset[TFD_QUEUE_BC_SIZE];
98} __attribute__ ((packed)); 95} __attribute__ ((packed));
99 96
97/**
98 * struct iwl5000_shared
99 * @rb_closed
100 * address is provided to FH_RSCSR_CHNL0_STTS_WPTR_REG
101 */
100struct iwl5000_shared { 102struct iwl5000_shared {
101 struct iwl5000_sched_queue_byte_cnt_tbl 103 struct iwl5000_schedq_bc_tbl queues_bc_tbls[IWL50_NUM_QUEUES];
102 queues_byte_cnt_tbls[IWL50_NUM_QUEUES];
103 __le32 rb_closed; 104 __le32 rb_closed;
104 105
105 /* __le32 rb_closed_stts_rb_num:12; */ 106 /* __le32 rb_closed_stts_rb_num:12; */
diff --git a/drivers/net/wireless/iwlwifi/iwl-5000.c b/drivers/net/wireless/iwlwifi/iwl-5000.c
index b4b7e8b2a429..0c9281e9f2ad 100644
--- a/drivers/net/wireless/iwlwifi/iwl-5000.c
+++ b/drivers/net/wireless/iwlwifi/iwl-5000.c
@@ -723,7 +723,7 @@ static int iwl5000_alive_notify(struct iwl_priv *priv)
723 723
724 iwl_write_prph(priv, IWL50_SCD_DRAM_BASE_ADDR, 724 iwl_write_prph(priv, IWL50_SCD_DRAM_BASE_ADDR,
725 (priv->shared_phys + 725 (priv->shared_phys +
726 offsetof(struct iwl5000_shared, queues_byte_cnt_tbls)) >> 10); 726 offsetof(struct iwl5000_shared, queues_bc_tbls)) >> 10);
727 iwl_write_prph(priv, IWL50_SCD_QUEUECHAIN_SEL, 727 iwl_write_prph(priv, IWL50_SCD_QUEUECHAIN_SEL,
728 IWL50_SCD_QUEUECHAIN_SEL_ALL( 728 IWL50_SCD_QUEUECHAIN_SEL_ALL(
729 priv->hw_params.max_txq_num)); 729 priv->hw_params.max_txq_num));
@@ -891,15 +891,17 @@ static void iwl5000_txq_update_byte_cnt_tbl(struct iwl_priv *priv,
891 u16 byte_cnt) 891 u16 byte_cnt)
892{ 892{
893 struct iwl5000_shared *shared_data = priv->shared_virt; 893 struct iwl5000_shared *shared_data = priv->shared_virt;
894 int write_ptr = txq->q.write_ptr;
894 int txq_id = txq->q.id; 895 int txq_id = txq->q.id;
895 u8 sec_ctl = 0; 896 u8 sec_ctl = 0;
896 u8 sta = 0; 897 u8 sta_id = 0;
897 int len; 898 u16 len = byte_cnt + IWL_TX_CRC_SIZE + IWL_TX_DELIMITER_SIZE;
899 __le16 bc_ent;
898 900
899 len = byte_cnt + IWL_TX_CRC_SIZE + IWL_TX_DELIMITER_SIZE; 901 WARN_ON(len > 0xFFF || write_ptr >= TFD_QUEUE_SIZE_MAX);
900 902
901 if (txq_id != IWL_CMD_QUEUE_NUM) { 903 if (txq_id != IWL_CMD_QUEUE_NUM) {
902 sta = txq->cmd[txq->q.write_ptr]->cmd.tx.sta_id; 904 sta_id = txq->cmd[txq->q.write_ptr]->cmd.tx.sta_id;
903 sec_ctl = txq->cmd[txq->q.write_ptr]->cmd.tx.sec_ctl; 905 sec_ctl = txq->cmd[txq->q.write_ptr]->cmd.tx.sec_ctl;
904 906
905 switch (sec_ctl & TX_CMD_SEC_MSK) { 907 switch (sec_ctl & TX_CMD_SEC_MSK) {
@@ -915,40 +917,36 @@ static void iwl5000_txq_update_byte_cnt_tbl(struct iwl_priv *priv,
915 } 917 }
916 } 918 }
917 919
918 IWL_SET_BITS16(shared_data->queues_byte_cnt_tbls[txq_id]. 920 bc_ent = cpu_to_le16((len & 0xFFF) | (sta_id << 12));
919 tfd_offset[txq->q.write_ptr], byte_cnt, len);
920 921
921 IWL_SET_BITS16(shared_data->queues_byte_cnt_tbls[txq_id]. 922 shared_data->queues_bc_tbls[txq_id].tfd_offset[write_ptr] = bc_ent;
922 tfd_offset[txq->q.write_ptr], sta_id, sta);
923 923
924 if (txq->q.write_ptr < IWL50_MAX_WIN_SIZE) { 924 if (txq->q.write_ptr < TFD_QUEUE_SIZE_BC_DUP)
925 IWL_SET_BITS16(shared_data->queues_byte_cnt_tbls[txq_id]. 925 shared_data->queues_bc_tbls[txq_id].
926 tfd_offset[IWL50_QUEUE_SIZE + txq->q.write_ptr], 926 tfd_offset[TFD_QUEUE_SIZE_MAX + write_ptr] = bc_ent;
927 byte_cnt, len);
928 IWL_SET_BITS16(shared_data->queues_byte_cnt_tbls[txq_id].
929 tfd_offset[IWL50_QUEUE_SIZE + txq->q.write_ptr],
930 sta_id, sta);
931 }
932} 927}
933 928
934static void iwl5000_txq_inval_byte_cnt_tbl(struct iwl_priv *priv, 929static void iwl5000_txq_inval_byte_cnt_tbl(struct iwl_priv *priv,
935 struct iwl_tx_queue *txq) 930 struct iwl_tx_queue *txq)
936{ 931{
937 int txq_id = txq->q.id;
938 struct iwl5000_shared *shared_data = priv->shared_virt; 932 struct iwl5000_shared *shared_data = priv->shared_virt;
939 u8 sta = 0; 933 int txq_id = txq->q.id;
934 int read_ptr = txq->q.read_ptr;
935 u8 sta_id = 0;
936 __le16 bc_ent;
937
938 WARN_ON(read_ptr >= TFD_QUEUE_SIZE_MAX);
940 939
941 if (txq_id != IWL_CMD_QUEUE_NUM) 940 if (txq_id != IWL_CMD_QUEUE_NUM)
942 sta = txq->cmd[txq->q.read_ptr]->cmd.tx.sta_id; 941 sta_id = txq->cmd[read_ptr]->cmd.tx.sta_id;
943 942
944 shared_data->queues_byte_cnt_tbls[txq_id].tfd_offset[txq->q.read_ptr]. 943 bc_ent = cpu_to_le16(1 | (sta_id << 12));
945 val = cpu_to_le16(1 | (sta << 12)); 944 shared_data->queues_bc_tbls[txq_id].
945 tfd_offset[read_ptr] = bc_ent;
946 946
947 if (txq->q.write_ptr < IWL50_MAX_WIN_SIZE) { 947 if (txq->q.write_ptr < TFD_QUEUE_SIZE_BC_DUP)
948 shared_data->queues_byte_cnt_tbls[txq_id]. 948 shared_data->queues_bc_tbls[txq_id].
949 tfd_offset[IWL50_QUEUE_SIZE + txq->q.read_ptr]. 949 tfd_offset[TFD_QUEUE_SIZE_MAX + read_ptr] = bc_ent;
950 val = cpu_to_le16(1 | (sta << 12));
951 }
952} 950}
953 951
954static int iwl5000_tx_queue_set_q2ratid(struct iwl_priv *priv, u16 ra_tid, 952static int iwl5000_tx_queue_set_q2ratid(struct iwl_priv *priv, u16 ra_tid,
diff --git a/drivers/net/wireless/iwlwifi/iwl-fh.h b/drivers/net/wireless/iwlwifi/iwl-fh.h
index 8c48d8870218..f2688d551830 100644
--- a/drivers/net/wireless/iwlwifi/iwl-fh.h
+++ b/drivers/net/wireless/iwlwifi/iwl-fh.h
@@ -393,4 +393,9 @@
393/* TCSR: tx_config register values */ 393/* TCSR: tx_config register values */
394#define FH_RSCSR_FRAME_SIZE_MSK (0x00003FFF) /* bits 0-13 */ 394#define FH_RSCSR_FRAME_SIZE_MSK (0x00003FFF) /* bits 0-13 */
395 395
396#define TFD_QUEUE_SIZE_MAX (256)
397#define TFD_QUEUE_SIZE_BC_DUP (64)
398#define TFD_QUEUE_BC_SIZE (TFD_QUEUE_SIZE_MAX + TFD_QUEUE_SIZE_BC_DUP)
399
400
396#endif /* !__iwl_fh_h__ */ 401#endif /* !__iwl_fh_h__ */
diff --git a/drivers/net/wireless/iwlwifi/iwl-helpers.h b/drivers/net/wireless/iwlwifi/iwl-helpers.h
index 029d19c7075c..4f0fa215d32e 100644
--- a/drivers/net/wireless/iwlwifi/iwl-helpers.h
+++ b/drivers/net/wireless/iwlwifi/iwl-helpers.h
@@ -32,108 +32,6 @@
32 32
33#include <linux/ctype.h> 33#include <linux/ctype.h>
34 34
35/*
36 * The structures defined by the hardware/uCode interface
37 * have bit-wise operations. For each bit-field there is
38 * a data symbol in the structure, the start bit position
39 * and the length of the bit-field.
40 *
41 * iwl_get_bits and iwl_set_bits will return or set the
42 * appropriate bits on a 32-bit value.
43 *
44 * IWL_GET_BITS and IWL_SET_BITS use symbol expansion to
45 * expand out to the appropriate call to iwl_get_bits
46 * and iwl_set_bits without having to reference all of the
47 * numerical constants and defines provided in the hardware
48 * definition
49 */
50
51/**
52 * iwl_get_bits - Extract a hardware bit-field value
53 * @src: source hardware value (__le32)
54 * @pos: bit-position (0-based) of first bit of value
55 * @len: length of bit-field
56 *
57 * iwl_get_bits will return the bit-field in cpu endian ordering.
58 *
59 * NOTE: If used from IWL_GET_BITS then pos and len are compile-constants and
60 * will collapse to minimal code by the compiler.
61 */
62static inline u32 iwl_get_bits(__le32 src, u8 pos, u8 len)
63{
64 u32 tmp = le32_to_cpu(src);
65
66 tmp >>= pos;
67 tmp &= (1UL << len) - 1;
68 return tmp;
69}
70
71/**
72 * iwl_set_bits - Set a hardware bit-field value
73 * @dst: Address of __le32 hardware value
74 * @pos: bit-position (0-based) of first bit of value
75 * @len: length of bit-field
76 * @val: cpu endian value to encode into the bit-field
77 *
78 * iwl_set_bits will encode val into dst, masked to be len bits long at bit
79 * position pos.
80 *
81 * NOTE: If used IWL_SET_BITS pos and len will be compile-constants and
82 * will collapse to minimal code by the compiler.
83 */
84static inline void iwl_set_bits(__le32 *dst, u8 pos, u8 len, int val)
85{
86 u32 tmp = le32_to_cpu(*dst);
87
88 tmp &= ~(((1UL << len) - 1) << pos);
89 tmp |= (val & ((1UL << len) - 1)) << pos;
90 *dst = cpu_to_le32(tmp);
91}
92
93static inline void iwl_set_bits16(__le16 *dst, u8 pos, u8 len, int val)
94{
95 u16 tmp = le16_to_cpu(*dst);
96
97 tmp &= ~((1UL << (pos + len)) - (1UL << pos));
98 tmp |= (val & ((1UL << len) - 1)) << pos;
99 *dst = cpu_to_le16(tmp);
100}
101
102/*
103 * The bit-field definitions in iwl-xxxx-hw.h are in the form of:
104 *
105 * struct example {
106 * __le32 val1;
107 * #define IWL_name_POS 8
108 * #define IWL_name_LEN 4
109 * #define IWL_name_SYM val1
110 * };
111 *
112 * The IWL_SET_BITS and IWL_GET_BITS macros are provided to allow the driver
113 * to call:
114 *
115 * struct example bar;
116 * u32 val = IWL_GET_BITS(bar, name);
117 * val = val * 2;
118 * IWL_SET_BITS(bar, name, val);
119 *
120 * All cpu / host ordering, masking, and shifts are performed by the macros
121 * and iwl_{get,set}_bits.
122 *
123 */
124#define IWL_SET_BITS(s, sym, v) \
125 iwl_set_bits(&(s).IWL_ ## sym ## _SYM, IWL_ ## sym ## _POS, \
126 IWL_ ## sym ## _LEN, (v))
127
128#define IWL_SET_BITS16(s, sym, v) \
129 iwl_set_bits16(&(s).IWL_ ## sym ## _SYM, IWL_ ## sym ## _POS, \
130 IWL_ ## sym ## _LEN, (v))
131
132#define IWL_GET_BITS(s, sym) \
133 iwl_get_bits((s).IWL_ ## sym ## _SYM, IWL_ ## sym ## _POS, \
134 IWL_ ## sym ## _LEN)
135
136
137#define KELVIN_TO_CELSIUS(x) ((x)-273) 35#define KELVIN_TO_CELSIUS(x) ((x)-273)
138#define CELSIUS_TO_KELVIN(x) ((x)+273) 36#define CELSIUS_TO_KELVIN(x) ((x)+273)
139#define IWL_MASK(lo, hi) ((1 << (hi)) | ((1 << (hi)) - (1 << (lo)))) 37#define IWL_MASK(lo, hi) ((1 << (hi)) | ((1 << (hi)) - (1 << (lo))))