2008-08-27 19:11:58 +04:00
/*
* libudev - interface to udev device information
*
2009-08-01 17:39:18 +04:00
* Copyright ( C ) 2008 - 2009 Kay Sievers < kay . sievers @ vrfy . org >
2008-08-27 19:11:58 +04:00
*
2009-03-26 21:29:36 +03:00
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2.1 of the License , or ( at your option ) any later version .
2008-08-27 19:11:58 +04:00
*/
# include <stdio.h>
# include <stdlib.h>
# include <stddef.h>
2008-08-31 01:07:50 +04:00
# include <stdarg.h>
2008-08-27 19:11:58 +04:00
# include <unistd.h>
# include <errno.h>
# include <string.h>
2008-09-06 17:45:31 +04:00
# include <ctype.h>
2008-08-27 19:11:58 +04:00
# include "libudev.h"
# include "libudev-private.h"
2009-06-15 19:09:43 +04:00
/**
* SECTION : libudev
* @ short_description : libudev context
*
* The context contains the default values read from the udev config file ,
* and is passed to all library operations .
*/
2009-06-15 15:22:38 +04:00
/**
* udev :
*
2009-06-15 19:09:43 +04:00
* Opaque object representing the library context .
2009-06-15 15:22:38 +04:00
*/
2008-08-29 01:05:01 +04:00
struct udev {
int refcount ;
void ( * log_fn ) ( struct udev * udev ,
int priority , const char * file , int line , const char * fn ,
const char * format , va_list args ) ;
2008-09-25 08:39:29 +04:00
void * userdata ;
2008-09-06 17:45:31 +04:00
char * sys_path ;
char * dev_path ;
char * rules_path ;
2008-10-16 15:51:29 +04:00
struct udev_list_node properties_list ;
2008-09-06 17:45:31 +04:00
int log_priority ;
2008-09-10 12:09:34 +04:00
int run ;
2008-08-29 01:05:01 +04:00
} ;
2008-08-27 19:11:58 +04:00
void udev_log ( struct udev * udev ,
int priority , const char * file , int line , const char * fn ,
const char * format , . . . )
{
va_list args ;
va_start ( args , format ) ;
udev - > log_fn ( udev , priority , file , line , fn , format , args ) ;
va_end ( args ) ;
}
static void log_stderr ( struct udev * udev ,
int priority , const char * file , int line , const char * fn ,
const char * format , va_list args )
{
2008-09-06 17:45:31 +04:00
fprintf ( stderr , " libudev: %s: " , fn ) ;
vfprintf ( stderr , format , args ) ;
2008-08-27 19:11:58 +04:00
}
2009-06-15 15:22:38 +04:00
/**
* udev_get_userdata :
* @ udev : udev library context
*
* Retrieve stored data pointer from library context . This might be useful
* to access from callbacks .
*
* Returns : stored userdata
* */
2008-09-25 08:39:29 +04:00
void * udev_get_userdata ( struct udev * udev )
{
if ( udev = = NULL )
return NULL ;
return udev - > userdata ;
}
2009-06-15 15:22:38 +04:00
/**
* udev_set_userdata :
* @ udev : udev library context
* @ userdata : data pointer
*
* Store custom @ userdata in the library context .
* */
2008-09-25 08:39:29 +04:00
void udev_set_userdata ( struct udev * udev , void * userdata )
{
if ( udev = = NULL )
return ;
udev - > userdata = userdata ;
}
2008-08-27 19:11:58 +04:00
/**
* udev_new :
*
2009-06-15 15:22:38 +04:00
* Create udev library context . This reads the udev configuration
* file , and fills in the default values .
2008-08-27 19:11:58 +04:00
*
* The initial refcount is 1 , and needs to be decremented to
2008-10-26 17:48:48 +03:00
* release the resources of the udev library context .
2008-08-27 19:11:58 +04:00
*
* Returns : a new udev library context
* */
struct udev * udev_new ( void )
{
struct udev * udev ;
2008-09-06 17:45:31 +04:00
const char * env ;
char * config_file ;
FILE * f ;
2008-08-27 19:11:58 +04:00
2008-10-21 14:10:32 +04:00
udev = calloc ( 1 , sizeof ( struct udev ) ) ;
2008-08-27 19:11:58 +04:00
if ( udev = = NULL )
return NULL ;
udev - > refcount = 1 ;
udev - > log_fn = log_stderr ;
2008-09-06 17:45:31 +04:00
udev - > log_priority = LOG_ERR ;
2008-10-16 15:51:29 +04:00
udev_list_init ( & udev - > properties_list ) ;
2008-09-06 17:45:31 +04:00
udev - > run = 1 ;
2009-06-17 04:25:07 +04:00
udev - > dev_path = strdup ( " /dev " ) ;
2008-09-06 17:45:31 +04:00
udev - > sys_path = strdup ( " /sys " ) ;
config_file = strdup ( SYSCONFDIR " /udev/udev.conf " ) ;
2008-09-09 02:08:42 +04:00
if ( udev - > dev_path = = NULL | |
udev - > sys_path = = NULL | |
config_file = = NULL )
2008-09-06 17:45:31 +04:00
goto err ;
/* settings by environment and config file */
env = getenv ( " SYSFS_PATH " ) ;
if ( env ! = NULL ) {
free ( udev - > sys_path ) ;
udev - > sys_path = strdup ( env ) ;
2008-09-10 19:08:24 +04:00
util_remove_trailing_chars ( udev - > sys_path , ' / ' ) ;
2008-10-16 15:51:29 +04:00
udev_add_property ( udev , " SYSFS_PATH " , udev - > sys_path ) ;
2008-09-06 17:45:31 +04:00
}
env = getenv ( " UDEV_RUN " ) ;
2008-09-10 19:08:24 +04:00
if ( env ! = NULL & & strcmp ( env , " 0 " ) = = 0 )
2008-09-06 17:45:31 +04:00
udev - > run = 0 ;
env = getenv ( " UDEV_CONFIG_FILE " ) ;
if ( env ! = NULL ) {
free ( config_file ) ;
config_file = strdup ( env ) ;
2008-09-10 19:08:24 +04:00
util_remove_trailing_chars ( config_file , ' / ' ) ;
2008-09-06 17:45:31 +04:00
}
if ( config_file = = NULL )
goto err ;
f = fopen ( config_file , " r " ) ;
if ( f ! = NULL ) {
2008-09-10 20:00:31 +04:00
char line [ UTIL_LINE_SIZE ] ;
2008-09-06 17:45:31 +04:00
int line_nr = 0 ;
while ( fgets ( line , sizeof ( line ) , f ) ) {
size_t len ;
char * key ;
char * val ;
line_nr + + ;
/* find key */
key = line ;
while ( isspace ( key [ 0 ] ) )
key + + ;
/* comment or empty line */
if ( key [ 0 ] = = ' # ' | | key [ 0 ] = = ' \0 ' )
continue ;
/* split key/value */
val = strchr ( key , ' = ' ) ;
if ( val = = NULL ) {
err ( udev , " missing <key>=<value> in '%s'[%i], skip line \n " , config_file , line_nr ) ;
continue ;
}
val [ 0 ] = ' \0 ' ;
val + + ;
/* find value */
while ( isspace ( val [ 0 ] ) )
val + + ;
/* terminate key */
len = strlen ( key ) ;
if ( len = = 0 )
continue ;
while ( isspace ( key [ len - 1 ] ) )
len - - ;
key [ len ] = ' \0 ' ;
/* terminate value */
len = strlen ( val ) ;
if ( len = = 0 )
continue ;
while ( isspace ( val [ len - 1 ] ) )
len - - ;
val [ len ] = ' \0 ' ;
if ( len = = 0 )
continue ;
/* unquote */
if ( val [ 0 ] = = ' " ' | | val [ 0 ] = = ' \' ' ) {
if ( val [ len - 1 ] ! = val [ 0 ] ) {
err ( udev , " inconsistent quoting in '%s'[%i], skip line \n " , config_file , line_nr ) ;
continue ;
}
val [ len - 1 ] = ' \0 ' ;
val + + ;
}
2009-05-21 01:45:32 +04:00
if ( strcmp ( key , " udev_log " ) = = 0 ) {
2008-10-16 20:51:05 +04:00
udev_set_log_priority ( udev , util_log_priority ( val ) ) ;
2008-09-06 17:45:31 +04:00
continue ;
}
2009-05-21 01:45:32 +04:00
if ( strcmp ( key , " udev_root " ) = = 0 ) {
2008-09-06 17:45:31 +04:00
free ( udev - > dev_path ) ;
udev - > dev_path = strdup ( val ) ;
2008-09-10 19:08:24 +04:00
util_remove_trailing_chars ( udev - > dev_path , ' / ' ) ;
2008-09-06 17:45:31 +04:00
continue ;
}
2009-05-21 01:45:32 +04:00
if ( strcmp ( key , " udev_rules " ) = = 0 ) {
2008-09-06 17:45:31 +04:00
free ( udev - > rules_path ) ;
udev - > rules_path = strdup ( val ) ;
2008-09-10 19:08:24 +04:00
util_remove_trailing_chars ( udev - > rules_path , ' / ' ) ;
2008-09-06 17:45:31 +04:00
continue ;
}
}
fclose ( f ) ;
}
env = getenv ( " UDEV_ROOT " ) ;
if ( env ! = NULL ) {
free ( udev - > dev_path ) ;
udev - > dev_path = strdup ( env ) ;
2008-09-10 19:08:24 +04:00
util_remove_trailing_chars ( udev - > dev_path , ' / ' ) ;
2008-10-16 15:51:29 +04:00
udev_add_property ( udev , " UDEV_ROOT " , udev - > dev_path ) ;
2008-09-06 17:45:31 +04:00
}
env = getenv ( " UDEV_LOG " ) ;
if ( env ! = NULL )
2008-10-16 15:51:29 +04:00
udev_set_log_priority ( udev , util_log_priority ( env ) ) ;
2008-09-06 17:45:31 +04:00
if ( udev - > dev_path = = NULL | | udev - > sys_path = = NULL )
goto err ;
2008-11-01 22:16:24 +03:00
dbg ( udev , " context %p created \n " , udev ) ;
dbg ( udev , " log_priority=%d \n " , udev - > log_priority ) ;
dbg ( udev , " config_file='%s' \n " , config_file ) ;
dbg ( udev , " dev_path='%s' \n " , udev - > dev_path ) ;
dbg ( udev , " sys_path='%s' \n " , udev - > sys_path ) ;
2008-09-06 17:45:31 +04:00
if ( udev - > rules_path ! = NULL )
2008-11-01 22:16:24 +03:00
dbg ( udev , " rules_path='%s' \n " , udev - > rules_path ) ;
2008-09-09 02:08:42 +04:00
free ( config_file ) ;
2008-08-27 19:11:58 +04:00
return udev ;
2008-09-06 17:45:31 +04:00
err :
2008-09-09 02:08:42 +04:00
free ( config_file ) ;
2008-09-06 17:45:31 +04:00
err ( udev , " context creation failed \n " ) ;
udev_unref ( udev ) ;
return NULL ;
2008-08-27 19:11:58 +04:00
}
/**
* udev_ref :
* @ udev : udev library context
*
* Take a reference of the udev library context .
*
* Returns : the passed udev library context
* */
struct udev * udev_ref ( struct udev * udev )
{
2008-08-29 01:05:01 +04:00
if ( udev = = NULL )
return NULL ;
2008-08-27 19:11:58 +04:00
udev - > refcount + + ;
return udev ;
}
/**
* udev_unref :
* @ udev : udev library context
*
* Drop a reference of the udev library context . If the refcount
2008-10-26 17:48:48 +03:00
* reaches zero , the resources of the context will be released .
2008-08-27 19:11:58 +04:00
*
* */
void udev_unref ( struct udev * udev )
{
2008-08-29 01:05:01 +04:00
if ( udev = = NULL )
return ;
2008-08-27 19:11:58 +04:00
udev - > refcount - - ;
if ( udev - > refcount > 0 )
return ;
2008-10-18 21:27:38 +04:00
udev_list_cleanup_entries ( udev , & udev - > properties_list ) ;
2008-09-06 17:45:31 +04:00
free ( udev - > dev_path ) ;
free ( udev - > sys_path ) ;
2008-09-09 02:08:42 +04:00
free ( udev - > rules_path ) ;
2008-11-01 22:16:24 +03:00
dbg ( udev , " context %p released \n " , udev ) ;
2008-08-27 19:11:58 +04:00
free ( udev ) ;
}
/**
* udev_set_log_fn :
* @ udev : udev library context
* @ log_fn : function to be called for logging messages
*
2008-10-26 17:48:48 +03:00
* The built - in logging writes to stderr . It can be
2008-08-27 19:11:58 +04:00
* overridden by a custom function , to plug log messages
2008-10-26 17:48:48 +03:00
* into the users ' logging functionality .
2008-08-27 19:11:58 +04:00
*
* */
void udev_set_log_fn ( struct udev * udev ,
void ( * log_fn ) ( struct udev * udev ,
int priority , const char * file , int line , const char * fn ,
const char * format , va_list args ) )
{
udev - > log_fn = log_fn ;
2008-09-06 17:45:31 +04:00
info ( udev , " custom logging function %p registered \n " , udev ) ;
}
2009-06-15 15:22:38 +04:00
/**
* udev_get_log_priority :
* @ udev : udev library context
*
2009-08-08 17:29:38 +04:00
* The initial syslog priority is read from the udev config file
2009-06-15 15:22:38 +04:00
* at startup .
*
* Returns : the current syslog priority
* */
2008-09-06 17:45:31 +04:00
int udev_get_log_priority ( struct udev * udev )
{
return udev - > log_priority ;
}
2009-06-15 15:22:38 +04:00
/**
* udev_set_log_priority :
* @ udev : udev library context
* @ priority : the new syslog priority
*
* Set the current syslog priority . The value controls which messages
* are send to syslog .
* */
2008-09-06 17:45:31 +04:00
void udev_set_log_priority ( struct udev * udev , int priority )
{
2008-10-16 15:51:29 +04:00
char num [ 32 ] ;
2008-09-06 17:45:31 +04:00
udev - > log_priority = priority ;
2008-10-16 15:51:29 +04:00
snprintf ( num , sizeof ( num ) , " %u " , udev - > log_priority ) ;
udev_add_property ( udev , " UDEV_LOG " , num ) ;
2008-09-06 17:45:31 +04:00
}
const char * udev_get_rules_path ( struct udev * udev )
{
return udev - > rules_path ;
}
int udev_get_run ( struct udev * udev )
{
return udev - > run ;
2008-08-27 19:11:58 +04:00
}
/**
* udev_get_sys_path :
* @ udev : udev library context
*
* Retrieve the sysfs mount point . The default is " /sys " . For
* testing purposes , it can be overridden with the environment
* variable SYSFS_PATH .
*
* Returns : the sys mount point
* */
const char * udev_get_sys_path ( struct udev * udev )
{
2008-08-29 01:05:01 +04:00
if ( udev = = NULL )
return NULL ;
2008-09-06 17:45:31 +04:00
return udev - > sys_path ;
2008-08-27 19:11:58 +04:00
}
/**
* udev_get_dev_path :
* @ udev : udev library context
*
* Retrieve the device directory path . The default value is " /dev " ,
* the actual value may be overridden in the udev configuration
* file .
*
* Returns : the device directory path
* */
const char * udev_get_dev_path ( struct udev * udev )
{
2008-08-29 01:05:01 +04:00
if ( udev = = NULL )
return NULL ;
2008-09-06 17:45:31 +04:00
return udev - > dev_path ;
2008-08-27 19:11:58 +04:00
}
2008-10-16 15:51:29 +04:00
struct udev_list_entry * udev_add_property ( struct udev * udev , const char * key , const char * value )
{
if ( value = = NULL ) {
struct udev_list_entry * list_entry ;
list_entry = udev_get_properties_list_entry ( udev ) ;
list_entry = udev_list_entry_get_by_name ( list_entry , key ) ;
if ( list_entry ! = NULL )
2008-11-04 22:19:01 +03:00
udev_list_entry_delete ( list_entry ) ;
2008-10-16 15:51:29 +04:00
return NULL ;
}
return udev_list_entry_add ( udev , & udev - > properties_list , key , value , 1 , 0 ) ;
}
struct udev_list_entry * udev_get_properties_list_entry ( struct udev * udev )
{
return udev_list_get_entry ( & udev - > properties_list ) ;
}