debugfs_key.c 8.4 KB
Newer Older
J
Jiri Benc 已提交
1 2 3 4 5 6 7 8 9 10 11
/*
 * Copyright 2003-2005	Devicescape Software, Inc.
 * Copyright (c) 2006	Jiri Benc <jbenc@suse.cz>
 * Copyright 2007	Johannes Berg <johannes@sipsolutions.net>
 *
 * 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/kobject.h>
12
#include <linux/slab.h>
J
Jiri Benc 已提交
13
#include "ieee80211_i.h"
J
Johannes Berg 已提交
14
#include "key.h"
J
Jiri Benc 已提交
15 16 17
#include "debugfs.h"
#include "debugfs_key.h"

18
#define KEY_READ(name, prop, buflen, format_string)			\
J
Jiri Benc 已提交
19 20 21 22 23 24
static ssize_t key_##name##_read(struct file *file,			\
				 char __user *userbuf,			\
				 size_t count, loff_t *ppos)		\
{									\
	char buf[buflen];						\
	struct ieee80211_key *key = file->private_data;			\
25
	int res = scnprintf(buf, buflen, format_string, key->prop);	\
J
Jiri Benc 已提交
26 27
	return simple_read_from_buffer(userbuf, count, ppos, buf, res);	\
}
28
#define KEY_READ_D(name) KEY_READ(name, name, 20, "%d\n")
J
Johannes Berg 已提交
29
#define KEY_READ_X(name) KEY_READ(name, name, 20, "0x%x\n")
J
Jiri Benc 已提交
30 31 32 33 34 35 36 37 38 39 40

#define KEY_OPS(name)							\
static const struct file_operations key_ ##name## _ops = {		\
	.read = key_##name##_read,					\
	.open = mac80211_open_file_generic,				\
}

#define KEY_FILE(name, format)						\
		 KEY_READ_##format(name)				\
		 KEY_OPS(name)

41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
#define KEY_CONF_READ(name, buflen, format_string)			\
	KEY_READ(conf_##name, conf.name, buflen, format_string)
#define KEY_CONF_READ_D(name) KEY_CONF_READ(name, 20, "%d\n")

#define KEY_CONF_OPS(name)						\
static const struct file_operations key_ ##name## _ops = {		\
	.read = key_conf_##name##_read,					\
	.open = mac80211_open_file_generic,				\
}

#define KEY_CONF_FILE(name, format)					\
		 KEY_CONF_READ_##format(name)				\
		 KEY_CONF_OPS(name)

KEY_CONF_FILE(keylen, D);
KEY_CONF_FILE(keyidx, D);
KEY_CONF_FILE(hw_key_idx, D);
J
Johannes Berg 已提交
58
KEY_FILE(flags, X);
J
Jiri Benc 已提交
59
KEY_FILE(tx_rx_count, D);
60
KEY_READ(ifindex, sdata->name, IFNAMSIZ + 2, "%s\n");
61
KEY_OPS(ifindex);
J
Jiri Benc 已提交
62 63 64 65 66 67 68 69

static ssize_t key_algorithm_read(struct file *file,
				  char __user *userbuf,
				  size_t count, loff_t *ppos)
{
	char *alg;
	struct ieee80211_key *key = file->private_data;

70
	switch (key->conf.alg) {
J
Jiri Benc 已提交
71 72 73 74 75 76 77 78 79
	case ALG_WEP:
		alg = "WEP\n";
		break;
	case ALG_TKIP:
		alg = "TKIP\n";
		break;
	case ALG_CCMP:
		alg = "CCMP\n";
		break;
80 81 82
	case ALG_AES_CMAC:
		alg = "AES-128-CMAC\n";
		break;
J
Jiri Benc 已提交
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
	default:
		return 0;
	}
	return simple_read_from_buffer(userbuf, count, ppos, alg, strlen(alg));
}
KEY_OPS(algorithm);

static ssize_t key_tx_spec_read(struct file *file, char __user *userbuf,
				size_t count, loff_t *ppos)
{
	const u8 *tpn;
	char buf[20];
	int len;
	struct ieee80211_key *key = file->private_data;

98
	switch (key->conf.alg) {
J
Jiri Benc 已提交
99 100
	case ALG_WEP:
		len = scnprintf(buf, sizeof(buf), "\n");
J
Johannes Berg 已提交
101
		break;
J
Jiri Benc 已提交
102 103
	case ALG_TKIP:
		len = scnprintf(buf, sizeof(buf), "%08x %04x\n",
104 105
				key->u.tkip.tx.iv32,
				key->u.tkip.tx.iv16);
J
Johannes Berg 已提交
106
		break;
J
Jiri Benc 已提交
107 108 109 110
	case ALG_CCMP:
		tpn = key->u.ccmp.tx_pn;
		len = scnprintf(buf, sizeof(buf), "%02x%02x%02x%02x%02x%02x\n",
				tpn[0], tpn[1], tpn[2], tpn[3], tpn[4], tpn[5]);
J
Johannes Berg 已提交
111
		break;
112 113 114 115 116 117
	case ALG_AES_CMAC:
		tpn = key->u.aes_cmac.tx_pn;
		len = scnprintf(buf, sizeof(buf), "%02x%02x%02x%02x%02x%02x\n",
				tpn[0], tpn[1], tpn[2], tpn[3], tpn[4],
				tpn[5]);
		break;
J
Jiri Benc 已提交
118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
	default:
		return 0;
	}
	return simple_read_from_buffer(userbuf, count, ppos, buf, len);
}
KEY_OPS(tx_spec);

static ssize_t key_rx_spec_read(struct file *file, char __user *userbuf,
				size_t count, loff_t *ppos)
{
	struct ieee80211_key *key = file->private_data;
	char buf[14*NUM_RX_DATA_QUEUES+1], *p = buf;
	int i, len;
	const u8 *rpn;

133
	switch (key->conf.alg) {
J
Jiri Benc 已提交
134 135
	case ALG_WEP:
		len = scnprintf(buf, sizeof(buf), "\n");
J
Johannes Berg 已提交
136
		break;
J
Jiri Benc 已提交
137 138 139 140
	case ALG_TKIP:
		for (i = 0; i < NUM_RX_DATA_QUEUES; i++)
			p += scnprintf(p, sizeof(buf)+buf-p,
				       "%08x %04x\n",
141 142
				       key->u.tkip.rx[i].iv32,
				       key->u.tkip.rx[i].iv16);
J
Jiri Benc 已提交
143
		len = p - buf;
J
Johannes Berg 已提交
144
		break;
J
Jiri Benc 已提交
145
	case ALG_CCMP:
146
		for (i = 0; i < NUM_RX_DATA_QUEUES + 1; i++) {
J
Jiri Benc 已提交
147 148 149 150 151 152 153
			rpn = key->u.ccmp.rx_pn[i];
			p += scnprintf(p, sizeof(buf)+buf-p,
				       "%02x%02x%02x%02x%02x%02x\n",
				       rpn[0], rpn[1], rpn[2],
				       rpn[3], rpn[4], rpn[5]);
		}
		len = p - buf;
J
Johannes Berg 已提交
154
		break;
155 156 157 158 159 160 161 162
	case ALG_AES_CMAC:
		rpn = key->u.aes_cmac.rx_pn;
		p += scnprintf(p, sizeof(buf)+buf-p,
			       "%02x%02x%02x%02x%02x%02x\n",
			       rpn[0], rpn[1], rpn[2],
			       rpn[3], rpn[4], rpn[5]);
		len = p - buf;
		break;
J
Jiri Benc 已提交
163 164 165 166 167 168 169 170 171 172 173 174 175 176
	default:
		return 0;
	}
	return simple_read_from_buffer(userbuf, count, ppos, buf, len);
}
KEY_OPS(rx_spec);

static ssize_t key_replays_read(struct file *file, char __user *userbuf,
				size_t count, loff_t *ppos)
{
	struct ieee80211_key *key = file->private_data;
	char buf[20];
	int len;

177 178 179 180 181 182 183 184 185
	switch (key->conf.alg) {
	case ALG_CCMP:
		len = scnprintf(buf, sizeof(buf), "%u\n", key->u.ccmp.replays);
		break;
	case ALG_AES_CMAC:
		len = scnprintf(buf, sizeof(buf), "%u\n",
				key->u.aes_cmac.replays);
		break;
	default:
J
Jiri Benc 已提交
186
		return 0;
187
	}
J
Jiri Benc 已提交
188 189 190 191
	return simple_read_from_buffer(userbuf, count, ppos, buf, len);
}
KEY_OPS(replays);

192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210
static ssize_t key_icverrors_read(struct file *file, char __user *userbuf,
				  size_t count, loff_t *ppos)
{
	struct ieee80211_key *key = file->private_data;
	char buf[20];
	int len;

	switch (key->conf.alg) {
	case ALG_AES_CMAC:
		len = scnprintf(buf, sizeof(buf), "%u\n",
				key->u.aes_cmac.icverrors);
		break;
	default:
		return 0;
	}
	return simple_read_from_buffer(userbuf, count, ppos, buf, len);
}
KEY_OPS(icverrors);

J
Jiri Benc 已提交
211 212 213 214
static ssize_t key_key_read(struct file *file, char __user *userbuf,
			    size_t count, loff_t *ppos)
{
	struct ieee80211_key *key = file->private_data;
215
	int i, res, bufsize = 2 * key->conf.keylen + 2;
J
Jiri Benc 已提交
216 217 218
	char *buf = kmalloc(bufsize, GFP_KERNEL);
	char *p = buf;

219 220
	for (i = 0; i < key->conf.keylen; i++)
		p += scnprintf(p, bufsize + buf - p, "%02x", key->conf.key[i]);
J
Jiri Benc 已提交
221 222 223 224 225 226 227 228
	p += scnprintf(p, bufsize+buf-p, "\n");
	res = simple_read_from_buffer(userbuf, count, ppos, buf, p - buf);
	kfree(buf);
	return res;
}
KEY_OPS(key);

#define DEBUGFS_ADD(name) \
229 230
	debugfs_create_file(#name, 0400, key->debugfs.dir, \
			    key, &key_##name##_ops);
J
Jiri Benc 已提交
231

232 233
void ieee80211_debugfs_key_add(struct ieee80211_key *key)
  {
J
Johannes Berg 已提交
234
	static int keycount;
235 236
	char buf[50];
	struct sta_info *sta;
J
Jiri Benc 已提交
237

238
	if (!key->local->debugfs.keys)
J
Jiri Benc 已提交
239 240
		return;

J
Johannes Berg 已提交
241
	sprintf(buf, "%d", keycount);
242
	key->debugfs.cnt = keycount;
J
Johannes Berg 已提交
243
	keycount++;
J
Jiri Benc 已提交
244
	key->debugfs.dir = debugfs_create_dir(buf,
245
					key->local->debugfs.keys);
J
Jiri Benc 已提交
246 247 248 249

	if (!key->debugfs.dir)
		return;

250 251 252
	rcu_read_lock();
	sta = rcu_dereference(key->sta);
	if (sta)
253
		sprintf(buf, "../../stations/%pM", sta->sta.addr);
254 255 256 257 258 259 260
	rcu_read_unlock();

	/* using sta as a boolean is fine outside RCU lock */
	if (sta)
		key->debugfs.stalink =
			debugfs_create_symlink("station", key->debugfs.dir, buf);

