diff options
Diffstat (limited to 'net/mac80211/mlme.c')
-rw-r--r-- | net/mac80211/mlme.c | 112 |
1 files changed, 45 insertions, 67 deletions
diff --git a/net/mac80211/mlme.c b/net/mac80211/mlme.c index 64d710a88b86..61d7f81bf45e 100644 --- a/net/mac80211/mlme.c +++ b/net/mac80211/mlme.c | |||
@@ -2406,8 +2406,6 @@ static int ieee80211_sta_join_ibss(struct net_device *dev, | |||
2406 | int res, rates, i, j; | 2406 | int res, rates, i, j; |
2407 | struct sk_buff *skb; | 2407 | struct sk_buff *skb; |
2408 | struct ieee80211_mgmt *mgmt; | 2408 | struct ieee80211_mgmt *mgmt; |
2409 | struct ieee80211_tx_info *control; | ||
2410 | struct rate_selection ratesel; | ||
2411 | u8 *pos; | 2409 | u8 *pos; |
2412 | struct ieee80211_sub_if_data *sdata; | 2410 | struct ieee80211_sub_if_data *sdata; |
2413 | struct ieee80211_supported_band *sband; | 2411 | struct ieee80211_supported_band *sband; |
@@ -2425,7 +2423,7 @@ static int ieee80211_sta_join_ibss(struct net_device *dev, | |||
2425 | local->ops->reset_tsf(local_to_hw(local)); | 2423 | local->ops->reset_tsf(local_to_hw(local)); |
2426 | } | 2424 | } |
2427 | memcpy(ifsta->bssid, bss->bssid, ETH_ALEN); | 2425 | memcpy(ifsta->bssid, bss->bssid, ETH_ALEN); |
2428 | res = ieee80211_if_config(dev); | 2426 | res = ieee80211_if_config(sdata, IEEE80211_IFCC_BSSID); |
2429 | if (res) | 2427 | if (res) |
2430 | return res; | 2428 | return res; |
2431 | 2429 | ||
@@ -2439,19 +2437,16 @@ static int ieee80211_sta_join_ibss(struct net_device *dev, | |||
2439 | if (res) | 2437 | if (res) |
2440 | return res; | 2438 | return res; |
2441 | 2439 | ||
2442 | /* Set beacon template */ | 2440 | /* Build IBSS probe response */ |
2443 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + 400); | 2441 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + 400); |
2444 | do { | 2442 | if (skb) { |
2445 | if (!skb) | ||
2446 | break; | ||
2447 | |||
2448 | skb_reserve(skb, local->hw.extra_tx_headroom); | 2443 | skb_reserve(skb, local->hw.extra_tx_headroom); |
2449 | 2444 | ||
2450 | mgmt = (struct ieee80211_mgmt *) | 2445 | mgmt = (struct ieee80211_mgmt *) |
2451 | skb_put(skb, 24 + sizeof(mgmt->u.beacon)); | 2446 | skb_put(skb, 24 + sizeof(mgmt->u.beacon)); |
2452 | memset(mgmt, 0, 24 + sizeof(mgmt->u.beacon)); | 2447 | memset(mgmt, 0, 24 + sizeof(mgmt->u.beacon)); |
2453 | mgmt->frame_control = IEEE80211_FC(IEEE80211_FTYPE_MGMT, | 2448 | mgmt->frame_control = IEEE80211_FC(IEEE80211_FTYPE_MGMT, |
2454 | IEEE80211_STYPE_BEACON); | 2449 | IEEE80211_STYPE_PROBE_RESP); |
2455 | memset(mgmt->da, 0xff, ETH_ALEN); | 2450 | memset(mgmt->da, 0xff, ETH_ALEN); |
2456 | memcpy(mgmt->sa, dev->dev_addr, ETH_ALEN); | 2451 | memcpy(mgmt->sa, dev->dev_addr, ETH_ALEN); |
2457 | memcpy(mgmt->bssid, ifsta->bssid, ETH_ALEN); | 2452 | memcpy(mgmt->bssid, ifsta->bssid, ETH_ALEN); |
@@ -2495,61 +2490,22 @@ static int ieee80211_sta_join_ibss(struct net_device *dev, | |||
2495 | memcpy(pos, &bss->supp_rates[8], rates); | 2490 | memcpy(pos, &bss->supp_rates[8], rates); |
2496 | } | 2491 | } |
2497 | 2492 | ||
2498 | control = IEEE80211_SKB_CB(skb); | 2493 | ifsta->probe_resp = skb; |
2499 | |||
2500 | rate_control_get_rate(dev, sband, skb, &ratesel); | ||
2501 | if (ratesel.rate_idx < 0) { | ||
2502 | printk(KERN_DEBUG "%s: Failed to determine TX rate " | ||
2503 | "for IBSS beacon\n", dev->name); | ||
2504 | break; | ||
2505 | } | ||
2506 | control->control.vif = &sdata->vif; | ||
2507 | control->tx_rate_idx = ratesel.rate_idx; | ||
2508 | if (sdata->bss_conf.use_short_preamble && | ||
2509 | sband->bitrates[ratesel.rate_idx].flags & IEEE80211_RATE_SHORT_PREAMBLE) | ||
2510 | control->flags |= IEEE80211_TX_CTL_SHORT_PREAMBLE; | ||
2511 | control->antenna_sel_tx = local->hw.conf.antenna_sel_tx; | ||
2512 | control->flags |= IEEE80211_TX_CTL_NO_ACK; | ||
2513 | control->flags |= IEEE80211_TX_CTL_DO_NOT_ENCRYPT; | ||
2514 | control->control.retry_limit = 1; | ||
2515 | |||
2516 | ifsta->probe_resp = skb_copy(skb, GFP_ATOMIC); | ||
2517 | if (ifsta->probe_resp) { | ||
2518 | mgmt = (struct ieee80211_mgmt *) | ||
2519 | ifsta->probe_resp->data; | ||
2520 | mgmt->frame_control = | ||
2521 | IEEE80211_FC(IEEE80211_FTYPE_MGMT, | ||
2522 | IEEE80211_STYPE_PROBE_RESP); | ||
2523 | } else { | ||
2524 | printk(KERN_DEBUG "%s: Could not allocate ProbeResp " | ||
2525 | "template for IBSS\n", dev->name); | ||
2526 | } | ||
2527 | |||
2528 | if (local->ops->beacon_update && | ||
2529 | local->ops->beacon_update(local_to_hw(local), skb) == 0) { | ||
2530 | printk(KERN_DEBUG "%s: Configured IBSS beacon " | ||
2531 | "template\n", dev->name); | ||
2532 | skb = NULL; | ||
2533 | } | ||
2534 | |||
2535 | rates = 0; | ||
2536 | sband = local->hw.wiphy->bands[local->hw.conf.channel->band]; | ||
2537 | for (i = 0; i < bss->supp_rates_len; i++) { | ||
2538 | int bitrate = (bss->supp_rates[i] & 0x7f) * 5; | ||
2539 | for (j = 0; j < sband->n_bitrates; j++) | ||
2540 | if (sband->bitrates[j].bitrate == bitrate) | ||
2541 | rates |= BIT(j); | ||
2542 | } | ||
2543 | ifsta->supp_rates_bits[local->hw.conf.channel->band] = rates; | ||
2544 | 2494 | ||
2545 | ieee80211_sta_def_wmm_params(dev, bss, 1); | 2495 | ieee80211_if_config(sdata, IEEE80211_IFCC_BEACON); |
2546 | } while (0); | 2496 | } |
2547 | 2497 | ||
2548 | if (skb) { | 2498 | rates = 0; |
2549 | printk(KERN_DEBUG "%s: Failed to configure IBSS beacon " | 2499 | sband = local->hw.wiphy->bands[local->hw.conf.channel->band]; |
2550 | "template\n", dev->name); | 2500 | for (i = 0; i < bss->supp_rates_len; i++) { |
2551 | dev_kfree_skb(skb); | 2501 | int bitrate = (bss->supp_rates[i] & 0x7f) * 5; |
2502 | for (j = 0; j < sband->n_bitrates; j++) | ||
2503 | if (sband->bitrates[j].bitrate == bitrate) | ||
2504 | rates |= BIT(j); | ||
2552 | } | 2505 | } |
2506 | ifsta->supp_rates_bits[local->hw.conf.channel->band] = rates; | ||
2507 | |||
2508 | ieee80211_sta_def_wmm_params(dev, bss, 1); | ||
2553 | 2509 | ||
2554 | ifsta->state = IEEE80211_IBSS_JOINED; | 2510 | ifsta->state = IEEE80211_IBSS_JOINED; |
2555 | mod_timer(&ifsta->timer, jiffies + IEEE80211_IBSS_MERGE_INTERVAL); | 2511 | mod_timer(&ifsta->timer, jiffies + IEEE80211_IBSS_MERGE_INTERVAL); |
@@ -3333,7 +3289,7 @@ static void ieee80211_mesh_housekeeping(struct net_device *dev, | |||
3333 | 3289 | ||
3334 | free_plinks = mesh_plink_availables(sdata); | 3290 | free_plinks = mesh_plink_availables(sdata); |
3335 | if (free_plinks != sdata->u.sta.accepting_plinks) | 3291 | if (free_plinks != sdata->u.sta.accepting_plinks) |
3336 | ieee80211_if_config_beacon(dev); | 3292 | ieee80211_if_config(sdata, IEEE80211_IFCC_BEACON); |
3337 | 3293 | ||
3338 | mod_timer(&ifsta->timer, jiffies + | 3294 | mod_timer(&ifsta->timer, jiffies + |
3339 | IEEE80211_MESH_HOUSEKEEPING_INTERVAL); | 3295 | IEEE80211_MESH_HOUSEKEEPING_INTERVAL); |
@@ -3757,28 +3713,45 @@ int ieee80211_sta_set_ssid(struct net_device *dev, char *ssid, size_t len) | |||
3757 | { | 3713 | { |
3758 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); | 3714 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); |
3759 | struct ieee80211_if_sta *ifsta; | 3715 | struct ieee80211_if_sta *ifsta; |
3716 | int res; | ||
3760 | 3717 | ||
3761 | if (len > IEEE80211_MAX_SSID_LEN) | 3718 | if (len > IEEE80211_MAX_SSID_LEN) |
3762 | return -EINVAL; | 3719 | return -EINVAL; |
3763 | 3720 | ||
3764 | ifsta = &sdata->u.sta; | 3721 | ifsta = &sdata->u.sta; |
3765 | 3722 | ||
3766 | if (ifsta->ssid_len != len || memcmp(ifsta->ssid, ssid, len) != 0) | 3723 | if (ifsta->ssid_len != len || memcmp(ifsta->ssid, ssid, len) != 0) { |
3724 | memset(ifsta->ssid, 0, sizeof(ifsta->ssid)); | ||
3725 | memcpy(ifsta->ssid, ssid, len); | ||
3726 | ifsta->ssid_len = len; | ||
3767 | ifsta->flags &= ~IEEE80211_STA_PREV_BSSID_SET; | 3727 | ifsta->flags &= ~IEEE80211_STA_PREV_BSSID_SET; |
3768 | memcpy(ifsta->ssid, ssid, len); | 3728 | |
3769 | memset(ifsta->ssid + len, 0, IEEE80211_MAX_SSID_LEN - len); | 3729 | res = 0; |
3770 | ifsta->ssid_len = len; | 3730 | /* |
3731 | * Hack! MLME code needs to be cleaned up to have different | ||
3732 | * entry points for configuration and internal selection change | ||
3733 | */ | ||
3734 | if (netif_running(sdata->dev)) | ||
3735 | res = ieee80211_if_config(sdata, IEEE80211_IFCC_SSID); | ||
3736 | if (res) { | ||
3737 | printk(KERN_DEBUG "%s: Failed to config new SSID to " | ||
3738 | "the low-level driver\n", dev->name); | ||
3739 | return res; | ||
3740 | } | ||
3741 | } | ||
3771 | 3742 | ||
3772 | if (len) | 3743 | if (len) |
3773 | ifsta->flags |= IEEE80211_STA_SSID_SET; | 3744 | ifsta->flags |= IEEE80211_STA_SSID_SET; |
3774 | else | 3745 | else |
3775 | ifsta->flags &= ~IEEE80211_STA_SSID_SET; | 3746 | ifsta->flags &= ~IEEE80211_STA_SSID_SET; |
3747 | |||
3776 | if (sdata->vif.type == IEEE80211_IF_TYPE_IBSS && | 3748 | if (sdata->vif.type == IEEE80211_IF_TYPE_IBSS && |
3777 | !(ifsta->flags & IEEE80211_STA_BSSID_SET)) { | 3749 | !(ifsta->flags & IEEE80211_STA_BSSID_SET)) { |
3778 | ifsta->ibss_join_req = jiffies; | 3750 | ifsta->ibss_join_req = jiffies; |
3779 | ifsta->state = IEEE80211_IBSS_SEARCH; | 3751 | ifsta->state = IEEE80211_IBSS_SEARCH; |
3780 | return ieee80211_sta_find_ibss(dev, ifsta); | 3752 | return ieee80211_sta_find_ibss(dev, ifsta); |
3781 | } | 3753 | } |
3754 | |||
3782 | return 0; | 3755 | return 0; |
3783 | } | 3756 | } |
3784 | 3757 | ||
@@ -3804,7 +3777,12 @@ int ieee80211_sta_set_bssid(struct net_device *dev, u8 *bssid) | |||
3804 | 3777 | ||
3805 | if (memcmp(ifsta->bssid, bssid, ETH_ALEN) != 0) { | 3778 | if (memcmp(ifsta->bssid, bssid, ETH_ALEN) != 0) { |
3806 | memcpy(ifsta->bssid, bssid, ETH_ALEN); | 3779 | memcpy(ifsta->bssid, bssid, ETH_ALEN); |
3807 | res = ieee80211_if_config(dev); | 3780 | res = 0; |
3781 | /* | ||
3782 | * Hack! See also ieee80211_sta_set_ssid. | ||
3783 | */ | ||
3784 | if (netif_running(sdata->dev)) | ||
3785 | res = ieee80211_if_config(sdata, IEEE80211_IFCC_BSSID); | ||
3808 | if (res) { | 3786 | if (res) { |
3809 | printk(KERN_DEBUG "%s: Failed to config new BSSID to " | 3787 | printk(KERN_DEBUG "%s: Failed to config new BSSID to " |
3810 | "the low-level driver\n", dev->name); | 3788 | "the low-level driver\n", dev->name); |