2003-07-20 20:48:48 -07:00
/*
* udev - add . c
*
* Userspace devfs
*
* 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 version 2 of the License .
*
* 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 <stdlib.h>
# include <string.h>
# include <stdio.h>
# include <fcntl.h>
# include <unistd.h>
# include <errno.h>
# include "udev.h"
# include "udev_version.h"
# include "namedev.h"
2003-08-05 23:57:23 -07:00
# include "udevdb.h"
2003-07-20 20:48:48 -07:00
# include "libsysfs/libsysfs.h"
/*
* Right now the major / minor of a device is stored in a file called
* " dev " in sysfs .
* The number is stored as :
2003-10-20 20:31:24 -07:00
* MM : mm
2003-07-20 20:48:48 -07:00
* MM is the major
* mm is the minor
2003-10-20 20:31:24 -07:00
* The value is in decimal .
2003-07-20 20:48:48 -07:00
*/
static int get_major_minor ( struct sysfs_class_device * class_dev , int * major , int * minor )
{
2003-08-04 21:20:19 -07:00
int retval = - ENODEV ;
2003-07-20 20:48:48 -07:00
char * dev ;
dev = sysfs_get_value_from_attributes ( class_dev - > directory - > attributes , " dev " ) ;
if ( dev = = NULL )
2003-08-04 21:20:19 -07:00
goto exit ;
2003-07-20 20:48:48 -07:00
dbg ( " dev = %s " , dev ) ;
2003-08-04 21:20:19 -07:00
if ( sscanf ( dev , " %u:%u " , major , minor ) ! = 2 )
goto exit ;
2003-07-20 20:48:48 -07:00
dbg ( " found major = %d, minor = %d " , * major , * minor ) ;
retval = 0 ;
2003-08-04 21:20:19 -07:00
exit :
2003-07-20 20:48:48 -07:00
return retval ;
}
/*
* We also want to add some permissions here , and possibly some symlinks
*/
2003-10-20 22:48:44 -07:00
static int create_node ( struct udevice * dev )
2003-07-20 20:48:48 -07:00
{
char filename [ 255 ] ;
int retval = 0 ;
2003-10-20 22:48:44 -07:00
2003-10-21 20:19:09 -07:00
strncpy ( filename , udev_root , sizeof ( filename ) ) ;
2003-10-20 22:48:44 -07:00
strncat ( filename , dev - > name , sizeof ( filename ) ) ;
switch ( dev - > type ) {
2003-08-04 21:59:50 -07:00
case ' b ' :
2003-10-20 22:48:44 -07:00
dev - > mode | = S_IFBLK ;
2003-08-04 21:59:50 -07:00
break ;
case ' c ' :
case ' u ' :
2003-10-20 22:48:44 -07:00
dev - > mode | = S_IFCHR ;
2003-08-04 21:59:50 -07:00
break ;
case ' p ' :
2003-10-20 22:48:44 -07:00
dev - > mode | = S_IFIFO ;
2003-08-04 21:59:50 -07:00
break ;
default :
2003-10-20 22:48:44 -07:00
dbg ( " unknown node type %c \n " , dev - > type ) ;
2003-08-04 21:59:50 -07:00
return - EINVAL ;
}
2003-08-04 21:39:33 -07:00
2003-10-20 22:48:44 -07:00
dbg ( " mknod(%s, %#o, %u, %u) " , filename , dev - > mode , dev - > major , dev - > minor ) ;
retval = mknod ( filename , dev - > mode , makedev ( dev - > major , dev - > minor ) ) ;
2003-08-04 21:59:50 -07:00
if ( retval )
dbg ( " mknod(%s, %#o, %u, %u) failed with error '%s' " ,
2003-10-20 22:48:44 -07:00
filename , dev - > mode , dev - > major , dev - > minor , strerror ( errno ) ) ;
// FIXME set the ownership of the node
2003-07-20 20:48:48 -07:00
return retval ;
}
2003-10-19 21:55:43 -07:00
static struct sysfs_class_device * get_class_dev ( char * device_name )
2003-07-20 20:48:48 -07:00
{
char dev_path [ SYSFS_PATH_MAX ] ;
2003-08-04 22:13:35 -07:00
struct sysfs_class_device * class_dev = NULL ;
2003-07-20 20:48:48 -07:00
strcpy ( dev_path , sysfs_path ) ;
strcat ( dev_path , device_name ) ;
dbg ( " looking at %s " , dev_path ) ;
/* open up the sysfs class device for this thing... */
class_dev = sysfs_open_class_device ( dev_path ) ;
if ( class_dev = = NULL ) {
dbg ( " sysfs_open_class_device failed " ) ;
2003-08-04 22:13:35 -07:00
goto exit ;
2003-07-20 20:48:48 -07:00
}
dbg ( " class_dev->name = %s " , class_dev - > name ) ;
2003-08-04 22:13:35 -07:00
exit :
2003-07-20 20:48:48 -07:00
return class_dev ;
}
2003-10-20 21:03:15 -07:00
/* wait for the "dev" file to show up in the directory in sysfs.
* If it doesn ' t happen in about 10 seconds , give up .
*/
# define SECONDS_TO_WAIT_FOR_DEV 10
2003-10-21 21:45:19 -07:00
static int sleep_for_dev ( char * path )
2003-10-20 21:03:15 -07:00
{
char filename [ SYSFS_PATH_MAX + 6 ] ;
struct stat buf ;
int loop = 0 ;
int retval = - ENODEV ;
strcpy ( filename , sysfs_path ) ;
2003-10-20 22:48:44 -07:00
strcat ( filename , path ) ;
2003-10-20 21:03:15 -07:00
strcat ( filename , " /dev " ) ;
while ( loop < SECONDS_TO_WAIT_FOR_DEV ) {
dbg ( " looking for %s " , filename ) ;
retval = stat ( filename , & buf ) ;
if ( retval = = 0 ) {
retval = 0 ;
goto exit ;
}
/* sleep for a second or two to give the kernel a chance to
* create the dev file */
sleep ( 1 ) ;
2003-10-21 21:40:53 -07:00
+ + loop ;
2003-10-20 21:03:15 -07:00
}
retval = - ENODEV ;
exit :
return retval ;
}
2003-10-20 22:48:44 -07:00
int udev_add_device ( char * path , char * subsystem )
2003-07-20 20:48:48 -07:00
{
struct sysfs_class_device * class_dev ;
2003-10-20 22:48:44 -07:00
struct udevice dev ;
2003-07-20 20:48:48 -07:00
int retval = - EINVAL ;
/* for now, the block layer is the only place where block devices are */
if ( strcmp ( subsystem , " block " ) = = 0 )
2003-10-20 22:48:44 -07:00
dev . type = ' b ' ;
2003-07-20 20:48:48 -07:00
else
2003-10-20 22:48:44 -07:00
dev . type = ' c ' ;
2003-07-20 20:48:48 -07:00
2003-10-20 22:48:44 -07:00
retval = sleep_for_dev ( path ) ;
2003-10-20 21:03:15 -07:00
if ( retval )
goto exit ;
2003-08-04 22:13:35 -07:00
2003-10-20 22:48:44 -07:00
class_dev = get_class_dev ( path ) ;
2003-07-20 20:48:48 -07:00
if ( class_dev = = NULL )
goto exit ;
2003-10-21 22:28:32 -07:00
retval = namedev_name_device ( class_dev , & dev ) ;
2003-07-20 20:48:48 -07:00
if ( retval )
return retval ;
2003-10-20 22:48:44 -07:00
retval = get_major_minor ( class_dev , & dev . major , & dev . minor ) ;
2003-07-20 20:48:48 -07:00
if ( retval ) {
2003-10-17 23:32:17 -07:00
dbg ( " get_major_minor failed " ) ;
2003-07-20 20:48:48 -07:00
goto exit ;
}
2003-10-21 22:28:32 -07:00
// strcpy(dev.name, attr.name);
// strcpy(dev.owner, attr.owner);
// strcpy(dev.group, attr.group);
// dev.mode = attr.mode;
2003-10-20 22:48:44 -07:00
retval = udevdb_add_dev ( path , & dev ) ;
2003-08-05 23:57:23 -07:00
if ( retval ! = 0 )
2003-10-20 22:48:44 -07:00
dbg ( " udevdb_add_dev failed, but we are going to try to create the node anyway. "
2003-10-17 23:32:17 -07:00
" But remove might not work properly for this device. " ) ;
sysfs_close_class_device ( class_dev ) ;
2003-08-05 23:57:23 -07:00
2003-10-20 22:48:44 -07:00
dbg ( " name = %s " , dev . name ) ;
retval = create_node ( & dev ) ;
2003-07-20 20:48:48 -07:00
exit :
return retval ;
}