2003-10-17 12:40:02 +04:00
/*
2004-11-12 08:32:19 +03:00
* udev_db . c
2003-10-17 12:40:02 +04:00
*
* Userspace devfs
*
* Copyright ( C ) 2003 Greg Kroah - Hartman < greg @ kroah . com >
2004-11-06 16:28:01 +03:00
* Copyright ( C ) 2004 Kay Sievers < kay . sievers @ vrfy . org >
2003-10-17 12:40:02 +04:00
*
* 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 .
*
*/
2004-11-06 16:28:01 +03:00
2003-08-06 10:57:23 +04:00
# include <stdlib.h>
# include <stdio.h>
2004-03-02 10:47:59 +03:00
# include <string.h>
2004-03-23 09:22:20 +03:00
# include <stddef.h>
2003-08-06 10:57:23 +04:00
# include <fcntl.h>
# include <string.h>
# include <errno.h>
2004-11-06 16:28:01 +03:00
# include <dirent.h>
2003-08-06 10:57:23 +04:00
2004-02-24 06:07:25 +03:00
# include "libsysfs/sysfs/libsysfs.h"
2003-10-16 10:50:13 +04:00
# include "udev.h"
2004-11-25 04:44:38 +03:00
# include "udev_utils.h"
[PATCH] add udev logging to info log
On Thu, Jan 15, 2004 at 05:14:16AM +0100, Kay Sievers wrote:
> On Wed, Jan 14, 2004 at 01:10:43PM -0800, Greg KH wrote:
> > On Wed, Jan 14, 2004 at 02:34:26PM -0600, Clay Haapala wrote:
> > > On Wed, 14 Jan 2004, Chris Friesen spake thusly:
> > > >
> > > > Maybe for ones with a matching rule, you could print something like:
> > > >
> > > >
> > > Is the act of printing/syslogging a rule in an of itself?
> >
> > No, as currently the only way stuff ends up in the syslog is if
> > DEBUG=true is used on the build line.
> >
> > But it's sounding like we might want to change that... :)
>
> How about this in the syslog after connect/disconnect?
>
> Jan 15 05:07:45 pim udev[28007]: configured rule in '/etc/udev/udev.rules' at line 17 applied, 'video*' becomes 'video/webcam%n'
> Jan 15 05:07:45 pim udev[28007]: creating device node '/udev/video/webcam0'
> Jan 15 05:07:47 pim udev[28015]: removing device node '/udev/video/webcam0'
Here is a slightly better version. I've created a logging.h file and
moved the debug macros from udev.h in there.
If you type:
'make' - you will get a binary that prints one or two lines to syslog
if a device node is created or deleted
'make LOG=false' - you get a binary that prints asolutely nothing
'make DEBUG=true' - the same as today, it will print all debug lines
2004-01-16 08:53:20 +03:00
# include "logging.h"
2004-11-12 08:32:19 +03:00
# include "udev_db.h"
2003-08-06 10:57:23 +04:00
2004-11-06 16:28:01 +03:00
# define PATH_TO_NAME_CHAR '@'
static int get_db_filename ( struct udevice * udev , char * filename , int len )
{
char devpath [ SYSFS_PATH_MAX ] ;
char * pos ;
/* replace '/' to transform path into a filename */
strfieldcpy ( devpath , udev - > devpath ) ;
pos = strchr ( & devpath [ 1 ] , ' / ' ) ;
while ( pos ) {
pos [ 0 ] = PATH_TO_NAME_CHAR ;
pos = strchr ( & pos [ 1 ] , ' / ' ) ;
}
2004-11-12 08:21:16 +03:00
snprintf ( filename , len , " %s%s " , udev_db_path , devpath ) ;
2004-11-06 16:28:01 +03:00
filename [ len - 1 ] = ' \0 ' ;
return 0 ;
}
2003-10-18 10:32:17 +04:00
2004-11-12 08:32:19 +03:00
int udev_db_add_device ( struct udevice * udev )
2003-10-18 10:32:17 +04:00
{
2004-11-06 16:28:01 +03:00
char filename [ SYSFS_PATH_MAX ] ;
FILE * f ;
2003-10-18 10:32:17 +04:00
2004-10-19 06:28:39 +04:00
if ( udev - > test_run )
return 0 ;
2004-11-06 16:28:01 +03:00
get_db_filename ( udev , filename , SYSFS_PATH_MAX ) ;
create_path ( filename ) ;
f = fopen ( filename , " w " ) ;
if ( f = = NULL ) {
dbg ( " unable to create db file '%s' " , filename ) ;
2004-10-14 10:13:26 +04:00
return - 1 ;
2004-11-06 16:28:01 +03:00
}
dbg ( " storing data for device '%s' in '%s' " , udev - > devpath , filename ) ;
2004-10-14 10:13:26 +04:00
2004-11-06 16:28:01 +03:00
fprintf ( f , " P:%s \n " , udev - > devpath ) ;
fprintf ( f , " N:%s \n " , udev - > name ) ;
fprintf ( f , " S:%s \n " , udev - > symlink ) ;
2004-11-13 16:43:24 +03:00
fprintf ( f , " A:%u \n " , udev - > partitions ) ;
fprintf ( f , " R:%u \n " , udev - > ignore_remove ) ;
2003-11-25 09:27:17 +03:00
2004-11-06 16:28:01 +03:00
fclose ( f ) ;
2004-02-13 09:51:44 +03:00
2004-11-06 16:28:01 +03:00
return 0 ;
2003-10-18 10:32:17 +04:00
}
2004-11-06 16:28:01 +03:00
static int parse_db_file ( struct udevice * udev , const char * filename )
2003-08-06 10:57:23 +04:00
{
2004-11-06 16:28:01 +03:00
char line [ NAME_SIZE ] ;
char * bufline ;
char * buf ;
size_t bufsize ;
size_t cur ;
size_t count ;
if ( file_map ( filename , & buf , & bufsize ) ! = 0 ) {
dbg ( " unable to read db file '%s' " , filename ) ;
2004-10-14 10:13:26 +04:00
return - 1 ;
2004-11-06 16:28:01 +03:00
}
2004-10-14 10:13:26 +04:00
2004-11-06 16:28:01 +03:00
cur = 0 ;
while ( cur < bufsize ) {
count = buf_get_line ( buf , bufsize , cur ) ;
bufline = & buf [ cur ] ;
cur + = count + 1 ;
switch ( bufline [ 0 ] ) {
case ' P ' :
if ( count > DEVPATH_SIZE )
count = DEVPATH_SIZE - 1 ;
strncpy ( udev - > devpath , & bufline [ 2 ] , count - 2 ) ;
2004-11-13 16:43:24 +03:00
udev - > devpath [ count - 2 ] = ' \0 ' ;
2004-11-06 16:28:01 +03:00
break ;
case ' N ' :
if ( count > NAME_SIZE )
count = NAME_SIZE - 1 ;
strncpy ( udev - > name , & bufline [ 2 ] , count - 2 ) ;
2004-11-13 16:43:24 +03:00
udev - > name [ count - 2 ] = ' \0 ' ;
2004-11-06 16:28:01 +03:00
break ;
case ' S ' :
if ( count > NAME_SIZE )
count = NAME_SIZE - 1 ;
strncpy ( udev - > symlink , & bufline [ 2 ] , count - 2 ) ;
2004-11-13 16:43:24 +03:00
udev - > symlink [ count - 2 ] = ' \0 ' ;
2004-11-06 16:28:01 +03:00
break ;
case ' A ' :
2004-11-13 16:43:24 +03:00
if ( count > NAME_SIZE )
count = NAME_SIZE - 1 ;
strncpy ( line , & bufline [ 2 ] , count - 2 ) ;
line [ count - 2 ] = ' \0 ' ;
2004-11-06 16:28:01 +03:00
udev - > partitions = atoi ( line ) ;
break ;
2004-11-13 16:43:24 +03:00
case ' R ' :
if ( count > NAME_SIZE )
count = NAME_SIZE - 1 ;
strncpy ( line , & bufline [ 2 ] , count - 2 ) ;
line [ count - 2 ] = ' \0 ' ;
udev - > ignore_remove = atoi ( line ) ;
break ;
2004-11-06 16:28:01 +03:00
}
}
2003-11-25 09:27:17 +03:00
2004-11-06 16:28:01 +03:00
if ( udev - > name [ 0 ] = = ' \0 ' )
return - 1 ;
2004-10-19 06:11:51 +04:00
2003-12-20 05:29:01 +03:00
return 0 ;
2003-08-06 10:57:23 +04:00
}
2004-11-12 08:32:19 +03:00
int udev_db_get_device ( struct udevice * udev )
2003-08-06 10:57:23 +04:00
{
2004-11-06 16:28:01 +03:00
char filename [ SYSFS_PATH_MAX ] ;
2003-10-21 07:28:42 +04:00
2004-11-06 16:28:01 +03:00
get_db_filename ( udev , filename , SYSFS_PATH_MAX ) ;
2003-10-21 07:28:42 +04:00
2004-11-06 16:28:01 +03:00
return parse_db_file ( udev , filename ) ;
2003-10-21 07:28:42 +04:00
}
2004-11-12 08:32:19 +03:00
int udev_db_delete_device ( struct udevice * udev )
2003-10-15 10:20:53 +04:00
{
2004-11-06 16:28:01 +03:00
char filename [ SYSFS_PATH_MAX ] ;
get_db_filename ( udev , filename , SYSFS_PATH_MAX ) ;
unlink ( filename ) ;
2003-10-15 10:20:53 +04:00
2003-10-21 09:48:44 +04:00
return 0 ;
2003-10-15 10:20:53 +04:00
}
2003-12-31 09:31:37 +03:00
2004-11-12 08:32:19 +03:00
int udev_db_get_device_byname ( struct udevice * udev , const char * name )
2003-12-31 09:31:37 +03:00
{
2004-11-06 16:28:01 +03:00
struct dirent * ent ;
DIR * dir ;
char filename [ NAME_SIZE ] ;
struct udevice db_udev ;
dir = opendir ( udev_db_path ) ;
if ( dir = = NULL ) {
dbg ( " unable to udev db '%s' " , udev_db_path ) ;
return - 1 ;
2004-01-13 12:01:19 +03:00
}
2004-11-06 16:28:01 +03:00
while ( 1 ) {
ent = readdir ( dir ) ;
if ( ent = = NULL | | ent - > d_name [ 0 ] = = ' \0 ' )
break ;
2004-01-13 12:01:19 +03:00
2004-11-06 16:28:01 +03:00
if ( ent - > d_name [ 0 ] = = ' . ' )
continue ;
2004-01-13 12:01:19 +03:00
2004-11-12 08:21:16 +03:00
snprintf ( filename , NAME_SIZE , " %s/%s " , udev_db_path , ent - > d_name ) ;
2004-11-06 16:28:01 +03:00
filename [ NAME_SIZE - 1 ] = ' \0 ' ;
2004-01-20 06:40:32 +03:00
2004-11-06 16:28:01 +03:00
memset ( & db_udev , 0x00 , sizeof ( struct udevice ) ) ;
if ( parse_db_file ( & db_udev , filename ) = = 0 ) {
char * pos ;
int len ;
2004-10-14 10:13:26 +04:00
2004-11-06 16:28:01 +03:00
if ( strncmp ( name , db_udev . name , NAME_SIZE ) = = 0 ) {
goto found ;
}
2004-01-20 06:40:32 +03:00
2004-11-06 16:28:01 +03:00
foreach_strpart ( db_udev . symlink , " " , pos , len ) {
if ( strncmp ( name , pos , len ) ! = 0 )
continue ;
2004-01-20 06:40:32 +03:00
2004-11-06 16:28:01 +03:00
if ( len = = strlen ( name ) )
goto found ;
}
}
2004-01-20 06:40:32 +03:00
}
2004-03-04 05:16:35 +03:00
2004-11-06 16:28:01 +03:00
closedir ( dir ) ;
2004-03-04 05:16:35 +03:00
2004-11-06 16:28:01 +03:00
return - 1 ;
2004-01-20 06:40:32 +03:00
2004-11-06 16:28:01 +03:00
found :
closedir ( dir ) ;
strfieldcpy ( udev - > devpath , db_udev . devpath ) ;
strfieldcpy ( udev - > name , db_udev . name ) ;
strfieldcpy ( udev - > symlink , db_udev . symlink ) ;
udev - > partitions = db_udev . partitions ;
return 0 ;
2004-01-20 06:40:32 +03:00
}