2005-04-16 15:20:36 -07:00
/*
2006-12-10 21:21:31 +01:00
i2c - dev . c - i2c - bus driver , char device interface
2005-04-16 15:20:36 -07:00
Copyright ( C ) 1995 - 97 Simon G . Vogl
Copyright ( C ) 1998 - 99 Frodo Looijaard < frodol @ dds . nl >
Copyright ( C ) 2003 Greg Kroah - Hartman < greg @ kroah . com >
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
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 .
*/
/* Note that this is a complete rewrite of Simon Vogl's i2c-dev module.
But I have used so much of his original code and ideas that it seems
only fair to recognize him as co - author - - Frodo */
/* The I2C_RDWR ioctl code is written by Kolja Waschk <waschk@telos.de> */
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/fs.h>
# include <linux/slab.h>
# include <linux/init.h>
2006-07-01 17:17:38 +02:00
# include <linux/list.h>
2005-04-16 15:20:36 -07:00
# include <linux/i2c.h>
# include <linux/i2c-dev.h>
# include <asm/uaccess.h>
2006-07-01 17:20:57 +02:00
static struct i2c_driver i2cdev_driver ;
2005-04-16 15:20:36 -07:00
2007-11-15 19:24:01 +01:00
/*
* An i2c_dev represents an i2c_adapter . . . an I2C or SMBus master , not a
* slave ( i2c_client ) with which messages will be exchanged . It ' s coupled
* with a character special file which is accessed by user mode drivers .
*
* The list of i2c_dev structures is parallel to the i2c_adapter lists
* maintained by the driver model , and is updated using notifications
* delivered to the i2cdev_driver .
*/
2005-04-16 15:20:36 -07:00
struct i2c_dev {
2006-07-01 17:17:38 +02:00
struct list_head list ;
2005-04-16 15:20:36 -07:00
struct i2c_adapter * adap ;
2006-07-03 13:46:24 -07:00
struct device * dev ;
2005-04-16 15:20:36 -07:00
} ;
# define I2C_MINORS 256
2006-07-01 17:17:38 +02:00
static LIST_HEAD ( i2c_dev_list ) ;
static DEFINE_SPINLOCK ( i2c_dev_list_lock ) ;
2005-04-16 15:20:36 -07:00
static struct i2c_dev * i2c_dev_get_by_minor ( unsigned index )
{
struct i2c_dev * i2c_dev ;
2006-07-01 17:17:38 +02:00
spin_lock ( & i2c_dev_list_lock ) ;
list_for_each_entry ( i2c_dev , & i2c_dev_list , list ) {
if ( i2c_dev - > adap - > nr = = index )
goto found ;
}
i2c_dev = NULL ;
found :
spin_unlock ( & i2c_dev_list_lock ) ;
2005-04-16 15:20:36 -07:00
return i2c_dev ;
}
static struct i2c_dev * get_free_i2c_dev ( struct i2c_adapter * adap )
{
struct i2c_dev * i2c_dev ;
2006-07-01 17:17:38 +02:00
if ( adap - > nr > = I2C_MINORS ) {
printk ( KERN_ERR " i2c-dev: Out of device minors (%d) \n " ,
adap - > nr ) ;
return ERR_PTR ( - ENODEV ) ;
}
2005-10-17 23:09:43 +02:00
i2c_dev = kzalloc ( sizeof ( * i2c_dev ) , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( ! i2c_dev )
return ERR_PTR ( - ENOMEM ) ;
2006-07-01 17:16:57 +02:00
i2c_dev - > adap = adap ;
2006-07-01 17:17:38 +02:00
spin_lock ( & i2c_dev_list_lock ) ;
list_add_tail ( & i2c_dev - > list , & i2c_dev_list ) ;
spin_unlock ( & i2c_dev_list_lock ) ;
2005-04-16 15:20:36 -07:00
return i2c_dev ;
}
static void return_i2c_dev ( struct i2c_dev * i2c_dev )
{
2006-07-01 17:17:38 +02:00
spin_lock ( & i2c_dev_list_lock ) ;
list_del ( & i2c_dev - > list ) ;
spin_unlock ( & i2c_dev_list_lock ) ;
2006-12-10 21:21:33 +01:00
kfree ( i2c_dev ) ;
2005-04-16 15:20:36 -07:00
}
2006-07-03 13:46:24 -07:00
static ssize_t show_adapter_name ( struct device * dev ,
struct device_attribute * attr , char * buf )
2005-04-16 15:20:36 -07:00
{
2006-07-03 13:46:24 -07:00
struct i2c_dev * i2c_dev = i2c_dev_get_by_minor ( MINOR ( dev - > devt ) ) ;
2005-12-06 15:33:15 -08:00
if ( ! i2c_dev )
return - ENODEV ;
2005-04-16 15:20:36 -07:00
return sprintf ( buf , " %s \n " , i2c_dev - > adap - > name ) ;
}
2006-07-03 13:46:24 -07:00
static DEVICE_ATTR ( name , S_IRUGO , show_adapter_name , NULL ) ;
2005-04-16 15:20:36 -07:00
2007-11-15 19:24:01 +01:00
/* ------------------------------------------------------------------------- */
/*
* After opening an instance of this character special file , a file
* descriptor starts out associated only with an i2c_adapter ( and bus ) .
*
* Using the I2C_RDWR ioctl ( ) , you can then * immediately * issue i2c_msg
* traffic to any devices on the bus used by that adapter . That ' s because
* the i2c_msg vectors embed all the addressing information they need , and
* are submitted directly to an i2c_adapter . However , SMBus - only adapters
* don ' t support that interface .
*
* To use read ( ) / write ( ) system calls on that file descriptor , or to use
* SMBus interfaces ( and work with SMBus - only hosts ! ) , you must first issue
* an I2C_SLAVE ( or I2C_SLAVE_FORCE ) ioctl . That configures an anonymous
* ( never registered ) i2c_client so it holds the addressing information
* needed by those system calls and by this SMBus interface .
*/
2005-04-16 15:20:36 -07:00
static ssize_t i2cdev_read ( struct file * file , char __user * buf , size_t count ,
loff_t * offset )
{
char * tmp ;
int ret ;
struct i2c_client * client = ( struct i2c_client * ) file - > private_data ;
if ( count > 8192 )
count = 8192 ;
tmp = kmalloc ( count , GFP_KERNEL ) ;
if ( tmp = = NULL )
return - ENOMEM ;
pr_debug ( " i2c-dev: i2c-%d reading %zd bytes. \n " ,
2006-12-08 02:37:08 -08:00
iminor ( file - > f_path . dentry - > d_inode ) , count ) ;
2005-04-16 15:20:36 -07:00
ret = i2c_master_recv ( client , tmp , count ) ;
if ( ret > = 0 )
ret = copy_to_user ( buf , tmp , count ) ? - EFAULT : ret ;
kfree ( tmp ) ;
return ret ;
}
static ssize_t i2cdev_write ( struct file * file , const char __user * buf , size_t count ,
loff_t * offset )
{
int ret ;
char * tmp ;
struct i2c_client * client = ( struct i2c_client * ) file - > private_data ;
if ( count > 8192 )
count = 8192 ;
tmp = kmalloc ( count , GFP_KERNEL ) ;
if ( tmp = = NULL )
return - ENOMEM ;
if ( copy_from_user ( tmp , buf , count ) ) {
kfree ( tmp ) ;
return - EFAULT ;
}
pr_debug ( " i2c-dev: i2c-%d writing %zd bytes. \n " ,
2006-12-08 02:37:08 -08:00
iminor ( file - > f_path . dentry - > d_inode ) , count ) ;
2005-04-16 15:20:36 -07:00
ret = i2c_master_send ( client , tmp , count ) ;
kfree ( tmp ) ;
return ret ;
}
static int i2cdev_ioctl ( struct inode * inode , struct file * file ,
unsigned int cmd , unsigned long arg )
{
struct i2c_client * client = ( struct i2c_client * ) file - > private_data ;
struct i2c_rdwr_ioctl_data rdwr_arg ;
struct i2c_smbus_ioctl_data data_arg ;
union i2c_smbus_data temp ;
struct i2c_msg * rdwr_pa ;
u8 __user * * data_ptrs ;
int i , datasize , res ;
unsigned long funcs ;
2005-10-07 23:06:27 +02:00
dev_dbg ( & client - > adapter - > dev , " ioctl, cmd=0x%02x, arg=0x%02lx \n " ,
cmd , arg ) ;
2005-04-16 15:20:36 -07:00
switch ( cmd ) {
case I2C_SLAVE :
case I2C_SLAVE_FORCE :
2007-11-15 19:24:01 +01:00
/* NOTE: devices set up to work with "new style" drivers
* can ' t use I2C_SLAVE , even when the device node is not
* bound to a driver . Only I2C_SLAVE_FORCE will work .
*
* Setting the PEC flag here won ' t affect kernel drivers ,
* which will be using the i2c_client node registered with
* the driver model core . Likewise , when that client has
* the PEC flag already set , the i2c - dev driver won ' t see
* ( or use ) this setting .
*/
2006-12-10 21:21:31 +01:00
if ( ( arg > 0x3ff ) | |
2005-04-16 15:20:36 -07:00
( ( ( client - > flags & I2C_M_TEN ) = = 0 ) & & arg > 0x7f ) )
return - EINVAL ;
if ( ( cmd = = I2C_SLAVE ) & & i2c_check_addr ( client - > adapter , arg ) )
return - EBUSY ;
client - > addr = arg ;
return 0 ;
case I2C_TENBIT :
if ( arg )
client - > flags | = I2C_M_TEN ;
else
client - > flags & = ~ I2C_M_TEN ;
return 0 ;
case I2C_PEC :
if ( arg )
client - > flags | = I2C_CLIENT_PEC ;
else
client - > flags & = ~ I2C_CLIENT_PEC ;
return 0 ;
case I2C_FUNCS :
funcs = i2c_get_functionality ( client - > adapter ) ;
2006-12-10 21:21:30 +01:00
return put_user ( funcs , ( unsigned long __user * ) arg ) ;
2005-04-16 15:20:36 -07:00
case I2C_RDWR :
2006-12-10 21:21:31 +01:00
if ( copy_from_user ( & rdwr_arg ,
( struct i2c_rdwr_ioctl_data __user * ) arg ,
2005-04-16 15:20:36 -07:00
sizeof ( rdwr_arg ) ) )
return - EFAULT ;
2005-05-19 21:41:47 +02:00
/* Put an arbitrary limit on the number of messages that can
2005-04-16 15:20:36 -07:00
* be sent at once */
if ( rdwr_arg . nmsgs > I2C_RDRW_IOCTL_MAX_MSGS )
return - EINVAL ;
2006-12-10 21:21:31 +01:00
2005-04-16 15:20:36 -07:00
rdwr_pa = ( struct i2c_msg * )
2006-12-10 21:21:31 +01:00
kmalloc ( rdwr_arg . nmsgs * sizeof ( struct i2c_msg ) ,
2005-04-16 15:20:36 -07:00
GFP_KERNEL ) ;
if ( rdwr_pa = = NULL ) return - ENOMEM ;
if ( copy_from_user ( rdwr_pa , rdwr_arg . msgs ,
rdwr_arg . nmsgs * sizeof ( struct i2c_msg ) ) ) {
kfree ( rdwr_pa ) ;
return - EFAULT ;
}
data_ptrs = kmalloc ( rdwr_arg . nmsgs * sizeof ( u8 __user * ) , GFP_KERNEL ) ;
if ( data_ptrs = = NULL ) {
kfree ( rdwr_pa ) ;
return - ENOMEM ;
}
res = 0 ;
for ( i = 0 ; i < rdwr_arg . nmsgs ; i + + ) {
2007-10-13 23:56:31 +02:00
/* Limit the size of the message to a sane amount;
* and don ' t let length change either . */
if ( ( rdwr_pa [ i ] . len > 8192 ) | |
( rdwr_pa [ i ] . flags & I2C_M_RECV_LEN ) ) {
2005-04-16 15:20:36 -07:00
res = - EINVAL ;
break ;
}
data_ptrs [ i ] = ( u8 __user * ) rdwr_pa [ i ] . buf ;
rdwr_pa [ i ] . buf = kmalloc ( rdwr_pa [ i ] . len , GFP_KERNEL ) ;
if ( rdwr_pa [ i ] . buf = = NULL ) {
res = - ENOMEM ;
break ;
}
if ( copy_from_user ( rdwr_pa [ i ] . buf ,
data_ptrs [ i ] ,
rdwr_pa [ i ] . len ) ) {
+ + i ; /* Needs to be kfreed too */
res = - EFAULT ;
break ;
}
}
if ( res < 0 ) {
int j ;
for ( j = 0 ; j < i ; + + j )
kfree ( rdwr_pa [ j ] . buf ) ;
kfree ( data_ptrs ) ;
kfree ( rdwr_pa ) ;
return res ;
}
res = i2c_transfer ( client - > adapter ,
rdwr_pa ,
rdwr_arg . nmsgs ) ;
while ( i - - > 0 ) {
if ( res > = 0 & & ( rdwr_pa [ i ] . flags & I2C_M_RD ) ) {
if ( copy_to_user (
data_ptrs [ i ] ,
rdwr_pa [ i ] . buf ,
rdwr_pa [ i ] . len ) ) {
res = - EFAULT ;
}
}
kfree ( rdwr_pa [ i ] . buf ) ;
}
kfree ( data_ptrs ) ;
kfree ( rdwr_pa ) ;
return res ;
case I2C_SMBUS :
if ( copy_from_user ( & data_arg ,
( struct i2c_smbus_ioctl_data __user * ) arg ,
sizeof ( struct i2c_smbus_ioctl_data ) ) )
return - EFAULT ;
2006-12-10 21:21:31 +01:00
if ( ( data_arg . size ! = I2C_SMBUS_BYTE ) & &
2005-04-16 15:20:36 -07:00
( data_arg . size ! = I2C_SMBUS_QUICK ) & &
2006-12-10 21:21:31 +01:00
( data_arg . size ! = I2C_SMBUS_BYTE_DATA ) & &
2005-04-16 15:20:36 -07:00
( data_arg . size ! = I2C_SMBUS_WORD_DATA ) & &
( data_arg . size ! = I2C_SMBUS_PROC_CALL ) & &
( data_arg . size ! = I2C_SMBUS_BLOCK_DATA ) & &
i2c: Fix the i2c_smbus_read_i2c_block_data() prototype
Let the drivers specify how many bytes they want to read with
i2c_smbus_read_i2c_block_data(). So far, the block count was
hard-coded to I2C_SMBUS_BLOCK_MAX (32), which did not make much sense.
Many driver authors complained about this before, and I believe it's
about time to fix it. Right now, authors have to do technically stupid
things, such as individual byte reads or full-fledged I2C messaging,
to work around the problem. We do not want to encourage that.
I even found that some bus drivers (e.g. i2c-amd8111) already
implemented I2C block read the "right" way, that is, they didn't
follow the old, broken standard. The fact that it was never noticed
before just shows how little i2c_smbus_read_i2c_block_data() was used,
which isn't that surprising given how broken its prototype was so far.
There are some obvious compatiblity considerations:
* This changes the i2c_smbus_read_i2c_block_data() prototype. Users
outside the kernel tree will notice at compilation time, and will
have to update their code.
* User-space has access to i2c_smbus_xfer() directly using i2c-dev, so
the changed expectations would affect tools such as i2cdump. In order
to preserve binary compatibility, we give I2C_SMBUS_I2C_BLOCK_DATA
a new numeric value, and define I2C_SMBUS_I2C_BLOCK_BROKEN with the
old numeric value. When i2c-dev receives a transaction with the
old value, it can convert it to the new format on the fly.
Signed-off-by: Jean Delvare <khali@linux-fr.org>
2007-07-12 14:12:29 +02:00
( data_arg . size ! = I2C_SMBUS_I2C_BLOCK_BROKEN ) & &
2005-04-16 15:20:36 -07:00
( data_arg . size ! = I2C_SMBUS_I2C_BLOCK_DATA ) & &
( data_arg . size ! = I2C_SMBUS_BLOCK_PROC_CALL ) ) {
dev_dbg ( & client - > adapter - > dev ,
" size out of range (%x) in ioctl I2C_SMBUS. \n " ,
data_arg . size ) ;
return - EINVAL ;
}
2006-12-10 21:21:31 +01:00
/* Note that I2C_SMBUS_READ and I2C_SMBUS_WRITE are 0 and 1,
2005-04-16 15:20:36 -07:00
so the check is valid if size = = I2C_SMBUS_QUICK too . */
2006-12-10 21:21:31 +01:00
if ( ( data_arg . read_write ! = I2C_SMBUS_READ ) & &
2005-04-16 15:20:36 -07:00
( data_arg . read_write ! = I2C_SMBUS_WRITE ) ) {
2006-12-10 21:21:31 +01:00
dev_dbg ( & client - > adapter - > dev ,
2005-04-16 15:20:36 -07:00
" read_write out of range (%x) in ioctl I2C_SMBUS. \n " ,
data_arg . read_write ) ;
return - EINVAL ;
}
/* Note that command values are always valid! */
if ( ( data_arg . size = = I2C_SMBUS_QUICK ) | |
2006-12-10 21:21:31 +01:00
( ( data_arg . size = = I2C_SMBUS_BYTE ) & &
2005-04-16 15:20:36 -07:00
( data_arg . read_write = = I2C_SMBUS_WRITE ) ) )
/* These are special: we do not use data */
return i2c_smbus_xfer ( client - > adapter , client - > addr ,
client - > flags ,
data_arg . read_write ,
data_arg . command ,
data_arg . size , NULL ) ;
if ( data_arg . data = = NULL ) {
dev_dbg ( & client - > adapter - > dev ,
" data is NULL pointer in ioctl I2C_SMBUS. \n " ) ;
return - EINVAL ;
}
if ( ( data_arg . size = = I2C_SMBUS_BYTE_DATA ) | |
( data_arg . size = = I2C_SMBUS_BYTE ) )
datasize = sizeof ( data_arg . data - > byte ) ;
2006-12-10 21:21:31 +01:00
else if ( ( data_arg . size = = I2C_SMBUS_WORD_DATA ) | |
2005-04-16 15:20:36 -07:00
( data_arg . size = = I2C_SMBUS_PROC_CALL ) )
datasize = sizeof ( data_arg . data - > word ) ;
else /* size == smbus block, i2c block, or block proc. call */
datasize = sizeof ( data_arg . data - > block ) ;
2006-12-10 21:21:31 +01:00
if ( ( data_arg . size = = I2C_SMBUS_PROC_CALL ) | |
( data_arg . size = = I2C_SMBUS_BLOCK_PROC_CALL ) | |
i2c: Fix the i2c_smbus_read_i2c_block_data() prototype
Let the drivers specify how many bytes they want to read with
i2c_smbus_read_i2c_block_data(). So far, the block count was
hard-coded to I2C_SMBUS_BLOCK_MAX (32), which did not make much sense.
Many driver authors complained about this before, and I believe it's
about time to fix it. Right now, authors have to do technically stupid
things, such as individual byte reads or full-fledged I2C messaging,
to work around the problem. We do not want to encourage that.
I even found that some bus drivers (e.g. i2c-amd8111) already
implemented I2C block read the "right" way, that is, they didn't
follow the old, broken standard. The fact that it was never noticed
before just shows how little i2c_smbus_read_i2c_block_data() was used,
which isn't that surprising given how broken its prototype was so far.
There are some obvious compatiblity considerations:
* This changes the i2c_smbus_read_i2c_block_data() prototype. Users
outside the kernel tree will notice at compilation time, and will
have to update their code.
* User-space has access to i2c_smbus_xfer() directly using i2c-dev, so
the changed expectations would affect tools such as i2cdump. In order
to preserve binary compatibility, we give I2C_SMBUS_I2C_BLOCK_DATA
a new numeric value, and define I2C_SMBUS_I2C_BLOCK_BROKEN with the
old numeric value. When i2c-dev receives a transaction with the
old value, it can convert it to the new format on the fly.
Signed-off-by: Jean Delvare <khali@linux-fr.org>
2007-07-12 14:12:29 +02:00
( data_arg . size = = I2C_SMBUS_I2C_BLOCK_DATA ) | |
2005-04-16 15:20:36 -07:00
( data_arg . read_write = = I2C_SMBUS_WRITE ) ) {
if ( copy_from_user ( & temp , data_arg . data , datasize ) )
return - EFAULT ;
}
i2c: Fix the i2c_smbus_read_i2c_block_data() prototype
Let the drivers specify how many bytes they want to read with
i2c_smbus_read_i2c_block_data(). So far, the block count was
hard-coded to I2C_SMBUS_BLOCK_MAX (32), which did not make much sense.
Many driver authors complained about this before, and I believe it's
about time to fix it. Right now, authors have to do technically stupid
things, such as individual byte reads or full-fledged I2C messaging,
to work around the problem. We do not want to encourage that.
I even found that some bus drivers (e.g. i2c-amd8111) already
implemented I2C block read the "right" way, that is, they didn't
follow the old, broken standard. The fact that it was never noticed
before just shows how little i2c_smbus_read_i2c_block_data() was used,
which isn't that surprising given how broken its prototype was so far.
There are some obvious compatiblity considerations:
* This changes the i2c_smbus_read_i2c_block_data() prototype. Users
outside the kernel tree will notice at compilation time, and will
have to update their code.
* User-space has access to i2c_smbus_xfer() directly using i2c-dev, so
the changed expectations would affect tools such as i2cdump. In order
to preserve binary compatibility, we give I2C_SMBUS_I2C_BLOCK_DATA
a new numeric value, and define I2C_SMBUS_I2C_BLOCK_BROKEN with the
old numeric value. When i2c-dev receives a transaction with the
old value, it can convert it to the new format on the fly.
Signed-off-by: Jean Delvare <khali@linux-fr.org>
2007-07-12 14:12:29 +02:00
if ( data_arg . size = = I2C_SMBUS_I2C_BLOCK_BROKEN ) {
/* Convert old I2C block commands to the new
convention . This preserves binary compatibility . */
data_arg . size = I2C_SMBUS_I2C_BLOCK_DATA ;
if ( data_arg . read_write = = I2C_SMBUS_READ )
temp . block [ 0 ] = I2C_SMBUS_BLOCK_MAX ;
}
2005-04-16 15:20:36 -07:00
res = i2c_smbus_xfer ( client - > adapter , client - > addr , client - > flags ,
data_arg . read_write ,
data_arg . command , data_arg . size , & temp ) ;
2006-12-10 21:21:31 +01:00
if ( ! res & & ( ( data_arg . size = = I2C_SMBUS_PROC_CALL ) | |
( data_arg . size = = I2C_SMBUS_BLOCK_PROC_CALL ) | |
2005-04-16 15:20:36 -07:00
( data_arg . read_write = = I2C_SMBUS_READ ) ) ) {
if ( copy_to_user ( data_arg . data , & temp , datasize ) )
return - EFAULT ;
}
return res ;
2007-10-13 23:56:32 +02:00
case I2C_RETRIES :
client - > adapter - > retries = arg ;
break ;
case I2C_TIMEOUT :
client - > adapter - > timeout = arg ;
break ;
2005-04-16 15:20:36 -07:00
default :
2007-10-13 23:56:32 +02:00
/* NOTE: returning a fault code here could cause trouble
* in buggy userspace code . Some old kernel bugs returned
* zero in this case , and userspace code might accidentally
* have depended on that bug .
*/
return - ENOTTY ;
2005-04-16 15:20:36 -07:00
}
return 0 ;
}
static int i2cdev_open ( struct inode * inode , struct file * file )
{
unsigned int minor = iminor ( inode ) ;
struct i2c_client * client ;
struct i2c_adapter * adap ;
struct i2c_dev * i2c_dev ;
i2c_dev = i2c_dev_get_by_minor ( minor ) ;
if ( ! i2c_dev )
return - ENODEV ;
adap = i2c_get_adapter ( i2c_dev - > adap - > nr ) ;
if ( ! adap )
return - ENODEV ;
2007-11-15 19:24:01 +01:00
/* This creates an anonymous i2c_client, which may later be
* pointed to some address using I2C_SLAVE or I2C_SLAVE_FORCE .
*
* This client is * * NEVER REGISTERED * * with the driver model
* or I2C core code ! ! It just holds private copies of addressing
* information and maybe a PEC flag .
*/
2006-07-01 17:20:57 +02:00
client = kzalloc ( sizeof ( * client ) , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( ! client ) {
i2c_put_adapter ( adap ) ;
return - ENOMEM ;
}
2006-07-01 17:20:57 +02:00
snprintf ( client - > name , I2C_NAME_SIZE , " i2c-dev %d " , adap - > nr ) ;
client - > driver = & i2cdev_driver ;
2005-04-16 15:20:36 -07:00
client - > adapter = adap ;
file - > private_data = client ;
return 0 ;
}
static int i2cdev_release ( struct inode * inode , struct file * file )
{
struct i2c_client * client = file - > private_data ;
i2c_put_adapter ( client - > adapter ) ;
kfree ( client ) ;
file - > private_data = NULL ;
return 0 ;
}
2007-02-12 00:55:32 -08:00
static const struct file_operations i2cdev_fops = {
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. read = i2cdev_read ,
. write = i2cdev_write ,
. ioctl = i2cdev_ioctl ,
. open = i2cdev_open ,
. release = i2cdev_release ,
} ;
2007-11-15 19:24:01 +01:00
/* ------------------------------------------------------------------------- */
/*
* The legacy " i2cdev_driver " is used primarily to get notifications when
* I2C adapters are added or removed , so that each one gets an i2c_dev
* and is thus made available to userspace driver code .
*/
2005-12-06 15:33:15 -08:00
static struct class * i2c_dev_class ;
2005-04-16 15:20:36 -07:00
static int i2cdev_attach_adapter ( struct i2c_adapter * adap )
{
struct i2c_dev * i2c_dev ;
2006-08-15 18:30:24 +02:00
int res ;
2005-04-16 15:20:36 -07:00
i2c_dev = get_free_i2c_dev ( adap ) ;
if ( IS_ERR ( i2c_dev ) )
return PTR_ERR ( i2c_dev ) ;
/* register this i2c device with the driver core */
2006-07-03 13:46:24 -07:00
i2c_dev - > dev = device_create ( i2c_dev_class , & adap - > dev ,
MKDEV ( I2C_MAJOR , adap - > nr ) ,
" i2c-%d " , adap - > nr ) ;
2006-12-10 21:21:33 +01:00
if ( IS_ERR ( i2c_dev - > dev ) ) {
res = PTR_ERR ( i2c_dev - > dev ) ;
2005-04-16 15:20:36 -07:00
goto error ;
2006-08-15 18:30:24 +02:00
}
2006-07-03 13:46:24 -07:00
res = device_create_file ( i2c_dev - > dev , & dev_attr_name ) ;
2006-08-15 18:30:24 +02:00
if ( res )
goto error_destroy ;
2006-09-03 22:19:25 +02:00
pr_debug ( " i2c-dev: adapter [%s] registered as minor %d \n " ,
adap - > name , adap - > nr ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
2006-08-15 18:30:24 +02:00
error_destroy :
2006-07-03 13:46:24 -07:00
device_destroy ( i2c_dev_class , MKDEV ( I2C_MAJOR , adap - > nr ) ) ;
2005-04-16 15:20:36 -07:00
error :
return_i2c_dev ( i2c_dev ) ;
2006-08-15 18:30:24 +02:00
return res ;
2005-04-16 15:20:36 -07:00
}
static int i2cdev_detach_adapter ( struct i2c_adapter * adap )
{
struct i2c_dev * i2c_dev ;
2006-07-01 17:16:57 +02:00
i2c_dev = i2c_dev_get_by_minor ( adap - > nr ) ;
2006-09-03 22:19:25 +02:00
if ( ! i2c_dev ) /* attach_adapter must have failed */
return 0 ;
2005-04-16 15:20:36 -07:00
2006-07-03 13:46:24 -07:00
device_remove_file ( i2c_dev - > dev , & dev_attr_name ) ;
2005-04-16 15:20:36 -07:00
return_i2c_dev ( i2c_dev ) ;
2006-07-03 13:46:24 -07:00
device_destroy ( i2c_dev_class , MKDEV ( I2C_MAJOR , adap - > nr ) ) ;
2005-04-16 15:20:36 -07:00
2005-07-31 19:02:53 +02:00
pr_debug ( " i2c-dev: adapter [%s] unregistered \n " , adap - > name ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int i2cdev_detach_client ( struct i2c_client * client )
{
return 0 ;
}
static struct i2c_driver i2cdev_driver = {
2005-11-26 20:34:05 +01:00
. driver = {
. name = " dev_driver " ,
} ,
2005-04-16 15:20:36 -07:00
. id = I2C_DRIVERID_I2CDEV ,
. attach_adapter = i2cdev_attach_adapter ,
. detach_adapter = i2cdev_detach_adapter ,
. detach_client = i2cdev_detach_client ,
} ;
2007-11-15 19:24:01 +01:00
/* ------------------------------------------------------------------------- */
/*
* module load / unload record keeping
*/
2005-04-16 15:20:36 -07:00
static int __init i2c_dev_init ( void )
{
int res ;
printk ( KERN_INFO " i2c /dev entries driver \n " ) ;
res = register_chrdev ( I2C_MAJOR , " i2c " , & i2cdev_fops ) ;
if ( res )
goto out ;
2005-12-06 15:33:15 -08:00
i2c_dev_class = class_create ( THIS_MODULE , " i2c-dev " ) ;
if ( IS_ERR ( i2c_dev_class ) )
2005-04-16 15:20:36 -07:00
goto out_unreg_chrdev ;
res = i2c_add_driver ( & i2cdev_driver ) ;
if ( res )
goto out_unreg_class ;
return 0 ;
out_unreg_class :
2005-12-06 15:33:15 -08:00
class_destroy ( i2c_dev_class ) ;
2005-04-16 15:20:36 -07:00
out_unreg_chrdev :
unregister_chrdev ( I2C_MAJOR , " i2c " ) ;
out :
printk ( KERN_ERR " %s: Driver Initialisation failed \n " , __FILE__ ) ;
return res ;
}
static void __exit i2c_dev_exit ( void )
{
i2c_del_driver ( & i2cdev_driver ) ;
2005-12-06 15:33:15 -08:00
class_destroy ( i2c_dev_class ) ;
2005-04-16 15:20:36 -07:00
unregister_chrdev ( I2C_MAJOR , " i2c " ) ;
}
MODULE_AUTHOR ( " Frodo Looijaard <frodol@dds.nl> and "
" Simon G. Vogl <simon@tk.uni-linz.ac.at> " ) ;
MODULE_DESCRIPTION ( " I2C /dev entries driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( i2c_dev_init ) ;
module_exit ( i2c_dev_exit ) ;