diff options
Diffstat (limited to 'net/mac80211/key.c')
-rw-r--r-- | net/mac80211/key.c | 51 |
1 files changed, 25 insertions, 26 deletions
diff --git a/net/mac80211/key.c b/net/mac80211/key.c index 999f7aa42326..b0a025c9b615 100644 --- a/net/mac80211/key.c +++ b/net/mac80211/key.c | |||
@@ -47,7 +47,6 @@ | |||
47 | */ | 47 | */ |
48 | 48 | ||
49 | static const u8 bcast_addr[ETH_ALEN] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }; | 49 | static const u8 bcast_addr[ETH_ALEN] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }; |
50 | static const u8 zero_addr[ETH_ALEN]; | ||
51 | 50 | ||
52 | /* key mutex: used to synchronise todo runners */ | 51 | /* key mutex: used to synchronise todo runners */ |
53 | static DEFINE_MUTEX(key_mutex); | 52 | static DEFINE_MUTEX(key_mutex); |
@@ -108,29 +107,18 @@ static void assert_key_lock(void) | |||
108 | WARN_ON(!mutex_is_locked(&key_mutex)); | 107 | WARN_ON(!mutex_is_locked(&key_mutex)); |
109 | } | 108 | } |
110 | 109 | ||
111 | static const u8 *get_mac_for_key(struct ieee80211_key *key) | 110 | static struct ieee80211_sta *get_sta_for_key(struct ieee80211_key *key) |
112 | { | 111 | { |
113 | const u8 *addr = bcast_addr; | ||
114 | |||
115 | /* | ||
116 | * If we're an AP we won't ever receive frames with a non-WEP | ||
117 | * group key so we tell the driver that by using the zero MAC | ||
118 | * address to indicate a transmit-only key. | ||
119 | */ | ||
120 | if (key->conf.alg != ALG_WEP && | ||
121 | (key->sdata->vif.type == NL80211_IFTYPE_AP || | ||
122 | key->sdata->vif.type == NL80211_IFTYPE_AP_VLAN)) | ||
123 | addr = zero_addr; | ||
124 | |||
125 | if (key->sta) | 112 | if (key->sta) |
126 | addr = key->sta->sta.addr; | 113 | return &key->sta->sta; |
127 | 114 | ||
128 | return addr; | 115 | return NULL; |
129 | } | 116 | } |
130 | 117 | ||
131 | static void ieee80211_key_enable_hw_accel(struct ieee80211_key *key) | 118 | static void ieee80211_key_enable_hw_accel(struct ieee80211_key *key) |
132 | { | 119 | { |
133 | const u8 *addr; | 120 | struct ieee80211_sub_if_data *sdata; |
121 | struct ieee80211_sta *sta; | ||
134 | int ret; | 122 | int ret; |
135 | 123 | ||
136 | assert_key_lock(); | 124 | assert_key_lock(); |
@@ -139,11 +127,16 @@ static void ieee80211_key_enable_hw_accel(struct ieee80211_key *key) | |||
139 | if (!key->local->ops->set_key) | 127 | if (!key->local->ops->set_key) |
140 | return; | 128 | return; |
141 | 129 | ||
142 | addr = get_mac_for_key(key); | 130 | sta = get_sta_for_key(key); |
131 | |||
132 | sdata = key->sdata; | ||
133 | if (sdata->vif.type == NL80211_IFTYPE_AP_VLAN) | ||
134 | sdata = container_of(sdata->bss, | ||
135 | struct ieee80211_sub_if_data, | ||
136 | u.ap); | ||
143 | 137 | ||
144 | ret = key->local->ops->set_key(local_to_hw(key->local), SET_KEY, | 138 | ret = key->local->ops->set_key(local_to_hw(key->local), SET_KEY, |
145 | key->sdata->dev->dev_addr, addr, | 139 | &sdata->vif, sta, &key->conf); |
146 | &key->conf); | ||
147 | 140 | ||
148 | if (!ret) { | 141 | if (!ret) { |
149 | spin_lock(&todo_lock); | 142 | spin_lock(&todo_lock); |
@@ -155,12 +148,13 @@ static void ieee80211_key_enable_hw_accel(struct ieee80211_key *key) | |||
155 | printk(KERN_ERR "mac80211-%s: failed to set key " | 148 | printk(KERN_ERR "mac80211-%s: failed to set key " |
156 | "(%d, %pM) to hardware (%d)\n", | 149 | "(%d, %pM) to hardware (%d)\n", |
157 | wiphy_name(key->local->hw.wiphy), | 150 | wiphy_name(key->local->hw.wiphy), |
158 | key->conf.keyidx, addr, ret); | 151 | key->conf.keyidx, sta ? sta->addr : bcast_addr, ret); |
159 | } | 152 | } |
160 | 153 | ||
161 | static void ieee80211_key_disable_hw_accel(struct ieee80211_key *key) | 154 | static void ieee80211_key_disable_hw_accel(struct ieee80211_key *key) |
162 | { | 155 | { |
163 | const u8 *addr; | 156 | struct ieee80211_sub_if_data *sdata; |
157 | struct ieee80211_sta *sta; | ||
164 | int ret; | 158 | int ret; |
165 | 159 | ||
166 | assert_key_lock(); | 160 | assert_key_lock(); |
@@ -176,17 +170,22 @@ static void ieee80211_key_disable_hw_accel(struct ieee80211_key *key) | |||
176 | } | 170 | } |
177 | spin_unlock(&todo_lock); | 171 | spin_unlock(&todo_lock); |
178 | 172 | ||
179 | addr = get_mac_for_key(key); | 173 | sta = get_sta_for_key(key); |
174 | sdata = key->sdata; | ||
175 | |||
176 | if (sdata->vif.type == NL80211_IFTYPE_AP_VLAN) | ||
177 | sdata = container_of(sdata->bss, | ||
178 | struct ieee80211_sub_if_data, | ||
179 | u.ap); | ||
180 | 180 | ||
181 | ret = key->local->ops->set_key(local_to_hw(key->local), DISABLE_KEY, | 181 | ret = key->local->ops->set_key(local_to_hw(key->local), DISABLE_KEY, |
182 | key->sdata->dev->dev_addr, addr, | 182 | &sdata->vif, sta, &key->conf); |
183 | &key->conf); | ||
184 | 183 | ||
185 | if (ret) | 184 | if (ret) |
186 | printk(KERN_ERR "mac80211-%s: failed to remove key " | 185 | printk(KERN_ERR "mac80211-%s: failed to remove key " |
187 | "(%d, %pM) from hardware (%d)\n", | 186 | "(%d, %pM) from hardware (%d)\n", |
188 | wiphy_name(key->local->hw.wiphy), | 187 | wiphy_name(key->local->hw.wiphy), |
189 | key->conf.keyidx, addr, ret); | 188 | key->conf.keyidx, sta ? sta->addr : bcast_addr, ret); |
190 | 189 | ||
191 | spin_lock(&todo_lock); | 190 | spin_lock(&todo_lock); |
192 | key->flags &= ~KEY_FLAG_UPLOADED_TO_HARDWARE; | 191 | key->flags &= ~KEY_FLAG_UPLOADED_TO_HARDWARE; |