aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/p54/eeprom.c
diff options
context:
space:
mode:
authorChristian Lamparter <chunkeey@web.de>2009-07-10 19:22:26 -0400
committerJohn W. Linville <linville@tuxdriver.com>2009-07-24 15:05:12 -0400
commit1a9b6679adfb8ef1f1f3dbb7ebd2ee72e2ea4b56 (patch)
tree2cb7507ba929d9214ef84d76dca2fe93285f7a35 /drivers/net/wireless/p54/eeprom.c
parent596a07c18b35c9df2fb212856241ae0dfe3162b9 (diff)
p54: generate channel list dynamically
This patch enhances the eeprom parser to generate customized channel list for every device. Signed-off-by: Christian Lamparter <chunkeey@web.de> Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net/wireless/p54/eeprom.c')
-rw-r--r--drivers/net/wireless/p54/eeprom.c323
1 files changed, 256 insertions, 67 deletions
diff --git a/drivers/net/wireless/p54/eeprom.c b/drivers/net/wireless/p54/eeprom.c
index a2a044ef1012..549ef2d19cd7 100644
--- a/drivers/net/wireless/p54/eeprom.c
+++ b/drivers/net/wireless/p54/eeprom.c
@@ -19,6 +19,7 @@
19#include <linux/init.h> 19#include <linux/init.h>
20#include <linux/firmware.h> 20#include <linux/firmware.h>
21#include <linux/etherdevice.h> 21#include <linux/etherdevice.h>
22#include <linux/sort.h>
22 23
23#include <net/mac80211.h> 24#include <net/mac80211.h>
24 25
@@ -41,30 +42,6 @@ static struct ieee80211_rate p54_bgrates[] = {
41 { .bitrate = 540, .hw_value = 11, }, 42 { .bitrate = 540, .hw_value = 11, },
42}; 43};
43 44
44static struct ieee80211_channel p54_bgchannels[] = {
45 { .center_freq = 2412, .hw_value = 1, },
46 { .center_freq = 2417, .hw_value = 2, },
47 { .center_freq = 2422, .hw_value = 3, },
48 { .center_freq = 2427, .hw_value = 4, },
49 { .center_freq = 2432, .hw_value = 5, },
50 { .center_freq = 2437, .hw_value = 6, },
51 { .center_freq = 2442, .hw_value = 7, },
52 { .center_freq = 2447, .hw_value = 8, },
53 { .center_freq = 2452, .hw_value = 9, },
54 { .center_freq = 2457, .hw_value = 10, },
55 { .center_freq = 2462, .hw_value = 11, },
56 { .center_freq = 2467, .hw_value = 12, },
57 { .center_freq = 2472, .hw_value = 13, },
58 { .center_freq = 2484, .hw_value = 14, },
59};
60
61static struct ieee80211_supported_band band_2GHz = {
62 .channels = p54_bgchannels,
63 .n_channels = ARRAY_SIZE(p54_bgchannels),
64 .bitrates = p54_bgrates,
65 .n_bitrates = ARRAY_SIZE(p54_bgrates),
66};
67
68static struct ieee80211_rate p54_arates[] = { 45static struct ieee80211_rate p54_arates[] = {
69 { .bitrate = 60, .hw_value = 4, }, 46 { .bitrate = 60, .hw_value = 4, },
70 { .bitrate = 90, .hw_value = 5, }, 47 { .bitrate = 90, .hw_value = 5, },
@@ -76,51 +53,257 @@ static struct ieee80211_rate p54_arates[] = {
76 { .bitrate = 540, .hw_value = 11, }, 53 { .bitrate = 540, .hw_value = 11, },
77}; 54};
78 55
79static struct ieee80211_channel p54_achannels[] = { 56#define CHAN_HAS_CAL BIT(0)
80 { .center_freq = 4920 }, 57#define CHAN_HAS_LIMIT BIT(1)
81 { .center_freq = 4940 }, 58#define CHAN_HAS_CURVE BIT(2)
82 { .center_freq = 4960 }, 59#define CHAN_HAS_ALL (CHAN_HAS_CAL | CHAN_HAS_LIMIT | CHAN_HAS_CURVE)
83 { .center_freq = 4980 }, 60
84 { .center_freq = 5040 }, 61struct p54_channel_entry {
85 { .center_freq = 5060 }, 62 u16 freq;
86 { .center_freq = 5080 }, 63 u16 data;
87 { .center_freq = 5170 }, 64 int index;
88 { .center_freq = 5180 }, 65 enum ieee80211_band band;
89 { .center_freq = 5190 },
90 { .center_freq = 5200 },
91 { .center_freq = 5210 },
92 { .center_freq = 5220 },
93 { .center_freq = 5230 },
94 { .center_freq = 5240 },
95 { .center_freq = 5260 },
96 { .center_freq = 5280 },
97 { .center_freq = 5300 },
98 { .center_freq = 5320 },
99 { .center_freq = 5500 },
100 { .center_freq = 5520 },
101 { .center_freq = 5540 },
102 { .center_freq = 5560 },
103 { .center_freq = 5580 },
104 { .center_freq = 5600 },
105 { .center_freq = 5620 },
106 { .center_freq = 5640 },
107 { .center_freq = 5660 },
108 { .center_freq = 5680 },
109 { .center_freq = 5700 },
110 { .center_freq = 5745 },
111 { .center_freq = 5765 },
112 { .center_freq = 5785 },
113 { .center_freq = 5805 },
114 { .center_freq = 5825 },
115}; 66};
116 67
117static struct ieee80211_supported_band band_5GHz = { 68struct p54_channel_list {
118 .channels = p54_achannels, 69 struct p54_channel_entry *channels;
119 .n_channels = ARRAY_SIZE(p54_achannels), 70 size_t entries;
120 .bitrates = p54_arates, 71 size_t max_entries;
121 .n_bitrates = ARRAY_SIZE(p54_arates), 72 size_t band_channel_num[IEEE80211_NUM_BANDS];
122}; 73};
123 74
75static int p54_get_band_from_freq(u16 freq)
76{
77 /* FIXME: sync these values with the 802.11 spec */
78
79 if ((freq >= 2412) && (freq <= 2484))
80 return IEEE80211_BAND_2GHZ;
81
82 if ((freq >= 4920) && (freq <= 5825))
83 return IEEE80211_BAND_5GHZ;
84
85 return -1;
86}
87
88static int p54_compare_channels(const void *_a,
89 const void *_b)
90{
91 const struct p54_channel_entry *a = _a;
92 const struct p54_channel_entry *b = _b;
93
94 return a->index - b->index;
95}
96
97static int p54_fill_band_bitrates(struct ieee80211_hw *dev,
98 struct ieee80211_supported_band *band_entry,
99 enum ieee80211_band band)
100{
101 /* TODO: generate rate array dynamically */
102
103 switch (band) {
104 case IEEE80211_BAND_2GHZ:
105 band_entry->bitrates = p54_bgrates;
106 band_entry->n_bitrates = ARRAY_SIZE(p54_bgrates);
107 break;
108 case IEEE80211_BAND_5GHZ:
109 band_entry->bitrates = p54_arates;
110 band_entry->n_bitrates = ARRAY_SIZE(p54_arates);
111 break;
112 default:
113 return -EINVAL;
114 }
115
116 return 0;
117}
118
119static int p54_generate_band(struct ieee80211_hw *dev,
120 struct p54_channel_list *list,
121 enum ieee80211_band band)
122{
123 struct p54_common *priv = dev->priv;
124 struct ieee80211_supported_band *tmp, *old;
125 unsigned int i, j;
126 int ret = -ENOMEM;
127
128 if ((!list->entries) || (!list->band_channel_num[band]))
129 return 0;
130
131 tmp = kzalloc(sizeof(*tmp), GFP_KERNEL);
132 if (!tmp)
133 goto err_out;
134
135 tmp->channels = kzalloc(sizeof(struct ieee80211_channel) *
136 list->band_channel_num[band], GFP_KERNEL);
137 if (!tmp->channels)
138 goto err_out;
139
140 ret = p54_fill_band_bitrates(dev, tmp, band);
141 if (ret)
142 goto err_out;
143
144 for (i = 0, j = 0; (j < list->band_channel_num[band]) &&
145 (i < list->entries); i++) {
146
147 if (list->channels[i].band != band)
148 continue;
149
150 if (list->channels[i].data != CHAN_HAS_ALL) {
151 printk(KERN_ERR "%s:%s%s%s is/are missing for "
152 "channel:%d [%d MHz].\n",
153 wiphy_name(dev->wiphy),
154 (list->channels[i].data & CHAN_HAS_CAL ? "" :
155 " [iqauto calibration data]"),
156 (list->channels[i].data & CHAN_HAS_LIMIT ? "" :
157 " [output power limits]"),
158 (list->channels[i].data & CHAN_HAS_CURVE ? "" :
159 " [curve data]"),
160 list->channels[i].index, list->channels[i].freq);
161 }
162
163 tmp->channels[j].band = list->channels[i].band;
164 tmp->channels[j].center_freq = list->channels[i].freq;
165 j++;
166 }
167
168 tmp->n_channels = list->band_channel_num[band];
169 old = priv->band_table[band];
170 priv->band_table[band] = tmp;
171 if (old) {
172 kfree(old->channels);
173 kfree(old);
174 }
175
176 return 0;
177
178err_out:
179 if (tmp) {
180 kfree(tmp->channels);
181 kfree(tmp);
182 }
183
184 return ret;
185}
186
187static void p54_update_channel_param(struct p54_channel_list *list,
188 u16 freq, u16 data)
189{
190 int band, i;
191
192 /*
193 * usually all lists in the eeprom are mostly sorted.
194 * so it's very likely that the entry we are looking for
195 * is right at the end of the list
196 */
197 for (i = list->entries; i >= 0; i--) {
198 if (freq == list->channels[i].freq) {
199 list->channels[i].data |= data;
200 break;
201 }
202 }
203
204 if ((i < 0) && (list->entries < list->max_entries)) {
205 /* entry does not exist yet. Initialize a new one. */
206 band = p54_get_band_from_freq(freq);
207
208 /*
209 * filter out frequencies which don't belong into
210 * any supported band.
211 */
212 if (band < 0)
213 return ;
214
215 i = list->entries++;
216 list->band_channel_num[band]++;
217
218 list->channels[i].freq = freq;
219 list->channels[i].data = data;
220 list->channels[i].band = band;
221 list->channels[i].index = ieee80211_frequency_to_channel(freq);
222 /* TODO: parse output_limit and fill max_power */
223 }
224}
225
226static int p54_generate_channel_lists(struct ieee80211_hw *dev)
227{
228 struct p54_common *priv = dev->priv;
229 struct p54_channel_list *list;
230 unsigned int i, j, max_channel_num;
231 int ret = -ENOMEM;
232 u16 freq;
233
234 if ((priv->iq_autocal_len != priv->curve_data->entries) ||
235 (priv->iq_autocal_len != priv->output_limit->entries))
236 printk(KERN_ERR "%s: EEPROM is damaged... you may not be able"
237 "to use all channels with this device.\n",
238 wiphy_name(dev->wiphy));
239
240 max_channel_num = max_t(unsigned int, priv->output_limit->entries,
241 priv->iq_autocal_len);
242 max_channel_num = max_t(unsigned int, max_channel_num,
243 priv->curve_data->entries);
244
245 list = kzalloc(sizeof(*list), GFP_KERNEL);
246 if (!list)
247 goto free;
248
249 list->max_entries = max_channel_num;
250 list->channels = kzalloc(sizeof(struct p54_channel_entry) *
251 max_channel_num, GFP_KERNEL);
252 if (!list->channels)
253 goto free;
254
255 for (i = 0; i < max_channel_num; i++) {
256 if (i < priv->iq_autocal_len) {
257 freq = le16_to_cpu(priv->iq_autocal[i].freq);
258 p54_update_channel_param(list, freq, CHAN_HAS_CAL);
259 }
260
261 if (i < priv->output_limit->entries) {
262 freq = le16_to_cpup((__le16 *) (i *
263 priv->output_limit->entry_size +
264 priv->output_limit->offset +
265 priv->output_limit->data));
266
267 p54_update_channel_param(list, freq, CHAN_HAS_LIMIT);
268 }
269
270 if (i < priv->curve_data->entries) {
271 freq = le16_to_cpup((__le16 *) (i *
272 priv->curve_data->entry_size +
273 priv->curve_data->offset +
274 priv->curve_data->data));
275
276 p54_update_channel_param(list, freq, CHAN_HAS_CURVE);
277 }
278 }
279
280 /* sort the list by the channel index */
281 sort(list->channels, list->entries, sizeof(struct p54_channel_entry),
282 p54_compare_channels, NULL);
283
284 for (i = 0, j = 0; i < IEEE80211_NUM_BANDS; i++) {
285 if (list->band_channel_num[i]) {
286 ret = p54_generate_band(dev, list, i);
287 if (ret)
288 goto free;
289
290 j++;
291 }
292 }
293 if (j == 0) {
294 /* no useable band available. */
295 ret = -EINVAL;
296 }
297
298free:
299 if (list) {
300 kfree(list->channels);
301 kfree(list);
302 }
303
304 return ret;
305}
306
124static int p54_convert_rev0(struct ieee80211_hw *dev, 307static int p54_convert_rev0(struct ieee80211_hw *dev,
125 struct pda_pa_curve_data *curve_data) 308 struct pda_pa_curve_data *curve_data)
126{ 309{
@@ -487,13 +670,19 @@ int p54_parse_eeprom(struct ieee80211_hw *dev, void *eeprom, int len)
487 goto err; 670 goto err;
488 } 671 }
489 672
673 err = p54_generate_channel_lists(dev);
674 if (err)
675 goto err;
676
490 priv->rxhw = synth & PDR_SYNTH_FRONTEND_MASK; 677 priv->rxhw = synth & PDR_SYNTH_FRONTEND_MASK;
491 if (priv->rxhw == PDR_SYNTH_FRONTEND_XBOW) 678 if (priv->rxhw == PDR_SYNTH_FRONTEND_XBOW)
492 p54_init_xbow_synth(priv); 679 p54_init_xbow_synth(priv);
493 if (!(synth & PDR_SYNTH_24_GHZ_DISABLED)) 680 if (!(synth & PDR_SYNTH_24_GHZ_DISABLED))
494 dev->wiphy->bands[IEEE80211_BAND_2GHZ] = &band_2GHz; 681 dev->wiphy->bands[IEEE80211_BAND_2GHZ] =
682 priv->band_table[IEEE80211_BAND_2GHZ];
495 if (!(synth & PDR_SYNTH_5_GHZ_DISABLED)) 683 if (!(synth & PDR_SYNTH_5_GHZ_DISABLED))
496 dev->wiphy->bands[IEEE80211_BAND_5GHZ] = &band_5GHz; 684 dev->wiphy->bands[IEEE80211_BAND_5GHZ] =
685 priv->band_table[IEEE80211_BAND_5GHZ];
497 if ((synth & PDR_SYNTH_RX_DIV_MASK) == PDR_SYNTH_RX_DIV_SUPPORTED) 686 if ((synth & PDR_SYNTH_RX_DIV_MASK) == PDR_SYNTH_RX_DIV_SUPPORTED)
498 priv->rx_diversity_mask = 3; 687 priv->rx_diversity_mask = 3;
499 if ((synth & PDR_SYNTH_TX_DIV_MASK) == PDR_SYNTH_TX_DIV_SUPPORTED) 688 if ((synth & PDR_SYNTH_TX_DIV_MASK) == PDR_SYNTH_TX_DIV_SUPPORTED)