162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci Copyright (C) 2004 - 2009 Ivo van Doorn <IvDoorn@gmail.com> 462306a36Sopenharmony_ci <http://rt2x00.serialmonkey.com> 562306a36Sopenharmony_ci 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci/* 962306a36Sopenharmony_ci Module: rt2x00lib 1062306a36Sopenharmony_ci Abstract: rt2x00 debugfs specific routines. 1162306a36Sopenharmony_ci */ 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#include <linux/debugfs.h> 1462306a36Sopenharmony_ci#include <linux/kernel.h> 1562306a36Sopenharmony_ci#include <linux/module.h> 1662306a36Sopenharmony_ci#include <linux/poll.h> 1762306a36Sopenharmony_ci#include <linux/sched.h> 1862306a36Sopenharmony_ci#include <linux/slab.h> 1962306a36Sopenharmony_ci#include <linux/uaccess.h> 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#include "rt2x00.h" 2262306a36Sopenharmony_ci#include "rt2x00lib.h" 2362306a36Sopenharmony_ci#include "rt2x00dump.h" 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci#define MAX_LINE_LENGTH 64 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_cistruct rt2x00debug_crypto { 2862306a36Sopenharmony_ci unsigned long success; 2962306a36Sopenharmony_ci unsigned long icv_error; 3062306a36Sopenharmony_ci unsigned long mic_error; 3162306a36Sopenharmony_ci unsigned long key_error; 3262306a36Sopenharmony_ci}; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cistruct rt2x00debug_intf { 3562306a36Sopenharmony_ci /* 3662306a36Sopenharmony_ci * Pointer to driver structure where 3762306a36Sopenharmony_ci * this debugfs entry belongs to. 3862306a36Sopenharmony_ci */ 3962306a36Sopenharmony_ci struct rt2x00_dev *rt2x00dev; 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_ci /* 4262306a36Sopenharmony_ci * Reference to the rt2x00debug structure 4362306a36Sopenharmony_ci * which can be used to communicate with 4462306a36Sopenharmony_ci * the registers. 4562306a36Sopenharmony_ci */ 4662306a36Sopenharmony_ci const struct rt2x00debug *debug; 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci /* 4962306a36Sopenharmony_ci * Debugfs entries for: 5062306a36Sopenharmony_ci * - driver folder 5162306a36Sopenharmony_ci * - driver file 5262306a36Sopenharmony_ci * - chipset file 5362306a36Sopenharmony_ci * - device state flags file 5462306a36Sopenharmony_ci * - device capability flags file 5562306a36Sopenharmony_ci * - hardware restart file 5662306a36Sopenharmony_ci * - register folder 5762306a36Sopenharmony_ci * - csr offset/value files 5862306a36Sopenharmony_ci * - eeprom offset/value files 5962306a36Sopenharmony_ci * - bbp offset/value files 6062306a36Sopenharmony_ci * - rf offset/value files 6162306a36Sopenharmony_ci * - rfcsr offset/value files 6262306a36Sopenharmony_ci * - queue folder 6362306a36Sopenharmony_ci * - frame dump file 6462306a36Sopenharmony_ci * - queue stats file 6562306a36Sopenharmony_ci * - crypto stats file 6662306a36Sopenharmony_ci */ 6762306a36Sopenharmony_ci struct dentry *driver_folder; 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci /* 7062306a36Sopenharmony_ci * The frame dump file only allows a single reader, 7162306a36Sopenharmony_ci * so we need to store the current state here. 7262306a36Sopenharmony_ci */ 7362306a36Sopenharmony_ci unsigned long frame_dump_flags; 7462306a36Sopenharmony_ci#define FRAME_DUMP_FILE_OPEN 1 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci /* 7762306a36Sopenharmony_ci * We queue each frame before dumping it to the user, 7862306a36Sopenharmony_ci * per read command we will pass a single skb structure 7962306a36Sopenharmony_ci * so we should be prepared to queue multiple sk buffers 8062306a36Sopenharmony_ci * before sending it to userspace. 8162306a36Sopenharmony_ci */ 8262306a36Sopenharmony_ci struct sk_buff_head frame_dump_skbqueue; 8362306a36Sopenharmony_ci wait_queue_head_t frame_dump_waitqueue; 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci /* 8662306a36Sopenharmony_ci * HW crypto statistics. 8762306a36Sopenharmony_ci * All statistics are stored separately per cipher type. 8862306a36Sopenharmony_ci */ 8962306a36Sopenharmony_ci struct rt2x00debug_crypto crypto_stats[CIPHER_MAX]; 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_ci /* 9262306a36Sopenharmony_ci * Driver and chipset files will use a data buffer 9362306a36Sopenharmony_ci * that has been created in advance. This will simplify 9462306a36Sopenharmony_ci * the code since we can use the debugfs functions. 9562306a36Sopenharmony_ci */ 9662306a36Sopenharmony_ci struct debugfs_blob_wrapper driver_blob; 9762306a36Sopenharmony_ci struct debugfs_blob_wrapper chipset_blob; 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci /* 10062306a36Sopenharmony_ci * Requested offset for each register type. 10162306a36Sopenharmony_ci */ 10262306a36Sopenharmony_ci unsigned int offset_csr; 10362306a36Sopenharmony_ci unsigned int offset_eeprom; 10462306a36Sopenharmony_ci unsigned int offset_bbp; 10562306a36Sopenharmony_ci unsigned int offset_rf; 10662306a36Sopenharmony_ci unsigned int offset_rfcsr; 10762306a36Sopenharmony_ci}; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_civoid rt2x00debug_update_crypto(struct rt2x00_dev *rt2x00dev, 11062306a36Sopenharmony_ci struct rxdone_entry_desc *rxdesc) 11162306a36Sopenharmony_ci{ 11262306a36Sopenharmony_ci struct rt2x00debug_intf *intf = rt2x00dev->debugfs_intf; 11362306a36Sopenharmony_ci enum cipher cipher = rxdesc->cipher; 11462306a36Sopenharmony_ci enum rx_crypto status = rxdesc->cipher_status; 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci if (cipher == CIPHER_TKIP_NO_MIC) 11762306a36Sopenharmony_ci cipher = CIPHER_TKIP; 11862306a36Sopenharmony_ci if (cipher == CIPHER_NONE || cipher >= CIPHER_MAX) 11962306a36Sopenharmony_ci return; 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci /* Remove CIPHER_NONE index */ 12262306a36Sopenharmony_ci cipher--; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci intf->crypto_stats[cipher].success += (status == RX_CRYPTO_SUCCESS); 12562306a36Sopenharmony_ci intf->crypto_stats[cipher].icv_error += (status == RX_CRYPTO_FAIL_ICV); 12662306a36Sopenharmony_ci intf->crypto_stats[cipher].mic_error += (status == RX_CRYPTO_FAIL_MIC); 12762306a36Sopenharmony_ci intf->crypto_stats[cipher].key_error += (status == RX_CRYPTO_FAIL_KEY); 12862306a36Sopenharmony_ci} 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_civoid rt2x00debug_dump_frame(struct rt2x00_dev *rt2x00dev, 13162306a36Sopenharmony_ci enum rt2x00_dump_type type, struct queue_entry *entry) 13262306a36Sopenharmony_ci{ 13362306a36Sopenharmony_ci struct rt2x00debug_intf *intf = rt2x00dev->debugfs_intf; 13462306a36Sopenharmony_ci struct sk_buff *skb = entry->skb; 13562306a36Sopenharmony_ci struct skb_frame_desc *skbdesc = get_skb_frame_desc(skb); 13662306a36Sopenharmony_ci struct sk_buff *skbcopy; 13762306a36Sopenharmony_ci struct rt2x00dump_hdr *dump_hdr; 13862306a36Sopenharmony_ci struct timespec64 timestamp; 13962306a36Sopenharmony_ci u32 data_len; 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_ci if (likely(!test_bit(FRAME_DUMP_FILE_OPEN, &intf->frame_dump_flags))) 14262306a36Sopenharmony_ci return; 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ci ktime_get_ts64(×tamp); 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci if (skb_queue_len(&intf->frame_dump_skbqueue) > 20) { 14762306a36Sopenharmony_ci rt2x00_dbg(rt2x00dev, "txrx dump queue length exceeded\n"); 14862306a36Sopenharmony_ci return; 14962306a36Sopenharmony_ci } 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci data_len = skb->len; 15262306a36Sopenharmony_ci if (skbdesc->flags & SKBDESC_DESC_IN_SKB) 15362306a36Sopenharmony_ci data_len -= skbdesc->desc_len; 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci skbcopy = alloc_skb(sizeof(*dump_hdr) + skbdesc->desc_len + data_len, 15662306a36Sopenharmony_ci GFP_ATOMIC); 15762306a36Sopenharmony_ci if (!skbcopy) { 15862306a36Sopenharmony_ci rt2x00_dbg(rt2x00dev, "Failed to copy skb for dump\n"); 15962306a36Sopenharmony_ci return; 16062306a36Sopenharmony_ci } 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci dump_hdr = skb_put(skbcopy, sizeof(*dump_hdr)); 16362306a36Sopenharmony_ci dump_hdr->version = cpu_to_le32(DUMP_HEADER_VERSION); 16462306a36Sopenharmony_ci dump_hdr->header_length = cpu_to_le32(sizeof(*dump_hdr)); 16562306a36Sopenharmony_ci dump_hdr->desc_length = cpu_to_le32(skbdesc->desc_len); 16662306a36Sopenharmony_ci dump_hdr->data_length = cpu_to_le32(data_len); 16762306a36Sopenharmony_ci dump_hdr->chip_rt = cpu_to_le16(rt2x00dev->chip.rt); 16862306a36Sopenharmony_ci dump_hdr->chip_rf = cpu_to_le16(rt2x00dev->chip.rf); 16962306a36Sopenharmony_ci dump_hdr->chip_rev = cpu_to_le16(rt2x00dev->chip.rev); 17062306a36Sopenharmony_ci dump_hdr->type = cpu_to_le16(type); 17162306a36Sopenharmony_ci dump_hdr->queue_index = entry->queue->qid; 17262306a36Sopenharmony_ci dump_hdr->entry_index = entry->entry_idx; 17362306a36Sopenharmony_ci dump_hdr->timestamp_sec = cpu_to_le32(timestamp.tv_sec); 17462306a36Sopenharmony_ci dump_hdr->timestamp_usec = cpu_to_le32(timestamp.tv_nsec / 17562306a36Sopenharmony_ci NSEC_PER_USEC); 17662306a36Sopenharmony_ci 17762306a36Sopenharmony_ci if (!(skbdesc->flags & SKBDESC_DESC_IN_SKB)) 17862306a36Sopenharmony_ci skb_put_data(skbcopy, skbdesc->desc, skbdesc->desc_len); 17962306a36Sopenharmony_ci skb_put_data(skbcopy, skb->data, skb->len); 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_ci skb_queue_tail(&intf->frame_dump_skbqueue, skbcopy); 18262306a36Sopenharmony_ci wake_up_interruptible(&intf->frame_dump_waitqueue); 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci /* 18562306a36Sopenharmony_ci * Verify that the file has not been closed while we were working. 18662306a36Sopenharmony_ci */ 18762306a36Sopenharmony_ci if (!test_bit(FRAME_DUMP_FILE_OPEN, &intf->frame_dump_flags)) 18862306a36Sopenharmony_ci skb_queue_purge(&intf->frame_dump_skbqueue); 18962306a36Sopenharmony_ci} 19062306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(rt2x00debug_dump_frame); 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_cistatic int rt2x00debug_file_open(struct inode *inode, struct file *file) 19362306a36Sopenharmony_ci{ 19462306a36Sopenharmony_ci struct rt2x00debug_intf *intf = inode->i_private; 19562306a36Sopenharmony_ci 19662306a36Sopenharmony_ci file->private_data = inode->i_private; 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci if (!try_module_get(intf->debug->owner)) 19962306a36Sopenharmony_ci return -EBUSY; 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_ci return 0; 20262306a36Sopenharmony_ci} 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_cistatic int rt2x00debug_file_release(struct inode *inode, struct file *file) 20562306a36Sopenharmony_ci{ 20662306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci module_put(intf->debug->owner); 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_ci return 0; 21162306a36Sopenharmony_ci} 21262306a36Sopenharmony_ci 21362306a36Sopenharmony_cistatic int rt2x00debug_open_queue_dump(struct inode *inode, struct file *file) 21462306a36Sopenharmony_ci{ 21562306a36Sopenharmony_ci struct rt2x00debug_intf *intf = inode->i_private; 21662306a36Sopenharmony_ci int retval; 21762306a36Sopenharmony_ci 21862306a36Sopenharmony_ci retval = rt2x00debug_file_open(inode, file); 21962306a36Sopenharmony_ci if (retval) 22062306a36Sopenharmony_ci return retval; 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_ci if (test_and_set_bit(FRAME_DUMP_FILE_OPEN, &intf->frame_dump_flags)) { 22362306a36Sopenharmony_ci rt2x00debug_file_release(inode, file); 22462306a36Sopenharmony_ci return -EBUSY; 22562306a36Sopenharmony_ci } 22662306a36Sopenharmony_ci 22762306a36Sopenharmony_ci return 0; 22862306a36Sopenharmony_ci} 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_cistatic int rt2x00debug_release_queue_dump(struct inode *inode, struct file *file) 23162306a36Sopenharmony_ci{ 23262306a36Sopenharmony_ci struct rt2x00debug_intf *intf = inode->i_private; 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_ci skb_queue_purge(&intf->frame_dump_skbqueue); 23562306a36Sopenharmony_ci 23662306a36Sopenharmony_ci clear_bit(FRAME_DUMP_FILE_OPEN, &intf->frame_dump_flags); 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_ci return rt2x00debug_file_release(inode, file); 23962306a36Sopenharmony_ci} 24062306a36Sopenharmony_ci 24162306a36Sopenharmony_cistatic ssize_t rt2x00debug_read_queue_dump(struct file *file, 24262306a36Sopenharmony_ci char __user *buf, 24362306a36Sopenharmony_ci size_t length, 24462306a36Sopenharmony_ci loff_t *offset) 24562306a36Sopenharmony_ci{ 24662306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 24762306a36Sopenharmony_ci struct sk_buff *skb; 24862306a36Sopenharmony_ci size_t status; 24962306a36Sopenharmony_ci int retval; 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci if (file->f_flags & O_NONBLOCK) 25262306a36Sopenharmony_ci return -EAGAIN; 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci retval = 25562306a36Sopenharmony_ci wait_event_interruptible(intf->frame_dump_waitqueue, 25662306a36Sopenharmony_ci (skb = 25762306a36Sopenharmony_ci skb_dequeue(&intf->frame_dump_skbqueue))); 25862306a36Sopenharmony_ci if (retval) 25962306a36Sopenharmony_ci return retval; 26062306a36Sopenharmony_ci 26162306a36Sopenharmony_ci status = min_t(size_t, skb->len, length); 26262306a36Sopenharmony_ci if (copy_to_user(buf, skb->data, status)) { 26362306a36Sopenharmony_ci status = -EFAULT; 26462306a36Sopenharmony_ci goto exit; 26562306a36Sopenharmony_ci } 26662306a36Sopenharmony_ci 26762306a36Sopenharmony_ci *offset += status; 26862306a36Sopenharmony_ci 26962306a36Sopenharmony_ciexit: 27062306a36Sopenharmony_ci kfree_skb(skb); 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_ci return status; 27362306a36Sopenharmony_ci} 27462306a36Sopenharmony_ci 27562306a36Sopenharmony_cistatic __poll_t rt2x00debug_poll_queue_dump(struct file *file, 27662306a36Sopenharmony_ci poll_table *wait) 27762306a36Sopenharmony_ci{ 27862306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 27962306a36Sopenharmony_ci 28062306a36Sopenharmony_ci poll_wait(file, &intf->frame_dump_waitqueue, wait); 28162306a36Sopenharmony_ci 28262306a36Sopenharmony_ci if (!skb_queue_empty(&intf->frame_dump_skbqueue)) 28362306a36Sopenharmony_ci return EPOLLOUT | EPOLLWRNORM; 28462306a36Sopenharmony_ci 28562306a36Sopenharmony_ci return 0; 28662306a36Sopenharmony_ci} 28762306a36Sopenharmony_ci 28862306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_fop_queue_dump = { 28962306a36Sopenharmony_ci .owner = THIS_MODULE, 29062306a36Sopenharmony_ci .read = rt2x00debug_read_queue_dump, 29162306a36Sopenharmony_ci .poll = rt2x00debug_poll_queue_dump, 29262306a36Sopenharmony_ci .open = rt2x00debug_open_queue_dump, 29362306a36Sopenharmony_ci .release = rt2x00debug_release_queue_dump, 29462306a36Sopenharmony_ci .llseek = default_llseek, 29562306a36Sopenharmony_ci}; 29662306a36Sopenharmony_ci 29762306a36Sopenharmony_cistatic ssize_t rt2x00debug_read_queue_stats(struct file *file, 29862306a36Sopenharmony_ci char __user *buf, 29962306a36Sopenharmony_ci size_t length, 30062306a36Sopenharmony_ci loff_t *offset) 30162306a36Sopenharmony_ci{ 30262306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 30362306a36Sopenharmony_ci struct data_queue *queue; 30462306a36Sopenharmony_ci unsigned long irqflags; 30562306a36Sopenharmony_ci unsigned int lines = 1 + intf->rt2x00dev->data_queues; 30662306a36Sopenharmony_ci size_t size; 30762306a36Sopenharmony_ci char *data; 30862306a36Sopenharmony_ci char *temp; 30962306a36Sopenharmony_ci 31062306a36Sopenharmony_ci if (*offset) 31162306a36Sopenharmony_ci return 0; 31262306a36Sopenharmony_ci 31362306a36Sopenharmony_ci data = kcalloc(lines, MAX_LINE_LENGTH, GFP_KERNEL); 31462306a36Sopenharmony_ci if (!data) 31562306a36Sopenharmony_ci return -ENOMEM; 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci temp = data + 31862306a36Sopenharmony_ci sprintf(data, "qid\tflags\t\tcount\tlimit\tlength\tindex\tdma done\tdone\n"); 31962306a36Sopenharmony_ci 32062306a36Sopenharmony_ci queue_for_each(intf->rt2x00dev, queue) { 32162306a36Sopenharmony_ci spin_lock_irqsave(&queue->index_lock, irqflags); 32262306a36Sopenharmony_ci 32362306a36Sopenharmony_ci temp += sprintf(temp, "%d\t0x%.8x\t%d\t%d\t%d\t%d\t%d\t\t%d\n", 32462306a36Sopenharmony_ci queue->qid, (unsigned int)queue->flags, 32562306a36Sopenharmony_ci queue->count, queue->limit, queue->length, 32662306a36Sopenharmony_ci queue->index[Q_INDEX], 32762306a36Sopenharmony_ci queue->index[Q_INDEX_DMA_DONE], 32862306a36Sopenharmony_ci queue->index[Q_INDEX_DONE]); 32962306a36Sopenharmony_ci 33062306a36Sopenharmony_ci spin_unlock_irqrestore(&queue->index_lock, irqflags); 33162306a36Sopenharmony_ci } 33262306a36Sopenharmony_ci 33362306a36Sopenharmony_ci size = strlen(data); 33462306a36Sopenharmony_ci size = min(size, length); 33562306a36Sopenharmony_ci 33662306a36Sopenharmony_ci if (copy_to_user(buf, data, size)) { 33762306a36Sopenharmony_ci kfree(data); 33862306a36Sopenharmony_ci return -EFAULT; 33962306a36Sopenharmony_ci } 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ci kfree(data); 34262306a36Sopenharmony_ci 34362306a36Sopenharmony_ci *offset += size; 34462306a36Sopenharmony_ci return size; 34562306a36Sopenharmony_ci} 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_fop_queue_stats = { 34862306a36Sopenharmony_ci .owner = THIS_MODULE, 34962306a36Sopenharmony_ci .read = rt2x00debug_read_queue_stats, 35062306a36Sopenharmony_ci .open = rt2x00debug_file_open, 35162306a36Sopenharmony_ci .release = rt2x00debug_file_release, 35262306a36Sopenharmony_ci .llseek = default_llseek, 35362306a36Sopenharmony_ci}; 35462306a36Sopenharmony_ci 35562306a36Sopenharmony_ci#ifdef CONFIG_RT2X00_LIB_CRYPTO 35662306a36Sopenharmony_cistatic ssize_t rt2x00debug_read_crypto_stats(struct file *file, 35762306a36Sopenharmony_ci char __user *buf, 35862306a36Sopenharmony_ci size_t length, 35962306a36Sopenharmony_ci loff_t *offset) 36062306a36Sopenharmony_ci{ 36162306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 36262306a36Sopenharmony_ci static const char * const name[] = { "WEP64", "WEP128", "TKIP", "AES" }; 36362306a36Sopenharmony_ci char *data; 36462306a36Sopenharmony_ci char *temp; 36562306a36Sopenharmony_ci size_t size; 36662306a36Sopenharmony_ci unsigned int i; 36762306a36Sopenharmony_ci 36862306a36Sopenharmony_ci if (*offset) 36962306a36Sopenharmony_ci return 0; 37062306a36Sopenharmony_ci 37162306a36Sopenharmony_ci data = kcalloc(1 + CIPHER_MAX, MAX_LINE_LENGTH, GFP_KERNEL); 37262306a36Sopenharmony_ci if (!data) 37362306a36Sopenharmony_ci return -ENOMEM; 37462306a36Sopenharmony_ci 37562306a36Sopenharmony_ci temp = data; 37662306a36Sopenharmony_ci temp += sprintf(data, "cipher\tsuccess\ticv err\tmic err\tkey err\n"); 37762306a36Sopenharmony_ci 37862306a36Sopenharmony_ci for (i = 0; i < CIPHER_MAX; i++) { 37962306a36Sopenharmony_ci temp += sprintf(temp, "%s\t%lu\t%lu\t%lu\t%lu\n", name[i], 38062306a36Sopenharmony_ci intf->crypto_stats[i].success, 38162306a36Sopenharmony_ci intf->crypto_stats[i].icv_error, 38262306a36Sopenharmony_ci intf->crypto_stats[i].mic_error, 38362306a36Sopenharmony_ci intf->crypto_stats[i].key_error); 38462306a36Sopenharmony_ci } 38562306a36Sopenharmony_ci 38662306a36Sopenharmony_ci size = strlen(data); 38762306a36Sopenharmony_ci size = min(size, length); 38862306a36Sopenharmony_ci 38962306a36Sopenharmony_ci if (copy_to_user(buf, data, size)) { 39062306a36Sopenharmony_ci kfree(data); 39162306a36Sopenharmony_ci return -EFAULT; 39262306a36Sopenharmony_ci } 39362306a36Sopenharmony_ci 39462306a36Sopenharmony_ci kfree(data); 39562306a36Sopenharmony_ci 39662306a36Sopenharmony_ci *offset += size; 39762306a36Sopenharmony_ci return size; 39862306a36Sopenharmony_ci} 39962306a36Sopenharmony_ci 40062306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_fop_crypto_stats = { 40162306a36Sopenharmony_ci .owner = THIS_MODULE, 40262306a36Sopenharmony_ci .read = rt2x00debug_read_crypto_stats, 40362306a36Sopenharmony_ci .open = rt2x00debug_file_open, 40462306a36Sopenharmony_ci .release = rt2x00debug_file_release, 40562306a36Sopenharmony_ci .llseek = default_llseek, 40662306a36Sopenharmony_ci}; 40762306a36Sopenharmony_ci#endif 40862306a36Sopenharmony_ci 40962306a36Sopenharmony_ci#define RT2X00DEBUGFS_OPS_READ(__name, __format, __type) \ 41062306a36Sopenharmony_cistatic ssize_t rt2x00debug_read_##__name(struct file *file, \ 41162306a36Sopenharmony_ci char __user *buf, \ 41262306a36Sopenharmony_ci size_t length, \ 41362306a36Sopenharmony_ci loff_t *offset) \ 41462306a36Sopenharmony_ci{ \ 41562306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; \ 41662306a36Sopenharmony_ci const struct rt2x00debug *debug = intf->debug; \ 41762306a36Sopenharmony_ci char line[16]; \ 41862306a36Sopenharmony_ci size_t size; \ 41962306a36Sopenharmony_ci unsigned int index = intf->offset_##__name; \ 42062306a36Sopenharmony_ci __type value; \ 42162306a36Sopenharmony_ci \ 42262306a36Sopenharmony_ci if (*offset) \ 42362306a36Sopenharmony_ci return 0; \ 42462306a36Sopenharmony_ci \ 42562306a36Sopenharmony_ci if (index >= debug->__name.word_count) \ 42662306a36Sopenharmony_ci return -EINVAL; \ 42762306a36Sopenharmony_ci \ 42862306a36Sopenharmony_ci index += (debug->__name.word_base / \ 42962306a36Sopenharmony_ci debug->__name.word_size); \ 43062306a36Sopenharmony_ci \ 43162306a36Sopenharmony_ci if (debug->__name.flags & RT2X00DEBUGFS_OFFSET) \ 43262306a36Sopenharmony_ci index *= debug->__name.word_size; \ 43362306a36Sopenharmony_ci \ 43462306a36Sopenharmony_ci value = debug->__name.read(intf->rt2x00dev, index); \ 43562306a36Sopenharmony_ci \ 43662306a36Sopenharmony_ci size = sprintf(line, __format, value); \ 43762306a36Sopenharmony_ci \ 43862306a36Sopenharmony_ci return simple_read_from_buffer(buf, length, offset, line, size); \ 43962306a36Sopenharmony_ci} 44062306a36Sopenharmony_ci 44162306a36Sopenharmony_ci#define RT2X00DEBUGFS_OPS_WRITE(__name, __type) \ 44262306a36Sopenharmony_cistatic ssize_t rt2x00debug_write_##__name(struct file *file, \ 44362306a36Sopenharmony_ci const char __user *buf,\ 44462306a36Sopenharmony_ci size_t length, \ 44562306a36Sopenharmony_ci loff_t *offset) \ 44662306a36Sopenharmony_ci{ \ 44762306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; \ 44862306a36Sopenharmony_ci const struct rt2x00debug *debug = intf->debug; \ 44962306a36Sopenharmony_ci char line[17]; \ 45062306a36Sopenharmony_ci size_t size; \ 45162306a36Sopenharmony_ci unsigned int index = intf->offset_##__name; \ 45262306a36Sopenharmony_ci __type value; \ 45362306a36Sopenharmony_ci \ 45462306a36Sopenharmony_ci if (*offset) \ 45562306a36Sopenharmony_ci return 0; \ 45662306a36Sopenharmony_ci \ 45762306a36Sopenharmony_ci if (index >= debug->__name.word_count) \ 45862306a36Sopenharmony_ci return -EINVAL; \ 45962306a36Sopenharmony_ci \ 46062306a36Sopenharmony_ci if (length > sizeof(line)) \ 46162306a36Sopenharmony_ci return -EINVAL; \ 46262306a36Sopenharmony_ci \ 46362306a36Sopenharmony_ci if (copy_from_user(line, buf, length)) \ 46462306a36Sopenharmony_ci return -EFAULT; \ 46562306a36Sopenharmony_ci line[16] = 0; \ 46662306a36Sopenharmony_ci \ 46762306a36Sopenharmony_ci size = strlen(line); \ 46862306a36Sopenharmony_ci value = simple_strtoul(line, NULL, 0); \ 46962306a36Sopenharmony_ci \ 47062306a36Sopenharmony_ci index += (debug->__name.word_base / \ 47162306a36Sopenharmony_ci debug->__name.word_size); \ 47262306a36Sopenharmony_ci \ 47362306a36Sopenharmony_ci if (debug->__name.flags & RT2X00DEBUGFS_OFFSET) \ 47462306a36Sopenharmony_ci index *= debug->__name.word_size; \ 47562306a36Sopenharmony_ci \ 47662306a36Sopenharmony_ci debug->__name.write(intf->rt2x00dev, index, value); \ 47762306a36Sopenharmony_ci \ 47862306a36Sopenharmony_ci *offset += size; \ 47962306a36Sopenharmony_ci return size; \ 48062306a36Sopenharmony_ci} 48162306a36Sopenharmony_ci 48262306a36Sopenharmony_ci#define RT2X00DEBUGFS_OPS(__name, __format, __type) \ 48362306a36Sopenharmony_ciRT2X00DEBUGFS_OPS_READ(__name, __format, __type); \ 48462306a36Sopenharmony_ciRT2X00DEBUGFS_OPS_WRITE(__name, __type); \ 48562306a36Sopenharmony_ci \ 48662306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_fop_##__name = {\ 48762306a36Sopenharmony_ci .owner = THIS_MODULE, \ 48862306a36Sopenharmony_ci .read = rt2x00debug_read_##__name, \ 48962306a36Sopenharmony_ci .write = rt2x00debug_write_##__name, \ 49062306a36Sopenharmony_ci .open = rt2x00debug_file_open, \ 49162306a36Sopenharmony_ci .release = rt2x00debug_file_release, \ 49262306a36Sopenharmony_ci .llseek = generic_file_llseek, \ 49362306a36Sopenharmony_ci}; 49462306a36Sopenharmony_ci 49562306a36Sopenharmony_ciRT2X00DEBUGFS_OPS(csr, "0x%.8x\n", u32); 49662306a36Sopenharmony_ciRT2X00DEBUGFS_OPS(eeprom, "0x%.4x\n", u16); 49762306a36Sopenharmony_ciRT2X00DEBUGFS_OPS(bbp, "0x%.2x\n", u8); 49862306a36Sopenharmony_ciRT2X00DEBUGFS_OPS(rf, "0x%.8x\n", u32); 49962306a36Sopenharmony_ciRT2X00DEBUGFS_OPS(rfcsr, "0x%.2x\n", u8); 50062306a36Sopenharmony_ci 50162306a36Sopenharmony_cistatic ssize_t rt2x00debug_read_dev_flags(struct file *file, 50262306a36Sopenharmony_ci char __user *buf, 50362306a36Sopenharmony_ci size_t length, 50462306a36Sopenharmony_ci loff_t *offset) 50562306a36Sopenharmony_ci{ 50662306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 50762306a36Sopenharmony_ci char line[16]; 50862306a36Sopenharmony_ci size_t size; 50962306a36Sopenharmony_ci 51062306a36Sopenharmony_ci if (*offset) 51162306a36Sopenharmony_ci return 0; 51262306a36Sopenharmony_ci 51362306a36Sopenharmony_ci size = sprintf(line, "0x%.8x\n", (unsigned int)intf->rt2x00dev->flags); 51462306a36Sopenharmony_ci 51562306a36Sopenharmony_ci return simple_read_from_buffer(buf, length, offset, line, size); 51662306a36Sopenharmony_ci} 51762306a36Sopenharmony_ci 51862306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_fop_dev_flags = { 51962306a36Sopenharmony_ci .owner = THIS_MODULE, 52062306a36Sopenharmony_ci .read = rt2x00debug_read_dev_flags, 52162306a36Sopenharmony_ci .open = rt2x00debug_file_open, 52262306a36Sopenharmony_ci .release = rt2x00debug_file_release, 52362306a36Sopenharmony_ci .llseek = default_llseek, 52462306a36Sopenharmony_ci}; 52562306a36Sopenharmony_ci 52662306a36Sopenharmony_cistatic ssize_t rt2x00debug_read_cap_flags(struct file *file, 52762306a36Sopenharmony_ci char __user *buf, 52862306a36Sopenharmony_ci size_t length, 52962306a36Sopenharmony_ci loff_t *offset) 53062306a36Sopenharmony_ci{ 53162306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 53262306a36Sopenharmony_ci char line[16]; 53362306a36Sopenharmony_ci size_t size; 53462306a36Sopenharmony_ci 53562306a36Sopenharmony_ci if (*offset) 53662306a36Sopenharmony_ci return 0; 53762306a36Sopenharmony_ci 53862306a36Sopenharmony_ci size = sprintf(line, "0x%.8x\n", (unsigned int)intf->rt2x00dev->cap_flags); 53962306a36Sopenharmony_ci 54062306a36Sopenharmony_ci return simple_read_from_buffer(buf, length, offset, line, size); 54162306a36Sopenharmony_ci} 54262306a36Sopenharmony_ci 54362306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_fop_cap_flags = { 54462306a36Sopenharmony_ci .owner = THIS_MODULE, 54562306a36Sopenharmony_ci .read = rt2x00debug_read_cap_flags, 54662306a36Sopenharmony_ci .open = rt2x00debug_file_open, 54762306a36Sopenharmony_ci .release = rt2x00debug_file_release, 54862306a36Sopenharmony_ci .llseek = default_llseek, 54962306a36Sopenharmony_ci}; 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_cistatic ssize_t rt2x00debug_write_restart_hw(struct file *file, 55262306a36Sopenharmony_ci const char __user *buf, 55362306a36Sopenharmony_ci size_t length, 55462306a36Sopenharmony_ci loff_t *offset) 55562306a36Sopenharmony_ci{ 55662306a36Sopenharmony_ci struct rt2x00debug_intf *intf = file->private_data; 55762306a36Sopenharmony_ci struct rt2x00_dev *rt2x00dev = intf->rt2x00dev; 55862306a36Sopenharmony_ci static unsigned long last_reset = INITIAL_JIFFIES; 55962306a36Sopenharmony_ci 56062306a36Sopenharmony_ci if (!rt2x00_has_cap_restart_hw(rt2x00dev)) 56162306a36Sopenharmony_ci return -EOPNOTSUPP; 56262306a36Sopenharmony_ci 56362306a36Sopenharmony_ci if (time_before(jiffies, last_reset + msecs_to_jiffies(2000))) 56462306a36Sopenharmony_ci return -EBUSY; 56562306a36Sopenharmony_ci 56662306a36Sopenharmony_ci last_reset = jiffies; 56762306a36Sopenharmony_ci 56862306a36Sopenharmony_ci ieee80211_restart_hw(rt2x00dev->hw); 56962306a36Sopenharmony_ci return length; 57062306a36Sopenharmony_ci} 57162306a36Sopenharmony_ci 57262306a36Sopenharmony_cistatic const struct file_operations rt2x00debug_restart_hw = { 57362306a36Sopenharmony_ci .owner = THIS_MODULE, 57462306a36Sopenharmony_ci .write = rt2x00debug_write_restart_hw, 57562306a36Sopenharmony_ci .open = simple_open, 57662306a36Sopenharmony_ci .llseek = generic_file_llseek, 57762306a36Sopenharmony_ci}; 57862306a36Sopenharmony_ci 57962306a36Sopenharmony_cistatic void rt2x00debug_create_file_driver(const char *name, 58062306a36Sopenharmony_ci struct rt2x00debug_intf *intf, 58162306a36Sopenharmony_ci struct debugfs_blob_wrapper *blob) 58262306a36Sopenharmony_ci{ 58362306a36Sopenharmony_ci char *data; 58462306a36Sopenharmony_ci 58562306a36Sopenharmony_ci data = kzalloc(3 * MAX_LINE_LENGTH, GFP_KERNEL); 58662306a36Sopenharmony_ci if (!data) 58762306a36Sopenharmony_ci return; 58862306a36Sopenharmony_ci 58962306a36Sopenharmony_ci blob->data = data; 59062306a36Sopenharmony_ci data += sprintf(data, "driver:\t%s\n", intf->rt2x00dev->ops->name); 59162306a36Sopenharmony_ci data += sprintf(data, "version:\t%s\n", DRV_VERSION); 59262306a36Sopenharmony_ci blob->size = strlen(blob->data); 59362306a36Sopenharmony_ci 59462306a36Sopenharmony_ci debugfs_create_blob(name, 0400, intf->driver_folder, blob); 59562306a36Sopenharmony_ci} 59662306a36Sopenharmony_ci 59762306a36Sopenharmony_cistatic void rt2x00debug_create_file_chipset(const char *name, 59862306a36Sopenharmony_ci struct rt2x00debug_intf *intf, 59962306a36Sopenharmony_ci struct debugfs_blob_wrapper *blob) 60062306a36Sopenharmony_ci{ 60162306a36Sopenharmony_ci const struct rt2x00debug *debug = intf->debug; 60262306a36Sopenharmony_ci char *data; 60362306a36Sopenharmony_ci 60462306a36Sopenharmony_ci data = kzalloc(9 * MAX_LINE_LENGTH, GFP_KERNEL); 60562306a36Sopenharmony_ci if (!data) 60662306a36Sopenharmony_ci return; 60762306a36Sopenharmony_ci 60862306a36Sopenharmony_ci blob->data = data; 60962306a36Sopenharmony_ci data += sprintf(data, "rt chip:\t%04x\n", intf->rt2x00dev->chip.rt); 61062306a36Sopenharmony_ci data += sprintf(data, "rf chip:\t%04x\n", intf->rt2x00dev->chip.rf); 61162306a36Sopenharmony_ci data += sprintf(data, "revision:\t%04x\n", intf->rt2x00dev->chip.rev); 61262306a36Sopenharmony_ci data += sprintf(data, "\n"); 61362306a36Sopenharmony_ci data += sprintf(data, "register\tbase\twords\twordsize\n"); 61462306a36Sopenharmony_ci#define RT2X00DEBUGFS_SPRINTF_REGISTER(__name) \ 61562306a36Sopenharmony_ci{ \ 61662306a36Sopenharmony_ci if (debug->__name.read) \ 61762306a36Sopenharmony_ci data += sprintf(data, __stringify(__name) \ 61862306a36Sopenharmony_ci "\t%d\t%d\t%d\n", \ 61962306a36Sopenharmony_ci debug->__name.word_base, \ 62062306a36Sopenharmony_ci debug->__name.word_count, \ 62162306a36Sopenharmony_ci debug->__name.word_size); \ 62262306a36Sopenharmony_ci} 62362306a36Sopenharmony_ci RT2X00DEBUGFS_SPRINTF_REGISTER(csr); 62462306a36Sopenharmony_ci RT2X00DEBUGFS_SPRINTF_REGISTER(eeprom); 62562306a36Sopenharmony_ci RT2X00DEBUGFS_SPRINTF_REGISTER(bbp); 62662306a36Sopenharmony_ci RT2X00DEBUGFS_SPRINTF_REGISTER(rf); 62762306a36Sopenharmony_ci RT2X00DEBUGFS_SPRINTF_REGISTER(rfcsr); 62862306a36Sopenharmony_ci#undef RT2X00DEBUGFS_SPRINTF_REGISTER 62962306a36Sopenharmony_ci 63062306a36Sopenharmony_ci blob->size = strlen(blob->data); 63162306a36Sopenharmony_ci 63262306a36Sopenharmony_ci debugfs_create_blob(name, 0400, intf->driver_folder, blob); 63362306a36Sopenharmony_ci} 63462306a36Sopenharmony_ci 63562306a36Sopenharmony_civoid rt2x00debug_register(struct rt2x00_dev *rt2x00dev) 63662306a36Sopenharmony_ci{ 63762306a36Sopenharmony_ci const struct rt2x00debug *debug = rt2x00dev->ops->debugfs; 63862306a36Sopenharmony_ci struct rt2x00debug_intf *intf; 63962306a36Sopenharmony_ci struct dentry *queue_folder; 64062306a36Sopenharmony_ci struct dentry *register_folder; 64162306a36Sopenharmony_ci 64262306a36Sopenharmony_ci intf = kzalloc(sizeof(struct rt2x00debug_intf), GFP_KERNEL); 64362306a36Sopenharmony_ci if (!intf) { 64462306a36Sopenharmony_ci rt2x00_err(rt2x00dev, "Failed to allocate debug handler\n"); 64562306a36Sopenharmony_ci return; 64662306a36Sopenharmony_ci } 64762306a36Sopenharmony_ci 64862306a36Sopenharmony_ci intf->debug = debug; 64962306a36Sopenharmony_ci intf->rt2x00dev = rt2x00dev; 65062306a36Sopenharmony_ci rt2x00dev->debugfs_intf = intf; 65162306a36Sopenharmony_ci 65262306a36Sopenharmony_ci intf->driver_folder = 65362306a36Sopenharmony_ci debugfs_create_dir(intf->rt2x00dev->ops->name, 65462306a36Sopenharmony_ci rt2x00dev->hw->wiphy->debugfsdir); 65562306a36Sopenharmony_ci 65662306a36Sopenharmony_ci rt2x00debug_create_file_driver("driver", intf, &intf->driver_blob); 65762306a36Sopenharmony_ci rt2x00debug_create_file_chipset("chipset", intf, &intf->chipset_blob); 65862306a36Sopenharmony_ci debugfs_create_file("dev_flags", 0400, intf->driver_folder, intf, 65962306a36Sopenharmony_ci &rt2x00debug_fop_dev_flags); 66062306a36Sopenharmony_ci debugfs_create_file("cap_flags", 0400, intf->driver_folder, intf, 66162306a36Sopenharmony_ci &rt2x00debug_fop_cap_flags); 66262306a36Sopenharmony_ci debugfs_create_file("restart_hw", 0200, intf->driver_folder, intf, 66362306a36Sopenharmony_ci &rt2x00debug_restart_hw); 66462306a36Sopenharmony_ci 66562306a36Sopenharmony_ci register_folder = debugfs_create_dir("register", intf->driver_folder); 66662306a36Sopenharmony_ci 66762306a36Sopenharmony_ci#define RT2X00DEBUGFS_CREATE_REGISTER_ENTRY(__intf, __name) \ 66862306a36Sopenharmony_ci({ \ 66962306a36Sopenharmony_ci if (debug->__name.read) { \ 67062306a36Sopenharmony_ci debugfs_create_u32(__stringify(__name) "_offset", 0600, \ 67162306a36Sopenharmony_ci register_folder, \ 67262306a36Sopenharmony_ci &(__intf)->offset_##__name); \ 67362306a36Sopenharmony_ci \ 67462306a36Sopenharmony_ci debugfs_create_file(__stringify(__name) "_value", 0600, \ 67562306a36Sopenharmony_ci register_folder, (__intf), \ 67662306a36Sopenharmony_ci &rt2x00debug_fop_##__name); \ 67762306a36Sopenharmony_ci } \ 67862306a36Sopenharmony_ci}) 67962306a36Sopenharmony_ci 68062306a36Sopenharmony_ci RT2X00DEBUGFS_CREATE_REGISTER_ENTRY(intf, csr); 68162306a36Sopenharmony_ci RT2X00DEBUGFS_CREATE_REGISTER_ENTRY(intf, eeprom); 68262306a36Sopenharmony_ci RT2X00DEBUGFS_CREATE_REGISTER_ENTRY(intf, bbp); 68362306a36Sopenharmony_ci RT2X00DEBUGFS_CREATE_REGISTER_ENTRY(intf, rf); 68462306a36Sopenharmony_ci RT2X00DEBUGFS_CREATE_REGISTER_ENTRY(intf, rfcsr); 68562306a36Sopenharmony_ci 68662306a36Sopenharmony_ci#undef RT2X00DEBUGFS_CREATE_REGISTER_ENTRY 68762306a36Sopenharmony_ci 68862306a36Sopenharmony_ci queue_folder = debugfs_create_dir("queue", intf->driver_folder); 68962306a36Sopenharmony_ci 69062306a36Sopenharmony_ci debugfs_create_file("dump", 0400, queue_folder, intf, 69162306a36Sopenharmony_ci &rt2x00debug_fop_queue_dump); 69262306a36Sopenharmony_ci 69362306a36Sopenharmony_ci skb_queue_head_init(&intf->frame_dump_skbqueue); 69462306a36Sopenharmony_ci init_waitqueue_head(&intf->frame_dump_waitqueue); 69562306a36Sopenharmony_ci 69662306a36Sopenharmony_ci debugfs_create_file("queue", 0400, queue_folder, intf, 69762306a36Sopenharmony_ci &rt2x00debug_fop_queue_stats); 69862306a36Sopenharmony_ci 69962306a36Sopenharmony_ci#ifdef CONFIG_RT2X00_LIB_CRYPTO 70062306a36Sopenharmony_ci if (rt2x00_has_cap_hw_crypto(rt2x00dev)) 70162306a36Sopenharmony_ci debugfs_create_file("crypto", 0444, queue_folder, intf, 70262306a36Sopenharmony_ci &rt2x00debug_fop_crypto_stats); 70362306a36Sopenharmony_ci#endif 70462306a36Sopenharmony_ci 70562306a36Sopenharmony_ci return; 70662306a36Sopenharmony_ci} 70762306a36Sopenharmony_ci 70862306a36Sopenharmony_civoid rt2x00debug_deregister(struct rt2x00_dev *rt2x00dev) 70962306a36Sopenharmony_ci{ 71062306a36Sopenharmony_ci struct rt2x00debug_intf *intf = rt2x00dev->debugfs_intf; 71162306a36Sopenharmony_ci 71262306a36Sopenharmony_ci if (unlikely(!intf)) 71362306a36Sopenharmony_ci return; 71462306a36Sopenharmony_ci 71562306a36Sopenharmony_ci skb_queue_purge(&intf->frame_dump_skbqueue); 71662306a36Sopenharmony_ci 71762306a36Sopenharmony_ci debugfs_remove_recursive(intf->driver_folder); 71862306a36Sopenharmony_ci kfree(intf->chipset_blob.data); 71962306a36Sopenharmony_ci kfree(intf->driver_blob.data); 72062306a36Sopenharmony_ci kfree(intf); 72162306a36Sopenharmony_ci 72262306a36Sopenharmony_ci rt2x00dev->debugfs_intf = NULL; 72362306a36Sopenharmony_ci} 724