* Apply more sane canceling logic to EHCI too

* Remove the leftover user_area - this is handeled inside the Transfer class by now

git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@21906 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Michael Lotz 2007-08-12 12:54:16 +00:00
parent d2e77702cb
commit c954d33405
2 changed files with 71 additions and 60 deletions

View File

@ -831,7 +831,7 @@ status_t
EHCI::AddPendingTransfer(Transfer *transfer, ehci_qh *queueHead, EHCI::AddPendingTransfer(Transfer *transfer, ehci_qh *queueHead,
ehci_qtd *dataDescriptor, bool directionIn) ehci_qtd *dataDescriptor, bool directionIn)
{ {
transfer_data *data = new(std::nothrow) transfer_data(); transfer_data *data = new(std::nothrow) transfer_data;
if (!data) if (!data)
return B_NO_MEMORY; return B_NO_MEMORY;
@ -842,8 +842,8 @@ EHCI::AddPendingTransfer(Transfer *transfer, ehci_qh *queueHead,
data->transfer = transfer; data->transfer = transfer;
data->queue_head = queueHead; data->queue_head = queueHead;
data->data_descriptor = dataDescriptor; data->data_descriptor = dataDescriptor;
data->user_area = -1;
data->incoming = directionIn; data->incoming = directionIn;
data->canceled = false;
data->link = NULL; data->link = NULL;
if (!Lock()) { if (!Lock()) {
@ -869,29 +869,21 @@ EHCI::CancelQueuedTransfers(Pipe *pipe)
if (!Lock()) if (!Lock())
return B_ERROR; return B_ERROR;
transfer_data *last = NULL;
transfer_data *current = fFirstTransfer; transfer_data *current = fFirstTransfer;
while (current) { while (current) {
if (current->transfer->TransferPipe() == pipe) { if (current->transfer->TransferPipe() == pipe) {
UnlinkQueueHead(current->queue_head, &fFreeListHead); // clear the active bit so the descriptors are canceled
current->transfer->Finished(B_CANCELED, 0); ehci_qtd *descriptor = (ehci_qtd *)current->queue_head->element_log;
delete current->transfer; while (descriptor) {
descriptor->token &= ~EHCI_QTD_STATUS_ACTIVE;
transfer_data *next = current->link; descriptor = (ehci_qtd *)descriptor->next_log;
if (last)
last->link = next;
else
fFirstTransfer = next;
if (fLastTransfer == current)
fLastTransfer = last;
delete current;
current = next;
} else {
last = current;
current = current->link;
} }
current->transfer->Finished(B_CANCELED, 0);
current->canceled = true;
}
current = current->link;
} }
Unlock(); Unlock();
@ -954,10 +946,7 @@ EHCI::FinishTransfers()
while (transfer) { while (transfer) {
bool transferDone = false; bool transferDone = false;
ehci_qtd *descriptor = (ehci_qtd *)transfer->queue_head->element_log; ehci_qtd *descriptor = (ehci_qtd *)transfer->queue_head->element_log;
status_t callbackStatus = B_OK;
#ifdef TRACE_USB
print_queue(transfer->queue_head);
#endif
while (descriptor) { while (descriptor) {
uint32 status = descriptor->token; uint32 status = descriptor->token;
@ -998,8 +987,6 @@ EHCI::FinishTransfers()
callbackStatus = B_DEV_STALLED; callbackStatus = B_DEV_STALLED;
} }
UnlinkQueueHead(transfer->queue_head, &fFreeListHead);
transfer->transfer->Finished(callbackStatus, 0);
transferDone = true; transferDone = true;
break; break;
} }
@ -1007,8 +994,41 @@ EHCI::FinishTransfers()
if (descriptor->next_phy & EHCI_QTD_TERMINATE) { if (descriptor->next_phy & EHCI_QTD_TERMINATE) {
// we arrived at the last (stray) descriptor, we're done // we arrived at the last (stray) descriptor, we're done
TRACE(("usb_ehci: qtd (0x%08lx) done\n", descriptor->this_phy)); TRACE(("usb_ehci: qtd (0x%08lx) done\n", descriptor->this_phy));
callbackStatus = B_OK;
transferDone = true;
break;
}
descriptor = (ehci_qtd *)descriptor->next_log;
}
if (!transferDone) {
lastTransfer = transfer;
transfer = transfer->link;
continue;
}
// remove the transfer from the list first so we are sure
// it doesn't get canceled while we still process it
transfer_data *next = transfer->link;
if (Lock()) {
if (lastTransfer)
lastTransfer->link = transfer->link;
if (transfer == fFirstTransfer)
fFirstTransfer = transfer->link;
if (transfer == fLastTransfer)
fLastTransfer = lastTransfer;
transfer->link = NULL;
Unlock();
}
// if canceled the callback has already been called
if (!transfer->canceled) {
size_t actualLength = 0; size_t actualLength = 0;
if (callbackStatus == B_OK) {
bool nextDataToggle = false; bool nextDataToggle = false;
if (transfer->data_descriptor && transfer->incoming) { if (transfer->data_descriptor && transfer->incoming) {
// data to read out // data to read out
@ -1027,56 +1047,47 @@ EHCI::FinishTransfers()
} }
transfer->transfer->TransferPipe()->SetDataToggle(nextDataToggle); transfer->transfer->TransferPipe()->SetDataToggle(nextDataToggle);
if (transfer->transfer->IsFragmented()) { if (transfer->transfer->IsFragmented()) {
// this transfer may still have data left // this transfer may still have data left
transfer->transfer->AdvanceByFragment(actualLength); transfer->transfer->AdvanceByFragment(actualLength);
if (transfer->transfer->VectorLength() > 0) { if (transfer->transfer->VectorLength() > 0) {
FreeDescriptorChain(transfer->data_descriptor); FreeDescriptorChain(transfer->data_descriptor);
transfer->transfer->PrepareKernelAccess(); transfer->transfer->PrepareKernelAccess();
FillQueueWithData(transfer->transfer, status_t result = FillQueueWithData(
transfer->transfer,
transfer->queue_head, transfer->queue_head,
&transfer->data_descriptor, NULL); &transfer->data_descriptor, NULL);
break;
if (result == B_OK && Lock()) {
// reappend the transfer
if (fLastTransfer)
fLastTransfer->link = transfer;
if (!fFirstTransfer)
fFirstTransfer = transfer;
fLastTransfer = transfer;
Unlock();
transfer = next;
continue;
}
} }
// the transfer is done, but we already set the // the transfer is done, but we already set the
// actualLength with AdvanceByFragment() // actualLength with AdvanceByFragment()
actualLength = 0; actualLength = 0;
} }
}
transfer->transfer->Finished(callbackStatus, actualLength);
}
// unlink hardware queue and delete the transfer
UnlinkQueueHead(transfer->queue_head, &fFreeListHead); UnlinkQueueHead(transfer->queue_head, &fFreeListHead);
transfer->transfer->Finished(B_OK, actualLength);
transferDone = true;
break;
}
descriptor = (ehci_qtd *)descriptor->next_log;
}
if (transferDone) {
if (Lock()) {
if (lastTransfer)
lastTransfer->link = transfer->link;
if (transfer == fFirstTransfer)
fFirstTransfer = transfer->link;
if (transfer == fLastTransfer)
fLastTransfer = lastTransfer;
transfer_data *next = transfer->link;
delete transfer->transfer; delete transfer->transfer;
delete transfer; delete transfer;
transfer = next; transfer = next;
Unlock();
}
} else {
if (Lock()) {
lastTransfer = transfer;
transfer = transfer->link;
Unlock();
}
}
release_sem(fCleanupSem); release_sem(fCleanupSem);
} }
} }

View File

@ -22,8 +22,8 @@ typedef struct transfer_data_s {
Transfer *transfer; Transfer *transfer;
ehci_qh *queue_head; ehci_qh *queue_head;
ehci_qtd *data_descriptor; ehci_qtd *data_descriptor;
area_id user_area;
bool incoming; bool incoming;
bool canceled;
transfer_data_s *link; transfer_data_s *link;
} transfer_data; } transfer_data;