wl1271: Update filters properly
This patch adds support for the filters configured by the mac80211 stack. Signed-off-by: Juuso Oikarinen <juuso.oikarinen@nokia.com> Reviewed-by: Teemu Paasikivi <ext-teemu.3.paasikivi@nokia.com> Signed-off-by: Luciano Coelho <luciano.coelho@nokia.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
parent
801a673ed1
commit
14b228a0d4
|
@ -54,6 +54,7 @@ enum {
|
||||||
DEBUG_CMD = BIT(12),
|
DEBUG_CMD = BIT(12),
|
||||||
DEBUG_ACX = BIT(13),
|
DEBUG_ACX = BIT(13),
|
||||||
DEBUG_SDIO = BIT(14),
|
DEBUG_SDIO = BIT(14),
|
||||||
|
DEBUG_FILTERS = BIT(15),
|
||||||
DEBUG_ALL = ~0,
|
DEBUG_ALL = ~0,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -458,6 +459,7 @@ struct wl1271 {
|
||||||
/* Default key (for WEP) */
|
/* Default key (for WEP) */
|
||||||
u32 default_key;
|
u32 default_key;
|
||||||
|
|
||||||
|
unsigned int filters;
|
||||||
unsigned int rx_config;
|
unsigned int rx_config;
|
||||||
unsigned int rx_filter;
|
unsigned int rx_filter;
|
||||||
|
|
||||||
|
|
|
@ -281,15 +281,6 @@ int wl1271_cmd_join(struct wl1271 *wl, u8 bss_type)
|
||||||
join->rx_filter_options = cpu_to_le32(wl->rx_filter);
|
join->rx_filter_options = cpu_to_le32(wl->rx_filter);
|
||||||
join->bss_type = bss_type;
|
join->bss_type = bss_type;
|
||||||
|
|
||||||
/*
|
|
||||||
* FIXME: disable temporarily all filters because after commit
|
|
||||||
* 9cef8737 "mac80211: fix managed mode BSSID handling" broke
|
|
||||||
* association. The filter logic needs to be implemented properly
|
|
||||||
* and once that is done, this hack can be removed.
|
|
||||||
*/
|
|
||||||
join->rx_config_options = cpu_to_le32(0);
|
|
||||||
join->rx_filter_options = cpu_to_le32(WL1271_DEFAULT_RX_FILTER);
|
|
||||||
|
|
||||||
if (wl->band == IEEE80211_BAND_2GHZ)
|
if (wl->band == IEEE80211_BAND_2GHZ)
|
||||||
join->basic_rate_set = cpu_to_le32(CONF_HW_BIT_RATE_1MBPS |
|
join->basic_rate_set = cpu_to_le32(CONF_HW_BIT_RATE_1MBPS |
|
||||||
CONF_HW_BIT_RATE_2MBPS |
|
CONF_HW_BIT_RATE_2MBPS |
|
||||||
|
|
|
@ -1080,6 +1080,7 @@ static void wl1271_op_remove_interface(struct ieee80211_hw *hw,
|
||||||
wl->sta_rate_set = 0;
|
wl->sta_rate_set = 0;
|
||||||
wl->flags = 0;
|
wl->flags = 0;
|
||||||
wl->vif = NULL;
|
wl->vif = NULL;
|
||||||
|
wl->filters = 0;
|
||||||
|
|
||||||
for (i = 0; i < NUM_TX_QUEUES; i++)
|
for (i = 0; i < NUM_TX_QUEUES; i++)
|
||||||
wl->tx_blocks_freed[i] = 0;
|
wl->tx_blocks_freed[i] = 0;
|
||||||
|
@ -1088,6 +1089,40 @@ static void wl1271_op_remove_interface(struct ieee80211_hw *hw,
|
||||||
mutex_unlock(&wl->mutex);
|
mutex_unlock(&wl->mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void wl1271_configure_filters(struct wl1271 *wl, unsigned int filters)
|
||||||
|
{
|
||||||
|
wl->rx_config = WL1271_DEFAULT_RX_CONFIG;
|
||||||
|
wl->rx_filter = WL1271_DEFAULT_RX_FILTER;
|
||||||
|
|
||||||
|
/* combine requested filters with current filter config */
|
||||||
|
filters = wl->filters | filters;
|
||||||
|
|
||||||
|
wl1271_debug(DEBUG_FILTERS, "RX filters set: ");
|
||||||
|
|
||||||
|
if (filters & FIF_PROMISC_IN_BSS) {
|
||||||
|
wl1271_debug(DEBUG_FILTERS, " - FIF_PROMISC_IN_BSS");
|
||||||
|
wl->rx_config &= ~CFG_UNI_FILTER_EN;
|
||||||
|
wl->rx_config |= CFG_BSSID_FILTER_EN;
|
||||||
|
}
|
||||||
|
if (filters & FIF_BCN_PRBRESP_PROMISC) {
|
||||||
|
wl1271_debug(DEBUG_FILTERS, " - FIF_BCN_PRBRESP_PROMISC");
|
||||||
|
wl->rx_config &= ~CFG_BSSID_FILTER_EN;
|
||||||
|
wl->rx_config &= ~CFG_SSID_FILTER_EN;
|
||||||
|
}
|
||||||
|
if (filters & FIF_OTHER_BSS) {
|
||||||
|
wl1271_debug(DEBUG_FILTERS, " - FIF_OTHER_BSS");
|
||||||
|
wl->rx_config &= ~CFG_BSSID_FILTER_EN;
|
||||||
|
}
|
||||||
|
if (filters & FIF_CONTROL) {
|
||||||
|
wl1271_debug(DEBUG_FILTERS, " - FIF_CONTROL");
|
||||||
|
wl->rx_filter |= CFG_RX_CTL_EN;
|
||||||
|
}
|
||||||
|
if (filters & FIF_FCSFAIL) {
|
||||||
|
wl1271_debug(DEBUG_FILTERS, " - FIF_FCSFAIL");
|
||||||
|
wl->rx_filter |= CFG_RX_FCS_ERROR;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int wl1271_join_channel(struct wl1271 *wl, int channel)
|
static int wl1271_join_channel(struct wl1271 *wl, int channel)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -1095,12 +1130,12 @@ static int wl1271_join_channel(struct wl1271 *wl, int channel)
|
||||||
static const u8 dummy_bssid[ETH_ALEN] = { 0x0b, 0xad, 0xde,
|
static const u8 dummy_bssid[ETH_ALEN] = { 0x0b, 0xad, 0xde,
|
||||||
0xad, 0xbe, 0xef };
|
0xad, 0xbe, 0xef };
|
||||||
|
|
||||||
/* disable mac filter, so we hear everything */
|
|
||||||
wl->rx_config &= ~CFG_BSSID_FILTER_EN;
|
|
||||||
|
|
||||||
wl->channel = channel;
|
wl->channel = channel;
|
||||||
memcpy(wl->bssid, dummy_bssid, ETH_ALEN);
|
memcpy(wl->bssid, dummy_bssid, ETH_ALEN);
|
||||||
|
|
||||||
|
/* pass through frames from all BSS */
|
||||||
|
wl1271_configure_filters(wl, FIF_OTHER_BSS);
|
||||||
|
|
||||||
/* the dummy join is performed always with STATION BSS type to allow
|
/* the dummy join is performed always with STATION BSS type to allow
|
||||||
also ad-hoc mode to listen to the surroundings without sending any
|
also ad-hoc mode to listen to the surroundings without sending any
|
||||||
beacons yet. */
|
beacons yet. */
|
||||||
|
@ -1126,7 +1161,9 @@ static int wl1271_unjoin_channel(struct wl1271 *wl)
|
||||||
clear_bit(WL1271_FLAG_JOINED, &wl->flags);
|
clear_bit(WL1271_FLAG_JOINED, &wl->flags);
|
||||||
wl->channel = 0;
|
wl->channel = 0;
|
||||||
memset(wl->bssid, 0, ETH_ALEN);
|
memset(wl->bssid, 0, ETH_ALEN);
|
||||||
wl->rx_config = WL1271_DEFAULT_RX_CONFIG;
|
|
||||||
|
/* stop filterting packets based on bssid */
|
||||||
|
wl1271_configure_filters(wl, FIF_OTHER_BSS);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -1299,14 +1336,14 @@ static void wl1271_op_configure_filter(struct ieee80211_hw *hw,
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out_sleep;
|
goto out_sleep;
|
||||||
|
|
||||||
kfree(fp);
|
|
||||||
|
|
||||||
/* FIXME: We still need to set our filters properly */
|
|
||||||
|
|
||||||
/* determine, whether supported filter values have changed */
|
/* determine, whether supported filter values have changed */
|
||||||
if (changed == 0)
|
if (changed == 0)
|
||||||
goto out_sleep;
|
goto out_sleep;
|
||||||
|
|
||||||
|
/* configure filters */
|
||||||
|
wl->filters = *total;
|
||||||
|
wl1271_configure_filters(wl, 0);
|
||||||
|
|
||||||
/* apply configured filters */
|
/* apply configured filters */
|
||||||
ret = wl1271_acx_rx_config(wl, wl->rx_config, wl->rx_filter);
|
ret = wl1271_acx_rx_config(wl, wl->rx_config, wl->rx_filter);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
@ -1317,6 +1354,7 @@ out_sleep:
|
||||||
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&wl->mutex);
|
mutex_unlock(&wl->mutex);
|
||||||
|
kfree(fp);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int wl1271_op_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
static int wl1271_op_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
||||||
|
@ -1580,7 +1618,6 @@ static void wl1271_op_bss_info_changed(struct ieee80211_hw *hw,
|
||||||
* and enable the BSSID filter
|
* and enable the BSSID filter
|
||||||
*/
|
*/
|
||||||
memcmp(wl->bssid, bss_conf->bssid, ETH_ALEN)) {
|
memcmp(wl->bssid, bss_conf->bssid, ETH_ALEN)) {
|
||||||
wl->rx_config |= CFG_BSSID_FILTER_EN;
|
|
||||||
memcpy(wl->bssid, bss_conf->bssid, ETH_ALEN);
|
memcpy(wl->bssid, bss_conf->bssid, ETH_ALEN);
|
||||||
ret = wl1271_cmd_build_null_data(wl);
|
ret = wl1271_cmd_build_null_data(wl);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
@ -1589,6 +1626,9 @@ static void wl1271_op_bss_info_changed(struct ieee80211_hw *hw,
|
||||||
goto out_sleep;
|
goto out_sleep;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* filter out all packets not from this BSSID */
|
||||||
|
wl1271_configure_filters(wl, 0);
|
||||||
|
|
||||||
/* Need to update the BSSID (for filtering etc) */
|
/* Need to update the BSSID (for filtering etc) */
|
||||||
do_join = true;
|
do_join = true;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue