2010-09-20 14:46:25 +02:00
/*
* Samba Unix / Linux SMB client library
*
* Copyright ( C ) Gregor Beck 2010
*
* 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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
/**
* @ brief Parser for dot . reg files
* @ file reg_parse . c
* @ author Gregor Beck < gb @ sernet . de >
* @ date Jun 2010
*
*/
# include "includes.h"
2011-02-25 23:20:06 +01:00
# include "system/filesys.h"
2010-09-20 14:46:25 +02:00
# include "cbuf.h"
# include "srprs.h"
# include "reg_parse_internal.h"
# include "reg_parse.h"
# include "reg_format.h"
# include <stdio.h>
# include <unistd.h>
# include <wchar.h>
# include <talloc.h>
# include <stdbool.h>
# include <string.h>
# include <sys/types.h>
# include <regex.h>
# include <assert.h>
# include <stdint.h>
enum reg_parse_state {
STATE_DEFAULT ,
STATE_KEY_OPEN ,
STATE_VAL_HEX_CONT ,
STATE_VAL_SZ_CONT
} ;
struct reg_parse {
struct reg_format_callback reg_format_callback ;
cbuf * key ;
cbuf * valname ;
uint32_t valtype ;
cbuf * valblob ;
cbuf * tmp ;
struct reg_parse_callback call ;
int ret ;
int linenum ;
enum reg_parse_state state ;
struct reg_parse_options * opt ;
smb_iconv_t str2UTF16 ;
unsigned flags ;
} ;
/**
* @ defgroup action Action
* @ {
*/
static bool act_key ( struct reg_parse * p , cbuf * keyname , bool del )
{
const char * name = cbuf_gets ( keyname , 0 ) ;
cbuf_swap ( p - > key , keyname ) ;
assert ( p - > state = = STATE_DEFAULT | | p - > state = = STATE_KEY_OPEN ) ;
p - > state = del ? STATE_DEFAULT : STATE_KEY_OPEN ;
assert ( p - > call . key ) ;
p - > ret = p - > call . key ( p - > call . data , & name , 1 , del ) ;
return p - > ret > = 0 ;
}
static bool value_callback ( struct reg_parse * p )
{
const char * name = cbuf_gets ( p - > valname , 0 ) ;
const uint8_t * val = ( const uint8_t * ) cbuf_gets ( p - > valblob , 0 ) ;
size_t len = cbuf_getpos ( p - > valblob ) ;
assert ( p - > call . val ) ;
p - > ret = p - > call . val ( p - > call . data , name , p - > valtype , val , len ) ;
return p - > ret > = 0 ;
}
static bool act_val_hex ( struct reg_parse * p , cbuf * value , bool cont )
{
cbuf_swap ( p - > valblob , value ) ;
assert ( ( p - > state = = STATE_KEY_OPEN ) | | ( p - > state = = STATE_VAL_HEX_CONT ) ) ;
if ( cont ) {
p - > state = STATE_VAL_HEX_CONT ;
} else {
p - > state = STATE_KEY_OPEN ;
switch ( p - > valtype ) {
case REG_EXPAND_SZ :
case REG_MULTI_SZ :
if ( p - > str2UTF16 ! = NULL ) {
char * dst = NULL ;
const char * src = cbuf_gets ( p - > valblob , 0 ) ;
const size_t slen = cbuf_getpos ( p - > valblob ) ;
size_t dlen = iconvert_talloc ( p ,
p - > str2UTF16 ,
src , slen ,
& dst ) ;
if ( dlen ! = - 1 ) {
cbuf_swapptr ( p - > valblob , & dst , dlen ) ;
} else {
DEBUG ( 0 , ( " iconvert_talloc failed \n " ) ) ;
}
talloc_free ( dst ) ;
}
default :
break ;
}
return value_callback ( p ) ;
}
return true ;
}
static bool act_val_dw ( struct reg_parse * p , uint32_t val )
{
assert ( p - > valtype = = REG_DWORD ) ;
assert ( p - > state = = STATE_KEY_OPEN ) ;
cbuf_clear ( p - > valblob ) ;
if ( cbuf_putdw ( p - > valblob , val ) < 0 ) {
return false ;
}
return value_callback ( p ) ;
}
static bool act_val_sz ( struct reg_parse * p , cbuf * value , bool cont )
{
cbuf_swap ( p - > valblob , value ) ;
assert ( p - > valtype = = REG_SZ ) ;
assert ( ( p - > state = = STATE_KEY_OPEN ) | | ( p - > state = = STATE_VAL_SZ_CONT ) ) ;
if ( cont ) {
p - > state = STATE_VAL_SZ_CONT ;
} else {
char * dst = NULL ;
size_t dlen ;
const char * src = cbuf_gets ( p - > valblob , 0 ) ;
p - > state = STATE_KEY_OPEN ;
if ( convert_string_talloc ( p - > valblob , CH_UNIX , CH_UTF16LE ,
src , strlen ( src ) + 1 ,
2011-03-24 10:59:41 +11:00
& dst , & dlen ) )
2010-09-20 14:46:25 +02:00
{
cbuf_swapptr ( p - > valblob , & dst , dlen ) ;
} else {
DEBUG ( 0 , ( " convert_string_talloc failed: >%s< \n "
" use it as is \t " , src ) ) ;
}
talloc_free ( dst ) ;
return value_callback ( p ) ;
}
return true ;
}
static bool act_val_del ( struct reg_parse * p )
{
const char * name = cbuf_gets ( p - > valname , 0 ) ;
assert ( p - > call . val_del ) ;
p - > ret = p - > call . val_del ( p - > call . data , name ) ;
return p - > ret > = 0 ;
}
static bool act_comment ( struct reg_parse * p , const char * txt )
{
assert ( p - > call . comment ) ;
p - > ret = p - > call . comment ( p - > call . data , txt ) ;
return p - > ret > = 0 ;
}
/**@}*/
static int nop ( void * data )
{
return 0 ;
}
struct reg_parse * reg_parse_new ( const void * ctx ,
struct reg_parse_callback cb ,
const char * str_enc , unsigned flags )
{
struct reg_parse * s = talloc_zero ( ctx , struct reg_parse ) ;
if ( s = = NULL )
return NULL ;
s - > key = cbuf_new ( s ) ;
s - > valname = cbuf_new ( s ) ;
s - > valblob = cbuf_new ( s ) ;
s - > tmp = cbuf_new ( s ) ;
if ( ( s - > tmp = = NULL ) | | ( s - > valblob = = NULL )
| | ( s - > valname = = NULL ) | | ( s - > key = = NULL ) )
{
goto fail ;
}
s - > reg_format_callback . writeline = ( reg_format_callback_writeline_t ) & reg_parse_line ;
s - > reg_format_callback . data = s ;
s - > valtype = 0 ;
if ( cb . key = = NULL ) {
cb . key = ( reg_parse_callback_key_t ) & nop ;
}
if ( cb . val = = NULL ) {
cb . val = ( reg_parse_callback_val_t ) & nop ;
}
if ( cb . val_del = = NULL ) {
cb . val_del = ( reg_parse_callback_val_del_t ) & nop ;
}
if ( cb . comment = = NULL ) {
cb . comment = ( reg_parse_callback_comment_t ) & nop ;
}
s - > call = cb ;
s - > linenum = 0 ;
s - > state = STATE_DEFAULT ;
s - > flags = flags ;
if ( str_enc & & ! set_iconv ( & s - > str2UTF16 , " UTF-16LE " , str_enc ) ) {
DEBUG ( 0 , ( " reg_parse_new: failed to set encoding: %s " ,
str_enc ) ) ;
goto fail ;
}
assert ( & s - > reg_format_callback = = ( struct reg_format_callback * ) s ) ;
return s ;
fail :
talloc_free ( s ) ;
return NULL ;
}
/**
* @ defgroup parse Parser Primitive
* @ ingroup internal
* @ {
*/
static bool srprs_key ( const char * * ptr , cbuf * key , bool * del )
{
const char * pos = * ptr ;
const char * closing_bracket_pos = NULL ;
2010-10-05 14:58:52 +02:00
size_t closing_bracket_idx = 0 ;
2010-09-20 14:46:25 +02:00
if ( ! srprs_skipws ( & pos ) | | ! srprs_char ( & pos , ' [ ' ) ) {
return false ;
}
* del = srprs_char ( & pos , ' - ' ) ;
cbuf_clear ( key ) ;
while ( true ) {
while ( srprs_charsetinv ( & pos , " ] \\ " , key ) )
;
switch ( * pos ) {
case ' ] ' :
closing_bracket_idx = cbuf_getpos ( key ) ;
closing_bracket_pos = pos ;
cbuf_putc ( key , ' ] ' ) ;
pos + + ;
break ;
case ' \\ ' :
cbuf_putc ( key , ' \\ ' ) ;
/* n++; */
/* cbuf_puts(subkeyidx, cbuf_getpos(key), sizeof(size_t)) */
while ( srprs_char ( & pos , ' \\ ' ) )
;
break ;
case ' \0 ' :
if ( closing_bracket_pos = = NULL ) {
return false ;
}
/* remove trailing backslash (if any) */
if ( * ( closing_bracket_pos - 1 ) = = ' \\ ' ) {
closing_bracket_idx - - ;
}
cbuf_setpos ( key , closing_bracket_idx ) ;
* ptr = closing_bracket_pos + 1 ;
return true ;
default :
assert ( false ) ;
}
}
}
static bool srprs_val_name ( const char * * ptr , cbuf * name )
{
const char * pos = * ptr ;
const size_t spos = cbuf_getpos ( name ) ;
if ( ! srprs_skipws ( & pos ) ) {
goto fail ;
}
if ( srprs_char ( & pos , ' @ ' ) ) {
cbuf_puts ( name , " " , - 1 ) ;
}
else if ( ! srprs_quoted_string ( & pos , name , NULL ) ) {
goto fail ;
}
if ( ! srprs_skipws ( & pos ) | | ! srprs_char ( & pos , ' = ' ) ) {
goto fail ;
}
* ptr = pos ;
return true ;
fail :
cbuf_setpos ( name , spos ) ;
return false ;
}
static bool srprs_val_dword ( const char * * ptr , uint32_t * type , uint32_t * val )
{
const char * pos = * ptr ;
if ( ! srprs_str ( & pos , " dword: " , - 1 ) ) {
return false ;
}
if ( ! srprs_hex ( & pos , 8 , val ) ) {
return false ;
}
* type = REG_DWORD ;
* ptr = pos ;
return true ;
}
static bool srprs_val_sz ( const char * * ptr , uint32_t * type , cbuf * val , bool * cont )
{
if ( ! srprs_quoted_string ( ptr , val , cont ) ) {
return false ;
}
* type = REG_SZ ;
return true ;
}
static bool srprs_nl_no_eos ( const char * * ptr , cbuf * str , bool eof )
{
const char * pos = * ptr ;
const size_t spos = cbuf_getpos ( str ) ;
if ( srprs_nl ( & pos , str ) & & ( eof | | * pos ! = ' \0 ' ) ) {
* ptr = pos ;
return true ;
}
cbuf_setpos ( str , spos ) ;
return false ;
}
static bool srprs_eol_cont ( const char * * ptr , bool * cont )
{
const char * pos = * ptr ;
bool bs = srprs_char ( & pos , ' \\ ' ) ;
if ( ! srprs_eol ( & pos , NULL ) ) {
return false ;
}
* cont = bs ;
* ptr = pos ;
return true ;
}
/* matches the empty string, for zero length lists */
static bool srprs_val_hex_values ( const char * * ptr , cbuf * val , bool * cont )
{
const char * pos = * ptr ;
unsigned u ;
do {
if ( ! srprs_skipws ( & pos ) | | ! srprs_hex ( & pos , 2 , & u ) | | ! srprs_skipws ( & pos ) ) {
break ;
}
cbuf_putc ( val , ( char ) u ) ;
} while ( srprs_char ( & pos , ' , ' ) ) ;
* ptr = pos ;
if ( srprs_skipws ( & pos ) & & srprs_eol_cont ( & pos , cont ) ) {
* ptr = pos ;
}
return true ;
}
static bool srprs_val_hex ( const char * * ptr , uint32_t * ptype , cbuf * val ,
bool * cont )
{
const char * pos = * ptr ;
uint32_t type ;
if ( ! srprs_str ( & pos , " hex " , - 1 ) ) {
return false ;
}
if ( srprs_char ( & pos , ' : ' ) ) {
type = REG_BINARY ;
}
else if ( ! srprs_char ( & pos , ' ( ' ) | |
! srprs_hex ( & pos , 8 , & type ) | |
! srprs_char ( & pos , ' ) ' ) | |
! srprs_char ( & pos , ' : ' ) )
{
return false ;
}
if ( ! srprs_val_hex_values ( & pos , val , cont ) ) {
return false ;
}
* ptype = type ;
* ptr = pos ;
return true ;
}
static bool srprs_comment ( const char * * ptr , cbuf * str )
{
return srprs_char ( ptr , ' ; ' ) & & srprs_line ( ptr , str ) ;
}
/**@}*/
int reg_parse_set_options ( struct reg_parse * parser , const char * options )
{
static const char * DEFAULT = " enc=unix,flags=0 " ;
int ret = 0 ;
char * key , * val ;
void * ctx = talloc_new ( parser ) ;
if ( options = = NULL ) {
options = DEFAULT ;
}
while ( srprs_option ( & options , ctx , & key , & val ) ) {
if ( ( strcmp ( key , " enc " ) = = 0 ) | | ( strcmp ( key , " strenc " ) = = 0 ) ) {
} else if ( ( strcmp ( key , " flags " ) = = 0 ) & & ( val ! = NULL ) ) {
char * end = NULL ;
if ( val ! = NULL ) {
parser - > flags = strtol ( val , & end , 0 ) ;
}
if ( ( end = = NULL ) | | ( * end ! = ' \0 ' ) ) {
DEBUG ( 0 , ( " Invalid flags format: %s \n " ,
val ? val : " <NULL> " ) ) ;
ret = - 1 ;
}
}
/* else if (strcmp(key, "hive") == 0) { */
/* if (strcmp(val, "short") == 0) { */
/* f->hive_fmt = REG_FMT_SHORT_HIVES; */
/* } else if (strcmp(val, "long") == 0) { */
/* f->hive_fmt = REG_FMT_LONG_HIVES; */
/* } else if (strcmp(val, "preserve") == 0) { */
/* f->hive_fmt = REG_FMT_PRESERVE_HIVES; */
/* } else { */
/* DEBUG(0, ("Invalid hive format: %s\n", val)); */
/* ret = -1; */
/* } */
/* } */
}
talloc_free ( ctx ) ;
return ret ;
}
int reg_parse_line ( struct reg_parse * parser , const char * line )
{
const char * pos ;
bool del = false ;
cbuf * tmp = cbuf_clear ( parser - > tmp ) ;
bool cb_ok = true ;
bool cont = true ;
if ( ! line ) {
return - 4 ;
}
parser - > linenum + + ;
pos = line ;
switch ( parser - > state ) {
case STATE_VAL_HEX_CONT :
if ( srprs_val_hex_values ( & pos , parser - > valblob , & cont ) ) {
cb_ok = act_val_hex ( parser , parser - > valblob , cont ) ;
}
goto done ;
case STATE_VAL_SZ_CONT :
if ( srprs_quoted_string ( & pos , parser - > valblob , & cont ) ) {
cb_ok = act_val_sz ( parser , parser - > valblob , cont ) ;
}
goto done ;
default :
cont = false ;
}
if ( ! srprs_skipws ( & pos ) ) {
return - 4 ;
}
/* empty line ?*/
if ( srprs_eol ( & pos , NULL ) ) {
return 0 ;
}
/* key line ?*/
else if ( srprs_key ( & pos , tmp , & del ) ) {
cb_ok = act_key ( parser , tmp , del ) ;
}
/* comment line ? */
else if ( srprs_comment ( & pos , tmp ) ) {
cb_ok = act_comment ( parser , cbuf_gets ( tmp , 0 ) ) ;
}
/* head line */
else if ( ( parser - > linenum = = 1 ) & & srprs_line ( & pos , tmp ) ) {
/* cb_ok = act_head(parser, cbuf_gets(tmp, 0)); */
}
/* value line ?*/
else if ( srprs_val_name ( & pos , tmp ) ) {
uint32_t dw ;
cbuf_swap ( parser - > valname , tmp ) ;
cbuf_clear ( tmp ) ;
if ( parser - > state ! = STATE_KEY_OPEN ) {
DEBUG ( 0 , ( " value \" %s \" without a key at line: %i " ,
cbuf_gets ( parser - > valname , 0 ) , parser - > linenum ) ) ;
return - 3 ;
}
if ( ! srprs_skipws ( & pos ) ) {
return - 4 ;
}
if ( srprs_char ( & pos , ' - ' ) ) {
cb_ok = act_val_del ( parser ) ;
}
else if ( srprs_val_dword ( & pos , & parser - > valtype , & dw ) ) {
cb_ok = act_val_dw ( parser , dw ) ;
}
else if ( srprs_val_sz ( & pos , & parser - > valtype , tmp , & cont ) ) {
cb_ok = act_val_sz ( parser , tmp , cont ) ;
}
else if ( srprs_val_hex ( & pos , & parser - > valtype , tmp , & cont ) ) {
cb_ok = act_val_hex ( parser , tmp , cont ) ;
}
else {
DEBUG ( 0 , ( " value \" %s \" parse error "
" at line: %i pos: %li : %s " ,
cbuf_gets ( parser - > valname , 0 ) , parser - > linenum ,
2010-09-25 14:59:42 -07:00
( long int ) ( pos - line ) , pos ) ) ;
2010-09-20 14:46:25 +02:00
return - 3 ;
}
}
else {
DEBUG ( 0 , ( " unrecognized line %i : %s \n " , parser - > linenum , line ) ) ;
return - 3 ;
}
done :
if ( ! cb_ok )
return - 2 ;
if ( ! srprs_skipws ( & pos ) | | ! srprs_eol ( & pos , NULL ) ) {
DEBUG ( 0 , ( " trailing garbage at line: %i pos: %li : %s \n " ,
2010-09-25 14:59:42 -07:00
parser - > linenum , ( long int ) ( pos - line ) , pos ) ) ;
2010-09-20 14:46:25 +02:00
return - 1 ;
}
return 0 ;
}
/******************************************************************************/
/**
* @ addtogroup misc
* @ {
*/
static bool lookslike_utf16 ( const char * line , size_t len , bool * little_endian )
{
static const uint16_t M_LE = 0xFF80 ;
static const uint16_t M_BE = 0x80FF ;
uint16_t mask ;
bool le ;
size_t l = MIN ( len / 2 , 64 ) ;
2011-05-05 16:19:49 -07:00
const uint16_t * u = ( const uint16_t * ) line ;
2010-09-20 14:46:25 +02:00
int i ;
assert ( len > = 2 ) ;
if ( u [ 0 ] & M_LE ) {
le = true ;
mask = M_LE ;
} else if ( u [ 0 ] & M_BE ) {
le = false ;
mask = M_BE ;
} else {
return false ;
}
for ( i = 1 ; i < l ; i + + ) {
if ( u [ i ] & mask ) {
return false ;
}
}
* little_endian = le ;
return true ;
}
static bool lookslike_dos ( const char * line , size_t len )
{
int i ;
for ( i = 0 ; i < len ; i + + ) {
if ( ( line [ i ] = = ' \0 ' ) | | ( line [ i ] & 0x80 ) ) {
return false ;
}
if ( ( line [ i ] = = ' \r ' ) & & ( i + 1 < len ) & & ( line [ i + 1 ] = = ' \n ' ) ) {
return true ;
}
}
return false ;
}
static bool guess_charset ( const char * * ptr ,
size_t * len ,
const char * * file_enc ,
const char * * str_enc )
{
const char * charset = NULL ;
const char * pos = * ptr ;
if ( * len < 4 ) {
return false ;
}
if ( srprs_bom ( & pos , & charset , NULL ) ) {
* len - = ( pos - * ptr ) ;
* ptr = pos ;
if ( * file_enc = = NULL ) {
* file_enc = charset ;
}
else if ( strcmp ( * file_enc , charset ) ! = 0 ) {
DEBUG ( 0 , ( " file encoding forced to %s \n " ,
* file_enc ) ) ;
}
}
else if ( * file_enc = = NULL ) {
bool le ;
if ( lookslike_utf16 ( * ptr , * len , & le ) ) {
* file_enc = le ? " UTF-16LE " : " UTF-16BE " ;
}
else if ( lookslike_dos ( * ptr , * len ) ) {
* file_enc = " dos " ;
}
else {
* file_enc = " unix " ;
}
}
if ( ( str_enc ! = NULL ) & & ( * str_enc = = NULL ) ) {
* str_enc = ( strncmp ( * ptr , " REGEDIT4 " , 8 ) = = 0 )
? * file_enc
: " UTF-16LE " ;
}
return true ;
}
/**@}*/
struct reg_parse_fd_opt {
const char * file_enc ;
const char * str_enc ;
unsigned flags ;
int fail_level ;
} ;
static struct reg_parse_fd_opt
reg_parse_fd_opt ( void * mem_ctx , const char * options )
{
struct reg_parse_fd_opt ret = {
. file_enc = NULL ,
. str_enc = NULL ,
. flags = 0 ,
} ;
void * ctx = talloc_new ( mem_ctx ) ;
char * key , * val ;
if ( options = = NULL ) {
goto done ;
}
while ( srprs_option ( & options , ctx , & key , & val ) ) {
if ( strcmp ( key , " enc " ) = = 0 ) {
ret . file_enc = talloc_steal ( mem_ctx , val ) ;
ret . str_enc = ret . file_enc ;
} else if ( strcmp ( key , " strenc " ) = = 0 ) {
ret . str_enc = talloc_steal ( mem_ctx , val ) ;
} else if ( strcmp ( key , " fileenc " ) = = 0 ) {
ret . file_enc = talloc_steal ( mem_ctx , val ) ;
} else if ( ( strcmp ( key , " flags " ) = = 0 ) & & ( val ! = NULL ) ) {
char * end = NULL ;
if ( val ! = NULL ) {
ret . flags = strtol ( val , & end , 0 ) ;
}
if ( ( end = = NULL ) | | ( * end ! = ' \0 ' ) ) {
DEBUG ( 0 , ( " Invalid format \" %s \" : %s \n " ,
key , val ? val : " <NULL> " ) ) ;
}
} else if ( ( strcmp ( key , " fail " ) = = 0 ) & & ( val ! = NULL ) ) {
char * end = NULL ;
if ( val ! = NULL ) {
ret . fail_level = - strtol ( val , & end , 0 ) ;
}
if ( ( end = = NULL ) | | ( * end ! = ' \0 ' ) ) {
DEBUG ( 0 , ( " Invalid format \" %s \" : %s \n " ,
key , val ? val : " <NULL> " ) ) ;
}
}
}
done :
talloc_free ( ctx ) ;
return ret ;
}
static void
handle_iconv_errno ( int err , const char * obuf , size_t linenum ,
smb_iconv_t cd , const char * * iptr , size_t * ilen ,
char * * optr , size_t * olen )
{
const char * pos = obuf ;
const char * ptr = obuf ;
switch ( err ) {
case EINVAL :
/* DEBUG(0, ("Incomplete multibyte sequence\n")); */
case E2BIG :
return ;
case EILSEQ :
break ;
default :
assert ( false ) ;
}
* * optr = ' \0 ' ;
while ( srprs_line ( & ptr , NULL ) & & srprs_nl ( & ptr , NULL ) ) {
pos = ptr ;
linenum + + ;
}
if ( pos = = * optr ) {
pos = MAX ( obuf , * optr - 60 ) ;
}
2010-09-25 14:59:42 -07:00
DEBUG ( 0 , ( " Illegal multibyte sequence at line %lu: %s " ,
( long unsigned ) ( linenum + 1 ) , pos ) ) ;
2010-09-20 14:46:25 +02:00
assert ( ilen > 0 ) ;
do {
size_t il = 1 ;
DEBUGADD ( 0 , ( " <%02x> " , ( unsigned char ) * * iptr ) ) ;
if ( olen > 0 ) {
* ( * optr ) + + = ' \? ' ;
( * iptr ) + + ;
/* Todo: parametrize, e.g. skip: *optr++ = *iptr++; */
( * ilen ) - - ;
}
if ( smb_iconv ( cd , iptr , & il , optr , olen ) ! = ( size_t ) - 1 | | ( errno ! = EILSEQ ) ) {
if ( il = = 0 )
( * ilen ) - - ;
break ;
}
}
while ( ( * ilen > 0 ) & & ( * olen > 0 ) ) ;
DEBUGADD ( 0 , ( " \n " ) ) ;
}
int reg_parse_fd ( int fd , const struct reg_parse_callback * cb , const char * opts )
{
void * mem_ctx = talloc_stackframe ( ) ;
cbuf * line = cbuf_new ( mem_ctx ) ;
smb_iconv_t cd = ( smb_iconv_t ) - 1 ;
struct reg_parse * parser = NULL ;
char buf_raw [ 1024 ] ;
char buf_unix [ 1025 ] ;
ssize_t nread ;
size_t nconv ;
const char * pos ;
const char * iptr ;
char * optr ;
size_t ilen ;
size_t olen ;
int ret = - 1 ;
bool eof = false ;
size_t linenum = 0 ;
struct reg_parse_fd_opt opt = reg_parse_fd_opt ( mem_ctx , opts ) ;
if ( cb = = NULL ) {
DEBUG ( 0 , ( " reg_parse_fd: NULL callback \n " ) ) ;
goto done ;
}
nread = read ( fd , buf_raw , sizeof ( buf_raw ) ) ;
if ( nread < 0 ) {
DEBUG ( 0 , ( " reg_parse_fd: read failed: %s \n " , strerror ( errno ) ) ) ;
ret = nread ;
goto done ;
}
iptr = & buf_raw [ 0 ] ;
ilen = nread ;
if ( ! guess_charset ( & iptr , & ilen ,
& opt . file_enc , & opt . str_enc ) )
{
DEBUG ( 0 , ( " reg_parse_fd: failed to guess encoding \n " ) ) ;
goto done ;
}
2010-12-09 11:25:17 +01:00
DEBUG ( 10 , ( " reg_parse_fd: encoding file: %s str: %s \n " ,
2010-09-20 14:46:25 +02:00
opt . file_enc , opt . str_enc ) ) ;
if ( ! set_iconv ( & cd , " unix " , opt . file_enc ) ) {
DEBUG ( 0 , ( " reg_parse_fd: failed to set file encoding %s \n " ,
opt . file_enc ) ) ;
goto done ;
}
parser = reg_parse_new ( mem_ctx , * cb , opt . str_enc , opt . flags ) ;
optr = & buf_unix [ 0 ] ;
while ( ! eof ) {
olen = sizeof ( buf_unix ) - ( optr - buf_unix ) - 1 ;
while ( olen > 0 ) {
memmove ( buf_raw , iptr , ilen ) ;
nread = read ( fd , buf_raw + ilen , sizeof ( buf_raw ) - ilen ) ;
if ( nread < 0 ) {
DEBUG ( 0 , ( " reg_parse_fd: read failed: %s \n " , strerror ( errno ) ) ) ;
ret = nread ;
goto done ;
}
iptr = buf_raw ;
ilen + = nread ;
if ( ilen = = 0 ) {
smb_iconv ( cd , NULL , NULL , & optr , & olen ) ;
eof = true ;
break ;
}
nconv = smb_iconv ( cd , & iptr , & ilen , & optr , & olen ) ;
if ( nconv = = ( size_t ) - 1 ) {
handle_iconv_errno ( errno , buf_unix , linenum ,
cd , & iptr , & ilen ,
& optr , & olen ) ;
break ;
}
}
/* process_lines: */
* optr = ' \0 ' ;
pos = & buf_unix [ 0 ] ;
while ( srprs_line ( & pos , line ) & & srprs_nl_no_eos ( & pos , line , eof ) ) {
linenum + + ;
ret = reg_parse_line ( parser , cbuf_gets ( line , 0 ) ) ;
if ( ret < opt . fail_level ) {
goto done ;
}
cbuf_clear ( line ) ;
}
memmove ( buf_unix , pos , optr - pos ) ;
optr - = ( pos - buf_unix ) ;
}
ret = 0 ;
done :
set_iconv ( & cd , NULL , NULL ) ;
talloc_free ( mem_ctx ) ;
return ret ;
}
int reg_parse_file ( const char * fname , const struct reg_parse_callback * cb ,
const char * opt )
{
int ret = - 1 ;
int fd ;
fd = open ( fname , O_RDONLY ) ;
if ( fd < 0 ) {
DEBUG ( 0 , ( " reg_parse_file: open failed: %s \n " , strerror ( errno ) ) ) ;
return - 1 ;
}
ret = reg_parse_fd ( fd , cb , opt ) ;
close ( fd ) ;
return ret ;
}
/* static struct registry_key *find_regkey_by_hnd(pipes_struct *p, */
/* struct policy_handle *hnd) */
/* { */
/* struct registry_key *regkey = NULL; */
/* if(!find_policy_by_hnd(p,hnd,(void **)(void *)®key)) { */
/* DEBUG(2,("find_regkey_index_by_hnd: Registry Key not found: ")); */
/* return NULL; */
/* } */
/* return regkey; */
/* } */