iface.c 6.1 KB
Newer Older
1 2 3 4
/*
 * Copyright 2002-2005, Instant802 Networks, Inc.
 * Copyright 2005-2006, Devicescape Software, Inc.
 * Copyright (c) 2006 Jiri Benc <jbenc@suse.cz>
5
 * Copyright 2008, Johannes Berg <johannes@sipsolutions.net>
6 7 8 9 10 11 12 13 14 15 16 17
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */
#include <linux/kernel.h>
#include <linux/if_arp.h>
#include <linux/netdevice.h>
#include <linux/rtnetlink.h>
#include <net/mac80211.h>
#include "ieee80211_i.h"
#include "sta_info.h"
J
Jiri Benc 已提交
18
#include "debugfs_netdev.h"
19
#include "mesh.h"
20

21 22 23 24 25
/*
 * Called when the netdev is removed or, by the code below, before
 * the interface type changes.
 */
static void ieee80211_teardown_sdata(struct net_device *dev)
26
{
27 28 29 30 31
	struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev);
	struct ieee80211_local *local = sdata->local;
	struct beacon_data *beacon;
	struct sk_buff *skb;
	int flushed;
32 33
	int i;

34 35 36
	/* free extra data */
	ieee80211_free_keys(sdata);

37 38
	ieee80211_debugfs_remove_netdev(sdata);

39
	for (i = 0; i < IEEE80211_FRAGMENT_MAX; i++)
40 41
		__skb_queue_purge(&sdata->fragments[i].skb_list);
	sdata->fragment_next = 0;
J
Johannes Berg 已提交
42

43 44 45 46 47 48
	switch (sdata->vif.type) {
	case IEEE80211_IF_TYPE_AP:
		beacon = sdata->u.ap.beacon;
		rcu_assign_pointer(sdata->u.ap.beacon, NULL);
		synchronize_rcu();
		kfree(beacon);
49

50 51 52 53 54 55 56 57
		while ((skb = skb_dequeue(&sdata->u.ap.ps_bc_buf))) {
			local->total_ps_buffered--;
			dev_kfree_skb(skb);
		}

		break;
	case IEEE80211_IF_TYPE_MESH_POINT:
		if (ieee80211_vif_is_mesh(&sdata->vif))
58
			mesh_rmc_free(sdata);
59
		break;
60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77
	case IEEE80211_IF_TYPE_STA:
	case IEEE80211_IF_TYPE_IBSS:
		kfree(sdata->u.sta.extra_ie);
		kfree(sdata->u.sta.assocreq_ies);
		kfree(sdata->u.sta.assocresp_ies);
		kfree_skb(sdata->u.sta.probe_resp);
		break;
	case IEEE80211_IF_TYPE_WDS:
	case IEEE80211_IF_TYPE_VLAN:
	case IEEE80211_IF_TYPE_MNTR:
		break;
	case IEEE80211_IF_TYPE_INVALID:
		BUG();
		break;
	}

	flushed = sta_info_flush(local, sdata);
	WARN_ON(flushed);
78 79
}

80 81 82 83 84
/*
 * Helper function to initialise an interface to a specific type.
 */
static void ieee80211_setup_sdata(struct ieee80211_sub_if_data *sdata,
				  enum ieee80211_if_types type)
85
{
86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
	/* clear type-dependent union */
	memset(&sdata->u, 0, sizeof(sdata->u));

	/* and set some type-dependent values */
	sdata->vif.type = type;

	/* only monitor differs */
	sdata->dev->type = ARPHRD_ETHER;

	switch (type) {
	case IEEE80211_IF_TYPE_AP:
		skb_queue_head_init(&sdata->u.ap.ps_bc_buf);
		INIT_LIST_HEAD(&sdata->u.ap.vlans);
		break;
	case IEEE80211_IF_TYPE_STA:
	case IEEE80211_IF_TYPE_IBSS:
J
Johannes Berg 已提交
102
		ieee80211_sta_setup_sdata(sdata);
103 104
		break;
	case IEEE80211_IF_TYPE_MESH_POINT:
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
		if (ieee80211_vif_is_mesh(&sdata->vif))
			ieee80211_mesh_init_sdata(sdata);
		break;
	case IEEE80211_IF_TYPE_MNTR:
		sdata->dev->type = ARPHRD_IEEE80211_RADIOTAP;
		sdata->dev->hard_start_xmit = ieee80211_monitor_start_xmit;
		sdata->u.mntr_flags = MONITOR_FLAG_CONTROL |
				      MONITOR_FLAG_OTHER_BSS;
		break;
	case IEEE80211_IF_TYPE_WDS:
	case IEEE80211_IF_TYPE_VLAN:
		break;
	case IEEE80211_IF_TYPE_INVALID:
		BUG();
		break;
	}

	ieee80211_debugfs_add_netdev(sdata);
}

