2008-08-28 00:02:41 +04:00
/*
* libudev - interface to udev device information
*
* Copyright ( C ) 2008 Kay Sievers < kay . sievers @ vrfy . org >
*
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-28 00:02:41 +04:00
*/
# include <stdio.h>
# include <stdlib.h>
# include <stddef.h>
# include <unistd.h>
# include <errno.h>
# include <string.h>
# include <dirent.h>
2008-09-10 20:24:39 +04:00
# include <ctype.h>
2008-09-13 23:09:28 +04:00
# include <fcntl.h>
2008-08-28 00:02:41 +04:00
# include <sys/stat.h>
# include "libudev.h"
# include "libudev-private.h"
2008-10-15 20:34:14 +04:00
static ssize_t get_sys_link ( struct udev * udev , const char * slink , const char * syspath , char * value , size_t size )
2008-08-28 00:02:41 +04:00
{
2008-09-10 20:00:31 +04:00
char path [ UTIL_PATH_SIZE ] ;
2008-08-28 00:02:41 +04:00
ssize_t len ;
const char * pos ;
2008-09-16 04:12:47 +04:00
util_strlcpy ( path , syspath , sizeof ( path ) ) ;
2008-09-10 20:00:31 +04:00
util_strlcat ( path , " / " , sizeof ( path ) ) ;
util_strlcat ( path , slink , sizeof ( path ) ) ;
2008-08-28 00:02:41 +04:00
len = readlink ( path , path , sizeof ( path ) ) ;
if ( len < 0 | | len > = ( ssize_t ) sizeof ( path ) )
return - 1 ;
path [ len ] = ' \0 ' ;
pos = strrchr ( path , ' / ' ) ;
if ( pos = = NULL )
return - 1 ;
pos = & pos [ 1 ] ;
2008-11-01 22:16:24 +03:00
dbg ( udev , " resolved link to: '%s' \n " , pos ) ;
2008-10-15 20:34:14 +04:00
return util_strlcpy ( value , pos , size ) ;
2008-08-28 00:02:41 +04:00
}
2008-09-09 20:14:54 +04:00
2008-09-16 04:12:47 +04:00
ssize_t util_get_sys_subsystem ( struct udev * udev , const char * syspath , char * subsystem , size_t size )
2008-09-09 20:14:54 +04:00
{
2008-09-16 04:12:47 +04:00
return get_sys_link ( udev , " subsystem " , syspath , subsystem , size ) ;
2008-09-09 20:14:54 +04:00
}
2008-09-16 04:12:47 +04:00
ssize_t util_get_sys_driver ( struct udev * udev , const char * syspath , char * driver , size_t size )
2008-09-09 20:14:54 +04:00
{
2008-09-16 04:12:47 +04:00
return get_sys_link ( udev , " driver " , syspath , driver , size ) ;
2008-09-09 20:14:54 +04:00
}
2008-09-16 04:12:47 +04:00
int util_resolve_sys_link ( struct udev * udev , char * syspath , size_t size )
2008-09-10 16:29:07 +04:00
{
2008-09-10 20:00:31 +04:00
char link_target [ UTIL_PATH_SIZE ] ;
2008-09-10 16:29:07 +04:00
int len ;
int i ;
int back ;
2008-09-16 04:12:47 +04:00
len = readlink ( syspath , link_target , sizeof ( link_target ) ) ;
2008-09-10 16:29:07 +04:00
if ( len < = 0 )
return - 1 ;
link_target [ len ] = ' \0 ' ;
2008-09-16 04:12:47 +04:00
dbg ( udev , " path link '%s' points to '%s' \n " , syspath , link_target ) ;
2008-09-10 16:29:07 +04:00
for ( back = 0 ; strncmp ( & link_target [ back * 3 ] , " ../ " , 3 ) = = 0 ; back + + )
;
2008-09-16 04:12:47 +04:00
dbg ( udev , " base '%s', tail '%s', back %i \n " , syspath , & link_target [ back * 3 ] , back ) ;
2008-09-10 16:29:07 +04:00
for ( i = 0 ; i < = back ; i + + ) {
2008-09-16 04:12:47 +04:00
char * pos = strrchr ( syspath , ' / ' ) ;
2008-09-10 16:29:07 +04:00
if ( pos = = NULL )
return - 1 ;
pos [ 0 ] = ' \0 ' ;
}
2008-09-16 04:12:47 +04:00
dbg ( udev , " after moving back '%s' \n " , syspath ) ;
util_strlcat ( syspath , " / " , size ) ;
util_strlcat ( syspath , & link_target [ back * 3 ] , size ) ;
2008-09-10 16:29:07 +04:00
return 0 ;
}
2008-09-10 19:08:24 +04:00
int util_log_priority ( const char * priority )
{
char * endptr ;
int prio ;
prio = strtol ( priority , & endptr , 10 ) ;
if ( endptr [ 0 ] = = ' \0 ' )
return prio ;
if ( strncasecmp ( priority , " err " , 3 ) = = 0 )
return LOG_ERR ;
if ( strcasecmp ( priority , " info " ) = = 0 )
return LOG_INFO ;
if ( strcasecmp ( priority , " debug " ) = = 0 )
return LOG_DEBUG ;
return 0 ;
}
2009-04-15 23:47:04 +04:00
size_t util_path_encode ( char * s , size_t size )
2008-09-10 19:08:24 +04:00
{
2009-04-15 23:47:04 +04:00
char t [ ( size * 4 ) + 1 ] ;
2008-09-10 19:08:24 +04:00
size_t i , j ;
2009-04-15 23:47:04 +04:00
for ( i = 0 , j = 0 ; s [ i ] ! = ' \0 ' & & i < size ; i + + ) {
2008-09-10 19:08:24 +04:00
if ( s [ i ] = = ' / ' ) {
memcpy ( & t [ j ] , " \\ x2f " , 4 ) ;
j + = 4 ;
} else if ( s [ i ] = = ' \\ ' ) {
memcpy ( & t [ j ] , " \\ x5c " , 4 ) ;
j + = 4 ;
} else {
t [ j ] = s [ i ] ;
j + + ;
}
}
2009-04-15 23:47:04 +04:00
if ( i > = size )
return 0 ;
if ( j > = size )
return 0 ;
memcpy ( s , t , j ) ;
s [ j ] = ' \0 ' ;
2008-09-10 19:08:24 +04:00
return j ;
}
size_t util_path_decode ( char * s )
{
size_t i , j ;
for ( i = 0 , j = 0 ; s [ i ] ! = ' \0 ' ; j + + ) {
if ( memcmp ( & s [ i ] , " \\ x2f " , 4 ) = = 0 ) {
s [ j ] = ' / ' ;
i + = 4 ;
2009-04-15 23:47:04 +04:00
} else if ( memcmp ( & s [ i ] , " \\ x5c " , 4 ) = = 0 ) {
2008-09-10 19:08:24 +04:00
s [ j ] = ' \\ ' ;
i + = 4 ;
} else {
s [ j ] = s [ i ] ;
i + + ;
}
}
s [ j ] = ' \0 ' ;
return j ;
}
void util_remove_trailing_chars ( char * path , char c )
{
size_t len ;
if ( path = = NULL )
return ;
len = strlen ( path ) ;
while ( len > 0 & & path [ len - 1 ] = = c )
path [ - - len ] = ' \0 ' ;
}
2008-09-10 20:00:31 +04:00
size_t util_strlcpy ( char * dst , const char * src , size_t size )
{
size_t bytes = 0 ;
char * q = dst ;
const char * p = src ;
char ch ;
while ( ( ch = * p + + ) ) {
if ( bytes + 1 < size )
* q + + = ch ;
bytes + + ;
}
/* If size == 0 there is no space for a final null... */
if ( size )
* q = ' \0 ' ;
return bytes ;
}
size_t util_strlcat ( char * dst , const char * src , size_t size )
{
size_t bytes = 0 ;
char * q = dst ;
const char * p = src ;
char ch ;
while ( bytes < size & & * q ) {
q + + ;
bytes + + ;
}
if ( bytes = = size )
return ( bytes + strlen ( src ) ) ;
while ( ( ch = * p + + ) ) {
if ( bytes + 1 < size )
* q + + = ch ;
bytes + + ;
}
* q = ' \0 ' ;
return bytes ;
}
2008-09-10 20:24:39 +04:00
/* count of characters used to encode one unicode char */
static int utf8_encoded_expected_len ( const char * str )
{
unsigned char c = ( unsigned char ) str [ 0 ] ;
if ( c < 0x80 )
return 1 ;
if ( ( c & 0xe0 ) = = 0xc0 )
return 2 ;
if ( ( c & 0xf0 ) = = 0xe0 )
return 3 ;
if ( ( c & 0xf8 ) = = 0xf0 )
return 4 ;
if ( ( c & 0xfc ) = = 0xf8 )
return 5 ;
if ( ( c & 0xfe ) = = 0xfc )
return 6 ;
return 0 ;
}
/* decode one unicode char */
static int utf8_encoded_to_unichar ( const char * str )
{
int unichar ;
int len ;
int i ;
len = utf8_encoded_expected_len ( str ) ;
switch ( len ) {
case 1 :
return ( int ) str [ 0 ] ;
case 2 :
unichar = str [ 0 ] & 0x1f ;
break ;
case 3 :
unichar = ( int ) str [ 0 ] & 0x0f ;
break ;
case 4 :
unichar = ( int ) str [ 0 ] & 0x07 ;
break ;
case 5 :
unichar = ( int ) str [ 0 ] & 0x03 ;
break ;
case 6 :
unichar = ( int ) str [ 0 ] & 0x01 ;
break ;
default :
return - 1 ;
}
for ( i = 1 ; i < len ; i + + ) {
if ( ( ( int ) str [ i ] & 0xc0 ) ! = 0x80 )
return - 1 ;
unichar < < = 6 ;
unichar | = ( int ) str [ i ] & 0x3f ;
}
return unichar ;
}
/* expected size used to encode one unicode char */
static int utf8_unichar_to_encoded_len ( int unichar )
{
if ( unichar < 0x80 )
return 1 ;
if ( unichar < 0x800 )
return 2 ;
if ( unichar < 0x10000 )
return 3 ;
if ( unichar < 0x200000 )
return 4 ;
if ( unichar < 0x4000000 )
return 5 ;
return 6 ;
}
/* check if unicode char has a valid numeric range */
static int utf8_unichar_valid_range ( int unichar )
{
if ( unichar > 0x10ffff )
return 0 ;
if ( ( unichar & 0xfffff800 ) = = 0xd800 )
return 0 ;
if ( ( unichar > 0xfdcf ) & & ( unichar < 0xfdf0 ) )
return 0 ;
if ( ( unichar & 0xffff ) = = 0xffff )
return 0 ;
return 1 ;
}
/* validate one encoded unicode char and return its length */
static int utf8_encoded_valid_unichar ( const char * str )
{
int len ;
int unichar ;
int i ;
len = utf8_encoded_expected_len ( str ) ;
if ( len = = 0 )
return - 1 ;
/* ascii is valid */
if ( len = = 1 )
return 1 ;
/* check if expected encoded chars are available */
for ( i = 0 ; i < len ; i + + )
if ( ( str [ i ] & 0x80 ) ! = 0x80 )
return - 1 ;
unichar = utf8_encoded_to_unichar ( str ) ;
/* check if encoded length matches encoded value */
if ( utf8_unichar_to_encoded_len ( unichar ) ! = len )
return - 1 ;
/* check if value has valid range */
if ( ! utf8_unichar_valid_range ( unichar ) )
return - 1 ;
return len ;
}
2008-11-05 23:49:52 +03:00
int udev_util_replace_whitespace ( const char * str , char * to , size_t len )
{
size_t i , j ;
/* strip trailing whitespace */
len = strnlen ( str , len ) ;
while ( len & & isspace ( str [ len - 1 ] ) )
len - - ;
/* strip leading whitespace */
i = 0 ;
while ( isspace ( str [ i ] ) & & ( i < len ) )
i + + ;
j = 0 ;
while ( i < len ) {
/* substitute multiple whitespace with a single '_' */
if ( isspace ( str [ i ] ) ) {
while ( isspace ( str [ i ] ) )
i + + ;
to [ j + + ] = ' _ ' ;
}
to [ j + + ] = str [ i + + ] ;
}
to [ j ] = ' \0 ' ;
return 0 ;
}
static int is_whitelisted ( char c , const char * white )
{
if ( ( c > = ' 0 ' & & c < = ' 9 ' ) | |
( c > = ' A ' & & c < = ' Z ' ) | |
( c > = ' a ' & & c < = ' z ' ) | |
strchr ( " #+-.:=@_ " , c ) ! = NULL | |
( white ! = NULL & & strchr ( white , c ) ! = NULL ) )
return 1 ;
return 0 ;
}
2008-09-10 20:24:39 +04:00
/* allow chars in whitelist, plain ascii, hex-escaping and valid utf8 */
2008-11-05 23:49:52 +03:00
int udev_util_replace_chars ( char * str , const char * white )
2008-09-10 20:24:39 +04:00
{
size_t i = 0 ;
int replaced = 0 ;
while ( str [ i ] ! = ' \0 ' ) {
int len ;
2008-11-05 23:49:52 +03:00
if ( is_whitelisted ( str [ i ] , white ) ) {
2008-09-10 20:24:39 +04:00
i + + ;
continue ;
}
/* accept hex encoding */
if ( str [ i ] = = ' \\ ' & & str [ i + 1 ] = = ' x ' ) {
i + = 2 ;
continue ;
}
/* accept valid utf8 */
len = utf8_encoded_valid_unichar ( & str [ i ] ) ;
if ( len > 1 ) {
i + = len ;
continue ;
}
/* if space is allowed, replace whitespace with ordinary space */
2008-11-20 04:43:34 +03:00
if ( isspace ( str [ i ] ) & & white ! = NULL & & strchr ( white , ' ' ) ! = NULL ) {
2008-09-10 20:24:39 +04:00
str [ i ] = ' ' ;
i + + ;
replaced + + ;
continue ;
}
/* everything else is replaced with '_' */
str [ i ] = ' _ ' ;
i + + ;
replaced + + ;
}
return replaced ;
}
2008-11-05 23:49:52 +03:00
/**
* util_encode_string :
* @ str : input string to be encoded
* @ str_enc : output string to store the encoded input string
* @ len : maximum size of the output string , which may be
* four times as long as the input string
*
* Encode all potentially unsafe characters of a string to the
* corresponding hex value prefixed by ' \x ' .
*
* Returns : 0 if the entire string was copied , non - zero otherwise .
* */
int udev_util_encode_string ( const char * str , char * str_enc , size_t len )
{
size_t i , j ;
if ( str = = NULL | | str_enc = = NULL | | len = = 0 )
return - 1 ;
str_enc [ 0 ] = ' \0 ' ;
for ( i = 0 , j = 0 ; str [ i ] ! = ' \0 ' ; i + + ) {
int seqlen ;
seqlen = utf8_encoded_valid_unichar ( & str [ i ] ) ;
if ( seqlen > 1 ) {
memcpy ( & str_enc [ j ] , & str [ i ] , seqlen ) ;
j + = seqlen ;
i + = ( seqlen - 1 ) ;
} else if ( str [ i ] = = ' \\ ' | | ! is_whitelisted ( str [ i ] , NULL ) ) {
sprintf ( & str_enc [ j ] , " \\ x%02x " , ( unsigned char ) str [ i ] ) ;
j + = 4 ;
} else {
str_enc [ j ] = str [ i ] ;
j + + ;
}
if ( j + 3 > = len )
goto err ;
}
str_enc [ j ] = ' \0 ' ;
return 0 ;
err :
return - 1 ;
}
2009-04-06 13:18:41 +04:00
void util_set_fd_cloexec ( int fd )
{
int flags ;
flags = fcntl ( fd , F_GETFD ) ;
if ( flags < 0 )
flags = FD_CLOEXEC ;
else
flags | = FD_CLOEXEC ;
fcntl ( fd , F_SETFD , flags ) ;
}
2009-04-22 05:50:11 +04:00
2009-04-22 20:38:16 +04:00
unsigned int util_string_hash32 ( const char * str )
2009-04-22 05:50:11 +04:00
{
2009-04-22 20:38:16 +04:00
unsigned int hash = 0 ;
while ( str [ 0 ] ! = ' \0 ' ) {
hash + = str [ 0 ] < < 4 ;
hash + = str [ 0 ] > > 4 ;
hash * = 11 ;
str + + ;
}
return hash ;
2009-04-22 05:50:11 +04:00
}