2009-04-27 19:52:22 -07:00
/*
* xHCI host controller driver
*
* Copyright ( C ) 2008 Intel Corp .
*
* Author : Sarah Sharp
* Some code borrowed from the Linux EHCI driver .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License
* for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "xhci.h"
# define XHCI_INIT_VALUE 0x0
/* Add verbose debugging later, just print everything for now */
void xhci_dbg_regs ( struct xhci_hcd * xhci )
{
u32 temp ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " // xHCI capability registers at %p: \n " ,
xhci - > cap_regs ) ;
2009-04-27 19:52:22 -07:00
temp = xhci_readl ( xhci , & xhci - > cap_regs - > hc_capbase ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " // @%p = 0x%x (CAPLENGTH AND HCIVERSION) \n " ,
& xhci - > cap_regs - > hc_capbase , temp ) ;
2009-04-27 19:52:22 -07:00
xhci_dbg ( xhci , " // CAPLENGTH: 0x%x \n " ,
( unsigned int ) HC_LENGTH ( temp ) ) ;
#if 0
xhci_dbg ( xhci , " // HCIVERSION: 0x%x \n " ,
( unsigned int ) HC_VERSION ( temp ) ) ;
# endif
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " // xHCI operational registers at %p: \n " , xhci - > op_regs ) ;
2009-04-27 19:52:22 -07:00
temp = xhci_readl ( xhci , & xhci - > cap_regs - > run_regs_off ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " // @%p = 0x%x RTSOFF \n " ,
& xhci - > cap_regs - > run_regs_off ,
2009-04-27 19:52:22 -07:00
( unsigned int ) temp & RTSOFF_MASK ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " // xHCI runtime registers at %p: \n " , xhci - > run_regs ) ;
2009-04-27 19:52:22 -07:00
temp = xhci_readl ( xhci , & xhci - > cap_regs - > db_off ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " // @%p = 0x%x DBOFF \n " , & xhci - > cap_regs - > db_off , temp ) ;
xhci_dbg ( xhci , " // Doorbell array at %p: \n " , xhci - > dba ) ;
2009-04-27 19:52:22 -07:00
}
2009-04-29 19:05:20 -07:00
static void xhci_print_cap_regs ( struct xhci_hcd * xhci )
2009-04-27 19:52:22 -07:00
{
u32 temp ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " xHCI capability registers at %p: \n " , xhci - > cap_regs ) ;
2009-04-27 19:52:22 -07:00
temp = xhci_readl ( xhci , & xhci - > cap_regs - > hc_capbase ) ;
xhci_dbg ( xhci , " CAPLENGTH AND HCIVERSION 0x%x: \n " ,
( unsigned int ) temp ) ;
xhci_dbg ( xhci , " CAPLENGTH: 0x%x \n " ,
( unsigned int ) HC_LENGTH ( temp ) ) ;
xhci_dbg ( xhci , " HCIVERSION: 0x%x \n " ,
( unsigned int ) HC_VERSION ( temp ) ) ;
temp = xhci_readl ( xhci , & xhci - > cap_regs - > hcs_params1 ) ;
xhci_dbg ( xhci , " HCSPARAMS 1: 0x%x \n " ,
( unsigned int ) temp ) ;
xhci_dbg ( xhci , " Max device slots: %u \n " ,
( unsigned int ) HCS_MAX_SLOTS ( temp ) ) ;
xhci_dbg ( xhci , " Max interrupters: %u \n " ,
( unsigned int ) HCS_MAX_INTRS ( temp ) ) ;
xhci_dbg ( xhci , " Max ports: %u \n " ,
( unsigned int ) HCS_MAX_PORTS ( temp ) ) ;
temp = xhci_readl ( xhci , & xhci - > cap_regs - > hcs_params2 ) ;
xhci_dbg ( xhci , " HCSPARAMS 2: 0x%x \n " ,
( unsigned int ) temp ) ;
xhci_dbg ( xhci , " Isoc scheduling threshold: %u \n " ,
( unsigned int ) HCS_IST ( temp ) ) ;
xhci_dbg ( xhci , " Maximum allowed segments in event ring: %u \n " ,
( unsigned int ) HCS_ERST_MAX ( temp ) ) ;
temp = xhci_readl ( xhci , & xhci - > cap_regs - > hcs_params3 ) ;
xhci_dbg ( xhci , " HCSPARAMS 3 0x%x: \n " ,
( unsigned int ) temp ) ;
xhci_dbg ( xhci , " Worst case U1 device exit latency: %u \n " ,
( unsigned int ) HCS_U1_LATENCY ( temp ) ) ;
xhci_dbg ( xhci , " Worst case U2 device exit latency: %u \n " ,
( unsigned int ) HCS_U2_LATENCY ( temp ) ) ;
temp = xhci_readl ( xhci , & xhci - > cap_regs - > hcc_params ) ;
xhci_dbg ( xhci , " HCC PARAMS 0x%x: \n " , ( unsigned int ) temp ) ;
xhci_dbg ( xhci , " HC generates %s bit addresses \n " ,
HCC_64BIT_ADDR ( temp ) ? " 64 " : " 32 " ) ;
/* FIXME */
xhci_dbg ( xhci , " FIXME: more HCCPARAMS debugging \n " ) ;
temp = xhci_readl ( xhci , & xhci - > cap_regs - > run_regs_off ) ;
xhci_dbg ( xhci , " RTSOFF 0x%x: \n " , temp & RTSOFF_MASK ) ;
}
2009-04-29 19:05:20 -07:00
static void xhci_print_command_reg ( struct xhci_hcd * xhci )
2009-04-27 19:52:22 -07:00
{
u32 temp ;
temp = xhci_readl ( xhci , & xhci - > op_regs - > command ) ;
xhci_dbg ( xhci , " USBCMD 0x%x: \n " , temp ) ;
xhci_dbg ( xhci , " HC is %s \n " ,
( temp & CMD_RUN ) ? " running " : " being stopped " ) ;
xhci_dbg ( xhci , " HC has %sfinished hard reset \n " ,
( temp & CMD_RESET ) ? " not " : " " ) ;
xhci_dbg ( xhci , " Event Interrupts %s \n " ,
( temp & CMD_EIE ) ? " enabled " : " disabled " ) ;
xhci_dbg ( xhci , " Host System Error Interrupts %s \n " ,
( temp & CMD_EIE ) ? " enabled " : " disabled " ) ;
xhci_dbg ( xhci , " HC has %sfinished light reset \n " ,
( temp & CMD_LRESET ) ? " not " : " " ) ;
}
2009-04-29 19:05:20 -07:00
static void xhci_print_status ( struct xhci_hcd * xhci )
2009-04-27 19:52:22 -07:00
{
u32 temp ;
temp = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
xhci_dbg ( xhci , " USBSTS 0x%x: \n " , temp ) ;
xhci_dbg ( xhci , " Event ring is %sempty \n " ,
( temp & STS_EINT ) ? " not " : " " ) ;
xhci_dbg ( xhci , " %sHost System Error \n " ,
( temp & STS_FATAL ) ? " WARNING: " : " No " ) ;
xhci_dbg ( xhci , " HC is %s \n " ,
( temp & STS_HALT ) ? " halted " : " running " ) ;
}
2009-04-29 19:05:20 -07:00
static void xhci_print_op_regs ( struct xhci_hcd * xhci )
2009-04-27 19:52:22 -07:00
{
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " xHCI operational registers at %p: \n " , xhci - > op_regs ) ;
2009-04-27 19:52:22 -07:00
xhci_print_command_reg ( xhci ) ;
xhci_print_status ( xhci ) ;
}
2009-04-29 19:05:20 -07:00
static void xhci_print_ports ( struct xhci_hcd * xhci )
2009-04-27 19:57:12 -07:00
{
2011-03-29 13:40:46 +11:00
__le32 __iomem * addr ;
2009-04-27 19:57:12 -07:00
int i , j ;
int ports ;
char * names [ NUM_PORT_REGS ] = {
" status " ,
" power " ,
" link " ,
" reserved " ,
} ;
ports = HCS_MAX_PORTS ( xhci - > hcs_params1 ) ;
addr = & xhci - > op_regs - > port_status_base ;
for ( i = 0 ; i < ports ; i + + ) {
for ( j = 0 ; j < NUM_PORT_REGS ; + + j ) {
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " %p port %s reg = 0x%x \n " ,
addr , names [ j ] ,
2009-04-27 19:57:12 -07:00
( unsigned int ) xhci_readl ( xhci , addr ) ) ;
addr + + ;
}
}
}
2011-02-08 16:29:33 -08:00
void xhci_print_ir_set ( struct xhci_hcd * xhci , int set_num )
2009-04-27 19:52:22 -07:00
{
2011-02-08 16:29:33 -08:00
struct xhci_intr_reg __iomem * ir_set = & xhci - > run_regs - > ir_set [ set_num ] ;
void __iomem * addr ;
2009-04-27 19:52:22 -07:00
u32 temp ;
2009-07-27 12:03:31 -07:00
u64 temp_64 ;
2009-04-27 19:52:22 -07:00
addr = & ir_set - > irq_pending ;
temp = xhci_readl ( xhci , addr ) ;
if ( temp = = XHCI_INIT_VALUE )
return ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " %p: ir_set[%i] \n " , ir_set , set_num ) ;
2009-04-27 19:52:22 -07:00
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " %p: ir_set.pending = 0x%x \n " , addr ,
( unsigned int ) temp ) ;
2009-04-27 19:52:22 -07:00
addr = & ir_set - > irq_control ;
temp = xhci_readl ( xhci , addr ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " %p: ir_set.control = 0x%x \n " , addr ,
( unsigned int ) temp ) ;
2009-04-27 19:52:22 -07:00
addr = & ir_set - > erst_size ;
temp = xhci_readl ( xhci , addr ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " %p: ir_set.erst_size = 0x%x \n " , addr ,
( unsigned int ) temp ) ;
2009-04-27 19:52:22 -07:00
addr = & ir_set - > rsvd ;
temp = xhci_readl ( xhci , addr ) ;
if ( temp ! = XHCI_INIT_VALUE )
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " WARN: %p: ir_set.rsvd = 0x%x \n " ,
addr , ( unsigned int ) temp ) ;
2009-04-27 19:52:22 -07:00
2009-07-27 12:03:31 -07:00
addr = & ir_set - > erst_base ;
temp_64 = xhci_read_64 ( xhci , addr ) ;
xhci_dbg ( xhci , " %p: ir_set.erst_base = @%08llx \n " ,
addr , temp_64 ) ;
2009-04-27 19:52:22 -07:00
2009-07-27 12:03:31 -07:00
addr = & ir_set - > erst_dequeue ;
temp_64 = xhci_read_64 ( xhci , addr ) ;
xhci_dbg ( xhci , " %p: ir_set.erst_dequeue = @%08llx \n " ,
addr , temp_64 ) ;
2009-04-27 19:52:22 -07:00
}
void xhci_print_run_regs ( struct xhci_hcd * xhci )
{
u32 temp ;
int i ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " xHCI runtime registers at %p: \n " , xhci - > run_regs ) ;
2009-04-27 19:52:22 -07:00
temp = xhci_readl ( xhci , & xhci - > run_regs - > microframe_index ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " %p: Microframe index = 0x%x \n " ,
& xhci - > run_regs - > microframe_index ,
2009-04-27 19:52:22 -07:00
( unsigned int ) temp ) ;
for ( i = 0 ; i < 7 ; + + i ) {
temp = xhci_readl ( xhci , & xhci - > run_regs - > rsvd [ i ] ) ;
if ( temp ! = XHCI_INIT_VALUE )
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " WARN: %p: Rsvd[%i] = 0x%x \n " ,
& xhci - > run_regs - > rsvd [ i ] ,
2009-04-27 19:52:22 -07:00
i , ( unsigned int ) temp ) ;
}
}
void xhci_print_registers ( struct xhci_hcd * xhci )
{
xhci_print_cap_regs ( xhci ) ;
xhci_print_op_regs ( xhci ) ;
2009-04-27 19:57:12 -07:00
xhci_print_ports ( xhci ) ;
2009-04-27 19:52:22 -07:00
}
2009-04-27 19:52:34 -07:00
2009-04-27 19:53:56 -07:00
void xhci_print_trb_offsets ( struct xhci_hcd * xhci , union xhci_trb * trb )
{
int i ;
for ( i = 0 ; i < 4 ; + + i )
xhci_dbg ( xhci , " Offset 0x%x = 0x%x \n " ,
i * 4 , trb - > generic . field [ i ] ) ;
}
/**
* Debug a transfer request block ( TRB ) .
*/
void xhci_debug_trb ( struct xhci_hcd * xhci , union xhci_trb * trb )
{
u64 address ;
2011-03-29 13:40:46 +11:00
u32 type = le32_to_cpu ( trb - > link . control ) & TRB_TYPE_BITMASK ;
2009-04-27 19:53:56 -07:00
switch ( type ) {
case TRB_TYPE ( TRB_LINK ) :
xhci_dbg ( xhci , " Link TRB: \n " ) ;
xhci_print_trb_offsets ( xhci , trb ) ;
2011-03-29 13:40:46 +11:00
address = le64_to_cpu ( trb - > link . segment_ptr ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Next ring segment DMA address = 0x%llx \n " , address ) ;
xhci_dbg ( xhci , " Interrupter target = 0x%x \n " ,
2011-03-29 13:40:46 +11:00
GET_INTR_TARGET ( le32_to_cpu ( trb - > link . intr_target ) ) ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Cycle bit = %u \n " ,
2011-03-29 13:40:46 +11:00
( unsigned int ) ( le32_to_cpu ( trb - > link . control ) & TRB_CYCLE ) ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Toggle cycle bit = %u \n " ,
2011-03-29 13:40:46 +11:00
( unsigned int ) ( le32_to_cpu ( trb - > link . control ) & LINK_TOGGLE ) ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " No Snoop bit = %u \n " ,
2011-03-29 13:40:46 +11:00
( unsigned int ) ( le32_to_cpu ( trb - > link . control ) & TRB_NO_SNOOP ) ) ;
2009-04-27 19:53:56 -07:00
break ;
case TRB_TYPE ( TRB_TRANSFER ) :
2011-03-29 13:40:46 +11:00
address = le64_to_cpu ( trb - > trans_event . buffer ) ;
2009-04-27 19:53:56 -07:00
/*
* FIXME : look at flags to figure out if it ' s an address or if
* the data is directly in the buffer field .
*/
xhci_dbg ( xhci , " DMA address or buffer contents= %llu \n " , address ) ;
break ;
case TRB_TYPE ( TRB_COMPLETION ) :
2011-03-29 13:40:46 +11:00
address = le64_to_cpu ( trb - > event_cmd . cmd_trb ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Command TRB pointer = %llu \n " , address ) ;
xhci_dbg ( xhci , " Completion status = %u \n " ,
2011-03-29 13:40:46 +11:00
( unsigned int ) GET_COMP_CODE ( le32_to_cpu ( trb - > event_cmd . status ) ) ) ;
xhci_dbg ( xhci , " Flags = 0x%x \n " ,
( unsigned int ) le32_to_cpu ( trb - > event_cmd . flags ) ) ;
2009-04-27 19:53:56 -07:00
break ;
default :
xhci_dbg ( xhci , " Unknown TRB with TRB type ID %u \n " ,
( unsigned int ) type > > 10 ) ;
xhci_print_trb_offsets ( xhci , trb ) ;
break ;
}
}
2009-04-27 19:52:34 -07:00
/**
* Debug a segment with an xHCI ring .
*
* @ return The Link TRB of the segment , or NULL if there is no Link TRB
* ( which is a bug , since all segments must have a Link TRB ) .
*
* Prints out all TRBs in the segment , even those after the Link TRB .
*
* XXX : should we print out TRBs that the HC owns ? As long as we don ' t
* write , that should be fine . . . We shouldn ' t expect that the memory pointed to
* by the TRB is valid at all . Do we care about ones the HC owns ? Probably ,
* for HC debugging .
*/
void xhci_debug_segment ( struct xhci_hcd * xhci , struct xhci_segment * seg )
{
int i ;
2011-03-29 13:40:46 +11:00
u64 addr = seg - > dma ;
2009-04-27 19:52:34 -07:00
union xhci_trb * trb = seg - > trbs ;
for ( i = 0 ; i < TRBS_PER_SEGMENT ; + + i ) {
trb = & seg - > trbs [ i ] ;
2011-03-29 13:40:46 +11:00
xhci_dbg ( xhci , " @%016llx %08x %08x %08x %08x \n " , addr ,
( u32 ) lower_32_bits ( le64_to_cpu ( trb - > link . segment_ptr ) ) ,
( u32 ) upper_32_bits ( le64_to_cpu ( trb - > link . segment_ptr ) ) ,
( unsigned int ) le32_to_cpu ( trb - > link . intr_target ) ,
( unsigned int ) le32_to_cpu ( trb - > link . control ) ) ;
2009-04-27 19:52:34 -07:00
addr + = sizeof ( * trb ) ;
}
}
2009-04-27 19:53:56 -07:00
void xhci_dbg_ring_ptrs ( struct xhci_hcd * xhci , struct xhci_ring * ring )
{
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " Ring deq = %p (virt), 0x%llx (dma) \n " ,
ring - > dequeue ,
2009-04-29 19:05:20 -07:00
( unsigned long long ) xhci_trb_virt_to_dma ( ring - > deq_seg ,
2009-04-29 19:14:08 -07:00
ring - > dequeue ) ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Ring deq updated %u times \n " ,
ring - > deq_updates ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " Ring enq = %p (virt), 0x%llx (dma) \n " ,
ring - > enqueue ,
2009-04-29 19:05:20 -07:00
( unsigned long long ) xhci_trb_virt_to_dma ( ring - > enq_seg ,
2009-04-29 19:14:08 -07:00
ring - > enqueue ) ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Ring enq updated %u times \n " ,
ring - > enq_updates ) ;
}
2009-04-27 19:52:34 -07:00
/**
* Debugging for an xHCI ring , which is a queue broken into multiple segments .
*
* Print out each segment in the ring . Check that the DMA address in
* each link segment actually matches the segment ' s stored DMA address .
* Check that the link end bit is only set at the end of the ring .
* Check that the dequeue and enqueue pointers point to real data in this ring
* ( not some other ring ) .
*/
void xhci_debug_ring ( struct xhci_hcd * xhci , struct xhci_ring * ring )
{
/* FIXME: Throw an error if any segment doesn't have a Link TRB */
struct xhci_segment * seg ;
struct xhci_segment * first_seg = ring - > first_seg ;
xhci_debug_segment ( xhci , first_seg ) ;
2009-04-27 19:53:56 -07:00
if ( ! ring - > enq_updates & & ! ring - > deq_updates ) {
xhci_dbg ( xhci , " Ring has not been updated \n " ) ;
return ;
}
2009-04-27 19:52:34 -07:00
for ( seg = first_seg - > next ; seg ! = first_seg ; seg = seg - > next )
xhci_debug_segment ( xhci , seg ) ;
}
2010-04-02 15:34:43 -07:00
void xhci_dbg_ep_rings ( struct xhci_hcd * xhci ,
unsigned int slot_id , unsigned int ep_index ,
struct xhci_virt_ep * ep )
{
int i ;
struct xhci_ring * ring ;
if ( ep - > ep_state & EP_HAS_STREAMS ) {
for ( i = 1 ; i < ep - > stream_info - > num_streams ; i + + ) {
ring = ep - > stream_info - > stream_rings [ i ] ;
xhci_dbg ( xhci , " Dev %d endpoint %d stream ID %d: \n " ,
slot_id , ep_index , i ) ;
xhci_debug_segment ( xhci , ring - > deq_seg ) ;
}
} else {
ring = ep - > ring ;
if ( ! ring )
return ;
xhci_dbg ( xhci , " Dev %d endpoint ring %d: \n " ,
slot_id , ep_index ) ;
xhci_debug_segment ( xhci , ring - > deq_seg ) ;
}
}
2009-04-27 19:52:34 -07:00
void xhci_dbg_erst ( struct xhci_hcd * xhci , struct xhci_erst * erst )
{
2011-03-29 13:40:46 +11:00
u64 addr = erst - > erst_dma_addr ;
2009-04-27 19:52:34 -07:00
int i ;
struct xhci_erst_entry * entry ;
for ( i = 0 ; i < erst - > num_entries ; + + i ) {
entry = & erst - > entries [ i ] ;
2011-03-29 13:40:46 +11:00
xhci_dbg ( xhci , " @%016llx %08x %08x %08x %08x \n " ,
addr ,
lower_32_bits ( le64_to_cpu ( entry - > seg_addr ) ) ,
upper_32_bits ( le64_to_cpu ( entry - > seg_addr ) ) ,
( unsigned int ) le32_to_cpu ( entry - > seg_size ) ,
( unsigned int ) le32_to_cpu ( entry - > rsvd ) ) ;
2009-04-27 19:52:34 -07:00
addr + = sizeof ( * entry ) ;
}
}
void xhci_dbg_cmd_ptrs ( struct xhci_hcd * xhci )
{
2009-07-27 12:03:31 -07:00
u64 val ;
2009-04-27 19:52:34 -07:00
2009-07-27 12:03:31 -07:00
val = xhci_read_64 ( xhci , & xhci - > op_regs - > cmd_ring ) ;
xhci_dbg ( xhci , " // xHC command ring deq ptr low bits + flags = @%08x \n " ,
lower_32_bits ( val ) ) ;
xhci_dbg ( xhci , " // xHC command ring deq ptr high bits = @%08x \n " ,
upper_32_bits ( val ) ) ;
2009-04-27 19:52:34 -07:00
}
2009-04-27 19:57:38 -07:00
2009-07-27 12:05:15 -07:00
/* Print the last 32 bytes for 64-byte contexts */
static void dbg_rsvd64 ( struct xhci_hcd * xhci , u64 * ctx , dma_addr_t dma )
{
int i ;
for ( i = 0 ; i < 4 ; + + i ) {
xhci_dbg ( xhci , " @%p (virt) @%08llx "
" (dma) %#08llx - rsvd64[%d] \n " ,
& ctx [ 4 + i ] , ( unsigned long long ) dma ,
ctx [ 4 + i ] , i ) ;
dma + = 8 ;
}
}
2010-01-04 12:20:17 -08:00
char * xhci_get_slot_state ( struct xhci_hcd * xhci ,
2009-12-09 15:59:13 -08:00
struct xhci_container_ctx * ctx )
{
struct xhci_slot_ctx * slot_ctx = xhci_get_slot_ctx ( xhci , ctx ) ;
2011-03-29 13:40:46 +11:00
switch ( GET_SLOT_STATE ( le32_to_cpu ( slot_ctx - > dev_state ) ) ) {
2011-06-01 23:27:49 +02:00
case SLOT_STATE_ENABLED :
2009-12-09 15:59:13 -08:00
return " enabled/disabled " ;
2011-06-01 23:27:49 +02:00
case SLOT_STATE_DEFAULT :
2009-12-09 15:59:13 -08:00
return " default " ;
2011-06-01 23:27:49 +02:00
case SLOT_STATE_ADDRESSED :
2009-12-09 15:59:13 -08:00
return " addressed " ;
2011-06-01 23:27:49 +02:00
case SLOT_STATE_CONFIGURED :
2009-12-09 15:59:13 -08:00
return " configured " ;
default :
return " reserved " ;
}
}
2011-02-08 13:55:59 -08:00
static void xhci_dbg_slot_ctx ( struct xhci_hcd * xhci , struct xhci_container_ctx * ctx )
2009-04-27 19:57:38 -07:00
{
/* Fields are 32 bits wide, DMA addresses are in bytes */
int field_size = 32 / 8 ;
2009-07-27 12:05:08 -07:00
int i ;
2009-04-27 19:57:38 -07:00
2009-07-27 12:05:15 -07:00
struct xhci_slot_ctx * slot_ctx = xhci_get_slot_ctx ( xhci , ctx ) ;
2009-08-07 14:04:40 -07:00
dma_addr_t dma = ctx - > dma +
( ( unsigned long ) slot_ctx - ( unsigned long ) ctx - > bytes ) ;
2009-07-27 12:05:15 -07:00
int csz = HCC_64BYTE_CONTEXT ( xhci - > hcc_params ) ;
2009-04-27 19:57:38 -07:00
xhci_dbg ( xhci , " Slot Context: \n " ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - dev_info \n " ,
2009-07-27 12:05:15 -07:00
& slot_ctx - > dev_info ,
( unsigned long long ) dma , slot_ctx - > dev_info ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - dev_info2 \n " ,
2009-07-27 12:05:15 -07:00
& slot_ctx - > dev_info2 ,
( unsigned long long ) dma , slot_ctx - > dev_info2 ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - tt_info \n " ,
2009-07-27 12:05:15 -07:00
& slot_ctx - > tt_info ,
( unsigned long long ) dma , slot_ctx - > tt_info ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - dev_state \n " ,
2009-07-27 12:05:15 -07:00
& slot_ctx - > dev_state ,
( unsigned long long ) dma , slot_ctx - > dev_state ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
2009-07-13 00:23:47 +02:00
for ( i = 0 ; i < 4 ; + + i ) {
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - rsvd[%d] \n " ,
2009-07-27 12:05:15 -07:00
& slot_ctx - > reserved [ i ] , ( unsigned long long ) dma ,
slot_ctx - > reserved [ i ] , i ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
}
2009-07-27 12:05:15 -07:00
if ( csz )
dbg_rsvd64 ( xhci , ( u64 * ) slot_ctx , dma ) ;
2009-07-27 12:05:08 -07:00
}
2011-02-08 13:55:59 -08:00
static void xhci_dbg_ep_ctx ( struct xhci_hcd * xhci ,
2009-07-27 12:05:15 -07:00
struct xhci_container_ctx * ctx ,
unsigned int last_ep )
2009-07-27 12:05:08 -07:00
{
int i , j ;
int last_ep_ctx = 31 ;
/* Fields are 32 bits wide, DMA addresses are in bytes */
int field_size = 32 / 8 ;
2009-07-27 12:05:15 -07:00
int csz = HCC_64BYTE_CONTEXT ( xhci - > hcc_params ) ;
2009-07-27 12:05:08 -07:00
2009-04-27 19:57:38 -07:00
if ( last_ep < 31 )
last_ep_ctx = last_ep + 1 ;
for ( i = 0 ; i < last_ep_ctx ; + + i ) {
2009-07-27 12:05:15 -07:00
struct xhci_ep_ctx * ep_ctx = xhci_get_ep_ctx ( xhci , ctx , i ) ;
dma_addr_t dma = ctx - > dma +
2009-08-07 14:04:40 -07:00
( ( unsigned long ) ep_ctx - ( unsigned long ) ctx - > bytes ) ;
2009-07-27 12:05:15 -07:00
2009-04-27 19:57:38 -07:00
xhci_dbg ( xhci , " Endpoint %02d Context: \n " , i ) ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - ep_info \n " ,
2009-07-27 12:05:15 -07:00
& ep_ctx - > ep_info ,
( unsigned long long ) dma , ep_ctx - > ep_info ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - ep_info2 \n " ,
2009-07-27 12:05:15 -07:00
& ep_ctx - > ep_info2 ,
( unsigned long long ) dma , ep_ctx - > ep_info2 ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
2009-07-27 12:03:31 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08llx - deq \n " ,
2009-07-27 12:05:15 -07:00
& ep_ctx - > deq ,
( unsigned long long ) dma , ep_ctx - > deq ) ;
2009-07-27 12:03:31 -07:00
dma + = 2 * field_size ;
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - tx_info \n " ,
2009-07-27 12:05:15 -07:00
& ep_ctx - > tx_info ,
( unsigned long long ) dma , ep_ctx - > tx_info ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
for ( j = 0 ; j < 3 ; + + j ) {
2009-04-29 19:14:08 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - rsvd[%d] \n " ,
2009-07-27 12:05:15 -07:00
& ep_ctx - > reserved [ j ] ,
2009-04-29 19:14:08 -07:00
( unsigned long long ) dma ,
2009-07-27 12:05:15 -07:00
ep_ctx - > reserved [ j ] , j ) ;
2009-04-27 19:57:38 -07:00
dma + = field_size ;
}
2009-07-27 12:05:15 -07:00
if ( csz )
dbg_rsvd64 ( xhci , ( u64 * ) ep_ctx , dma ) ;
2009-04-27 19:57:38 -07:00
}
2009-07-27 12:05:08 -07:00
}
2009-07-27 12:05:15 -07:00
void xhci_dbg_ctx ( struct xhci_hcd * xhci ,
struct xhci_container_ctx * ctx ,
unsigned int last_ep )
2009-07-27 12:05:08 -07:00
{
int i ;
/* Fields are 32 bits wide, DMA addresses are in bytes */
int field_size = 32 / 8 ;
2009-07-27 12:05:15 -07:00
struct xhci_slot_ctx * slot_ctx ;
dma_addr_t dma = ctx - > dma ;
int csz = HCC_64BYTE_CONTEXT ( xhci - > hcc_params ) ;
if ( ctx - > type = = XHCI_CTX_TYPE_INPUT ) {
struct xhci_input_control_ctx * ctrl_ctx =
xhci_get_input_control_ctx ( xhci , ctx ) ;
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - drop flags \n " ,
& ctrl_ctx - > drop_flags , ( unsigned long long ) dma ,
ctrl_ctx - > drop_flags ) ;
2009-07-27 12:05:08 -07:00
dma + = field_size ;
2009-07-27 12:05:15 -07:00
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - add flags \n " ,
& ctrl_ctx - > add_flags , ( unsigned long long ) dma ,
ctrl_ctx - > add_flags ) ;
dma + = field_size ;
for ( i = 0 ; i < 6 ; + + i ) {
xhci_dbg ( xhci , " @%p (virt) @%08llx (dma) %#08x - rsvd2[%d] \n " ,
& ctrl_ctx - > rsvd2 [ i ] , ( unsigned long long ) dma ,
ctrl_ctx - > rsvd2 [ i ] , i ) ;
dma + = field_size ;
}
if ( csz )
dbg_rsvd64 ( xhci , ( u64 * ) ctrl_ctx , dma ) ;
2009-07-27 12:05:08 -07:00
}
2009-07-27 12:05:15 -07:00
slot_ctx = xhci_get_slot_ctx ( xhci , ctx ) ;
xhci_dbg_slot_ctx ( xhci , ctx ) ;
xhci_dbg_ep_ctx ( xhci , ctx , last_ep ) ;
2009-04-27 19:57:38 -07:00
}