J
Jiri Benc 已提交
261
	DEBUGFS_ADD(keylen);
262
	DEBUGFS_ADD(flags);
J
Jiri Benc 已提交
263 264 265 266 267 268 269
	DEBUGFS_ADD(keyidx);
	DEBUGFS_ADD(hw_key_idx);
	DEBUGFS_ADD(tx_rx_count);
	DEBUGFS_ADD(algorithm);
	DEBUGFS_ADD(tx_spec);
	DEBUGFS_ADD(rx_spec);
	DEBUGFS_ADD(replays);
270
	DEBUGFS_ADD(icverrors);
J
Jiri Benc 已提交
271
	DEBUGFS_ADD(key);
272
	DEBUGFS_ADD(ifindex);
J
Jiri Benc 已提交
273 274 275 276 277 278 279
};

void ieee80211_debugfs_key_remove(struct ieee80211_key *key)
{
	if (!key)
		return;

280
	debugfs_remove_recursive(key->debugfs.dir);
J
Jiri Benc 已提交
281 282 283 284 285
	key->debugfs.dir = NULL;
}
void ieee80211_debugfs_key_add_default(struct ieee80211_sub_if_data *sdata)
{
	char buf[50];
286
	struct ieee80211_key *key;
J
Jiri Benc 已提交
287

288
	if (!sdata->debugfs.dir)
J
Jiri Benc 已提交
289 290
		return;

291 292 293 294 295
	/* this is running under the key lock */

	key = sdata->default_key;
	if (key) {
		sprintf(buf, "../keys/%d", key->debugfs.cnt);
296
		sdata->debugfs.default_key =
297
			debugfs_create_symlink("default_key",
298
					       sdata->debugfs.dir, buf);
299 300
	} else
		ieee80211_debugfs_key_remove_default(sdata);
J
Jiri Benc 已提交
301
}
302

