Commit fa62f99c authored by David Woodhouse's avatar David Woodhouse Committed by John W. Linville

libertas: convert 802_11_SCAN to a direct command

Signed-off-by: default avatarDavid Woodhouse <dwmw2@infradead.org>
Acked-by: default avatarDan Williams <dcbw@redhat.com>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent c5562e98
...@@ -1382,10 +1382,6 @@ int lbs_prepare_and_send_command(struct lbs_private *priv, ...@@ -1382,10 +1382,6 @@ int lbs_prepare_and_send_command(struct lbs_private *priv,
ret = lbs_cmd_802_11_ps_mode(priv, cmdptr, cmd_action); ret = lbs_cmd_802_11_ps_mode(priv, cmdptr, cmd_action);
break; break;
case CMD_802_11_SCAN:
ret = lbs_cmd_80211_scan(priv, cmdptr, pdata_buf);
break;
case CMD_MAC_CONTROL: case CMD_MAC_CONTROL:
ret = lbs_cmd_mac_control(priv, cmdptr); ret = lbs_cmd_mac_control(priv, cmdptr);
break; break;
......
...@@ -352,10 +352,6 @@ static inline int handle_cmd_response(struct lbs_private *priv, ...@@ -352,10 +352,6 @@ static inline int handle_cmd_response(struct lbs_private *priv,
ret = lbs_ret_reg_access(priv, respcmd, resp); ret = lbs_ret_reg_access(priv, respcmd, resp);
break; break;
case CMD_RET(CMD_802_11_SCAN):
ret = lbs_ret_80211_scan(priv, resp);
break;
case CMD_RET(CMD_802_11_GET_LOG): case CMD_RET(CMD_802_11_GET_LOG):
ret = lbs_ret_get_log(priv, resp); ret = lbs_ret_get_log(priv, resp);
break; break;
......
...@@ -174,9 +174,11 @@ struct cmd_ds_802_11_subscribe_event { ...@@ -174,9 +174,11 @@ struct cmd_ds_802_11_subscribe_event {
* Define data structure for CMD_802_11_SCAN * Define data structure for CMD_802_11_SCAN
*/ */
struct cmd_ds_802_11_scan { struct cmd_ds_802_11_scan {
u8 bsstype; struct cmd_header hdr;
u8 bssid[ETH_ALEN];
u8 tlvbuffer[1]; uint8_t bsstype;
uint8_t bssid[ETH_ALEN];
uint8_t tlvbuffer[0];
#if 0 #if 0
mrvlietypes_ssidparamset_t ssidParamSet; mrvlietypes_ssidparamset_t ssidParamSet;
mrvlietypes_chanlistparamset_t ChanListParamSet; mrvlietypes_chanlistparamset_t ChanListParamSet;
...@@ -185,9 +187,11 @@ struct cmd_ds_802_11_scan { ...@@ -185,9 +187,11 @@ struct cmd_ds_802_11_scan {
}; };
struct cmd_ds_802_11_scan_rsp { struct cmd_ds_802_11_scan_rsp {
struct cmd_header hdr;
__le16 bssdescriptsize; __le16 bssdescriptsize;
u8 nr_sets; uint8_t nr_sets;
u8 bssdesc_and_tlvbuffer[1]; uint8_t bssdesc_and_tlvbuffer[0];
}; };
struct cmd_ds_802_11_get_log { struct cmd_ds_802_11_get_log {
...@@ -691,8 +695,6 @@ struct cmd_ds_command { ...@@ -691,8 +695,6 @@ struct cmd_ds_command {
/* command Body */ /* command Body */
union { union {
struct cmd_ds_802_11_ps_mode psmode; struct cmd_ds_802_11_ps_mode psmode;
struct cmd_ds_802_11_scan scan;
struct cmd_ds_802_11_scan_rsp scanresp;
struct cmd_ds_mac_control macctrl; struct cmd_ds_mac_control macctrl;
struct cmd_ds_802_11_associate associate; struct cmd_ds_802_11_associate associate;
struct cmd_ds_802_11_deauthenticate deauth; struct cmd_ds_802_11_deauthenticate deauth;
......
...@@ -20,6 +20,7 @@ ...@@ -20,6 +20,7 @@
#include "dev.h" #include "dev.h"
#include "scan.h" #include "scan.h"
#include "join.h" #include "join.h"
#include "cmd.h"
//! Approximate amount of data needed to pass a scan result back to iwlist //! Approximate amount of data needed to pass a scan result back to iwlist
#define MAX_SCAN_CELL_SIZE (IW_EV_ADDR_LEN \ #define MAX_SCAN_CELL_SIZE (IW_EV_ADDR_LEN \
...@@ -39,10 +40,9 @@ ...@@ -39,10 +40,9 @@
//! Memory needed to store a max number/size SSID TLV for a firmware scan //! Memory needed to store a max number/size SSID TLV for a firmware scan
#define SSID_TLV_MAX_SIZE (1 * sizeof(struct mrvlietypes_ssidparamset)) #define SSID_TLV_MAX_SIZE (1 * sizeof(struct mrvlietypes_ssidparamset))
//! Maximum memory needed for a lbs_scan_cmd_config with all TLVs at max //! Maximum memory needed for a cmd_ds_802_11_scan with all TLVs at max
#define MAX_SCAN_CFG_ALLOC (sizeof(struct lbs_scan_cmd_config) \ #define MAX_SCAN_CFG_ALLOC (sizeof(struct cmd_ds_802_11_scan) \
+ CHAN_TLV_MAX_SIZE \ + CHAN_TLV_MAX_SIZE + SSID_TLV_MAX_SIZE)
+ SSID_TLV_MAX_SIZE)
//! The maximum number of channels the firmware can scan per command //! The maximum number of channels the firmware can scan per command
#define MRVDRV_MAX_CHANNELS_PER_SCAN 14 #define MRVDRV_MAX_CHANNELS_PER_SCAN 14
...@@ -64,8 +64,8 @@ ...@@ -64,8 +64,8 @@
static const u8 zeromac[ETH_ALEN] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; static const u8 zeromac[ETH_ALEN] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
static const u8 bcastmac[ETH_ALEN] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }; static const u8 bcastmac[ETH_ALEN] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
static int lbs_ret_80211_scan(struct lbs_private *priv, unsigned long dummy,
struct cmd_header *resp);
/*********************************************************************/ /*********************************************************************/
/* */ /* */
...@@ -492,24 +492,22 @@ static int lbs_scan_add_rates_tlv(u8 *tlv) ...@@ -492,24 +492,22 @@ static int lbs_scan_add_rates_tlv(u8 *tlv)
* Generate the CMD_802_11_SCAN command with the proper tlv * Generate the CMD_802_11_SCAN command with the proper tlv
* for a bunch of channels. * for a bunch of channels.
*/ */
static int lbs_do_scan(struct lbs_private *priv, static int lbs_do_scan(struct lbs_private *priv, uint8_t bsstype,
u8 bsstype, struct chanscanparamset *chan_list, int chan_count,
struct chanscanparamset *chan_list,
int chan_count,
const struct lbs_ioctl_user_scan_cfg *user_cfg) const struct lbs_ioctl_user_scan_cfg *user_cfg)
{ {
int ret = -ENOMEM; int ret = -ENOMEM;
struct lbs_scan_cmd_config *scan_cmd; struct cmd_ds_802_11_scan *scan_cmd;
u8 *tlv; /* pointer into our current, growing TLV storage area */ uint8_t *tlv; /* pointer into our current, growing TLV storage area */
lbs_deb_enter_args(LBS_DEB_SCAN, "bsstype %d, chanlist[].chan %d, " lbs_deb_enter_args(LBS_DEB_SCAN, "bsstype %d, chanlist[].chan %d, chan_count %d",
"chan_count %d",
bsstype, chan_list[0].channumber, chan_count); bsstype, chan_list[0].channumber, chan_count);
/* create the fixed part for scan command */ /* create the fixed part for scan command */
scan_cmd = kzalloc(MAX_SCAN_CFG_ALLOC, GFP_KERNEL); scan_cmd = kzalloc(MAX_SCAN_CFG_ALLOC, GFP_KERNEL);
if (scan_cmd == NULL) if (scan_cmd == NULL)
goto out; goto out;
tlv = scan_cmd->tlvbuffer; tlv = scan_cmd->tlvbuffer;
if (user_cfg) if (user_cfg)
memcpy(scan_cmd->bssid, user_cfg->bssid, ETH_ALEN); memcpy(scan_cmd->bssid, user_cfg->bssid, ETH_ALEN);
...@@ -523,13 +521,16 @@ static int lbs_do_scan(struct lbs_private *priv, ...@@ -523,13 +521,16 @@ static int lbs_do_scan(struct lbs_private *priv,
tlv += lbs_scan_add_rates_tlv(tlv); tlv += lbs_scan_add_rates_tlv(tlv);
/* This is the final data we are about to send */ /* This is the final data we are about to send */
scan_cmd->tlvbufferlen = tlv - scan_cmd->tlvbuffer; scan_cmd->hdr.size = cpu_to_le16(tlv - (uint8_t *)scan_cmd);
lbs_deb_hex(LBS_DEB_SCAN, "SCAN_CMD", (void *)scan_cmd, 1+6); lbs_deb_hex(LBS_DEB_SCAN, "SCAN_CMD", (void *)scan_cmd,
sizeof(*scan_cmd));
lbs_deb_hex(LBS_DEB_SCAN, "SCAN_TLV", scan_cmd->tlvbuffer, lbs_deb_hex(LBS_DEB_SCAN, "SCAN_TLV", scan_cmd->tlvbuffer,
scan_cmd->tlvbufferlen); tlv - scan_cmd->tlvbuffer);
ret = __lbs_cmd(priv, CMD_802_11_SCAN, &scan_cmd->hdr,
le16_to_cpu(scan_cmd->hdr.size),
lbs_ret_80211_scan, 0);
ret = lbs_prepare_and_send_command(priv, CMD_802_11_SCAN, 0,
CMD_OPTION_WAITFORRSP, 0, scan_cmd);
out: out:
kfree(scan_cmd); kfree(scan_cmd);
lbs_deb_leave_args(LBS_DEB_SCAN, "ret %d", ret); lbs_deb_leave_args(LBS_DEB_SCAN, "ret %d", ret);
...@@ -1484,44 +1485,6 @@ int lbs_get_scan(struct net_device *dev, struct iw_request_info *info, ...@@ -1484,44 +1485,6 @@ int lbs_get_scan(struct net_device *dev, struct iw_request_info *info,
/*********************************************************************/ /*********************************************************************/
/**
* @brief Prepare a scan command to be sent to the firmware
*
* Called via lbs_prepare_and_send_command(priv, CMD_802_11_SCAN, ...)
* from cmd.c
*
* Sends a fixed length data part (specifying the BSS type and BSSID filters)
* as well as a variable number/length of TLVs to the firmware.
*
* @param priv A pointer to struct lbs_private structure
* @param cmd A pointer to cmd_ds_command structure to be sent to
* firmware with the cmd_DS_801_11_SCAN structure
* @param pdata_buf Void pointer cast of a lbs_scan_cmd_config struct used
* to set the fields/TLVs for the command sent to firmware
*
* @return 0 or -1
*/
int lbs_cmd_80211_scan(struct lbs_private *priv,
struct cmd_ds_command *cmd, void *pdata_buf)
{
struct cmd_ds_802_11_scan *pscan = &cmd->params.scan;
struct lbs_scan_cmd_config *pscancfg = pdata_buf;
lbs_deb_enter(LBS_DEB_SCAN);
/* Set fixed field variables in scan command */
pscan->bsstype = pscancfg->bsstype;
memcpy(pscan->bssid, pscancfg->bssid, ETH_ALEN);
memcpy(pscan->tlvbuffer, pscancfg->tlvbuffer, pscancfg->tlvbufferlen);
/* size is equal to the sizeof(fixed portions) + the TLV len + header */
cmd->size = cpu_to_le16(sizeof(pscan->bsstype) + ETH_ALEN
+ pscancfg->tlvbufferlen + S_DS_GEN);
lbs_deb_leave(LBS_DEB_SCAN);
return 0;
}
/** /**
* @brief This function handles the command response of scan * @brief This function handles the command response of scan
* *
...@@ -1548,13 +1511,14 @@ int lbs_cmd_80211_scan(struct lbs_private *priv, ...@@ -1548,13 +1511,14 @@ int lbs_cmd_80211_scan(struct lbs_private *priv,
* *
* @return 0 or -1 * @return 0 or -1
*/ */
int lbs_ret_80211_scan(struct lbs_private *priv, struct cmd_ds_command *resp) static int lbs_ret_80211_scan(struct lbs_private *priv, unsigned long dummy,
struct cmd_header *resp)
{ {
struct cmd_ds_802_11_scan_rsp *pscan; struct cmd_ds_802_11_scan_rsp *scanresp = (void *)resp;
struct bss_descriptor * iter_bss; struct bss_descriptor * iter_bss;
struct bss_descriptor * safe; struct bss_descriptor * safe;
u8 *pbssinfo; uint8_t *bssinfo;
u16 scanrespsize; uint16_t scanrespsize;
int bytesleft; int bytesleft;
int idx; int idx;
int tlvbufsize; int tlvbufsize;
...@@ -1571,48 +1535,45 @@ int lbs_ret_80211_scan(struct lbs_private *priv, struct cmd_ds_command *resp) ...@@ -1571,48 +1535,45 @@ int lbs_ret_80211_scan(struct lbs_private *priv, struct cmd_ds_command *resp)
clear_bss_descriptor(iter_bss); clear_bss_descriptor(iter_bss);
} }
pscan = &resp->params.scanresp; if (scanresp->nr_sets > MAX_NETWORK_COUNT) {
lbs_deb_scan("SCAN_RESP: too many scan results (%d, max %d)\n",
if (pscan->nr_sets > MAX_NETWORK_COUNT) { scanresp->nr_sets, MAX_NETWORK_COUNT);
lbs_deb_scan(
"SCAN_RESP: too many scan results (%d, max %d)!!\n",
pscan->nr_sets, MAX_NETWORK_COUNT);
ret = -1; ret = -1;
goto done; goto done;
} }
bytesleft = le16_to_cpu(pscan->bssdescriptsize); bytesleft = le16_to_cpu(scanresp->bssdescriptsize);
lbs_deb_scan("SCAN_RESP: bssdescriptsize %d\n", bytesleft); lbs_deb_scan("SCAN_RESP: bssdescriptsize %d\n", bytesleft);
scanrespsize = le16_to_cpu(resp->size); scanrespsize = le16_to_cpu(resp->size);
lbs_deb_scan("SCAN_RESP: scan results %d\n", pscan->nr_sets); lbs_deb_scan("SCAN_RESP: scan results %d\n", scanresp->nr_sets);
pbssinfo = pscan->bssdesc_and_tlvbuffer; bssinfo = scanresp->bssdesc_and_tlvbuffer;
/* The size of the TLV buffer is equal to the entire command response /* The size of the TLV buffer is equal to the entire command response
* size (scanrespsize) minus the fixed fields (sizeof()'s), the * size (scanrespsize) minus the fixed fields (sizeof()'s), the
* BSS Descriptions (bssdescriptsize as bytesLef) and the command * BSS Descriptions (bssdescriptsize as bytesLef) and the command
* response header (S_DS_GEN) * response header (S_DS_GEN)
*/ */
tlvbufsize = scanrespsize - (bytesleft + sizeof(pscan->bssdescriptsize) tlvbufsize = scanrespsize - (bytesleft + sizeof(scanresp->bssdescriptsize)
+ sizeof(pscan->nr_sets) + sizeof(scanresp->nr_sets)
+ S_DS_GEN); + S_DS_GEN);
/* /*
* Process each scan response returned (pscan->nr_sets). Save * Process each scan response returned (scanresp->nr_sets). Save
* the information in the newbssentry and then insert into the * the information in the newbssentry and then insert into the
* driver scan table either as an update to an existing entry * driver scan table either as an update to an existing entry
* or as an addition at the end of the table * or as an addition at the end of the table
*/ */
for (idx = 0; idx < pscan->nr_sets && bytesleft; idx++) { for (idx = 0; idx < scanresp->nr_sets && bytesleft; idx++) {
struct bss_descriptor new; struct bss_descriptor new;
struct bss_descriptor * found = NULL; struct bss_descriptor *found = NULL;
struct bss_descriptor * oldest = NULL; struct bss_descriptor *oldest = NULL;
DECLARE_MAC_BUF(mac); DECLARE_MAC_BUF(mac);
/* Process the data fields and IEs returned for this BSS */ /* Process the data fields and IEs returned for this BSS */
memset(&new, 0, sizeof (struct bss_descriptor)); memset(&new, 0, sizeof (struct bss_descriptor));
if (lbs_process_bss(&new, &pbssinfo, &bytesleft) != 0) { if (lbs_process_bss(&new, &bssinfo, &bytesleft) != 0) {
/* error parsing the scan response, skipped */ /* error parsing the scan response, skipped */
lbs_deb_scan("SCAN_RESP: process_bss returned ERROR\n"); lbs_deb_scan("SCAN_RESP: process_bss returned ERROR\n");
continue; continue;
...@@ -1647,8 +1608,7 @@ int lbs_ret_80211_scan(struct lbs_private *priv, struct cmd_ds_command *resp) ...@@ -1647,8 +1608,7 @@ int lbs_ret_80211_scan(struct lbs_private *priv, struct cmd_ds_command *resp)
continue; continue;
} }
lbs_deb_scan("SCAN_RESP: BSSID %s\n", lbs_deb_scan("SCAN_RESP: BSSID %s\n", print_mac(mac, new.bssid));
print_mac(mac, new.bssid));
/* Copy the locally created newbssentry to the scan table */ /* Copy the locally created newbssentry to the scan table */
memcpy(found, &new, offsetof(struct bss_descriptor, list)); memcpy(found, &new, offsetof(struct bss_descriptor, list));
......
...@@ -17,56 +17,15 @@ ...@@ -17,56 +17,15 @@
*/ */
#define LBS_IOCTL_USER_SCAN_CHAN_MAX 50 #define LBS_IOCTL_USER_SCAN_CHAN_MAX 50
//! Infrastructure BSS scan type in lbs_scan_cmd_config //! Infrastructure BSS scan type in cmd_ds_802_11_scan
#define LBS_SCAN_BSS_TYPE_BSS 1 #define LBS_SCAN_BSS_TYPE_BSS 1
//! Adhoc BSS scan type in lbs_scan_cmd_config //! Adhoc BSS scan type in cmd_ds_802_11_scan
#define LBS_SCAN_BSS_TYPE_IBSS 2 #define LBS_SCAN_BSS_TYPE_IBSS 2
//! Adhoc or Infrastructure BSS scan type in lbs_scan_cmd_config, no filter //! Adhoc or Infrastructure BSS scan type in cmd_ds_802_11_scan, no filter
#define LBS_SCAN_BSS_TYPE_ANY 3 #define LBS_SCAN_BSS_TYPE_ANY 3
/**
* @brief Structure used internally in the wlan driver to configure a scan.
*
* Sent to the command processing module to configure the firmware
* scan command prepared by lbs_cmd_80211_scan.
*
* @sa lbs_scan_networks
*
*/
struct lbs_scan_cmd_config {
/**
* @brief BSS type to be sent in the firmware command
*
* Field can be used to restrict the types of networks returned in the
* scan. valid settings are:
*
* - LBS_SCAN_BSS_TYPE_BSS (infrastructure)
* - LBS_SCAN_BSS_TYPE_IBSS (adhoc)
* - LBS_SCAN_BSS_TYPE_ANY (unrestricted, adhoc and infrastructure)
*/
u8 bsstype;
/**
* @brief Specific BSSID used to filter scan results in the firmware
*/
u8 bssid[ETH_ALEN];
/**
* @brief length of TLVs sent in command starting at tlvBuffer
*/
int tlvbufferlen;
/**
* @brief SSID TLV(s) and ChanList TLVs to be sent in the firmware command
*
* @sa TLV_TYPE_CHANLIST, mrvlietypes_chanlistparamset_t
* @sa TLV_TYPE_SSID, mrvlietypes_ssidparamset_t
*/
u8 tlvbuffer[1]; //!< SSID TLV(s) and ChanList TLVs are stored here
};
/** /**
* @brief IOCTL channel sub-structure sent in lbs_ioctl_user_scan_cfg * @brief IOCTL channel sub-structure sent in lbs_ioctl_user_scan_cfg
* *
...@@ -179,13 +138,6 @@ int lbs_find_best_network_ssid(struct lbs_private *priv, u8 *out_ssid, ...@@ -179,13 +138,6 @@ int lbs_find_best_network_ssid(struct lbs_private *priv, u8 *out_ssid,
int lbs_send_specific_ssid_scan(struct lbs_private *priv, u8 *ssid, int lbs_send_specific_ssid_scan(struct lbs_private *priv, u8 *ssid,
u8 ssid_len, u8 clear_ssid); u8 ssid_len, u8 clear_ssid);
int lbs_cmd_80211_scan(struct lbs_private *priv,
struct cmd_ds_command *cmd,
void *pdata_buf);
int lbs_ret_80211_scan(struct lbs_private *priv,
struct cmd_ds_command *resp);
int lbs_scan_networks(struct lbs_private *priv, int lbs_scan_networks(struct lbs_private *priv,
const struct lbs_ioctl_user_scan_cfg *puserscanin, const struct lbs_ioctl_user_scan_cfg *puserscanin,
int full_scan); int full_scan);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment