2012-05-11 18:25:46 +04:00
# include <linux/kernel.h>
2013-03-30 04:46:17 +04:00
# include <linux/device.h>
# include <linux/types.h>
# include <linux/spinlock.h>
2013-03-30 14:53:51 +04:00
# include <linux/debugfs.h>
# include <linux/seq_file.h>
# include <linux/uaccess.h>
2012-05-11 18:25:46 +04:00
# include <linux/usb/ch9.h>
# include <linux/usb/gadget.h>
# include "ci.h"
# include "udc.h"
# include "bits.h"
# include "debug.h"
/**
2013-03-30 14:53:51 +04:00
* ci_device_show : prints information about device capabilities and status
2012-05-11 18:25:46 +04:00
*/
2013-03-30 14:53:51 +04:00
static int ci_device_show ( struct seq_file * s , void * data )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
struct ci13xxx * ci = s - > private ;
struct usb_gadget * gadget = & ci - > gadget ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
seq_printf ( s , " speed = %d \n " , gadget - > speed ) ;
seq_printf ( s , " max_speed = %d \n " , gadget - > max_speed ) ;
seq_printf ( s , " is_otg = %d \n " , gadget - > is_otg ) ;
seq_printf ( s , " is_a_peripheral = %d \n " , gadget - > is_a_peripheral ) ;
seq_printf ( s , " b_hnp_enable = %d \n " , gadget - > b_hnp_enable ) ;
seq_printf ( s , " a_hnp_support = %d \n " , gadget - > a_hnp_support ) ;
seq_printf ( s , " a_alt_hnp_support = %d \n " , gadget - > a_alt_hnp_support ) ;
seq_printf ( s , " name = %s \n " ,
( gadget - > name ? gadget - > name : " " ) ) ;
if ( ! ci - > driver )
return 0 ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
seq_printf ( s , " gadget function = %s \n " ,
( ci - > driver - > function ? ci - > driver - > function : " " ) ) ;
seq_printf ( s , " gadget max speed = %d \n " , ci - > driver - > max_speed ) ;
2012-05-11 18:25:46 +04:00
return 0 ;
}
2013-03-30 14:53:51 +04:00
static int ci_device_open ( struct inode * inode , struct file * file )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
return single_open ( file , ci_device_show , inode - > i_private ) ;
2012-05-11 18:25:46 +04:00
}
2013-03-30 14:53:51 +04:00
static const struct file_operations ci_device_fops = {
. open = ci_device_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
2012-05-11 18:25:46 +04:00
/**
2013-03-30 14:53:51 +04:00
* ci_port_test_show : reads port test mode
2012-05-11 18:25:46 +04:00
*/
2013-03-30 14:53:51 +04:00
static int ci_port_test_show ( struct seq_file * s , void * data )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
struct ci13xxx * ci = s - > private ;
2012-05-11 18:25:46 +04:00
unsigned long flags ;
unsigned mode ;
2012-07-07 18:56:40 +04:00
spin_lock_irqsave ( & ci - > lock , flags ) ;
mode = hw_port_test_get ( ci ) ;
spin_unlock_irqrestore ( & ci - > lock , flags ) ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
seq_printf ( s , " mode = %u \n " , mode ) ;
return 0 ;
2012-05-11 18:25:46 +04:00
}
/**
2013-03-30 14:53:51 +04:00
* ci_port_test_write : writes port test mode
2012-05-11 18:25:46 +04:00
*/
2013-03-30 14:53:51 +04:00
static ssize_t ci_port_test_write ( struct file * file , const char __user * ubuf ,
size_t count , loff_t * ppos )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
struct seq_file * s = file - > private_data ;
struct ci13xxx * ci = s - > private ;
2012-05-11 18:25:46 +04:00
unsigned long flags ;
unsigned mode ;
2013-03-30 14:53:51 +04:00
char buf [ 32 ] ;
int ret ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
if ( copy_from_user ( buf , ubuf , min_t ( size_t , sizeof ( buf ) - 1 , count ) ) )
return - EFAULT ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
if ( sscanf ( buf , " %u " , & mode ) ! = 1 )
return - EINVAL ;
2012-05-11 18:25:46 +04:00
2012-07-07 18:56:40 +04:00
spin_lock_irqsave ( & ci - > lock , flags ) ;
2013-03-30 14:53:51 +04:00
ret = hw_port_test_set ( ci , mode ) ;
2012-07-07 18:56:40 +04:00
spin_unlock_irqrestore ( & ci - > lock , flags ) ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
return ret ? ret : count ;
2012-05-11 18:25:46 +04:00
}
2013-03-30 14:53:51 +04:00
static int ci_port_test_open ( struct inode * inode , struct file * file )
{
return single_open ( file , ci_port_test_show , inode - > i_private ) ;
}
static const struct file_operations ci_port_test_fops = {
. open = ci_port_test_open ,
. write = ci_port_test_write ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
2012-05-11 18:25:46 +04:00
/**
2013-03-30 14:53:51 +04:00
* ci_qheads_show : DMA contents of all queue heads
2012-05-11 18:25:46 +04:00
*/
2013-03-30 14:53:51 +04:00
static int ci_qheads_show ( struct seq_file * s , void * data )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
struct ci13xxx * ci = s - > private ;
2012-05-11 18:25:46 +04:00
unsigned long flags ;
2013-03-30 14:53:51 +04:00
unsigned i , j ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
if ( ci - > role ! = CI_ROLE_GADGET ) {
seq_printf ( s , " not in gadget mode \n " ) ;
2012-05-11 18:25:46 +04:00
return 0 ;
}
2012-07-07 18:56:40 +04:00
spin_lock_irqsave ( & ci - > lock , flags ) ;
for ( i = 0 ; i < ci - > hw_ep_max / 2 ; i + + ) {
struct ci13xxx_ep * mEpRx = & ci - > ci13xxx_ep [ i ] ;
2012-05-11 18:25:46 +04:00
struct ci13xxx_ep * mEpTx =
2012-07-07 18:56:40 +04:00
& ci - > ci13xxx_ep [ i + ci - > hw_ep_max / 2 ] ;
2013-03-30 14:53:51 +04:00
seq_printf ( s , " EP=%02i: RX=%08X TX=%08X \n " ,
i , ( u32 ) mEpRx - > qh . dma , ( u32 ) mEpTx - > qh . dma ) ;
for ( j = 0 ; j < ( sizeof ( struct ci13xxx_qh ) / sizeof ( u32 ) ) ; j + + )
seq_printf ( s , " %04X: %08X %08X \n " , j ,
* ( ( u32 * ) mEpRx - > qh . ptr + j ) ,
* ( ( u32 * ) mEpTx - > qh . ptr + j ) ) ;
2012-05-11 18:25:46 +04:00
}
2012-07-07 18:56:40 +04:00
spin_unlock_irqrestore ( & ci - > lock , flags ) ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
return 0 ;
2012-05-11 18:25:46 +04:00
}
2013-03-30 14:53:51 +04:00
static int ci_qheads_open ( struct inode * inode , struct file * file )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
return single_open ( file , ci_qheads_show , inode - > i_private ) ;
2012-05-11 18:25:46 +04:00
}
2013-03-30 14:53:51 +04:00
static const struct file_operations ci_qheads_fops = {
. open = ci_qheads_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
2012-05-11 18:25:46 +04:00
/**
2013-03-30 14:53:51 +04:00
* ci_requests_show : DMA contents of all requests currently queued ( all endpts )
2012-05-11 18:25:46 +04:00
*/
2013-03-30 14:53:51 +04:00
static int ci_requests_show ( struct seq_file * s , void * data )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
struct ci13xxx * ci = s - > private ;
2012-05-11 18:25:46 +04:00
unsigned long flags ;
struct list_head * ptr = NULL ;
struct ci13xxx_req * req = NULL ;
2013-03-30 14:53:51 +04:00
unsigned i , j , qsize = sizeof ( struct ci13xxx_td ) / sizeof ( u32 ) ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
if ( ci - > role ! = CI_ROLE_GADGET ) {
seq_printf ( s , " not in gadget mode \n " ) ;
2012-05-11 18:25:46 +04:00
return 0 ;
}
2012-07-07 18:56:40 +04:00
spin_lock_irqsave ( & ci - > lock , flags ) ;
for ( i = 0 ; i < ci - > hw_ep_max ; i + + )
2013-03-30 14:53:51 +04:00
list_for_each ( ptr , & ci - > ci13xxx_ep [ i ] . qh . queue ) {
2012-05-11 18:25:46 +04:00
req = list_entry ( ptr , struct ci13xxx_req , queue ) ;
2013-03-30 14:53:51 +04:00
seq_printf ( s , " EP=%02i: TD=%08X %s \n " ,
2013-03-30 14:53:54 +04:00
i % ( ci - > hw_ep_max / 2 ) , ( u32 ) req - > dma ,
2013-03-30 14:53:51 +04:00
( ( i < ci - > hw_ep_max / 2 ) ? " RX " : " TX " ) ) ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
for ( j = 0 ; j < qsize ; j + + )
seq_printf ( s , " %04X: %08X \n " , j ,
* ( ( u32 * ) req - > ptr + j ) ) ;
2012-05-11 18:25:46 +04:00
}
2012-07-07 18:56:40 +04:00
spin_unlock_irqrestore ( & ci - > lock , flags ) ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:51 +04:00
return 0 ;
}
static int ci_requests_open ( struct inode * inode , struct file * file )
{
return single_open ( file , ci_requests_show , inode - > i_private ) ;
2012-05-11 18:25:46 +04:00
}
2013-03-30 14:53:51 +04:00
static const struct file_operations ci_requests_fops = {
. open = ci_requests_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
2012-05-11 18:25:46 +04:00
2013-03-30 14:53:52 +04:00
static int ci_role_show ( struct seq_file * s , void * data )
{
struct ci13xxx * ci = s - > private ;
seq_printf ( s , " %s \n " , ci_role ( ci ) - > name ) ;
return 0 ;
}
static ssize_t ci_role_write ( struct file * file , const char __user * ubuf ,
size_t count , loff_t * ppos )
{
struct seq_file * s = file - > private_data ;
struct ci13xxx * ci = s - > private ;
enum ci_role role ;
char buf [ 8 ] ;
int ret ;
if ( copy_from_user ( buf , ubuf , min_t ( size_t , sizeof ( buf ) - 1 , count ) ) )
return - EFAULT ;
for ( role = CI_ROLE_HOST ; role < CI_ROLE_END ; role + + )
if ( ci - > roles [ role ] & &
! strncmp ( buf , ci - > roles [ role ] - > name ,
strlen ( ci - > roles [ role ] - > name ) ) )
break ;
if ( role = = CI_ROLE_END | | role = = ci - > role )
return - EINVAL ;
ci_role_stop ( ci ) ;
ret = ci_role_start ( ci , role ) ;
return ret ? ret : count ;
}
static int ci_role_open ( struct inode * inode , struct file * file )
{
return single_open ( file , ci_role_show , inode - > i_private ) ;
}
static const struct file_operations ci_role_fops = {
. open = ci_role_open ,
. write = ci_role_write ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
2012-05-11 18:25:46 +04:00
/**
* dbg_create_files : initializes the attribute interface
2013-03-30 14:53:51 +04:00
* @ ci : device
2012-05-11 18:25:46 +04:00
*
* This function returns an error code
*/
2013-03-30 14:53:51 +04:00
int dbg_create_files ( struct ci13xxx * ci )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
struct dentry * dent ;
ci - > debugfs = debugfs_create_dir ( dev_name ( ci - > dev ) , NULL ) ;
if ( ! ci - > debugfs )
return - ENOMEM ;
dent = debugfs_create_file ( " device " , S_IRUGO , ci - > debugfs , ci ,
& ci_device_fops ) ;
if ( ! dent )
goto err ;
dent = debugfs_create_file ( " port_test " , S_IRUGO | S_IWUSR , ci - > debugfs ,
ci , & ci_port_test_fops ) ;
if ( ! dent )
goto err ;
dent = debugfs_create_file ( " qheads " , S_IRUGO , ci - > debugfs , ci ,
& ci_qheads_fops ) ;
if ( ! dent )
goto err ;
dent = debugfs_create_file ( " requests " , S_IRUGO , ci - > debugfs , ci ,
& ci_requests_fops ) ;
2013-03-30 14:53:52 +04:00
if ( ! dent )
goto err ;
dent = debugfs_create_file ( " role " , S_IRUGO | S_IWUSR , ci - > debugfs , ci ,
& ci_role_fops ) ;
2013-03-30 14:53:51 +04:00
if ( dent )
return 0 ;
err :
debugfs_remove_recursive ( ci - > debugfs ) ;
return - ENOMEM ;
2012-05-11 18:25:46 +04:00
}
/**
* dbg_remove_files : destroys the attribute interface
2013-03-30 14:53:51 +04:00
* @ ci : device
2012-05-11 18:25:46 +04:00
*/
2013-03-30 14:53:51 +04:00
void dbg_remove_files ( struct ci13xxx * ci )
2012-05-11 18:25:46 +04:00
{
2013-03-30 14:53:51 +04:00
debugfs_remove_recursive ( ci - > debugfs ) ;
2012-05-11 18:25:46 +04:00
}