2005-04-16 15:20:36 -07:00
/*
* UHCI - specific debugging code . Invaluable when something
* goes wrong , but don ' t get in my face .
*
2005-11-30 17:16:19 -05:00
* Kernel visible pointers are surrounded in [ ] s and bus
* visible pointers are surrounded in ( ) s
2005-04-16 15:20:36 -07:00
*
* ( C ) Copyright 1999 Linus Torvalds
* ( C ) Copyright 1999 - 2001 Johannes Erdfelt
*/
# include <linux/kernel.h>
# include <linux/debugfs.h>
# include <linux/smp_lock.h>
# include <asm/io.h>
# include "uhci-hcd.h"
2006-08-05 20:37:11 -03:00
# define uhci_debug_operations (* (const struct file_operations *) NULL)
2005-12-17 18:03:37 -05:00
static struct dentry * uhci_debugfs_root ;
# ifdef DEBUG
2005-04-16 15:20:36 -07:00
2005-11-30 17:16:19 -05:00
/* Handle REALLY large printks so we don't overflow buffers */
2005-12-17 18:03:37 -05:00
static void lprintk ( char * buf )
2005-04-16 15:20:36 -07:00
{
char * p ;
/* Just write one line at a time */
while ( buf ) {
p = strchr ( buf , ' \n ' ) ;
if ( p )
* p = 0 ;
printk ( KERN_DEBUG " %s \n " , buf ) ;
buf = p ;
if ( buf )
buf + + ;
}
}
static int uhci_show_td ( struct uhci_td * td , char * buf , int len , int space )
{
char * out = buf ;
char * spid ;
u32 status , token ;
/* Try to make sure there's enough memory */
if ( len < 160 )
return 0 ;
status = td_status ( td ) ;
out + = sprintf ( out , " %*s[%p] link (%08x) " , space , " " , td , le32_to_cpu ( td - > link ) ) ;
out + = sprintf ( out , " e%d %s%s%s%s%s%s%s%s%s%sLength=%x " ,
( ( status > > 27 ) & 3 ) ,
( status & TD_CTRL_SPD ) ? " SPD " : " " ,
( status & TD_CTRL_LS ) ? " LS " : " " ,
( status & TD_CTRL_IOC ) ? " IOC " : " " ,
( status & TD_CTRL_ACTIVE ) ? " Active " : " " ,
( status & TD_CTRL_STALLED ) ? " Stalled " : " " ,
( status & TD_CTRL_DBUFERR ) ? " DataBufErr " : " " ,
( status & TD_CTRL_BABBLE ) ? " Babble " : " " ,
( status & TD_CTRL_NAK ) ? " NAK " : " " ,
( status & TD_CTRL_CRCTIMEO ) ? " CRC/Timeo " : " " ,
( status & TD_CTRL_BITSTUFF ) ? " BitStuff " : " " ,
status & 0x7ff ) ;
token = td_token ( td ) ;
switch ( uhci_packetid ( token ) ) {
case USB_PID_SETUP :
spid = " SETUP " ;
break ;
case USB_PID_OUT :
spid = " OUT " ;
break ;
case USB_PID_IN :
spid = " IN " ;
break ;
default :
spid = " ? " ;
break ;
}
out + = sprintf ( out , " MaxLen=%x DT%d EndPt=%x Dev=%x, PID=%x(%s) " ,
token > > 21 ,
( ( token > > 19 ) & 1 ) ,
( token > > 15 ) & 15 ,
( token > > 8 ) & 127 ,
( token & 0xff ) ,
spid ) ;
out + = sprintf ( out , " (buf=%08x) \n " , le32_to_cpu ( td - > buffer ) ) ;
return out - buf ;
}
2005-12-17 17:58:46 -05:00
static int uhci_show_urbp ( struct urb_priv * urbp , char * buf , int len , int space )
2005-04-16 15:20:36 -07:00
{
char * out = buf ;
struct uhci_td * td ;
2005-12-17 17:58:46 -05:00
int i , nactive , ninactive ;
2006-05-05 16:26:58 -04:00
char * ptype ;
2005-12-17 17:58:46 -05:00
if ( len < 200 )
return 0 ;
out + = sprintf ( out , " urb_priv [%p] " , urbp ) ;
out + = sprintf ( out , " urb [%p] " , urbp - > urb ) ;
out + = sprintf ( out , " qh [%p] " , urbp - > qh ) ;
out + = sprintf ( out , " Dev=%d " , usb_pipedevice ( urbp - > urb - > pipe ) ) ;
out + = sprintf ( out , " EP=%x(%s) " , usb_pipeendpoint ( urbp - > urb - > pipe ) ,
( usb_pipein ( urbp - > urb - > pipe ) ? " IN " : " OUT " ) ) ;
switch ( usb_pipetype ( urbp - > urb - > pipe ) ) {
2006-05-05 16:26:58 -04:00
case PIPE_ISOCHRONOUS : ptype = " ISO " ; break ;
case PIPE_INTERRUPT : ptype = " INT " ; break ;
case PIPE_BULK : ptype = " BLK " ; break ;
default :
case PIPE_CONTROL : ptype = " CTL " ; break ;
2005-12-17 17:58:46 -05:00
}
2006-05-05 16:26:58 -04:00
out + = sprintf ( out , " %s%s " , ptype , ( urbp - > fsbr ? " FSBR " : " " ) ) ;
2006-05-12 11:19:19 -04:00
out + = sprintf ( out , " Actlen=%d " , urbp - > urb - > actual_length ) ;
2005-12-17 17:58:46 -05:00
2007-08-21 15:40:36 -04:00
if ( urbp - > urb - > unlinked )
out + = sprintf ( out , " Unlinked=%d " , urbp - > urb - > unlinked ) ;
2005-12-17 17:58:46 -05:00
out + = sprintf ( out , " \n " ) ;
i = nactive = ninactive = 0 ;
list_for_each_entry ( td , & urbp - > td_list , list ) {
2006-05-19 16:52:35 -04:00
if ( urbp - > qh - > type ! = USB_ENDPOINT_XFER_ISOC & &
( + + i < = 10 | | debug > 2 ) ) {
2005-12-17 17:58:46 -05:00
out + = sprintf ( out , " %*s%d: " , space + 2 , " " , i ) ;
out + = uhci_show_td ( td , out , len - ( out - buf ) , 0 ) ;
} else {
if ( td_status ( td ) & TD_CTRL_ACTIVE )
+ + nactive ;
else
+ + ninactive ;
}
}
if ( nactive + ninactive > 0 )
out + = sprintf ( out , " %*s[skipped %d inactive and %d active "
" TDs] \n " ,
space , " " , ninactive , nactive ) ;
return out - buf ;
}
2007-03-19 15:31:42 -04:00
static int uhci_show_qh ( struct uhci_hcd * uhci ,
struct uhci_qh * qh , char * buf , int len , int space )
2005-12-17 17:58:46 -05:00
{
char * out = buf ;
int i , nurbs ;
2005-04-16 15:20:36 -07:00
__le32 element = qh_element ( qh ) ;
2006-05-05 16:26:58 -04:00
char * qtype ;
2005-04-16 15:20:36 -07:00
/* Try to make sure there's enough memory */
2006-05-19 16:44:55 -04:00
if ( len < 80 * 7 )
2005-04-16 15:20:36 -07:00
return 0 ;
2006-05-05 16:26:58 -04:00
switch ( qh - > type ) {
case USB_ENDPOINT_XFER_ISOC : qtype = " ISO " ; break ;
case USB_ENDPOINT_XFER_INT : qtype = " INT " ; break ;
case USB_ENDPOINT_XFER_BULK : qtype = " BLK " ; break ;
case USB_ENDPOINT_XFER_CONTROL : qtype = " CTL " ; break ;
default : qtype = " Skel " ; break ;
}
out + = sprintf ( out , " %*s[%p] %s QH link (%08x) element (%08x) \n " ,
space , " " , qh , qtype ,
le32_to_cpu ( qh - > link ) , le32_to_cpu ( element ) ) ;
2006-05-19 16:44:55 -04:00
if ( qh - > type = = USB_ENDPOINT_XFER_ISOC )
2007-01-16 11:56:32 -05:00
out + = sprintf ( out , " %*s period %d phase %d load %d us, "
" frame %x desc [%p] \n " ,
space , " " , qh - > period , qh - > phase , qh - > load ,
qh - > iso_frame , qh - > iso_packet_desc ) ;
else if ( qh - > type = = USB_ENDPOINT_XFER_INT )
out + = sprintf ( out , " %*s period %d phase %d load %d us \n " ,
space , " " , qh - > period , qh - > phase , qh - > load ) ;
2005-04-16 15:20:36 -07:00
if ( element & UHCI_PTR_QH )
out + = sprintf ( out , " %*s Element points to QH (bug?) \n " , space , " " ) ;
if ( element & UHCI_PTR_DEPTH )
out + = sprintf ( out , " %*s Depth traverse \n " , space , " " ) ;
if ( element & cpu_to_le32 ( 8 ) )
out + = sprintf ( out , " %*s Bit 3 set (bug?) \n " , space , " " ) ;
if ( ! ( element & ~ ( UHCI_PTR_QH | UHCI_PTR_DEPTH ) ) )
out + = sprintf ( out , " %*s Element is NULL (bug?) \n " , space , " " ) ;
2005-12-17 17:58:46 -05:00
if ( list_empty ( & qh - > queue ) ) {
out + = sprintf ( out , " %*s queue is empty \n " , space , " " ) ;
2007-03-19 15:31:42 -04:00
if ( qh = = uhci - > skel_async_qh )
out + = uhci_show_td ( uhci - > term_td , out ,
len - ( out - buf ) , 0 ) ;
2005-12-17 17:58:46 -05:00
} else {
struct urb_priv * urbp = list_entry ( qh - > queue . next ,
struct urb_priv , node ) ;
struct uhci_td * td = list_entry ( urbp - > td_list . next ,
struct uhci_td , list ) ;
2007-02-19 15:51:51 -05:00
if ( element ! = LINK_TO_TD ( td ) )
2005-12-17 17:58:46 -05:00
out + = sprintf ( out , " %*s Element != First TD \n " ,
space , " " ) ;
i = nurbs = 0 ;
list_for_each_entry ( urbp , & qh - > queue , node ) {
if ( + + i < = 10 )
out + = uhci_show_urbp ( urbp , out ,
len - ( out - buf ) , space + 2 ) ;
else
+ + nurbs ;
2005-04-16 15:20:36 -07:00
}
2005-12-17 17:58:46 -05:00
if ( nurbs > 0 )
out + = sprintf ( out , " %*s Skipped %d URBs \n " ,
space , " " , nurbs ) ;
2005-04-16 15:20:36 -07:00
}
2007-01-08 12:01:43 -05:00
if ( qh - > dummy_td ) {
2005-12-17 18:00:12 -05:00
out + = sprintf ( out , " %*s Dummy TD \n " , space , " " ) ;
out + = uhci_show_td ( qh - > dummy_td , out , len - ( out - buf ) , 0 ) ;
}
2005-04-16 15:20:36 -07:00
return out - buf ;
}
static int uhci_show_sc ( int port , unsigned short status , char * buf , int len )
{
char * out = buf ;
/* Try to make sure there's enough memory */
if ( len < 160 )
return 0 ;
out + = sprintf ( out , " stat%d = %04x %s%s%s%s%s%s%s%s%s%s \n " ,
port ,
status ,
( status & USBPORTSC_SUSP ) ? " Suspend " : " " ,
( status & USBPORTSC_OCC ) ? " OverCurrentChange " : " " ,
( status & USBPORTSC_OC ) ? " OverCurrent " : " " ,
( status & USBPORTSC_PR ) ? " Reset " : " " ,
( status & USBPORTSC_LSDA ) ? " LowSpeed " : " " ,
( status & USBPORTSC_RD ) ? " ResumeDetect " : " " ,
( status & USBPORTSC_PEC ) ? " EnableChange " : " " ,
( status & USBPORTSC_PE ) ? " Enabled " : " " ,
( status & USBPORTSC_CSC ) ? " ConnectChange " : " " ,
( status & USBPORTSC_CCS ) ? " Connected " : " " ) ;
return out - buf ;
}
2005-04-09 17:27:32 -04:00
static int uhci_show_root_hub_state ( struct uhci_hcd * uhci , char * buf , int len )
{
char * out = buf ;
char * rh_state ;
/* Try to make sure there's enough memory */
if ( len < 60 )
return 0 ;
switch ( uhci - > rh_state ) {
case UHCI_RH_RESET :
rh_state = " reset " ; break ;
case UHCI_RH_SUSPENDED :
rh_state = " suspended " ; break ;
case UHCI_RH_AUTO_STOPPED :
rh_state = " auto-stopped " ; break ;
case UHCI_RH_RESUMING :
rh_state = " resuming " ; break ;
case UHCI_RH_SUSPENDING :
rh_state = " suspending " ; break ;
case UHCI_RH_RUNNING :
rh_state = " running " ; break ;
case UHCI_RH_RUNNING_NODEVS :
rh_state = " running, no devs " ; break ;
default :
rh_state = " ? " ; break ;
}
2006-05-12 11:35:45 -04:00
out + = sprintf ( out , " Root-hub state: %s FSBR: %d \n " ,
rh_state , uhci - > fsbr_is_on ) ;
2005-04-09 17:27:32 -04:00
return out - buf ;
}
2005-04-16 15:20:36 -07:00
static int uhci_show_status ( struct uhci_hcd * uhci , char * buf , int len )
{
char * out = buf ;
unsigned long io_addr = uhci - > io_addr ;
unsigned short usbcmd , usbstat , usbint , usbfrnum ;
unsigned int flbaseadd ;
unsigned char sof ;
unsigned short portsc1 , portsc2 ;
/* Try to make sure there's enough memory */
2006-05-19 16:34:57 -04:00
if ( len < 80 * 9 )
2005-04-16 15:20:36 -07:00
return 0 ;
usbcmd = inw ( io_addr + 0 ) ;
usbstat = inw ( io_addr + 2 ) ;
usbint = inw ( io_addr + 4 ) ;
usbfrnum = inw ( io_addr + 6 ) ;
flbaseadd = inl ( io_addr + 8 ) ;
sof = inb ( io_addr + 12 ) ;
portsc1 = inw ( io_addr + 16 ) ;
portsc2 = inw ( io_addr + 18 ) ;
out + = sprintf ( out , " usbcmd = %04x %s%s%s%s%s%s%s%s \n " ,
usbcmd ,
( usbcmd & USBCMD_MAXP ) ? " Maxp64 " : " Maxp32 " ,
( usbcmd & USBCMD_CF ) ? " CF " : " " ,
( usbcmd & USBCMD_SWDBG ) ? " SWDBG " : " " ,
( usbcmd & USBCMD_FGR ) ? " FGR " : " " ,
( usbcmd & USBCMD_EGSM ) ? " EGSM " : " " ,
( usbcmd & USBCMD_GRESET ) ? " GRESET " : " " ,
( usbcmd & USBCMD_HCRESET ) ? " HCRESET " : " " ,
( usbcmd & USBCMD_RS ) ? " RS " : " " ) ;
out + = sprintf ( out , " usbstat = %04x %s%s%s%s%s%s \n " ,
usbstat ,
( usbstat & USBSTS_HCH ) ? " HCHalted " : " " ,
( usbstat & USBSTS_HCPE ) ? " HostControllerProcessError " : " " ,
( usbstat & USBSTS_HSE ) ? " HostSystemError " : " " ,
( usbstat & USBSTS_RD ) ? " ResumeDetect " : " " ,
( usbstat & USBSTS_ERROR ) ? " USBError " : " " ,
( usbstat & USBSTS_USBINT ) ? " USBINT " : " " ) ;
out + = sprintf ( out , " usbint = %04x \n " , usbint ) ;
out + = sprintf ( out , " usbfrnum = (%d)%03x \n " , ( usbfrnum > > 10 ) & 1 ,
0xfff & ( 4 * ( unsigned int ) usbfrnum ) ) ;
out + = sprintf ( out , " flbaseadd = %08x \n " , flbaseadd ) ;
out + = sprintf ( out , " sof = %02x \n " , sof ) ;
out + = uhci_show_sc ( 1 , portsc1 , out , len - ( out - buf ) ) ;
out + = uhci_show_sc ( 2 , portsc2 , out , len - ( out - buf ) ) ;
2006-05-19 16:52:35 -04:00
out + = sprintf ( out , " Most recent frame: %x (%d) "
" Last ISO frame: %x (%d) \n " ,
uhci - > frame_number , uhci - > frame_number & 1023 ,
uhci - > last_iso_frame , uhci - > last_iso_frame & 1023 ) ;
2005-04-16 15:20:36 -07:00
return out - buf ;
}
static int uhci_sprint_schedule ( struct uhci_hcd * uhci , char * buf , int len )
{
char * out = buf ;
int i , j ;
struct uhci_qh * qh ;
struct uhci_td * td ;
struct list_head * tmp , * head ;
2007-01-08 12:00:28 -05:00
int nframes , nerrs ;
2007-02-19 15:52:45 -05:00
__le32 link ;
2007-03-19 15:31:42 -04:00
__le32 fsbr_link ;
2007-02-19 15:52:45 -05:00
static const char * const qh_names [ ] = {
" unlink " , " iso " , " int128 " , " int64 " , " int32 " , " int16 " ,
" int8 " , " int4 " , " int2 " , " async " , " term "
} ;
2005-04-16 15:20:36 -07:00
2005-04-09 17:27:32 -04:00
out + = uhci_show_root_hub_state ( uhci , out , len - ( out - buf ) ) ;
2005-04-16 15:20:36 -07:00
out + = sprintf ( out , " HC status \n " ) ;
out + = uhci_show_status ( uhci , out , len - ( out - buf ) ) ;
2007-01-16 11:56:32 -05:00
out + = sprintf ( out , " Periodic load table \n " ) ;
for ( i = 0 ; i < MAX_PHASE ; + + i ) {
out + = sprintf ( out , " \t %d " , uhci - > load [ i ] ) ;
if ( i % 8 = = 7 )
* out + + = ' \n ' ;
}
out + = sprintf ( out , " Total: %d, #INT: %d, #ISO: %d \n " ,
uhci - > total_load ,
uhci_to_hcd ( uhci ) - > self . bandwidth_int_reqs ,
uhci_to_hcd ( uhci ) - > self . bandwidth_isoc_reqs ) ;
2005-12-17 17:58:46 -05:00
if ( debug < = 1 )
return out - buf ;
2005-04-16 15:20:36 -07:00
out + = sprintf ( out , " Frame List \n " ) ;
2007-01-08 12:00:28 -05:00
nframes = 10 ;
nerrs = 0 ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < UHCI_NUMFRAMES ; + + i ) {
2007-02-19 15:52:45 -05:00
__le32 qh_dma ;
2007-01-08 12:00:28 -05:00
j = 0 ;
2005-09-16 14:22:51 -04:00
td = uhci - > frame_cpu [ i ] ;
2007-01-08 12:00:28 -05:00
link = uhci - > frame [ i ] ;
2005-04-16 15:20:36 -07:00
if ( ! td )
2007-01-08 12:00:28 -05:00
goto check_link ;
2005-04-16 15:20:36 -07:00
2007-01-08 12:00:28 -05:00
if ( nframes > 0 ) {
out + = sprintf ( out , " - Frame %d -> (%08x) \n " ,
i , le32_to_cpu ( link ) ) ;
j = 1 ;
}
2005-04-16 15:20:36 -07:00
head = & td - > fl_list ;
tmp = head ;
do {
td = list_entry ( tmp , struct uhci_td , fl_list ) ;
tmp = tmp - > next ;
2007-02-19 15:51:51 -05:00
if ( link ! = LINK_TO_TD ( td ) ) {
2007-01-08 12:00:28 -05:00
if ( nframes > 0 )
out + = sprintf ( out , " link does "
" not match list entry! \n " ) ;
else
+ + nerrs ;
}
if ( nframes > 0 )
out + = uhci_show_td ( td , out ,
len - ( out - buf ) , 4 ) ;
link = td - > link ;
2005-04-16 15:20:36 -07:00
} while ( tmp ! = head ) ;
2007-01-08 12:00:28 -05:00
check_link :
qh_dma = uhci_frame_skel_link ( uhci , i ) ;
if ( link ! = qh_dma ) {
if ( nframes > 0 ) {
if ( ! j ) {
out + = sprintf ( out ,
" - Frame %d -> (%08x) \n " ,
i , le32_to_cpu ( link ) ) ;
j = 1 ;
}
out + = sprintf ( out , " link does not match "
" QH (%08x)! \n " , le32_to_cpu ( qh_dma ) ) ;
} else
+ + nerrs ;
}
nframes - = j ;
2005-04-16 15:20:36 -07:00
}
2007-01-08 12:00:28 -05:00
if ( nerrs > 0 )
out + = sprintf ( out , " Skipped %d bad links \n " , nerrs ) ;
2005-04-16 15:20:36 -07:00
2005-11-30 17:16:19 -05:00
out + = sprintf ( out , " Skeleton QHs \n " ) ;
2005-04-16 15:20:36 -07:00
2007-03-19 15:31:42 -04:00
fsbr_link = 0 ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < UHCI_NUM_SKELQH ; + + i ) {
2005-12-17 17:58:46 -05:00
int cnt = 0 ;
2005-04-16 15:20:36 -07:00
qh = uhci - > skelqh [ i ] ;
2007-02-19 15:52:45 -05:00
out + = sprintf ( out , " - skel_%s_qh \n " , qh_names [ i ] ) ; \
2007-03-19 15:31:42 -04:00
out + = uhci_show_qh ( uhci , qh , out , len - ( out - buf ) , 4 ) ;
2005-04-16 15:20:36 -07:00
/* Last QH is the Terminating QH, it's different */
2007-02-19 15:52:45 -05:00
if ( i = = SKEL_TERM ) {
2007-02-19 15:51:51 -05:00
if ( qh_element ( qh ) ! = LINK_TO_TD ( uhci - > term_td ) )
2005-04-16 15:20:36 -07:00
out + = sprintf ( out , " skel_term_qh element is not set to term_td! \n " ) ;
2007-03-19 15:31:42 -04:00
link = fsbr_link ;
if ( ! link )
link = LINK_TO_QH ( uhci - > skel_term_qh ) ;
goto check_qh_link ;
2005-04-16 15:20:36 -07:00
}
2005-12-17 17:58:46 -05:00
head = & qh - > node ;
2005-04-16 15:20:36 -07:00
tmp = head - > next ;
while ( tmp ! = head ) {
2005-12-17 17:58:46 -05:00
qh = list_entry ( tmp , struct uhci_qh , node ) ;
2005-04-16 15:20:36 -07:00
tmp = tmp - > next ;
2005-12-17 17:58:46 -05:00
if ( + + cnt < = 10 )
2007-03-19 15:31:42 -04:00
out + = uhci_show_qh ( uhci , qh , out ,
2005-12-17 17:58:46 -05:00
len - ( out - buf ) , 4 ) ;
2007-02-19 15:52:45 -05:00
if ( ! fsbr_link & & qh - > skel > = SKEL_FSBR )
fsbr_link = LINK_TO_QH ( qh ) ;
2005-04-16 15:20:36 -07:00
}
2005-12-17 17:58:46 -05:00
if ( ( cnt - = 10 ) > 0 )
out + = sprintf ( out , " Skipped %d QHs \n " , cnt ) ;
2005-04-16 15:20:36 -07:00
2007-02-19 15:52:45 -05:00
link = UHCI_PTR_TERM ;
if ( i < = SKEL_ISO )
;
else if ( i < SKEL_ASYNC )
link = LINK_TO_QH ( uhci - > skel_async_qh ) ;
else if ( ! uhci - > fsbr_is_on )
;
else
link = LINK_TO_QH ( uhci - > skel_term_qh ) ;
check_qh_link :
if ( qh - > link ! = link )
out + = sprintf ( out , " last QH not linked to next skeleton! \n " ) ;
2005-04-16 15:20:36 -07:00
}
return out - buf ;
}
2005-12-17 18:03:37 -05:00
# ifdef CONFIG_DEBUG_FS
2005-04-16 15:20:36 -07:00
# define MAX_OUTPUT (64 * 1024)
struct uhci_debug {
int size ;
char * data ;
} ;
static int uhci_debug_open ( struct inode * inode , struct file * file )
{
2006-09-27 01:50:46 -07:00
struct uhci_hcd * uhci = inode - > i_private ;
2005-04-16 15:20:36 -07:00
struct uhci_debug * up ;
int ret = - ENOMEM ;
2005-12-17 17:58:46 -05:00
unsigned long flags ;
2005-04-16 15:20:36 -07:00
lock_kernel ( ) ;
up = kmalloc ( sizeof ( * up ) , GFP_KERNEL ) ;
if ( ! up )
goto out ;
up - > data = kmalloc ( MAX_OUTPUT , GFP_KERNEL ) ;
if ( ! up - > data ) {
kfree ( up ) ;
goto out ;
}
2005-12-17 18:03:37 -05:00
up - > size = 0 ;
2005-12-17 17:58:46 -05:00
spin_lock_irqsave ( & uhci - > lock , flags ) ;
2005-12-17 18:03:37 -05:00
if ( uhci - > is_initialized )
up - > size = uhci_sprint_schedule ( uhci , up - > data , MAX_OUTPUT ) ;
2005-12-17 17:58:46 -05:00
spin_unlock_irqrestore ( & uhci - > lock , flags ) ;
2005-04-16 15:20:36 -07:00
file - > private_data = up ;
ret = 0 ;
out :
unlock_kernel ( ) ;
return ret ;
}
static loff_t uhci_debug_lseek ( struct file * file , loff_t off , int whence )
{
struct uhci_debug * up ;
loff_t new = - 1 ;
lock_kernel ( ) ;
up = file - > private_data ;
switch ( whence ) {
case 0 :
new = off ;
break ;
case 1 :
new = file - > f_pos + off ;
break ;
}
if ( new < 0 | | new > up - > size ) {
unlock_kernel ( ) ;
return - EINVAL ;
}
unlock_kernel ( ) ;
return ( file - > f_pos = new ) ;
}
static ssize_t uhci_debug_read ( struct file * file , char __user * buf ,
size_t nbytes , loff_t * ppos )
{
struct uhci_debug * up = file - > private_data ;
return simple_read_from_buffer ( buf , nbytes , ppos , up - > data , up - > size ) ;
}
static int uhci_debug_release ( struct inode * inode , struct file * file )
{
struct uhci_debug * up = file - > private_data ;
kfree ( up - > data ) ;
kfree ( up ) ;
return 0 ;
}
2005-12-17 18:03:37 -05:00
# undef uhci_debug_operations
2006-08-05 20:37:11 -03:00
static const struct file_operations uhci_debug_operations = {
2005-12-17 18:03:37 -05:00
. owner = THIS_MODULE ,
2005-04-16 15:20:36 -07:00
. open = uhci_debug_open ,
. llseek = uhci_debug_lseek ,
. read = uhci_debug_read ,
. release = uhci_debug_release ,
} ;
2005-12-17 18:03:37 -05:00
# endif /* CONFIG_DEBUG_FS */
2005-04-16 15:20:36 -07:00
2005-12-17 18:03:37 -05:00
# else /* DEBUG */
static inline void lprintk ( char * buf )
{ }
2007-03-19 15:31:42 -04:00
static inline int uhci_show_qh ( struct uhci_hcd * uhci ,
struct uhci_qh * qh , char * buf , int len , int space )
2005-12-17 18:03:37 -05:00
{
return 0 ;
}
static inline int uhci_sprint_schedule ( struct uhci_hcd * uhci ,
char * buf , int len )
{
return 0 ;
}
2005-04-16 15:20:36 -07:00
# endif