adb: avoid potential race conditions in detecting device disconnects
I am hoping this will fix b/2767449 and b/2752393
Change-Id: I230310d37f50eb2a2e1bad31e374427fd44393a5
Signed-off-by: Mike Lockwood <lockwood@android.com>
diff --git a/adb/usb_linux.c b/adb/usb_linux.c
index 7e510df..31fe6f6 100644
--- a/adb/usb_linux.c
+++ b/adb/usb_linux.c
@@ -81,13 +81,41 @@
static void kick_disconnected_device(const char *devname, void *client_data)
{
- usb_handle *usb;
+ usb_handle *h;
adb_mutex_lock(&usb_lock);
/* kick the device if it is in our list */
- for (usb = handle_list.next; usb != &handle_list; usb = usb->next) {
- if (!strcmp(devname, usb_device_get_name(usb->device)))
- usb_kick(usb);
+ for (h = handle_list.next; h != &handle_list; h = h->next) {
+ if (!strcmp(devname, usb_device_get_name(h->device))) {
+ D("[ kicking %p (fd = %s) ]\n", h, usb_device_get_name(h->device));
+ adb_mutex_lock(&h->lock);
+ if(h->dead == 0) {
+ h->dead = 1;
+
+ if (usb_device_is_writeable(h->device)) {
+ /* HACK ALERT!
+ ** Sometimes we get stuck in ioctl(USBDEVFS_REAPURB).
+ ** This is a workaround for that problem.
+ */
+ if (h->reaper_thread) {
+ pthread_kill(h->reaper_thread, SIGALRM);
+ }
+
+ /* cancel any pending transactions
+ ** these will quietly fail if the txns are not active,
+ ** but this ensures that a reader blocked on REAPURB
+ ** will get unblocked
+ */
+ usb_endpoint_cancel(h->ep_in);
+ usb_endpoint_cancel(h->ep_out);
+ adb_cond_broadcast(&h->notify_in);
+ adb_cond_broadcast(&h->notify_out);
+ } else {
+ unregister_usb_transport(h);
+ }
+ }
+ adb_mutex_unlock(&h->lock);
+ }
}
adb_mutex_unlock(&usb_lock);
@@ -405,34 +433,7 @@
void usb_kick(usb_handle *h)
{
- D("[ kicking %p (fd = %s) ]\n", h, usb_device_get_name(h->device));
- adb_mutex_lock(&h->lock);
- if(h->dead == 0) {
- h->dead = 1;
-
- if (usb_device_is_writeable(h->device)) {
- /* HACK ALERT!
- ** Sometimes we get stuck in ioctl(USBDEVFS_REAPURB).
- ** This is a workaround for that problem.
- */
- if (h->reaper_thread) {
- pthread_kill(h->reaper_thread, SIGALRM);
- }
-
- /* cancel any pending transactions
- ** these will quietly fail if the txns are not active,
- ** but this ensures that a reader blocked on REAPURB
- ** will get unblocked
- */
- usb_endpoint_cancel(h->ep_in);
- usb_endpoint_cancel(h->ep_out);
- adb_cond_broadcast(&h->notify_in);
- adb_cond_broadcast(&h->notify_out);
- } else {
- unregister_usb_transport(h);
- }
- }
- adb_mutex_unlock(&h->lock);
+ // do nothing here. we kick in kick_disconnected_devices instead.
}
int usb_close(usb_handle *h)