Apply patch in PR kern/44907 (crash due to race in ehci.c):

- make sure to remove abort_task in ehci_freex
 - always initialize abort_task in ehci_allocx,
   not in ehci_timeout just before adding the task
Also apply similar fixes to ohci and uhci.

XXX: should we also call abort_task handler before removing it from queue
     if *hci_freex() is called for usbd_xfer_handle with queued abort_task?
This commit is contained in:
tsutsui 2011-05-27 19:04:24 +00:00
parent b9e08c16fb
commit a1d8675863
4 changed files with 34 additions and 21 deletions

View File

@ -1,4 +1,4 @@
/* $NetBSD: ehci.c,v 1.175 2011/05/27 17:19:18 drochner Exp $ */
/* $NetBSD: ehci.c,v 1.176 2011/05/27 19:04:24 tsutsui Exp $ */
/*
* Copyright (c) 2004-2008 The NetBSD Foundation, Inc.
@ -52,7 +52,7 @@
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: ehci.c,v 1.175 2011/05/27 17:19:18 drochner Exp $");
__KERNEL_RCSID(0, "$NetBSD: ehci.c,v 1.176 2011/05/27 19:04:24 tsutsui Exp $");
#include "ohci.h"
#include "uhci.h"
@ -1284,6 +1284,7 @@ ehci_allocx(struct usbd_bus *bus)
{
struct ehci_softc *sc = bus->hci_private;
usbd_xfer_handle xfer;
struct ehci_xfer *exfer;
xfer = SIMPLEQ_FIRST(&sc->sc_free_xfers);
if (xfer != NULL) {
@ -1295,12 +1296,14 @@ ehci_allocx(struct usbd_bus *bus)
}
#endif
} else {
xfer = malloc(sizeof(struct ehci_xfer), M_USB, M_NOWAIT);
xfer = malloc(sizeof(*exfer), M_USB, M_NOWAIT);
}
if (xfer != NULL) {
memset(xfer, 0, sizeof(struct ehci_xfer));
exfer = EXFER(xfer);
memset(exfer, 0, sizeof(*exfer));
usb_init_task(&exfer->abort_task, ehci_timeout_task, exfer);
#ifdef DIAGNOSTIC
EXFER(xfer)->isdone = 1;
exfer->isdone = 1;
xfer->busy_free = XFER_BUSY;
#endif
}
@ -1311,6 +1314,7 @@ Static void
ehci_freex(struct usbd_bus *bus, usbd_xfer_handle xfer)
{
struct ehci_softc *sc = bus->hci_private;
struct ehci_xfer *exfer = EXFER(xfer);
#ifdef DIAGNOSTIC
if (xfer->busy_free != XFER_BUSY) {
@ -1318,10 +1322,11 @@ ehci_freex(struct usbd_bus *bus, usbd_xfer_handle xfer)
xfer->busy_free);
}
xfer->busy_free = XFER_FREE;
if (!EXFER(xfer)->isdone) {
if (!exfer->isdone) {
printf("ehci_freex: !isdone\n");
}
#endif
usb_rem_task(xfer->pipe->device, &exfer->abort_task);
SIMPLEQ_INSERT_HEAD(&sc->sc_free_xfers, xfer, next);
}
@ -3150,7 +3155,6 @@ ehci_timeout(void *addr)
}
/* Execute the abort in a process context. */
usb_init_task(&exfer->abort_task, ehci_timeout_task, addr);
usb_add_task(exfer->xfer.pipe->device, &exfer->abort_task,
USB_TASKQ_HC);
}

View File

@ -1,4 +1,4 @@
/* $NetBSD: ohci.c,v 1.212 2010/12/22 01:34:19 macallan Exp $ */
/* $NetBSD: ohci.c,v 1.213 2011/05/27 19:04:24 tsutsui Exp $ */
/* $FreeBSD: src/sys/dev/usb/ohci.c,v 1.22 1999/11/17 22:33:40 n_hibma Exp $ */
/*
@ -41,7 +41,7 @@
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: ohci.c,v 1.212 2010/12/22 01:34:19 macallan Exp $");
__KERNEL_RCSID(0, "$NetBSD: ohci.c,v 1.213 2011/05/27 19:04:24 tsutsui Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@ -945,6 +945,7 @@ ohci_allocx(struct usbd_bus *bus)
{
struct ohci_softc *sc = bus->hci_private;
usbd_xfer_handle xfer;
struct ohci_xfer *oxfer;
xfer = SIMPLEQ_FIRST(&sc->sc_free_xfers);
if (xfer != NULL) {
@ -956,10 +957,12 @@ ohci_allocx(struct usbd_bus *bus)
}
#endif
} else {
xfer = malloc(sizeof(struct ohci_xfer), M_USB, M_NOWAIT);
xfer = malloc(sizeof(*oxfer), M_USB, M_NOWAIT);
}
if (xfer != NULL) {
memset(xfer, 0, sizeof (struct ohci_xfer));
oxfer = OXFER(xfer);
memset(oxfer, 0, sizeof(*oxfer));
usb_init_task(&oxfer->abort_task, ohci_timeout_task, oxfer);
#ifdef DIAGNOSTIC
xfer->busy_free = XFER_BUSY;
#endif
@ -971,6 +974,7 @@ void
ohci_freex(struct usbd_bus *bus, usbd_xfer_handle xfer)
{
struct ohci_softc *sc = bus->hci_private;
struct ohci_xfer *oxfer = OXFER(xfer);
#ifdef DIAGNOSTIC
if (xfer->busy_free != XFER_BUSY) {
@ -979,6 +983,7 @@ ohci_freex(struct usbd_bus *bus, usbd_xfer_handle xfer)
}
xfer->busy_free = XFER_FREE;
#endif
usb_rem_task(xfer->pipe->device, &oxfer->abort_task);
SIMPLEQ_INSERT_HEAD(&sc->sc_free_xfers, xfer, next);
}
@ -1948,7 +1953,6 @@ ohci_timeout(void *addr)
}
/* Execute the abort in a process context. */
usb_init_task(&oxfer->abort_task, ohci_timeout_task, addr);
usb_add_task(oxfer->xfer.pipe->device, &oxfer->abort_task,
USB_TASKQ_HC);
}

