rate.c 6.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
/*
 * Copyright 2002-2005, Instant802 Networks, Inc.
 * Copyright 2005-2006, Devicescape Software, Inc.
 * Copyright (c) 2006 Jiri Benc <jbenc@suse.cz>
 *
 * 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>
12
#include <linux/rtnetlink.h>
J
Johannes Berg 已提交
13
#include "rate.h"
14
#include "ieee80211_i.h"
15
#include "debugfs.h"
16 17 18 19 20 21 22 23 24

struct rate_control_alg {
	struct list_head list;
	struct rate_control_ops *ops;
};

static LIST_HEAD(rate_ctrl_algs);
static DEFINE_MUTEX(rate_ctrl_mutex);

25 26 27 28 29
static char *ieee80211_default_rc_algo = CONFIG_MAC80211_RC_DEFAULT;
module_param(ieee80211_default_rc_algo, charp, 0644);
MODULE_PARM_DESC(ieee80211_default_rc_algo,
		 "Default rate control algorithm for mac80211 to use");

30 31 32 33
int ieee80211_rate_control_register(struct rate_control_ops *ops)
{
	struct rate_control_alg *alg;

34 35 36
	if (!ops->name)
		return -EINVAL;

37 38 39 40 41
	mutex_lock(&rate_ctrl_mutex);
	list_for_each_entry(alg, &rate_ctrl_algs, list) {
		if (!strcmp(alg->ops->name, ops->name)) {
			/* don't register an algorithm twice */
			WARN_ON(1);
C
Cyrill Gorcunov 已提交
42
			mutex_unlock(&rate_ctrl_mutex);
43 44 45 46
			return -EALREADY;
		}
	}

47
	alg = kzalloc(sizeof(*alg), GFP_KERNEL);
48
	if (alg == NULL) {
49
		mutex_unlock(&rate_ctrl_mutex);
50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
		return -ENOMEM;
	}
	alg->ops = ops;

	list_add_tail(&alg->list, &rate_ctrl_algs);
	mutex_unlock(&rate_ctrl_mutex);

	return 0;
}
EXPORT_SYMBOL(ieee80211_rate_control_register);

void ieee80211_rate_control_unregister(struct rate_control_ops *ops)
{
	struct rate_control_alg *alg;

	mutex_lock(&rate_ctrl_mutex);
	list_for_each_entry(alg, &rate_ctrl_algs, list) {
		if (alg->ops == ops) {
			list_del(&alg->list);
69
			kfree(alg);
70 71 72 73 74 75 76 77 78 79 80 81 82
			break;
		}
	}
	mutex_unlock(&rate_ctrl_mutex);
}
EXPORT_SYMBOL(ieee80211_rate_control_unregister);

static struct rate_control_ops *
ieee80211_try_rate_control_ops_get(const char *name)
{
	struct rate_control_alg *alg;
	struct rate_control_ops *ops = NULL;

83 84 85
	if (!name)
		return NULL;

86 87
	mutex_lock(&rate_ctrl_mutex);
	list_for_each_entry(alg, &rate_ctrl_algs, list) {
88
		if (!strcmp(alg->ops->name, name))
89 90 91 92 93 94 95 96 97
			if (try_module_get(alg->ops->module)) {
				ops = alg->ops;
				break;
			}
	}
	mutex_unlock(&rate_ctrl_mutex);
	return ops;
}

98
/* Get the rate control algorithm. */
99 100 101 102
static struct rate_control_ops *
ieee80211_rate_control_ops_get(const char *name)
{
	struct rate_control_ops *ops;
103
	const char *alg_name;
104

105
	if (!name)
106 107 108
		alg_name = ieee80211_default_rc_algo;
	else
		alg_name = name;
109

110
	ops = ieee80211_try_rate_control_ops_get(alg_name);
111
	if (!ops) {
112 113
		request_module("rc80211_%s", alg_name);
		ops = ieee80211_try_rate_control_ops_get(alg_name);
114
	}
115 116 117 118
	if (!ops && name)
		/* try default if specific alg requested but not found */
		ops = ieee80211_try_rate_control_ops_get(ieee80211_default_rc_algo);

119 120 121 122
	/* try built-in one if specific alg requested but not found */
	if (!ops && strlen(CONFIG_MAC80211_RC_DEFAULT))
		ops = ieee80211_try_rate_control_ops_get(CONFIG_MAC80211_RC_DEFAULT);

123 124 125 126 127 128 129 130
	return ops;
}

static void ieee80211_rate_control_ops_put(struct rate_control_ops *ops)
{
	module_put(ops->module);
}

131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
#ifdef CONFIG_MAC80211_DEBUGFS
static ssize_t rcname_read(struct file *file, char __user *userbuf,
			   size_t count, loff_t *ppos)
{
	struct rate_control_ref *ref = file->private_data;
	int len = strlen(ref->ops->name);

	return simple_read_from_buffer(userbuf, count, ppos,
				       ref->ops->name, len);
}

static const struct file_operations rcname_ops = {
	.read = rcname_read,
	.open = mac80211_open_file_generic,
};
#endif

148 149 150
struct rate_control_ref *rate_control_alloc(const char *name,
					    struct ieee80211_local *local)
{
151
	struct dentry *debugfsdir = NULL;
152 153 154 155 156 157
	struct rate_control_ref *ref;

