mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-10 15:10:38 +00:00
rt2x00usb: fix rx queue hang
Since commit ed194d136769 ("usb: core: remove local_irq_save() around ->complete() handler") the handler rt2x00usb_interrupt_rxdone() is not running with interrupts disabled anymore. So this completion handler is not guaranteed to run completely before workqueue processing starts for the same queue entry. Be sure to set all other flags in the entry correctly before marking this entry ready for workqueue processing. This way we cannot miss error conditions that need to be signalled from the completion handler to the worker thread. Note that rt2x00usb_work_rxdone() processes all available entries, not only such for which queue_work() was called. This patch is similar to what commit df71c9cfceea ("rt2x00: fix order of entry flags modification") did for TX processing. This fixes a regression on a RT5370 based wifi stick in AP mode, which suddenly stopped data transmission after some period of heavy load. Also stopping the hanging hostapd resulted in the error message "ieee80211 phy0: rt2x00queue_flush_queue: Warning - Queue 14 failed to flush". Other operation modes are probably affected as well, this just was the used testcase. Fixes: ed194d136769 ("usb: core: remove local_irq_save() around ->complete() handler") Cc: stable@vger.kernel.org # 4.20+ Signed-off-by: Soeren Moch <smoch@web.de> Acked-by: Stanislaw Gruszka <sgruszka@redhat.com> Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
This commit is contained in:
parent
ff414f31ce
commit
41a531ffa4
@ -372,14 +372,9 @@ static void rt2x00usb_interrupt_rxdone(struct urb *urb)
|
|||||||
struct queue_entry *entry = (struct queue_entry *)urb->context;
|
struct queue_entry *entry = (struct queue_entry *)urb->context;
|
||||||
struct rt2x00_dev *rt2x00dev = entry->queue->rt2x00dev;
|
struct rt2x00_dev *rt2x00dev = entry->queue->rt2x00dev;
|
||||||
|
|
||||||
if (!test_and_clear_bit(ENTRY_OWNER_DEVICE_DATA, &entry->flags))
|
if (!test_bit(ENTRY_OWNER_DEVICE_DATA, &entry->flags))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/*
|
|
||||||
* Report the frame as DMA done
|
|
||||||
*/
|
|
||||||
rt2x00lib_dmadone(entry);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Check if the received data is simply too small
|
* Check if the received data is simply too small
|
||||||
* to be actually valid, or if the urb is signaling
|
* to be actually valid, or if the urb is signaling
|
||||||
@ -388,6 +383,11 @@ static void rt2x00usb_interrupt_rxdone(struct urb *urb)
|
|||||||
if (urb->actual_length < entry->queue->desc_size || urb->status)
|
if (urb->actual_length < entry->queue->desc_size || urb->status)
|
||||||
set_bit(ENTRY_DATA_IO_FAILED, &entry->flags);
|
set_bit(ENTRY_DATA_IO_FAILED, &entry->flags);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Report the frame as DMA done
|
||||||
|
*/
|
||||||
|
rt2x00lib_dmadone(entry);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Schedule the delayed work for reading the RX status
|
* Schedule the delayed work for reading the RX status
|
||||||
* from the device.
|
* from the device.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user