blob: 60f28740f6afed42ab9e01c1077ae2130646ac69 [file] [log] [blame]
Jeff Garzikb4538722005-05-12 22:48:20 -04001/*******************************************************************************
2
James Ketrenosebeaddc2005-09-21 11:58:43 -05003 Copyright(c) 2004-2005 Intel Corporation. All rights reserved.
Jeff Garzikb4538722005-05-12 22:48:20 -04004
5 Portions of this file are based on the WEP enablement code provided by the
6 Host AP project hostap-drivers v0.1.3
7 Copyright (c) 2001-2002, SSH Communications Security Corp and Jouni Malinen
Jouni Malinen85d32e72007-03-24 17:15:30 -07008 <j@w1.fi>
9 Copyright (c) 2002-2003, Jouni Malinen <j@w1.fi>
Jeff Garzikb4538722005-05-12 22:48:20 -040010
11 This program is free software; you can redistribute it and/or modify it
12 under the terms of version 2 of the GNU General Public License as
13 published by the Free Software Foundation.
14
15 This program is distributed in the hope that it will be useful, but WITHOUT
16 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
17 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
18 more details.
19
20 You should have received a copy of the GNU General Public License along with
21 this program; if not, write to the Free Software Foundation, Inc., 59
22 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23
24 The full GNU General Public License is included in this distribution in the
25 file called LICENSE.
26
27 Contact Information:
Reinette Chatrec1eb2c82009-08-21 13:34:26 -070028 Intel Linux Wireless <ilw@linux.intel.com>
Jeff Garzikb4538722005-05-12 22:48:20 -040029 Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
30
31*******************************************************************************/
32
33#include <linux/compiler.h>
Jeff Garzikb4538722005-05-12 22:48:20 -040034#include <linux/errno.h>
35#include <linux/if_arp.h>
36#include <linux/in6.h>
37#include <linux/in.h>
38#include <linux/ip.h>
39#include <linux/kernel.h>
40#include <linux/module.h>
41#include <linux/netdevice.h>
Jeff Garzikb4538722005-05-12 22:48:20 -040042#include <linux/proc_fs.h>
43#include <linux/skbuff.h>
44#include <linux/slab.h>
45#include <linux/tcp.h>
46#include <linux/types.h>
Jeff Garzikb4538722005-05-12 22:48:20 -040047#include <linux/wireless.h>
48#include <linux/etherdevice.h>
49#include <asm/uaccess.h>
Eric W. Biederman457c4cb2007-09-12 12:01:34 +020050#include <net/net_namespace.h>
Jeff Garzikb4538722005-05-12 22:48:20 -040051#include <net/arp.h>
52
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040053#include "libipw.h"
Jeff Garzikb4538722005-05-12 22:48:20 -040054
James Ketrenos31696162005-09-21 11:58:46 -050055#define DRV_DESCRIPTION "802.11 data/management/control stack"
Pavel Roskin27ae60f2010-03-12 00:01:22 -050056#define DRV_NAME "libipw"
Linus Torvalds151f52f2010-11-05 18:57:04 -070057#define DRV_PROCNAME "ieee80211"
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040058#define DRV_VERSION LIBIPW_VERSION
James Ketrenos31696162005-09-21 11:58:46 -050059#define DRV_COPYRIGHT "Copyright (C) 2004-2005 Intel Corporation <jketreno@linux.intel.com>"
Jeff Garzikb4538722005-05-12 22:48:20 -040060
James Ketrenos31696162005-09-21 11:58:46 -050061MODULE_VERSION(DRV_VERSION);
62MODULE_DESCRIPTION(DRV_DESCRIPTION);
63MODULE_AUTHOR(DRV_COPYRIGHT);
64MODULE_LICENSE("GPL");
Jeff Garzikb4538722005-05-12 22:48:20 -040065
John W. Linvillecc40cc52010-07-20 14:14:03 -040066static struct cfg80211_ops libipw_config_ops = { };
67static void *libipw_wiphy_privid = &libipw_wiphy_privid;
John W. Linvillea3caa992009-08-25 14:12:25 -040068
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040069static int libipw_networks_allocate(struct libipw_device *ieee)
Jeff Garzikb4538722005-05-12 22:48:20 -040070{
Zhu Yi8e593402010-03-08 13:18:03 +080071 int i, j;
Jeff Garzikb4538722005-05-12 22:48:20 -040072
Zhu Yi8e593402010-03-08 13:18:03 +080073 for (i = 0; i < MAX_NETWORK_COUNT; i++) {
74 ieee->networks[i] = kzalloc(sizeof(struct libipw_network),
75 GFP_KERNEL);
76 if (!ieee->networks[i]) {
77 LIBIPW_ERROR("Out of memory allocating beacons\n");
78 for (j = 0; j < i; j++)
79 kfree(ieee->networks[j]);
80 return -ENOMEM;
81 }
Jeff Garzikb4538722005-05-12 22:48:20 -040082 }
83
Jeff Garzikb4538722005-05-12 22:48:20 -040084 return 0;
85}
86
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040087static inline void libipw_networks_free(struct libipw_device *ieee)
Jeff Garzikb4538722005-05-12 22:48:20 -040088{
Zhu Yi15f38592006-01-19 16:21:35 +080089 int i;
90
Dan Carpenter2aa01652014-10-09 12:57:42 +030091 for (i = 0; i < MAX_NETWORK_COUNT; i++)
Zhu Yi8e593402010-03-08 13:18:03 +080092 kfree(ieee->networks[i]);
Jeff Garzikb4538722005-05-12 22:48:20 -040093}
94
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040095void libipw_networks_age(struct libipw_device *ieee,
Dan Williamsc3d72b92009-02-11 13:26:06 -050096 unsigned long age_secs)
97{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -040098 struct libipw_network *network = NULL;
Dan Williamsc3d72b92009-02-11 13:26:06 -050099 unsigned long flags;
100 unsigned long age_jiffies = msecs_to_jiffies(age_secs * MSEC_PER_SEC);
101
102 spin_lock_irqsave(&ieee->lock, flags);
103 list_for_each_entry(network, &ieee->network_list, list) {
104 network->last_scanned -= age_jiffies;
105 }
106 spin_unlock_irqrestore(&ieee->lock, flags);
107}
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400108EXPORT_SYMBOL(libipw_networks_age);
Dan Williamsc3d72b92009-02-11 13:26:06 -0500109
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400110static void libipw_networks_initialize(struct libipw_device *ieee)
Jeff Garzikb4538722005-05-12 22:48:20 -0400111{
112 int i;
113
114 INIT_LIST_HEAD(&ieee->network_free_list);
115 INIT_LIST_HEAD(&ieee->network_list);
116 for (i = 0; i < MAX_NETWORK_COUNT; i++)
Zhu Yi8e593402010-03-08 13:18:03 +0800117 list_add_tail(&ieee->networks[i]->list,
Jeff Garzik0edd5b42005-09-07 00:48:31 -0400118 &ieee->network_free_list);
Jeff Garzikb4538722005-05-12 22:48:20 -0400119}
120
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400121int libipw_change_mtu(struct net_device *dev, int new_mtu)
matthieu castet42a4cf92006-09-28 19:57:25 +0200122{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400123 if ((new_mtu < 68) || (new_mtu > LIBIPW_DATA_LEN))
matthieu castet42a4cf92006-09-28 19:57:25 +0200124 return -EINVAL;
125 dev->mtu = new_mtu;
126 return 0;
127}
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400128EXPORT_SYMBOL(libipw_change_mtu);
matthieu castet42a4cf92006-09-28 19:57:25 +0200129
Pavel Roskin27ae60f2010-03-12 00:01:22 -0500130struct net_device *alloc_libipw(int sizeof_priv, int monitor)
Jeff Garzikb4538722005-05-12 22:48:20 -0400131{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400132 struct libipw_device *ieee;
Jeff Garzikb4538722005-05-12 22:48:20 -0400133 struct net_device *dev;
134 int err;
135
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400136 LIBIPW_DEBUG_INFO("Initializing...\n");
Jeff Garzikb4538722005-05-12 22:48:20 -0400137
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400138 dev = alloc_etherdev(sizeof(struct libipw_device) + sizeof_priv);
Joe Perches41de8d42012-01-29 13:47:52 +0000139 if (!dev)
Jeff Garzikb4538722005-05-12 22:48:20 -0400140 goto failed;
Joe Perches41de8d42012-01-29 13:47:52 +0000141
Jeff Garzikb4538722005-05-12 22:48:20 -0400142 ieee = netdev_priv(dev);
Jeff Garzikb4538722005-05-12 22:48:20 -0400143
144 ieee->dev = dev;
145
John W. Linvillea3caa992009-08-25 14:12:25 -0400146 if (!monitor) {
147 ieee->wdev.wiphy = wiphy_new(&libipw_config_ops, 0);
148 if (!ieee->wdev.wiphy) {
149 LIBIPW_ERROR("Unable to allocate wiphy.\n");
150 goto failed_free_netdev;
151 }
152
153 ieee->dev->ieee80211_ptr = &ieee->wdev;
154 ieee->wdev.iftype = NL80211_IFTYPE_STATION;
155
156 /* Fill-out wiphy structure bits we know... Not enough info
157 here to call set_wiphy_dev or set MAC address or channel info
158 -- have to do that in ->ndo_init... */
159 ieee->wdev.wiphy->privid = libipw_wiphy_privid;
160
161 ieee->wdev.wiphy->max_scan_ssids = 1;
162 ieee->wdev.wiphy->max_scan_ie_len = 0;
163 ieee->wdev.wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION)
164 | BIT(NL80211_IFTYPE_ADHOC);
165 }
166
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400167 err = libipw_networks_allocate(ieee);
Jeff Garzikb4538722005-05-12 22:48:20 -0400168 if (err) {
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400169 LIBIPW_ERROR("Unable to allocate beacon storage: %d\n", err);
John W. Linvillea3caa992009-08-25 14:12:25 -0400170 goto failed_free_wiphy;
Jeff Garzikb4538722005-05-12 22:48:20 -0400171 }
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400172 libipw_networks_initialize(ieee);
Jeff Garzikb4538722005-05-12 22:48:20 -0400173
174 /* Default fragmentation threshold is maximum payload size */
175 ieee->fts = DEFAULT_FTS;
James Ketrenos3cdd00c2005-09-21 11:54:43 -0500176 ieee->rts = DEFAULT_FTS;
Jeff Garzikb4538722005-05-12 22:48:20 -0400177 ieee->scan_age = DEFAULT_MAX_SCAN_AGE;
178 ieee->open_wep = 1;
179
180 /* Default to enabling full open WEP with host based encrypt/decrypt */
181 ieee->host_encrypt = 1;
182 ieee->host_decrypt = 1;
James Ketrenosccd0fda2005-09-21 11:58:32 -0500183 ieee->host_mc_decrypt = 1;
184
André Goddard Rosaaf901ca2009-11-14 13:09:05 -0200185 /* Host fragmentation in Open mode. Default is enabled.
James Ketrenos1264fc02005-09-21 11:54:53 -0500186 * Note: host fragmentation is always enabled if host encryption
187 * is enabled. For cards can do hardware encryption, they must do
188 * hardware fragmentation as well. So we don't need a variable
189 * like host_enc_frag. */
190 ieee->host_open_frag = 1;
Jeff Garzik0edd5b42005-09-07 00:48:31 -0400191 ieee->ieee802_1x = 1; /* Default to supporting 802.1x */
Jeff Garzikb4538722005-05-12 22:48:20 -0400192
Jeff Garzikb4538722005-05-12 22:48:20 -0400193 spin_lock_init(&ieee->lock);
194
John W. Linville2ba4b322008-11-11 16:00:06 -0500195 lib80211_crypt_info_init(&ieee->crypt_info, dev->name, &ieee->lock);
John W. Linville274bfb82008-10-29 11:35:05 -0400196
Jeff Garzik0edd5b42005-09-07 00:48:31 -0400197 ieee->wpa_enabled = 0;
Jeff Garzik0edd5b42005-09-07 00:48:31 -0400198 ieee->drop_unencrypted = 0;
199 ieee->privacy_invoked = 0;
Jeff Garzikb4538722005-05-12 22:48:20 -0400200
201 return dev;
202
John W. Linvillea3caa992009-08-25 14:12:25 -0400203failed_free_wiphy:
204 if (!monitor)
205 wiphy_free(ieee->wdev.wiphy);
Julia Lawalld92a8e82008-07-16 16:34:54 +0200206failed_free_netdev:
207 free_netdev(dev);
208failed:
Jeff Garzikb4538722005-05-12 22:48:20 -0400209 return NULL;
210}
Pavel Roskin27ae60f2010-03-12 00:01:22 -0500211EXPORT_SYMBOL(alloc_libipw);
Jeff Garzikb4538722005-05-12 22:48:20 -0400212
Pavel Roskin27ae60f2010-03-12 00:01:22 -0500213void free_libipw(struct net_device *dev, int monitor)
Jeff Garzikb4538722005-05-12 22:48:20 -0400214{
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400215 struct libipw_device *ieee = netdev_priv(dev);
Jeff Garzikb4538722005-05-12 22:48:20 -0400216
John W. Linville2ba4b322008-11-11 16:00:06 -0500217 lib80211_crypt_info_free(&ieee->crypt_info);
Jeff Garzikb4538722005-05-12 22:48:20 -0400218
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400219 libipw_networks_free(ieee);
John W. Linvillea3caa992009-08-25 14:12:25 -0400220
221 /* free cfg80211 resources */
222 if (!monitor)
223 wiphy_free(ieee->wdev.wiphy);
224
Jeff Garzikb4538722005-05-12 22:48:20 -0400225 free_netdev(dev);
226}
Pavel Roskin27ae60f2010-03-12 00:01:22 -0500227EXPORT_SYMBOL(free_libipw);
Jeff Garzikb4538722005-05-12 22:48:20 -0400228
Helmut Schaa37561622009-03-06 12:02:25 +0100229#ifdef CONFIG_LIBIPW_DEBUG
Jeff Garzikb4538722005-05-12 22:48:20 -0400230
231static int debug = 0;
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400232u32 libipw_debug_level = 0;
233EXPORT_SYMBOL_GPL(libipw_debug_level);
234static struct proc_dir_entry *libipw_proc = NULL;
Jeff Garzikb4538722005-05-12 22:48:20 -0400235
Alexey Dobriyanb8d83392009-11-25 10:14:12 +0300236static int debug_level_proc_show(struct seq_file *m, void *v)
Jeff Garzikb4538722005-05-12 22:48:20 -0400237{
Alexey Dobriyanb8d83392009-11-25 10:14:12 +0300238 seq_printf(m, "0x%08X\n", libipw_debug_level);
239 return 0;
Jeff Garzikb4538722005-05-12 22:48:20 -0400240}
241
Alexey Dobriyanb8d83392009-11-25 10:14:12 +0300242static int debug_level_proc_open(struct inode *inode, struct file *file)
243{
244 return single_open(file, debug_level_proc_show, NULL);
245}
246
247static ssize_t debug_level_proc_write(struct file *file,
248 const char __user *buffer, size_t count, loff_t *pos)
Jeff Garzikb4538722005-05-12 22:48:20 -0400249{
James Ketrenos262d8e42005-09-13 17:42:53 -0500250 char buf[] = "0x00000000\n";
Alexey Dobriyanb8d83392009-11-25 10:14:12 +0300251 size_t len = min(sizeof(buf) - 1, count);
Jeff Garzikb4538722005-05-12 22:48:20 -0400252 unsigned long val;
253
James Ketrenos262d8e42005-09-13 17:42:53 -0500254 if (copy_from_user(buf, buffer, len))
Jeff Garzikb4538722005-05-12 22:48:20 -0400255 return count;
James Ketrenos262d8e42005-09-13 17:42:53 -0500256 buf[len] = 0;
257 if (sscanf(buf, "%li", &val) != 1)
Jeff Garzikb4538722005-05-12 22:48:20 -0400258 printk(KERN_INFO DRV_NAME
259 ": %s is not in hex or decimal form.\n", buf);
260 else
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400261 libipw_debug_level = val;
Jeff Garzikb4538722005-05-12 22:48:20 -0400262
James Ketrenos262d8e42005-09-13 17:42:53 -0500263 return strnlen(buf, len);
Jeff Garzikb4538722005-05-12 22:48:20 -0400264}
Alexey Dobriyanb8d83392009-11-25 10:14:12 +0300265
266static const struct file_operations debug_level_proc_fops = {
267 .owner = THIS_MODULE,
268 .open = debug_level_proc_open,
269 .read = seq_read,
270 .llseek = seq_lseek,
271 .release = single_release,
272 .write = debug_level_proc_write,
273};
Helmut Schaa37561622009-03-06 12:02:25 +0100274#endif /* CONFIG_LIBIPW_DEBUG */
Jeff Garzikb4538722005-05-12 22:48:20 -0400275
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400276static int __init libipw_init(void)
Jeff Garzikb4538722005-05-12 22:48:20 -0400277{
Helmut Schaa37561622009-03-06 12:02:25 +0100278#ifdef CONFIG_LIBIPW_DEBUG
Jeff Garzikb4538722005-05-12 22:48:20 -0400279 struct proc_dir_entry *e;
280
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400281 libipw_debug_level = debug;
Linus Torvalds151f52f2010-11-05 18:57:04 -0700282 libipw_proc = proc_mkdir(DRV_PROCNAME, init_net.proc_net);
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400283 if (libipw_proc == NULL) {
Linus Torvalds151f52f2010-11-05 18:57:04 -0700284 LIBIPW_ERROR("Unable to create " DRV_PROCNAME
Jeff Garzikb4538722005-05-12 22:48:20 -0400285 " proc directory\n");
286 return -EIO;
287 }
Alexey Dobriyanb8d83392009-11-25 10:14:12 +0300288 e = proc_create("debug_level", S_IRUGO | S_IWUSR, libipw_proc,
289 &debug_level_proc_fops);
Jeff Garzikb4538722005-05-12 22:48:20 -0400290 if (!e) {
Linus Torvalds151f52f2010-11-05 18:57:04 -0700291 remove_proc_entry(DRV_PROCNAME, init_net.proc_net);
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400292 libipw_proc = NULL;
Jeff Garzikb4538722005-05-12 22:48:20 -0400293 return -EIO;
294 }
Helmut Schaa37561622009-03-06 12:02:25 +0100295#endif /* CONFIG_LIBIPW_DEBUG */
James Ketrenos31696162005-09-21 11:58:46 -0500296
297 printk(KERN_INFO DRV_NAME ": " DRV_DESCRIPTION ", " DRV_VERSION "\n");
298 printk(KERN_INFO DRV_NAME ": " DRV_COPYRIGHT "\n");
Jeff Garzikb4538722005-05-12 22:48:20 -0400299
300 return 0;
301}
302
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400303static void __exit libipw_exit(void)
Jeff Garzikb4538722005-05-12 22:48:20 -0400304{
Helmut Schaa37561622009-03-06 12:02:25 +0100305#ifdef CONFIG_LIBIPW_DEBUG
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400306 if (libipw_proc) {
307 remove_proc_entry("debug_level", libipw_proc);
Linus Torvalds151f52f2010-11-05 18:57:04 -0700308 remove_proc_entry(DRV_PROCNAME, init_net.proc_net);
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400309 libipw_proc = NULL;
Jeff Garzikb4538722005-05-12 22:48:20 -0400310 }
Helmut Schaa37561622009-03-06 12:02:25 +0100311#endif /* CONFIG_LIBIPW_DEBUG */
Jeff Garzikb4538722005-05-12 22:48:20 -0400312}
313
Helmut Schaa37561622009-03-06 12:02:25 +0100314#ifdef CONFIG_LIBIPW_DEBUG
Jeff Garzikb4538722005-05-12 22:48:20 -0400315#include <linux/moduleparam.h>
316module_param(debug, int, 0444);
317MODULE_PARM_DESC(debug, "debug output mask");
Helmut Schaa37561622009-03-06 12:02:25 +0100318#endif /* CONFIG_LIBIPW_DEBUG */
Jeff Garzikb4538722005-05-12 22:48:20 -0400319
John W. Linvilleb0a4e7d2009-08-20 14:48:03 -0400320module_exit(libipw_exit);
321module_init(libipw_init);