summaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/zd1211rw/zd_ieee80211.c
blob: 77cfe5e78230017e2f0ee86d093e353269019139 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
/* zd_ieee80211.c
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 */

/*
 * In the long term, we'll probably find a better way of handling regulatory
 * requirements outside of the driver.
 */

#include <linux/kernel.h>
#include <net/mac80211.h>

#include "zd_ieee80211.h"
#include "zd_mac.h"

struct channel_range {
	u8 regdomain;
	u8 start;
	u8 end; /* exclusive (channel must be less than end) */
};

static const struct channel_range channel_ranges[] = {
	{ ZD_REGDOMAIN_FCC,		1, 12 },
	{ ZD_REGDOMAIN_IC,		1, 12 },
	{ ZD_REGDOMAIN_ETSI,		1, 14 },
	{ ZD_REGDOMAIN_JAPAN,		1, 14 },
	{ ZD_REGDOMAIN_SPAIN,		1, 14 },
	{ ZD_REGDOMAIN_FRANCE,		1, 14 },

	/* Japan originally only had channel 14 available (see CHNL_ID 0x40 in
	 * 802.11). However, in 2001 the range was extended to include channels
	 * 1-13. The ZyDAS devices still use the old region code but are
	 * designed to allow the extra channel access in Japan. */
	{ ZD_REGDOMAIN_JAPAN_ADD,	1, 15 },
};

static const struct channel_range *zd_channel_range(u8 regdomain)
{
	int i;
	for (i = 0; i < ARRAY_SIZE(channel_ranges); i++) {
		const struct channel_range *range = &channel_ranges[i];
		if (range->regdomain == regdomain)
			return range;
	}
	return NULL;
}

#define CHAN_TO_IDX(chan) ((chan) - 1)

static void unmask_bg_channels(struct ieee80211_hw *hw,
	const struct channel_range *range,
	struct ieee80211_hw_mode *mode)
{
	u8 channel;

	for (channel = range->start; channel < range->end; channel++) {
		struct ieee80211_channel *chan =
			&mode->channels[CHAN_TO_IDX(channel)];
		chan->flag |= IEEE80211_CHAN_W_SCAN |
			IEEE80211_CHAN_W_ACTIVE_SCAN |
			IEEE80211_CHAN_W_IBSS;
	}
}

void zd_geo_init(struct ieee80211_hw *hw, u8 regdomain)
{
	struct zd_mac *mac = zd_hw_mac(hw);
	const struct channel_range *range;

	dev_dbg(zd_mac_dev(mac), "regdomain %#02x\n", regdomain);

	range = zd_channel_range(regdomain);
	if (!range) {
		/* The vendor driver overrides the regulatory domain and
		 * allowed channel registers and unconditionally restricts
		 * available channels to 1-11 everywhere. Match their
		 * questionable behaviour only for regdomains which we don't
		 * recognise. */
		dev_warn(zd_mac_dev(mac), "Unrecognised regulatory domain: "
			"%#02x. Defaulting to FCC.\n", regdomain);
		range = zd_channel_range(ZD_REGDOMAIN_FCC);
	}

	unmask_bg_channels(hw, range, &mac->modes[0]);
	unmask_bg_channels(hw, range, &mac->modes[1]);
}