debugfs_key.c 6.8 KB
Newer Older
J
Jiri Benc 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
/*
 * 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"
#include "ieee80211_key.h"
#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
Jiri Benc 已提交
28 29 30 31 32 33 34 35 36 37 38

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

39 40 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_READ_X(name) KEY_CONF_READ(name, 20, "0x%x\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);
KEY_CONF_FILE(flags, X);
J
Jiri Benc 已提交
58 59 60 61 62 63 64 65 66
KEY_FILE(tx_rx_count, D);

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;

67
	switch (key->conf.alg) {
J
Jiri Benc 已提交
68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
	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;

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

121
	switch (key->conf.alg) {
J
Jiri Benc 已提交
122 123
	case ALG_WEP:
		len = scnprintf(buf, sizeof(buf), "\n");
J
Johannes Berg 已提交
124
		break;
J
Jiri Benc 已提交
125 126 127 128 129 130 131
	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 已提交
132
		break;
J
Jiri Benc 已提交
133 134 135 136 137 138 139 140 141
	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 已提交
142
		break;
J
Jiri Benc 已提交
143 144 145 146 147 148 149 150 151 152 153 154 155 156
	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;

157
	if (key->conf.alg != ALG_CCMP)
J
Jiri Benc 已提交
158 159 160 161 162 163 164 165 166 167
		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;
168
	int i, res, bufsize = 2 * key->conf.keylen + 2;
J
Jiri Benc 已提交
169 170 171
	char *buf = kmalloc(bufsize, GFP_KERNEL);
	char *p = buf;

172 173
	for (i = 0; i < key->conf.keylen; i++)
		p += scnprintf(p, bufsize + buf - p, "%02x", key->conf.key[i]);
J
Jiri Benc 已提交
174 175 176 177 178 179 180 181 182 183 184 185 186 187
	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);

void ieee80211_debugfs_key_add(struct ieee80211_local *local,
			       struct ieee80211_key *key)
{
J
Johannes Berg 已提交
188
	static int keycount;
J
Jiri Benc 已提交
189 190 191 192 193
	char buf[20];

	if (!local->debugfs.keys)
		return;

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

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

	DEBUGFS_ADD(keylen);
203
	DEBUGFS_ADD(flags);
J
Jiri Benc 已提交
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222
	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);
};

#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);
223
	DEBUGFS_DEL(flags);
J
Jiri Benc 已提交
224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244
	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);

	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;

245
	sprintf(buf, "../keys/%d", sdata->default_key->conf.keyidx);
J
Jiri Benc 已提交
246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264
	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_link(struct ieee80211_key *key,
				    struct sta_info *sta)
{
	char buf[50];

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

J
Johannes Berg 已提交
265
	sprintf(buf, "../../stations/" MAC_FMT, MAC_ARG(sta->addr));
J
Jiri Benc 已提交
266 267 268 269 270 271 272 273 274 275
	key->debugfs.stalink =
		debugfs_create_symlink("station", key->debugfs.dir, buf);
}

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