1d2912cb1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
21ac61302SLuis R. Rodriguez /*
31ac61302SLuis R. Rodriguez * cfg80211 debugfs
41ac61302SLuis R. Rodriguez *
51ac61302SLuis R. Rodriguez * Copyright 2009 Luis R. Rodriguez <lrodriguez@atheros.com>
61ac61302SLuis R. Rodriguez * Copyright 2007 Johannes Berg <johannes@sipsolutions.net>
71ac61302SLuis R. Rodriguez */
81ac61302SLuis R. Rodriguez
95a0e3ad6STejun Heo #include <linux/slab.h>
101ac61302SLuis R. Rodriguez #include "core.h"
111ac61302SLuis R. Rodriguez #include "debugfs.h"
121ac61302SLuis R. Rodriguez
131ac61302SLuis R. Rodriguez #define DEBUGFS_READONLY_FILE(name, buflen, fmt, value...) \
141ac61302SLuis R. Rodriguez static ssize_t name## _read(struct file *file, char __user *userbuf, \
151ac61302SLuis R. Rodriguez size_t count, loff_t *ppos) \
161ac61302SLuis R. Rodriguez { \
171ac61302SLuis R. Rodriguez struct wiphy *wiphy = file->private_data; \
181ac61302SLuis R. Rodriguez char buf[buflen]; \
191ac61302SLuis R. Rodriguez int res; \
201ac61302SLuis R. Rodriguez \
211ac61302SLuis R. Rodriguez res = scnprintf(buf, buflen, fmt "\n", ##value); \
221ac61302SLuis R. Rodriguez return simple_read_from_buffer(userbuf, count, ppos, buf, res); \
231ac61302SLuis R. Rodriguez } \
241ac61302SLuis R. Rodriguez \
251ac61302SLuis R. Rodriguez static const struct file_operations name## _ops = { \
261ac61302SLuis R. Rodriguez .read = name## _read, \
27234e3405SStephen Boyd .open = simple_open, \
282b18ab36SArnd Bergmann .llseek = generic_file_llseek, \
29b699b71dSPichugin Dmitry }
301ac61302SLuis R. Rodriguez
311ac61302SLuis R. Rodriguez DEBUGFS_READONLY_FILE(rts_threshold, 20, "%d",
32b699b71dSPichugin Dmitry wiphy->rts_threshold);
331ac61302SLuis R. Rodriguez DEBUGFS_READONLY_FILE(fragmentation_threshold, 20, "%d",
341ac61302SLuis R. Rodriguez wiphy->frag_threshold);
351ac61302SLuis R. Rodriguez DEBUGFS_READONLY_FILE(short_retry_limit, 20, "%d",
36b699b71dSPichugin Dmitry wiphy->retry_short);
371ac61302SLuis R. Rodriguez DEBUGFS_READONLY_FILE(long_retry_limit, 20, "%d",
381ac61302SLuis R. Rodriguez wiphy->retry_long);
391ac61302SLuis R. Rodriguez
ht_print_chan(struct ieee80211_channel * chan,char * buf,int buf_size,int offset)4080a3511dSLuis R. Rodriguez static int ht_print_chan(struct ieee80211_channel *chan,
4180a3511dSLuis R. Rodriguez char *buf, int buf_size, int offset)
4280a3511dSLuis R. Rodriguez {
4380a3511dSLuis R. Rodriguez if (WARN_ON(offset > buf_size))
4480a3511dSLuis R. Rodriguez return 0;
4580a3511dSLuis R. Rodriguez
4680a3511dSLuis R. Rodriguez if (chan->flags & IEEE80211_CHAN_DISABLED)
47f364ef99SEliad Peller return scnprintf(buf + offset,
4880a3511dSLuis R. Rodriguez buf_size - offset,
4980a3511dSLuis R. Rodriguez "%d Disabled\n",
5080a3511dSLuis R. Rodriguez chan->center_freq);
5180a3511dSLuis R. Rodriguez
52f364ef99SEliad Peller return scnprintf(buf + offset,
5380a3511dSLuis R. Rodriguez buf_size - offset,
5480a3511dSLuis R. Rodriguez "%d HT40 %c%c\n",
5580a3511dSLuis R. Rodriguez chan->center_freq,
56f364ef99SEliad Peller (chan->flags & IEEE80211_CHAN_NO_HT40MINUS) ?
57f364ef99SEliad Peller ' ' : '-',
58f364ef99SEliad Peller (chan->flags & IEEE80211_CHAN_NO_HT40PLUS) ?
59f364ef99SEliad Peller ' ' : '+');
6080a3511dSLuis R. Rodriguez }
6180a3511dSLuis R. Rodriguez
ht40allow_map_read(struct file * file,char __user * user_buf,size_t count,loff_t * ppos)6280a3511dSLuis R. Rodriguez static ssize_t ht40allow_map_read(struct file *file,
6380a3511dSLuis R. Rodriguez char __user *user_buf,
6480a3511dSLuis R. Rodriguez size_t count, loff_t *ppos)
6580a3511dSLuis R. Rodriguez {
6680a3511dSLuis R. Rodriguez struct wiphy *wiphy = file->private_data;
6780a3511dSLuis R. Rodriguez char *buf;
68*d776763fSDan Carpenter unsigned int offset = 0, buf_size = PAGE_SIZE, i;
6957fbcce3SJohannes Berg enum nl80211_band band;
7080a3511dSLuis R. Rodriguez struct ieee80211_supported_band *sband;
71*d776763fSDan Carpenter ssize_t r;
7280a3511dSLuis R. Rodriguez
7380a3511dSLuis R. Rodriguez buf = kzalloc(buf_size, GFP_KERNEL);
7480a3511dSLuis R. Rodriguez if (!buf)
7580a3511dSLuis R. Rodriguez return -ENOMEM;
7680a3511dSLuis R. Rodriguez
7757fbcce3SJohannes Berg for (band = 0; band < NUM_NL80211_BANDS; band++) {
7880a3511dSLuis R. Rodriguez sband = wiphy->bands[band];
7980a3511dSLuis R. Rodriguez if (!sband)
8080a3511dSLuis R. Rodriguez continue;
8180a3511dSLuis R. Rodriguez for (i = 0; i < sband->n_channels; i++)
8280a3511dSLuis R. Rodriguez offset += ht_print_chan(&sband->channels[i],
8380a3511dSLuis R. Rodriguez buf, buf_size, offset);
8480a3511dSLuis R. Rodriguez }
8580a3511dSLuis R. Rodriguez
8680a3511dSLuis R. Rodriguez r = simple_read_from_buffer(user_buf, count, ppos, buf, offset);
8780a3511dSLuis R. Rodriguez
8880a3511dSLuis R. Rodriguez kfree(buf);
8980a3511dSLuis R. Rodriguez
9080a3511dSLuis R. Rodriguez return r;
9180a3511dSLuis R. Rodriguez }
9280a3511dSLuis R. Rodriguez
9380a3511dSLuis R. Rodriguez static const struct file_operations ht40allow_map_ops = {
9480a3511dSLuis R. Rodriguez .read = ht40allow_map_read,
95234e3405SStephen Boyd .open = simple_open,
966038f373SArnd Bergmann .llseek = default_llseek,
9780a3511dSLuis R. Rodriguez };
9880a3511dSLuis R. Rodriguez
991ac61302SLuis R. Rodriguez #define DEBUGFS_ADD(name) \
100b699b71dSPichugin Dmitry debugfs_create_file(#name, 0444, phyd, &rdev->wiphy, &name## _ops)
1011ac61302SLuis R. Rodriguez
cfg80211_debugfs_rdev_add(struct cfg80211_registered_device * rdev)10279c97e97SJohannes Berg void cfg80211_debugfs_rdev_add(struct cfg80211_registered_device *rdev)
1031ac61302SLuis R. Rodriguez {
10479c97e97SJohannes Berg struct dentry *phyd = rdev->wiphy.debugfsdir;
1051ac61302SLuis R. Rodriguez
1061ac61302SLuis R. Rodriguez DEBUGFS_ADD(rts_threshold);
1071ac61302SLuis R. Rodriguez DEBUGFS_ADD(fragmentation_threshold);
1081ac61302SLuis R. Rodriguez DEBUGFS_ADD(short_retry_limit);
1091ac61302SLuis R. Rodriguez DEBUGFS_ADD(long_retry_limit);
11080a3511dSLuis R. Rodriguez DEBUGFS_ADD(ht40allow_map);
1111ac61302SLuis R. Rodriguez }
112