diff options
| author | Stanislaw Gruszka <sgruszka@redhat.com> | 2011-01-10 06:56:05 -0500 |
|---|---|---|
| committer | John W. Linville <linville@tuxdriver.com> | 2011-01-10 15:40:53 -0500 |
| commit | 4e5518ca53be29c1ec3c00089c97bef36bfed515 (patch) | |
| tree | 99c9405c40b0e8b36c730b40a2760db91142d0b4 | |
| parent | 3c4a8cc46e8cc17910020964689f3faf6bffb8ad (diff) | |
hostap_cs: fix sleeping function called from invalid context
pcmcia_request_irq() and pcmcia_enable_device() are intended
to be called from process context (first function allocate memory
with GFP_KERNEL, second take a mutex). We can not take spin lock
and call them.
It's safe to move spin lock after pcmcia_enable_device() as we
still hold off IRQ until dev->base_addr is 0 and driver will
not proceed with interrupts when is not ready.
Patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=643758
Reported-and-tested-by: rbugz@biobind.com
Cc: stable@kernel.org # 2.6.34+
Signed-off-by: Stanislaw Gruszka <sgruszka@redhat.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
| -rw-r--r-- | drivers/net/wireless/hostap/hostap_cs.c | 15 |
1 files changed, 6 insertions, 9 deletions
diff --git a/drivers/net/wireless/hostap/hostap_cs.c b/drivers/net/wireless/hostap/hostap_cs.c index bd8a4134ede..2176edede39 100644 --- a/drivers/net/wireless/hostap/hostap_cs.c +++ b/drivers/net/wireless/hostap/hostap_cs.c | |||
| @@ -518,22 +518,21 @@ static int prism2_config(struct pcmcia_device *link) | |||
| 518 | hw_priv->link = link; | 518 | hw_priv->link = link; |
| 519 | 519 | ||
| 520 | /* | 520 | /* |
| 521 | * Make sure the IRQ handler cannot proceed until at least | 521 | * We enable IRQ here, but IRQ handler will not proceed |
| 522 | * dev->base_addr is initialized. | 522 | * until dev->base_addr is set below. This protect us from |
| 523 | * receive interrupts when driver is not initialized. | ||
| 523 | */ | 524 | */ |
| 524 | spin_lock_irqsave(&local->irq_init_lock, flags); | ||
| 525 | |||
| 526 | ret = pcmcia_request_irq(link, prism2_interrupt); | 525 | ret = pcmcia_request_irq(link, prism2_interrupt); |
| 527 | if (ret) | 526 | if (ret) |
| 528 | goto failed_unlock; | 527 | goto failed; |
| 529 | 528 | ||
| 530 | ret = pcmcia_enable_device(link); | 529 | ret = pcmcia_enable_device(link); |
| 531 | if (ret) | 530 | if (ret) |
| 532 | goto failed_unlock; | 531 | goto failed; |
| 533 | 532 | ||
| 533 | spin_lock_irqsave(&local->irq_init_lock, flags); | ||
| 534 | dev->irq = link->irq; | 534 | dev->irq = link->irq; |
| 535 | dev->base_addr = link->resource[0]->start; | 535 | dev->base_addr = link->resource[0]->start; |
| 536 | |||
| 537 | spin_unlock_irqrestore(&local->irq_init_lock, flags); | 536 | spin_unlock_irqrestore(&local->irq_init_lock, flags); |
| 538 | 537 | ||
| 539 | local->shutdown = 0; | 538 | local->shutdown = 0; |
| @@ -546,8 +545,6 @@ static int prism2_config(struct pcmcia_device *link) | |||
| 546 | 545 | ||
| 547 | return ret; | 546 | return ret; |
| 548 | 547 | ||
| 549 | failed_unlock: | ||
| 550 | spin_unlock_irqrestore(&local->irq_init_lock, flags); | ||
| 551 | failed: | 548 | failed: |
| 552 | kfree(hw_priv); | 549 | kfree(hw_priv); |
| 553 | prism2_release((u_long)link); | 550 | prism2_release((u_long)link); |