125 126
int ieee80211_if_change_type(struct ieee80211_sub_if_data *sdata,
			     enum ieee80211_if_types type)
127
{
128 129 130 131 132 133 134 135 136 137 138 139 140 141
	ASSERT_RTNL();

	if (type == sdata->vif.type)
		return 0;

	/*
	 * We could, here, on changes between IBSS/STA/MESH modes,
	 * invoke an MLME function instead that disassociates etc.
	 * and goes into the requested mode.
	 */

	if (netif_running(sdata->dev))
		return -EBUSY;

142 143 144 145 146 147 148
	/* Purge and reset type-dependent state. */
	ieee80211_teardown_sdata(sdata->dev);
	ieee80211_setup_sdata(sdata, type);

	/* reset some values that shouldn't be kept across type changes */
	sdata->basic_rates = 0;
	sdata->drop_unencrypted = 0;
149 150

	return 0;
151 152
}

153
int ieee80211_if_add(struct ieee80211_local *local, const char *name,
154
		     struct net_device **new_dev, enum ieee80211_if_types type,
155
		     struct vif_params *params)
156 157 158
{
	struct net_device *ndev;
	struct ieee80211_sub_if_data *sdata = NULL;
159
	int ret, i;
160 161

	ASSERT_RTNL();
162

163
	ndev = alloc_netdev(sizeof(*sdata) + local->hw.vif_data_size,
164 165 166 167
			    name, ieee80211_if_setup);
	if (!ndev)
		return -ENOMEM;

168 169 170 171 172 173 174 175 176
	ndev->needed_headroom = local->tx_headroom +
				4*6 /* four MAC addresses */
				+ 2 + 2 + 2 + 2 /* ctl, dur, seq, qos */
				+ 6 /* mesh */
				+ 8 /* rfc1042/bridge tunnel */
				- ETH_HLEN /* ethernet hard_header_len */
				+ IEEE80211_ENCRYPT_HEADROOM;
	ndev->needed_tailroom = IEEE80211_ENCRYPT_TAILROOM;

177 178 179 180 181 182 183
	ret = dev_alloc_name(ndev, ndev->name);
	if (ret < 0)
		goto fail;

	memcpy(ndev->dev_addr, local->hw.wiphy->perm_addr, ETH_ALEN);
	SET_NETDEV_DEV(ndev, wiphy_dev(local->hw.wiphy));

184 185
	/* don't use IEEE80211_DEV_TO_SUB_IF because it checks too much */
	sdata = netdev_priv(ndev);
186
	ndev->ieee80211_ptr = &sdata->wdev;
187 188

	/* initialise type-independent data */
189 190
	sdata->wdev.wiphy = local->hw.wiphy;
	sdata->local = local;
191 192 193 194 195 196 197 198 199 200 201 202
	sdata->dev = ndev;

	for (i = 0; i < IEEE80211_FRAGMENT_MAX; i++)
		skb_queue_head_init(&sdata->fragments[i].skb_list);

	INIT_LIST_HEAD(&sdata->key_list);

	sdata->force_unicast_rateidx = -1;
	sdata->max_ratectrl_rateidx = -1;

	/* setup type-dependent data */
	ieee80211_setup_sdata(sdata, type);
203 204 205 206 207

	ret = register_netdevice(ndev);
	if (ret)
		goto fail;

208
	ndev->uninit = ieee80211_teardown_sdata;
209

J
Johannes Berg 已提交
210 211
	if (ieee80211_vif_is_mesh(&sdata->vif) &&
	    params && params->mesh_id_len)
212 213 214
		ieee80211_sdata_set_mesh_id(sdata,
					    params->mesh_id_len,
					    params->mesh_id);
215

216 217
	list_add_tail_rcu(&sdata->list, &local->interfaces);

218 219 220 221 222
	if (new_dev)
		*new_dev = ndev;

	return 0;

223
 fail:
224 225 226 227
	free_netdev(ndev);
	return ret;
}

228
void ieee80211_if_remove(struct ieee80211_sub_if_data *sdata)
229 230
{
	ASSERT_RTNL();
J
Johannes Berg 已提交
231

232 233
	list_del_rcu(&sdata->list);
	synchronize_rcu();
234
	unregister_netdevice(sdata->dev);
235 236
}

237 238 239 240 241
/*
 * Remove all interfaces, may only be called at hardware unregistration
 * time because it doesn't do RCU-safe list removals.
 */
void ieee80211_remove_interfaces(struct ieee80211_local *local)
242
{
243
	struct ieee80211_sub_if_data *sdata, *tmp;
244 245 246

	ASSERT_RTNL();

247 248 249
	list_for_each_entry_safe(sdata, tmp, &local->interfaces, list) {
		list_del(&sdata->list);
		unregister_netdevice(sdata->dev);
250 251
	}
}