J
Jiri Benc 已提交
303 304 305 306 307
void ieee80211_debugfs_key_remove_default(struct ieee80211_sub_if_data *sdata)
{
	if (!sdata)
		return;

308 309
	debugfs_remove(sdata->debugfs.default_key);
	sdata->debugfs.default_key = NULL;
J
Jiri Benc 已提交
310 311
}

312 313 314 315 316
void ieee80211_debugfs_key_add_mgmt_default(struct ieee80211_sub_if_data *sdata)
{
	char buf[50];
	struct ieee80211_key *key;

317
	if (!sdata->debugfs.dir)
318 319 320 321 322 323 324
		return;

	/* this is running under the key lock */

	key = sdata->default_mgmt_key;
	if (key) {
		sprintf(buf, "../keys/%d", key->debugfs.cnt);
325
		sdata->debugfs.default_mgmt_key =
326
			debugfs_create_symlink("default_mgmt_key",
327
					       sdata->debugfs.dir, buf);
328 329 330 331 332 333 334 335 336
	} else
		ieee80211_debugfs_key_remove_mgmt_default(sdata);
}

void ieee80211_debugfs_key_remove_mgmt_default(struct ieee80211_sub_if_data *sdata)
{
	if (!sdata)
		return;

337 338
	debugfs_remove(sdata->debugfs.default_mgmt_key);
	sdata->debugfs.default_mgmt_key = NULL;
339 340
}

J
Jiri Benc 已提交
341 342 343 344 345 346
void ieee80211_debugfs_key_sta_del(struct ieee80211_key *key,
				   struct sta_info *sta)
{
	debugfs_remove(key->debugfs.stalink);
	key->debugfs.stalink = NULL;
}