blob: c9fe3c99cb003c087729b12380b16df014827f8b [file] [log] [blame]
James Ketrenos02cda6a2005-09-21 11:56:38 -05001/******************************************************************************
2
3 Copyright(c) 2005 Intel Corporation. All rights reserved.
4
5 This program is free software; you can redistribute it and/or modify it
6 under the terms of version 2 of the GNU General Public License as
7 published by the Free Software Foundation.
8
9 This program is distributed in the hope that it will be useful, but WITHOUT
10 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
12 more details.
13
14 You should have received a copy of the GNU General Public License along with
15 this program; if not, write to the Free Software Foundation, Inc., 59
16 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17
18 The full GNU General Public License is included in this distribution in the
19 file called LICENSE.
20
21 Contact Information:
Reinette Chatrec1eb2c82009-08-21 13:34:26 -070022 Intel Linux Wireless <ilw@linux.intel.com>
James Ketrenos02cda6a2005-09-21 11:56:38 -050023 Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
24
25******************************************************************************/
26#include <linux/compiler.h>
James Ketrenos02cda6a2005-09-21 11:56:38 -050027#include <linux/errno.h>
28#include <linux/if_arp.h>
29#include <linux/in6.h>
30#include <linux/in.h>
31#include <linux/ip.h>
32#include <linux/kernel.h>
33#include <linux/module.h>
34#include <linux/netdevice.h>
35#include <linux/proc_fs.h>
36#include <linux/skbuff.h>
James Ketrenos02cda6a2005-09-21 11:56:38 -050037#include <linux/tcp.h>
38#include <linux/types.h>
James Ketrenos02cda6a2005-09-21 11:56:38 -050039#include <linux/wireless.h>
40#include <linux/etherdevice.h>
41#include <asm/uaccess.h>
42
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040043#include "libipw.h"
James Ketrenos02cda6a2005-09-21 11:56:38 -050044
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040045int libipw_is_valid_channel(struct libipw_device *ieee, u8 channel)
James Ketrenos02cda6a2005-09-21 11:56:38 -050046{
47 int i;
48
49 /* Driver needs to initialize the geography map before using
50 * these helper functions */
Pete Zaitcev07981aa2006-02-26 16:29:51 -080051 if (ieee->geo.bg_channels == 0 && ieee->geo.a_channels == 0)
52 return 0;
James Ketrenos02cda6a2005-09-21 11:56:38 -050053
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040054 if (ieee->freq_band & LIBIPW_24GHZ_BAND)
James Ketrenos02cda6a2005-09-21 11:56:38 -050055 for (i = 0; i < ieee->geo.bg_channels; i++)
56 /* NOTE: If G mode is currently supported but
57 * this is a B only channel, we don't see it
58 * as valid. */
59 if ((ieee->geo.bg[i].channel == channel) &&
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040060 !(ieee->geo.bg[i].flags & LIBIPW_CH_INVALID) &&
James Ketrenos02cda6a2005-09-21 11:56:38 -050061 (!(ieee->mode & IEEE_G) ||
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040062 !(ieee->geo.bg[i].flags & LIBIPW_CH_B_ONLY)))
63 return LIBIPW_24GHZ_BAND;
James Ketrenos02cda6a2005-09-21 11:56:38 -050064
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040065 if (ieee->freq_band & LIBIPW_52GHZ_BAND)
James Ketrenos02cda6a2005-09-21 11:56:38 -050066 for (i = 0; i < ieee->geo.a_channels; i++)
Zhu Yid128f6c2006-01-19 16:21:45 +080067 if ((ieee->geo.a[i].channel == channel) &&
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040068 !(ieee->geo.a[i].flags & LIBIPW_CH_INVALID))
69 return LIBIPW_52GHZ_BAND;
James Ketrenos02cda6a2005-09-21 11:56:38 -050070
71 return 0;
72}
73
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040074int libipw_channel_to_index(struct libipw_device *ieee, u8 channel)
James Ketrenos02cda6a2005-09-21 11:56:38 -050075{
76 int i;
77
78 /* Driver needs to initialize the geography map before using
79 * these helper functions */
Pete Zaitcev07981aa2006-02-26 16:29:51 -080080 if (ieee->geo.bg_channels == 0 && ieee->geo.a_channels == 0)
81 return -1;
James Ketrenos02cda6a2005-09-21 11:56:38 -050082
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040083 if (ieee->freq_band & LIBIPW_24GHZ_BAND)
James Ketrenos02cda6a2005-09-21 11:56:38 -050084 for (i = 0; i < ieee->geo.bg_channels; i++)
85 if (ieee->geo.bg[i].channel == channel)
86 return i;
87
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040088 if (ieee->freq_band & LIBIPW_52GHZ_BAND)
James Ketrenos02cda6a2005-09-21 11:56:38 -050089 for (i = 0; i < ieee->geo.a_channels; i++)
90 if (ieee->geo.a[i].channel == channel)
91 return i;
92
93 return -1;
94}
95
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040096u32 libipw_channel_to_freq(struct libipw_device * ieee, u8 channel)
Larry Fingerf5cdf302007-04-21 17:56:29 -050097{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040098 const struct libipw_channel * ch;
Larry Fingerf5cdf302007-04-21 17:56:29 -050099
100 /* Driver needs to initialize the geography map before using
101 * these helper functions */
102 if (ieee->geo.bg_channels == 0 && ieee->geo.a_channels == 0)
103 return 0;
104
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400105 ch = libipw_get_channel(ieee, channel);
Larry Fingerf5cdf302007-04-21 17:56:29 -0500106 if (!ch->channel)
107 return 0;
108 return ch->freq;
109}
110
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400111u8 libipw_freq_to_channel(struct libipw_device * ieee, u32 freq)
James Ketrenos02cda6a2005-09-21 11:56:38 -0500112{
113 int i;
114
115 /* Driver needs to initialize the geography map before using
116 * these helper functions */
Pete Zaitcev07981aa2006-02-26 16:29:51 -0800117 if (ieee->geo.bg_channels == 0 && ieee->geo.a_channels == 0)
118 return 0;
James Ketrenos02cda6a2005-09-21 11:56:38 -0500119
120 freq /= 100000;
121
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400122 if (ieee->freq_band & LIBIPW_24GHZ_BAND)
James Ketrenos02cda6a2005-09-21 11:56:38 -0500123 for (i = 0; i < ieee->geo.bg_channels; i++)
124 if (ieee->geo.bg[i].freq == freq)
125 return ieee->geo.bg[i].channel;
126
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400127 if (ieee->freq_band & LIBIPW_52GHZ_BAND)
James Ketrenos02cda6a2005-09-21 11:56:38 -0500128 for (i = 0; i < ieee->geo.a_channels; i++)
129 if (ieee->geo.a[i].freq == freq)
130 return ieee->geo.a[i].channel;
131
132 return 0;
133}
134
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400135int libipw_set_geo(struct libipw_device *ieee,
136 const struct libipw_geo *geo)
James Ketrenos02cda6a2005-09-21 11:56:38 -0500137{
138 memcpy(ieee->geo.name, geo->name, 3);
139 ieee->geo.name[3] = '\0';
140 ieee->geo.bg_channels = geo->bg_channels;
141 ieee->geo.a_channels = geo->a_channels;
142 memcpy(ieee->geo.bg, geo->bg, geo->bg_channels *
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400143 sizeof(struct libipw_channel));
James Ketrenos02cda6a2005-09-21 11:56:38 -0500144 memcpy(ieee->geo.a, geo->a, ieee->geo.a_channels *
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400145 sizeof(struct libipw_channel));
James Ketrenos02cda6a2005-09-21 11:56:38 -0500146 return 0;
147}
148
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400149const struct libipw_geo *libipw_get_geo(struct libipw_device *ieee)
James Ketrenos02cda6a2005-09-21 11:56:38 -0500150{
151 return &ieee->geo;
152}
153
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400154u8 libipw_get_channel_flags(struct libipw_device * ieee, u8 channel)
Zhu Yid128f6c2006-01-19 16:21:45 +0800155{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400156 int index = libipw_channel_to_index(ieee, channel);
Zhu Yid128f6c2006-01-19 16:21:45 +0800157
158 if (index == -1)
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400159 return LIBIPW_CH_INVALID;
Zhu Yid128f6c2006-01-19 16:21:45 +0800160
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400161 if (channel <= LIBIPW_24GHZ_CHANNELS)
Zhu Yid128f6c2006-01-19 16:21:45 +0800162 return ieee->geo.bg[index].flags;
163
164 return ieee->geo.a[index].flags;
165}
166
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400167static const struct libipw_channel bad_channel = {
Zhu Yid128f6c2006-01-19 16:21:45 +0800168 .channel = 0,
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400169 .flags = LIBIPW_CH_INVALID,
Zhu Yid128f6c2006-01-19 16:21:45 +0800170 .max_power = 0,
171};
172
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400173const struct libipw_channel *libipw_get_channel(struct libipw_device
Zhu Yid128f6c2006-01-19 16:21:45 +0800174 *ieee, u8 channel)
175{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400176 int index = libipw_channel_to_index(ieee, channel);
Zhu Yid128f6c2006-01-19 16:21:45 +0800177
178 if (index == -1)
179 return &bad_channel;
180
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400181 if (channel <= LIBIPW_24GHZ_CHANNELS)
Zhu Yid128f6c2006-01-19 16:21:45 +0800182 return &ieee->geo.bg[index];
183
184 return &ieee->geo.a[index];
185}
186
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400187EXPORT_SYMBOL(libipw_get_channel);
188EXPORT_SYMBOL(libipw_get_channel_flags);
189EXPORT_SYMBOL(libipw_is_valid_channel);
190EXPORT_SYMBOL(libipw_freq_to_channel);
191EXPORT_SYMBOL(libipw_channel_to_freq);
192EXPORT_SYMBOL(libipw_channel_to_index);
193EXPORT_SYMBOL(libipw_set_geo);
194EXPORT_SYMBOL(libipw_get_geo);