#include <linux/module.h> #include <linux/dcache.h> #include <linux/debugfs.h> #include <linux/delay.h> #include <linux/mm.h> #include <net/iw_handler.h> #include "dev.h" #include "decl.h" #include "host.h" #include "debugfs.h" static struct dentry *libertas_dir = NULL; static char *szStates[] = { "Connected", "Disconnected" }; #ifdef PROC_DEBUG static void libertas_debug_init(wlan_private * priv, struct net_device *dev); #endif static int open_file_generic(struct inode *inode, struct file *file) { file->private_data = inode->i_private; return 0; } static ssize_t write_file_dummy(struct file *file, const char __user *buf, size_t count, loff_t *ppos) { return -EINVAL; } static const size_t len = PAGE_SIZE; static ssize_t libertas_dev_info(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; size_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; ssize_t res; pos += snprintf(buf+pos, len-pos, "state = %s\n", szStates[priv->adapter->connect_status]); pos += snprintf(buf+pos, len-pos, "region_code = %02x\n", (u32) priv->adapter->regioncode); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_getscantable(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; size_t pos = 0; int numscansdone = 0, res; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; struct bss_descriptor * iter_bss; pos += snprintf(buf+pos, len-pos, "# | ch | rssi | bssid | cap | Qual | SSID \n"); mutex_lock(&priv->adapter->lock); list_for_each_entry (iter_bss, &priv->adapter->network_list, list) { u16 ibss = (iter_bss->capability & WLAN_CAPABILITY_IBSS); u16 privacy = (iter_bss->capability & WLAN_CAPABILITY_PRIVACY); u16 spectrum_mgmt = (iter_bss->capability & WLAN_CAPABILITY_SPECTRUM_MGMT); pos += snprintf(buf+pos, len-pos, "%02u| %03d | %04ld | " MAC_FMT " |", numscansdone, iter_bss->channel, iter_bss->rssi, MAC_ARG(iter_bss->bssid)); pos += snprintf(buf+pos, len-pos, " %04x-", iter_bss->capability); pos += snprintf(buf+pos, len-pos, "%c%c%c |", ibss ? 'A' : 'I', privacy ? 'P' : ' ', spectrum_mgmt ? 'S' : ' '); pos += snprintf(buf+pos, len-pos, " %04d |", SCAN_RSSI(iter_bss->rssi)); pos += snprintf(buf+pos, len-pos, " %s\n", escape_essid(iter_bss->ssid, iter_bss->ssid_len)); numscansdone++; } mutex_unlock(&priv->adapter->lock); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_sleepparams_write(struct file *file, const char __user *user_buf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t buf_size, res; int p1, p2, p3, p4, p5, p6; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, user_buf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d %d %d %d", &p1, &p2, &p3, &p4, &p5, &p6); if (res != 6) { res = -EFAULT; goto out_unlock; } priv->adapter->sp.sp_error = p1; priv->adapter->sp.sp_offset = p2; priv->adapter->sp.sp_stabletime = p3; priv->adapter->sp.sp_calcontrol = p4; priv->adapter->sp.sp_extsleepclk = p5; priv->adapter->sp.sp_reserved = p6; res = libertas_prepare_and_send_command(priv, CMD_802_11_SLEEP_PARAMS, CMD_ACT_SET, CMD_OPTION_WAITFORRSP, 0, NULL); if (!res) res = count; else res = -EINVAL; out_unlock: free_page(addr); return res; } static ssize_t libertas_sleepparams_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res; size_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_prepare_and_send_command(priv, CMD_802_11_SLEEP_PARAMS, CMD_ACT_GET, CMD_OPTION_WAITFORRSP, 0, NULL); if (res) { res = -EFAULT; goto out_unlock; } pos += snprintf(buf, len, "%d %d %d %d %d %d\n", adapter->sp.sp_error, adapter->sp.sp_offset, adapter->sp.sp_stabletime, adapter->sp.sp_calcontrol, adapter->sp.sp_extsleepclk, adapter->sp.sp_reserved); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); out_unlock: free_page(addr); return res; } static ssize_t libertas_extscan(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; union iwreq_data wrqu; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } libertas_send_specific_ssid_scan(priv, buf, strlen(buf)-1, 0); memset(&wrqu, 0, sizeof(union iwreq_data)); wireless_send_event(priv->dev, SIOCGIWSCAN, &wrqu, NULL); out_unlock: free_page(addr); return count; } static int libertas_parse_chan(char *buf, size_t count, struct wlan_ioctl_user_scan_cfg *scan_cfg, int dur) { char *start, *end, *hold, *str; int i = 0; start = strstr(buf, "chan="); if (!start) return -EINVAL; start += 5; end = strstr(start, " "); if (!end) end = buf + count; hold = kzalloc((end - start)+1, GFP_KERNEL); if (!hold) return -ENOMEM; strncpy(hold, start, end - start); hold[(end-start)+1] = '\0'; while(hold && (str = strsep(&hold, ","))) { int chan; char band, passive = 0; sscanf(str, "%d%c%c", &chan, &band, &passive); scan_cfg->chanlist[i].channumber = chan; scan_cfg->chanlist[i].scantype = passive ? 1 : 0; if (band == 'b' || band == 'g') scan_cfg->chanlist[i].radiotype = 0; else if (band == 'a') scan_cfg->chanlist[i].radiotype = 1; scan_cfg->chanlist[i].scantime = dur; i++; } kfree(hold); return i; } static void libertas_parse_bssid(char *buf, size_t count, struct wlan_ioctl_user_scan_cfg *scan_cfg) { char *hold; unsigned int mac[ETH_ALEN]; hold = strstr(buf, "bssid="); if (!hold) return; hold += 6; sscanf(hold, MAC_FMT, mac, mac+1, mac+2, mac+3, mac+4, mac+5); memcpy(scan_cfg->bssid, mac, ETH_ALEN); } static void libertas_parse_ssid(char *buf, size_t count, struct wlan_ioctl_user_scan_cfg *scan_cfg) { char *hold, *end; ssize_t size; hold = strstr(buf, "ssid="); if (!hold) return; hold += 5; end = strstr(hold, " "); if (!end) end = buf + count - 1; size = min((size_t)IW_ESSID_MAX_SIZE, (size_t) (end - hold)); strncpy(scan_cfg->ssid, hold, size); return; } static int libertas_parse_clear(char *buf, size_t count, const char *tag) { char *hold; int val; hold = strstr(buf, tag); if (!hold) return 0; hold += strlen(tag); sscanf(hold, "%d", &val); if (val != 0) val = 1; return val; } static int libertas_parse_dur(char *buf, size_t count, struct wlan_ioctl_user_scan_cfg *scan_cfg) { char *hold; int val; hold = strstr(buf, "dur="); if (!hold) return 0; hold += 4; sscanf(hold, "%d", &val); return val; } static void libertas_parse_probes(char *buf, size_t count, struct wlan_ioctl_user_scan_cfg *scan_cfg) { char *hold; int val; hold = strstr(buf, "probes="); if (!hold) return; hold += 7; sscanf(hold, "%d", &val); scan_cfg->numprobes = val; return; } static void libertas_parse_type(char *buf, size_t count, struct wlan_ioctl_user_scan_cfg *scan_cfg) { char *hold; int val; hold = strstr(buf, "type="); if (!hold) return; hold += 5; sscanf(hold, "%d", &val); /* type=1,2 or 3 */ if (val < 1 || val > 3) return; scan_cfg->bsstype = val; return; } static ssize_t libertas_setuserscan(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; struct wlan_ioctl_user_scan_cfg *scan_cfg; union iwreq_data wrqu; int dur; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; scan_cfg = kzalloc(sizeof(struct wlan_ioctl_user_scan_cfg), GFP_KERNEL); if (!scan_cfg) return -ENOMEM; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } scan_cfg->bsstype = WLAN_SCAN_BSS_TYPE_ANY; dur = libertas_parse_dur(buf, count, scan_cfg); libertas_parse_chan(buf, count, scan_cfg, dur); libertas_parse_bssid(buf, count, scan_cfg); scan_cfg->clear_bssid = libertas_parse_clear(buf, count, "clear_bssid="); libertas_parse_ssid(buf, count, scan_cfg); scan_cfg->clear_ssid = libertas_parse_clear(buf, count, "clear_ssid="); libertas_parse_probes(buf, count, scan_cfg); libertas_parse_type(buf, count, scan_cfg); wlan_scan_networks(priv, scan_cfg, 1); wait_event_interruptible(priv->adapter->cmd_pending, !priv->adapter->nr_cmd_pending); memset(&wrqu, 0x00, sizeof(union iwreq_data)); wireless_send_event(priv->dev, SIOCGIWSCAN, &wrqu, NULL); out_unlock: free_page(addr); kfree(scan_cfg); return count; } static int libertas_event_initcmd(wlan_private *priv, void **response_buf, struct cmd_ctrl_node **cmdnode, struct cmd_ds_command **cmd) { u16 wait_option = CMD_OPTION_WAITFORRSP; if (!(*cmdnode = libertas_get_free_cmd_ctrl_node(priv))) { lbs_deb_debugfs("failed libertas_get_free_cmd_ctrl_node\n"); return -ENOMEM; } if (!(*response_buf = kmalloc(3000, GFP_KERNEL))) { lbs_deb_debugfs("failed to allocate response buffer!\n"); return -ENOMEM; } libertas_set_cmd_ctrl_node(priv, *cmdnode, 0, wait_option, NULL); init_waitqueue_head(&(*cmdnode)->cmdwait_q); (*cmdnode)->pdata_buf = *response_buf; (*cmdnode)->cmdflags |= CMD_F_HOSTCMD; (*cmdnode)->cmdwaitqwoken = 0; *cmd = (struct cmd_ds_command *)(*cmdnode)->bufvirtualaddr; (*cmd)->command = cpu_to_le16(CMD_802_11_SUBSCRIBE_EVENT); (*cmd)->seqnum = cpu_to_le16(++priv->adapter->seqnum); (*cmd)->result = 0; return 0; } static ssize_t libertas_lowrssi_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res, cmd_len; ssize_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) { free_page(addr); return res; } event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); event = (void *)(response_buf + S_DS_GEN); while (cmd_len < le16_to_cpu(pcmdptr->size)) { struct mrvlietypesheader *header = (void *)(response_buf + cmd_len); switch (header->type) { struct mrvlietypes_rssithreshold *Lowrssi; case __constant_cpu_to_le16(TLV_TYPE_RSSI_LOW): Lowrssi = (void *)(response_buf + cmd_len); pos += snprintf(buf+pos, len-pos, "%d %d %d\n", Lowrssi->rssivalue, Lowrssi->rssifreq, (event->events & cpu_to_le16(0x0001))?1:0); default: cmd_len += sizeof(struct mrvlietypes_snrthreshold); break; } } kfree(response_buf); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static u16 libertas_get_events_bitmap(wlan_private *priv) { wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res; u16 event_bitmap; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) return res; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); return 0; } if (pcmdptr->command != CMD_RET(CMD_802_11_SUBSCRIBE_EVENT)) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); return 0; } event = (struct cmd_ds_802_11_subscribe_event *)(response_buf + S_DS_GEN); event_bitmap = le16_to_cpu(event->events); kfree(response_buf); return event_bitmap; } static ssize_t libertas_lowrssi_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res, buf_size; int value, freq, subscribed, cmd_len; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; struct mrvlietypes_rssithreshold *rssi_threshold; void *response_buf; u16 event_bitmap; u8 *ptr; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d", &value, &freq, &subscribed); if (res != 3) { res = -EFAULT; goto out_unlock; } event_bitmap = libertas_get_events_bitmap(priv); res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) goto out_unlock; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_SET); pcmdptr->size = cpu_to_le16(S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event) + sizeof(struct mrvlietypes_rssithreshold)); cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); ptr = (u8*) pcmdptr+cmd_len; rssi_threshold = (struct mrvlietypes_rssithreshold *)(ptr); rssi_threshold->header.type = cpu_to_le16(0x0104); rssi_threshold->header.len = cpu_to_le16(2); rssi_threshold->rssivalue = value; rssi_threshold->rssifreq = freq; event_bitmap |= subscribed ? 0x0001 : 0x0; event->events = cpu_to_le16(event_bitmap); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_lowsnr_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res, cmd_len; ssize_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) { free_page(addr); return res; } event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); event = (void *)(response_buf + S_DS_GEN); while (cmd_len < le16_to_cpu(pcmdptr->size)) { struct mrvlietypesheader *header = (void *)(response_buf + cmd_len); switch (header->type) { struct mrvlietypes_snrthreshold *LowSnr; case __constant_cpu_to_le16(TLV_TYPE_SNR_LOW): LowSnr = (void *)(response_buf + cmd_len); pos += snprintf(buf+pos, len-pos, "%d %d %d\n", LowSnr->snrvalue, LowSnr->snrfreq, (event->events & cpu_to_le16(0x0002))?1:0); default: cmd_len += sizeof(struct mrvlietypes_snrthreshold); break; } } kfree(response_buf); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_lowsnr_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res, buf_size; int value, freq, subscribed, cmd_len; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; struct mrvlietypes_snrthreshold *snr_threshold; void *response_buf; u16 event_bitmap; u8 *ptr; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d", &value, &freq, &subscribed); if (res != 3) { res = -EFAULT; goto out_unlock; } event_bitmap = libertas_get_events_bitmap(priv); res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) goto out_unlock; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_SET); pcmdptr->size = cpu_to_le16(S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event) + sizeof(struct mrvlietypes_snrthreshold)); cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); ptr = (u8*) pcmdptr+cmd_len; snr_threshold = (struct mrvlietypes_snrthreshold *)(ptr); snr_threshold->header.type = cpu_to_le16(TLV_TYPE_SNR_LOW); snr_threshold->header.len = cpu_to_le16(2); snr_threshold->snrvalue = value; snr_threshold->snrfreq = freq; event_bitmap |= subscribed ? 0x0002 : 0x0; event->events = cpu_to_le16(event_bitmap); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_failcount_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res, cmd_len; ssize_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) { free_page(addr); return res; } event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); event = (void *)(response_buf + S_DS_GEN); while (cmd_len < le16_to_cpu(pcmdptr->size)) { struct mrvlietypesheader *header = (void *)(response_buf + cmd_len); switch (header->type) { struct mrvlietypes_failurecount *failcount; case __constant_cpu_to_le16(TLV_TYPE_FAILCOUNT): failcount = (void *)(response_buf + cmd_len); pos += snprintf(buf+pos, len-pos, "%d %d %d\n", failcount->failvalue, failcount->Failfreq, (event->events & cpu_to_le16(0x0004))?1:0); default: cmd_len += sizeof(struct mrvlietypes_failurecount); break; } } kfree(response_buf); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_failcount_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res, buf_size; int value, freq, subscribed, cmd_len; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; struct mrvlietypes_failurecount *failcount; void *response_buf; u16 event_bitmap; u8 *ptr; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d", &value, &freq, &subscribed); if (res != 3) { res = -EFAULT; goto out_unlock; } event_bitmap = libertas_get_events_bitmap(priv); res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) goto out_unlock; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_SET); pcmdptr->size = cpu_to_le16(S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event) + sizeof(struct mrvlietypes_failurecount)); cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); ptr = (u8*) pcmdptr+cmd_len; failcount = (struct mrvlietypes_failurecount *)(ptr); failcount->header.type = cpu_to_le16(TLV_TYPE_FAILCOUNT); failcount->header.len = cpu_to_le16(2); failcount->failvalue = value; failcount->Failfreq = freq; event_bitmap |= subscribed ? 0x0004 : 0x0; event->events = cpu_to_le16(event_bitmap); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = (struct cmd_ds_command *)response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_bcnmiss_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res, cmd_len; ssize_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) { free_page(addr); return res; } event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); free_page(addr); kfree(response_buf); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); free_page(addr); kfree(response_buf); return 0; } cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); event = (void *)(response_buf + S_DS_GEN); while (cmd_len < le16_to_cpu(pcmdptr->size)) { struct mrvlietypesheader *header = (void *)(response_buf + cmd_len); switch (header->type) { struct mrvlietypes_beaconsmissed *bcnmiss; case __constant_cpu_to_le16(TLV_TYPE_BCNMISS): bcnmiss = (void *)(response_buf + cmd_len); pos += snprintf(buf+pos, len-pos, "%d N/A %d\n", bcnmiss->beaconmissed, (event->events & cpu_to_le16(0x0008))?1:0); default: cmd_len += sizeof(struct mrvlietypes_beaconsmissed); break; } } kfree(response_buf); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_bcnmiss_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res, buf_size; int value, freq, subscribed, cmd_len; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; struct mrvlietypes_beaconsmissed *bcnmiss; void *response_buf; u16 event_bitmap; u8 *ptr; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d", &value, &freq, &subscribed); if (res != 3) { res = -EFAULT; goto out_unlock; } event_bitmap = libertas_get_events_bitmap(priv); res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) goto out_unlock; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_SET); pcmdptr->size = cpu_to_le16(S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event) + sizeof(struct mrvlietypes_beaconsmissed)); cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); ptr = (u8*) pcmdptr+cmd_len; bcnmiss = (struct mrvlietypes_beaconsmissed *)(ptr); bcnmiss->header.type = cpu_to_le16(TLV_TYPE_BCNMISS); bcnmiss->header.len = cpu_to_le16(2); bcnmiss->beaconmissed = value; event_bitmap |= subscribed ? 0x0008 : 0x0; event->events = cpu_to_le16(event_bitmap); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); free_page(addr); kfree(response_buf); return 0; } res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_highrssi_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res, cmd_len; ssize_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) { free_page(addr); return res; } event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); event = (void *)(response_buf + S_DS_GEN); while (cmd_len < le16_to_cpu(pcmdptr->size)) { struct mrvlietypesheader *header = (void *)(response_buf + cmd_len); switch (header->type) { struct mrvlietypes_rssithreshold *Highrssi; case __constant_cpu_to_le16(TLV_TYPE_RSSI_HIGH): Highrssi = (void *)(response_buf + cmd_len); pos += snprintf(buf+pos, len-pos, "%d %d %d\n", Highrssi->rssivalue, Highrssi->rssifreq, (event->events & cpu_to_le16(0x0010))?1:0); default: cmd_len += sizeof(struct mrvlietypes_snrthreshold); break; } } kfree(response_buf); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_highrssi_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res, buf_size; int value, freq, subscribed, cmd_len; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; struct mrvlietypes_rssithreshold *rssi_threshold; void *response_buf; u16 event_bitmap; u8 *ptr; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d", &value, &freq, &subscribed); if (res != 3) { res = -EFAULT; goto out_unlock; } event_bitmap = libertas_get_events_bitmap(priv); res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) goto out_unlock; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_SET); pcmdptr->size = cpu_to_le16(S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event) + sizeof(struct mrvlietypes_rssithreshold)); cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); ptr = (u8*) pcmdptr+cmd_len; rssi_threshold = (struct mrvlietypes_rssithreshold *)(ptr); rssi_threshold->header.type = cpu_to_le16(TLV_TYPE_RSSI_HIGH); rssi_threshold->header.len = cpu_to_le16(2); rssi_threshold->rssivalue = value; rssi_threshold->rssifreq = freq; event_bitmap |= subscribed ? 0x0010 : 0x0; event->events = cpu_to_le16(event_bitmap); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); return 0; } res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_highsnr_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; void *response_buf; int res, cmd_len; ssize_t pos = 0; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) { free_page(addr); return res; } event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_GET); pcmdptr->size = cpu_to_le16(sizeof(*event) + S_DS_GEN); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); event = (void *)(response_buf + S_DS_GEN); while (cmd_len < le16_to_cpu(pcmdptr->size)) { struct mrvlietypesheader *header = (void *)(response_buf + cmd_len); switch (header->type) { struct mrvlietypes_snrthreshold *HighSnr; case __constant_cpu_to_le16(TLV_TYPE_SNR_HIGH): HighSnr = (void *)(response_buf + cmd_len); pos += snprintf(buf+pos, len-pos, "%d %d %d\n", HighSnr->snrvalue, HighSnr->snrfreq, (event->events & cpu_to_le16(0x0020))?1:0); default: cmd_len += sizeof(struct mrvlietypes_snrthreshold); break; } } kfree(response_buf); res = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return res; } static ssize_t libertas_highsnr_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; ssize_t res, buf_size; int value, freq, subscribed, cmd_len; struct cmd_ctrl_node *pcmdnode; struct cmd_ds_command *pcmdptr; struct cmd_ds_802_11_subscribe_event *event; struct mrvlietypes_snrthreshold *snr_threshold; void *response_buf; u16 event_bitmap; u8 *ptr; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%d %d %d", &value, &freq, &subscribed); if (res != 3) { res = -EFAULT; goto out_unlock; } event_bitmap = libertas_get_events_bitmap(priv); res = libertas_event_initcmd(priv, &response_buf, &pcmdnode, &pcmdptr); if (res < 0) goto out_unlock; event = &pcmdptr->params.subscribe_event; event->action = cpu_to_le16(CMD_ACT_SET); pcmdptr->size = cpu_to_le16(S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event) + sizeof(struct mrvlietypes_snrthreshold)); cmd_len = S_DS_GEN + sizeof(struct cmd_ds_802_11_subscribe_event); ptr = (u8*) pcmdptr+cmd_len; snr_threshold = (struct mrvlietypes_snrthreshold *)(ptr); snr_threshold->header.type = cpu_to_le16(TLV_TYPE_SNR_HIGH); snr_threshold->header.len = cpu_to_le16(2); snr_threshold->snrvalue = value; snr_threshold->snrfreq = freq; event_bitmap |= subscribed ? 0x0020 : 0x0; event->events = cpu_to_le16(event_bitmap); libertas_queue_cmd(adapter, pcmdnode, 1); wake_up_interruptible(&priv->waitq); /* Sleep until response is generated by FW */ wait_event_interruptible(pcmdnode->cmdwait_q, pcmdnode->cmdwaitqwoken); pcmdptr = response_buf; if (pcmdptr->result) { lbs_pr_err("%s: fail, result=%d\n", __func__, le16_to_cpu(pcmdptr->result)); kfree(response_buf); free_page(addr); return 0; } if (pcmdptr->command != cpu_to_le16(CMD_RET(CMD_802_11_SUBSCRIBE_EVENT))) { lbs_pr_err("command response incorrect!\n"); kfree(response_buf); free_page(addr); return 0; } res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_rdmac_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct wlan_offset_value offval; ssize_t pos = 0; int ret; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; offval.offset = priv->mac_offset; offval.value = 0; ret = libertas_prepare_and_send_command(priv, CMD_MAC_REG_ACCESS, 0, CMD_OPTION_WAITFORRSP, 0, &offval); mdelay(10); pos += snprintf(buf+pos, len-pos, "MAC[0x%x] = 0x%08x\n", priv->mac_offset, adapter->offsetvalue.value); ret = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return ret; } static ssize_t libertas_rdmac_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } priv->mac_offset = simple_strtoul((char *)buf, NULL, 16); res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_wrmac_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; u32 offset, value; struct wlan_offset_value offval; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%x %x", &offset, &value); if (res != 2) { res = -EFAULT; goto out_unlock; } offval.offset = offset; offval.value = value; res = libertas_prepare_and_send_command(priv, CMD_MAC_REG_ACCESS, 1, CMD_OPTION_WAITFORRSP, 0, &offval); mdelay(10); res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_rdbbp_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct wlan_offset_value offval; ssize_t pos = 0; int ret; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; offval.offset = priv->bbp_offset; offval.value = 0; ret = libertas_prepare_and_send_command(priv, CMD_BBP_REG_ACCESS, 0, CMD_OPTION_WAITFORRSP, 0, &offval); mdelay(10); pos += snprintf(buf+pos, len-pos, "BBP[0x%x] = 0x%08x\n", priv->bbp_offset, adapter->offsetvalue.value); ret = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return ret; } static ssize_t libertas_rdbbp_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } priv->bbp_offset = simple_strtoul((char *)buf, NULL, 16); res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_wrbbp_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; u32 offset, value; struct wlan_offset_value offval; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%x %x", &offset, &value); if (res != 2) { res = -EFAULT; goto out_unlock; } offval.offset = offset; offval.value = value; res = libertas_prepare_and_send_command(priv, CMD_BBP_REG_ACCESS, 1, CMD_OPTION_WAITFORRSP, 0, &offval); mdelay(10); res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_rdrf_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; wlan_adapter *adapter = priv->adapter; struct wlan_offset_value offval; ssize_t pos = 0; int ret; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; offval.offset = priv->rf_offset; offval.value = 0; ret = libertas_prepare_and_send_command(priv, CMD_RF_REG_ACCESS, 0, CMD_OPTION_WAITFORRSP, 0, &offval); mdelay(10); pos += snprintf(buf+pos, len-pos, "RF[0x%x] = 0x%08x\n", priv->rf_offset, adapter->offsetvalue.value); ret = simple_read_from_buffer(userbuf, count, ppos, buf, pos); free_page(addr); return ret; } static ssize_t libertas_rdrf_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } priv->rf_offset = simple_strtoul((char *)buf, NULL, 16); res = count; out_unlock: free_page(addr); return res; } static ssize_t libertas_wrrf_write(struct file *file, const char __user *userbuf, size_t count, loff_t *ppos) { wlan_private *priv = file->private_data; ssize_t res, buf_size; u32 offset, value; struct wlan_offset_value offval; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; buf_size = min(count, len - 1); if (copy_from_user(buf, userbuf, buf_size)) { res = -EFAULT; goto out_unlock; } res = sscanf(buf, "%x %x", &offset, &value); if (res != 2) { res = -EFAULT; goto out_unlock; } offval.offset = offset; offval.value = value; res = libertas_prepare_and_send_command(priv, CMD_RF_REG_ACCESS, 1, CMD_OPTION_WAITFORRSP, 0, &offval); mdelay(10); res = count; out_unlock: free_page(addr); return res; } #define FOPS(fread, fwrite) { \ .owner = THIS_MODULE, \ .open = open_file_generic, \ .read = (fread), \ .write = (fwrite), \ } struct libertas_debugfs_files { char *name; int perm; struct file_operations fops; }; static struct libertas_debugfs_files debugfs_files[] = { { "info", 0444, FOPS(libertas_dev_info, write_file_dummy), }, { "getscantable", 0444, FOPS(libertas_getscantable, write_file_dummy), }, { "sleepparams", 0644, FOPS(libertas_sleepparams_read, libertas_sleepparams_write), }, { "extscan", 0600, FOPS(NULL, libertas_extscan), }, { "setuserscan", 0600, FOPS(NULL, libertas_setuserscan), }, }; static struct libertas_debugfs_files debugfs_events_files[] = { {"low_rssi", 0644, FOPS(libertas_lowrssi_read, libertas_lowrssi_write), }, {"low_snr", 0644, FOPS(libertas_lowsnr_read, libertas_lowsnr_write), }, {"failure_count", 0644, FOPS(libertas_failcount_read, libertas_failcount_write), }, {"beacon_missed", 0644, FOPS(libertas_bcnmiss_read, libertas_bcnmiss_write), }, {"high_rssi", 0644, FOPS(libertas_highrssi_read, libertas_highrssi_write), }, {"high_snr", 0644, FOPS(libertas_highsnr_read, libertas_highsnr_write), }, }; static struct libertas_debugfs_files debugfs_regs_files[] = { {"rdmac", 0644, FOPS(libertas_rdmac_read, libertas_rdmac_write), }, {"wrmac", 0600, FOPS(NULL, libertas_wrmac_write), }, {"rdbbp", 0644, FOPS(libertas_rdbbp_read, libertas_rdbbp_write), }, {"wrbbp", 0600, FOPS(NULL, libertas_wrbbp_write), }, {"rdrf", 0644, FOPS(libertas_rdrf_read, libertas_rdrf_write), }, {"wrrf", 0600, FOPS(NULL, libertas_wrrf_write), }, }; void libertas_debugfs_init(void) { if (!libertas_dir) libertas_dir = debugfs_create_dir("libertas_wireless", NULL); return; } void libertas_debugfs_remove(void) { if (libertas_dir) debugfs_remove(libertas_dir); return; } void libertas_debugfs_init_one(wlan_private *priv, struct net_device *dev) { int i; struct libertas_debugfs_files *files; if (!libertas_dir) goto exit; priv->debugfs_dir = debugfs_create_dir(dev->name, libertas_dir); if (!priv->debugfs_dir) goto exit; for (i=0; i<ARRAY_SIZE(debugfs_files); i++) { files = &debugfs_files[i]; priv->debugfs_files[i] = debugfs_create_file(files->name, files->perm, priv->debugfs_dir, priv, &files->fops); } priv->events_dir = debugfs_create_dir("subscribed_events", priv->debugfs_dir); if (!priv->events_dir) goto exit; for (i=0; i<ARRAY_SIZE(debugfs_events_files); i++) { files = &debugfs_events_files[i]; priv->debugfs_events_files[i] = debugfs_create_file(files->name, files->perm, priv->events_dir, priv, &files->fops); } priv->regs_dir = debugfs_create_dir("registers", priv->debugfs_dir); if (!priv->regs_dir) goto exit; for (i=0; i<ARRAY_SIZE(debugfs_regs_files); i++) { files = &debugfs_regs_files[i]; priv->debugfs_regs_files[i] = debugfs_create_file(files->name, files->perm, priv->regs_dir, priv, &files->fops); } #ifdef PROC_DEBUG libertas_debug_init(priv, dev); #endif exit: return; } void libertas_debugfs_remove_one(wlan_private *priv) { int i; for(i=0; i<ARRAY_SIZE(debugfs_regs_files); i++) debugfs_remove(priv->debugfs_regs_files[i]); debugfs_remove(priv->regs_dir); for(i=0; i<ARRAY_SIZE(debugfs_events_files); i++) debugfs_remove(priv->debugfs_events_files[i]); debugfs_remove(priv->events_dir); #ifdef PROC_DEBUG debugfs_remove(priv->debugfs_debug); #endif for(i=0; i<ARRAY_SIZE(debugfs_files); i++) debugfs_remove(priv->debugfs_files[i]); debugfs_remove(priv->debugfs_dir); } /* debug entry */ #ifdef PROC_DEBUG #define item_size(n) (FIELD_SIZEOF(wlan_adapter, n)) #define item_addr(n) (offsetof(wlan_adapter, n)) struct debug_data { char name[32]; u32 size; size_t addr; }; /* To debug any member of wlan_adapter, simply add one line here. */ static struct debug_data items[] = { {"intcounter", item_size(intcounter), item_addr(intcounter)}, {"psmode", item_size(psmode), item_addr(psmode)}, {"psstate", item_size(psstate), item_addr(psstate)}, }; static int num_of_items = ARRAY_SIZE(items); /** * @brief proc read function * * @param page pointer to buffer * @param s read data starting position * @param off offset * @param cnt counter * @param eof end of file flag * @param data data to output * @return number of output data */ static ssize_t wlan_debugfs_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { int val = 0; size_t pos = 0; ssize_t res; char *p; int i; struct debug_data *d; unsigned long addr = get_zeroed_page(GFP_KERNEL); char *buf = (char *)addr; p = buf; d = (struct debug_data *)file->private_data; for (i = 0; i < num_of_items; i++) { if (d[i].size == 1) val = *((u8 *) d[i].addr); else if (d[i].size == 2) val = *((u16 *) d[i].addr); else if (d[i].size == 4) val = *((u32 *) d[i].addr); else if (d[i].size == 8) val = *((u64 *) d[i].addr); pos += sprintf(p + pos, "%s=%d\n", d[i].name, val); } res = simple_read_from_buffer(userbuf, count, ppos, p, pos); free_page(addr); return res; } /** * @brief proc write function * * @param f file pointer * @param buf pointer to data buffer * @param cnt data number to write * @param data data to write * @return number of data */ static ssize_t wlan_debugfs_write(struct file *f, const char __user *buf, size_t cnt, loff_t *ppos) { int r, i; char *pdata; char *p; char *p0; char *p1; char *p2; struct debug_data *d = (struct debug_data *)f->private_data; pdata = (char *)kmalloc(cnt, GFP_KERNEL); if (pdata == NULL) return 0; if (copy_from_user(pdata, buf, cnt)) { lbs_deb_debugfs("Copy from user failed\n"); kfree(pdata); return 0; } p0 = pdata; for (i = 0; i < num_of_items; i++) { do { p = strstr(p0, d[i].name); if (p == NULL) break; p1 = strchr(p, '\n'); if (p1 == NULL) break; p0 = p1++; p2 = strchr(p, '='); if (!p2) break; p2++; r = simple_strtoul(p2, NULL, 0); if (d[i].size == 1) *((u8 *) d[i].addr) = (u8) r; else if (d[i].size == 2) *((u16 *) d[i].addr) = (u16) r; else if (d[i].size == 4) *((u32 *) d[i].addr) = (u32) r; else if (d[i].size == 8) *((u64 *) d[i].addr) = (u64) r; break; } while (1); } kfree(pdata); return (ssize_t)cnt; } static struct file_operations libertas_debug_fops = { .owner = THIS_MODULE, .open = open_file_generic, .write = wlan_debugfs_write, .read = wlan_debugfs_read, }; /** * @brief create debug proc file * * @param priv pointer wlan_private * @param dev pointer net_device * @return N/A */ static void libertas_debug_init(wlan_private * priv, struct net_device *dev) { int i; if (!priv->debugfs_dir) return; for (i = 0; i < num_of_items; i++) items[i].addr += (size_t) priv->adapter; priv->debugfs_debug = debugfs_create_file("debug", 0644, priv->debugfs_dir, &items[0], &libertas_debug_fops); } #endif