Commit 1b4cd43b authored by Alan Stern's avatar Alan Stern Committed by Greg Kroah-Hartman

isp116x-hcd: prepare for urb->status

This patch (as931b), adapted from a patch by Olav Kongas, makes a small
set of conservative changes to the isp116x-hcd driver in preparation
for the removal of urb->status.

	finish_request() is moved up in the source and is called
	as soon as the URB is known to have completed, rather than
	after all the active endpoints have been scanned.

	The status of a completed URB is kept in a local variable
	and copied to urb->status only when the URB is about to be
	given back.

	-EREMOTEIO error status for control transfers is set after
	the status stage rather than when the short packet arrives.

	Some unnecessary uses of urb->lock are removed.
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Cc: Olav Kongas <ok@artecdesign.ee>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 10e48522
...@@ -228,7 +228,6 @@ static void preproc_atl_queue(struct isp116x *isp116x) ...@@ -228,7 +228,6 @@ static void preproc_atl_queue(struct isp116x *isp116x)
struct urb, urb_list); struct urb, urb_list);
ptd = &ep->ptd; ptd = &ep->ptd;
len = ep->length; len = ep->length;
spin_lock(&urb->lock);
ep->data = (unsigned char *)urb->transfer_buffer ep->data = (unsigned char *)urb->transfer_buffer
+ urb->actual_length; + urb->actual_length;
...@@ -264,7 +263,6 @@ static void preproc_atl_queue(struct isp116x *isp116x) ...@@ -264,7 +263,6 @@ static void preproc_atl_queue(struct isp116x *isp116x)
| PTD_EP(ep->epnum); | PTD_EP(ep->epnum);
ptd->len = PTD_LEN(len) | PTD_DIR(dir); ptd->len = PTD_LEN(len) | PTD_DIR(dir);
ptd->faddr = PTD_FA(usb_pipedevice(urb->pipe)); ptd->faddr = PTD_FA(usb_pipedevice(urb->pipe));
spin_unlock(&urb->lock);
if (!ep->active) { if (!ep->active) {
ptd->mps |= PTD_LAST_MSK; ptd->mps |= PTD_LAST_MSK;
isp116x->atl_last_dir = dir; isp116x->atl_last_dir = dir;
...@@ -274,6 +272,61 @@ static void preproc_atl_queue(struct isp116x *isp116x) ...@@ -274,6 +272,61 @@ static void preproc_atl_queue(struct isp116x *isp116x)
} }
} }
/*
Take done or failed requests out of schedule. Give back
processed urbs.
*/
static void finish_request(struct isp116x *isp116x, struct isp116x_ep *ep,
struct urb *urb)
__releases(isp116x->lock) __acquires(isp116x->lock)
{
unsigned i;
urb->hcpriv = NULL;
ep->error_count = 0;
if (usb_pipecontrol(urb->pipe))
ep->nextpid = USB_PID_SETUP;
urb_dbg(urb, "Finish");
spin_unlock(&isp116x->lock);
usb_hcd_giveback_urb(isp116x_to_hcd(isp116x), urb);
spin_lock(&isp116x->lock);
/* take idle endpoints out of the schedule */
if (!list_empty(&ep->hep->urb_list))
return;
/* async deschedule */
if (!list_empty(&ep->schedule)) {
list_del_init(&ep->schedule);
return;
}
/* periodic deschedule */
DBG("deschedule qh%d/%p branch %d\n", ep->period, ep, ep->branch);
for (i = ep->branch; i < PERIODIC_SIZE; i += ep->period) {
struct isp116x_ep *temp;
struct isp116x_ep **prev = &isp116x->periodic[i];
while (*prev && ((temp = *prev) != ep))
prev = &temp->next;
if (*prev)
*prev = ep->next;
isp116x->load[i] -= ep->load;
}
ep->branch = PERIODIC_SIZE;
isp116x_to_hcd(isp116x)->self.bandwidth_allocated -=
ep->load / ep->period;
/* switch irq type? */
if (!--isp116x->periodic_count) {
isp116x->irqenb &= ~HCuPINT_SOF;
isp116x->irqenb |= HCuPINT_ATL;
}
}
/* /*
Analyze transfer results, handle partial transfers and errors Analyze transfer results, handle partial transfers and errors
*/ */
...@@ -284,6 +337,7 @@ static void postproc_atl_queue(struct isp116x *isp116x) ...@@ -284,6 +337,7 @@ static void postproc_atl_queue(struct isp116x *isp116x)
struct usb_device *udev; struct usb_device *udev;
struct ptd *ptd; struct ptd *ptd;
int short_not_ok; int short_not_ok;
int status;
u8 cc; u8 cc;
for (ep = isp116x->atl_active; ep; ep = ep->active) { for (ep = isp116x->atl_active; ep; ep = ep->active) {
...@@ -294,7 +348,7 @@ static void postproc_atl_queue(struct isp116x *isp116x) ...@@ -294,7 +348,7 @@ static void postproc_atl_queue(struct isp116x *isp116x)
ptd = &ep->ptd; ptd = &ep->ptd;
cc = PTD_GET_CC(ptd); cc = PTD_GET_CC(ptd);
short_not_ok = 1; short_not_ok = 1;
spin_lock(&urb->lock); status = -EINPROGRESS;
/* Data underrun is special. For allowed underrun /* Data underrun is special. For allowed underrun
we clear the error and continue as normal. For we clear the error and continue as normal. For
...@@ -302,47 +356,36 @@ static void postproc_atl_queue(struct isp116x *isp116x) ...@@ -302,47 +356,36 @@ static void postproc_atl_queue(struct isp116x *isp116x)
immediately while for control transfer, immediately while for control transfer,
we do a STATUS stage. */ we do a STATUS stage. */
if (cc == TD_DATAUNDERRUN) { if (cc == TD_DATAUNDERRUN) {
if (!(urb->transfer_flags & URB_SHORT_NOT_OK)) { if (!(urb->transfer_flags & URB_SHORT_NOT_OK) ||
DBG("Allowed data underrun\n"); usb_pipecontrol(urb->pipe)) {
DBG("Allowed or control data underrun\n");
cc = TD_CC_NOERROR; cc = TD_CC_NOERROR;
short_not_ok = 0; short_not_ok = 0;
} else { } else {
ep->error_count = 1; ep->error_count = 1;
if (usb_pipecontrol(urb->pipe)) usb_settoggle(udev, ep->epnum,
ep->nextpid = USB_PID_ACK; ep->nextpid == USB_PID_OUT,
else PTD_GET_TOGGLE(ptd));
usb_settoggle(udev, ep->epnum,
ep->nextpid ==
USB_PID_OUT,
PTD_GET_TOGGLE(ptd));
urb->actual_length += PTD_GET_COUNT(ptd); urb->actual_length += PTD_GET_COUNT(ptd);
urb->status = cc_to_error[TD_DATAUNDERRUN]; status = cc_to_error[TD_DATAUNDERRUN];
spin_unlock(&urb->lock); goto done;
continue;
} }
} }
/* Keep underrun error through the STATUS stage */
if (urb->status == cc_to_error[TD_DATAUNDERRUN])
cc = TD_DATAUNDERRUN;
if (cc != TD_CC_NOERROR && cc != TD_NOTACCESSED if (cc != TD_CC_NOERROR && cc != TD_NOTACCESSED
&& (++ep->error_count >= 3 || cc == TD_CC_STALL && (++ep->error_count >= 3 || cc == TD_CC_STALL
|| cc == TD_DATAOVERRUN)) { || cc == TD_DATAOVERRUN)) {
if (urb->status == -EINPROGRESS) status = cc_to_error[cc];
urb->status = cc_to_error[cc];
if (ep->nextpid == USB_PID_ACK) if (ep->nextpid == USB_PID_ACK)
ep->nextpid = 0; ep->nextpid = 0;
spin_unlock(&urb->lock); goto done;
continue;
} }
/* According to usb spec, zero-length Int transfer signals /* According to usb spec, zero-length Int transfer signals
finishing of the urb. Hey, does this apply only finishing of the urb. Hey, does this apply only
for IN endpoints? */ for IN endpoints? */
if (usb_pipeint(urb->pipe) && !PTD_GET_LEN(ptd)) { if (usb_pipeint(urb->pipe) && !PTD_GET_LEN(ptd)) {
if (urb->status == -EINPROGRESS) status = 0;
urb->status = 0; goto done;
spin_unlock(&urb->lock);
continue;
} }
/* Relax after previously failed, but later succeeded /* Relax after previously failed, but later succeeded
...@@ -381,8 +424,8 @@ static void postproc_atl_queue(struct isp116x *isp116x) ...@@ -381,8 +424,8 @@ static void postproc_atl_queue(struct isp116x *isp116x)
/* All data for this URB is transferred, let's finish */ /* All data for this URB is transferred, let's finish */
if (usb_pipecontrol(urb->pipe)) if (usb_pipecontrol(urb->pipe))
ep->nextpid = USB_PID_ACK; ep->nextpid = USB_PID_ACK;
else if (urb->status == -EINPROGRESS) else
urb->status = 0; status = 0;
break; break;
case USB_PID_SETUP: case USB_PID_SETUP:
if (PTD_GET_ACTIVE(ptd) if (PTD_GET_ACTIVE(ptd)
...@@ -402,69 +445,27 @@ static void postproc_atl_queue(struct isp116x *isp116x) ...@@ -402,69 +445,27 @@ static void postproc_atl_queue(struct isp116x *isp116x)
if (PTD_GET_ACTIVE(ptd) if (PTD_GET_ACTIVE(ptd)
|| (cc != TD_CC_NOERROR && cc < 0x0E)) || (cc != TD_CC_NOERROR && cc < 0x0E))
break; break;
if (urb->status == -EINPROGRESS) if ((urb->transfer_flags & URB_SHORT_NOT_OK) &&
urb->status = 0; urb->actual_length <
urb->transfer_buffer_length)
status = -EREMOTEIO;
else
status = 0;
ep->nextpid = 0; ep->nextpid = 0;
break; break;
default: default:
BUG(); BUG();
} }
spin_unlock(&urb->lock);
}
}
/*
Take done or failed requests out of schedule. Give back
processed urbs.
*/
static void finish_request(struct isp116x *isp116x, struct isp116x_ep *ep,
struct urb *urb)
__releases(isp116x->lock) __acquires(isp116x->lock)
{
unsigned i;
urb->hcpriv = NULL;
ep->error_count = 0;
if (usb_pipecontrol(urb->pipe))
ep->nextpid = USB_PID_SETUP;
urb_dbg(urb, "Finish");
spin_unlock(&isp116x->lock);
usb_hcd_giveback_urb(isp116x_to_hcd(isp116x), urb);
spin_lock(&isp116x->lock);
/* take idle endpoints out of the schedule */
if (!list_empty(&ep->hep->urb_list))
return;
/* async deschedule */
if (!list_empty(&ep->schedule)) {
list_del_init(&ep->schedule);
return;
}
/* periodic deschedule */ done:
DBG("deschedule qh%d/%p branch %d\n", ep->period, ep, ep->branch); if (status != -EINPROGRESS) {
for (i = ep->branch; i < PERIODIC_SIZE; i += ep->period) { spin_lock(&urb->lock);
struct isp116x_ep *temp; if (urb->status == -EINPROGRESS)
struct isp116x_ep **prev = &isp116x->periodic[i]; urb->status = status;
spin_unlock(&urb->lock);
while (*prev && ((temp = *prev) != ep)) }
prev = &temp->next; if (urb->status != -EINPROGRESS)
if (*prev) finish_request(isp116x, ep, urb);
*prev = ep->next;
isp116x->load[i] -= ep->load;
}
ep->branch = PERIODIC_SIZE;
isp116x_to_hcd(isp116x)->self.bandwidth_allocated -=
ep->load / ep->period;
/* switch irq type? */
if (!--isp116x->periodic_count) {
isp116x->irqenb &= ~HCuPINT_SOF;
isp116x->irqenb |= HCuPINT_ATL;
} }
} }
...@@ -570,9 +571,6 @@ static void start_atl_transfers(struct isp116x *isp116x) ...@@ -570,9 +571,6 @@ static void start_atl_transfers(struct isp116x *isp116x)
*/ */
static void finish_atl_transfers(struct isp116x *isp116x) static void finish_atl_transfers(struct isp116x *isp116x)
{ {
struct isp116x_ep *ep;
struct urb *urb;
if (!isp116x->atl_active) if (!isp116x->atl_active)
return; return;
/* Fifo not ready? */ /* Fifo not ready? */
...@@ -582,16 +580,6 @@ static void finish_atl_transfers(struct isp116x *isp116x) ...@@ -582,16 +580,6 @@ static void finish_atl_transfers(struct isp116x *isp116x)
atomic_inc(&isp116x->atl_finishing); atomic_inc(&isp116x->atl_finishing);
unpack_fifo(isp116x); unpack_fifo(isp116x);
postproc_atl_queue(isp116x); postproc_atl_queue(isp116x);
for (ep = isp116x->atl_active; ep; ep = ep->active) {
urb =
container_of(ep->hep->urb_list.next, struct urb, urb_list);
/* USB_PID_ACK check here avoids finishing of
control transfers, for which TD_DATAUNDERRUN
occured, while URB_SHORT_NOT_OK was set */
if (urb && urb->status != -EINPROGRESS
&& ep->nextpid != USB_PID_ACK)
finish_request(isp116x, ep, urb);
}
atomic_dec(&isp116x->atl_finishing); atomic_dec(&isp116x->atl_finishing);
} }
...@@ -821,15 +809,12 @@ static int isp116x_urb_enqueue(struct usb_hcd *hcd, ...@@ -821,15 +809,12 @@ static int isp116x_urb_enqueue(struct usb_hcd *hcd,
} }
/* in case of unlink-during-submit */ /* in case of unlink-during-submit */
spin_lock(&urb->lock);
if (urb->status != -EINPROGRESS) { if (urb->status != -EINPROGRESS) {
spin_unlock(&urb->lock);
finish_request(isp116x, ep, urb); finish_request(isp116x, ep, urb);
ret = 0; ret = 0;
goto fail; goto fail;
} }
urb->hcpriv = hep; urb->hcpriv = hep;
spin_unlock(&urb->lock);
start_atl_transfers(isp116x); start_atl_transfers(isp116x);
fail: fail:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment