2009-04-27 19:52:28 -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 <linux/irq.h>
# include <linux/module.h>
# include "xhci.h"
# define DRIVER_AUTHOR "Sarah Sharp"
# define DRIVER_DESC "'eXtensible' Host Controller (xHC) Driver"
/* TODO: copied from ehci-hcd.c - can this be refactored? */
/*
* handshake - spin reading hc until handshake completes or fails
* @ ptr : address of hc register to be read
* @ mask : bits to look at in result of read
* @ done : value of those bits when handshake succeeds
* @ usec : timeout in microseconds
*
* Returns negative errno , or zero on success
*
* Success happens when the " mask " bits have the specified value ( hardware
* handshake done ) . There are two failure modes : " usec " have passed ( major
* hardware flakeout ) , or the register reads as all - ones ( hardware removed ) .
*/
static int handshake ( struct xhci_hcd * xhci , void __iomem * ptr ,
u32 mask , u32 done , int usec )
{
u32 result ;
do {
result = xhci_readl ( xhci , ptr ) ;
if ( result = = ~ ( u32 ) 0 ) /* card removed */
return - ENODEV ;
result & = mask ;
if ( result = = done )
return 0 ;
udelay ( 1 ) ;
usec - - ;
} while ( usec > 0 ) ;
return - ETIMEDOUT ;
}
/*
* Force HC into halt state .
*
* Disable any IRQs and clear the run / stop bit .
* HC will complete any current and actively pipelined transactions , and
* should halt within 16 microframes of the run / stop bit being cleared .
* Read HC Halted bit in the status register to see when the HC is finished .
* XXX : shouldn ' t we set HC_STATE_HALT here somewhere ?
*/
int xhci_halt ( struct xhci_hcd * xhci )
{
u32 halted ;
u32 cmd ;
u32 mask ;
xhci_dbg ( xhci , " // Halt the HC \n " ) ;
/* Disable all interrupts from the host controller */
mask = ~ ( XHCI_IRQS ) ;
halted = xhci_readl ( xhci , & xhci - > op_regs - > status ) & STS_HALT ;
if ( ! halted )
mask & = ~ CMD_RUN ;
cmd = xhci_readl ( xhci , & xhci - > op_regs - > command ) ;
cmd & = mask ;
xhci_writel ( xhci , cmd , & xhci - > op_regs - > command ) ;
return handshake ( xhci , & xhci - > op_regs - > status ,
STS_HALT , STS_HALT , XHCI_MAX_HALT_USEC ) ;
}
/*
* Reset a halted HC , and set the internal HC state to HC_STATE_HALT .
*
* This resets pipelines , timers , counters , state machines , etc .
* Transactions will be terminated immediately , and operational registers
* will be set to their defaults .
*/
int xhci_reset ( struct xhci_hcd * xhci )
{
u32 command ;
u32 state ;
state = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
BUG_ON ( ( state & STS_HALT ) = = 0 ) ;
xhci_dbg ( xhci , " // Reset the HC \n " ) ;
command = xhci_readl ( xhci , & xhci - > op_regs - > command ) ;
command | = CMD_RESET ;
xhci_writel ( xhci , command , & xhci - > op_regs - > command ) ;
/* XXX: Why does EHCI set this here? Shouldn't other code do this? */
xhci_to_hcd ( xhci ) - > state = HC_STATE_HALT ;
return handshake ( xhci , & xhci - > op_regs - > command , CMD_RESET , 0 , 250 * 1000 ) ;
}
/*
* Stop the HC from processing the endpoint queues .
*/
static void xhci_quiesce ( struct xhci_hcd * xhci )
{
/*
* Queues are per endpoint , so we need to disable an endpoint or slot .
*
* To disable a slot , we need to insert a disable slot command on the
* command ring and ring the doorbell . This will also free any internal
* resources associated with the slot ( which might not be what we want ) .
*
* A Release Endpoint command sounds better - doesn ' t free internal HC
* memory , but removes the endpoints from the schedule and releases the
* bandwidth , disables the doorbells , and clears the endpoint enable
* flag . Usually used prior to a set interface command .
*
* TODO : Implement after command ring code is done .
*/
BUG_ON ( ! HC_IS_RUNNING ( xhci_to_hcd ( xhci ) - > state ) ) ;
xhci_dbg ( xhci , " Finished quiescing -- code not written yet \n " ) ;
}
#if 0
/* Set up MSI-X table for entry 0 (may claim other entries later) */
static int xhci_setup_msix ( struct xhci_hcd * xhci )
{
int ret ;
struct pci_dev * pdev = to_pci_dev ( xhci_to_hcd ( xhci ) - > self . controller ) ;
xhci - > msix_count = 0 ;
/* XXX: did I do this right? ixgbe does kcalloc for more than one */
xhci - > msix_entries = kmalloc ( sizeof ( struct msix_entry ) , GFP_KERNEL ) ;
if ( ! xhci - > msix_entries ) {
xhci_err ( xhci , " Failed to allocate MSI-X entries \n " ) ;
return - ENOMEM ;
}
xhci - > msix_entries [ 0 ] . entry = 0 ;
ret = pci_enable_msix ( pdev , xhci - > msix_entries , xhci - > msix_count ) ;
if ( ret ) {
xhci_err ( xhci , " Failed to enable MSI-X \n " ) ;
goto free_entries ;
}
/*
* Pass the xhci pointer value as the request_irq " cookie " .
* If more irqs are added , this will need to be unique for each one .
*/
ret = request_irq ( xhci - > msix_entries [ 0 ] . vector , & xhci_irq , 0 ,
" xHCI " , xhci_to_hcd ( xhci ) ) ;
if ( ret ) {
xhci_err ( xhci , " Failed to allocate MSI-X interrupt \n " ) ;
goto disable_msix ;
}
xhci_dbg ( xhci , " Finished setting up MSI-X \n " ) ;
return 0 ;
disable_msix :
pci_disable_msix ( pdev ) ;
free_entries :
kfree ( xhci - > msix_entries ) ;
xhci - > msix_entries = NULL ;
return ret ;
}
/* XXX: code duplication; can xhci_setup_msix call this? */
/* Free any IRQs and disable MSI-X */
static void xhci_cleanup_msix ( struct xhci_hcd * xhci )
{
struct pci_dev * pdev = to_pci_dev ( xhci_to_hcd ( xhci ) - > self . controller ) ;
if ( ! xhci - > msix_entries )
return ;
free_irq ( xhci - > msix_entries [ 0 ] . vector , xhci ) ;
pci_disable_msix ( pdev ) ;
kfree ( xhci - > msix_entries ) ;
xhci - > msix_entries = NULL ;
xhci_dbg ( xhci , " Finished cleaning up MSI-X \n " ) ;
}
# endif
/*
* Initialize memory for HCD and xHC ( one - time init ) .
*
* Program the PAGESIZE register , initialize the device context array , create
* device contexts ( ? ) , set up a command ring segment ( or two ? ) , create event
* ring ( one for now ) .
*/
int xhci_init ( struct usb_hcd * hcd )
{
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
int retval = 0 ;
xhci_dbg ( xhci , " xhci_init \n " ) ;
spin_lock_init ( & xhci - > lock ) ;
retval = xhci_mem_init ( xhci , GFP_KERNEL ) ;
xhci_dbg ( xhci , " Finished xhci_init \n " ) ;
return retval ;
}
2009-04-27 19:53:56 -07:00
/*
* Called in interrupt context when there might be work
* queued on the event ring
*
* xhci - > lock must be held by caller .
*/
static void xhci_work ( struct xhci_hcd * xhci )
{
u32 temp ;
/*
* Clear the op reg interrupt status first ,
* so we can receive interrupts from other MSI - X interrupters .
* Write 1 to clear the interrupt status .
*/
temp = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
temp | = STS_EINT ;
xhci_writel ( xhci , temp , & xhci - > op_regs - > status ) ;
/* FIXME when MSI-X is supported and there are multiple vectors */
/* Clear the MSI-X event interrupt status */
/* Acknowledge the interrupt */
temp = xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
temp | = 0x3 ;
xhci_writel ( xhci , temp , & xhci - > ir_set - > irq_pending ) ;
/* Flush posted writes */
xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
/* FIXME this should be a delayed service routine that clears the EHB */
handle_event ( xhci ) ;
/* Clear the event handler busy flag; the event ring should be empty. */
temp = xhci_readl ( xhci , & xhci - > ir_set - > erst_dequeue [ 0 ] ) ;
xhci_writel ( xhci , temp & ~ ERST_EHB , & xhci - > ir_set - > erst_dequeue [ 0 ] ) ;
/* Flush posted writes -- FIXME is this necessary? */
xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
}
/*-------------------------------------------------------------------------*/
/*
* xHCI spec says we can get an interrupt , and if the HC has an error condition ,
* we might get bad data out of the event ring . Section 4.10 .2 .7 has a list of
* indicators of an event TRB error , but we check the status * first * to be safe .
*/
irqreturn_t xhci_irq ( struct usb_hcd * hcd )
{
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
u32 temp , temp2 ;
spin_lock ( & xhci - > lock ) ;
/* Check if the xHC generated the interrupt, or the irq is shared */
temp = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
temp2 = xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
if ( ! ( temp & STS_EINT ) & & ! ER_IRQ_PENDING ( temp2 ) ) {
spin_unlock ( & xhci - > lock ) ;
return IRQ_NONE ;
}
temp = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
if ( temp & STS_FATAL ) {
xhci_warn ( xhci , " WARNING: Host System Error \n " ) ;
xhci_halt ( xhci ) ;
xhci_to_hcd ( xhci ) - > state = HC_STATE_HALT ;
return - ESHUTDOWN ;
}
xhci_work ( xhci ) ;
spin_unlock ( & xhci - > lock ) ;
return IRQ_HANDLED ;
}
# ifdef CONFIG_USB_XHCI_HCD_DEBUGGING
void event_ring_work ( unsigned long arg )
{
unsigned long flags ;
int temp ;
struct xhci_hcd * xhci = ( struct xhci_hcd * ) arg ;
int i , j ;
xhci_dbg ( xhci , " Poll event ring: %lu \n " , jiffies ) ;
spin_lock_irqsave ( & xhci - > lock , flags ) ;
temp = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
xhci_dbg ( xhci , " op reg status = 0x%x \n " , temp ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
xhci_dbg ( xhci , " ir_set 0 pending = 0x%x \n " , temp ) ;
xhci_dbg ( xhci , " No-op commands handled = %d \n " , xhci - > noops_handled ) ;
xhci_dbg ( xhci , " HC error bitmask = 0x%x \n " , xhci - > error_bitmask ) ;
xhci - > error_bitmask = 0 ;
xhci_dbg ( xhci , " Event ring: \n " ) ;
xhci_debug_segment ( xhci , xhci - > event_ring - > deq_seg ) ;
xhci_dbg_ring_ptrs ( xhci , xhci - > event_ring ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > erst_dequeue [ 0 ] ) ;
temp & = ERST_PTR_MASK ;
xhci_dbg ( xhci , " ERST deq = 0x%x \n " , temp ) ;
xhci_dbg ( xhci , " Command ring: \n " ) ;
xhci_debug_segment ( xhci , xhci - > cmd_ring - > deq_seg ) ;
xhci_dbg_ring_ptrs ( xhci , xhci - > cmd_ring ) ;
xhci_dbg_cmd_ptrs ( xhci ) ;
if ( xhci - > noops_submitted ! = NUM_TEST_NOOPS )
if ( setup_one_noop ( xhci ) )
ring_cmd_db ( xhci ) ;
spin_unlock_irqrestore ( & xhci - > lock , flags ) ;
if ( ! xhci - > zombie )
mod_timer ( & xhci - > event_ring_timer , jiffies + POLL_TIMEOUT * HZ ) ;
else
xhci_dbg ( xhci , " Quit polling the event ring. \n " ) ;
}
# endif
2009-04-27 19:52:28 -07:00
/*
* Start the HC after it was halted .
*
* This function is called by the USB core when the HC driver is added .
* Its opposite is xhci_stop ( ) .
*
* xhci_init ( ) must be called once before this function can be called .
* Reset the HC , enable device slot contexts , program DCBAAP , and
* set command ring pointer and event ring pointer .
*
* Setup MSI - X vectors and enable interrupts .
*/
int xhci_run ( struct usb_hcd * hcd )
{
u32 temp ;
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
2009-04-27 19:53:56 -07:00
void ( * doorbell ) ( struct xhci_hcd * ) = NULL ;
2009-04-27 19:52:28 -07:00
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " xhci_run \n " ) ;
2009-04-27 19:52:28 -07:00
#if 0 /* FIXME: MSI not setup yet */
/* Do this at the very last minute */
ret = xhci_setup_msix ( xhci ) ;
if ( ! ret )
return ret ;
return - ENOSYS ;
# endif
2009-04-27 19:53:56 -07:00
# ifdef CONFIG_USB_XHCI_HCD_DEBUGGING
init_timer ( & xhci - > event_ring_timer ) ;
xhci - > event_ring_timer . data = ( unsigned long ) xhci ;
xhci - > event_ring_timer . function = event_ring_work ;
/* Poll the event ring */
xhci - > event_ring_timer . expires = jiffies + POLL_TIMEOUT * HZ ;
xhci - > zombie = 0 ;
xhci_dbg ( xhci , " Setting event ring polling timer \n " ) ;
add_timer ( & xhci - > event_ring_timer ) ;
# endif
2009-04-27 19:52:28 -07:00
xhci_dbg ( xhci , " // Set the interrupt modulation register \n " ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > irq_control ) ;
temp & = 0xffff ;
temp | = ( u32 ) 160 ;
xhci_writel ( xhci , temp , & xhci - > ir_set - > irq_control ) ;
/* Set the HCD state before we enable the irqs */
hcd - > state = HC_STATE_RUNNING ;
temp = xhci_readl ( xhci , & xhci - > op_regs - > command ) ;
temp | = ( CMD_EIE ) ;
xhci_dbg ( xhci , " // Enable interrupts, cmd = 0x%x. \n " ,
temp ) ;
xhci_writel ( xhci , temp , & xhci - > op_regs - > command ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
xhci_dbg ( xhci , " // Enabling event ring interrupter 0x%x "
" by writing 0x%x to irq_pending \n " ,
( unsigned int ) xhci - > ir_set ,
( unsigned int ) ER_IRQ_ENABLE ( temp ) ) ;
xhci_writel ( xhci , ER_IRQ_ENABLE ( temp ) ,
& xhci - > ir_set - > irq_pending ) ;
xhci_print_ir_set ( xhci , xhci - > ir_set , 0 ) ;
2009-04-27 19:53:56 -07:00
if ( NUM_TEST_NOOPS > 0 )
doorbell = setup_one_noop ( xhci ) ;
2009-04-27 19:52:34 -07:00
xhci_dbg ( xhci , " Command ring memory map follows: \n " ) ;
xhci_debug_ring ( xhci , xhci - > cmd_ring ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg_ring_ptrs ( xhci , xhci - > cmd_ring ) ;
xhci_dbg_cmd_ptrs ( xhci ) ;
2009-04-27 19:52:34 -07:00
xhci_dbg ( xhci , " ERST memory map follows: \n " ) ;
xhci_dbg_erst ( xhci , & xhci - > erst ) ;
2009-04-27 19:53:56 -07:00
xhci_dbg ( xhci , " Event ring: \n " ) ;
xhci_debug_ring ( xhci , xhci - > event_ring ) ;
xhci_dbg_ring_ptrs ( xhci , xhci - > event_ring ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > erst_dequeue [ 1 ] ) ;
xhci_dbg ( xhci , " ERST deq upper = 0x%x \n " , temp ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > erst_dequeue [ 0 ] ) ;
temp & = ERST_PTR_MASK ;
xhci_dbg ( xhci , " ERST deq = 0x%x \n " , temp ) ;
2009-04-27 19:52:34 -07:00
2009-04-27 19:52:28 -07:00
temp = xhci_readl ( xhci , & xhci - > op_regs - > command ) ;
temp | = ( CMD_RUN ) ;
xhci_dbg ( xhci , " // Turn on HC, cmd = 0x%x. \n " ,
temp ) ;
xhci_writel ( xhci , temp , & xhci - > op_regs - > command ) ;
/* Flush PCI posted writes */
temp = xhci_readl ( xhci , & xhci - > op_regs - > command ) ;
xhci_dbg ( xhci , " // @%x = 0x%x \n " ,
( unsigned int ) & xhci - > op_regs - > command , temp ) ;
2009-04-27 19:53:56 -07:00
if ( doorbell )
( * doorbell ) ( xhci ) ;
2009-04-27 19:52:28 -07:00
xhci_dbg ( xhci , " Finished xhci_run \n " ) ;
return 0 ;
}
/*
* Stop xHCI driver .
*
* This function is called by the USB core when the HC driver is removed .
* Its opposite is xhci_run ( ) .
*
* Disable device contexts , disable IRQs , and quiesce the HC .
* Reset the HC , finish any completed transactions , and cleanup memory .
*/
void xhci_stop ( struct usb_hcd * hcd )
{
u32 temp ;
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
spin_lock_irq ( & xhci - > lock ) ;
if ( HC_IS_RUNNING ( hcd - > state ) )
xhci_quiesce ( xhci ) ;
xhci_halt ( xhci ) ;
xhci_reset ( xhci ) ;
spin_unlock_irq ( & xhci - > lock ) ;
#if 0 /* No MSI yet */
xhci_cleanup_msix ( xhci ) ;
# endif
2009-04-27 19:53:56 -07:00
# ifdef CONFIG_USB_XHCI_HCD_DEBUGGING
/* Tell the event ring poll function not to reschedule */
xhci - > zombie = 1 ;
del_timer_sync ( & xhci - > event_ring_timer ) ;
# endif
2009-04-27 19:52:28 -07:00
xhci_dbg ( xhci , " // Disabling event ring interrupts \n " ) ;
temp = xhci_readl ( xhci , & xhci - > op_regs - > status ) ;
xhci_writel ( xhci , temp & ~ STS_EINT , & xhci - > op_regs - > status ) ;
temp = xhci_readl ( xhci , & xhci - > ir_set - > irq_pending ) ;
xhci_writel ( xhci , ER_IRQ_DISABLE ( temp ) ,
& xhci - > ir_set - > irq_pending ) ;
xhci_print_ir_set ( xhci , xhci - > ir_set , 0 ) ;
xhci_dbg ( xhci , " cleaning up memory \n " ) ;
xhci_mem_cleanup ( xhci ) ;
xhci_dbg ( xhci , " xhci_stop completed - status = %x \n " ,
xhci_readl ( xhci , & xhci - > op_regs - > status ) ) ;
}
/*
* Shutdown HC ( not bus - specific )
*
* This is called when the machine is rebooting or halting . We assume that the
* machine will be powered off , and the HC ' s internal state will be reset .
* Don ' t bother to free memory .
*/
void xhci_shutdown ( struct usb_hcd * hcd )
{
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
spin_lock_irq ( & xhci - > lock ) ;
xhci_halt ( xhci ) ;
spin_unlock_irq ( & xhci - > lock ) ;
#if 0
xhci_cleanup_msix ( xhci ) ;
# endif
xhci_dbg ( xhci , " xhci_shutdown completed - status = %x \n " ,
xhci_readl ( xhci , & xhci - > op_regs - > status ) ) ;
}
2009-04-27 19:53:56 -07:00
/*-------------------------------------------------------------------------*/
2009-04-27 19:52:28 -07:00
int xhci_get_frame ( struct usb_hcd * hcd )
{
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
/* EHCI mods by the periodic size. Why? */
return xhci_readl ( xhci , & xhci - > run_regs - > microframe_index ) > > 3 ;
}
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_AUTHOR ( DRIVER_AUTHOR ) ;
MODULE_LICENSE ( " GPL " ) ;
static int __init xhci_hcd_init ( void )
{
# ifdef CONFIG_PCI
int retval = 0 ;
retval = xhci_register_pci ( ) ;
if ( retval < 0 ) {
printk ( KERN_DEBUG " Problem registering PCI driver. " ) ;
return retval ;
}
# endif
return 0 ;
}
module_init ( xhci_hcd_init ) ;
static void __exit xhci_hcd_cleanup ( void )
{
# ifdef CONFIG_PCI
xhci_unregister_pci ( ) ;
# endif
}
module_exit ( xhci_hcd_cleanup ) ;