core: Implement new transfer completion API
authorChris Dickens <christopher.a.dickens@gmail.com>
Wed, 21 Jan 2015 08:18:40 +0000 (00:18 -0800)
committerChris Dickens <christopher.a.dickens@gmail.com>
Tue, 27 Jan 2015 02:57:04 +0000 (18:57 -0800)
Signed-off-by: Chris Dickens <christopher.a.dickens@gmail.com>
libusb/io.c
libusb/libusbi.h
libusb/version_nano.h

index d8a2bd6c84ac82ba0e20f23c5d8329fac6cbface..ba4e06c4ddb01528bd4656be8ab29b80b168d7e7 100644 (file)
@@ -1126,6 +1126,7 @@ int usbi_io_init(struct libusb_context *ctx)
        list_init(&ctx->flying_transfers);
        list_init(&ctx->ipollfds);
        list_init(&ctx->hotplug_msgs);
+       list_init(&ctx->completed_transfers);
 
        /* FIXME should use an eventfd on kernels that support it */
        r = usbi_pipe(ctx->event_pipe);
@@ -1632,6 +1633,22 @@ int usbi_handle_transfer_cancellation(struct usbi_transfer *transfer)
        return usbi_handle_transfer_completion(transfer, LIBUSB_TRANSFER_CANCELLED);
 }
 
+/* Add a completed transfer to the completed_transfers list of the
+ * context and signal the event. The backend's handle_transfer_completion()
+ * function will be called the next time an event handler runs. */
+void usbi_signal_transfer_completion(struct usbi_transfer *transfer)
+{
+       struct libusb_context *ctx = ITRANSFER_CTX(transfer);
+       int pending_events;
+
+       usbi_mutex_lock(&ctx->event_data_lock);
+       pending_events = usbi_pending_events(ctx);
+       list_add_tail(&transfer->completed_list, &ctx->completed_transfers);
+       if (!pending_events)
+               usbi_signal_event(ctx);
+       usbi_mutex_unlock(&ctx->event_data_lock);
+}
+
 /** \ingroup poll
  * Attempt to acquire the event handling lock. This lock is used to ensure that
  * only one thread is monitoring libusb event sources at any one time.
@@ -2056,6 +2073,8 @@ redo_poll:
        /* fds[0] is always the event pipe */
        if (fds[0].revents) {
                libusb_hotplug_message *message = NULL;
+               struct usbi_transfer *itransfer;
+               int ret = 0;
 
                usbi_dbg("caught a fish on the event pipe");
 
@@ -2078,6 +2097,17 @@ redo_poll:
                        list_del(&message->list);
                }
 
+               /* complete any pending transfers */
+               while (ret == 0 && !list_empty(&ctx->completed_transfers)) {
+                       itransfer = list_first_entry(&ctx->completed_transfers, struct usbi_transfer, completed_list);
+                       list_del(&itransfer->completed_list);
+                       usbi_mutex_unlock(&ctx->event_data_lock);
+                       ret = usbi_backend->handle_transfer_completion(itransfer);
+                       if (ret)
+                               usbi_err(ctx, "backend handle_transfer_completion failed with error %d", ret);
+                       usbi_mutex_lock(&ctx->event_data_lock);
+               }
+
                /* if no further pending events, clear the event pipe */
                if (!usbi_pending_events(ctx))
                        usbi_clear_event(ctx);
@@ -2095,6 +2125,12 @@ redo_poll:
                        free(message);
                }
 
+               if (ret) {
+                       /* return error code */
+                       r = ret;
+                       goto handled;
+               }
+
                if (0 == --r)
                        goto handled;
        }
index 757e1ec33232eae5bb2ea657f58faacb89a3b041..deb1570ff1133297b0172997d039fe12f8bf2c51 100644 (file)
@@ -394,6 +394,7 @@ enum {
 struct usbi_transfer {
        int num_iso_packets;
        struct list_head list;
+       struct list_head completed_list;
        struct timeval timeout;
        int transferred;
        uint32_t stream_id;
index a291c7437645e2ccd9d303880833eb00846d875d..b9f504cd8416abab15d6239b1a25f09ec245be71 100644 (file)
@@ -1 +1 @@
-#define LIBUSB_NANO 10953
+#define LIBUSB_NANO 10954