aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net
diff options
context:
space:
mode:
authorJohannes Berg <johannes.berg@intel.com>2010-07-31 11:34:10 -0400
committerJohn W. Linville <linville@tuxdriver.com>2010-08-16 15:26:38 -0400
commita85d7cca1204f2dba86d2f61693f0fe8c48f0fa5 (patch)
tree78613f0415c0feca672b382d9c98f1c69c25b7e9 /drivers/net
parent07d4f1ad2c21273f0f89a3d2202d22fcaf901439 (diff)
iwlwifi: track IBSS manager status
Only the IBSS manager, ie. the station that sent the IBSS beacon last, should be replying to probe responses. This requires implementing the mac80211 tx_last_beacon callback, which we can do thanks to the ucode beacon notification. Signed-off-by: Johannes Berg <johannes.berg@intel.com> Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net')
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-agn.c5
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-commands.h5
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-core.c8
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-core.h1
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-dev.h3
-rw-r--r--drivers/net/wireless/iwlwifi/iwl3945-base.c5
6 files changed, 25 insertions, 2 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl-agn.c b/drivers/net/wireless/iwlwifi/iwl-agn.c
index 8684f2639716..43e078b87378 100644
--- a/drivers/net/wireless/iwlwifi/iwl-agn.c
+++ b/drivers/net/wireless/iwlwifi/iwl-agn.c
@@ -764,10 +764,10 @@ static void iwl_bg_ucode_trace(unsigned long data)
764static void iwl_rx_beacon_notif(struct iwl_priv *priv, 764static void iwl_rx_beacon_notif(struct iwl_priv *priv,
765 struct iwl_rx_mem_buffer *rxb) 765 struct iwl_rx_mem_buffer *rxb)
766{ 766{
767#ifdef CONFIG_IWLWIFI_DEBUG
768 struct iwl_rx_packet *pkt = rxb_addr(rxb); 767 struct iwl_rx_packet *pkt = rxb_addr(rxb);
769 struct iwl4965_beacon_notif *beacon = 768 struct iwl4965_beacon_notif *beacon =
770 (struct iwl4965_beacon_notif *)pkt->u.raw; 769 (struct iwl4965_beacon_notif *)pkt->u.raw;
770#ifdef CONFIG_IWLWIFI_DEBUG
771 u8 rate = iwl_hw_get_rate(beacon->beacon_notify_hdr.rate_n_flags); 771 u8 rate = iwl_hw_get_rate(beacon->beacon_notify_hdr.rate_n_flags);
772 772
773 IWL_DEBUG_RX(priv, "beacon status %x retries %d iss %d " 773 IWL_DEBUG_RX(priv, "beacon status %x retries %d iss %d "
@@ -779,6 +779,8 @@ static void iwl_rx_beacon_notif(struct iwl_priv *priv,
779 le32_to_cpu(beacon->low_tsf), rate); 779 le32_to_cpu(beacon->low_tsf), rate);
780#endif 780#endif
781 781
782 priv->ibss_manager = le32_to_cpu(beacon->ibss_mgr_status);
783
782 if ((priv->iw_mode == NL80211_IFTYPE_AP) && 784 if ((priv->iw_mode == NL80211_IFTYPE_AP) &&
783 (!test_bit(STATUS_EXIT_PENDING, &priv->status))) 785 (!test_bit(STATUS_EXIT_PENDING, &priv->status)))
784 queue_work(priv->workqueue, &priv->beacon_update); 786 queue_work(priv->workqueue, &priv->beacon_update);
@@ -3881,6 +3883,7 @@ static struct ieee80211_ops iwl_hw_ops = {
3881 .sta_remove = iwl_mac_sta_remove, 3883 .sta_remove = iwl_mac_sta_remove,
3882 .channel_switch = iwl_mac_channel_switch, 3884 .channel_switch = iwl_mac_channel_switch,
3883 .flush = iwl_mac_flush, 3885 .flush = iwl_mac_flush,
3886 .tx_last_beacon = iwl_mac_tx_last_beacon,
3884}; 3887};
3885 3888
3886static void iwl_hw_detect(struct iwl_priv *priv) 3889static void iwl_hw_detect(struct iwl_priv *priv)
diff --git a/drivers/net/wireless/iwlwifi/iwl-commands.h b/drivers/net/wireless/iwlwifi/iwl-commands.h
index 60725a5c1b69..9435fd934d5e 100644
--- a/drivers/net/wireless/iwlwifi/iwl-commands.h
+++ b/drivers/net/wireless/iwlwifi/iwl-commands.h
@@ -2919,6 +2919,11 @@ struct iwl_scancomplete_notification {
2919 * 2919 *
2920 *****************************************************************************/ 2920 *****************************************************************************/
2921 2921
2922enum iwl_ibss_manager {
2923 IWL_NOT_IBSS_MANAGER = 0,
2924 IWL_IBSS_MANAGER = 1,
2925};
2926
2922/* 2927/*
2923 * BEACON_NOTIFICATION = 0x90 (notification only, not a command) 2928 * BEACON_NOTIFICATION = 0x90 (notification only, not a command)
2924 */ 2929 */
diff --git a/drivers/net/wireless/iwlwifi/iwl-core.c b/drivers/net/wireless/iwlwifi/iwl-core.c
index 78882f9ab459..ccdf3c02bb3c 100644
--- a/drivers/net/wireless/iwlwifi/iwl-core.c
+++ b/drivers/net/wireless/iwlwifi/iwl-core.c
@@ -1666,6 +1666,14 @@ int iwl_mac_conf_tx(struct ieee80211_hw *hw, u16 queue,
1666} 1666}
1667EXPORT_SYMBOL(iwl_mac_conf_tx); 1667EXPORT_SYMBOL(iwl_mac_conf_tx);
1668 1668
1669int iwl_mac_tx_last_beacon(struct ieee80211_hw *hw)
1670{
1671 struct iwl_priv *priv = hw->priv;
1672
1673 return priv->ibss_manager == IWL_IBSS_MANAGER;
1674}
1675EXPORT_SYMBOL_GPL(iwl_mac_tx_last_beacon);
1676
1669static void iwl_ht_conf(struct iwl_priv *priv, 1677static void iwl_ht_conf(struct iwl_priv *priv,
1670 struct ieee80211_vif *vif) 1678 struct ieee80211_vif *vif)
1671{ 1679{
diff --git a/drivers/net/wireless/iwlwifi/iwl-core.h b/drivers/net/wireless/iwlwifi/iwl-core.h
index cca7428c238f..7f1aaf206078 100644
--- a/drivers/net/wireless/iwlwifi/iwl-core.h
+++ b/drivers/net/wireless/iwlwifi/iwl-core.h
@@ -350,6 +350,7 @@ struct ieee80211_hw *iwl_alloc_all(struct iwl_cfg *cfg,
350void iwl_activate_qos(struct iwl_priv *priv); 350void iwl_activate_qos(struct iwl_priv *priv);
351int iwl_mac_conf_tx(struct ieee80211_hw *hw, u16 queue, 351int iwl_mac_conf_tx(struct ieee80211_hw *hw, u16 queue,
352 const struct ieee80211_tx_queue_params *params); 352 const struct ieee80211_tx_queue_params *params);
353int iwl_mac_tx_last_beacon(struct ieee80211_hw *hw);
353void iwl_set_rxon_hwcrypto(struct iwl_priv *priv, int hw_decrypt); 354void iwl_set_rxon_hwcrypto(struct iwl_priv *priv, int hw_decrypt);
354int iwl_check_rxon_cmd(struct iwl_priv *priv); 355int iwl_check_rxon_cmd(struct iwl_priv *priv);
355int iwl_full_rxon_required(struct iwl_priv *priv); 356int iwl_full_rxon_required(struct iwl_priv *priv);
diff --git a/drivers/net/wireless/iwlwifi/iwl-dev.h b/drivers/net/wireless/iwlwifi/iwl-dev.h
index f35bcad56e36..599635547021 100644
--- a/drivers/net/wireless/iwlwifi/iwl-dev.h
+++ b/drivers/net/wireless/iwlwifi/iwl-dev.h
@@ -1109,6 +1109,9 @@ struct iwl_priv {
1109 u32 ucode_beacon_time; 1109 u32 ucode_beacon_time;
1110 int missed_beacon_threshold; 1110 int missed_beacon_threshold;
1111 1111
1112 /* track IBSS manager (last beacon) status */
1113 u32 ibss_manager;
1114
1112 /* storing the jiffies when the plcp error rate is received */ 1115 /* storing the jiffies when the plcp error rate is received */
1113 unsigned long plcp_jiffies; 1116 unsigned long plcp_jiffies;
1114 1117
diff --git a/drivers/net/wireless/iwlwifi/iwl3945-base.c b/drivers/net/wireless/iwlwifi/iwl3945-base.c
index de6d592af748..b14eaf91c7c2 100644
--- a/drivers/net/wireless/iwlwifi/iwl3945-base.c
+++ b/drivers/net/wireless/iwlwifi/iwl3945-base.c
@@ -814,9 +814,9 @@ static void iwl3945_bg_beacon_update(struct work_struct *work)
814static void iwl3945_rx_beacon_notif(struct iwl_priv *priv, 814static void iwl3945_rx_beacon_notif(struct iwl_priv *priv,
815 struct iwl_rx_mem_buffer *rxb) 815 struct iwl_rx_mem_buffer *rxb)
816{ 816{
817#ifdef CONFIG_IWLWIFI_DEBUG
818 struct iwl_rx_packet *pkt = rxb_addr(rxb); 817 struct iwl_rx_packet *pkt = rxb_addr(rxb);
819 struct iwl3945_beacon_notif *beacon = &(pkt->u.beacon_status); 818 struct iwl3945_beacon_notif *beacon = &(pkt->u.beacon_status);
819#ifdef CONFIG_IWLWIFI_DEBUG
820 u8 rate = beacon->beacon_notify_hdr.rate; 820 u8 rate = beacon->beacon_notify_hdr.rate;
821 821
822 IWL_DEBUG_RX(priv, "beacon status %x retries %d iss %d " 822 IWL_DEBUG_RX(priv, "beacon status %x retries %d iss %d "
@@ -828,6 +828,8 @@ static void iwl3945_rx_beacon_notif(struct iwl_priv *priv,
828 le32_to_cpu(beacon->low_tsf), rate); 828 le32_to_cpu(beacon->low_tsf), rate);
829#endif 829#endif
830 830
831 priv->ibss_manager = le32_to_cpu(beacon->ibss_mgr_status);
832
831 if ((priv->iw_mode == NL80211_IFTYPE_AP) && 833 if ((priv->iw_mode == NL80211_IFTYPE_AP) &&
832 (!test_bit(STATUS_EXIT_PENDING, &priv->status))) 834 (!test_bit(STATUS_EXIT_PENDING, &priv->status)))
833 queue_work(priv->workqueue, &priv->beacon_update); 835 queue_work(priv->workqueue, &priv->beacon_update);
@@ -3803,6 +3805,7 @@ static struct ieee80211_ops iwl3945_hw_ops = {
3803 .hw_scan = iwl_mac_hw_scan, 3805 .hw_scan = iwl_mac_hw_scan,
3804 .sta_add = iwl3945_mac_sta_add, 3806 .sta_add = iwl3945_mac_sta_add,
3805 .sta_remove = iwl_mac_sta_remove, 3807 .sta_remove = iwl_mac_sta_remove,
3808 .tx_last_beacon = iwl_mac_tx_last_beacon,
3806}; 3809};
3807 3810
3808static int iwl3945_init_drv(struct iwl_priv *priv) 3811static int iwl3945_init_drv(struct iwl_priv *priv)