rt2x00: Add "flush" queue command
authorIvo van Doorn <ivdoorn@gmail.com>
Mon, 13 Dec 2010 11:35:40 +0000 (12:35 +0100)
committerJohn W. Linville <linville@tuxdriver.com>
Mon, 13 Dec 2010 20:23:35 +0000 (15:23 -0500)
Add a new command to the queue handlers: "flush",
this moves the flush() callback from mac80211
into rt2x00queue and adds support for flushing
the RX queue as well.

Signed-off-by: Ivo van Doorn <IvDoorn@gmail.com>
Acked-by: Helmut Schaa <helmut.schaa@googlemail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>

drivers/net/wireless/rt2x00/rt2500usb.c
drivers/net/wireless/rt2x00/rt2800usb.c
drivers/net/wireless/rt2x00/rt2x00.h
drivers/net/wireless/rt2x00/rt2x00dev.c
drivers/net/wireless/rt2x00/rt2x00mac.c
drivers/net/wireless/rt2x00/rt2x00queue.c
drivers/net/wireless/rt2x00/rt2x00usb.c
drivers/net/wireless/rt2x00/rt2x00usb.h
drivers/net/wireless/rt2x00/rt73usb.c

index a56b38f..6b3b1de 100644 (file)
@@ -785,8 +785,6 @@ static void rt2500usb_stop_queue(struct data_queue *queue)
        default:
                break;
        }
-
-       rt2x00usb_stop_queue(queue);
 }
 
 /*
@@ -1842,6 +1840,7 @@ static const struct rt2x00lib_ops rt2500usb_rt2x00_ops = {
        .start_queue            = rt2500usb_start_queue,
        .kick_queue             = rt2x00usb_kick_queue,
        .stop_queue             = rt2500usb_stop_queue,
+       .flush_queue            = rt2x00usb_flush_queue,
        .write_tx_desc          = rt2500usb_write_tx_desc,
        .write_beacon           = rt2500usb_write_beacon,
        .get_tx_data_len        = rt2500usb_get_tx_data_len,
index 60b5503..3e0205d 100644 (file)
@@ -96,8 +96,6 @@ static void rt2800usb_stop_queue(struct data_queue *queue)
        default:
                break;
        }
-
-       rt2x00usb_stop_queue(queue);
 }
 
 /*
@@ -623,6 +621,7 @@ static const struct rt2x00lib_ops rt2800usb_rt2x00_ops = {
        .start_queue            = rt2800usb_start_queue,
        .kick_queue             = rt2x00usb_kick_queue,
        .stop_queue             = rt2800usb_stop_queue,
+       .flush_queue            = rt2x00usb_flush_queue,
        .write_tx_desc          = rt2800usb_write_tx_desc,
        .write_tx_data          = rt2800usb_write_tx_data,
        .write_beacon           = rt2800_write_beacon,
index ac7c3d8..1d7b481 100644 (file)
@@ -575,6 +575,7 @@ struct rt2x00lib_ops {
        void (*start_queue) (struct data_queue *queue);
        void (*kick_queue) (struct data_queue *queue);
        void (*stop_queue) (struct data_queue *queue);
+       void (*flush_queue) (struct data_queue *queue);
 
        /*
         * TX control handlers
@@ -1109,6 +1110,16 @@ void rt2x00queue_start_queue(struct data_queue *queue);
 void rt2x00queue_stop_queue(struct data_queue *queue);
 
 /**
+ * rt2x00queue_flush_queue - Flush a data queue
+ * @queue: Pointer to &struct data_queue.
+ * @drop: True to drop all pending frames.
+ *
+ * This function will flush the queue. After this call
+ * the queue is guarenteed to be empty.
+ */
+void rt2x00queue_flush_queue(struct data_queue *queue, bool drop);
+
+/**
  * rt2x00queue_start_queues - Start all data queues
  * @rt2x00dev: Pointer to &struct rt2x00_dev.
  *
@@ -1125,6 +1136,16 @@ void rt2x00queue_start_queues(struct rt2x00_dev *rt2x00dev);
  */
 void rt2x00queue_stop_queues(struct rt2x00_dev *rt2x00dev);
 
+/**
+ * rt2x00queue_flush_queues - Flush all data queues
+ * @rt2x00dev: Pointer to &struct rt2x00_dev.
+ * @drop: True to drop all pending frames.
+ *
+ * This function will loop through all available queues to flush
+ * any pending frames.
+ */
+void rt2x00queue_flush_queues(struct rt2x00_dev *rt2x00dev, bool drop);
+
 /*
  * Debugfs handlers.
  */
index e428162..9ef5a24 100644 (file)
@@ -94,6 +94,7 @@ void rt2x00lib_disable_radio(struct rt2x00_dev *rt2x00dev)
         */
        rt2x00link_stop_tuner(rt2x00dev);
        rt2x00queue_stop_queues(rt2x00dev);
+       rt2x00queue_flush_queues(rt2x00dev, true);
 
        /*
         * Disable radio.
index c4abb20..4cac7ad 100644 (file)
@@ -718,36 +718,8 @@ void rt2x00mac_flush(struct ieee80211_hw *hw, bool drop)
 {
        struct rt2x00_dev *rt2x00dev = hw->priv;
        struct data_queue *queue;
-       unsigned int i = 0;
 
-       ieee80211_stop_queues(hw);
-
-       /*
-        * Run over all queues to kick them, this will force
-        * any pending frames to be transmitted.
-        */
-       tx_queue_for_each(rt2x00dev, queue) {
-               rt2x00dev->ops->lib->kick_queue(queue);
-       }
-
-       /**
-        * All queues have been kicked, now wait for each queue
-        * to become empty. With a bit of luck, we only have to wait
-        * for the first queue to become empty, because while waiting
-        * for the that queue, the other queues will have transmitted
-        * all their frames as well (since they were already kicked).
-        */
-       tx_queue_for_each(rt2x00dev, queue) {
-               for (i = 0; i < 10; i++) {
-                       if (rt2x00queue_empty(queue))
-                               break;
-                       msleep(100);
-               }
-
-               if (!rt2x00queue_empty(queue))
-                       WARNING(rt2x00dev, "Failed to flush queue %d\n", queue->qid);
-       }
-
-       ieee80211_wake_queues(hw);
+       tx_queue_for_each(rt2x00dev, queue)
+               rt2x00queue_flush_queue(queue, drop);
 }
 EXPORT_SYMBOL_GPL(rt2x00mac_flush);
index 558965f..313a8fa 100644 (file)
@@ -780,6 +780,12 @@ void rt2x00queue_unpause_queue(struct data_queue *queue)
                 */
                ieee80211_wake_queue(queue->rt2x00dev->hw, queue->qid);
                break;
+       case QID_RX:
+               /*
+                * For RX we need to kick the queue now in order to
+                * receive frames.
+                */
+               queue->rt2x00dev->ops->lib->kick_queue(queue);
        default:
                break;
        }
@@ -823,6 +829,74 @@ void rt2x00queue_stop_queue(struct data_queue *queue)
 }
 EXPORT_SYMBOL_GPL(rt2x00queue_stop_queue);
 
+void rt2x00queue_flush_queue(struct data_queue *queue, bool drop)
+{
+       unsigned int i;
+       bool started;
+       bool tx_queue =
+               (queue->qid == QID_AC_BE) ||
+               (queue->qid == QID_AC_BK) ||
+               (queue->qid == QID_AC_VI) ||
+               (queue->qid == QID_AC_VO);
+
+       mutex_lock(&queue->status_lock);
+
+       /*
+        * If the queue has been started, we must stop it temporarily
+        * to prevent any new frames to be queued on the device. If
+        * we are not dropping the pending frames, the queue must
+        * only be stopped in the software and not the hardware,
+        * otherwise the queue will never become empty on its own.
+        */
+       started = test_bit(QUEUE_STARTED, &queue->flags);
+       if (started) {
+               /*
+                * Pause the queue
+                */
+               rt2x00queue_pause_queue(queue);
+
+               /*
+                * If we are not supposed to drop any pending
+                * frames, this means we must force a start (=kick)
+                * to the queue to make sure the hardware will
+                * start transmitting.
+                */
+               if (!drop && tx_queue)
+                       queue->rt2x00dev->ops->lib->kick_queue(queue);
+       }
+
+       /*
+        * Check if driver supports flushing, we can only guarentee
+        * full support for flushing if the driver is able
+        * to cancel all pending frames (drop = true).
+        */
+       if (drop && queue->rt2x00dev->ops->lib->flush_queue)
+               queue->rt2x00dev->ops->lib->flush_queue(queue);
+
+       /*
+        * When we don't want to drop any frames, or when
+        * the driver doesn't fully flush the queue correcly,
+        * we must wait for the queue to become empty.
+        */
+       for (i = 0; !rt2x00queue_empty(queue) && i < 100; i++)
+               msleep(10);
+
+       /*
+        * The queue flush has failed...
+        */
+       if (unlikely(!rt2x00queue_empty(queue)))
+               WARNING(queue->rt2x00dev, "Queue %d failed to flush", queue->qid);
+
+       /*
+        * Restore the queue to the previous status
+        */
+       if (started)
+               rt2x00queue_unpause_queue(queue);
+
+       mutex_unlock(&queue->status_lock);
+}
+EXPORT_SYMBOL_GPL(rt2x00queue_flush_queue);
+
 void rt2x00queue_start_queues(struct rt2x00_dev *rt2x00dev)
 {
        struct data_queue *queue;
@@ -857,6 +931,17 @@ void rt2x00queue_stop_queues(struct rt2x00_dev *rt2x00dev)
 }
 EXPORT_SYMBOL_GPL(rt2x00queue_stop_queues);
 
