Commit 8cda74c4 authored by Ulrich Kunitz's avatar Ulrich Kunitz Committed by Chris Wright

[PATCH] zd1211rw: Call ieee80211_rx in tasklet

The driver called ieee80211_rx in hardware interrupt context.  This has
been against the intention of the ieee80211_rx function.  It caused a bug
in the crypto routines used by WPA.  This patch calls ieee80211_rx in a
tasklet.
Signed-off-by: default avatarUlrich Kunitz <kune@deine-taler.de>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
Signed-off-by: default avatarChris Wright <chrisw@sous-sol.org>
parent 39e06a69
...@@ -37,6 +37,8 @@ static void housekeeping_init(struct zd_mac *mac); ...@@ -37,6 +37,8 @@ static void housekeeping_init(struct zd_mac *mac);
static void housekeeping_enable(struct zd_mac *mac); static void housekeeping_enable(struct zd_mac *mac);
static void housekeeping_disable(struct zd_mac *mac); static void housekeeping_disable(struct zd_mac *mac);
static void do_rx(unsigned long mac_ptr);
int zd_mac_init(struct zd_mac *mac, int zd_mac_init(struct zd_mac *mac,
struct net_device *netdev, struct net_device *netdev,
struct usb_interface *intf) struct usb_interface *intf)
...@@ -47,6 +49,10 @@ int zd_mac_init(struct zd_mac *mac, ...@@ -47,6 +49,10 @@ int zd_mac_init(struct zd_mac *mac,
spin_lock_init(&mac->lock); spin_lock_init(&mac->lock);
mac->netdev = netdev; mac->netdev = netdev;
skb_queue_head_init(&mac->rx_queue);
tasklet_init(&mac->rx_tasklet, do_rx, (unsigned long)mac);
tasklet_disable(&mac->rx_tasklet);
ieee_init(ieee); ieee_init(ieee);
softmac_init(ieee80211_priv(netdev)); softmac_init(ieee80211_priv(netdev));
zd_chip_init(&mac->chip, netdev, intf); zd_chip_init(&mac->chip, netdev, intf);
...@@ -132,6 +138,8 @@ int zd_mac_init_hw(struct zd_mac *mac, u8 device_type) ...@@ -132,6 +138,8 @@ int zd_mac_init_hw(struct zd_mac *mac, u8 device_type)
void zd_mac_clear(struct zd_mac *mac) void zd_mac_clear(struct zd_mac *mac)
{ {
skb_queue_purge(&mac->rx_queue);
tasklet_kill(&mac->rx_tasklet);
zd_chip_clear(&mac->chip); zd_chip_clear(&mac->chip);
ZD_ASSERT(!spin_is_locked(&mac->lock)); ZD_ASSERT(!spin_is_locked(&mac->lock));
ZD_MEMCLEAR(mac, sizeof(struct zd_mac)); ZD_MEMCLEAR(mac, sizeof(struct zd_mac));
...@@ -160,6 +168,8 @@ int zd_mac_open(struct net_device *netdev) ...@@ -160,6 +168,8 @@ int zd_mac_open(struct net_device *netdev)
struct zd_chip *chip = &mac->chip; struct zd_chip *chip = &mac->chip;
int r; int r;
tasklet_enable(&mac->rx_tasklet);
r = zd_chip_enable_int(chip); r = zd_chip_enable_int(chip);
if (r < 0) if (r < 0)
goto out; goto out;
...@@ -210,6 +220,8 @@ int zd_mac_stop(struct net_device *netdev) ...@@ -210,6 +220,8 @@ int zd_mac_stop(struct net_device *netdev)
*/ */
zd_chip_disable_rx(chip); zd_chip_disable_rx(chip);
skb_queue_purge(&mac->rx_queue);
tasklet_disable(&mac->rx_tasklet);
housekeeping_disable(mac); housekeeping_disable(mac);
ieee80211softmac_stop(netdev); ieee80211softmac_stop(netdev);
...@@ -873,45 +885,78 @@ static int fill_rx_stats(struct ieee80211_rx_stats *stats, ...@@ -873,45 +885,78 @@ static int fill_rx_stats(struct ieee80211_rx_stats *stats,
return 0; return 0;
} }
int zd_mac_rx(struct zd_mac *mac, const u8 *buffer, unsigned int length) static void zd_mac_rx(struct zd_mac *mac, struct sk_buff *skb)
{ {
int r; int r;
struct ieee80211_device *ieee = zd_mac_to_ieee80211(mac); struct ieee80211_device *ieee = zd_mac_to_ieee80211(mac);
struct ieee80211_rx_stats stats; struct ieee80211_rx_stats stats;
const struct rx_status *status; const struct rx_status *status;
struct sk_buff *skb;
if (length < ZD_PLCP_HEADER_SIZE + IEEE80211_1ADDR_LEN + if (skb->len < ZD_PLCP_HEADER_SIZE + IEEE80211_1ADDR_LEN +
IEEE80211_FCS_LEN + sizeof(struct rx_status)) IEEE80211_FCS_LEN + sizeof(struct rx_status))
return -EINVAL; {
dev_dbg_f(zd_mac_dev(mac), "Packet with length %u to small.\n",
skb->len);
goto free_skb;
}
r = fill_rx_stats(&stats, &status, mac, buffer, length); r = fill_rx_stats(&stats, &status, mac, skb->data, skb->len);
if (r) if (r) {
return r; /* Only packets with rx errors are included here. */
goto free_skb;
}
length -= ZD_PLCP_HEADER_SIZE+IEEE80211_FCS_LEN+ __skb_pull(skb, ZD_PLCP_HEADER_SIZE);
sizeof(struct rx_status); __skb_trim(skb, skb->len -
buffer += ZD_PLCP_HEADER_SIZE; (IEEE80211_FCS_LEN + sizeof(struct rx_status)));
update_qual_rssi(mac, buffer, length, stats.signal, stats.rssi); update_qual_rssi(mac, skb->data, skb->len, stats.signal,
status->signal_strength);
r = filter_rx(ieee, buffer, length, &stats);
if (r <= 0)
return r;
skb = dev_alloc_skb(sizeof(struct zd_rt_hdr) + length); r = filter_rx(ieee, skb->data, skb->len, &stats);
if (!skb) if (r <= 0) {
return -ENOMEM; if (r < 0)
dev_dbg_f(zd_mac_dev(mac), "Error in packet.\n");
goto free_skb;
}
if (ieee->iw_mode == IW_MODE_MONITOR) if (ieee->iw_mode == IW_MODE_MONITOR)
fill_rt_header(skb_put(skb, sizeof(struct zd_rt_hdr)), mac, fill_rt_header(skb_push(skb, sizeof(struct zd_rt_hdr)), mac,
&stats, status); &stats, status);
memcpy(skb_put(skb, length), buffer, length);
r = ieee80211_rx(ieee, skb, &stats); r = ieee80211_rx(ieee, skb, &stats);
if (!r) { if (r)
ZD_ASSERT(in_irq()); return;
dev_kfree_skb_irq(skb);
free_skb:
/* We are always in a soft irq. */
dev_kfree_skb(skb);
}
static void do_rx(unsigned long mac_ptr)
{
struct zd_mac *mac = (struct zd_mac *)mac_ptr;
struct sk_buff *skb;
while ((skb = skb_dequeue(&mac->rx_queue)) != NULL)
zd_mac_rx(mac, skb);
}
int zd_mac_rx_irq(struct zd_mac *mac, const u8 *buffer, unsigned int length)
{
struct sk_buff *skb;
skb = dev_alloc_skb(sizeof(struct zd_rt_hdr) + length);
if (!skb) {
dev_warn(zd_mac_dev(mac), "Could not allocate skb.\n");
return -ENOMEM;
} }
skb_reserve(skb, sizeof(struct zd_rt_hdr));
memcpy(__skb_put(skb, length), buffer, length);
skb_queue_tail(&mac->rx_queue, skb);
tasklet_schedule(&mac->rx_tasklet);
return 0; return 0;
} }
......
...@@ -133,6 +133,10 @@ struct zd_mac { ...@@ -133,6 +133,10 @@ struct zd_mac {
/* Unlocked reading possible */ /* Unlocked reading possible */
struct iw_statistics iw_stats; struct iw_statistics iw_stats;
struct housekeeping housekeeping; struct housekeeping housekeeping;
struct tasklet_struct rx_tasklet;
struct sk_buff_head rx_queue;
unsigned int stats_count; unsigned int stats_count;
u8 qual_buffer[ZD_MAC_STATS_BUFFER_SIZE]; u8 qual_buffer[ZD_MAC_STATS_BUFFER_SIZE];
u8 rssi_buffer[ZD_MAC_STATS_BUFFER_SIZE]; u8 rssi_buffer[ZD_MAC_STATS_BUFFER_SIZE];
...@@ -174,7 +178,7 @@ int zd_mac_open(struct net_device *netdev); ...@@ -174,7 +178,7 @@ int zd_mac_open(struct net_device *netdev);
int zd_mac_stop(struct net_device *netdev); int zd_mac_stop(struct net_device *netdev);
int zd_mac_set_mac_address(struct net_device *dev, void *p); int zd_mac_set_mac_address(struct net_device *dev, void *p);
int zd_mac_rx(struct zd_mac *mac, const u8 *buffer, unsigned int length); int zd_mac_rx_irq(struct zd_mac *mac, const u8 *buffer, unsigned int length);
int zd_mac_set_regdomain(struct zd_mac *zd_mac, u8 regdomain); int zd_mac_set_regdomain(struct zd_mac *zd_mac, u8 regdomain);
u8 zd_mac_get_regdomain(struct zd_mac *zd_mac); u8 zd_mac_get_regdomain(struct zd_mac *zd_mac);
......
...@@ -599,13 +599,13 @@ static void handle_rx_packet(struct zd_usb *usb, const u8 *buffer, ...@@ -599,13 +599,13 @@ static void handle_rx_packet(struct zd_usb *usb, const u8 *buffer,
n = l+k; n = l+k;
if (n > length) if (n > length)
return; return;
zd_mac_rx(mac, buffer+l, k); zd_mac_rx_irq(mac, buffer+l, k);
if (i >= 2) if (i >= 2)
return; return;
l = (n+3) & ~3; l = (n+3) & ~3;
} }
} else { } else {
zd_mac_rx(mac, buffer, length); zd_mac_rx_irq(mac, buffer, length);
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment