USB EHCI QH handling patch

This commit is contained in:
starous@volny.cz 2013-09-21 15:39:51 +02:00
parent 63653cfdae
commit 306950e137

View file

@ -670,7 +670,7 @@ grub_ehci_pci_iter (grub_pci_device_t dev, grub_pci_id_t pciid,
grub_cpu_to_le32 (GRUB_EHCI_TERMINATE); grub_cpu_to_le32 (GRUB_EHCI_TERMINATE);
e->tdfree_virt = e->td_virt; e->tdfree_virt = e->td_virt;
/* Set Terminate in first QH, which is used in framelist */ /* Set Terminate in first QH, which is used in framelist */
e->qh_virt[0].qh_hptr = grub_cpu_to_le32 (GRUB_EHCI_TERMINATE); e->qh_virt[0].qh_hptr = grub_cpu_to_le32 (GRUB_EHCI_TERMINATE | GRUB_EHCI_HPTR_TYPE_QH);
e->qh_virt[0].td_overlay.next_td = grub_cpu_to_le32 (GRUB_EHCI_TERMINATE); e->qh_virt[0].td_overlay.next_td = grub_cpu_to_le32 (GRUB_EHCI_TERMINATE);
e->qh_virt[0].td_overlay.alt_next_td = e->qh_virt[0].td_overlay.alt_next_td =
grub_cpu_to_le32 (GRUB_EHCI_TERMINATE); grub_cpu_to_le32 (GRUB_EHCI_TERMINATE);
@ -977,6 +977,10 @@ grub_ehci_find_qh (struct grub_ehci *e, grub_usb_transfer_t transfer)
int i; int i;
grub_ehci_qh_t qh = e->qh_virt; grub_ehci_qh_t qh = e->qh_virt;
grub_ehci_qh_t head; grub_ehci_qh_t head;
grub_uint32_t qh_phys;
grub_uint32_t qh_terminate =
GRUB_EHCI_TERMINATE | GRUB_EHCI_HPTR_TYPE_QH;
grub_ehci_qh_t qh_iter;
/* Prepare part of EP Characteristic to find existing QH */ /* Prepare part of EP Characteristic to find existing QH */
target = ((transfer->endpoint << GRUB_EHCI_EP_NUM_OFF) | target = ((transfer->endpoint << GRUB_EHCI_EP_NUM_OFF) |
@ -984,21 +988,58 @@ grub_ehci_find_qh (struct grub_ehci *e, grub_usb_transfer_t transfer)
target = grub_cpu_to_le32 (target); target = grub_cpu_to_le32 (target);
mask = grub_cpu_to_le32 (GRUB_EHCI_TARGET_MASK); mask = grub_cpu_to_le32 (GRUB_EHCI_TARGET_MASK);
/* First try to find existing QH with proper target */ /* low speed interrupt transfers are linked to the periodic */
/* schedule, everything else to the asynchronous schedule */
if (transfer->dev->speed == GRUB_USB_SPEED_LOW
&& transfer->type != GRUB_USB_TRANSACTION_TYPE_CONTROL)
head = &qh[0];
else
head = &qh[1];
/* First try to find existing QH with proper target in proper list */
qh_phys = grub_le_to_cpu32( head->qh_hptr );
if (qh_phys != qh_terminate)
qh_iter = grub_dma_phys2virt ( qh_phys & GRUB_EHCI_QHTDPTR_MASK,
e->qh_chunk );
else
qh_iter = NULL;
for (
i = 0;
(qh_phys != qh_terminate) && (qh_iter != NULL) &&
(qh_iter != head) && (i < GRUB_EHCI_N_QH);
i++ )
{
if (target == (qh_iter->ep_char & mask))
{
/* Found proper existing (and linked) QH, do setup of QH */
grub_dprintf ("ehci", "find_qh: found, QH=%p\n", qh_iter);
grub_ehci_setup_qh (qh_iter, transfer);
return qh_iter;
}
qh_phys = grub_le_to_cpu32( qh_iter->qh_hptr );
if (qh_phys != qh_terminate)
qh_iter = grub_dma_phys2virt ( qh_phys & GRUB_EHCI_QHTDPTR_MASK,
e->qh_chunk );
else
qh_iter = NULL;
}
/* variable "i" should be never equal to GRUB_EHCI_N_QH here */
if (i >= GRUB_EHCI_N_QH)
{ /* Something very bad happened in QH list(s) ! */
grub_dprintf ("ehci", "find_qh: Mismatch in QH list! head=%p\n",
head);
}
/* QH with target_addr does not exist, we have to find and add it */
for (i = 2; i < GRUB_EHCI_N_QH; i++) /* We ignore zero and first QH */ for (i = 2; i < GRUB_EHCI_N_QH; i++) /* We ignore zero and first QH */
{ {
if (!qh[i].ep_char) if (!qh[i].ep_char)
break; /* Found first not-allocated QH, finish */ break; /* Found first not-allocated QH, finish */
if (target == (qh[i].ep_char & mask))
{
/* Found proper existing (and linked) QH, do setup of QH */
grub_dprintf ("ehci", "find_qh: found, i=%d, QH=%p\n",
i, &qh[i]);
grub_ehci_setup_qh (&qh[i], transfer);
return &qh[i];
} }
}
/* QH with target_addr does not exist, we have to add it */
/* Have we any free QH in array ? */ /* Have we any free QH in array ? */
if (i >= GRUB_EHCI_N_QH) /* No. */ if (i >= GRUB_EHCI_N_QH) /* No. */
{ {
@ -1013,14 +1054,6 @@ grub_ehci_find_qh (struct grub_ehci *e, grub_usb_transfer_t transfer)
/* We should preset new QH and link it into AL */ /* We should preset new QH and link it into AL */
grub_ehci_setup_qh (&qh[i], transfer); grub_ehci_setup_qh (&qh[i], transfer);
/* low speed interrupt transfers are linked to the periodic
* scheudle, everything else to the asynchronous schedule */
if (transfer->dev->speed == GRUB_USB_SPEED_LOW
&& transfer->type != GRUB_USB_TRANSACTION_TYPE_CONTROL)
head = &qh[0];
else
head = &qh[1];
/* Linking - this new (last) QH will copy the QH from the head QH */ /* Linking - this new (last) QH will copy the QH from the head QH */
qh[i].qh_hptr = head->qh_hptr; qh[i].qh_hptr = head->qh_hptr;
/* Linking - the head QH will point to this new QH */ /* Linking - the head QH will point to this new QH */
@ -1538,17 +1571,20 @@ grub_ehci_cancel_transfer (grub_usb_controller_t dev,
int i; int i;
grub_uint64_t maxtime; grub_uint64_t maxtime;
grub_uint32_t qh_phys; grub_uint32_t qh_phys;
grub_uint32_t interrupt =
cdata->qh_virt->ep_cap & GRUB_EHCI_SMASK_MASK;
/* QH can be active and should be de-activated and halted */ /* QH can be active and should be de-activated and halted */
grub_dprintf ("ehci", "cancel_transfer: begin\n"); grub_dprintf ("ehci", "cancel_transfer: begin\n");
/* First check if EHCI is running and AL is enabled and if not, /* First check if EHCI is running - if not, there is no problem */
* there is no problem... */ /* to cancel any transfer. Or, if transfer is asynchronous, check */
if (((grub_ehci_oper_read32 (e, GRUB_EHCI_STATUS) /* if AL is enabled - if not, transfer can be canceled also. */
& GRUB_EHCI_ST_HC_HALTED) != 0) || if (((grub_ehci_oper_read32 (e, GRUB_EHCI_STATUS) &
((grub_ehci_oper_read32 (e, GRUB_EHCI_STATUS) GRUB_EHCI_ST_HC_HALTED) != 0) ||
& (GRUB_EHCI_ST_AS_STATUS | GRUB_EHCI_ST_PS_STATUS)) == 0)) (!interrupt && ((grub_ehci_oper_read32 (e, GRUB_EHCI_STATUS) &
(GRUB_EHCI_ST_AS_STATUS | GRUB_EHCI_ST_PS_STATUS)) == 0)))
{ {
grub_ehci_pre_finish_transfer (transfer); grub_ehci_pre_finish_transfer (transfer);
grub_ehci_free_tds (e, cdata->td_first_virt, transfer, &actual); grub_ehci_free_tds (e, cdata->td_first_virt, transfer, &actual);
@ -1558,13 +1594,14 @@ grub_ehci_cancel_transfer (grub_usb_controller_t dev,
return GRUB_USB_ERR_NONE; return GRUB_USB_ERR_NONE;
} }
/* EHCI and AL are running. What to do? /* EHCI and (AL or SL) are running. What to do? */
* Try to Halt QH via de-scheduling QH. */ /* Try to Halt QH via de-scheduling QH. */
/* Find index of previous QH */ /* Find index of previous QH */
qh_phys = grub_dma_virt2phys(cdata->qh_virt, e->qh_chunk); qh_phys = grub_dma_virt2phys(cdata->qh_virt, e->qh_chunk);
for (i = 0; i < GRUB_EHCI_N_QH; i++) for (i = 0; i < GRUB_EHCI_N_QH; i++)
{ {
if ((e->qh_virt[i].qh_hptr & GRUB_EHCI_QHTDPTR_MASK) == qh_phys) if ((grub_le_to_cpu32(e->qh_virt[i].qh_hptr)
& GRUB_EHCI_QHTDPTR_MASK) == qh_phys)
break; break;
} }
if (i == GRUB_EHCI_N_QH) if (i == GRUB_EHCI_N_QH)
@ -1618,24 +1655,12 @@ grub_ehci_cancel_transfer (grub_usb_controller_t dev,
grub_ehci_free_tds (e, cdata->td_first_virt, transfer, &actual); grub_ehci_free_tds (e, cdata->td_first_virt, transfer, &actual);
grub_ehci_free_td (e, cdata->td_alt_virt); grub_ehci_free_td (e, cdata->td_alt_virt);
/* FIXME Putting the QH back on the list should work, but for some /* "Free" the QH - link it to itself */
* strange reason doing that will affect other QHs on the periodic
* list. So free the QH instead of putting it back on the list
* which does seem to work, but I would like to know why. */
#if 0
/* Finaly we should return QH back to the AL... */
e->qh_virt[i].qh_hptr =
grub_cpu_to_le32 (grub_dma_virt2phys
(cdata->qh_virt, e->qh_chunk));
#else
/* Free the QH */
cdata->qh_virt->ep_char = 0; cdata->qh_virt->ep_char = 0;
cdata->qh_virt->qh_hptr = cdata->qh_virt->qh_hptr =
grub_cpu_to_le32 ((grub_dma_virt2phys (cdata->qh_virt, grub_cpu_to_le32 ((grub_dma_virt2phys (cdata->qh_virt,
e->qh_chunk) e->qh_chunk)
& GRUB_EHCI_POINTER_MASK) | GRUB_EHCI_HPTR_TYPE_QH); & GRUB_EHCI_POINTER_MASK) | GRUB_EHCI_HPTR_TYPE_QH);
#endif
grub_free (cdata); grub_free (cdata);