+void rt2x00queue_flush_queues(struct rt2x00_dev *rt2x00dev, bool drop)
+{
+       struct data_queue *queue;
+
+       tx_queue_for_each(rt2x00dev, queue)
+               rt2x00queue_flush_queue(queue, drop);
+
+       rt2x00queue_flush_queue(rt2x00dev->rx, drop);
+}
+EXPORT_SYMBOL_GPL(rt2x00queue_flush_queues);
+
 static void rt2x00queue_reset(struct data_queue *queue)
 {
        unsigned long irqflags;
index fca29ae..cd80eec 100644 (file)
@@ -366,7 +366,7 @@ void rt2x00usb_kick_queue(struct data_queue *queue)
 }
 EXPORT_SYMBOL_GPL(rt2x00usb_kick_queue);
 
-static void rt2x00usb_kill_entry(struct queue_entry *entry)
+static void rt2x00usb_flush_entry(struct queue_entry *entry)
 {
        struct rt2x00_dev *rt2x00dev = entry->queue->rt2x00dev;
        struct queue_entry_priv_usb *entry_priv = entry->priv_data;
@@ -385,37 +385,61 @@ static void rt2x00usb_kill_entry(struct queue_entry *entry)
                usb_kill_urb(bcn_priv->guardian_urb);
 }
 
-void rt2x00usb_stop_queue(struct data_queue *queue)
+void rt2x00usb_flush_queue(struct data_queue *queue)
 {
+       struct work_struct *completion;
+       unsigned int i;
+
        rt2x00queue_for_each_entry(queue, Q_INDEX_DONE, Q_INDEX,
-                                  rt2x00usb_kill_entry);
+                                  rt2x00usb_flush_entry);
+
+       /*
+        * Obtain the queue completion handler
+        */
+       switch (queue->qid) {
+       case QID_AC_BE:
+       case QID_AC_BK:
+       case QID_AC_VI:
+       case QID_AC_VO:
+               completion = &queue->rt2x00dev->txdone_work;
+               break;
+       case QID_RX:
+               completion = &queue->rt2x00dev->rxdone_work;
+               break;
+       default:
+               return;
+       }
+
+       for (i = 0; i < 20; i++) {
+               /*
+                * Check if the driver is already done, otherwise we
+                * have to sleep a little while to give the driver/hw
+                * the oppurtunity to complete interrupt process itself.
+                */
+               if (rt2x00queue_empty(queue))
+                       break;
+
+               /*
+                * Schedule the completion handler manually, when this
+                * worker function runs, it should cleanup the queue.
+                */
+               ieee80211_queue_work(queue->rt2x00dev->hw, completion);
+
+               /*
+                * Wait for a little while to give the driver
+                * the oppurtunity to recover itself.
+                */
+               msleep(10);
+       }
 }
-EXPORT_SYMBOL_GPL(rt2x00usb_stop_queue);
+EXPORT_SYMBOL_GPL(rt2x00usb_flush_queue);
 
 static void rt2x00usb_watchdog_tx_dma(struct data_queue *queue)
 {
-       struct rt2x00_dev *rt2x00dev = queue->rt2x00dev;
-
        WARNING(queue->rt2x00dev, "TX queue %d DMA timed out,"
                " invoke forced forced reset\n", queue->qid);
 
-       /*
-        * Temporarily disable the TX queue, this will force mac80211
-        * to use the other queues until this queue has been restored.
-        */
-       rt2x00queue_stop_queue(queue);
-
-       /*
-        * In case that a driver has overriden the txdone_work
-        * function, we invoke the TX done through there.
-        */
-       rt2x00dev->txdone_work.func(&rt2x00dev->txdone_work);
-
-       /*
-        * The queue has been reset, and mac80211 is allowed to use the
-        * queue again.
-        */
-       rt2x00queue_start_queue(queue);
+       rt2x00queue_flush_queue(queue, true);
 }
 
 static void rt2x00usb_watchdog_tx_status(struct data_queue *queue)
index 05a5424..6aaf51f 100644 (file)
@@ -387,13 +387,13 @@ struct queue_entry_priv_usb_bcn {
 void rt2x00usb_kick_queue(struct data_queue *queue);
 
 /**
- * rt2x00usb_stop_queue - Stop data queue
+ * rt2x00usb_flush_queue - Flush data queue
  * @queue: Data queue to stop
  *
  * This will walk through all entries of the queue and kill all
  * URB's which were send to the device.
  */
-void rt2x00usb_stop_queue(struct data_queue *queue);
+void rt2x00usb_flush_queue(struct data_queue *queue);
 
 /**
  * rt2x00usb_watchdog - Watchdog for USB communication
index f55e74e..0b3959b 100644 (file)
@@ -1077,8 +1077,6 @@ static void rt73usb_stop_queue(struct data_queue *queue)
        default:
                break;
        }
-
-       rt2x00usb_stop_queue(queue);
 }
 
 /*
@@ -2309,6 +2307,7 @@ static const struct rt2x00lib_ops rt73usb_rt2x00_ops = {
        .start_queue            = rt73usb_start_queue,
        .kick_queue             = rt2x00usb_kick_queue,
        .stop_queue             = rt73usb_stop_queue,
+       .flush_queue            = rt2x00usb_flush_queue,
        .write_tx_desc          = rt73usb_write_tx_desc,
        .write_beacon           = rt73usb_write_beacon,
        .get_tx_data_len        = rt73usb_get_tx_data_len,