aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--net/mac80211/ieee80211_i.h7
-rw-r--r--net/mac80211/tx.c322
-rw-r--r--net/mac80211/util.c19
-rw-r--r--net/mac80211/wep.c21
-rw-r--r--net/mac80211/wpa.c28
5 files changed, 173 insertions, 224 deletions
diff --git a/net/mac80211/ieee80211_i.h b/net/mac80211/ieee80211_i.h
index acba78e1a5ca..785f6363a6fc 100644
--- a/net/mac80211/ieee80211_i.h
+++ b/net/mac80211/ieee80211_i.h
@@ -149,11 +149,6 @@ struct ieee80211_tx_data {
149 149
150 struct ieee80211_channel *channel; 150 struct ieee80211_channel *channel;
151 151
152 /* Extra fragments (in addition to the first fragment
153 * in skb) */
154 struct sk_buff **extra_frag;
155 int num_extra_frag;
156
157 u16 ethertype; 152 u16 ethertype;
158 unsigned int flags; 153 unsigned int flags;
159}; 154};
@@ -191,8 +186,6 @@ struct ieee80211_rx_data {
191 186
192struct ieee80211_tx_stored_packet { 187struct ieee80211_tx_stored_packet {
193 struct sk_buff *skb; 188 struct sk_buff *skb;
194 struct sk_buff **extra_frag;
195 int num_extra_frag;
196}; 189};
197 190
198struct beacon_data { 191struct beacon_data {
diff --git a/net/mac80211/tx.c b/net/mac80211/tx.c
index f3f240c69018..51bf49cc75bc 100644
--- a/net/mac80211/tx.c
+++ b/net/mac80211/tx.c
@@ -34,8 +34,7 @@
34 34
35#define IEEE80211_TX_OK 0 35#define IEEE80211_TX_OK 0
36#define IEEE80211_TX_AGAIN 1 36#define IEEE80211_TX_AGAIN 1
37#define IEEE80211_TX_FRAG_AGAIN 2 37#define IEEE80211_TX_PENDING 2
38#define IEEE80211_TX_PENDING 3
39 38
40/* misc utils */ 39/* misc utils */
41 40
@@ -702,17 +701,62 @@ ieee80211_tx_h_sequence(struct ieee80211_tx_data *tx)
702 return TX_CONTINUE; 701 return TX_CONTINUE;
703} 702}
704 703
704static int ieee80211_fragment(struct ieee80211_local *local,
705 struct sk_buff *skb, int hdrlen,
706 int frag_threshold)
707{
708 struct sk_buff *tail = skb, *tmp;
709 int per_fragm = frag_threshold - hdrlen - FCS_LEN;
710 int pos = hdrlen + per_fragm;
711 int rem = skb->len - hdrlen - per_fragm;
712
713 if (WARN_ON(rem < 0))
714 return -EINVAL;
715
716 while (rem) {
717 int fraglen = per_fragm;
718
719 if (fraglen > rem)
720 fraglen = rem;
721 rem -= fraglen;
722 tmp = dev_alloc_skb(local->tx_headroom +
723 frag_threshold +
724 IEEE80211_ENCRYPT_HEADROOM +
725 IEEE80211_ENCRYPT_TAILROOM);
726 if (!tmp)
727 return -ENOMEM;
728 tail->next = tmp;
729 tail = tmp;
730 skb_reserve(tmp, local->tx_headroom +
731 IEEE80211_ENCRYPT_HEADROOM);
732 /* copy control information */
733 memcpy(tmp->cb, skb->cb, sizeof(tmp->cb));
734 skb_copy_queue_mapping(tmp, skb);
735 tmp->priority = skb->priority;
736 tmp->do_not_encrypt = skb->do_not_encrypt;
737 tmp->dev = skb->dev;
738 tmp->iif = skb->iif;
739
740 /* copy header and data */
741 memcpy(skb_put(tmp, hdrlen), skb->data, hdrlen);
742 memcpy(skb_put(tmp, fraglen), skb->data + pos, fraglen);
743
744 pos += fraglen;
745 }
746
747 skb->len = hdrlen + per_fragm;
748 return 0;
749}
750
705static ieee80211_tx_result debug_noinline 751static ieee80211_tx_result debug_noinline
706ieee80211_tx_h_fragment(struct ieee80211_tx_data *tx) 752ieee80211_tx_h_fragment(struct ieee80211_tx_data *tx)
707{ 753{
708 struct ieee80211_tx_info *info = IEEE80211_SKB_CB(tx->skb); 754 struct sk_buff *skb = tx->skb;
709 struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)tx->skb->data; 755 struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
710 size_t hdrlen, per_fragm, num_fragm, payload_len, left; 756 struct ieee80211_hdr *hdr = (void *)skb->data;
711 struct sk_buff **frags, *first, *frag;
712 int i;
713 u16 seq;
714 u8 *pos;
715 int frag_threshold = tx->local->fragmentation_threshold; 757 int frag_threshold = tx->local->fragmentation_threshold;
758 int hdrlen;
759 int fragnum;
716 760
717 if (!(tx->flags & IEEE80211_TX_FRAGMENTED)) 761 if (!(tx->flags & IEEE80211_TX_FRAGMENTED))
718 return TX_CONTINUE; 762 return TX_CONTINUE;
@@ -725,58 +769,35 @@ ieee80211_tx_h_fragment(struct ieee80211_tx_data *tx)
725 if (WARN_ON(info->flags & IEEE80211_TX_CTL_AMPDU)) 769 if (WARN_ON(info->flags & IEEE80211_TX_CTL_AMPDU))
726 return TX_DROP; 770 return TX_DROP;
727 771
728 first = tx->skb;
729
730 hdrlen = ieee80211_hdrlen(hdr->frame_control); 772 hdrlen = ieee80211_hdrlen(hdr->frame_control);
731 payload_len = first->len - hdrlen;
732 per_fragm = frag_threshold - hdrlen - FCS_LEN;
733 num_fragm = DIV_ROUND_UP(payload_len, per_fragm);
734
735 frags = kzalloc(num_fragm * sizeof(struct sk_buff *), GFP_ATOMIC);
736 if (!frags)
737 goto fail;
738
739 hdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_MOREFRAGS);
740 seq = le16_to_cpu(hdr->seq_ctrl) & IEEE80211_SCTL_SEQ;
741 pos = first->data + hdrlen + per_fragm;
742 left = payload_len - per_fragm;
743 for (i = 0; i < num_fragm - 1; i++) {
744 struct ieee80211_hdr *fhdr;
745 size_t copylen;
746
747 if (left <= 0)
748 goto fail;
749 773
750 /* reserve enough extra head and tail room for possible 774 /* internal error, why is TX_FRAGMENTED set? */
751 * encryption */ 775 if (WARN_ON(skb->len <= frag_threshold))
752 frag = frags[i] = 776 return TX_DROP;
753 dev_alloc_skb(tx->local->tx_headroom +
754 frag_threshold +
755 IEEE80211_ENCRYPT_HEADROOM +
756 IEEE80211_ENCRYPT_TAILROOM);
757 if (!frag)
758 goto fail;
759
760 /* Make sure that all fragments use the same priority so
761 * that they end up using the same TX queue */
762 frag->priority = first->priority;
763 777
764 skb_reserve(frag, tx->local->tx_headroom + 778 /*
765 IEEE80211_ENCRYPT_HEADROOM); 779 * Now fragment the frame. This will allocate all the fragments and
780 * chain them (using skb as the first fragment) to skb->next.
781 * During transmission, we will remove the successfully transmitted
782 * fragments from this list. When the low-level driver rejects one
783 * of the fragments then we will simply pretend to accept the skb
784 * but store it away as pending.
785 */
786 if (ieee80211_fragment(tx->local, skb, hdrlen, frag_threshold))
787 return TX_DROP;
766 788
767 /* copy TX information */ 789 /* update duration/seq/flags of fragments */
768 info = IEEE80211_SKB_CB(frag); 790 fragnum = 0;
769 memcpy(info, first->cb, sizeof(frag->cb)); 791 do {
792 int next_len;
793 const __le16 morefrags = cpu_to_le16(IEEE80211_FCTL_MOREFRAGS);
770 794
771 /* copy/fill in 802.11 header */ 795 hdr = (void *)skb->data;
772 fhdr = (struct ieee80211_hdr *) skb_put(frag, hdrlen); 796 info = IEEE80211_SKB_CB(skb);
773 memcpy(fhdr, first->data, hdrlen);
774 fhdr->seq_ctrl = cpu_to_le16(seq | ((i + 1) & IEEE80211_SCTL_FRAG));
775 797
776 if (i == num_fragm - 2) { 798 if (skb->next) {
777 /* clear MOREFRAGS bit for the last fragment */ 799 hdr->frame_control |= morefrags;
778 fhdr->frame_control &= cpu_to_le16(~IEEE80211_FCTL_MOREFRAGS); 800 next_len = skb->next->len;
779 } else {
780 /* 801 /*
781 * No multi-rate retries for fragmented frames, that 802 * No multi-rate retries for fragmented frames, that
782 * would completely throw off the NAV at other STAs. 803 * would completely throw off the NAV at other STAs.
@@ -787,37 +808,16 @@ ieee80211_tx_h_fragment(struct ieee80211_tx_data *tx)
787 info->control.rates[4].idx = -1; 808 info->control.rates[4].idx = -1;
788 BUILD_BUG_ON(IEEE80211_TX_MAX_RATES != 5); 809 BUILD_BUG_ON(IEEE80211_TX_MAX_RATES != 5);
789 info->flags &= ~IEEE80211_TX_CTL_RATE_CTRL_PROBE; 810 info->flags &= ~IEEE80211_TX_CTL_RATE_CTRL_PROBE;
811 } else {
812 hdr->frame_control &= ~morefrags;
813 next_len = 0;
790 } 814 }
791 815 hdr->duration_id = ieee80211_duration(tx, 0, next_len);
792 /* copy data */ 816 hdr->seq_ctrl |= cpu_to_le16(fragnum & IEEE80211_SCTL_FRAG);
793 copylen = left > per_fragm ? per_fragm : left; 817 fragnum++;
794 memcpy(skb_put(frag, copylen), pos, copylen); 818 } while ((skb = skb->next));
795
796 skb_copy_queue_mapping(frag, first);
797
798 frag->do_not_encrypt = first->do_not_encrypt;
799 frag->dev = first->dev;
800 frag->iif = first->iif;
801
802 pos += copylen;
803 left -= copylen;
804 }
805 skb_trim(first, hdrlen + per_fragm);
806
807 tx->num_extra_frag = num_fragm - 1;
808 tx->extra_frag = frags;
809 819
810 return TX_CONTINUE; 820 return TX_CONTINUE;
811
812 fail:
813 if (frags) {
814 for (i = 0; i < num_fragm - 1; i++)
815 if (frags[i])
816 dev_kfree_skb(frags[i]);
817 kfree(frags);
818 }
819 I802_DEBUG_INC(tx->local->tx_handlers_drop_fragment);
820 return TX_DROP;
821} 821}
822 822
823static ieee80211_tx_result debug_noinline 823static ieee80211_tx_result debug_noinline
@@ -845,27 +845,19 @@ ieee80211_tx_h_encrypt(struct ieee80211_tx_data *tx)
845static ieee80211_tx_result debug_noinline 845static ieee80211_tx_result debug_noinline
846ieee80211_tx_h_calculate_duration(struct ieee80211_tx_data *tx) 846ieee80211_tx_h_calculate_duration(struct ieee80211_tx_data *tx)
847{ 847{
848 struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)tx->skb->data; 848 struct sk_buff *skb = tx->skb;
849 int next_len, i; 849 struct ieee80211_hdr *hdr;
850 int group_addr = is_multicast_ether_addr(hdr->addr1); 850 int next_len;
851 851 bool group_addr;
852 if (!(tx->flags & IEEE80211_TX_FRAGMENTED)) {
853 hdr->duration_id = ieee80211_duration(tx, group_addr, 0);
854 return TX_CONTINUE;
855 }
856
857 hdr->duration_id = ieee80211_duration(tx, group_addr,
858 tx->extra_frag[0]->len);
859 852
860 for (i = 0; i < tx->num_extra_frag; i++) { 853 do {
861 if (i + 1 < tx->num_extra_frag) 854 hdr = (void *) skb->data;
862 next_len = tx->extra_frag[i + 1]->len; 855 next_len = skb->next ? skb->next->len : 0;
863 else 856 group_addr = is_multicast_ether_addr(hdr->addr1);
864 next_len = 0;
865 857
866 hdr = (struct ieee80211_hdr *)tx->extra_frag[i]->data; 858 hdr->duration_id =
867 hdr->duration_id = ieee80211_duration(tx, 0, next_len); 859 ieee80211_duration(tx, group_addr, next_len);
868 } 860 } while ((skb = skb->next));
869 861
870 return TX_CONTINUE; 862 return TX_CONTINUE;
871} 863}
@@ -873,19 +865,16 @@ ieee80211_tx_h_calculate_duration(struct ieee80211_tx_data *tx)
873static ieee80211_tx_result debug_noinline 865static ieee80211_tx_result debug_noinline
874ieee80211_tx_h_stats(struct ieee80211_tx_data *tx) 866ieee80211_tx_h_stats(struct ieee80211_tx_data *tx)
875{ 867{
876 int i; 868 struct sk_buff *skb = tx->skb;
877 869
878 if (!tx->sta) 870 if (!tx->sta)
879 return TX_CONTINUE; 871 return TX_CONTINUE;
880 872
881 tx->sta->tx_packets++; 873 tx->sta->tx_packets++;
882 tx->sta->tx_fragments++; 874 do {
883 tx->sta->tx_bytes += tx->skb->len; 875 tx->sta->tx_fragments++;
884 if (tx->extra_frag) { 876 tx->sta->tx_bytes += skb->len;
885 tx->sta->tx_fragments += tx->num_extra_frag; 877 } while ((skb = skb->next));
886 for (i = 0; i < tx->num_extra_frag; i++)
887 tx->sta->tx_bytes += tx->extra_frag[i]->len;
888 }
889 878
890 return TX_CONTINUE; 879 return TX_CONTINUE;
891} 880}
@@ -1099,45 +1088,36 @@ static int ieee80211_tx_prepare(struct ieee80211_local *local,
1099 return 0; 1088 return 0;
1100} 1089}
1101 1090
1102static int __ieee80211_tx(struct ieee80211_local *local, struct sk_buff *skb, 1091static int __ieee80211_tx(struct ieee80211_local *local,
1103 struct ieee80211_tx_data *tx) 1092 struct ieee80211_tx_data *tx)
1104{ 1093{
1094 struct sk_buff *skb = tx->skb, *next;
1105 struct ieee80211_tx_info *info; 1095 struct ieee80211_tx_info *info;
1106 int ret, i; 1096 int ret;
1097 bool fragm = false;
1107 1098
1108 if (skb) { 1099 local->mdev->trans_start = jiffies;
1100
1101 while (skb) {
1109 if (ieee80211_queue_stopped(&local->hw, 1102 if (ieee80211_queue_stopped(&local->hw,
1110 skb_get_queue_mapping(skb))) 1103 skb_get_queue_mapping(skb)))
1111 return IEEE80211_TX_PENDING; 1104 return IEEE80211_TX_PENDING;
1112 1105
1113 ret = local->ops->tx(local_to_hw(local), skb); 1106 if (fragm) {
1114 if (ret) 1107 info = IEEE80211_SKB_CB(skb);
1115 return IEEE80211_TX_AGAIN;
1116 local->mdev->trans_start = jiffies;
1117 ieee80211_led_tx(local, 1);
1118 }
1119 if (tx->extra_frag) {
1120 for (i = 0; i < tx->num_extra_frag; i++) {
1121 if (!tx->extra_frag[i])
1122 continue;
1123 info = IEEE80211_SKB_CB(tx->extra_frag[i]);
1124 info->flags &= ~(IEEE80211_TX_CTL_CLEAR_PS_FILT | 1108 info->flags &= ~(IEEE80211_TX_CTL_CLEAR_PS_FILT |
1125 IEEE80211_TX_CTL_FIRST_FRAGMENT); 1109 IEEE80211_TX_CTL_FIRST_FRAGMENT);
1126 if (ieee80211_queue_stopped(&local->hw,
1127 skb_get_queue_mapping(tx->extra_frag[i])))
1128 return IEEE80211_TX_FRAG_AGAIN;
1129
1130 ret = local->ops->tx(local_to_hw(local),
1131 tx->extra_frag[i]);
1132 if (ret)
1133 return IEEE80211_TX_FRAG_AGAIN;
1134 local->mdev->trans_start = jiffies;
1135 ieee80211_led_tx(local, 1);
1136 tx->extra_frag[i] = NULL;
1137 } 1110 }
1138 kfree(tx->extra_frag); 1111
1139 tx->extra_frag = NULL; 1112 next = skb->next;
1113 ret = local->ops->tx(local_to_hw(local), skb);
1114 if (ret != NETDEV_TX_OK)
1115 return IEEE80211_TX_AGAIN;
1116 tx->skb = skb = next;
1117 ieee80211_led_tx(local, 1);
1118 fragm = true;
1140 } 1119 }
1120
1141 return IEEE80211_TX_OK; 1121 return IEEE80211_TX_OK;
1142} 1122}
1143 1123
@@ -1149,7 +1129,6 @@ static int invoke_tx_handlers(struct ieee80211_tx_data *tx)
1149{ 1129{
1150 struct sk_buff *skb = tx->skb; 1130 struct sk_buff *skb = tx->skb;
1151 ieee80211_tx_result res = TX_DROP; 1131 ieee80211_tx_result res = TX_DROP;
1152 int i;
1153 1132
1154#define CALL_TXH(txh) \ 1133#define CALL_TXH(txh) \
1155 res = txh(tx); \ 1134 res = txh(tx); \
@@ -1173,11 +1152,13 @@ static int invoke_tx_handlers(struct ieee80211_tx_data *tx)
1173 txh_done: 1152 txh_done:
1174 if (unlikely(res == TX_DROP)) { 1153 if (unlikely(res == TX_DROP)) {
1175 I802_DEBUG_INC(tx->local->tx_handlers_drop); 1154 I802_DEBUG_INC(tx->local->tx_handlers_drop);
1176 dev_kfree_skb(skb); 1155 while (skb) {
1177 for (i = 0; i < tx->num_extra_frag; i++) 1156 struct sk_buff *next;
1178 if (tx->extra_frag[i]) 1157
1179 dev_kfree_skb(tx->extra_frag[i]); 1158 next = skb->next;
1180 kfree(tx->extra_frag); 1159 dev_kfree_skb(skb);
1160 skb = next;
1161 }
1181 return -1; 1162 return -1;
1182 } else if (unlikely(res == TX_QUEUED)) { 1163 } else if (unlikely(res == TX_QUEUED)) {
1183 I802_DEBUG_INC(tx->local->tx_handlers_queued); 1164 I802_DEBUG_INC(tx->local->tx_handlers_queued);
@@ -1194,7 +1175,7 @@ static int ieee80211_tx(struct net_device *dev, struct sk_buff *skb)
1194 struct ieee80211_tx_data tx; 1175 struct ieee80211_tx_data tx;
1195 ieee80211_tx_result res_prepare; 1176 ieee80211_tx_result res_prepare;
1196 struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); 1177 struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
1197 int ret, i; 1178 int ret;
1198 u16 queue; 1179 u16 queue;
1199 1180
1200 queue = skb_get_queue_mapping(skb); 1181 queue = skb_get_queue_mapping(skb);
@@ -1225,7 +1206,7 @@ static int ieee80211_tx(struct net_device *dev, struct sk_buff *skb)
1225 goto out; 1206 goto out;
1226 1207
1227retry: 1208retry:
1228 ret = __ieee80211_tx(local, skb, &tx); 1209 ret = __ieee80211_tx(local, &tx);
1229 if (ret) { 1210 if (ret) {
1230 struct ieee80211_tx_stored_packet *store; 1211 struct ieee80211_tx_stored_packet *store;
1231 1212
@@ -1240,9 +1221,6 @@ retry:
1240 1221
1241 store = &local->pending_packet[queue]; 1222 store = &local->pending_packet[queue];
1242 1223
1243 if (ret == IEEE80211_TX_FRAG_AGAIN)
1244 skb = NULL;
1245
1246 set_bit(queue, local->queues_pending); 1224 set_bit(queue, local->queues_pending);
1247 smp_mb(); 1225 smp_mb();
1248 /* 1226 /*
@@ -1260,22 +1238,23 @@ retry:
1260 clear_bit(queue, local->queues_pending); 1238 clear_bit(queue, local->queues_pending);
1261 goto retry; 1239 goto retry;
1262 } 1240 }
1263 store->skb = skb; 1241 store->skb = tx.skb;
1264 store->extra_frag = tx.extra_frag;
1265 store->num_extra_frag = tx.num_extra_frag;
1266 } 1242 }
1267 out: 1243 out:
1268 rcu_read_unlock(); 1244 rcu_read_unlock();
1269 return 0; 1245 return 0;
1270 1246
1271 drop: 1247 drop:
1272 if (skb)
1273 dev_kfree_skb(skb);
1274 for (i = 0; i < tx.num_extra_frag; i++)
1275 if (tx.extra_frag[i])
1276 dev_kfree_skb(tx.extra_frag[i]);
1277 kfree(tx.extra_frag);
1278 rcu_read_unlock(); 1248 rcu_read_unlock();
1249
1250 skb = tx.skb;
1251 while (skb) {
1252 struct sk_buff *next;
1253
1254 next = skb->next;
1255 dev_kfree_skb(skb);
1256 skb = next;
1257 }
1279 return 0; 1258 return 0;
1280} 1259}
1281 1260
@@ -1810,17 +1789,21 @@ int ieee80211_subif_start_xmit(struct sk_buff *skb,
1810 */ 1789 */
1811void ieee80211_clear_tx_pending(struct ieee80211_local *local) 1790void ieee80211_clear_tx_pending(struct ieee80211_local *local)
1812{ 1791{
1813 int i, j; 1792 struct sk_buff *skb;
1814 struct ieee80211_tx_stored_packet *store; 1793 int i;
1815 1794
1816 for (i = 0; i < local->hw.queues; i++) { 1795 for (i = 0; i < local->hw.queues; i++) {
1817 if (!test_bit(i, local->queues_pending)) 1796 if (!test_bit(i, local->queues_pending))
1818 continue; 1797 continue;
1819 store = &local->pending_packet[i]; 1798
1820 kfree_skb(store->skb); 1799 skb = local->pending_packet[i].skb;
1821 for (j = 0; j < store->num_extra_frag; j++) 1800 while (skb) {
1822 kfree_skb(store->extra_frag[j]); 1801 struct sk_buff *next;
1823 kfree(store->extra_frag); 1802
1803 next = skb->next;
1804 dev_kfree_skb(skb);
1805 skb = next;
1806 }
1824 clear_bit(i, local->queues_pending); 1807 clear_bit(i, local->queues_pending);
1825 } 1808 }
1826} 1809}
@@ -1854,14 +1837,11 @@ void ieee80211_tx_pending(unsigned long data)
1854 netif_start_subqueue(local->mdev, i); 1837 netif_start_subqueue(local->mdev, i);
1855 1838
1856 store = &local->pending_packet[i]; 1839 store = &local->pending_packet[i];
1857 tx.extra_frag = store->extra_frag;
1858 tx.num_extra_frag = store->num_extra_frag;
1859 tx.flags = 0; 1840 tx.flags = 0;
1860 ret = __ieee80211_tx(local, store->skb, &tx); 1841 tx.skb = store->skb;
1861 if (ret) { 1842 ret = __ieee80211_tx(local, &tx);
1862 if (ret == IEEE80211_TX_FRAG_AGAIN) 1843 store->skb = tx.skb;
1863 store->skb = NULL; 1844 if (!ret) {
1864 } else {
1865 clear_bit(i, local->queues_pending); 1845 clear_bit(i, local->queues_pending);
1866 ieee80211_wake_queue(&local->hw, i); 1846 ieee80211_wake_queue(&local->hw, i);
1867 } 1847 }
diff --git a/net/mac80211/util.c b/net/mac80211/util.c
index 444bb14c95e1..021166c8cce2 100644
--- a/net/mac80211/util.c
+++ b/net/mac80211/util.c
@@ -166,18 +166,13 @@ int ieee80211_get_mesh_hdrlen(struct ieee80211s_hdr *meshhdr)
166 166
167void ieee80211_tx_set_protected(struct ieee80211_tx_data *tx) 167void ieee80211_tx_set_protected(struct ieee80211_tx_data *tx)
168{ 168{
169 struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) tx->skb->data; 169 struct sk_buff *skb = tx->skb;
170 170 struct ieee80211_hdr *hdr;
171 hdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_PROTECTED); 171
172 if (tx->extra_frag) { 172 do {
173 struct ieee80211_hdr *fhdr; 173 hdr = (struct ieee80211_hdr *) skb->data;
174 int i; 174 hdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_PROTECTED);
175 for (i = 0; i < tx->num_extra_frag; i++) { 175 } while ((skb = skb->next));
176 fhdr = (struct ieee80211_hdr *)
177 tx->extra_frag[i]->data;
178 fhdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_PROTECTED);
179 }
180 }
181} 176}
182 177
183int ieee80211_frame_duration(struct ieee80211_local *local, size_t len, 178int ieee80211_frame_duration(struct ieee80211_local *local, size_t len,
diff --git a/net/mac80211/wep.c b/net/mac80211/wep.c
index 7043ddc75498..ef73105b3061 100644
--- a/net/mac80211/wep.c
+++ b/net/mac80211/wep.c
@@ -329,24 +329,17 @@ static int wep_encrypt_skb(struct ieee80211_tx_data *tx, struct sk_buff *skb)
329ieee80211_tx_result 329ieee80211_tx_result
330ieee80211_crypto_wep_encrypt(struct ieee80211_tx_data *tx) 330ieee80211_crypto_wep_encrypt(struct ieee80211_tx_data *tx)
331{ 331{
332 int i; 332 struct sk_buff *skb;
333 333
334 ieee80211_tx_set_protected(tx); 334 ieee80211_tx_set_protected(tx);
335 335
336 if (wep_encrypt_skb(tx, tx->skb) < 0) { 336 skb = tx->skb;
337 I802_DEBUG_INC(tx->local->tx_handlers_drop_wep); 337 do {
338 return TX_DROP; 338 if (wep_encrypt_skb(tx, skb) < 0) {
339 } 339 I802_DEBUG_INC(tx->local->tx_handlers_drop_wep);
340 340 return TX_DROP;
341 if (tx->extra_frag) {
342 for (i = 0; i < tx->num_extra_frag; i++) {
343 if (wep_encrypt_skb(tx, tx->extra_frag[i])) {
344 I802_DEBUG_INC(tx->local->
345 tx_handlers_drop_wep);
346 return TX_DROP;
347 }
348 } 341 }
349 } 342 } while ((skb = skb->next));
350 343
351 return TX_CONTINUE; 344 return TX_CONTINUE;
352} 345}
diff --git a/net/mac80211/wpa.c b/net/mac80211/wpa.c
index 9101b48ec2ae..4f8bfea278f2 100644
--- a/net/mac80211/wpa.c
+++ b/net/mac80211/wpa.c
@@ -196,19 +196,13 @@ ieee80211_tx_result
196ieee80211_crypto_tkip_encrypt(struct ieee80211_tx_data *tx) 196ieee80211_crypto_tkip_encrypt(struct ieee80211_tx_data *tx)
197{ 197{
198 struct sk_buff *skb = tx->skb; 198 struct sk_buff *skb = tx->skb;
199 int i;
200 199
201 ieee80211_tx_set_protected(tx); 200 ieee80211_tx_set_protected(tx);
202 201
203 if (tkip_encrypt_skb(tx, skb) < 0) 202 do {
204 return TX_DROP; 203 if (tkip_encrypt_skb(tx, skb) < 0)
205 204 return TX_DROP;
206 if (tx->extra_frag) { 205 } while ((skb = skb->next));
207 for (i = 0; i < tx->num_extra_frag; i++) {
208 if (tkip_encrypt_skb(tx, tx->extra_frag[i]))
209 return TX_DROP;
210 }
211 }
212 206
213 return TX_CONTINUE; 207 return TX_CONTINUE;
214} 208}
@@ -428,19 +422,13 @@ ieee80211_tx_result
428ieee80211_crypto_ccmp_encrypt(struct ieee80211_tx_data *tx) 422ieee80211_crypto_ccmp_encrypt(struct ieee80211_tx_data *tx)
429{ 423{
430 struct sk_buff *skb = tx->skb; 424 struct sk_buff *skb = tx->skb;
431 int i;
432 425
433 ieee80211_tx_set_protected(tx); 426 ieee80211_tx_set_protected(tx);
434 427
435 if (ccmp_encrypt_skb(tx, skb) < 0) 428 do {
436 return TX_DROP; 429 if (ccmp_encrypt_skb(tx, skb) < 0)
437 430 return TX_DROP;
438 if (tx->extra_frag) { 431 } while ((skb = skb->next));
439 for (i = 0; i < tx->num_extra_frag; i++) {
440 if (ccmp_encrypt_skb(tx, tx->extra_frag[i]))
441 return TX_DROP;
442 }
443 }
444 432
445 return TX_CONTINUE; 433 return TX_CONTINUE;
446} 434}