xhci: Use command structures when queuing commands on the command ring
To create a global command queue we require that each command put on the command ring is submitted with a command structure. Functions that queue commands and wait for completion need to allocate a command before submitting it, and free it once completed. The following command queuing functions need to be modified. xhci_configure_endpoint() xhci_address_device() xhci_queue_slot_control() xhci_queue_stop_endpoint() xhci_queue_new_dequeue_state() xhci_queue_reset_ep() xhci_configure_endpoint() xhci_configure_endpoint() could already be called with a command structure, and only xhci_check_maxpacket and xhci_check_bandwidth did not do so. These are changed and a command structure is now required. This change also simplifies the configure endpoint command completion handling and the "goto bandwidth_change" handling code can be removed. In some cases the command queuing function is called in interrupt context. These commands needs to be allocated atomically, and they can't wait for completion. These commands will in this patch be freed directly after queuing, but freeing will be moved to the command completion event handler in a later patch once we get the global command queue up.(Just so that we won't leak memory in the middle of the patch set) Signed-off-by: Mathias Nyman <mathias.nyman@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
be9820383b
commit
ddba5cd0ae
|
@ -20,7 +20,8 @@
|
||||||
* Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
* Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/gfp.h>
|
|
||||||
|
#include <linux/slab.h>
|
||||||
#include <asm/unaligned.h>
|
#include <asm/unaligned.h>
|
||||||
|
|
||||||
#include "xhci.h"
|
#include "xhci.h"
|
||||||
|
@ -284,12 +285,22 @@ static int xhci_stop_device(struct xhci_hcd *xhci, int slot_id, int suspend)
|
||||||
|
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
for (i = LAST_EP_INDEX; i > 0; i--) {
|
for (i = LAST_EP_INDEX; i > 0; i--) {
|
||||||
if (virt_dev->eps[i].ring && virt_dev->eps[i].ring->dequeue)
|
if (virt_dev->eps[i].ring && virt_dev->eps[i].ring->dequeue) {
|
||||||
xhci_queue_stop_endpoint(xhci, slot_id, i, suspend);
|
struct xhci_command *command;
|
||||||
|
command = xhci_alloc_command(xhci, false, false,
|
||||||
|
GFP_NOIO);
|
||||||
|
if (!command) {
|
||||||
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
xhci_free_command(xhci, cmd);
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
}
|
||||||
|
xhci_queue_stop_endpoint(xhci, command, slot_id, i,
|
||||||
|
suspend);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
cmd->command_trb = xhci_find_next_enqueue(xhci->cmd_ring);
|
|
||||||
list_add_tail(&cmd->cmd_list, &virt_dev->cmd_list);
|
list_add_tail(&cmd->cmd_list, &virt_dev->cmd_list);
|
||||||
xhci_queue_stop_endpoint(xhci, slot_id, 0, suspend);
|
xhci_queue_stop_endpoint(xhci, cmd, slot_id, 0, suspend);
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
|
||||||
|
|
|
@ -123,16 +123,6 @@ static int enqueue_is_link_trb(struct xhci_ring *ring)
|
||||||
return TRB_TYPE_LINK_LE32(link->control);
|
return TRB_TYPE_LINK_LE32(link->control);
|
||||||
}
|
}
|
||||||
|
|
||||||
union xhci_trb *xhci_find_next_enqueue(struct xhci_ring *ring)
|
|
||||||
{
|
|
||||||
/* Enqueue pointer can be left pointing to the link TRB,
|
|
||||||
* we must handle that
|
|
||||||
*/
|
|
||||||
if (TRB_TYPE_LINK_LE32(ring->enqueue->link.control))
|
|
||||||
return ring->enq_seg->next->trbs;
|
|
||||||
return ring->enqueue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Updates trb to point to the next TRB in the ring, and updates seg if the next
|
/* Updates trb to point to the next TRB in the ring, and updates seg if the next
|
||||||
* TRB is in a new segment. This does not skip over link TRBs, and it does not
|
* TRB is in a new segment. This does not skip over link TRBs, and it does not
|
||||||
* effect the ring dequeue or enqueue pointers.
|
* effect the ring dequeue or enqueue pointers.
|
||||||
|
@ -684,12 +674,14 @@ static void td_to_noop(struct xhci_hcd *xhci, struct xhci_ring *ep_ring,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int queue_set_tr_deq(struct xhci_hcd *xhci, int slot_id,
|
static int queue_set_tr_deq(struct xhci_hcd *xhci,
|
||||||
|
struct xhci_command *cmd, int slot_id,
|
||||||
unsigned int ep_index, unsigned int stream_id,
|
unsigned int ep_index, unsigned int stream_id,
|
||||||
struct xhci_segment *deq_seg,
|
struct xhci_segment *deq_seg,
|
||||||
union xhci_trb *deq_ptr, u32 cycle_state);
|
union xhci_trb *deq_ptr, u32 cycle_state);
|
||||||
|
|
||||||
void xhci_queue_new_dequeue_state(struct xhci_hcd *xhci,
|
void xhci_queue_new_dequeue_state(struct xhci_hcd *xhci,
|
||||||
|
struct xhci_command *cmd,
|
||||||
unsigned int slot_id, unsigned int ep_index,
|
unsigned int slot_id, unsigned int ep_index,
|
||||||
unsigned int stream_id,
|
unsigned int stream_id,
|
||||||
struct xhci_dequeue_state *deq_state)
|
struct xhci_dequeue_state *deq_state)
|
||||||
|
@ -704,7 +696,7 @@ void xhci_queue_new_dequeue_state(struct xhci_hcd *xhci,
|
||||||
deq_state->new_deq_ptr,
|
deq_state->new_deq_ptr,
|
||||||
(unsigned long long)xhci_trb_virt_to_dma(deq_state->new_deq_seg, deq_state->new_deq_ptr),
|
(unsigned long long)xhci_trb_virt_to_dma(deq_state->new_deq_seg, deq_state->new_deq_ptr),
|
||||||
deq_state->new_cycle_state);
|
deq_state->new_cycle_state);
|
||||||
queue_set_tr_deq(xhci, slot_id, ep_index, stream_id,
|
queue_set_tr_deq(xhci, cmd, slot_id, ep_index, stream_id,
|
||||||
deq_state->new_deq_seg,
|
deq_state->new_deq_seg,
|
||||||
deq_state->new_deq_ptr,
|
deq_state->new_deq_ptr,
|
||||||
(u32) deq_state->new_cycle_state);
|
(u32) deq_state->new_cycle_state);
|
||||||
|
@ -858,7 +850,9 @@ remove_finished_td:
|
||||||
|
|
||||||
/* If necessary, queue a Set Transfer Ring Dequeue Pointer command */
|
/* If necessary, queue a Set Transfer Ring Dequeue Pointer command */
|
||||||
if (deq_state.new_deq_ptr && deq_state.new_deq_seg) {
|
if (deq_state.new_deq_ptr && deq_state.new_deq_seg) {
|
||||||
xhci_queue_new_dequeue_state(xhci,
|
struct xhci_command *command;
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_ATOMIC);
|
||||||
|
xhci_queue_new_dequeue_state(xhci, command,
|
||||||
slot_id, ep_index,
|
slot_id, ep_index,
|
||||||
ep->stopped_td->urb->stream_id,
|
ep->stopped_td->urb->stream_id,
|
||||||
&deq_state);
|
&deq_state);
|
||||||
|
@ -1206,9 +1200,11 @@ static void xhci_handle_cmd_reset_ep(struct xhci_hcd *xhci, int slot_id,
|
||||||
* because the HW can't handle two commands being queued in a row.
|
* because the HW can't handle two commands being queued in a row.
|
||||||
*/
|
*/
|
||||||
if (xhci->quirks & XHCI_RESET_EP_QUIRK) {
|
if (xhci->quirks & XHCI_RESET_EP_QUIRK) {
|
||||||
|
struct xhci_command *command;
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_ATOMIC);
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
||||||
"Queueing configure endpoint command");
|
"Queueing configure endpoint command");
|
||||||
xhci_queue_configure_endpoint(xhci,
|
xhci_queue_configure_endpoint(xhci, command,
|
||||||
xhci->devs[slot_id]->in_ctx->dma, slot_id,
|
xhci->devs[slot_id]->in_ctx->dma, slot_id,
|
||||||
false);
|
false);
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
|
@ -1465,7 +1461,7 @@ static void xhci_handle_cmd_config_ep(struct xhci_hcd *xhci, int slot_id,
|
||||||
add_flags - SLOT_FLAG == drop_flags) {
|
add_flags - SLOT_FLAG == drop_flags) {
|
||||||
ep_state = virt_dev->eps[ep_index].ep_state;
|
ep_state = virt_dev->eps[ep_index].ep_state;
|
||||||
if (!(ep_state & EP_HALTED))
|
if (!(ep_state & EP_HALTED))
|
||||||
goto bandwidth_change;
|
return;
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
||||||
"Completed config ep cmd - "
|
"Completed config ep cmd - "
|
||||||
"last ep index = %d, state = %d",
|
"last ep index = %d, state = %d",
|
||||||
|
@ -1475,11 +1471,6 @@ static void xhci_handle_cmd_config_ep(struct xhci_hcd *xhci, int slot_id,
|
||||||
ring_doorbell_for_active_rings(xhci, slot_id, ep_index);
|
ring_doorbell_for_active_rings(xhci, slot_id, ep_index);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
bandwidth_change:
|
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_context_change,
|
|
||||||
"Completed config ep cmd");
|
|
||||||
virt_dev->cmd_status = cmd_comp_code;
|
|
||||||
complete(&virt_dev->cmd_completion);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1938,11 +1929,16 @@ static void xhci_cleanup_halted_endpoint(struct xhci_hcd *xhci,
|
||||||
struct xhci_td *td, union xhci_trb *event_trb)
|
struct xhci_td *td, union xhci_trb *event_trb)
|
||||||
{
|
{
|
||||||
struct xhci_virt_ep *ep = &xhci->devs[slot_id]->eps[ep_index];
|
struct xhci_virt_ep *ep = &xhci->devs[slot_id]->eps[ep_index];
|
||||||
|
struct xhci_command *command;
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_ATOMIC);
|
||||||
|
if (!command)
|
||||||
|
return;
|
||||||
|
|
||||||
ep->ep_state |= EP_HALTED;
|
ep->ep_state |= EP_HALTED;
|
||||||
ep->stopped_td = td;
|
ep->stopped_td = td;
|
||||||
ep->stopped_stream = stream_id;
|
ep->stopped_stream = stream_id;
|
||||||
|
|
||||||
xhci_queue_reset_ep(xhci, slot_id, ep_index);
|
xhci_queue_reset_ep(xhci, command, slot_id, ep_index);
|
||||||
xhci_cleanup_stalled_ring(xhci, td->urb->dev, ep_index);
|
xhci_cleanup_stalled_ring(xhci, td->urb->dev, ep_index);
|
||||||
|
|
||||||
ep->stopped_td = NULL;
|
ep->stopped_td = NULL;
|
||||||
|
@ -2654,7 +2650,7 @@ static int handle_tx_event(struct xhci_hcd *xhci,
|
||||||
* successful event after a short transfer.
|
* successful event after a short transfer.
|
||||||
* Ignore it.
|
* Ignore it.
|
||||||
*/
|
*/
|
||||||
if ((xhci->quirks & XHCI_SPURIOUS_SUCCESS) &&
|
if ((xhci->quirks & XHCI_SPURIOUS_SUCCESS) &&
|
||||||
ep_ring->last_td_was_short) {
|
ep_ring->last_td_was_short) {
|
||||||
ep_ring->last_td_was_short = false;
|
ep_ring->last_td_was_short = false;
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
@ -3996,8 +3992,9 @@ int xhci_queue_isoc_tx_prepare(struct xhci_hcd *xhci, gfp_t mem_flags,
|
||||||
* Don't decrement xhci->cmd_ring_reserved_trbs after we've queued the TRB
|
* Don't decrement xhci->cmd_ring_reserved_trbs after we've queued the TRB
|
||||||
* because the command event handler may want to resubmit a failed command.
|
* because the command event handler may want to resubmit a failed command.
|
||||||
*/
|
*/
|
||||||
static int queue_command(struct xhci_hcd *xhci, u32 field1, u32 field2,
|
static int queue_command(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
u32 field3, u32 field4, bool command_must_succeed)
|
u32 field1, u32 field2,
|
||||||
|
u32 field3, u32 field4, bool command_must_succeed)
|
||||||
{
|
{
|
||||||
int reserved_trbs = xhci->cmd_ring_reserved_trbs;
|
int reserved_trbs = xhci->cmd_ring_reserved_trbs;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -4014,57 +4011,65 @@ static int queue_command(struct xhci_hcd *xhci, u32 field1, u32 field2,
|
||||||
"unfailable commands failed.\n");
|
"unfailable commands failed.\n");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
if (cmd->completion)
|
||||||
|
cmd->command_trb = xhci->cmd_ring->enqueue;
|
||||||
|
else
|
||||||
|
kfree(cmd);
|
||||||
|
|
||||||
queue_trb(xhci, xhci->cmd_ring, false, field1, field2, field3,
|
queue_trb(xhci, xhci->cmd_ring, false, field1, field2, field3,
|
||||||
field4 | xhci->cmd_ring->cycle_state);
|
field4 | xhci->cmd_ring->cycle_state);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Queue a slot enable or disable request on the command ring */
|
/* Queue a slot enable or disable request on the command ring */
|
||||||
int xhci_queue_slot_control(struct xhci_hcd *xhci, u32 trb_type, u32 slot_id)
|
int xhci_queue_slot_control(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
|
u32 trb_type, u32 slot_id)
|
||||||
{
|
{
|
||||||
return queue_command(xhci, 0, 0, 0,
|
return queue_command(xhci, cmd, 0, 0, 0,
|
||||||
TRB_TYPE(trb_type) | SLOT_ID_FOR_TRB(slot_id), false);
|
TRB_TYPE(trb_type) | SLOT_ID_FOR_TRB(slot_id), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Queue an address device command TRB */
|
/* Queue an address device command TRB */
|
||||||
int xhci_queue_address_device(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
int xhci_queue_address_device(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
u32 slot_id, enum xhci_setup_dev setup)
|
dma_addr_t in_ctx_ptr, u32 slot_id, enum xhci_setup_dev setup)
|
||||||
{
|
{
|
||||||
return queue_command(xhci, lower_32_bits(in_ctx_ptr),
|
return queue_command(xhci, cmd, lower_32_bits(in_ctx_ptr),
|
||||||
upper_32_bits(in_ctx_ptr), 0,
|
upper_32_bits(in_ctx_ptr), 0,
|
||||||
TRB_TYPE(TRB_ADDR_DEV) | SLOT_ID_FOR_TRB(slot_id)
|
TRB_TYPE(TRB_ADDR_DEV) | SLOT_ID_FOR_TRB(slot_id)
|
||||||
| (setup == SETUP_CONTEXT_ONLY ? TRB_BSR : 0), false);
|
| (setup == SETUP_CONTEXT_ONLY ? TRB_BSR : 0), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
int xhci_queue_vendor_command(struct xhci_hcd *xhci,
|
int xhci_queue_vendor_command(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
u32 field1, u32 field2, u32 field3, u32 field4)
|
u32 field1, u32 field2, u32 field3, u32 field4)
|
||||||
{
|
{
|
||||||
return queue_command(xhci, field1, field2, field3, field4, false);
|
return queue_command(xhci, cmd, field1, field2, field3, field4, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Queue a reset device command TRB */
|
/* Queue a reset device command TRB */
|
||||||
int xhci_queue_reset_device(struct xhci_hcd *xhci, u32 slot_id)
|
int xhci_queue_reset_device(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
|
u32 slot_id)
|
||||||
{
|
{
|
||||||
return queue_command(xhci, 0, 0, 0,
|
return queue_command(xhci, cmd, 0, 0, 0,
|
||||||
TRB_TYPE(TRB_RESET_DEV) | SLOT_ID_FOR_TRB(slot_id),
|
TRB_TYPE(TRB_RESET_DEV) | SLOT_ID_FOR_TRB(slot_id),
|
||||||
false);
|
false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Queue a configure endpoint command TRB */
|
/* Queue a configure endpoint command TRB */
|
||||||
int xhci_queue_configure_endpoint(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
int xhci_queue_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
|
struct xhci_command *cmd, dma_addr_t in_ctx_ptr,
|
||||||
u32 slot_id, bool command_must_succeed)
|
u32 slot_id, bool command_must_succeed)
|
||||||
{
|
{
|
||||||
return queue_command(xhci, lower_32_bits(in_ctx_ptr),
|
return queue_command(xhci, cmd, lower_32_bits(in_ctx_ptr),
|
||||||
upper_32_bits(in_ctx_ptr), 0,
|
upper_32_bits(in_ctx_ptr), 0,
|
||||||
TRB_TYPE(TRB_CONFIG_EP) | SLOT_ID_FOR_TRB(slot_id),
|
TRB_TYPE(TRB_CONFIG_EP) | SLOT_ID_FOR_TRB(slot_id),
|
||||||
command_must_succeed);
|
command_must_succeed);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Queue an evaluate context command TRB */
|
/* Queue an evaluate context command TRB */
|
||||||
int xhci_queue_evaluate_context(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
int xhci_queue_evaluate_context(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
u32 slot_id, bool command_must_succeed)
|
dma_addr_t in_ctx_ptr, u32 slot_id, bool command_must_succeed)
|
||||||
{
|
{
|
||||||
return queue_command(xhci, lower_32_bits(in_ctx_ptr),
|
return queue_command(xhci, cmd, lower_32_bits(in_ctx_ptr),
|
||||||
upper_32_bits(in_ctx_ptr), 0,
|
upper_32_bits(in_ctx_ptr), 0,
|
||||||
TRB_TYPE(TRB_EVAL_CONTEXT) | SLOT_ID_FOR_TRB(slot_id),
|
TRB_TYPE(TRB_EVAL_CONTEXT) | SLOT_ID_FOR_TRB(slot_id),
|
||||||
command_must_succeed);
|
command_must_succeed);
|
||||||
|
@ -4074,25 +4079,26 @@ int xhci_queue_evaluate_context(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
||||||
* Suspend is set to indicate "Stop Endpoint Command" is being issued to stop
|
* Suspend is set to indicate "Stop Endpoint Command" is being issued to stop
|
||||||
* activity on an endpoint that is about to be suspended.
|
* activity on an endpoint that is about to be suspended.
|
||||||
*/
|
*/
|
||||||
int xhci_queue_stop_endpoint(struct xhci_hcd *xhci, int slot_id,
|
int xhci_queue_stop_endpoint(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
unsigned int ep_index, int suspend)
|
int slot_id, unsigned int ep_index, int suspend)
|
||||||
{
|
{
|
||||||
u32 trb_slot_id = SLOT_ID_FOR_TRB(slot_id);
|
u32 trb_slot_id = SLOT_ID_FOR_TRB(slot_id);
|
||||||
u32 trb_ep_index = EP_ID_FOR_TRB(ep_index);
|
u32 trb_ep_index = EP_ID_FOR_TRB(ep_index);
|
||||||
u32 type = TRB_TYPE(TRB_STOP_RING);
|
u32 type = TRB_TYPE(TRB_STOP_RING);
|
||||||
u32 trb_suspend = SUSPEND_PORT_FOR_TRB(suspend);
|
u32 trb_suspend = SUSPEND_PORT_FOR_TRB(suspend);
|
||||||
|
|
||||||
return queue_command(xhci, 0, 0, 0,
|
return queue_command(xhci, cmd, 0, 0, 0,
|
||||||
trb_slot_id | trb_ep_index | type | trb_suspend, false);
|
trb_slot_id | trb_ep_index | type | trb_suspend, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set Transfer Ring Dequeue Pointer command.
|
/* Set Transfer Ring Dequeue Pointer command.
|
||||||
* This should not be used for endpoints that have streams enabled.
|
* This should not be used for endpoints that have streams enabled.
|
||||||
*/
|
*/
|
||||||
static int queue_set_tr_deq(struct xhci_hcd *xhci, int slot_id,
|
static int queue_set_tr_deq(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
unsigned int ep_index, unsigned int stream_id,
|
int slot_id,
|
||||||
struct xhci_segment *deq_seg,
|
unsigned int ep_index, unsigned int stream_id,
|
||||||
union xhci_trb *deq_ptr, u32 cycle_state)
|
struct xhci_segment *deq_seg,
|
||||||
|
union xhci_trb *deq_ptr, u32 cycle_state)
|
||||||
{
|
{
|
||||||
dma_addr_t addr;
|
dma_addr_t addr;
|
||||||
u32 trb_slot_id = SLOT_ID_FOR_TRB(slot_id);
|
u32 trb_slot_id = SLOT_ID_FOR_TRB(slot_id);
|
||||||
|
@ -4119,18 +4125,19 @@ static int queue_set_tr_deq(struct xhci_hcd *xhci, int slot_id,
|
||||||
ep->queued_deq_ptr = deq_ptr;
|
ep->queued_deq_ptr = deq_ptr;
|
||||||
if (stream_id)
|
if (stream_id)
|
||||||
trb_sct = SCT_FOR_TRB(SCT_PRI_TR);
|
trb_sct = SCT_FOR_TRB(SCT_PRI_TR);
|
||||||
return queue_command(xhci, lower_32_bits(addr) | trb_sct | cycle_state,
|
return queue_command(xhci, cmd,
|
||||||
|
lower_32_bits(addr) | trb_sct | cycle_state,
|
||||||
upper_32_bits(addr), trb_stream_id,
|
upper_32_bits(addr), trb_stream_id,
|
||||||
trb_slot_id | trb_ep_index | type, false);
|
trb_slot_id | trb_ep_index | type, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
int xhci_queue_reset_ep(struct xhci_hcd *xhci, int slot_id,
|
int xhci_queue_reset_ep(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
unsigned int ep_index)
|
int slot_id, unsigned int ep_index)
|
||||||
{
|
{
|
||||||
u32 trb_slot_id = SLOT_ID_FOR_TRB(slot_id);
|
u32 trb_slot_id = SLOT_ID_FOR_TRB(slot_id);
|
||||||
u32 trb_ep_index = EP_ID_FOR_TRB(ep_index);
|
u32 trb_ep_index = EP_ID_FOR_TRB(ep_index);
|
||||||
u32 type = TRB_TYPE(TRB_RESET_EP);
|
u32 type = TRB_TYPE(TRB_RESET_EP);
|
||||||
|
|
||||||
return queue_command(xhci, 0, 0, 0, trb_slot_id | trb_ep_index | type,
|
return queue_command(xhci, cmd, 0, 0, 0,
|
||||||
false);
|
trb_slot_id | trb_ep_index | type, false);
|
||||||
}
|
}
|
||||||
|
|
|
@ -641,10 +641,14 @@ int xhci_run(struct usb_hcd *hcd)
|
||||||
writel(ER_IRQ_ENABLE(temp), &xhci->ir_set->irq_pending);
|
writel(ER_IRQ_ENABLE(temp), &xhci->ir_set->irq_pending);
|
||||||
xhci_print_ir_set(xhci, 0);
|
xhci_print_ir_set(xhci, 0);
|
||||||
|
|
||||||
if (xhci->quirks & XHCI_NEC_HOST)
|
if (xhci->quirks & XHCI_NEC_HOST) {
|
||||||
xhci_queue_vendor_command(xhci, 0, 0, 0,
|
struct xhci_command *command;
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_KERNEL);
|
||||||
|
if (!command)
|
||||||
|
return -ENOMEM;
|
||||||
|
xhci_queue_vendor_command(xhci, command, 0, 0, 0,
|
||||||
TRB_TYPE(TRB_NEC_GET_FW));
|
TRB_TYPE(TRB_NEC_GET_FW));
|
||||||
|
}
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_init,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_init,
|
||||||
"Finished xhci_run for USB2 roothub");
|
"Finished xhci_run for USB2 roothub");
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1187,10 +1191,10 @@ static int xhci_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
static int xhci_check_maxpacket(struct xhci_hcd *xhci, unsigned int slot_id,
|
static int xhci_check_maxpacket(struct xhci_hcd *xhci, unsigned int slot_id,
|
||||||
unsigned int ep_index, struct urb *urb)
|
unsigned int ep_index, struct urb *urb)
|
||||||
{
|
{
|
||||||
struct xhci_container_ctx *in_ctx;
|
|
||||||
struct xhci_container_ctx *out_ctx;
|
struct xhci_container_ctx *out_ctx;
|
||||||
struct xhci_input_control_ctx *ctrl_ctx;
|
struct xhci_input_control_ctx *ctrl_ctx;
|
||||||
struct xhci_ep_ctx *ep_ctx;
|
struct xhci_ep_ctx *ep_ctx;
|
||||||
|
struct xhci_command *command;
|
||||||
int max_packet_size;
|
int max_packet_size;
|
||||||
int hw_max_packet_size;
|
int hw_max_packet_size;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -1215,18 +1219,24 @@ static int xhci_check_maxpacket(struct xhci_hcd *xhci, unsigned int slot_id,
|
||||||
/* FIXME: This won't work if a non-default control endpoint
|
/* FIXME: This won't work if a non-default control endpoint
|
||||||
* changes max packet sizes.
|
* changes max packet sizes.
|
||||||
*/
|
*/
|
||||||
in_ctx = xhci->devs[slot_id]->in_ctx;
|
|
||||||
ctrl_ctx = xhci_get_input_control_ctx(xhci, in_ctx);
|
command = xhci_alloc_command(xhci, false, true, GFP_KERNEL);
|
||||||
|
if (!command)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
command->in_ctx = xhci->devs[slot_id]->in_ctx;
|
||||||
|
ctrl_ctx = xhci_get_input_control_ctx(xhci, command->in_ctx);
|
||||||
if (!ctrl_ctx) {
|
if (!ctrl_ctx) {
|
||||||
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
||||||
__func__);
|
__func__);
|
||||||
return -ENOMEM;
|
ret = -ENOMEM;
|
||||||
|
goto command_cleanup;
|
||||||
}
|
}
|
||||||
/* Set up the modified control endpoint 0 */
|
/* Set up the modified control endpoint 0 */
|
||||||
xhci_endpoint_copy(xhci, xhci->devs[slot_id]->in_ctx,
|
xhci_endpoint_copy(xhci, xhci->devs[slot_id]->in_ctx,
|
||||||
xhci->devs[slot_id]->out_ctx, ep_index);
|
xhci->devs[slot_id]->out_ctx, ep_index);
|
||||||
|
|
||||||
ep_ctx = xhci_get_ep_ctx(xhci, in_ctx, ep_index);
|
ep_ctx = xhci_get_ep_ctx(xhci, command->in_ctx, ep_index);
|
||||||
ep_ctx->ep_info2 &= cpu_to_le32(~MAX_PACKET_MASK);
|
ep_ctx->ep_info2 &= cpu_to_le32(~MAX_PACKET_MASK);
|
||||||
ep_ctx->ep_info2 |= cpu_to_le32(MAX_PACKET(max_packet_size));
|
ep_ctx->ep_info2 |= cpu_to_le32(MAX_PACKET(max_packet_size));
|
||||||
|
|
||||||
|
@ -1234,17 +1244,20 @@ static int xhci_check_maxpacket(struct xhci_hcd *xhci, unsigned int slot_id,
|
||||||
ctrl_ctx->drop_flags = 0;
|
ctrl_ctx->drop_flags = 0;
|
||||||
|
|
||||||
xhci_dbg(xhci, "Slot %d input context\n", slot_id);
|
xhci_dbg(xhci, "Slot %d input context\n", slot_id);
|
||||||
xhci_dbg_ctx(xhci, in_ctx, ep_index);
|
xhci_dbg_ctx(xhci, command->in_ctx, ep_index);
|
||||||
xhci_dbg(xhci, "Slot %d output context\n", slot_id);
|
xhci_dbg(xhci, "Slot %d output context\n", slot_id);
|
||||||
xhci_dbg_ctx(xhci, out_ctx, ep_index);
|
xhci_dbg_ctx(xhci, out_ctx, ep_index);
|
||||||
|
|
||||||
ret = xhci_configure_endpoint(xhci, urb->dev, NULL,
|
ret = xhci_configure_endpoint(xhci, urb->dev, command,
|
||||||
true, false);
|
true, false);
|
||||||
|
|
||||||
/* Clean up the input context for later use by bandwidth
|
/* Clean up the input context for later use by bandwidth
|
||||||
* functions.
|
* functions.
|
||||||
*/
|
*/
|
||||||
ctrl_ctx->add_flags = cpu_to_le32(SLOT_FLAG);
|
ctrl_ctx->add_flags = cpu_to_le32(SLOT_FLAG);
|
||||||
|
command_cleanup:
|
||||||
|
kfree(command->completion);
|
||||||
|
kfree(command);
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -1465,6 +1478,7 @@ int xhci_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, int status)
|
||||||
unsigned int ep_index;
|
unsigned int ep_index;
|
||||||
struct xhci_ring *ep_ring;
|
struct xhci_ring *ep_ring;
|
||||||
struct xhci_virt_ep *ep;
|
struct xhci_virt_ep *ep;
|
||||||
|
struct xhci_command *command;
|
||||||
|
|
||||||
xhci = hcd_to_xhci(hcd);
|
xhci = hcd_to_xhci(hcd);
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
|
@ -1534,12 +1548,14 @@ int xhci_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, int status)
|
||||||
* the first cancellation to be handled.
|
* the first cancellation to be handled.
|
||||||
*/
|
*/
|
||||||
if (!(ep->ep_state & EP_HALT_PENDING)) {
|
if (!(ep->ep_state & EP_HALT_PENDING)) {
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_ATOMIC);
|
||||||
ep->ep_state |= EP_HALT_PENDING;
|
ep->ep_state |= EP_HALT_PENDING;
|
||||||
ep->stop_cmds_pending++;
|
ep->stop_cmds_pending++;
|
||||||
ep->stop_cmd_timer.expires = jiffies +
|
ep->stop_cmd_timer.expires = jiffies +
|
||||||
XHCI_STOP_EP_CMD_TIMEOUT * HZ;
|
XHCI_STOP_EP_CMD_TIMEOUT * HZ;
|
||||||
add_timer(&ep->stop_cmd_timer);
|
add_timer(&ep->stop_cmd_timer);
|
||||||
xhci_queue_stop_endpoint(xhci, urb->dev->slot_id, ep_index, 0);
|
xhci_queue_stop_endpoint(xhci, command, urb->dev->slot_id,
|
||||||
|
ep_index, 0);
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
}
|
}
|
||||||
done:
|
done:
|
||||||
|
@ -2576,21 +2592,16 @@ static int xhci_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
int ret;
|
int ret;
|
||||||
int timeleft;
|
int timeleft;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct xhci_container_ctx *in_ctx;
|
|
||||||
struct xhci_input_control_ctx *ctrl_ctx;
|
struct xhci_input_control_ctx *ctrl_ctx;
|
||||||
struct completion *cmd_completion;
|
|
||||||
u32 *cmd_status;
|
|
||||||
struct xhci_virt_device *virt_dev;
|
struct xhci_virt_device *virt_dev;
|
||||||
union xhci_trb *cmd_trb;
|
|
||||||
|
if (!command)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
virt_dev = xhci->devs[udev->slot_id];
|
virt_dev = xhci->devs[udev->slot_id];
|
||||||
|
|
||||||
if (command)
|
ctrl_ctx = xhci_get_input_control_ctx(xhci, command->in_ctx);
|
||||||
in_ctx = command->in_ctx;
|
|
||||||
else
|
|
||||||
in_ctx = virt_dev->in_ctx;
|
|
||||||
ctrl_ctx = xhci_get_input_control_ctx(xhci, in_ctx);
|
|
||||||
if (!ctrl_ctx) {
|
if (!ctrl_ctx) {
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
||||||
|
@ -2607,7 +2618,7 @@ static int xhci_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
if ((xhci->quirks & XHCI_SW_BW_CHECKING) &&
|
if ((xhci->quirks & XHCI_SW_BW_CHECKING) &&
|
||||||
xhci_reserve_bandwidth(xhci, virt_dev, in_ctx)) {
|
xhci_reserve_bandwidth(xhci, virt_dev, command->in_ctx)) {
|
||||||
if ((xhci->quirks & XHCI_EP_LIMIT_QUIRK))
|
if ((xhci->quirks & XHCI_EP_LIMIT_QUIRK))
|
||||||
xhci_free_host_resources(xhci, ctrl_ctx);
|
xhci_free_host_resources(xhci, ctrl_ctx);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
@ -2615,27 +2626,18 @@ static int xhci_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (command) {
|
list_add_tail(&command->cmd_list, &virt_dev->cmd_list);
|
||||||
cmd_completion = command->completion;
|
|
||||||
cmd_status = &command->status;
|
|
||||||
command->command_trb = xhci_find_next_enqueue(xhci->cmd_ring);
|
|
||||||
list_add_tail(&command->cmd_list, &virt_dev->cmd_list);
|
|
||||||
} else {
|
|
||||||
cmd_completion = &virt_dev->cmd_completion;
|
|
||||||
cmd_status = &virt_dev->cmd_status;
|
|
||||||
}
|
|
||||||
init_completion(cmd_completion);
|
|
||||||
|
|
||||||
cmd_trb = xhci_find_next_enqueue(xhci->cmd_ring);
|
|
||||||
if (!ctx_change)
|
if (!ctx_change)
|
||||||
ret = xhci_queue_configure_endpoint(xhci, in_ctx->dma,
|
ret = xhci_queue_configure_endpoint(xhci, command,
|
||||||
|
command->in_ctx->dma,
|
||||||
udev->slot_id, must_succeed);
|
udev->slot_id, must_succeed);
|
||||||
else
|
else
|
||||||
ret = xhci_queue_evaluate_context(xhci, in_ctx->dma,
|
ret = xhci_queue_evaluate_context(xhci, command,
|
||||||
|
command->in_ctx->dma,
|
||||||
udev->slot_id, must_succeed);
|
udev->slot_id, must_succeed);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
if (command)
|
list_del(&command->cmd_list);
|
||||||
list_del(&command->cmd_list);
|
|
||||||
if ((xhci->quirks & XHCI_EP_LIMIT_QUIRK))
|
if ((xhci->quirks & XHCI_EP_LIMIT_QUIRK))
|
||||||
xhci_free_host_resources(xhci, ctrl_ctx);
|
xhci_free_host_resources(xhci, ctrl_ctx);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
@ -2648,7 +2650,7 @@ static int xhci_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
|
|
||||||
/* Wait for the configure endpoint command to complete */
|
/* Wait for the configure endpoint command to complete */
|
||||||
timeleft = wait_for_completion_interruptible_timeout(
|
timeleft = wait_for_completion_interruptible_timeout(
|
||||||
cmd_completion,
|
command->completion,
|
||||||
XHCI_CMD_DEFAULT_TIMEOUT);
|
XHCI_CMD_DEFAULT_TIMEOUT);
|
||||||
if (timeleft <= 0) {
|
if (timeleft <= 0) {
|
||||||
xhci_warn(xhci, "%s while waiting for %s command\n",
|
xhci_warn(xhci, "%s while waiting for %s command\n",
|
||||||
|
@ -2657,16 +2659,18 @@ static int xhci_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
"configure endpoint" :
|
"configure endpoint" :
|
||||||
"evaluate context");
|
"evaluate context");
|
||||||
/* cancel the configure endpoint command */
|
/* cancel the configure endpoint command */
|
||||||
ret = xhci_cancel_cmd(xhci, command, cmd_trb);
|
ret = xhci_cancel_cmd(xhci, command, command->command_trb);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
return -ETIME;
|
return -ETIME;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ctx_change)
|
if (!ctx_change)
|
||||||
ret = xhci_configure_endpoint_result(xhci, udev, cmd_status);
|
ret = xhci_configure_endpoint_result(xhci, udev,
|
||||||
|
&command->status);
|
||||||
else
|
else
|
||||||
ret = xhci_evaluate_context_result(xhci, udev, cmd_status);
|
ret = xhci_evaluate_context_result(xhci, udev,
|
||||||
|
&command->status);
|
||||||
|
|
||||||
if ((xhci->quirks & XHCI_EP_LIMIT_QUIRK)) {
|
if ((xhci->quirks & XHCI_EP_LIMIT_QUIRK)) {
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
|
@ -2714,6 +2718,7 @@ int xhci_check_bandwidth(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
struct xhci_virt_device *virt_dev;
|
struct xhci_virt_device *virt_dev;
|
||||||
struct xhci_input_control_ctx *ctrl_ctx;
|
struct xhci_input_control_ctx *ctrl_ctx;
|
||||||
struct xhci_slot_ctx *slot_ctx;
|
struct xhci_slot_ctx *slot_ctx;
|
||||||
|
struct xhci_command *command;
|
||||||
|
|
||||||
ret = xhci_check_args(hcd, udev, NULL, 0, true, __func__);
|
ret = xhci_check_args(hcd, udev, NULL, 0, true, __func__);
|
||||||
if (ret <= 0)
|
if (ret <= 0)
|
||||||
|
@ -2725,12 +2730,19 @@ int xhci_check_bandwidth(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
xhci_dbg(xhci, "%s called for udev %p\n", __func__, udev);
|
xhci_dbg(xhci, "%s called for udev %p\n", __func__, udev);
|
||||||
virt_dev = xhci->devs[udev->slot_id];
|
virt_dev = xhci->devs[udev->slot_id];
|
||||||
|
|
||||||
|
command = xhci_alloc_command(xhci, false, true, GFP_KERNEL);
|
||||||
|
if (!command)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
command->in_ctx = virt_dev->in_ctx;
|
||||||
|
|
||||||
/* See section 4.6.6 - A0 = 1; A1 = D0 = D1 = 0 */
|
/* See section 4.6.6 - A0 = 1; A1 = D0 = D1 = 0 */
|
||||||
ctrl_ctx = xhci_get_input_control_ctx(xhci, virt_dev->in_ctx);
|
ctrl_ctx = xhci_get_input_control_ctx(xhci, command->in_ctx);
|
||||||
if (!ctrl_ctx) {
|
if (!ctrl_ctx) {
|
||||||
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
||||||
__func__);
|
__func__);
|
||||||
return -ENOMEM;
|
ret = -ENOMEM;
|
||||||
|
goto command_cleanup;
|
||||||
}
|
}
|
||||||
ctrl_ctx->add_flags |= cpu_to_le32(SLOT_FLAG);
|
ctrl_ctx->add_flags |= cpu_to_le32(SLOT_FLAG);
|
||||||
ctrl_ctx->add_flags &= cpu_to_le32(~EP0_FLAG);
|
ctrl_ctx->add_flags &= cpu_to_le32(~EP0_FLAG);
|
||||||
|
@ -2738,20 +2750,20 @@ int xhci_check_bandwidth(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
|
|
||||||
/* Don't issue the command if there's no endpoints to update. */
|
/* Don't issue the command if there's no endpoints to update. */
|
||||||
if (ctrl_ctx->add_flags == cpu_to_le32(SLOT_FLAG) &&
|
if (ctrl_ctx->add_flags == cpu_to_le32(SLOT_FLAG) &&
|
||||||
ctrl_ctx->drop_flags == 0)
|
ctrl_ctx->drop_flags == 0) {
|
||||||
return 0;
|
ret = 0;
|
||||||
|
goto command_cleanup;
|
||||||
|
}
|
||||||
xhci_dbg(xhci, "New Input Control Context:\n");
|
xhci_dbg(xhci, "New Input Control Context:\n");
|
||||||
slot_ctx = xhci_get_slot_ctx(xhci, virt_dev->in_ctx);
|
slot_ctx = xhci_get_slot_ctx(xhci, virt_dev->in_ctx);
|
||||||
xhci_dbg_ctx(xhci, virt_dev->in_ctx,
|
xhci_dbg_ctx(xhci, virt_dev->in_ctx,
|
||||||
LAST_CTX_TO_EP_NUM(le32_to_cpu(slot_ctx->dev_info)));
|
LAST_CTX_TO_EP_NUM(le32_to_cpu(slot_ctx->dev_info)));
|
||||||
|
|
||||||
ret = xhci_configure_endpoint(xhci, udev, NULL,
|
ret = xhci_configure_endpoint(xhci, udev, command,
|
||||||
false, false);
|
false, false);
|
||||||
if (ret) {
|
if (ret)
|
||||||
/* Callee should call reset_bandwidth() */
|
/* Callee should call reset_bandwidth() */
|
||||||
return ret;
|
goto command_cleanup;
|
||||||
}
|
|
||||||
|
|
||||||
xhci_dbg(xhci, "Output context after successful config ep cmd:\n");
|
xhci_dbg(xhci, "Output context after successful config ep cmd:\n");
|
||||||
xhci_dbg_ctx(xhci, virt_dev->out_ctx,
|
xhci_dbg_ctx(xhci, virt_dev->out_ctx,
|
||||||
|
@ -2783,6 +2795,9 @@ int xhci_check_bandwidth(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
virt_dev->eps[i].ring = virt_dev->eps[i].new_ring;
|
virt_dev->eps[i].ring = virt_dev->eps[i].new_ring;
|
||||||
virt_dev->eps[i].new_ring = NULL;
|
virt_dev->eps[i].new_ring = NULL;
|
||||||
}
|
}
|
||||||
|
command_cleanup:
|
||||||
|
kfree(command->completion);
|
||||||
|
kfree(command);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -2884,9 +2899,14 @@ void xhci_cleanup_stalled_ring(struct xhci_hcd *xhci,
|
||||||
* issue a configure endpoint command later.
|
* issue a configure endpoint command later.
|
||||||
*/
|
*/
|
||||||
if (!(xhci->quirks & XHCI_RESET_EP_QUIRK)) {
|
if (!(xhci->quirks & XHCI_RESET_EP_QUIRK)) {
|
||||||
|
struct xhci_command *command;
|
||||||
|
/* Can't sleep if we're called from cleanup_halted_endpoint() */
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_ATOMIC);
|
||||||
|
if (!command)
|
||||||
|
return;
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_reset_ep,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_reset_ep,
|
||||||
"Queueing new dequeue state");
|
"Queueing new dequeue state");
|
||||||
xhci_queue_new_dequeue_state(xhci, udev->slot_id,
|
xhci_queue_new_dequeue_state(xhci, command, udev->slot_id,
|
||||||
ep_index, ep->stopped_stream, &deq_state);
|
ep_index, ep->stopped_stream, &deq_state);
|
||||||
} else {
|
} else {
|
||||||
/* Better hope no one uses the input context between now and the
|
/* Better hope no one uses the input context between now and the
|
||||||
|
@ -2917,6 +2937,7 @@ void xhci_endpoint_reset(struct usb_hcd *hcd,
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int ret;
|
int ret;
|
||||||
struct xhci_virt_ep *virt_ep;
|
struct xhci_virt_ep *virt_ep;
|
||||||
|
struct xhci_command *command;
|
||||||
|
|
||||||
xhci = hcd_to_xhci(hcd);
|
xhci = hcd_to_xhci(hcd);
|
||||||
udev = (struct usb_device *) ep->hcpriv;
|
udev = (struct usb_device *) ep->hcpriv;
|
||||||
|
@ -2939,10 +2960,14 @@ void xhci_endpoint_reset(struct usb_hcd *hcd,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_ATOMIC);
|
||||||
|
if (!command)
|
||||||
|
return;
|
||||||
|
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_reset_ep,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_reset_ep,
|
||||||
"Queueing reset endpoint command");
|
"Queueing reset endpoint command");
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
ret = xhci_queue_reset_ep(xhci, udev->slot_id, ep_index);
|
ret = xhci_queue_reset_ep(xhci, command, udev->slot_id, ep_index);
|
||||||
/*
|
/*
|
||||||
* Can't change the ring dequeue pointer until it's transitioned to the
|
* Can't change the ring dequeue pointer until it's transitioned to the
|
||||||
* stopped state, which is only upon a successful reset endpoint
|
* stopped state, which is only upon a successful reset endpoint
|
||||||
|
@ -3473,10 +3498,9 @@ int xhci_discover_or_reset_device(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
|
|
||||||
/* Attempt to submit the Reset Device command to the command ring */
|
/* Attempt to submit the Reset Device command to the command ring */
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
reset_device_cmd->command_trb = xhci_find_next_enqueue(xhci->cmd_ring);
|
|
||||||
|
|
||||||
list_add_tail(&reset_device_cmd->cmd_list, &virt_dev->cmd_list);
|
list_add_tail(&reset_device_cmd->cmd_list, &virt_dev->cmd_list);
|
||||||
ret = xhci_queue_reset_device(xhci, slot_id);
|
ret = xhci_queue_reset_device(xhci, reset_device_cmd, slot_id);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
xhci_dbg(xhci, "FIXME: allocate a command ring segment\n");
|
xhci_dbg(xhci, "FIXME: allocate a command ring segment\n");
|
||||||
list_del(&reset_device_cmd->cmd_list);
|
list_del(&reset_device_cmd->cmd_list);
|
||||||
|
@ -3589,6 +3613,11 @@ void xhci_free_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
u32 state;
|
u32 state;
|
||||||
int i, ret;
|
int i, ret;
|
||||||
|
struct xhci_command *command;
|
||||||
|
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_KERNEL);
|
||||||
|
if (!command)
|
||||||
|
return;
|
||||||
|
|
||||||
#ifndef CONFIG_USB_DEFAULT_PERSIST
|
#ifndef CONFIG_USB_DEFAULT_PERSIST
|
||||||
/*
|
/*
|
||||||
|
@ -3604,8 +3633,10 @@ void xhci_free_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
/* If the host is halted due to driver unload, we still need to free the
|
/* If the host is halted due to driver unload, we still need to free the
|
||||||
* device.
|
* device.
|
||||||
*/
|
*/
|
||||||
if (ret <= 0 && ret != -ENODEV)
|
if (ret <= 0 && ret != -ENODEV) {
|
||||||
|
kfree(command);
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
virt_dev = xhci->devs[udev->slot_id];
|
virt_dev = xhci->devs[udev->slot_id];
|
||||||
|
|
||||||
|
@ -3622,16 +3653,19 @@ void xhci_free_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
(xhci->xhc_state & XHCI_STATE_HALTED)) {
|
(xhci->xhc_state & XHCI_STATE_HALTED)) {
|
||||||
xhci_free_virt_device(xhci, udev->slot_id);
|
xhci_free_virt_device(xhci, udev->slot_id);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
kfree(command);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (xhci_queue_slot_control(xhci, TRB_DISABLE_SLOT, udev->slot_id)) {
|
if (xhci_queue_slot_control(xhci, command, TRB_DISABLE_SLOT,
|
||||||
|
udev->slot_id)) {
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
xhci_dbg(xhci, "FIXME: allocate a command ring segment\n");
|
xhci_dbg(xhci, "FIXME: allocate a command ring segment\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Event command completion handler will free any data structures
|
* Event command completion handler will free any data structures
|
||||||
* associated with the slot. XXX Can free sleep?
|
* associated with the slot. XXX Can free sleep?
|
||||||
|
@ -3671,27 +3705,35 @@ int xhci_alloc_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int timeleft;
|
int timeleft;
|
||||||
int ret;
|
int ret;
|
||||||
union xhci_trb *cmd_trb;
|
struct xhci_command *command;
|
||||||
|
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_KERNEL);
|
||||||
|
if (!command)
|
||||||
|
return 0;
|
||||||
|
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
cmd_trb = xhci_find_next_enqueue(xhci->cmd_ring);
|
command->completion = &xhci->addr_dev;
|
||||||
ret = xhci_queue_slot_control(xhci, TRB_ENABLE_SLOT, 0);
|
ret = xhci_queue_slot_control(xhci, command, TRB_ENABLE_SLOT, 0);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
xhci_dbg(xhci, "FIXME: allocate a command ring segment\n");
|
xhci_dbg(xhci, "FIXME: allocate a command ring segment\n");
|
||||||
|
kfree(command);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
|
||||||
/* XXX: how much time for xHC slot assignment? */
|
/* XXX: how much time for xHC slot assignment? */
|
||||||
timeleft = wait_for_completion_interruptible_timeout(&xhci->addr_dev,
|
timeleft = wait_for_completion_interruptible_timeout(
|
||||||
|
command->completion,
|
||||||
XHCI_CMD_DEFAULT_TIMEOUT);
|
XHCI_CMD_DEFAULT_TIMEOUT);
|
||||||
if (timeleft <= 0) {
|
if (timeleft <= 0) {
|
||||||
xhci_warn(xhci, "%s while waiting for a slot\n",
|
xhci_warn(xhci, "%s while waiting for a slot\n",
|
||||||
timeleft == 0 ? "Timeout" : "Signal");
|
timeleft == 0 ? "Timeout" : "Signal");
|
||||||
/* cancel the enable slot request */
|
/* cancel the enable slot request */
|
||||||
return xhci_cancel_cmd(xhci, NULL, cmd_trb);
|
ret = xhci_cancel_cmd(xhci, NULL, command->command_trb);
|
||||||
|
kfree(command);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!xhci->slot_id) {
|
if (!xhci->slot_id) {
|
||||||
|
@ -3699,6 +3741,7 @@ int xhci_alloc_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
xhci_err(xhci, "Max number of devices this xHCI host supports is %u.\n",
|
xhci_err(xhci, "Max number of devices this xHCI host supports is %u.\n",
|
||||||
HCS_MAX_SLOTS(
|
HCS_MAX_SLOTS(
|
||||||
readl(&xhci->cap_regs->hcs_params1)));
|
readl(&xhci->cap_regs->hcs_params1)));
|
||||||
|
kfree(command);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3733,6 +3776,8 @@ int xhci_alloc_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
pm_runtime_get_noresume(hcd->self.controller);
|
pm_runtime_get_noresume(hcd->self.controller);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
kfree(command);
|
||||||
/* Is this a LS or FS device under a HS hub? */
|
/* Is this a LS or FS device under a HS hub? */
|
||||||
/* Hub or peripherial? */
|
/* Hub or peripherial? */
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -3740,7 +3785,10 @@ int xhci_alloc_dev(struct usb_hcd *hcd, struct usb_device *udev)
|
||||||
disable_slot:
|
disable_slot:
|
||||||
/* Disable slot, if we can do it without mem alloc */
|
/* Disable slot, if we can do it without mem alloc */
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
if (!xhci_queue_slot_control(xhci, TRB_DISABLE_SLOT, udev->slot_id))
|
command->completion = NULL;
|
||||||
|
command->status = 0;
|
||||||
|
if (!xhci_queue_slot_control(xhci, command, TRB_DISABLE_SLOT,
|
||||||
|
udev->slot_id))
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -3764,7 +3812,7 @@ static int xhci_setup_device(struct usb_hcd *hcd, struct usb_device *udev,
|
||||||
struct xhci_slot_ctx *slot_ctx;
|
struct xhci_slot_ctx *slot_ctx;
|
||||||
struct xhci_input_control_ctx *ctrl_ctx;
|
struct xhci_input_control_ctx *ctrl_ctx;
|
||||||
u64 temp_64;
|
u64 temp_64;
|
||||||
union xhci_trb *cmd_trb;
|
struct xhci_command *command;
|
||||||
|
|
||||||
if (!udev->slot_id) {
|
if (!udev->slot_id) {
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_address,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_address,
|
||||||
|
@ -3785,11 +3833,19 @@ static int xhci_setup_device(struct usb_hcd *hcd, struct usb_device *udev,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
command = xhci_alloc_command(xhci, false, false, GFP_KERNEL);
|
||||||
|
if (!command)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
command->in_ctx = virt_dev->in_ctx;
|
||||||
|
command->completion = &xhci->addr_dev;
|
||||||
|
|
||||||
slot_ctx = xhci_get_slot_ctx(xhci, virt_dev->in_ctx);
|
slot_ctx = xhci_get_slot_ctx(xhci, virt_dev->in_ctx);
|
||||||
ctrl_ctx = xhci_get_input_control_ctx(xhci, virt_dev->in_ctx);
|
ctrl_ctx = xhci_get_input_control_ctx(xhci, virt_dev->in_ctx);
|
||||||
if (!ctrl_ctx) {
|
if (!ctrl_ctx) {
|
||||||
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
xhci_warn(xhci, "%s: Could not get input context, bad type.\n",
|
||||||
__func__);
|
__func__);
|
||||||
|
kfree(command);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
|
@ -3811,21 +3867,21 @@ static int xhci_setup_device(struct usb_hcd *hcd, struct usb_device *udev,
|
||||||
le32_to_cpu(slot_ctx->dev_info) >> 27);
|
le32_to_cpu(slot_ctx->dev_info) >> 27);
|
||||||
|
|
||||||
spin_lock_irqsave(&xhci->lock, flags);
|
spin_lock_irqsave(&xhci->lock, flags);
|
||||||
cmd_trb = xhci_find_next_enqueue(xhci->cmd_ring);
|
ret = xhci_queue_address_device(xhci, command, virt_dev->in_ctx->dma,
|
||||||
ret = xhci_queue_address_device(xhci, virt_dev->in_ctx->dma,
|
|
||||||
udev->slot_id, setup);
|
udev->slot_id, setup);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_address,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_address,
|
||||||
"FIXME: allocate a command ring segment");
|
"FIXME: allocate a command ring segment");
|
||||||
|
kfree(command);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
xhci_ring_cmd_db(xhci);
|
xhci_ring_cmd_db(xhci);
|
||||||
spin_unlock_irqrestore(&xhci->lock, flags);
|
spin_unlock_irqrestore(&xhci->lock, flags);
|
||||||
|
|
||||||
/* ctrl tx can take up to 5 sec; XXX: need more time for xHC? */
|
/* ctrl tx can take up to 5 sec; XXX: need more time for xHC? */
|
||||||
timeleft = wait_for_completion_interruptible_timeout(&xhci->addr_dev,
|
timeleft = wait_for_completion_interruptible_timeout(
|
||||||
XHCI_CMD_DEFAULT_TIMEOUT);
|
command->completion, XHCI_CMD_DEFAULT_TIMEOUT);
|
||||||
/* FIXME: From section 4.3.4: "Software shall be responsible for timing
|
/* FIXME: From section 4.3.4: "Software shall be responsible for timing
|
||||||
* the SetAddress() "recovery interval" required by USB and aborting the
|
* the SetAddress() "recovery interval" required by USB and aborting the
|
||||||
* command on a timeout.
|
* command on a timeout.
|
||||||
|
@ -3834,7 +3890,8 @@ static int xhci_setup_device(struct usb_hcd *hcd, struct usb_device *udev,
|
||||||
xhci_warn(xhci, "%s while waiting for setup %s command\n",
|
xhci_warn(xhci, "%s while waiting for setup %s command\n",
|
||||||
timeleft == 0 ? "Timeout" : "Signal", act);
|
timeleft == 0 ? "Timeout" : "Signal", act);
|
||||||
/* cancel the address device command */
|
/* cancel the address device command */
|
||||||
ret = xhci_cancel_cmd(xhci, NULL, cmd_trb);
|
ret = xhci_cancel_cmd(xhci, NULL, command->command_trb);
|
||||||
|
kfree(command);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
return -ETIME;
|
return -ETIME;
|
||||||
|
@ -3871,6 +3928,7 @@ static int xhci_setup_device(struct usb_hcd *hcd, struct usb_device *udev,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
kfree(command);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
temp_64 = xhci_read_64(xhci, &xhci->op_regs->dcbaa_ptr);
|
temp_64 = xhci_read_64(xhci, &xhci->op_regs->dcbaa_ptr);
|
||||||
|
@ -3905,7 +3963,7 @@ static int xhci_setup_device(struct usb_hcd *hcd, struct usb_device *udev,
|
||||||
xhci_dbg_trace(xhci, trace_xhci_dbg_address,
|
xhci_dbg_trace(xhci, trace_xhci_dbg_address,
|
||||||
"Internal device address = %d",
|
"Internal device address = %d",
|
||||||
le32_to_cpu(slot_ctx->dev_state) & DEV_ADDR_MASK);
|
le32_to_cpu(slot_ctx->dev_state) & DEV_ADDR_MASK);
|
||||||
|
kfree(command);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1807,13 +1807,14 @@ struct xhci_segment *trb_in_td(struct xhci_segment *start_seg,
|
||||||
dma_addr_t suspect_dma);
|
dma_addr_t suspect_dma);
|
||||||
int xhci_is_vendor_info_code(struct xhci_hcd *xhci, unsigned int trb_comp_code);
|
int xhci_is_vendor_info_code(struct xhci_hcd *xhci, unsigned int trb_comp_code);
|
||||||
void xhci_ring_cmd_db(struct xhci_hcd *xhci);
|
void xhci_ring_cmd_db(struct xhci_hcd *xhci);
|
||||||
int xhci_queue_slot_control(struct xhci_hcd *xhci, u32 trb_type, u32 slot_id);
|
int xhci_queue_slot_control(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
int xhci_queue_address_device(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
u32 trb_type, u32 slot_id);
|
||||||
u32 slot_id, enum xhci_setup_dev);
|
int xhci_queue_address_device(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
int xhci_queue_vendor_command(struct xhci_hcd *xhci,
|
dma_addr_t in_ctx_ptr, u32 slot_id, enum xhci_setup_dev);
|
||||||
|
int xhci_queue_vendor_command(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
u32 field1, u32 field2, u32 field3, u32 field4);
|
u32 field1, u32 field2, u32 field3, u32 field4);
|
||||||
int xhci_queue_stop_endpoint(struct xhci_hcd *xhci, int slot_id,
|
int xhci_queue_stop_endpoint(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
unsigned int ep_index, int suspend);
|
int slot_id, unsigned int ep_index, int suspend);
|
||||||
int xhci_queue_ctrl_tx(struct xhci_hcd *xhci, gfp_t mem_flags, struct urb *urb,
|
int xhci_queue_ctrl_tx(struct xhci_hcd *xhci, gfp_t mem_flags, struct urb *urb,
|
||||||
int slot_id, unsigned int ep_index);
|
int slot_id, unsigned int ep_index);
|
||||||
int xhci_queue_bulk_tx(struct xhci_hcd *xhci, gfp_t mem_flags, struct urb *urb,
|
int xhci_queue_bulk_tx(struct xhci_hcd *xhci, gfp_t mem_flags, struct urb *urb,
|
||||||
|
@ -1822,18 +1823,21 @@ int xhci_queue_intr_tx(struct xhci_hcd *xhci, gfp_t mem_flags, struct urb *urb,
|
||||||
int slot_id, unsigned int ep_index);
|
int slot_id, unsigned int ep_index);
|
||||||
int xhci_queue_isoc_tx_prepare(struct xhci_hcd *xhci, gfp_t mem_flags,
|
int xhci_queue_isoc_tx_prepare(struct xhci_hcd *xhci, gfp_t mem_flags,
|
||||||
struct urb *urb, int slot_id, unsigned int ep_index);
|
struct urb *urb, int slot_id, unsigned int ep_index);
|
||||||
int xhci_queue_configure_endpoint(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
int xhci_queue_configure_endpoint(struct xhci_hcd *xhci,
|
||||||
u32 slot_id, bool command_must_succeed);
|
struct xhci_command *cmd, dma_addr_t in_ctx_ptr, u32 slot_id,
|
||||||
int xhci_queue_evaluate_context(struct xhci_hcd *xhci, dma_addr_t in_ctx_ptr,
|
bool command_must_succeed);
|
||||||
u32 slot_id, bool command_must_succeed);
|
int xhci_queue_evaluate_context(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
int xhci_queue_reset_ep(struct xhci_hcd *xhci, int slot_id,
|
dma_addr_t in_ctx_ptr, u32 slot_id, bool command_must_succeed);
|
||||||
unsigned int ep_index);
|
int xhci_queue_reset_ep(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
int xhci_queue_reset_device(struct xhci_hcd *xhci, u32 slot_id);
|
int slot_id, unsigned int ep_index);
|
||||||
|
int xhci_queue_reset_device(struct xhci_hcd *xhci, struct xhci_command *cmd,
|
||||||
|
u32 slot_id);
|
||||||
void xhci_find_new_dequeue_state(struct xhci_hcd *xhci,
|
void xhci_find_new_dequeue_state(struct xhci_hcd *xhci,
|
||||||
unsigned int slot_id, unsigned int ep_index,
|
unsigned int slot_id, unsigned int ep_index,
|
||||||
unsigned int stream_id, struct xhci_td *cur_td,
|
unsigned int stream_id, struct xhci_td *cur_td,
|
||||||
struct xhci_dequeue_state *state);
|
struct xhci_dequeue_state *state);
|
||||||
void xhci_queue_new_dequeue_state(struct xhci_hcd *xhci,
|
void xhci_queue_new_dequeue_state(struct xhci_hcd *xhci,
|
||||||
|
struct xhci_command *cmd,
|
||||||
unsigned int slot_id, unsigned int ep_index,
|
unsigned int slot_id, unsigned int ep_index,
|
||||||
unsigned int stream_id,
|
unsigned int stream_id,
|
||||||
struct xhci_dequeue_state *deq_state);
|
struct xhci_dequeue_state *deq_state);
|
||||||
|
@ -1847,7 +1851,6 @@ int xhci_cancel_cmd(struct xhci_hcd *xhci, struct xhci_command *command,
|
||||||
union xhci_trb *cmd_trb);
|
union xhci_trb *cmd_trb);
|
||||||
void xhci_ring_ep_doorbell(struct xhci_hcd *xhci, unsigned int slot_id,
|
void xhci_ring_ep_doorbell(struct xhci_hcd *xhci, unsigned int slot_id,
|
||||||
unsigned int ep_index, unsigned int stream_id);
|
unsigned int ep_index, unsigned int stream_id);
|
||||||
union xhci_trb *xhci_find_next_enqueue(struct xhci_ring *ring);
|
|
||||||
|
|
||||||
/* xHCI roothub code */
|
/* xHCI roothub code */
|
||||||
void xhci_set_link_state(struct xhci_hcd *xhci, __le32 __iomem **port_array,
|
void xhci_set_link_state(struct xhci_hcd *xhci, __le32 __iomem **port_array,
|
||||||
|
|
Loading…
Reference in New Issue