diff options
Diffstat (limited to 'drivers/net/wireless/libertas/assoc.c')
-rw-r--r-- | drivers/net/wireless/libertas/assoc.c | 58 |
1 files changed, 27 insertions, 31 deletions
diff --git a/drivers/net/wireless/libertas/assoc.c b/drivers/net/wireless/libertas/assoc.c index a9440a96b2e7..dd8732611ba9 100644 --- a/drivers/net/wireless/libertas/assoc.c +++ b/drivers/net/wireless/libertas/assoc.c | |||
@@ -34,7 +34,8 @@ static const u8 bssid_off[ETH_ALEN] __attribute__ ((aligned (2))) = | |||
34 | * | 34 | * |
35 | * @param priv A pointer to struct lbs_private structure | 35 | * @param priv A pointer to struct lbs_private structure |
36 | * @param rates the buffer which keeps input and output | 36 | * @param rates the buffer which keeps input and output |
37 | * @param rates_size the size of rate1 buffer; new size of buffer on return | 37 | * @param rates_size the size of rates buffer; new size of buffer on return, |
38 | * which will be less than or equal to original rates_size | ||
38 | * | 39 | * |
39 | * @return 0 on success, or -1 on error | 40 | * @return 0 on success, or -1 on error |
40 | */ | 41 | */ |
@@ -42,46 +43,41 @@ static int get_common_rates(struct lbs_private *priv, | |||
42 | u8 *rates, | 43 | u8 *rates, |
43 | u16 *rates_size) | 44 | u16 *rates_size) |
44 | { | 45 | { |
45 | u8 *card_rates = lbs_bg_rates; | ||
46 | size_t num_card_rates = sizeof(lbs_bg_rates); | ||
47 | int i, j; | 46 | int i, j; |
48 | u8 *tmp; | 47 | u8 intersection[MAX_RATES]; |
49 | size_t tmp_size = 0; | 48 | u16 intersection_size; |
49 | u16 num_rates = 0; | ||
50 | 50 | ||
51 | tmp = kzalloc(MAX_RATES * ARRAY_SIZE(lbs_bg_rates), GFP_KERNEL); | 51 | intersection_size = min_t(u16, *rates_size, ARRAY_SIZE(intersection)); |
52 | if (!tmp) | ||
53 | return -1; | ||
54 | 52 | ||
55 | /* For each rate in card_rates that exists in rate1, copy to tmp */ | 53 | /* Allow each rate from 'rates' that is supported by the hardware */ |
56 | for (i = 0; card_rates[i] && (i < num_card_rates); i++) { | 54 | for (i = 0; i < ARRAY_SIZE(lbs_bg_rates) && lbs_bg_rates[i]; i++) { |
57 | for (j = 0; rates[j] && (j < *rates_size); j++) { | 55 | for (j = 0; j < intersection_size && rates[j]; j++) { |
58 | if (rates[j] == card_rates[i]) | 56 | if (rates[j] == lbs_bg_rates[i]) |
59 | tmp[tmp_size++] = card_rates[i]; | 57 | intersection[num_rates++] = rates[j]; |
60 | } | 58 | } |
61 | } | 59 | } |
62 | 60 | ||
63 | lbs_deb_hex(LBS_DEB_JOIN, "AP rates ", rates, *rates_size); | 61 | lbs_deb_hex(LBS_DEB_JOIN, "AP rates ", rates, *rates_size); |
64 | lbs_deb_hex(LBS_DEB_JOIN, "card rates ", card_rates, num_card_rates); | 62 | lbs_deb_hex(LBS_DEB_JOIN, "card rates ", lbs_bg_rates, |
65 | lbs_deb_hex(LBS_DEB_JOIN, "common rates", tmp, tmp_size); | 63 | ARRAY_SIZE(lbs_bg_rates)); |
64 | lbs_deb_hex(LBS_DEB_JOIN, "common rates", intersection, num_rates); | ||
66 | lbs_deb_join("TX data rate 0x%02x\n", priv->cur_rate); | 65 | lbs_deb_join("TX data rate 0x%02x\n", priv->cur_rate); |
67 | 66 | ||
68 | memset(rates, 0, *rates_size); | ||
69 | *rates_size = min_t(u16, tmp_size, *rates_size); | ||
70 | memcpy(rates, tmp, *rates_size); | ||
71 | |||
72 | if (!priv->enablehwauto) { | 67 | if (!priv->enablehwauto) { |
73 | for (i = 0; i < tmp_size; i++) { | 68 | for (i = 0; i < num_rates; i++) { |
74 | if (tmp[i] == priv->cur_rate) | 69 | if (intersection[i] == priv->cur_rate) |
75 | break; | 70 | goto done; |
76 | } | ||
77 | if (i == tmp_size) { | ||
78 | lbs_pr_alert("Previously set fixed data rate %#x isn't " | ||
79 | "compatible with the network.\n", | ||
80 | priv->cur_rate); | ||
81 | return -1; | ||
82 | } | 71 | } |
72 | lbs_pr_alert("Previously set fixed data rate %#x isn't " | ||
73 | "compatible with the network.\n", priv->cur_rate); | ||
74 | return -1; | ||
83 | } | 75 | } |
84 | kfree(tmp); | 76 | |
77 | done: | ||
78 | memset(rates, 0, *rates_size); | ||
79 | *rates_size = num_rates; | ||
80 | memcpy(rates, intersection, num_rates); | ||
85 | return 0; | 81 | return 0; |
86 | } | 82 | } |
87 | 83 | ||
@@ -324,7 +320,7 @@ static int lbs_associate(struct lbs_private *priv, | |||
324 | 320 | ||
325 | rates = (struct mrvl_ie_rates_param_set *) pos; | 321 | rates = (struct mrvl_ie_rates_param_set *) pos; |
326 | rates->header.type = cpu_to_le16(TLV_TYPE_RATES); | 322 | rates->header.type = cpu_to_le16(TLV_TYPE_RATES); |
327 | tmplen = MAX_RATES; | 323 | tmplen = min_t(u16, ARRAY_SIZE(bss->rates), MAX_RATES); |
328 | memcpy(&rates->rates, &bss->rates, tmplen); | 324 | memcpy(&rates->rates, &bss->rates, tmplen); |
329 | if (get_common_rates(priv, rates->rates, &tmplen)) { | 325 | if (get_common_rates(priv, rates->rates, &tmplen)) { |
330 | ret = -1; | 326 | ret = -1; |
@@ -599,7 +595,7 @@ static int lbs_adhoc_join(struct lbs_private *priv, | |||
599 | 595 | ||
600 | /* Copy Data rates from the rates recorded in scan response */ | 596 | /* Copy Data rates from the rates recorded in scan response */ |
601 | memset(cmd.bss.rates, 0, sizeof(cmd.bss.rates)); | 597 | memset(cmd.bss.rates, 0, sizeof(cmd.bss.rates)); |
602 | ratesize = min_t(u16, sizeof(cmd.bss.rates), MAX_RATES); | 598 | ratesize = min_t(u16, ARRAY_SIZE(cmd.bss.rates), ARRAY_SIZE (bss->rates)); |
603 | memcpy(cmd.bss.rates, bss->rates, ratesize); | 599 | memcpy(cmd.bss.rates, bss->rates, ratesize); |
604 | if (get_common_rates(priv, cmd.bss.rates, &ratesize)) { | 600 | if (get_common_rates(priv, cmd.bss.rates, &ratesize)) { |
605 | lbs_deb_join("ADHOC_JOIN: get_common_rates returned error.\n"); | 601 | lbs_deb_join("ADHOC_JOIN: get_common_rates returned error.\n"); |