	ref = kmalloc(sizeof(struct rate_control_ref), GFP_KERNEL);
	if (!ref)
		goto fail_ref;
	kref_init(&ref->kref);
158
	ref->local = local;
159 160 161
	ref->ops = ieee80211_rate_control_ops_get(name);
	if (!ref->ops)
		goto fail_ops;
162 163 164 165 166 167 168 169 170

#ifdef CONFIG_MAC80211_DEBUGFS
	debugfsdir = debugfs_create_dir("rc", local->hw.wiphy->debugfsdir);
	local->debugfs.rcdir = debugfsdir;
	local->debugfs.rcname = debugfs_create_file("name", 0400, debugfsdir,
						    ref, &rcname_ops);
#endif

	ref->priv = ref->ops->alloc(&local->hw, debugfsdir);
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
	if (!ref->priv)
		goto fail_priv;
	return ref;

fail_priv:
	ieee80211_rate_control_ops_put(ref->ops);
fail_ops:
	kfree(ref);
fail_ref:
	return NULL;
}

static void rate_control_release(struct kref *kref)
{
	struct rate_control_ref *ctrl_ref;

	ctrl_ref = container_of(kref, struct rate_control_ref, kref);
	ctrl_ref->ops->free(ctrl_ref->priv);
189 190 191 192 193 194 195 196

#ifdef CONFIG_MAC80211_DEBUGFS
	debugfs_remove(ctrl_ref->local->debugfs.rcname);
	ctrl_ref->local->debugfs.rcname = NULL;
	debugfs_remove(ctrl_ref->local->debugfs.rcdir);
	ctrl_ref->local->debugfs.rcdir = NULL;
#endif

197 198 199 200
	ieee80211_rate_control_ops_put(ctrl_ref->ops);
	kfree(ctrl_ref);
}

201
void rate_control_get_rate(struct ieee80211_sub_if_data *sdata,
202 203
			   struct sta_info *sta,
			   struct ieee80211_tx_rate_control *txrc)
204
{
205 206 207
	struct rate_control_ref *ref = sdata->local->rate_ctrl;
	void *priv_sta = NULL;
	struct ieee80211_sta *ista = NULL;
208
	struct ieee80211_tx_info *info = IEEE80211_SKB_CB(txrc->skb);
209 210
	int i;

211 212 213 214 215
	if (sta) {
		ista = &sta->sta;
		priv_sta = sta->rate_ctrl_priv;
	}

216 217 218 219 220 221
	for (i = 0; i < IEEE80211_TX_MAX_RATES; i++) {
		info->control.rates[i].idx = -1;
		info->control.rates[i].flags = 0;
		info->control.rates[i].count = 1;
	}

222
	if (sta && sdata->force_unicast_rateidx > -1)
223
		info->control.rates[0].idx = sdata->force_unicast_rateidx;
224
	else
225 226 227 228 229 230 231 232 233 234 235 236
		ref->ops->get_rate(ref->priv, ista, priv_sta, txrc);

	/*
	 * try to enforce the maximum rate the user wanted
	 */
	if (sdata->max_ratectrl_rateidx > -1)
		for (i = 0; i < IEEE80211_TX_MAX_RATES; i++) {
			if (info->control.rates[i].flags & IEEE80211_TX_RC_MCS)
				continue;
			info->control.rates[i].idx =
				min_t(s8, info->control.rates[i].idx,
				      sdata->max_ratectrl_rateidx);
237
	}
238 239

	BUG_ON(info->control.rates[0].idx < 0);
240 241
}

242 243 244 245 246 247 248 249 250 251
struct rate_control_ref *rate_control_get(struct rate_control_ref *ref)
{
	kref_get(&ref->kref);
	return ref;
}

void rate_control_put(struct rate_control_ref *ref)
{
	kref_put(&ref->kref, rate_control_release);
}
252 253 254 255 256 257 258

int ieee80211_init_rate_ctrl_alg(struct ieee80211_local *local,
				 const char *name)
{
	struct rate_control_ref *ref, *old;

	ASSERT_RTNL();
259
	if (local->open_count || netif_running(local->mdev))
260 261 262 263 264
		return -EBUSY;

	ref = rate_control_alloc(name, local);
	if (!ref) {
		printk(KERN_WARNING "%s: Failed to select rate control "
265
		       "algorithm\n", wiphy_name(local->hw.wiphy));
266 267 268 269 270 271 272 273 274 275 276
		return -ENOENT;
	}

	old = local->rate_ctrl;
	local->rate_ctrl = ref;
	if (old) {
		rate_control_put(old);
		sta_info_flush(local, NULL);
	}

	printk(KERN_DEBUG "%s: Selected rate control "
277
	       "algorithm '%s'\n", wiphy_name(local->hw.wiphy),
278 279 280 281 282 283 284 285 286 287 288 289 290 291
	       ref->ops->name);


	return 0;
}

void rate_control_deinitialize(struct ieee80211_local *local)
{
	struct rate_control_ref *ref;

	ref = local->rate_ctrl;
	local->rate_ctrl = NULL;
	rate_control_put(ref);
}
292