debugfs_key.c 7.0 KB
Newer Older
J
Jiri Benc 已提交
1 2 3 4 5 6 7 8 9 10 11 12
/*
 * 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>
#include "ieee80211_i.h"
J
Johannes Berg 已提交
13
#include "key.h"
J
Jiri Benc 已提交
14 15 16
#include "debugfs.h"
#include "debugfs_key.h"

17
#define KEY_READ(name, prop, buflen, format_string)			\
J
Jiri Benc 已提交
18 19 20 21 22 23
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;			\
24
	int res = scnprintf(buf, buflen, format_string, key->prop);	\
J
Jiri Benc 已提交
25 26
	return simple_read_from_buffer(userbuf, count, ppos, buf, res);	\
}
27
#define KEY_READ_D(name) KEY_READ(name, name, 20, "%d\n")
J
Johannes Berg 已提交
28
#define KEY_READ_X(name) KEY_READ(name, name, 20, "0x%x\n")
J
Jiri Benc 已提交
29 30 31 32 33 34 35 36 37 38 39

#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)

40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
#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 已提交
57
KEY_FILE(flags, X);
J
Jiri Benc 已提交
58
KEY_FILE(tx_rx_count, D);
59 60
KEY_READ(ifindex, sdata->dev->ifindex, 20, "%d\n");
KEY_OPS(ifindex);
J
Jiri Benc 已提交
61 62 63 64 65 66 67 68

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;

69
	switch (key->conf.alg) {
J
Jiri Benc 已提交
70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
	case ALG_WEP:
		alg = "WEP\n";
		break;
	case ALG_TKIP:
		alg = "TKIP\n";
		break;
	case ALG_CCMP:
		alg = "CCMP\n";
		break;
	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;

94
	switch (key->conf.alg) {
J
Jiri Benc 已提交
95 96
	case ALG_WEP:
		len = scnprintf(buf, sizeof(buf), "\n");
J
Johannes Berg 已提交
97
		break;
J
Jiri Benc 已提交
98 99 100 101
	case ALG_TKIP:
		len = scnprintf(buf, sizeof(buf), "%08x %04x\n",
				key->u.tkip.iv32,
				key->u.tkip.iv16);
J
Johannes Berg 已提交
102
		break;
J
Jiri Benc 已提交
103 104 105 106
	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 已提交
107
		break;
J
Jiri Benc 已提交
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
	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;

123
	switch (key->conf.alg) {
J
Jiri Benc 已提交
124 125
	case ALG_WEP:
		len = scnprintf(buf, sizeof(buf), "\n");
J
Johannes Berg 已提交
126
		break;
J
Jiri Benc 已提交
127 128 129 130 131 132 133
	case ALG_TKIP:
		for (i = 0; i < NUM_RX_DATA_QUEUES; i++)
			p += scnprintf(p, sizeof(buf)+buf-p,
				       "%08x %04x\n",
				       key->u.tkip.iv32_rx[i],
				       key->u.tkip.iv16_rx[i]);
		len = p - buf;
J
Johannes Berg 已提交
134
		break;
J
Jiri Benc 已提交
135 136 137 138 139 140 141 142 143
	case ALG_CCMP:
		for (i = 0; i < NUM_RX_DATA_QUEUES; i++) {
			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 已提交
144
		break;
J
Jiri Benc 已提交
145 146 147 148 149 150 151 152 153 154 155 156 157 158
	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;

159
	if (key->conf.alg != ALG_CCMP)
J
Jiri Benc 已提交
160 161 162 163 164 165 166 167 168 169
		return 0;
	len = scnprintf(buf, sizeof(buf), "%u\n", key->u.ccmp.replays);
	return simple_read_from_buffer(userbuf, count, ppos, buf, len);
}
KEY_OPS(replays);

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;
170
	int i, res, bufsize = 2 * key->conf.keylen + 2;
J
Jiri Benc 已提交
171 172 173
	char *buf = kmalloc(bufsize, GFP_KERNEL);
	char *p = buf;

174 175
	for (i = 0; i < key->conf.keylen; i++)
		p += scnprintf(p, bufsize + buf - p, "%02x", key->conf.key[i]);
J
Jiri Benc 已提交
176 177 178 179 180 181 182 183 184 185 186
	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) \
	key->debugfs.name = debugfs_create_file(#name, 0400,\
				key->debugfs.dir, key, &key_##name##_ops);

187 188
void ieee80211_debugfs_key_add(struct ieee80211_key *key)
  {
J
Johannes Berg 已提交
189
	static int keycount;
190 191 192
	char buf[50];
	DECLARE_MAC_BUF(mac);
	struct sta_info *sta;
J
Jiri Benc 已提交
193

194
	if (!key->local->debugfs.keys)
J
Jiri Benc 已提交
195 196
		return;

J
Johannes Berg 已提交
197 198
	sprintf(buf, "%d", keycount);
	keycount++;
J
Jiri Benc 已提交
199
	key->debugfs.dir = debugfs_create_dir(buf,
200
					key->local->debugfs.keys);
J
Jiri Benc 已提交
201 202 203 204

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

205 206 207 208 209 210 211 212 213 214 215
	rcu_read_lock();
	sta = rcu_dereference(key->sta);
	if (sta)
		sprintf(buf, "../../stations/%s", print_mac(mac, sta->addr));
	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 已提交
216
	DEBUGFS_ADD(keylen);
217
	DEBUGFS_ADD(flags);
J
Jiri Benc 已提交
218 219 220 221 222 223 224 225
	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);
	DEBUGFS_ADD(key);
226
	DEBUGFS_ADD(ifindex);
J
Jiri Benc 已提交
227 228 229 230 231 232 233 234 235 236 237
};

#define DEBUGFS_DEL(name) \
	debugfs_remove(key->debugfs.name); key->debugfs.name = NULL;

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

	DEBUGFS_DEL(keylen);
238
	DEBUGFS_DEL(flags);
J
Jiri Benc 已提交
239 240 241 242 243 244 245 246
	DEBUGFS_DEL(keyidx);
	DEBUGFS_DEL(hw_key_idx);
	DEBUGFS_DEL(tx_rx_count);
	DEBUGFS_DEL(algorithm);
	DEBUGFS_DEL(tx_spec);
	DEBUGFS_DEL(rx_spec);
	DEBUGFS_DEL(replays);
	DEBUGFS_DEL(key);
247
	DEBUGFS_DEL(ifindex);
J
Jiri Benc 已提交
248 249 250 251 252 253 254 255 256 257 258 259 260

	debugfs_remove(key->debugfs.stalink);
	key->debugfs.stalink = NULL;
	debugfs_remove(key->debugfs.dir);
	key->debugfs.dir = NULL;
}
void ieee80211_debugfs_key_add_default(struct ieee80211_sub_if_data *sdata)
{
	char buf[50];

	if (!sdata->debugfsdir)
		return;

261
	sprintf(buf, "../keys/%d", sdata->default_key->conf.keyidx);
J
Jiri Benc 已提交
262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279
	sdata->debugfs.default_key =
		debugfs_create_symlink("default_key", sdata->debugfsdir, buf);
}
void ieee80211_debugfs_key_remove_default(struct ieee80211_sub_if_data *sdata)
{
	if (!sdata)
		return;

	debugfs_remove(sdata->debugfs.default_key);
	sdata->debugfs.default_key = NULL;
}

void ieee80211_debugfs_key_sta_del(struct ieee80211_key *key,
				   struct sta_info *sta)
{
	debugfs_remove(key->debugfs.stalink);
	key->debugfs.stalink = NULL;
}