提交 b1869000 编写于 作者: A Alan Stern 提交者: Greg Kroah-Hartman

[PATCH] UHCI: Common result routine for Control/Bulk/Interrupt

This patch (as679) combines the result routine for Control URBs with the
routine for Bulk/Interrupt URBs.  Along the way I eliminated the
debugging printouts for Control transfers unless the debugging level is
set higher than 1.  I also eliminated a long-unused (#ifdef'ed-out)
section that works around some buggy old APC BackUPS devices.
Signed-off-by: NAlan Stern <stern@rowland.harvard.edu>
Signed-off-by: NGreg Kroah-Hartman <gregkh@suse.de>
上级 0f28b55d
...@@ -618,134 +618,6 @@ static int uhci_submit_control(struct uhci_hcd *uhci, struct urb *urb, ...@@ -618,134 +618,6 @@ static int uhci_submit_control(struct uhci_hcd *uhci, struct urb *urb,
return -ENOMEM; return -ENOMEM;
} }
/*
* If control-IN transfer was short, the status packet wasn't sent.
* This routine changes the element pointer in the QH to point at the
* status TD. It's safe to do this even while the QH is live, because
* the hardware only updates the element pointer following a successful
* transfer. The inactive TD for the short packet won't cause an update,
* so the pointer won't get overwritten. The next time the controller
* sees this QH, it will send the status packet.
*/
static int usb_control_retrigger_status(struct uhci_hcd *uhci, struct urb *urb)
{
struct urb_priv *urbp = (struct urb_priv *)urb->hcpriv;
struct uhci_td *td;
urbp->short_transfer = 1;
td = list_entry(urbp->td_list.prev, struct uhci_td, list);
urbp->qh->element = cpu_to_le32(td->dma_handle);
return -EINPROGRESS;
}
static int uhci_result_control(struct uhci_hcd *uhci, struct urb *urb)
{
struct list_head *tmp, *head;
struct urb_priv *urbp = urb->hcpriv;
struct uhci_td *td;
unsigned int status;
int ret = 0;
head = &urbp->td_list;
if (urbp->short_transfer) {
tmp = head->prev;
goto status_stage;
}
urb->actual_length = 0;
tmp = head->next;
td = list_entry(tmp, struct uhci_td, list);
/* The first TD is the SETUP stage, check the status, but skip */
/* the count */
status = uhci_status_bits(td_status(td));
if (status & TD_CTRL_ACTIVE)
return -EINPROGRESS;
if (status)
goto td_error;
/* The rest of the TDs (but the last) are data */
tmp = tmp->next;
while (tmp != head && tmp->next != head) {
unsigned int ctrlstat;
td = list_entry(tmp, struct uhci_td, list);
tmp = tmp->next;
ctrlstat = td_status(td);
status = uhci_status_bits(ctrlstat);
if (status & TD_CTRL_ACTIVE)
return -EINPROGRESS;
urb->actual_length += uhci_actual_length(ctrlstat);
if (status)
goto td_error;
/* Check to see if we received a short packet */
if (uhci_actual_length(ctrlstat) <
uhci_expected_length(td_token(td))) {
if (urb->transfer_flags & URB_SHORT_NOT_OK) {
ret = -EREMOTEIO;
goto err;
}
return usb_control_retrigger_status(uhci, urb);
}
}
status_stage:
td = list_entry(tmp, struct uhci_td, list);
/* Control status stage */
status = td_status(td);
#ifdef I_HAVE_BUGGY_APC_BACKUPS
/* APC BackUPS Pro kludge */
/* It tries to send all of the descriptor instead of the amount */
/* we requested */
if (status & TD_CTRL_IOC && /* IOC is masked out by uhci_status_bits */
status & TD_CTRL_ACTIVE &&
status & TD_CTRL_NAK)
return 0;
#endif
status = uhci_status_bits(status);
if (status & TD_CTRL_ACTIVE)
return -EINPROGRESS;
if (status)
goto td_error;
return 0;
td_error:
ret = uhci_map_status(status, uhci_packetout(td_token(td)));
err:
if ((debug == 1 && ret != -EPIPE) || debug > 1) {
/* Some debugging code */
dev_dbg(uhci_dev(uhci), "%s: failed with status %x\n",
__FUNCTION__, status);
if (errbuf) {
/* Print the chain for debugging purposes */
uhci_show_qh(urbp->qh, errbuf, ERRBUF_LEN, 0);
lprintk(errbuf);
}
}
/* Note that the queue has stopped */
urbp->qh->element = UHCI_PTR_TERM;
urbp->qh->is_stopped = 1;
return ret;
}
/* /*
* Common submit for bulk and interrupt * Common submit for bulk and interrupt
*/ */
...@@ -864,113 +736,156 @@ static int uhci_submit_common(struct uhci_hcd *uhci, struct urb *urb, ...@@ -864,113 +736,156 @@ static int uhci_submit_common(struct uhci_hcd *uhci, struct urb *urb,
return -ENOMEM; return -ENOMEM;
} }
static inline int uhci_submit_bulk(struct uhci_hcd *uhci, struct urb *urb,
struct uhci_qh *qh)
{
int ret;
/* Can't have low-speed bulk transfers */
if (urb->dev->speed == USB_SPEED_LOW)
return -EINVAL;
qh->skel = uhci->skel_bulk_qh;
ret = uhci_submit_common(uhci, urb, qh);
if (ret == 0)
uhci_inc_fsbr(uhci, urb);
return ret;
}
static inline int uhci_submit_interrupt(struct uhci_hcd *uhci, struct urb *urb,
struct uhci_qh *qh)
{
/* USB 1.1 interrupt transfers only involve one packet per interval.
* Drivers can submit URBs of any length, but longer ones will need
* multiple intervals to complete.
*/
qh->skel = uhci->skelqh[__interval_to_skel(urb->interval)];
return uhci_submit_common(uhci, urb, qh);
}
/* /*
* Common result for bulk and interrupt * Fix up the data structures following a short transfer
*/ */
static int uhci_result_common(struct uhci_hcd *uhci, struct urb *urb) static int uhci_fixup_short_transfer(struct uhci_hcd *uhci,
struct uhci_qh *qh, struct urb_priv *urbp,
struct uhci_td *short_td)
{ {
struct urb_priv *urbp = urb->hcpriv;
struct uhci_td *td; struct uhci_td *td;
unsigned int status = 0;
int ret = 0; int ret = 0;
urb->actual_length = 0; td = list_entry(urbp->td_list.prev, struct uhci_td, list);
if (qh->type == USB_ENDPOINT_XFER_CONTROL) {
list_for_each_entry(td, &urbp->td_list, list) { urbp->short_transfer = 1;
unsigned int ctrlstat = td_status(td);
status = uhci_status_bits(ctrlstat); /* When a control transfer is short, we have to restart
if (status & TD_CTRL_ACTIVE) * the queue at the status stage transaction, which is
return -EINPROGRESS; * the last TD. */
qh->element = cpu_to_le32(td->dma_handle);
ret = -EINPROGRESS;
urb->actual_length += uhci_actual_length(ctrlstat); } else if (!urbp->short_transfer) {
urbp->short_transfer = 1;
if (status) /* When a bulk/interrupt transfer is short, we have to
goto td_error; * fix up the toggles of the following URBs on the queue
* before restarting the queue at the next URB. */
qh->initial_toggle = uhci_toggle(td_token(short_td)) ^ 1;
uhci_fixup_toggles(qh, 1);
if (uhci_actual_length(ctrlstat) < qh->element = td->link;
uhci_expected_length(td_token(td))) {
if (urb->transfer_flags & URB_SHORT_NOT_OK) {
ret = -EREMOTEIO;
goto err;
} }
/* return ret;
* This URB stopped short of its end. We have to }
* fix up the toggles of the following URBs on the
* queue and restart the queue. /*
* * Common result for control, bulk, and interrupt
* Do this only the first time we encounter the
* short URB.
*/ */
if (!urbp->short_transfer) { static int uhci_result_common(struct uhci_hcd *uhci, struct urb *urb)
urbp->short_transfer = 1; {
urbp->qh->initial_toggle = struct urb_priv *urbp = urb->hcpriv;
uhci_toggle(td_token(td)) ^ 1; struct uhci_qh *qh = urbp->qh;
uhci_fixup_toggles(urbp->qh, 1); struct uhci_td *td;
struct list_head *tmp;
unsigned status;
int ret = 0;
td = list_entry(urbp->td_list.prev, tmp = urbp->td_list.next;
struct uhci_td, list);
urbp->qh->element = td->link;
}
break;
}
}
return 0; if (qh->type == USB_ENDPOINT_XFER_CONTROL) {
if (urbp->short_transfer)
tmp = urbp->td_list.prev;
else
urb->actual_length = -8; /* SETUP packet */
} else
urb->actual_length = 0;
while (tmp != &urbp->td_list) {
unsigned int ctrlstat;
int len;
td = list_entry(tmp, struct uhci_td, list);
tmp = tmp->next;
ctrlstat = td_status(td);
status = uhci_status_bits(ctrlstat);
if (status & TD_CTRL_ACTIVE)
return -EINPROGRESS;
td_error: len = uhci_actual_length(ctrlstat);
ret = uhci_map_status(status, uhci_packetout(td_token(td))); urb->actual_length += len;
if (status) {
ret = uhci_map_status(status,
uhci_packetout(td_token(td)));
if ((debug == 1 && ret != -EPIPE) || debug > 1) { if ((debug == 1 && ret != -EPIPE) || debug > 1) {
/* Some debugging code */ /* Some debugging code */
dev_dbg(uhci_dev(uhci), "%s: failed with status %x\n", dev_dbg(uhci_dev(uhci),
"%s: failed with status %x\n",
__FUNCTION__, status); __FUNCTION__, status);
if (debug > 1 && errbuf) { if (debug > 1 && errbuf) {
/* Print the chain for debugging purposes */ /* Print the chain for debugging */
uhci_show_qh(urbp->qh, errbuf, ERRBUF_LEN, 0); uhci_show_qh(urbp->qh, errbuf,
ERRBUF_LEN, 0);
lprintk(errbuf); lprintk(errbuf);
} }
} }
err:
/* Note that the queue has stopped and save the next toggle value */ } else if (len < uhci_expected_length(td_token(td))) {
urbp->qh->element = UHCI_PTR_TERM;
urbp->qh->is_stopped = 1; /* We received a short packet */
urbp->qh->needs_fixup = 1; if (urb->transfer_flags & URB_SHORT_NOT_OK)
urbp->qh->initial_toggle = uhci_toggle(td_token(td)) ^ ret = -EREMOTEIO;
(ret == -EREMOTEIO); else if (ctrlstat & TD_CTRL_SPD)
ret = 1;
}
if (ret != 0)
goto err;
}
return ret; return ret;
}
static inline int uhci_submit_bulk(struct uhci_hcd *uhci, struct urb *urb, err:
struct uhci_qh *qh) if (ret < 0) {
{ /* In case a control transfer gets an error
int ret; * during the setup stage */
urb->actual_length = max(urb->actual_length, 0);
/* Can't have low-speed bulk transfers */ /* Note that the queue has stopped and save
if (urb->dev->speed == USB_SPEED_LOW) * the next toggle value */
return -EINVAL; qh->element = UHCI_PTR_TERM;
qh->is_stopped = 1;
qh->needs_fixup = (qh->type != USB_ENDPOINT_XFER_CONTROL);
qh->initial_toggle = uhci_toggle(td_token(td)) ^
(ret == -EREMOTEIO);
qh->skel = uhci->skel_bulk_qh; } else /* Short packet received */
ret = uhci_submit_common(uhci, urb, qh); ret = uhci_fixup_short_transfer(uhci, qh, urbp, td);
if (ret == 0)
uhci_inc_fsbr(uhci, urb);
return ret; return ret;
} }
static inline int uhci_submit_interrupt(struct uhci_hcd *uhci, struct urb *urb,
struct uhci_qh *qh)
{
/* USB 1.1 interrupt transfers only involve one packet per interval.
* Drivers can submit URBs of any length, but longer ones will need
* multiple intervals to complete.
*/
qh->skel = uhci->skelqh[__interval_to_skel(urb->interval)];
return uhci_submit_common(uhci, urb, qh);
}
/* /*
* Isochronous transfers * Isochronous transfers
*/ */
...@@ -1276,17 +1191,10 @@ static void uhci_scan_qh(struct uhci_hcd *uhci, struct uhci_qh *qh, ...@@ -1276,17 +1191,10 @@ static void uhci_scan_qh(struct uhci_hcd *uhci, struct uhci_qh *qh,
urbp = list_entry(qh->queue.next, struct urb_priv, node); urbp = list_entry(qh->queue.next, struct urb_priv, node);
urb = urbp->urb; urb = urbp->urb;
switch (qh->type) { if (qh->type == USB_ENDPOINT_XFER_ISOC)
case USB_ENDPOINT_XFER_CONTROL:
status = uhci_result_control(uhci, urb);
break;
case USB_ENDPOINT_XFER_ISOC:
status = uhci_result_isochronous(uhci, urb); status = uhci_result_isochronous(uhci, urb);
break; else
default: /* USB_ENDPOINT_XFER_BULK or _INT */
status = uhci_result_common(uhci, urb); status = uhci_result_common(uhci, urb);
break;
}
if (status == -EINPROGRESS) if (status == -EINPROGRESS)
break; break;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册