View File

@ -1,4 +1,4 @@
/* $NetBSD: ohcivar.h,v 1.49 2010/11/03 22:34:23 dyoung Exp $ */
/* $NetBSD: ohcivar.h,v 1.50 2011/05/27 19:04:24 tsutsui Exp $ */
/* $FreeBSD: src/sys/dev/usb/ohcivar.h,v 1.13 1999/11/17 22:33:41 n_hibma Exp $ */
/*
@ -140,6 +140,7 @@ struct ohci_xfer {
struct usbd_xfer xfer;
struct usb_task abort_task;
};
#define OXFER(xfer) ((struct ohci_xfer *)(xfer))
usbd_status ohci_init(ohci_softc_t *);
int ohci_intr(void *);

View File

@ -1,4 +1,4 @@
/* $NetBSD: uhci.c,v 1.236 2011/05/27 17:19:18 drochner Exp $ */
/* $NetBSD: uhci.c,v 1.237 2011/05/27 19:04:24 tsutsui Exp $ */
/* $FreeBSD: src/sys/dev/usb/uhci.c,v 1.33 1999/11/17 22:33:41 n_hibma Exp $ */
/*
@ -42,7 +42,7 @@
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: uhci.c,v 1.236 2011/05/27 17:19:18 drochner Exp $");
__KERNEL_RCSID(0, "$NetBSD: uhci.c,v 1.237 2011/05/27 19:04:24 tsutsui Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@ -640,6 +640,7 @@ uhci_allocx(struct usbd_bus *bus)
{
struct uhci_softc *sc = bus->hci_private;
usbd_xfer_handle xfer;
struct uhci_xfer *uxfer;
xfer = SIMPLEQ_FIRST(&sc->sc_free_xfers);
if (xfer != NULL) {
@ -651,13 +652,15 @@ uhci_allocx(struct usbd_bus *bus)
}
#endif
} else {
xfer = malloc(sizeof(struct uhci_xfer), M_USB, M_NOWAIT);
xfer = malloc(sizeof(*uxfer), M_USB, M_NOWAIT);
}
if (xfer != NULL) {
memset(xfer, 0, sizeof (struct uhci_xfer));
UXFER(xfer)->iinfo.sc = sc;
uxfer = UXFER(xfer);
memset(uxfer, 0, sizeof(*uxfer));
uxfer->iinfo.sc = sc;
usb_init_task(&uxfer->abort_task, uhci_timeout_task, uxfer);
#ifdef DIAGNOSTIC
UXFER(xfer)->iinfo.isdone = 1;
uxfer->iinfo.isdone = 1;
xfer->busy_free = XFER_BUSY;
#endif
}
@ -668,6 +671,7 @@ void
uhci_freex(struct usbd_bus *bus, usbd_xfer_handle xfer)
{
struct uhci_softc *sc = bus->hci_private;
struct uhci_xfer *uxfer = UXFER(xfer);
#ifdef DIAGNOSTIC
if (xfer->busy_free != XFER_BUSY) {
@ -675,10 +679,11 @@ uhci_freex(struct usbd_bus *bus, usbd_xfer_handle xfer)
xfer->busy_free);
}
xfer->busy_free = XFER_FREE;
if (!UXFER(xfer)->iinfo.isdone) {
if (!uxfer->iinfo.isdone) {
printf("uhci_freex: !isdone\n");
}
#endif
usb_rem_task(xfer->pipe->device, &uxfer->abort_task);
SIMPLEQ_INSERT_HEAD(&sc->sc_free_xfers, xfer, next);
}
@ -1619,7 +1624,6 @@ uhci_timeout(void *addr)
}
/* Execute the abort in a process context. */
usb_init_task(&uxfer->abort_task, uhci_timeout_task, ii->xfer);
usb_add_task(uxfer->xfer.pipe->device, &uxfer->abort_task,
USB_TASKQ_HC);
}