openwrt/package/d80211/src/ieee80211_sysfs_sta.c
2006-12-30 18:47:07 +00:00

438 lines
11 KiB
C

/*
* Copyright 2003-2005, 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/kobject.h>
#include <linux/sysfs.h>
#include "ieee80211_i.h"
#include "ieee80211_key.h"
#include "sta_info.h"
static ssize_t sta_sysfs_show(struct kobject *, struct attribute *, char *);
static ssize_t key_sysfs_show(struct kobject *, struct attribute *, char *);
static struct sysfs_ops sta_ktype_ops = {
.show = sta_sysfs_show,
};
static struct sysfs_ops key_ktype_ops = {
.show = key_sysfs_show,
};
/* sta attributtes */
#define STA_SHOW(name, field, format_string) \
static ssize_t show_sta_##name(const struct sta_info *sta, char *buf) \
{ \
return sprintf(buf, format_string, sta->field); \
}
#define STA_SHOW_D(name, field) STA_SHOW(name, field, "%d\n")
#define STA_SHOW_U(name, field) STA_SHOW(name, field, "%u\n")
#define STA_SHOW_LU(name, field) STA_SHOW(name, field, "%lu\n")
#define STA_SHOW_S(name, field) STA_SHOW(name, field, "%s\n")
#define STA_SHOW_RATE(name, field) \
static ssize_t show_sta_##name(const struct sta_info *sta, char *buf) \
{ \
struct ieee80211_local *local = sta->dev->ieee80211_ptr; \
return sprintf(buf, "%d\n", \
(sta->field >= 0 && \
sta->field < local->num_curr_rates) ? \
local->curr_rates[sta->field].rate : -1); \
}
#define __STA_ATTR(name) \
static struct sta_attribute sta_attr_##name = \
__ATTR(name, S_IRUGO, show_sta_##name, NULL)
#define STA_ATTR(name, field, format) \
STA_SHOW_##format(name, field) \
__STA_ATTR(name)
STA_ATTR(aid, aid, D);
STA_ATTR(key_idx_compression, key_idx_compression, D);
STA_ATTR(dev, dev->name, S);
STA_ATTR(vlan_id, vlan_id, D);
STA_ATTR(rx_packets, rx_packets, LU);
STA_ATTR(tx_packets, tx_packets, LU);
STA_ATTR(rx_bytes, rx_bytes, LU);
STA_ATTR(tx_bytes, tx_bytes, LU);
STA_ATTR(rx_duplicates, num_duplicates, LU);
STA_ATTR(rx_fragments, rx_fragments, LU);
STA_ATTR(rx_dropped, rx_dropped, LU);
STA_ATTR(tx_fragments, tx_fragments, LU);
STA_ATTR(tx_filtered, tx_filtered_count, LU);
STA_ATTR(txrate, txrate, RATE);
STA_ATTR(last_txrate, last_txrate, RATE);
STA_ATTR(tx_retry_failed, tx_retry_failed, LU);
STA_ATTR(tx_retry_count, tx_retry_count, LU);
STA_ATTR(last_rssi, last_rssi, D);
STA_ATTR(last_signal, last_signal, D);
STA_ATTR(last_noise, last_noise, D);
STA_ATTR(channel_use, channel_use, D);
STA_ATTR(wep_weak_iv_count, wep_weak_iv_count, D);
static ssize_t show_sta_flags(const struct sta_info *sta, char *buf)
{
return sprintf(buf, "%s%s%s%s%s%s%s%s%s",
sta->flags & WLAN_STA_AUTH ? "AUTH\n" : "",
sta->flags & WLAN_STA_ASSOC ? "ASSOC\n" : "",
sta->flags & WLAN_STA_PS ? "PS\n" : "",
sta->flags & WLAN_STA_TIM ? "TIM\n" : "",
sta->flags & WLAN_STA_PERM ? "PERM\n" : "",
sta->flags & WLAN_STA_AUTHORIZED ? "AUTHORIZED\n" : "",
sta->flags & WLAN_STA_SHORT_PREAMBLE ?
"SHORT PREAMBLE\n" : "",
sta->flags & WLAN_STA_WME ? "WME\n" : "",
sta->flags & WLAN_STA_WDS ? "WDS\n" : "");
}
__STA_ATTR(flags);
static ssize_t show_sta_num_ps_buf_frames(const struct sta_info *sta, char *buf)
{
return sprintf(buf, "%u\n", skb_queue_len(&sta->ps_tx_buf));
}
__STA_ATTR(num_ps_buf_frames);
static ssize_t show_sta_last_ack_rssi(const struct sta_info *sta, char *buf)
{
return sprintf(buf, "%d %d %d\n", sta->last_ack_rssi[0],
sta->last_ack_rssi[1], sta->last_ack_rssi[2]);
}
__STA_ATTR(last_ack_rssi);
static ssize_t show_sta_last_ack_ms(const struct sta_info *sta, char *buf)
{
return sprintf(buf, "%d\n", sta->last_ack ?
jiffies_to_msecs(jiffies - sta->last_ack) : -1);
}
__STA_ATTR(last_ack_ms);
static ssize_t show_sta_inactive_ms(const struct sta_info *sta, char *buf)
{
return sprintf(buf, "%d\n", jiffies_to_msecs(jiffies - sta->last_rx));
}
__STA_ATTR(inactive_ms);
static ssize_t show_sta_last_seq_ctrl(const struct sta_info *sta, char *buf)
{
int i;
char *p = buf;
for (i = 0; i < NUM_RX_DATA_QUEUES; i++)
p += sprintf(p, "%x ", sta->last_seq_ctrl[i]);
p += sprintf(p, "\n");
return (p - buf);
}
__STA_ATTR(last_seq_ctrl);
#ifdef CONFIG_D80211_DEBUG_COUNTERS
static ssize_t show_sta_wme_rx_queue(const struct sta_info *sta, char *buf)
{
int i;
char *p = buf;
for (i = 0; i < NUM_RX_DATA_QUEUES; i++)
p += sprintf(p, "%u ", sta->wme_rx_queue[i]);
p += sprintf(p, "\n");
return (p - buf);
}
__STA_ATTR(wme_rx_queue);
static ssize_t show_sta_wme_tx_queue(const struct sta_info *sta, char *buf)
{
int i;
char *p = buf;
for (i = 0; i < NUM_RX_DATA_QUEUES; i++)
p += sprintf(p, "%u ", sta->wme_tx_queue[i]);
p += sprintf(p, "\n");
return (p - buf);
}
__STA_ATTR(wme_tx_queue);
#endif
static struct attribute *sta_ktype_attrs[] = {
&sta_attr_aid.attr,
&sta_attr_key_idx_compression.attr,
&sta_attr_dev.attr,
&sta_attr_vlan_id.attr,
&sta_attr_rx_packets.attr,
&sta_attr_tx_packets.attr,
&sta_attr_rx_bytes.attr,
&sta_attr_tx_bytes.attr,
&sta_attr_rx_duplicates.attr,
&sta_attr_rx_fragments.attr,
&sta_attr_rx_dropped.attr,
&sta_attr_tx_fragments.attr,
&sta_attr_tx_filtered.attr,
&sta_attr_txrate.attr,
&sta_attr_last_txrate.attr,
&sta_attr_tx_retry_failed.attr,
&sta_attr_tx_retry_count.attr,
&sta_attr_last_rssi.attr,
&sta_attr_last_signal.attr,
&sta_attr_last_noise.attr,
&sta_attr_channel_use.attr,
&sta_attr_wep_weak_iv_count.attr,
&sta_attr_flags.attr,
&sta_attr_num_ps_buf_frames.attr,
&sta_attr_last_ack_rssi.attr,
&sta_attr_last_ack_ms.attr,
&sta_attr_inactive_ms.attr,
&sta_attr_last_seq_ctrl.attr,
#ifdef CONFIG_D80211_DEBUG_COUNTERS
&sta_attr_wme_rx_queue.attr,
&sta_attr_wme_tx_queue.attr,
#endif
NULL
};
/* keys attributtes */
struct key_attribute {
struct attribute attr;
ssize_t (*show)(const struct ieee80211_key *, char *buf);
ssize_t (*store)(struct ieee80211_key *, const char *buf,
size_t count);
};
#define KEY_SHOW(name, field, format_string) \
static ssize_t show_key_##name(const struct ieee80211_key *key, char *buf)\
{ \
return sprintf(buf, format_string, key->field); \
}
#define KEY_SHOW_D(name, field) KEY_SHOW(name, field, "%d\n")
#define __KEY_ATTR(name) \
static struct key_attribute key_attr_##name = \
__ATTR(name, S_IRUSR, show_key_##name, NULL)
#define KEY_ATTR(name, field, format) \
KEY_SHOW_##format(name, field) \
__KEY_ATTR(name)
KEY_ATTR(length, keylen, D);
KEY_ATTR(sw_encrypt, force_sw_encrypt, D);
KEY_ATTR(index, keyidx, D);
KEY_ATTR(hw_index, hw_key_idx, D);
KEY_ATTR(tx_rx_count, tx_rx_count, D);
static ssize_t show_key_algorithm(const struct ieee80211_key *key, char *buf)
{
char *alg;
switch (key->alg) {
case ALG_WEP:
alg = "WEP";
break;
case ALG_TKIP:
alg = "TKIP";
break;
case ALG_CCMP:
alg = "CCMP";
break;
default:
return 0;
}
return sprintf(buf, "%s\n", alg);
}
__KEY_ATTR(algorithm);
static ssize_t show_key_tx_spec(const struct ieee80211_key *key, char *buf)
{
const u8 *tpn;
switch (key->alg) {
case ALG_WEP:
return sprintf(buf, "\n");
case ALG_TKIP:
return sprintf(buf, "%08x %04x\n", key->u.tkip.iv32,
key->u.tkip.iv16);
case ALG_CCMP:
tpn = key->u.ccmp.tx_pn;
return sprintf(buf, "%02x%02x%02x%02x%02x%02x\n", tpn[0],
tpn[1], tpn[2], tpn[3], tpn[4], tpn[5]);
default:
return 0;
}
}
__KEY_ATTR(tx_spec);
static ssize_t show_key_rx_spec(const struct ieee80211_key *key, char *buf)
{
int i;
const u8 *rpn;
char *p = buf;
switch (key->alg) {
case ALG_WEP:
return sprintf(buf, "\n");
case ALG_TKIP:
for (i = 0; i < NUM_RX_DATA_QUEUES; i++)
p += sprintf(p, "%08x %04x\n",
key->u.tkip.iv32_rx[i],
key->u.tkip.iv16_rx[i]);
return (p - buf);
case ALG_CCMP:
for (i = 0; i < NUM_RX_DATA_QUEUES; i++) {
rpn = key->u.ccmp.rx_pn[i];
p += sprintf(p, "%02x%02x%02x%02x%02x%02x\n", rpn[0],
rpn[1], rpn[2], rpn[3], rpn[4], rpn[5]);
}
return (p - buf);
default:
return 0;
}
}
__KEY_ATTR(rx_spec);
static ssize_t show_key_replays(const struct ieee80211_key *key, char *buf)
{
if (key->alg != ALG_CCMP)
return 0;
return sprintf(buf, "%u\n", key->u.ccmp.replays);
}
__KEY_ATTR(replays);
static ssize_t show_key_key(const struct ieee80211_key *key, char *buf)
{
int i;
char *p = buf;
for (i = 0; i < key->keylen; i++)
p += sprintf(p, "%02x", key->key[i]);
p += sprintf(p, "\n");
return (p - buf);
}
__KEY_ATTR(key);
static struct attribute *key_ktype_attrs[] = {
&key_attr_length.attr,
&key_attr_sw_encrypt.attr,
&key_attr_index.attr,
&key_attr_hw_index.attr,
&key_attr_tx_rx_count.attr,
&key_attr_algorithm.attr,
&key_attr_tx_spec.attr,
&key_attr_rx_spec.attr,
&key_attr_replays.attr,
&key_attr_key.attr,
NULL
};
/* structures and functions */
static struct kobj_type sta_ktype = {
.release = sta_info_release,
.sysfs_ops = &sta_ktype_ops,
.default_attrs = sta_ktype_attrs,
};
static struct kobj_type key_ktype = {
.release = ieee80211_key_release,
.sysfs_ops = &key_ktype_ops,
.default_attrs = key_ktype_attrs,
};
static ssize_t sta_sysfs_show(struct kobject *kobj, struct attribute *attr,
char *buf)
{
struct sta_attribute *sta_attr;
struct sta_info *sta;
sta_attr = container_of(attr, struct sta_attribute, attr);
sta = container_of(kobj, struct sta_info, kobj);
return sta_attr->show(sta, buf);
}
static ssize_t key_sysfs_show(struct kobject *kobj, struct attribute *attr,
char *buf)
{
struct key_attribute *key_attr;
struct ieee80211_key *key;
key_attr = container_of(attr, struct key_attribute, attr);
key = container_of(kobj, struct ieee80211_key, kobj);
return key_attr->show(key, buf);
}
int ieee80211_sta_kset_sysfs_register(struct ieee80211_local *local)
{
int res;
res = kobject_set_name(&local->sta_kset.kobj, "sta");
if (res)
return res;
local->sta_kset.kobj.parent = &local->class_dev.kobj;
local->sta_kset.ktype = &sta_ktype;
return kset_register(&local->sta_kset);
}
void ieee80211_sta_kset_sysfs_unregister(struct ieee80211_local *local)
{
kset_unregister(&local->sta_kset);
}
int ieee80211_key_kset_sysfs_register(struct ieee80211_sub_if_data *sdata)
{
int res;
res = kobject_set_name(&sdata->key_kset.kobj, "keys");
if (res)
return res;
sdata->key_kset.kobj.parent = &sdata->dev->class_dev.kobj;
sdata->key_kset.ktype = &key_ktype;
return kset_register(&sdata->key_kset);
}
void ieee80211_key_kset_sysfs_unregister(struct ieee80211_sub_if_data *sdata)
{
kset_unregister(&sdata->key_kset);
}
int ieee80211_sta_sysfs_add(struct sta_info *sta)
{
return kobject_add(&sta->kobj);
}
void ieee80211_sta_sysfs_remove(struct sta_info *sta)
{
kobject_del(&sta->kobj);
}
void ieee80211_key_sysfs_set_kset(struct ieee80211_key *key, struct kset *kset)
{
key->kobj.kset = kset;
if (!kset)
key->kobj.ktype = &key_ktype;
}
int ieee80211_key_sysfs_add(struct ieee80211_key *key)
{
return kobject_add(&key->kobj);
}
void ieee80211_key_sysfs_remove(struct ieee80211_key *key)
{
if (key)
kobject_del(&key->kobj);
}
int ieee80211_key_sysfs_add_default(struct ieee80211_sub_if_data *sdata)
{
return sysfs_create_link(&sdata->key_kset.kobj,
&sdata->default_key->kobj, "default");
}
void ieee80211_key_sysfs_remove_default(struct ieee80211_sub_if_data *sdata)
{
sysfs_remove_link(&sdata->key_kset.kobj, "default");
}